blob: 141adaf29fd0d8410a4db06902a5c03a60e0ede0 [file] [log] [blame]
reed@android.com8a1c16f2008-12-17 15:59:43 +00001/*
epoger@google.comec3ed6a2011-07-28 14:26:00 +00002 * Copyright 2008 The Android Open Source Project
reed@android.com8a1c16f2008-12-17 15:59:43 +00003 *
epoger@google.comec3ed6a2011-07-28 14:26:00 +00004 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
reed@android.com8a1c16f2008-12-17 15:59:43 +00006 */
7
Herb Derbyd7b34a52017-03-20 11:19:23 -04008#include "SkInterpolator.h"
epoger@google.comec3ed6a2011-07-28 14:26:00 +00009
benjaminwagner6c71e0a2016-04-07 08:49:31 -070010#include "SkFixed.h"
Hal Canaryfdcfb8b2018-06-13 09:42:32 -040011#include "SkMalloc.h"
Hal Canaryc640d0d2018-06-13 09:59:02 -040012#include "SkMath.h"
reed@android.com8a1c16f2008-12-17 15:59:43 +000013#include "SkTSearch.h"
Hal Canaryc640d0d2018-06-13 09:59:02 -040014#include "SkTo.h"
reed@android.com8a1c16f2008-12-17 15:59:43 +000015
16SkInterpolatorBase::SkInterpolatorBase() {
halcanary96fcdcc2015-08-27 07:41:13 -070017 fStorage = nullptr;
18 fTimes = nullptr;
19 SkDEBUGCODE(fTimesArray = nullptr;)
reed@android.com8a1c16f2008-12-17 15:59:43 +000020}
21
22SkInterpolatorBase::~SkInterpolatorBase() {
23 if (fStorage) {
24 sk_free(fStorage);
25 }
26}
27
28void SkInterpolatorBase::reset(int elemCount, int frameCount) {
29 fFlags = 0;
30 fElemCount = SkToU8(elemCount);
31 fFrameCount = SkToS16(frameCount);
32 fRepeat = SK_Scalar1;
33 if (fStorage) {
34 sk_free(fStorage);
halcanary96fcdcc2015-08-27 07:41:13 -070035 fStorage = nullptr;
36 fTimes = nullptr;
37 SkDEBUGCODE(fTimesArray = nullptr);
reed@android.com8a1c16f2008-12-17 15:59:43 +000038 }
39}
40
41/* Each value[] run is formated as:
42 <time (in msec)>
43 <blend>
44 <data[fElemCount]>
45
46 Totaling fElemCount+2 entries per keyframe
47*/
48
49bool SkInterpolatorBase::getDuration(SkMSec* startTime, SkMSec* endTime) const {
50 if (fFrameCount == 0) {
51 return false;
52 }
53
54 if (startTime) {
55 *startTime = fTimes[0].fTime;
56 }
57 if (endTime) {
58 *endTime = fTimes[fFrameCount - 1].fTime;
59 }
60 return true;
61}
62
63SkScalar SkInterpolatorBase::ComputeRelativeT(SkMSec time, SkMSec prevTime,
64 SkMSec nextTime, const SkScalar blend[4]) {
65 SkASSERT(time > prevTime && time < nextTime);
66
reed80ea19c2015-05-12 10:37:34 -070067 SkScalar t = (SkScalar)(time - prevTime) / (SkScalar)(nextTime - prevTime);
reed@android.com8a1c16f2008-12-17 15:59:43 +000068 return blend ?
69 SkUnitCubicInterp(t, blend[0], blend[1], blend[2], blend[3]) : t;
70}
71
72SkInterpolatorBase::Result SkInterpolatorBase::timeToT(SkMSec time, SkScalar* T,
reed9a878a02015-12-27 12:47:25 -080073 int* indexPtr, bool* exactPtr) const {
reed@android.com8a1c16f2008-12-17 15:59:43 +000074 SkASSERT(fFrameCount > 0);
75 Result result = kNormal_Result;
76 if (fRepeat != SK_Scalar1) {
reed@android.comf523e252009-01-26 23:15:37 +000077 SkMSec startTime = 0, endTime = 0; // initialize to avoid warning
reed@android.com8a1c16f2008-12-17 15:59:43 +000078 this->getDuration(&startTime, &endTime);
79 SkMSec totalTime = endTime - startTime;
80 SkMSec offsetTime = time - startTime;
reed@google.com8015cdd2013-12-18 15:49:32 +000081 endTime = SkScalarFloorToInt(fRepeat * totalTime);
reed@android.com8a1c16f2008-12-17 15:59:43 +000082 if (offsetTime >= endTime) {
83 SkScalar fraction = SkScalarFraction(fRepeat);
84 offsetTime = fraction == 0 && fRepeat > 0 ? totalTime :
reed@google.com8015cdd2013-12-18 15:49:32 +000085 (SkMSec) SkScalarFloorToInt(fraction * totalTime);
reed@android.com8a1c16f2008-12-17 15:59:43 +000086 result = kFreezeEnd_Result;
87 } else {
88 int mirror = fFlags & kMirror;
89 offsetTime = offsetTime % (totalTime << mirror);
90 if (offsetTime > totalTime) { // can only be true if fMirror is true
91 offsetTime = (totalTime << 1) - offsetTime;
92 }
93 }
94 time = offsetTime + startTime;
95 }
96
97 int index = SkTSearch<SkMSec>(&fTimes[0].fTime, fFrameCount, time,
98 sizeof(SkTimeCode));
99
100 bool exact = true;
101
102 if (index < 0) {
103 index = ~index;
104 if (index == 0) {
105 result = kFreezeStart_Result;
106 } else if (index == fFrameCount) {
107 if (fFlags & kReset) {
108 index = 0;
109 } else {
110 index -= 1;
111 }
112 result = kFreezeEnd_Result;
113 } else {
114 exact = false;
115 }
116 }
117 SkASSERT(index < fFrameCount);
118 const SkTimeCode* nextTime = &fTimes[index];
119 SkMSec nextT = nextTime[0].fTime;
120 if (exact) {
121 *T = 0;
122 } else {
123 SkMSec prevT = nextTime[-1].fTime;
124 *T = ComputeRelativeT(time, prevT, nextT, nextTime[-1].fBlend);
125 }
126 *indexPtr = index;
127 *exactPtr = exact;
128 return result;
129}
130
131
132SkInterpolator::SkInterpolator() {
133 INHERITED::reset(0, 0);
halcanary96fcdcc2015-08-27 07:41:13 -0700134 fValues = nullptr;
135 SkDEBUGCODE(fScalarsArray = nullptr;)
reed@android.com8a1c16f2008-12-17 15:59:43 +0000136}
137
138SkInterpolator::SkInterpolator(int elemCount, int frameCount) {
139 SkASSERT(elemCount > 0);
140 this->reset(elemCount, frameCount);
141}
142
143void SkInterpolator::reset(int elemCount, int frameCount) {
144 INHERITED::reset(elemCount, frameCount);
145 fStorage = sk_malloc_throw((sizeof(SkScalar) * elemCount +
146 sizeof(SkTimeCode)) * frameCount);
147 fTimes = (SkTimeCode*) fStorage;
148 fValues = (SkScalar*) ((char*) fStorage + sizeof(SkTimeCode) * frameCount);
149#ifdef SK_DEBUG
150 fTimesArray = (SkTimeCode(*)[10]) fTimes;
151 fScalarsArray = (SkScalar(*)[10]) fValues;
152#endif
153}
154
155#define SK_Fixed1Third (SK_Fixed1/3)
156#define SK_Fixed2Third (SK_Fixed1*2/3)
157
158static const SkScalar gIdentityBlend[4] = {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000159 0.33333333f, 0.33333333f, 0.66666667f, 0.66666667f
reed@android.com8a1c16f2008-12-17 15:59:43 +0000160};
161
162bool SkInterpolator::setKeyFrame(int index, SkMSec time,
163 const SkScalar values[], const SkScalar blend[4]) {
halcanary96fcdcc2015-08-27 07:41:13 -0700164 SkASSERT(values != nullptr);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000165
halcanary96fcdcc2015-08-27 07:41:13 -0700166 if (blend == nullptr) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000167 blend = gIdentityBlend;
168 }
169
170 bool success = ~index == SkTSearch<SkMSec>(&fTimes->fTime, index, time,
171 sizeof(SkTimeCode));
172 SkASSERT(success);
173 if (success) {
174 SkTimeCode* timeCode = &fTimes[index];
175 timeCode->fTime = time;
176 memcpy(timeCode->fBlend, blend, sizeof(timeCode->fBlend));
177 SkScalar* dst = &fValues[fElemCount * index];
178 memcpy(dst, values, fElemCount * sizeof(SkScalar));
179 }
180 return success;
181}
182
183SkInterpolator::Result SkInterpolator::timeToValues(SkMSec time,
184 SkScalar values[]) const {
185 SkScalar T;
186 int index;
reed9a878a02015-12-27 12:47:25 -0800187 bool exact;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000188 Result result = timeToT(time, &T, &index, &exact);
189 if (values) {
190 const SkScalar* nextSrc = &fValues[index * fElemCount];
191
192 if (exact) {
193 memcpy(values, nextSrc, fElemCount * sizeof(SkScalar));
194 } else {
195 SkASSERT(index > 0);
196
197 const SkScalar* prevSrc = nextSrc - fElemCount;
198
199 for (int i = fElemCount - 1; i >= 0; --i) {
200 values[i] = SkScalarInterp(prevSrc[i], nextSrc[i], T);
201 }
202 }
203 }
204 return result;
205}
206
207///////////////////////////////////////////////////////////////////////////////
208
209typedef int Dot14;
210#define Dot14_ONE (1 << 14)
211#define Dot14_HALF (1 << 13)
212
213#define Dot14ToFloat(x) ((x) / 16384.f)
214
215static inline Dot14 Dot14Mul(Dot14 a, Dot14 b) {
216 return (a * b + Dot14_HALF) >> 14;
217}
218
219static inline Dot14 eval_cubic(Dot14 t, Dot14 A, Dot14 B, Dot14 C) {
220 return Dot14Mul(Dot14Mul(Dot14Mul(C, t) + B, t) + A, t);
221}
222
223static inline Dot14 pin_and_convert(SkScalar x) {
224 if (x <= 0) {
225 return 0;
226 }
227 if (x >= SK_Scalar1) {
228 return Dot14_ONE;
229 }
230 return SkScalarToFixed(x) >> 2;
231}
232
233SkScalar SkUnitCubicInterp(SkScalar value, SkScalar bx, SkScalar by,
234 SkScalar cx, SkScalar cy) {
235 // pin to the unit-square, and convert to 2.14
236 Dot14 x = pin_and_convert(value);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000237
reed@android.com8a1c16f2008-12-17 15:59:43 +0000238 if (x == 0) return 0;
239 if (x == Dot14_ONE) return SK_Scalar1;
rmistry@google.comd6176b02012-08-23 18:14:13 +0000240
reed@android.com8a1c16f2008-12-17 15:59:43 +0000241 Dot14 b = pin_and_convert(bx);
242 Dot14 c = pin_and_convert(cx);
rmistry@google.comd6176b02012-08-23 18:14:13 +0000243
reed@android.com8a1c16f2008-12-17 15:59:43 +0000244 // Now compute our coefficients from the control points
245 // t -> 3b
246 // t^2 -> 3c - 6b
247 // t^3 -> 3b - 3c + 1
248 Dot14 A = 3*b;
249 Dot14 B = 3*(c - 2*b);
250 Dot14 C = 3*(b - c) + Dot14_ONE;
251
252 // Now search for a t value given x
253 Dot14 t = Dot14_HALF;
254 Dot14 dt = Dot14_HALF;
255 for (int i = 0; i < 13; i++) {
256 dt >>= 1;
257 Dot14 guess = eval_cubic(t, A, B, C);
258 if (x < guess) {
259 t -= dt;
260 } else {
261 t += dt;
262 }
263 }
rmistry@google.comd6176b02012-08-23 18:14:13 +0000264
reed@android.com8a1c16f2008-12-17 15:59:43 +0000265 // Now we have t, so compute the coeff for Y and evaluate
266 b = pin_and_convert(by);
267 c = pin_and_convert(cy);
268 A = 3*b;
269 B = 3*(c - 2*b);
270 C = 3*(b - c) + Dot14_ONE;
271 return SkFixedToScalar(eval_cubic(t, A, B, C) << 2);
272}