blob: d9fc39c3bc1198bd2d7ed246cf8f96ee4396fb9e [file] [log] [blame]
rileya@google.com589708b2012-07-26 20:04:23 +00001/*
2 * Copyright 2012 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
8#include "SkTwoPointConicalGradient.h"
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00009#include "SkTwoPointConicalGradient_gpu.h"
10
commit-bot@chromium.org87fcd952014-04-23 19:10:51 +000011struct TwoPtRadialContext {
12 const TwoPtRadial& fRec;
13 float fRelX, fRelY;
14 const float fIncX, fIncY;
15 float fB;
16 const float fDB;
17
18 TwoPtRadialContext(const TwoPtRadial& rec, SkScalar fx, SkScalar fy,
19 SkScalar dfx, SkScalar dfy);
20 SkFixed nextT();
21};
22
rileya@google.com589708b2012-07-26 20:04:23 +000023static int valid_divide(float numer, float denom, float* ratio) {
24 SkASSERT(ratio);
25 if (0 == denom) {
26 return 0;
27 }
28 *ratio = numer / denom;
29 return 1;
30}
31
32// Return the number of distinct real roots, and write them into roots[] in
33// ascending order
commit-bot@chromium.org44d83c12014-04-21 13:10:25 +000034static int find_quad_roots(float A, float B, float C, float roots[2], bool descendingOrder = false) {
rileya@google.com589708b2012-07-26 20:04:23 +000035 SkASSERT(roots);
rmistry@google.comfbfcd562012-08-23 18:09:54 +000036
rileya@google.com589708b2012-07-26 20:04:23 +000037 if (A == 0) {
38 return valid_divide(-C, B, roots);
39 }
rmistry@google.comfbfcd562012-08-23 18:09:54 +000040
rileya@google.com589708b2012-07-26 20:04:23 +000041 float R = B*B - 4*A*C;
42 if (R < 0) {
43 return 0;
44 }
45 R = sk_float_sqrt(R);
46
47#if 1
48 float Q = B;
49 if (Q < 0) {
50 Q -= R;
51 } else {
52 Q += R;
53 }
54#else
55 // on 10.6 this was much slower than the above branch :(
56 float Q = B + copysignf(R, B);
57#endif
58 Q *= -0.5f;
59 if (0 == Q) {
60 roots[0] = 0;
61 return 1;
62 }
63
64 float r0 = Q / A;
65 float r1 = C / Q;
66 roots[0] = r0 < r1 ? r0 : r1;
67 roots[1] = r0 > r1 ? r0 : r1;
commit-bot@chromium.org44d83c12014-04-21 13:10:25 +000068 if (descendingOrder) {
69 SkTSwap(roots[0], roots[1]);
70 }
rileya@google.com589708b2012-07-26 20:04:23 +000071 return 2;
72}
73
74static float lerp(float x, float dx, float t) {
75 return x + t * dx;
76}
77
78static float sqr(float x) { return x * x; }
79
80void TwoPtRadial::init(const SkPoint& center0, SkScalar rad0,
commit-bot@chromium.org44d83c12014-04-21 13:10:25 +000081 const SkPoint& center1, SkScalar rad1,
82 bool flipped) {
rileya@google.com589708b2012-07-26 20:04:23 +000083 fCenterX = SkScalarToFloat(center0.fX);
84 fCenterY = SkScalarToFloat(center0.fY);
85 fDCenterX = SkScalarToFloat(center1.fX) - fCenterX;
86 fDCenterY = SkScalarToFloat(center1.fY) - fCenterY;
87 fRadius = SkScalarToFloat(rad0);
88 fDRadius = SkScalarToFloat(rad1) - fRadius;
89
90 fA = sqr(fDCenterX) + sqr(fDCenterY) - sqr(fDRadius);
91 fRadius2 = sqr(fRadius);
92 fRDR = fRadius * fDRadius;
commit-bot@chromium.org44d83c12014-04-21 13:10:25 +000093
94 fFlipped = flipped;
rileya@google.com589708b2012-07-26 20:04:23 +000095}
96
commit-bot@chromium.org87fcd952014-04-23 19:10:51 +000097TwoPtRadialContext::TwoPtRadialContext(const TwoPtRadial& rec, SkScalar fx, SkScalar fy,
98 SkScalar dfx, SkScalar dfy)
99 : fRec(rec)
100 , fRelX(SkScalarToFloat(fx) - rec.fCenterX)
101 , fRelY(SkScalarToFloat(fy) - rec.fCenterY)
102 , fIncX(SkScalarToFloat(dfx))
103 , fIncY(SkScalarToFloat(dfy))
104 , fB(-2 * (rec.fDCenterX * fRelX + rec.fDCenterY * fRelY + rec.fRDR))
105 , fDB(-2 * (rec.fDCenterX * fIncX + rec.fDCenterY * fIncY)) {}
rileya@google.com589708b2012-07-26 20:04:23 +0000106
commit-bot@chromium.org87fcd952014-04-23 19:10:51 +0000107SkFixed TwoPtRadialContext::nextT() {
rileya@google.com589708b2012-07-26 20:04:23 +0000108 float roots[2];
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000109
commit-bot@chromium.org87fcd952014-04-23 19:10:51 +0000110 float C = sqr(fRelX) + sqr(fRelY) - fRec.fRadius2;
111 int countRoots = find_quad_roots(fRec.fA, fB, C, roots, fRec.fFlipped);
rileya@google.com589708b2012-07-26 20:04:23 +0000112
113 fRelX += fIncX;
114 fRelY += fIncY;
115 fB += fDB;
116
117 if (0 == countRoots) {
commit-bot@chromium.org87fcd952014-04-23 19:10:51 +0000118 return TwoPtRadial::kDontDrawT;
rileya@google.com589708b2012-07-26 20:04:23 +0000119 }
120
121 // Prefer the bigger t value if both give a radius(t) > 0
122 // find_quad_roots returns the values sorted, so we start with the last
123 float t = roots[countRoots - 1];
commit-bot@chromium.org87fcd952014-04-23 19:10:51 +0000124 float r = lerp(fRec.fRadius, fRec.fDRadius, t);
fmalita5edf82e2016-03-03 06:41:54 -0800125 if (r < 0) {
rileya@google.com589708b2012-07-26 20:04:23 +0000126 t = roots[0]; // might be the same as roots[countRoots-1]
commit-bot@chromium.org87fcd952014-04-23 19:10:51 +0000127 r = lerp(fRec.fRadius, fRec.fDRadius, t);
fmalita5edf82e2016-03-03 06:41:54 -0800128 if (r < 0) {
commit-bot@chromium.org87fcd952014-04-23 19:10:51 +0000129 return TwoPtRadial::kDontDrawT;
rileya@google.com589708b2012-07-26 20:04:23 +0000130 }
131 }
132 return SkFloatToFixed(t);
133}
134
commit-bot@chromium.org87fcd952014-04-23 19:10:51 +0000135typedef void (*TwoPointConicalProc)(TwoPtRadialContext* rec, SkPMColor* dstC,
reed@google.com60040292013-02-04 18:21:23 +0000136 const SkPMColor* cache, int toggle, int count);
rileya@google.com589708b2012-07-26 20:04:23 +0000137
commit-bot@chromium.org87fcd952014-04-23 19:10:51 +0000138static void twopoint_clamp(TwoPtRadialContext* rec, SkPMColor* SK_RESTRICT dstC,
reed@google.com60040292013-02-04 18:21:23 +0000139 const SkPMColor* SK_RESTRICT cache, int toggle,
140 int count) {
rileya@google.com589708b2012-07-26 20:04:23 +0000141 for (; count > 0; --count) {
142 SkFixed t = rec->nextT();
143 if (TwoPtRadial::DontDrawT(t)) {
144 *dstC++ = 0;
145 } else {
146 SkFixed index = SkClampMax(t, 0xFFFF);
147 SkASSERT(index <= 0xFFFF);
reed@google.com60040292013-02-04 18:21:23 +0000148 *dstC++ = cache[toggle +
149 (index >> SkGradientShaderBase::kCache32Shift)];
rileya@google.com589708b2012-07-26 20:04:23 +0000150 }
reed@google.com60040292013-02-04 18:21:23 +0000151 toggle = next_dither_toggle(toggle);
rileya@google.com589708b2012-07-26 20:04:23 +0000152 }
153}
154
commit-bot@chromium.org87fcd952014-04-23 19:10:51 +0000155static void twopoint_repeat(TwoPtRadialContext* rec, SkPMColor* SK_RESTRICT dstC,
reed@google.com60040292013-02-04 18:21:23 +0000156 const SkPMColor* SK_RESTRICT cache, int toggle,
157 int count) {
rileya@google.com589708b2012-07-26 20:04:23 +0000158 for (; count > 0; --count) {
159 SkFixed t = rec->nextT();
160 if (TwoPtRadial::DontDrawT(t)) {
161 *dstC++ = 0;
162 } else {
163 SkFixed index = repeat_tileproc(t);
164 SkASSERT(index <= 0xFFFF);
reed@google.com60040292013-02-04 18:21:23 +0000165 *dstC++ = cache[toggle +
166 (index >> SkGradientShaderBase::kCache32Shift)];
rileya@google.com589708b2012-07-26 20:04:23 +0000167 }
reed@google.com60040292013-02-04 18:21:23 +0000168 toggle = next_dither_toggle(toggle);
rileya@google.com589708b2012-07-26 20:04:23 +0000169 }
170}
171
commit-bot@chromium.org87fcd952014-04-23 19:10:51 +0000172static void twopoint_mirror(TwoPtRadialContext* rec, SkPMColor* SK_RESTRICT dstC,
reed@google.com60040292013-02-04 18:21:23 +0000173 const SkPMColor* SK_RESTRICT cache, int toggle,
174 int count) {
rileya@google.com589708b2012-07-26 20:04:23 +0000175 for (; count > 0; --count) {
176 SkFixed t = rec->nextT();
177 if (TwoPtRadial::DontDrawT(t)) {
178 *dstC++ = 0;
179 } else {
180 SkFixed index = mirror_tileproc(t);
181 SkASSERT(index <= 0xFFFF);
reed@google.com60040292013-02-04 18:21:23 +0000182 *dstC++ = cache[toggle +
183 (index >> SkGradientShaderBase::kCache32Shift)];
rileya@google.com589708b2012-07-26 20:04:23 +0000184 }
reed@google.com60040292013-02-04 18:21:23 +0000185 toggle = next_dither_toggle(toggle);
rileya@google.com589708b2012-07-26 20:04:23 +0000186 }
187}
188
rileya@google.com98e8b6d2012-07-31 20:38:06 +0000189/////////////////////////////////////////////////////////////////////
190
rileya@google.com589708b2012-07-26 20:04:23 +0000191SkTwoPointConicalGradient::SkTwoPointConicalGradient(
reed@google.com3d3a8602013-05-24 14:58:44 +0000192 const SkPoint& start, SkScalar startRadius,
193 const SkPoint& end, SkScalar endRadius,
reedaddf2ed2014-08-11 08:28:24 -0700194 bool flippedGrad, const Descriptor& desc)
mtkleincc695fe2014-12-10 10:29:19 -0800195 : SkGradientShaderBase(desc, SkMatrix::I())
reedaddf2ed2014-08-11 08:28:24 -0700196 , fCenter1(start)
197 , fCenter2(end)
198 , fRadius1(startRadius)
199 , fRadius2(endRadius)
200 , fFlippedGrad(flippedGrad)
201{
rileya@google.com589708b2012-07-26 20:04:23 +0000202 // this is degenerate, and should be caught by our caller
203 SkASSERT(fCenter1 != fCenter2 || fRadius1 != fRadius2);
mtkleincc695fe2014-12-10 10:29:19 -0800204 fRec.init(fCenter1, fRadius1, fCenter2, fRadius2, fFlippedGrad);
rileya@google.com589708b2012-07-26 20:04:23 +0000205}
206
commit-bot@chromium.org3fbab822013-03-20 00:49:57 +0000207bool SkTwoPointConicalGradient::isOpaque() const {
robertphillips@google.comcb6d97c2013-07-09 13:50:09 +0000208 // Because areas outside the cone are left untouched, we cannot treat the
209 // shader as opaque even if the gradient itself is opaque.
210 // TODO(junov): Compute whether the cone fills the plane crbug.com/222380
211 return false;
commit-bot@chromium.org3fbab822013-03-20 00:49:57 +0000212}
213
reed773ceda2016-03-03 18:18:25 -0800214size_t SkTwoPointConicalGradient::onContextSize(const ContextRec&) const {
reeda0cee5f2016-03-04 07:38:11 -0800215 return sizeof(TwoPointConicalGradientContext);
commit-bot@chromium.org87fcd952014-04-23 19:10:51 +0000216}
217
commit-bot@chromium.orgce56d962014-05-05 18:39:18 +0000218SkShader::Context* SkTwoPointConicalGradient::onCreateContext(const ContextRec& rec,
219 void* storage) const {
halcanary385fe4d2015-08-26 13:07:48 -0700220 return new (storage) TwoPointConicalGradientContext(*this, rec);
commit-bot@chromium.org87fcd952014-04-23 19:10:51 +0000221}
222
223SkTwoPointConicalGradient::TwoPointConicalGradientContext::TwoPointConicalGradientContext(
commit-bot@chromium.orge901b6d2014-05-01 19:31:31 +0000224 const SkTwoPointConicalGradient& shader, const ContextRec& rec)
225 : INHERITED(shader, rec)
commit-bot@chromium.org87fcd952014-04-23 19:10:51 +0000226{
commit-bot@chromium.org87fcd952014-04-23 19:10:51 +0000227 // in general, we might discard based on computed-radius, so clear
228 // this flag (todo: sometimes we can detect that we never discard...)
229 fFlags &= ~kOpaqueAlpha_Flag;
230}
231
232void SkTwoPointConicalGradient::TwoPointConicalGradientContext::shadeSpan(
233 int x, int y, SkPMColor* dstCParam, int count) {
234 const SkTwoPointConicalGradient& twoPointConicalGradient =
235 static_cast<const SkTwoPointConicalGradient&>(fShader);
236
reed@google.com60040292013-02-04 18:21:23 +0000237 int toggle = init_dither_toggle(x, y);
reed@google.com60040292013-02-04 18:21:23 +0000238
rileya@google.com589708b2012-07-26 20:04:23 +0000239 SkASSERT(count > 0);
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000240
rileya@google.com589708b2012-07-26 20:04:23 +0000241 SkPMColor* SK_RESTRICT dstC = dstCParam;
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000242
rileya@google.com589708b2012-07-26 20:04:23 +0000243 SkMatrix::MapXYProc dstProc = fDstToIndexProc;
bsalomon@google.com100abf42012-09-05 17:40:04 +0000244
commit-bot@chromium.org87fcd952014-04-23 19:10:51 +0000245 const SkPMColor* SK_RESTRICT cache = fCache->getCache32();
rileya@google.com589708b2012-07-26 20:04:23 +0000246
reed@google.com60040292013-02-04 18:21:23 +0000247 TwoPointConicalProc shadeProc = twopoint_repeat;
commit-bot@chromium.org87fcd952014-04-23 19:10:51 +0000248 if (SkShader::kClamp_TileMode == twoPointConicalGradient.fTileMode) {
rileya@google.com589708b2012-07-26 20:04:23 +0000249 shadeProc = twopoint_clamp;
commit-bot@chromium.org87fcd952014-04-23 19:10:51 +0000250 } else if (SkShader::kMirror_TileMode == twoPointConicalGradient.fTileMode) {
rileya@google.com589708b2012-07-26 20:04:23 +0000251 shadeProc = twopoint_mirror;
252 } else {
commit-bot@chromium.org87fcd952014-04-23 19:10:51 +0000253 SkASSERT(SkShader::kRepeat_TileMode == twoPointConicalGradient.fTileMode);
rileya@google.com589708b2012-07-26 20:04:23 +0000254 }
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000255
rileya@google.com589708b2012-07-26 20:04:23 +0000256 if (fDstToIndexClass != kPerspective_MatrixClass) {
257 SkPoint srcPt;
258 dstProc(fDstToIndex, SkIntToScalar(x) + SK_ScalarHalf,
259 SkIntToScalar(y) + SK_ScalarHalf, &srcPt);
260 SkScalar dx, fx = srcPt.fX;
261 SkScalar dy, fy = srcPt.fY;
rmistry@google.comfbfcd562012-08-23 18:09:54 +0000262
rileya@google.com589708b2012-07-26 20:04:23 +0000263 if (fDstToIndexClass == kFixedStepInX_MatrixClass) {
benjaminwagner8e175562016-02-16 10:09:40 -0800264 const auto step = fDstToIndex.fixedStepInX(SkIntToScalar(y));
265 dx = step.fX;
266 dy = step.fY;
rileya@google.com589708b2012-07-26 20:04:23 +0000267 } else {
268 SkASSERT(fDstToIndexClass == kLinear_MatrixClass);
269 dx = fDstToIndex.getScaleX();
270 dy = fDstToIndex.getSkewY();
271 }
272
commit-bot@chromium.org87fcd952014-04-23 19:10:51 +0000273 TwoPtRadialContext rec(twoPointConicalGradient.fRec, fx, fy, dx, dy);
274 (*shadeProc)(&rec, dstC, cache, toggle, count);
rileya@google.com589708b2012-07-26 20:04:23 +0000275 } else { // perspective case
mike@reedtribe.org139a2352013-11-14 20:15:51 +0000276 SkScalar dstX = SkIntToScalar(x) + SK_ScalarHalf;
277 SkScalar dstY = SkIntToScalar(y) + SK_ScalarHalf;
rileya@google.com589708b2012-07-26 20:04:23 +0000278 for (; count > 0; --count) {
279 SkPoint srcPt;
280 dstProc(fDstToIndex, dstX, dstY, &srcPt);
commit-bot@chromium.org87fcd952014-04-23 19:10:51 +0000281 TwoPtRadialContext rec(twoPointConicalGradient.fRec, srcPt.fX, srcPt.fY, 0, 0);
282 (*shadeProc)(&rec, dstC, cache, toggle, 1);
mike@reedtribe.org139a2352013-11-14 20:15:51 +0000283
284 dstX += SK_Scalar1;
reed@google.com60040292013-02-04 18:21:23 +0000285 toggle = next_dither_toggle(toggle);
mike@reedtribe.org139a2352013-11-14 20:15:51 +0000286 dstC += 1;
rileya@google.com589708b2012-07-26 20:04:23 +0000287 }
288 }
289}
290
commit-bot@chromium.org44d83c12014-04-21 13:10:25 +0000291// Returns the original non-sorted version of the gradient
rileya@google.com589708b2012-07-26 20:04:23 +0000292SkShader::GradientType SkTwoPointConicalGradient::asAGradient(
293 GradientInfo* info) const {
294 if (info) {
commit-bot@chromium.org44d83c12014-04-21 13:10:25 +0000295 commonAsAGradient(info, fFlippedGrad);
rileya@google.com589708b2012-07-26 20:04:23 +0000296 info->fPoint[0] = fCenter1;
297 info->fPoint[1] = fCenter2;
298 info->fRadius[0] = fRadius1;
299 info->fRadius[1] = fRadius2;
commit-bot@chromium.org44d83c12014-04-21 13:10:25 +0000300 if (fFlippedGrad) {
301 SkTSwap(info->fPoint[0], info->fPoint[1]);
302 SkTSwap(info->fRadius[0], info->fRadius[1]);
303 }
rileya@google.com589708b2012-07-26 20:04:23 +0000304 }
305 return kConical_GradientType;
306}
307
reed9fa60da2014-08-21 07:59:51 -0700308SkFlattenable* SkTwoPointConicalGradient::CreateProc(SkReadBuffer& buffer) {
309 DescriptorScope desc;
310 if (!desc.unflatten(buffer)) {
halcanary96fcdcc2015-08-27 07:41:13 -0700311 return nullptr;
reed9fa60da2014-08-21 07:59:51 -0700312 }
313 SkPoint c1 = buffer.readPoint();
314 SkPoint c2 = buffer.readPoint();
315 SkScalar r1 = buffer.readScalar();
316 SkScalar r2 = buffer.readScalar();
317
318 if (buffer.readBool()) { // flipped
319 SkTSwap(c1, c2);
320 SkTSwap(r1, r2);
321
322 SkColor* colors = desc.mutableColors();
323 SkScalar* pos = desc.mutablePos();
324 const int last = desc.fCount - 1;
325 const int half = desc.fCount >> 1;
326 for (int i = 0; i < half; ++i) {
327 SkTSwap(colors[i], colors[last - i]);
328 if (pos) {
329 SkScalar tmp = pos[i];
330 pos[i] = SK_Scalar1 - pos[last - i];
331 pos[last - i] = SK_Scalar1 - tmp;
332 }
333 }
334 if (pos) {
335 if (desc.fCount & 1) {
336 pos[half] = SK_Scalar1 - pos[half];
337 }
338 }
339 }
340
reed8a21c9f2016-03-08 18:50:00 -0800341 return SkGradientShader::MakeTwoPointConical(c1, r1, c2, r2, desc.fColors, desc.fPos,
342 desc.fCount, desc.fTileMode, desc.fGradFlags,
343 desc.fLocalMatrix).release();
reed9fa60da2014-08-21 07:59:51 -0700344}
345
346void SkTwoPointConicalGradient::flatten(SkWriteBuffer& buffer) const {
rileya@google.com589708b2012-07-26 20:04:23 +0000347 this->INHERITED::flatten(buffer);
348 buffer.writePoint(fCenter1);
349 buffer.writePoint(fCenter2);
350 buffer.writeScalar(fRadius1);
351 buffer.writeScalar(fRadius2);
commit-bot@chromium.org44d83c12014-04-21 13:10:25 +0000352 buffer.writeBool(fFlippedGrad);
rileya@google.com589708b2012-07-26 20:04:23 +0000353}
354
bsalomon@google.comcf8fb1f2012-08-02 14:03:32 +0000355#if SK_SUPPORT_GPU
356
dandov9de5b512014-06-10 14:38:28 -0700357#include "SkGr.h"
358
bsalomonc21b09e2015-08-28 18:46:56 -0700359const GrFragmentProcessor* SkTwoPointConicalGradient::asFragmentProcessor(
360 GrContext* context,
361 const SkMatrix& viewM,
362 const SkMatrix* localMatrix,
bsalomon4a339522015-10-06 08:40:50 -0700363 SkFilterQuality) const {
bsalomon49f085d2014-09-05 13:34:00 -0700364 SkASSERT(context);
bsalomon@google.comf94b3a42012-10-31 18:09:01 +0000365 SkASSERT(fPtsToUnit.isIdentity());
bsalomonc21b09e2015-08-28 18:46:56 -0700366 SkAutoTUnref<const GrFragmentProcessor> inner(
bsalomon4a339522015-10-06 08:40:50 -0700367 Gr2PtConicalGradientEffect::Create(context, *this, fTileMode, localMatrix));
bsalomonf1b7a1d2015-09-28 06:26:28 -0700368 return GrFragmentProcessor::MulOutputByInputAlpha(inner);
bsalomon@google.comcf8fb1f2012-08-02 14:03:32 +0000369}
370
twiz@google.coma5e65ec2012-08-02 15:15:16 +0000371#endif
robertphillips@google.com76f9e932013-01-15 20:17:47 +0000372
commit-bot@chromium.org0f10f7b2014-03-13 18:02:17 +0000373#ifndef SK_IGNORE_TO_STRING
robertphillips@google.com76f9e932013-01-15 20:17:47 +0000374void SkTwoPointConicalGradient::toString(SkString* str) const {
375 str->append("SkTwoPointConicalGradient: (");
376
377 str->append("center1: (");
378 str->appendScalar(fCenter1.fX);
379 str->append(", ");
380 str->appendScalar(fCenter1.fY);
381 str->append(") radius1: ");
382 str->appendScalar(fRadius1);
383 str->append(" ");
384
385 str->append("center2: (");
386 str->appendScalar(fCenter2.fX);
387 str->append(", ");
388 str->appendScalar(fCenter2.fY);
389 str->append(") radius2: ");
390 str->appendScalar(fRadius2);
391 str->append(" ");
392
393 this->INHERITED::toString(str);
394
395 str->append(")");
396}
397#endif