blob: 38b266d67ae6762ecc2c7fe847a488922bfed72d [file] [log] [blame]
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00002/*
3 * Copyright 2014 Google Inc.
4 *
5 * Use of this source code is governed by a BSD-style license that can be
6 * found in the LICENSE file.
7 */
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +00008
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00009#include "SkTwoPointConicalGradient_gpu.h"
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +000010
11#include "SkTwoPointConicalGradient.h"
12
commit-bot@chromium.orgef93d292014-04-10 15:37:52 +000013#if SK_SUPPORT_GPU
joshualitt8ca93e72015-07-08 06:51:43 -070014#include "GrPaint.h"
joshualitt30ba4362014-08-21 20:18:45 -070015#include "gl/builders/GrGLProgramBuilder.h"
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +000016// For brevity
kkinnunen7510b222014-07-30 00:04:16 -070017typedef GrGLProgramDataManager::UniformHandle UniformHandle;
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +000018
commit-bot@chromium.org80894672014-04-22 21:24:22 +000019static const SkScalar kErrorTol = 0.00001f;
egdaniel8405ef92014-06-09 11:57:28 -070020static const SkScalar kEdgeErrorTol = 5.f * kErrorTol;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +000021
22/**
23 * We have three general cases for 2pt conical gradients. First we always assume that
24 * the start radius <= end radius. Our first case (kInside_) is when the start circle
25 * is completely enclosed by the end circle. The second case (kOutside_) is the case
26 * when the start circle is either completely outside the end circle or the circles
27 * overlap. The final case (kEdge_) is when the start circle is inside the end one,
28 * but the two are just barely touching at 1 point along their edges.
29 */
30enum ConicalType {
31 kInside_ConicalType,
32 kOutside_ConicalType,
33 kEdge_ConicalType,
34};
35
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000036//////////////////////////////////////////////////////////////////////////////
37
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +000038static void set_matrix_edge_conical(const SkTwoPointConicalGradient& shader,
39 SkMatrix* invLMatrix) {
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000040 // Inverse of the current local matrix is passed in then,
41 // translate to center1, rotate so center2 is on x axis.
42 const SkPoint& center1 = shader.getStartCenter();
43 const SkPoint& center2 = shader.getEndCenter();
44
45 invLMatrix->postTranslate(-center1.fX, -center1.fY);
46
47 SkPoint diff = center2 - center1;
48 SkScalar diffLen = diff.length();
49 if (0 != diffLen) {
50 SkScalar invDiffLen = SkScalarInvert(diffLen);
51 SkMatrix rot;
52 rot.setSinCos(-SkScalarMul(invDiffLen, diff.fY),
53 SkScalarMul(invDiffLen, diff.fX));
54 invLMatrix->postConcat(rot);
55 }
56}
57
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +000058class Edge2PtConicalEffect : public GrGradientEffect {
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +000059public:
60
joshualittb0a8a372014-09-23 09:50:21 -070061 static GrFragmentProcessor* Create(GrContext* ctx,
joshualittb2456052015-07-08 09:36:59 -070062 GrShaderDataManager* shaderDataManager,
joshualittb0a8a372014-09-23 09:50:21 -070063 const SkTwoPointConicalGradient& shader,
64 const SkMatrix& matrix,
65 SkShader::TileMode tm) {
joshualittb2456052015-07-08 09:36:59 -070066 return SkNEW_ARGS(Edge2PtConicalEffect, (ctx, shaderDataManager, shader, matrix, tm));
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000067 }
68
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +000069 virtual ~Edge2PtConicalEffect() {}
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000070
mtklein36352bf2015-03-25 18:17:31 -070071 const char* name() const override {
joshualitteb2a6762014-12-04 11:35:33 -080072 return "Two-Point Conical Gradient Edge Touching";
73 }
74
jvanverthcfc18862015-04-28 08:48:20 -070075 void getGLProcessorKey(const GrGLSLCaps&, GrProcessorKeyBuilder*) const override;
joshualitteb2a6762014-12-04 11:35:33 -080076
mtklein36352bf2015-03-25 18:17:31 -070077 GrGLFragmentProcessor* createGLInstance() const override;
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000078
79 // The radial gradient parameters can collapse to a linear (instead of quadratic) equation.
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000080 SkScalar center() const { return fCenterX1; }
81 SkScalar diffRadius() const { return fDiffRadius; }
82 SkScalar radius() const { return fRadius0; }
83
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000084private:
mtklein36352bf2015-03-25 18:17:31 -070085 bool onIsEqual(const GrFragmentProcessor& sBase) const override {
joshualitt49586be2014-09-16 08:21:41 -070086 const Edge2PtConicalEffect& s = sBase.cast<Edge2PtConicalEffect>();
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000087 return (INHERITED::onIsEqual(sBase) &&
88 this->fCenterX1 == s.fCenterX1 &&
89 this->fRadius0 == s.fRadius0 &&
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +000090 this->fDiffRadius == s.fDiffRadius);
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000091 }
92
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +000093 Edge2PtConicalEffect(GrContext* ctx,
joshualittb2456052015-07-08 09:36:59 -070094 GrShaderDataManager* shaderDataManager,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +000095 const SkTwoPointConicalGradient& shader,
96 const SkMatrix& matrix,
97 SkShader::TileMode tm)
joshualittb2456052015-07-08 09:36:59 -070098 : INHERITED(ctx, shaderDataManager, shader, matrix, tm),
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000099 fCenterX1(shader.getCenterX1()),
100 fRadius0(shader.getStartRadius()),
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000101 fDiffRadius(shader.getDiffRadius()){
joshualitteb2a6762014-12-04 11:35:33 -0800102 this->initClassID<Edge2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000103 // We should only be calling this shader if we are degenerate case with touching circles
egdaniel8405ef92014-06-09 11:57:28 -0700104 // When deciding if we are in edge case, we scaled by the end radius for cases when the
joshualitt01258472014-09-22 10:29:30 -0700105 // start radius was close to zero, otherwise we scaled by the start radius. In addition
106 // Our test for the edge case in set_matrix_circle_conical has a higher tolerance so we
107 // need the sqrt value below
108 SkASSERT(SkScalarAbs(SkScalarAbs(fDiffRadius) - fCenterX1) <
109 (fRadius0 < kErrorTol ? shader.getEndRadius() * kEdgeErrorTol :
110 fRadius0 * sqrt(kEdgeErrorTol)));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000111
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +0000112 // We pass the linear part of the quadratic as a varying.
113 // float b = -2.0 * (fCenterX1 * x + fRadius0 * fDiffRadius * z)
114 fBTransform = this->getCoordTransform();
115 SkMatrix& bMatrix = *fBTransform.accessMatrix();
116 SkScalar r0dr = SkScalarMul(fRadius0, fDiffRadius);
117 bMatrix[SkMatrix::kMScaleX] = -2 * (SkScalarMul(fCenterX1, bMatrix[SkMatrix::kMScaleX]) +
118 SkScalarMul(r0dr, bMatrix[SkMatrix::kMPersp0]));
119 bMatrix[SkMatrix::kMSkewX] = -2 * (SkScalarMul(fCenterX1, bMatrix[SkMatrix::kMSkewX]) +
120 SkScalarMul(r0dr, bMatrix[SkMatrix::kMPersp1]));
121 bMatrix[SkMatrix::kMTransX] = -2 * (SkScalarMul(fCenterX1, bMatrix[SkMatrix::kMTransX]) +
122 SkScalarMul(r0dr, bMatrix[SkMatrix::kMPersp2]));
123 this->addCoordTransform(&fBTransform);
124 }
125
joshualittb0a8a372014-09-23 09:50:21 -0700126 GR_DECLARE_FRAGMENT_PROCESSOR_TEST;
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +0000127
128 // @{
129 // Cache of values - these can change arbitrarily, EXCEPT
130 // we shouldn't change between degenerate and non-degenerate?!
131
132 GrCoordTransform fBTransform;
133 SkScalar fCenterX1;
134 SkScalar fRadius0;
135 SkScalar fDiffRadius;
136
137 // @}
138
139 typedef GrGradientEffect INHERITED;
140};
141
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000142class GLEdge2PtConicalEffect : public GrGLGradientEffect {
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +0000143public:
joshualitteb2a6762014-12-04 11:35:33 -0800144 GLEdge2PtConicalEffect(const GrProcessor&);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000145 virtual ~GLEdge2PtConicalEffect() { }
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000146
joshualitt15988992014-10-09 15:04:05 -0700147 virtual void emitCode(GrGLFPBuilder*,
joshualittb0a8a372014-09-23 09:50:21 -0700148 const GrFragmentProcessor&,
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000149 const char* outputColor,
150 const char* inputColor,
151 const TransformedCoordsArray&,
mtklein36352bf2015-03-25 18:17:31 -0700152 const TextureSamplerArray&) override;
153 void setData(const GrGLProgramDataManager&, const GrProcessor&) override;
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000154
jvanverthcfc18862015-04-28 08:48:20 -0700155 static void GenKey(const GrProcessor&, const GrGLSLCaps& caps, GrProcessorKeyBuilder* b);
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000156
157protected:
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000158 UniformHandle fParamUni;
159
160 const char* fVSVaryingName;
161 const char* fFSVaryingName;
162
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000163 // @{
164 /// Values last uploaded as uniforms
165
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000166 SkScalar fCachedRadius;
167 SkScalar fCachedDiffRadius;
168
169 // @}
170
171private:
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000172 typedef GrGLGradientEffect INHERITED;
173
174};
175
jvanverthcfc18862015-04-28 08:48:20 -0700176void Edge2PtConicalEffect::getGLProcessorKey(const GrGLSLCaps& caps,
joshualitteb2a6762014-12-04 11:35:33 -0800177 GrProcessorKeyBuilder* b) const {
178 GLEdge2PtConicalEffect::GenKey(*this, caps, b);
179}
180
181GrGLFragmentProcessor* Edge2PtConicalEffect::createGLInstance() const {
182 return SkNEW_ARGS(GLEdge2PtConicalEffect, (*this));
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000183}
skia.committer@gmail.com221b9112014-04-04 03:04:32 +0000184
joshualittb0a8a372014-09-23 09:50:21 -0700185GR_DEFINE_FRAGMENT_PROCESSOR_TEST(Edge2PtConicalEffect);
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000186
joshualitt01258472014-09-22 10:29:30 -0700187/*
188 * All Two point conical gradient test create functions may occasionally create edge case shaders
189 */
joshualitt0067ff52015-07-08 14:26:19 -0700190GrFragmentProcessor* Edge2PtConicalEffect::TestCreate(GrProcessorTestData* d) {
191 SkPoint center1 = {d->fRandom->nextUScalar1(), d->fRandom->nextUScalar1()};
192 SkScalar radius1 = d->fRandom->nextUScalar1();
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000193 SkPoint center2;
194 SkScalar radius2;
195 do {
joshualitt0067ff52015-07-08 14:26:19 -0700196 center2.set(d->fRandom->nextUScalar1(), d->fRandom->nextUScalar1());
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000197 // If the circles are identical the factory will give us an empty shader.
198 // This will happen if we pick identical centers
199 } while (center1 == center2);
200
201 // Below makes sure that circle one is contained within circle two
202 // and both circles are touching on an edge
203 SkPoint diff = center2 - center1;
204 SkScalar diffLen = diff.length();
205 radius2 = radius1 + diffLen;
206
207 SkColor colors[kMaxRandomGradientColors];
208 SkScalar stopsArray[kMaxRandomGradientColors];
209 SkScalar* stops = stopsArray;
210 SkShader::TileMode tm;
joshualitt0067ff52015-07-08 14:26:19 -0700211 int colorCount = RandomGradientParams(d->fRandom, colors, &stops, &tm);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000212 SkAutoTUnref<SkShader> shader(SkGradientShader::CreateTwoPointConical(center1, radius1,
213 center2, radius2,
214 colors, stops, colorCount,
215 tm));
216 SkPaint paint;
joshualittb0a8a372014-09-23 09:50:21 -0700217 GrFragmentProcessor* fp;
bsalomon83d081a2014-07-08 09:56:10 -0700218 GrColor paintColor;
joshualitt0067ff52015-07-08 14:26:19 -0700219 SkAssertResult(shader->asFragmentProcessor(d->fContext, paint,
220 GrTest::TestMatrix(d->fRandom), NULL,
221 &paintColor, d->fShaderDataManager, &fp));
joshualittb0a8a372014-09-23 09:50:21 -0700222 return fp;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000223}
224
joshualitteb2a6762014-12-04 11:35:33 -0800225GLEdge2PtConicalEffect::GLEdge2PtConicalEffect(const GrProcessor&)
226 : fVSVaryingName(NULL)
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000227 , fFSVaryingName(NULL)
228 , fCachedRadius(-SK_ScalarMax)
229 , fCachedDiffRadius(-SK_ScalarMax) {}
230
joshualitt15988992014-10-09 15:04:05 -0700231void GLEdge2PtConicalEffect::emitCode(GrGLFPBuilder* builder,
joshualitt60030bc2014-11-25 14:21:55 -0800232 const GrFragmentProcessor& fp,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000233 const char* outputColor,
234 const char* inputColor,
235 const TransformedCoordsArray& coords,
236 const TextureSamplerArray& samplers) {
joshualitteb2a6762014-12-04 11:35:33 -0800237 const Edge2PtConicalEffect& ge = fp.cast<Edge2PtConicalEffect>();
joshualitt60030bc2014-11-25 14:21:55 -0800238 this->emitUniforms(builder, ge);
joshualitt30ba4362014-08-21 20:18:45 -0700239 fParamUni = builder->addUniformArray(GrGLProgramBuilder::kFragment_Visibility,
bsalomon422f56f2014-12-09 10:18:12 -0800240 kFloat_GrSLType, kDefault_GrSLPrecision,
241 "Conical2FSParams", 3);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000242
243 SkString cName("c");
244 SkString tName("t");
245 SkString p0; // start radius
246 SkString p1; // start radius squared
247 SkString p2; // difference in radii (r1 - r0)
248
249 builder->getUniformVariable(fParamUni).appendArrayAccess(0, &p0);
250 builder->getUniformVariable(fParamUni).appendArrayAccess(1, &p1);
251 builder->getUniformVariable(fParamUni).appendArrayAccess(2, &p2);
252
253 // We interpolate the linear component in coords[1].
joshualitt23e280d2014-09-18 12:26:38 -0700254 SkASSERT(coords[0].getType() == coords[1].getType());
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000255 const char* coords2D;
256 SkString bVar;
egdaniel29bee0f2015-04-29 11:54:42 -0700257 GrGLFragmentBuilder* fsBuilder = builder->getFragmentShaderBuilder();
joshualitt23e280d2014-09-18 12:26:38 -0700258 if (kVec3f_GrSLType == coords[0].getType()) {
joshualitt30ba4362014-08-21 20:18:45 -0700259 fsBuilder->codeAppendf("\tvec3 interpolants = vec3(%s.xy / %s.z, %s.x / %s.z);\n",
joshualitt49586be2014-09-16 08:21:41 -0700260 coords[0].c_str(), coords[0].c_str(), coords[1].c_str(),
261 coords[1].c_str());
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000262 coords2D = "interpolants.xy";
263 bVar = "interpolants.z";
264 } else {
265 coords2D = coords[0].c_str();
266 bVar.printf("%s.x", coords[1].c_str());
267 }
268
269 // output will default to transparent black (we simply won't write anything
270 // else to it if invalid, instead of discarding or returning prematurely)
joshualitt30ba4362014-08-21 20:18:45 -0700271 fsBuilder->codeAppendf("\t%s = vec4(0.0,0.0,0.0,0.0);\n", outputColor);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000272
273 // c = (x^2)+(y^2) - params[1]
joshualitt30ba4362014-08-21 20:18:45 -0700274 fsBuilder->codeAppendf("\tfloat %s = dot(%s, %s) - %s;\n",
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000275 cName.c_str(), coords2D, coords2D, p1.c_str());
276
277 // linear case: t = -c/b
joshualitt30ba4362014-08-21 20:18:45 -0700278 fsBuilder->codeAppendf("\tfloat %s = -(%s / %s);\n", tName.c_str(),
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000279 cName.c_str(), bVar.c_str());
280
281 // if r(t) > 0, then t will be the x coordinate
joshualitt30ba4362014-08-21 20:18:45 -0700282 fsBuilder->codeAppendf("\tif (%s * %s + %s > 0.0) {\n", tName.c_str(),
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000283 p2.c_str(), p0.c_str());
joshualitt30ba4362014-08-21 20:18:45 -0700284 fsBuilder->codeAppend("\t");
joshualitt60030bc2014-11-25 14:21:55 -0800285 this->emitColor(builder, ge, tName.c_str(), outputColor, inputColor, samplers);
joshualitt30ba4362014-08-21 20:18:45 -0700286 fsBuilder->codeAppend("\t}\n");
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000287}
288
kkinnunen7510b222014-07-30 00:04:16 -0700289void GLEdge2PtConicalEffect::setData(const GrGLProgramDataManager& pdman,
joshualittb0a8a372014-09-23 09:50:21 -0700290 const GrProcessor& processor) {
291 INHERITED::setData(pdman, processor);
292 const Edge2PtConicalEffect& data = processor.cast<Edge2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000293 SkScalar radius0 = data.radius();
294 SkScalar diffRadius = data.diffRadius();
295
296 if (fCachedRadius != radius0 ||
297 fCachedDiffRadius != diffRadius) {
298
299 float values[3] = {
300 SkScalarToFloat(radius0),
301 SkScalarToFloat(SkScalarMul(radius0, radius0)),
302 SkScalarToFloat(diffRadius)
303 };
304
kkinnunen7510b222014-07-30 00:04:16 -0700305 pdman.set1fv(fParamUni, 3, values);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000306 fCachedRadius = radius0;
307 fCachedDiffRadius = diffRadius;
308 }
309}
310
joshualittb0a8a372014-09-23 09:50:21 -0700311void GLEdge2PtConicalEffect::GenKey(const GrProcessor& processor,
jvanverthcfc18862015-04-28 08:48:20 -0700312 const GrGLSLCaps&, GrProcessorKeyBuilder* b) {
joshualittb0a8a372014-09-23 09:50:21 -0700313 b->add32(GenBaseGradientKey(processor));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000314}
315
316//////////////////////////////////////////////////////////////////////////////
317// Focal Conical Gradients
318//////////////////////////////////////////////////////////////////////////////
319
320static ConicalType set_matrix_focal_conical(const SkTwoPointConicalGradient& shader,
321 SkMatrix* invLMatrix, SkScalar* focalX) {
322 // Inverse of the current local matrix is passed in then,
323 // translate, scale, and rotate such that endCircle is unit circle on x-axis,
324 // and focal point is at the origin.
325 ConicalType conicalType;
326 const SkPoint& focal = shader.getStartCenter();
327 const SkPoint& centerEnd = shader.getEndCenter();
328 SkScalar radius = shader.getEndRadius();
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000329 SkScalar invRadius = 1.f / radius;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000330
331 SkMatrix matrix;
332
333 matrix.setTranslate(-centerEnd.fX, -centerEnd.fY);
334 matrix.postScale(invRadius, invRadius);
335
336 SkPoint focalTrans;
337 matrix.mapPoints(&focalTrans, &focal, 1);
338 *focalX = focalTrans.length();
339
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000340 if (0.f != *focalX) {
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000341 SkScalar invFocalX = SkScalarInvert(*focalX);
342 SkMatrix rot;
343 rot.setSinCos(-SkScalarMul(invFocalX, focalTrans.fY),
344 SkScalarMul(invFocalX, focalTrans.fX));
345 matrix.postConcat(rot);
346 }
347
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000348 matrix.postTranslate(-(*focalX), 0.f);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000349
350 // If the focal point is touching the edge of the circle it will
351 // cause a degenerate case that must be handled separately
egdaniel8405ef92014-06-09 11:57:28 -0700352 // kEdgeErrorTol = 5 * kErrorTol was picked after manual testing the
353 // stability trade off versus the linear approx used in the Edge Shader
354 if (SkScalarAbs(1.f - (*focalX)) < kEdgeErrorTol) {
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000355 return kEdge_ConicalType;
356 }
357
358 // Scale factor 1 / (1 - focalX * focalX)
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000359 SkScalar oneMinusF2 = 1.f - SkScalarMul(*focalX, *focalX);
reed80ea19c2015-05-12 10:37:34 -0700360 SkScalar s = SkScalarInvert(oneMinusF2);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000361
362
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000363 if (s >= 0.f) {
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000364 conicalType = kInside_ConicalType;
365 matrix.postScale(s, s * SkScalarSqrt(oneMinusF2));
366 } else {
367 conicalType = kOutside_ConicalType;
368 matrix.postScale(s, s);
369 }
370
371 invLMatrix->postConcat(matrix);
372
373 return conicalType;
374}
375
376//////////////////////////////////////////////////////////////////////////////
377
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000378class FocalOutside2PtConicalEffect : public GrGradientEffect {
379public:
380
joshualittb0a8a372014-09-23 09:50:21 -0700381 static GrFragmentProcessor* Create(GrContext* ctx,
joshualittb2456052015-07-08 09:36:59 -0700382 GrShaderDataManager* shaderDataManager,
joshualittb0a8a372014-09-23 09:50:21 -0700383 const SkTwoPointConicalGradient& shader,
384 const SkMatrix& matrix,
385 SkShader::TileMode tm,
386 SkScalar focalX) {
joshualittb2456052015-07-08 09:36:59 -0700387 return SkNEW_ARGS(FocalOutside2PtConicalEffect, (ctx, shaderDataManager, shader, matrix, tm,
388 focalX));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000389 }
390
391 virtual ~FocalOutside2PtConicalEffect() { }
392
mtklein36352bf2015-03-25 18:17:31 -0700393 const char* name() const override {
joshualitteb2a6762014-12-04 11:35:33 -0800394 return "Two-Point Conical Gradient Focal Outside";
395 }
396
jvanverthcfc18862015-04-28 08:48:20 -0700397 void getGLProcessorKey(const GrGLSLCaps&, GrProcessorKeyBuilder*) const override;
joshualitteb2a6762014-12-04 11:35:33 -0800398
mtklein36352bf2015-03-25 18:17:31 -0700399 GrGLFragmentProcessor* createGLInstance() const override;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000400
401 bool isFlipped() const { return fIsFlipped; }
402 SkScalar focal() const { return fFocalX; }
403
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000404private:
mtklein36352bf2015-03-25 18:17:31 -0700405 bool onIsEqual(const GrFragmentProcessor& sBase) const override {
joshualitt49586be2014-09-16 08:21:41 -0700406 const FocalOutside2PtConicalEffect& s = sBase.cast<FocalOutside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000407 return (INHERITED::onIsEqual(sBase) &&
408 this->fFocalX == s.fFocalX &&
409 this->fIsFlipped == s.fIsFlipped);
410 }
411
412 FocalOutside2PtConicalEffect(GrContext* ctx,
joshualittb2456052015-07-08 09:36:59 -0700413 GrShaderDataManager* shaderDataManager,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000414 const SkTwoPointConicalGradient& shader,
415 const SkMatrix& matrix,
416 SkShader::TileMode tm,
417 SkScalar focalX)
joshualittb2456052015-07-08 09:36:59 -0700418 : INHERITED(ctx, shaderDataManager, shader, matrix, tm)
419 , fFocalX(focalX)
420 , fIsFlipped(shader.isFlippedGrad()) {
joshualitteb2a6762014-12-04 11:35:33 -0800421 this->initClassID<FocalOutside2PtConicalEffect>();
422 }
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000423
joshualittb0a8a372014-09-23 09:50:21 -0700424 GR_DECLARE_FRAGMENT_PROCESSOR_TEST;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000425
426 SkScalar fFocalX;
427 bool fIsFlipped;
428
429 typedef GrGradientEffect INHERITED;
430};
431
432class GLFocalOutside2PtConicalEffect : public GrGLGradientEffect {
433public:
joshualitteb2a6762014-12-04 11:35:33 -0800434 GLFocalOutside2PtConicalEffect(const GrProcessor&);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000435 virtual ~GLFocalOutside2PtConicalEffect() { }
436
joshualitt15988992014-10-09 15:04:05 -0700437 virtual void emitCode(GrGLFPBuilder*,
joshualittb0a8a372014-09-23 09:50:21 -0700438 const GrFragmentProcessor&,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000439 const char* outputColor,
440 const char* inputColor,
441 const TransformedCoordsArray&,
mtklein36352bf2015-03-25 18:17:31 -0700442 const TextureSamplerArray&) override;
443 void setData(const GrGLProgramDataManager&, const GrProcessor&) override;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000444
jvanverthcfc18862015-04-28 08:48:20 -0700445 static void GenKey(const GrProcessor&, const GrGLSLCaps& caps, GrProcessorKeyBuilder* b);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000446
447protected:
448 UniformHandle fParamUni;
449
450 const char* fVSVaryingName;
451 const char* fFSVaryingName;
452
453 bool fIsFlipped;
454
455 // @{
456 /// Values last uploaded as uniforms
457
458 SkScalar fCachedFocal;
459
460 // @}
461
462private:
463 typedef GrGLGradientEffect INHERITED;
464
465};
466
jvanverthcfc18862015-04-28 08:48:20 -0700467void FocalOutside2PtConicalEffect::getGLProcessorKey(const GrGLSLCaps& caps,
joshualitteb2a6762014-12-04 11:35:33 -0800468 GrProcessorKeyBuilder* b) const {
469 GLFocalOutside2PtConicalEffect::GenKey(*this, caps, b);
470}
471
472GrGLFragmentProcessor* FocalOutside2PtConicalEffect::createGLInstance() const {
473 return SkNEW_ARGS(GLFocalOutside2PtConicalEffect, (*this));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000474}
475
joshualittb0a8a372014-09-23 09:50:21 -0700476GR_DEFINE_FRAGMENT_PROCESSOR_TEST(FocalOutside2PtConicalEffect);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000477
joshualitt01258472014-09-22 10:29:30 -0700478/*
479 * All Two point conical gradient test create functions may occasionally create edge case shaders
480 */
joshualitt0067ff52015-07-08 14:26:19 -0700481GrFragmentProcessor* FocalOutside2PtConicalEffect::TestCreate(GrProcessorTestData* d) {
482 SkPoint center1 = {d->fRandom->nextUScalar1(), d->fRandom->nextUScalar1()};
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000483 SkScalar radius1 = 0.f;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000484 SkPoint center2;
485 SkScalar radius2;
486 do {
joshualitt0067ff52015-07-08 14:26:19 -0700487 center2.set(d->fRandom->nextUScalar1(), d->fRandom->nextUScalar1());
skia.committer@gmail.comede0c5c2014-04-23 03:04:11 +0000488 // Need to make sure the centers are not the same or else focal point will be inside
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000489 } while (center1 == center2);
490 SkPoint diff = center2 - center1;
491 SkScalar diffLen = diff.length();
492 // Below makes sure that the focal point is not contained within circle two
joshualitt0067ff52015-07-08 14:26:19 -0700493 radius2 = d->fRandom->nextRangeF(0.f, diffLen);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000494
495 SkColor colors[kMaxRandomGradientColors];
496 SkScalar stopsArray[kMaxRandomGradientColors];
497 SkScalar* stops = stopsArray;
498 SkShader::TileMode tm;
joshualitt0067ff52015-07-08 14:26:19 -0700499 int colorCount = RandomGradientParams(d->fRandom, colors, &stops, &tm);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000500 SkAutoTUnref<SkShader> shader(SkGradientShader::CreateTwoPointConical(center1, radius1,
501 center2, radius2,
502 colors, stops, colorCount,
503 tm));
504 SkPaint paint;
joshualittb0a8a372014-09-23 09:50:21 -0700505 GrFragmentProcessor* effect;
bsalomon83d081a2014-07-08 09:56:10 -0700506 GrColor paintColor;
joshualitt8ca93e72015-07-08 06:51:43 -0700507 GrPaint grPaint;
joshualitt0067ff52015-07-08 14:26:19 -0700508 SkAssertResult(shader->asFragmentProcessor(d->fContext, paint,
509 GrTest::TestMatrix(d->fRandom), NULL,
510 &paintColor, d->fShaderDataManager,
joshualitt8ca93e72015-07-08 06:51:43 -0700511 &effect));
dandov9de5b512014-06-10 14:38:28 -0700512 return effect;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000513}
514
joshualitteb2a6762014-12-04 11:35:33 -0800515GLFocalOutside2PtConicalEffect::GLFocalOutside2PtConicalEffect(const GrProcessor& processor)
516 : fVSVaryingName(NULL)
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000517 , fFSVaryingName(NULL)
518 , fCachedFocal(SK_ScalarMax) {
joshualittb0a8a372014-09-23 09:50:21 -0700519 const FocalOutside2PtConicalEffect& data = processor.cast<FocalOutside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000520 fIsFlipped = data.isFlipped();
521}
522
joshualitt15988992014-10-09 15:04:05 -0700523void GLFocalOutside2PtConicalEffect::emitCode(GrGLFPBuilder* builder,
joshualitt60030bc2014-11-25 14:21:55 -0800524 const GrFragmentProcessor& fp,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000525 const char* outputColor,
526 const char* inputColor,
527 const TransformedCoordsArray& coords,
528 const TextureSamplerArray& samplers) {
joshualitteb2a6762014-12-04 11:35:33 -0800529 const FocalOutside2PtConicalEffect& ge = fp.cast<FocalOutside2PtConicalEffect>();
joshualitt60030bc2014-11-25 14:21:55 -0800530 this->emitUniforms(builder, ge);
joshualitt30ba4362014-08-21 20:18:45 -0700531 fParamUni = builder->addUniformArray(GrGLProgramBuilder::kFragment_Visibility,
bsalomon422f56f2014-12-09 10:18:12 -0800532 kFloat_GrSLType, kDefault_GrSLPrecision,
533 "Conical2FSParams", 2);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000534 SkString tName("t");
535 SkString p0; // focalX
536 SkString p1; // 1 - focalX * focalX
537
538 builder->getUniformVariable(fParamUni).appendArrayAccess(0, &p0);
539 builder->getUniformVariable(fParamUni).appendArrayAccess(1, &p1);
540
541 // if we have a vec3 from being in perspective, convert it to a vec2 first
egdaniel29bee0f2015-04-29 11:54:42 -0700542 GrGLFragmentBuilder* fsBuilder = builder->getFragmentShaderBuilder();
joshualitt30ba4362014-08-21 20:18:45 -0700543 SkString coords2DString = fsBuilder->ensureFSCoords2D(coords, 0);
skia.committer@gmail.comede0c5c2014-04-23 03:04:11 +0000544 const char* coords2D = coords2DString.c_str();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000545
546 // t = p.x * focal.x +/- sqrt(p.x^2 + (1 - focal.x^2) * p.y^2)
547
548 // output will default to transparent black (we simply won't write anything
549 // else to it if invalid, instead of discarding or returning prematurely)
joshualitt30ba4362014-08-21 20:18:45 -0700550 fsBuilder->codeAppendf("\t%s = vec4(0.0,0.0,0.0,0.0);\n", outputColor);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000551
joshualitt30ba4362014-08-21 20:18:45 -0700552 fsBuilder->codeAppendf("\tfloat xs = %s.x * %s.x;\n", coords2D, coords2D);
553 fsBuilder->codeAppendf("\tfloat ys = %s.y * %s.y;\n", coords2D, coords2D);
554 fsBuilder->codeAppendf("\tfloat d = xs + %s * ys;\n", p1.c_str());
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000555
556 // Must check to see if we flipped the circle order (to make sure start radius < end radius)
557 // If so we must also flip sign on sqrt
558 if (!fIsFlipped) {
joshualitt30ba4362014-08-21 20:18:45 -0700559 fsBuilder->codeAppendf("\tfloat %s = %s.x * %s + sqrt(d);\n", tName.c_str(),
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000560 coords2D, p0.c_str());
561 } else {
joshualitt30ba4362014-08-21 20:18:45 -0700562 fsBuilder->codeAppendf("\tfloat %s = %s.x * %s - sqrt(d);\n", tName.c_str(),
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000563 coords2D, p0.c_str());
564 }
565
joshualitt30ba4362014-08-21 20:18:45 -0700566 fsBuilder->codeAppendf("\tif (%s >= 0.0 && d >= 0.0) {\n", tName.c_str());
567 fsBuilder->codeAppend("\t\t");
joshualitt60030bc2014-11-25 14:21:55 -0800568 this->emitColor(builder, ge, tName.c_str(), outputColor, inputColor, samplers);
joshualitt30ba4362014-08-21 20:18:45 -0700569 fsBuilder->codeAppend("\t}\n");
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000570}
571
kkinnunen7510b222014-07-30 00:04:16 -0700572void GLFocalOutside2PtConicalEffect::setData(const GrGLProgramDataManager& pdman,
joshualittb0a8a372014-09-23 09:50:21 -0700573 const GrProcessor& processor) {
574 INHERITED::setData(pdman, processor);
575 const FocalOutside2PtConicalEffect& data = processor.cast<FocalOutside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000576 SkASSERT(data.isFlipped() == fIsFlipped);
577 SkScalar focal = data.focal();
578
579 if (fCachedFocal != focal) {
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000580 SkScalar oneMinus2F = 1.f - SkScalarMul(focal, focal);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000581
582 float values[2] = {
583 SkScalarToFloat(focal),
584 SkScalarToFloat(oneMinus2F),
585 };
586
kkinnunen7510b222014-07-30 00:04:16 -0700587 pdman.set1fv(fParamUni, 2, values);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000588 fCachedFocal = focal;
589 }
590}
591
joshualittb0a8a372014-09-23 09:50:21 -0700592void GLFocalOutside2PtConicalEffect::GenKey(const GrProcessor& processor,
jvanverthcfc18862015-04-28 08:48:20 -0700593 const GrGLSLCaps&, GrProcessorKeyBuilder* b) {
bsalomon63e99f72014-07-21 08:03:14 -0700594 uint32_t* key = b->add32n(2);
joshualittb0a8a372014-09-23 09:50:21 -0700595 key[0] = GenBaseGradientKey(processor);
596 key[1] = processor.cast<FocalOutside2PtConicalEffect>().isFlipped();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000597}
598
599//////////////////////////////////////////////////////////////////////////////
600
601class GLFocalInside2PtConicalEffect;
602
603class FocalInside2PtConicalEffect : public GrGradientEffect {
604public:
605
joshualittb0a8a372014-09-23 09:50:21 -0700606 static GrFragmentProcessor* Create(GrContext* ctx,
joshualittb2456052015-07-08 09:36:59 -0700607 GrShaderDataManager* shaderDataManager,
joshualittb0a8a372014-09-23 09:50:21 -0700608 const SkTwoPointConicalGradient& shader,
609 const SkMatrix& matrix,
610 SkShader::TileMode tm,
611 SkScalar focalX) {
joshualittb2456052015-07-08 09:36:59 -0700612 return SkNEW_ARGS(FocalInside2PtConicalEffect, (ctx, shaderDataManager, shader, matrix, tm,
613 focalX));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000614 }
615
616 virtual ~FocalInside2PtConicalEffect() {}
617
mtklein36352bf2015-03-25 18:17:31 -0700618 const char* name() const override {
joshualitteb2a6762014-12-04 11:35:33 -0800619 return "Two-Point Conical Gradient Focal Inside";
620 }
621
jvanverthcfc18862015-04-28 08:48:20 -0700622 void getGLProcessorKey(const GrGLSLCaps&, GrProcessorKeyBuilder*) const override;
joshualitteb2a6762014-12-04 11:35:33 -0800623
mtklein36352bf2015-03-25 18:17:31 -0700624 GrGLFragmentProcessor* createGLInstance() const override;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000625
626 SkScalar focal() const { return fFocalX; }
627
joshualittb0a8a372014-09-23 09:50:21 -0700628 typedef GLFocalInside2PtConicalEffect GLProcessor;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000629
630private:
mtklein36352bf2015-03-25 18:17:31 -0700631 bool onIsEqual(const GrFragmentProcessor& sBase) const override {
joshualitt49586be2014-09-16 08:21:41 -0700632 const FocalInside2PtConicalEffect& s = sBase.cast<FocalInside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000633 return (INHERITED::onIsEqual(sBase) &&
634 this->fFocalX == s.fFocalX);
635 }
636
637 FocalInside2PtConicalEffect(GrContext* ctx,
joshualittb2456052015-07-08 09:36:59 -0700638 GrShaderDataManager* shaderDataManager,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000639 const SkTwoPointConicalGradient& shader,
640 const SkMatrix& matrix,
641 SkShader::TileMode tm,
642 SkScalar focalX)
joshualittb2456052015-07-08 09:36:59 -0700643 : INHERITED(ctx, shaderDataManager, shader, matrix, tm), fFocalX(focalX) {
joshualitteb2a6762014-12-04 11:35:33 -0800644 this->initClassID<FocalInside2PtConicalEffect>();
645 }
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000646
joshualittb0a8a372014-09-23 09:50:21 -0700647 GR_DECLARE_FRAGMENT_PROCESSOR_TEST;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000648
649 SkScalar fFocalX;
650
651 typedef GrGradientEffect INHERITED;
652};
653
654class GLFocalInside2PtConicalEffect : public GrGLGradientEffect {
655public:
joshualitteb2a6762014-12-04 11:35:33 -0800656 GLFocalInside2PtConicalEffect(const GrProcessor&);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000657 virtual ~GLFocalInside2PtConicalEffect() {}
658
joshualitt15988992014-10-09 15:04:05 -0700659 virtual void emitCode(GrGLFPBuilder*,
joshualittb0a8a372014-09-23 09:50:21 -0700660 const GrFragmentProcessor&,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000661 const char* outputColor,
662 const char* inputColor,
663 const TransformedCoordsArray&,
mtklein36352bf2015-03-25 18:17:31 -0700664 const TextureSamplerArray&) override;
665 void setData(const GrGLProgramDataManager&, const GrProcessor&) override;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000666
jvanverthcfc18862015-04-28 08:48:20 -0700667 static void GenKey(const GrProcessor&, const GrGLSLCaps& caps, GrProcessorKeyBuilder* b);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000668
669protected:
670 UniformHandle fFocalUni;
671
672 const char* fVSVaryingName;
673 const char* fFSVaryingName;
674
675 // @{
676 /// Values last uploaded as uniforms
677
678 SkScalar fCachedFocal;
679
680 // @}
681
682private:
683 typedef GrGLGradientEffect INHERITED;
684
685};
686
jvanverthcfc18862015-04-28 08:48:20 -0700687void FocalInside2PtConicalEffect::getGLProcessorKey(const GrGLSLCaps& caps,
joshualitteb2a6762014-12-04 11:35:33 -0800688 GrProcessorKeyBuilder* b) const {
689 GLFocalInside2PtConicalEffect::GenKey(*this, caps, b);
690}
691
692GrGLFragmentProcessor* FocalInside2PtConicalEffect::createGLInstance() const {
693 return SkNEW_ARGS(GLFocalInside2PtConicalEffect, (*this));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000694}
695
joshualittb0a8a372014-09-23 09:50:21 -0700696GR_DEFINE_FRAGMENT_PROCESSOR_TEST(FocalInside2PtConicalEffect);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000697
joshualitt01258472014-09-22 10:29:30 -0700698/*
699 * All Two point conical gradient test create functions may occasionally create edge case shaders
700 */
joshualitt0067ff52015-07-08 14:26:19 -0700701GrFragmentProcessor* FocalInside2PtConicalEffect::TestCreate(GrProcessorTestData* d) {
702 SkPoint center1 = {d->fRandom->nextUScalar1(), d->fRandom->nextUScalar1()};
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000703 SkScalar radius1 = 0.f;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000704 SkPoint center2;
705 SkScalar radius2;
706 do {
joshualitt0067ff52015-07-08 14:26:19 -0700707 center2.set(d->fRandom->nextUScalar1(), d->fRandom->nextUScalar1());
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000708 // Below makes sure radius2 is larger enouch such that the focal point
709 // is inside the end circle
joshualitt0067ff52015-07-08 14:26:19 -0700710 SkScalar increase = d->fRandom->nextUScalar1();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000711 SkPoint diff = center2 - center1;
712 SkScalar diffLen = diff.length();
713 radius2 = diffLen + increase;
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000714 // If the circles are identical the factory will give us an empty shader.
715 } while (radius1 == radius2 && center1 == center2);
716
717 SkColor colors[kMaxRandomGradientColors];
718 SkScalar stopsArray[kMaxRandomGradientColors];
719 SkScalar* stops = stopsArray;
720 SkShader::TileMode tm;
joshualitt0067ff52015-07-08 14:26:19 -0700721 int colorCount = RandomGradientParams(d->fRandom, colors, &stops, &tm);
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000722 SkAutoTUnref<SkShader> shader(SkGradientShader::CreateTwoPointConical(center1, radius1,
723 center2, radius2,
724 colors, stops, colorCount,
725 tm));
726 SkPaint paint;
bsalomon83d081a2014-07-08 09:56:10 -0700727 GrColor paintColor;
joshualittb0a8a372014-09-23 09:50:21 -0700728 GrFragmentProcessor* fp;
joshualitt0067ff52015-07-08 14:26:19 -0700729 SkAssertResult(shader->asFragmentProcessor(d->fContext, paint,
730 GrTest::TestMatrix(d->fRandom), NULL,
731 &paintColor, d->fShaderDataManager, &fp));
joshualittb0a8a372014-09-23 09:50:21 -0700732 return fp;
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000733}
734
joshualitteb2a6762014-12-04 11:35:33 -0800735GLFocalInside2PtConicalEffect::GLFocalInside2PtConicalEffect(const GrProcessor&)
736 : fVSVaryingName(NULL)
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000737 , fFSVaryingName(NULL)
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000738 , fCachedFocal(SK_ScalarMax) {}
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000739
joshualitt15988992014-10-09 15:04:05 -0700740void GLFocalInside2PtConicalEffect::emitCode(GrGLFPBuilder* builder,
joshualitt60030bc2014-11-25 14:21:55 -0800741 const GrFragmentProcessor& fp,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000742 const char* outputColor,
743 const char* inputColor,
744 const TransformedCoordsArray& coords,
745 const TextureSamplerArray& samplers) {
joshualitteb2a6762014-12-04 11:35:33 -0800746 const FocalInside2PtConicalEffect& ge = fp.cast<FocalInside2PtConicalEffect>();
joshualitt60030bc2014-11-25 14:21:55 -0800747 this->emitUniforms(builder, ge);
joshualitt30ba4362014-08-21 20:18:45 -0700748 fFocalUni = builder->addUniform(GrGLProgramBuilder::kFragment_Visibility,
bsalomon422f56f2014-12-09 10:18:12 -0800749 kFloat_GrSLType, kDefault_GrSLPrecision,
750 "Conical2FSParams");
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000751 SkString tName("t");
752
753 // this is the distance along x-axis from the end center to focal point in
754 // transformed coordinates
755 GrGLShaderVar focal = builder->getUniformVariable(fFocalUni);
756
757 // if we have a vec3 from being in perspective, convert it to a vec2 first
egdaniel29bee0f2015-04-29 11:54:42 -0700758 GrGLFragmentBuilder* fsBuilder = builder->getFragmentShaderBuilder();
joshualitt30ba4362014-08-21 20:18:45 -0700759 SkString coords2DString = fsBuilder->ensureFSCoords2D(coords, 0);
skia.committer@gmail.comede0c5c2014-04-23 03:04:11 +0000760 const char* coords2D = coords2DString.c_str();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000761
762 // t = p.x * focalX + length(p)
joshualitt30ba4362014-08-21 20:18:45 -0700763 fsBuilder->codeAppendf("\tfloat %s = %s.x * %s + length(%s);\n", tName.c_str(),
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000764 coords2D, focal.c_str(), coords2D);
765
joshualitt60030bc2014-11-25 14:21:55 -0800766 this->emitColor(builder, ge, tName.c_str(), outputColor, inputColor, samplers);
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000767}
768
kkinnunen7510b222014-07-30 00:04:16 -0700769void GLFocalInside2PtConicalEffect::setData(const GrGLProgramDataManager& pdman,
joshualittb0a8a372014-09-23 09:50:21 -0700770 const GrProcessor& processor) {
771 INHERITED::setData(pdman, processor);
772 const FocalInside2PtConicalEffect& data = processor.cast<FocalInside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000773 SkScalar focal = data.focal();
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000774
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000775 if (fCachedFocal != focal) {
kkinnunen7510b222014-07-30 00:04:16 -0700776 pdman.set1f(fFocalUni, SkScalarToFloat(focal));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000777 fCachedFocal = focal;
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000778 }
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000779}
780
joshualittb0a8a372014-09-23 09:50:21 -0700781void GLFocalInside2PtConicalEffect::GenKey(const GrProcessor& processor,
jvanverthcfc18862015-04-28 08:48:20 -0700782 const GrGLSLCaps&, GrProcessorKeyBuilder* b) {
joshualittb0a8a372014-09-23 09:50:21 -0700783 b->add32(GenBaseGradientKey(processor));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000784}
785
786//////////////////////////////////////////////////////////////////////////////
787// Circle Conical Gradients
788//////////////////////////////////////////////////////////////////////////////
789
790struct CircleConicalInfo {
791 SkPoint fCenterEnd;
792 SkScalar fA;
793 SkScalar fB;
794 SkScalar fC;
795};
796
797// Returns focal distance along x-axis in transformed coords
798static ConicalType set_matrix_circle_conical(const SkTwoPointConicalGradient& shader,
799 SkMatrix* invLMatrix, CircleConicalInfo* info) {
800 // Inverse of the current local matrix is passed in then,
801 // translate and scale such that start circle is on the origin and has radius 1
802 const SkPoint& centerStart = shader.getStartCenter();
803 const SkPoint& centerEnd = shader.getEndCenter();
804 SkScalar radiusStart = shader.getStartRadius();
805 SkScalar radiusEnd = shader.getEndRadius();
806
807 SkMatrix matrix;
808
809 matrix.setTranslate(-centerStart.fX, -centerStart.fY);
810
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000811 SkScalar invStartRad = 1.f / radiusStart;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000812 matrix.postScale(invStartRad, invStartRad);
813
814 radiusEnd /= radiusStart;
815
816 SkPoint centerEndTrans;
817 matrix.mapPoints(&centerEndTrans, &centerEnd, 1);
818
819 SkScalar A = centerEndTrans.fX * centerEndTrans.fX + centerEndTrans.fY * centerEndTrans.fY
820 - radiusEnd * radiusEnd + 2 * radiusEnd - 1;
821
822 // Check to see if start circle is inside end circle with edges touching.
823 // If touching we return that it is of kEdge_ConicalType, and leave the matrix setting
egdaniel8405ef92014-06-09 11:57:28 -0700824 // to the edge shader. kEdgeErrorTol = 5 * kErrorTol was picked after manual testing
825 // so that C = 1 / A is stable, and the linear approximation used in the Edge shader is
826 // still accurate.
827 if (SkScalarAbs(A) < kEdgeErrorTol) {
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000828 return kEdge_ConicalType;
829 }
830
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000831 SkScalar C = 1.f / A;
832 SkScalar B = (radiusEnd - 1.f) * C;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000833
834 matrix.postScale(C, C);
835
836 invLMatrix->postConcat(matrix);
837
838 info->fCenterEnd = centerEndTrans;
839 info->fA = A;
840 info->fB = B;
841 info->fC = C;
842
843 // if A ends up being negative, the start circle is contained completely inside the end cirlce
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000844 if (A < 0.f) {
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000845 return kInside_ConicalType;
846 }
847 return kOutside_ConicalType;
848}
849
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000850class CircleInside2PtConicalEffect : public GrGradientEffect {
851public:
852
joshualittb0a8a372014-09-23 09:50:21 -0700853 static GrFragmentProcessor* Create(GrContext* ctx,
joshualittb2456052015-07-08 09:36:59 -0700854 GrShaderDataManager* shaderDataManager,
joshualittb0a8a372014-09-23 09:50:21 -0700855 const SkTwoPointConicalGradient& shader,
856 const SkMatrix& matrix,
857 SkShader::TileMode tm,
858 const CircleConicalInfo& info) {
joshualittb2456052015-07-08 09:36:59 -0700859 return SkNEW_ARGS(CircleInside2PtConicalEffect, (ctx, shaderDataManager, shader, matrix, tm,
860 info));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000861 }
862
863 virtual ~CircleInside2PtConicalEffect() {}
864
mtklein36352bf2015-03-25 18:17:31 -0700865 const char* name() const override { return "Two-Point Conical Gradient Inside"; }
joshualitteb2a6762014-12-04 11:35:33 -0800866
jvanverthcfc18862015-04-28 08:48:20 -0700867 virtual void getGLProcessorKey(const GrGLSLCaps& caps,
mtklein36352bf2015-03-25 18:17:31 -0700868 GrProcessorKeyBuilder* b) const override;
joshualitteb2a6762014-12-04 11:35:33 -0800869
mtklein36352bf2015-03-25 18:17:31 -0700870 GrGLFragmentProcessor* createGLInstance() const override;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000871
872 SkScalar centerX() const { return fInfo.fCenterEnd.fX; }
873 SkScalar centerY() const { return fInfo.fCenterEnd.fY; }
874 SkScalar A() const { return fInfo.fA; }
875 SkScalar B() const { return fInfo.fB; }
876 SkScalar C() const { return fInfo.fC; }
877
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000878private:
mtklein36352bf2015-03-25 18:17:31 -0700879 bool onIsEqual(const GrFragmentProcessor& sBase) const override {
joshualitt49586be2014-09-16 08:21:41 -0700880 const CircleInside2PtConicalEffect& s = sBase.cast<CircleInside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000881 return (INHERITED::onIsEqual(sBase) &&
882 this->fInfo.fCenterEnd == s.fInfo.fCenterEnd &&
883 this->fInfo.fA == s.fInfo.fA &&
884 this->fInfo.fB == s.fInfo.fB &&
885 this->fInfo.fC == s.fInfo.fC);
886 }
887
888 CircleInside2PtConicalEffect(GrContext* ctx,
joshualittb2456052015-07-08 09:36:59 -0700889 GrShaderDataManager* shaderDataManager,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000890 const SkTwoPointConicalGradient& shader,
891 const SkMatrix& matrix,
892 SkShader::TileMode tm,
893 const CircleConicalInfo& info)
joshualittb2456052015-07-08 09:36:59 -0700894 : INHERITED(ctx, shaderDataManager, shader, matrix, tm), fInfo(info) {
joshualitteb2a6762014-12-04 11:35:33 -0800895 this->initClassID<CircleInside2PtConicalEffect>();
896 }
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000897
joshualittb0a8a372014-09-23 09:50:21 -0700898 GR_DECLARE_FRAGMENT_PROCESSOR_TEST;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000899
900 const CircleConicalInfo fInfo;
901
902 typedef GrGradientEffect INHERITED;
903};
904
905class GLCircleInside2PtConicalEffect : public GrGLGradientEffect {
906public:
joshualitteb2a6762014-12-04 11:35:33 -0800907 GLCircleInside2PtConicalEffect(const GrProcessor&);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000908 virtual ~GLCircleInside2PtConicalEffect() {}
909
joshualitt15988992014-10-09 15:04:05 -0700910 virtual void emitCode(GrGLFPBuilder*,
joshualittb0a8a372014-09-23 09:50:21 -0700911 const GrFragmentProcessor&,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000912 const char* outputColor,
913 const char* inputColor,
914 const TransformedCoordsArray&,
mtklein36352bf2015-03-25 18:17:31 -0700915 const TextureSamplerArray&) override;
916 void setData(const GrGLProgramDataManager&, const GrProcessor&) override;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000917
jvanverthcfc18862015-04-28 08:48:20 -0700918 static void GenKey(const GrProcessor&, const GrGLSLCaps& caps, GrProcessorKeyBuilder* b);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000919
920protected:
921 UniformHandle fCenterUni;
922 UniformHandle fParamUni;
923
924 const char* fVSVaryingName;
925 const char* fFSVaryingName;
926
927 // @{
928 /// Values last uploaded as uniforms
929
930 SkScalar fCachedCenterX;
931 SkScalar fCachedCenterY;
932 SkScalar fCachedA;
933 SkScalar fCachedB;
934 SkScalar fCachedC;
935
936 // @}
937
938private:
939 typedef GrGLGradientEffect INHERITED;
940
941};
942
jvanverthcfc18862015-04-28 08:48:20 -0700943void CircleInside2PtConicalEffect::getGLProcessorKey(const GrGLSLCaps& caps,
joshualitteb2a6762014-12-04 11:35:33 -0800944 GrProcessorKeyBuilder* b) const {
945 GLCircleInside2PtConicalEffect::GenKey(*this, caps, b);
946}
947
948GrGLFragmentProcessor* CircleInside2PtConicalEffect::createGLInstance() const {
949 return SkNEW_ARGS(GLCircleInside2PtConicalEffect, (*this));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000950}
951
joshualittb0a8a372014-09-23 09:50:21 -0700952GR_DEFINE_FRAGMENT_PROCESSOR_TEST(CircleInside2PtConicalEffect);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000953
joshualitt01258472014-09-22 10:29:30 -0700954/*
955 * All Two point conical gradient test create functions may occasionally create edge case shaders
956 */
joshualitt0067ff52015-07-08 14:26:19 -0700957GrFragmentProcessor*
958CircleInside2PtConicalEffect::TestCreate(GrProcessorTestData* d) {
959 SkPoint center1 = {d->fRandom->nextUScalar1(), d->fRandom->nextUScalar1()};
960 SkScalar radius1 = d->fRandom->nextUScalar1() + 0.0001f; // make sure radius1 != 0
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000961 SkPoint center2;
962 SkScalar radius2;
963 do {
joshualitt0067ff52015-07-08 14:26:19 -0700964 center2.set(d->fRandom->nextUScalar1(), d->fRandom->nextUScalar1());
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000965 // Below makes sure that circle one is contained within circle two
joshualitt0067ff52015-07-08 14:26:19 -0700966 SkScalar increase = d->fRandom->nextUScalar1();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000967 SkPoint diff = center2 - center1;
968 SkScalar diffLen = diff.length();
969 radius2 = radius1 + diffLen + increase;
970 // If the circles are identical the factory will give us an empty shader.
971 } while (radius1 == radius2 && center1 == center2);
972
973 SkColor colors[kMaxRandomGradientColors];
974 SkScalar stopsArray[kMaxRandomGradientColors];
975 SkScalar* stops = stopsArray;
976 SkShader::TileMode tm;
joshualitt0067ff52015-07-08 14:26:19 -0700977 int colorCount = RandomGradientParams(d->fRandom, colors, &stops, &tm);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000978 SkAutoTUnref<SkShader> shader(SkGradientShader::CreateTwoPointConical(center1, radius1,
979 center2, radius2,
980 colors, stops, colorCount,
981 tm));
982 SkPaint paint;
bsalomon83d081a2014-07-08 09:56:10 -0700983 GrColor paintColor;
joshualitt8ca93e72015-07-08 06:51:43 -0700984 GrFragmentProcessor* fp;
joshualitt0067ff52015-07-08 14:26:19 -0700985 SkAssertResult(shader->asFragmentProcessor(d->fContext, paint,
986 GrTest::TestMatrix(d->fRandom), NULL,
987 &paintColor, d->fShaderDataManager, &fp));
joshualitt8ca93e72015-07-08 06:51:43 -0700988 return fp;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000989}
990
joshualitteb2a6762014-12-04 11:35:33 -0800991GLCircleInside2PtConicalEffect::GLCircleInside2PtConicalEffect(const GrProcessor& processor)
992 : fVSVaryingName(NULL)
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000993 , fFSVaryingName(NULL)
994 , fCachedCenterX(SK_ScalarMax)
995 , fCachedCenterY(SK_ScalarMax)
996 , fCachedA(SK_ScalarMax)
997 , fCachedB(SK_ScalarMax)
998 , fCachedC(SK_ScalarMax) {}
999
joshualitt15988992014-10-09 15:04:05 -07001000void GLCircleInside2PtConicalEffect::emitCode(GrGLFPBuilder* builder,
joshualitt60030bc2014-11-25 14:21:55 -08001001 const GrFragmentProcessor& fp,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001002 const char* outputColor,
1003 const char* inputColor,
1004 const TransformedCoordsArray& coords,
1005 const TextureSamplerArray& samplers) {
joshualitteb2a6762014-12-04 11:35:33 -08001006 const CircleInside2PtConicalEffect& ge = fp.cast<CircleInside2PtConicalEffect>();
joshualitt60030bc2014-11-25 14:21:55 -08001007 this->emitUniforms(builder, ge);
joshualitt30ba4362014-08-21 20:18:45 -07001008 fCenterUni = builder->addUniform(GrGLProgramBuilder::kFragment_Visibility,
bsalomon422f56f2014-12-09 10:18:12 -08001009 kVec2f_GrSLType, kDefault_GrSLPrecision,
1010 "Conical2FSCenter");
joshualitt30ba4362014-08-21 20:18:45 -07001011 fParamUni = builder->addUniform(GrGLProgramBuilder::kFragment_Visibility,
bsalomon422f56f2014-12-09 10:18:12 -08001012 kVec3f_GrSLType, kDefault_GrSLPrecision,
1013 "Conical2FSParams");
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001014 SkString tName("t");
1015
1016 GrGLShaderVar center = builder->getUniformVariable(fCenterUni);
1017 // params.x = A
1018 // params.y = B
1019 // params.z = C
1020 GrGLShaderVar params = builder->getUniformVariable(fParamUni);
1021
1022 // if we have a vec3 from being in perspective, convert it to a vec2 first
egdaniel29bee0f2015-04-29 11:54:42 -07001023 GrGLFragmentBuilder* fsBuilder = builder->getFragmentShaderBuilder();
joshualitt30ba4362014-08-21 20:18:45 -07001024 SkString coords2DString = fsBuilder->ensureFSCoords2D(coords, 0);
skia.committer@gmail.comede0c5c2014-04-23 03:04:11 +00001025 const char* coords2D = coords2DString.c_str();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001026
1027 // p = coords2D
1028 // e = center end
1029 // r = radius end
1030 // A = dot(e, e) - r^2 + 2 * r - 1
1031 // B = (r -1) / A
1032 // C = 1 / A
1033 // d = dot(e, p) + B
1034 // t = d +/- sqrt(d^2 - A * dot(p, p) + C)
joshualitt30ba4362014-08-21 20:18:45 -07001035 fsBuilder->codeAppendf("\tfloat pDotp = dot(%s, %s);\n", coords2D, coords2D);
joshualitt49586be2014-09-16 08:21:41 -07001036 fsBuilder->codeAppendf("\tfloat d = dot(%s, %s) + %s.y;\n", coords2D, center.c_str(),
1037 params.c_str());
joshualitt30ba4362014-08-21 20:18:45 -07001038 fsBuilder->codeAppendf("\tfloat %s = d + sqrt(d * d - %s.x * pDotp + %s.z);\n",
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001039 tName.c_str(), params.c_str(), params.c_str());
1040
joshualitt60030bc2014-11-25 14:21:55 -08001041 this->emitColor(builder, ge, tName.c_str(), outputColor, inputColor, samplers);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001042}
1043
kkinnunen7510b222014-07-30 00:04:16 -07001044void GLCircleInside2PtConicalEffect::setData(const GrGLProgramDataManager& pdman,
joshualittb0a8a372014-09-23 09:50:21 -07001045 const GrProcessor& processor) {
1046 INHERITED::setData(pdman, processor);
1047 const CircleInside2PtConicalEffect& data = processor.cast<CircleInside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001048 SkScalar centerX = data.centerX();
1049 SkScalar centerY = data.centerY();
1050 SkScalar A = data.A();
1051 SkScalar B = data.B();
1052 SkScalar C = data.C();
1053
1054 if (fCachedCenterX != centerX || fCachedCenterY != centerY ||
1055 fCachedA != A || fCachedB != B || fCachedC != C) {
1056
kkinnunen7510b222014-07-30 00:04:16 -07001057 pdman.set2f(fCenterUni, SkScalarToFloat(centerX), SkScalarToFloat(centerY));
1058 pdman.set3f(fParamUni, SkScalarToFloat(A), SkScalarToFloat(B), SkScalarToFloat(C));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001059
1060 fCachedCenterX = centerX;
1061 fCachedCenterY = centerY;
1062 fCachedA = A;
1063 fCachedB = B;
1064 fCachedC = C;
1065 }
1066}
1067
joshualittb0a8a372014-09-23 09:50:21 -07001068void GLCircleInside2PtConicalEffect::GenKey(const GrProcessor& processor,
jvanverthcfc18862015-04-28 08:48:20 -07001069 const GrGLSLCaps&, GrProcessorKeyBuilder* b) {
joshualittb0a8a372014-09-23 09:50:21 -07001070 b->add32(GenBaseGradientKey(processor));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001071}
1072
1073//////////////////////////////////////////////////////////////////////////////
1074
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001075class CircleOutside2PtConicalEffect : public GrGradientEffect {
1076public:
1077
joshualittb0a8a372014-09-23 09:50:21 -07001078 static GrFragmentProcessor* Create(GrContext* ctx,
joshualittb2456052015-07-08 09:36:59 -07001079 GrShaderDataManager* shaderDataManager,
joshualittb0a8a372014-09-23 09:50:21 -07001080 const SkTwoPointConicalGradient& shader,
1081 const SkMatrix& matrix,
1082 SkShader::TileMode tm,
1083 const CircleConicalInfo& info) {
joshualittb2456052015-07-08 09:36:59 -07001084 return SkNEW_ARGS(CircleOutside2PtConicalEffect, (ctx, shaderDataManager, shader, matrix,
1085 tm, info));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001086 }
1087
1088 virtual ~CircleOutside2PtConicalEffect() {}
1089
mtklein36352bf2015-03-25 18:17:31 -07001090 const char* name() const override { return "Two-Point Conical Gradient Outside"; }
joshualitteb2a6762014-12-04 11:35:33 -08001091
jvanverthcfc18862015-04-28 08:48:20 -07001092 void getGLProcessorKey(const GrGLSLCaps&, GrProcessorKeyBuilder*) const override;
joshualitteb2a6762014-12-04 11:35:33 -08001093
mtklein36352bf2015-03-25 18:17:31 -07001094 GrGLFragmentProcessor* createGLInstance() const override;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001095
1096 SkScalar centerX() const { return fInfo.fCenterEnd.fX; }
1097 SkScalar centerY() const { return fInfo.fCenterEnd.fY; }
1098 SkScalar A() const { return fInfo.fA; }
1099 SkScalar B() const { return fInfo.fB; }
1100 SkScalar C() const { return fInfo.fC; }
1101 SkScalar tLimit() const { return fTLimit; }
1102 bool isFlipped() const { return fIsFlipped; }
1103
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001104private:
mtklein36352bf2015-03-25 18:17:31 -07001105 bool onIsEqual(const GrFragmentProcessor& sBase) const override {
joshualitt49586be2014-09-16 08:21:41 -07001106 const CircleOutside2PtConicalEffect& s = sBase.cast<CircleOutside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001107 return (INHERITED::onIsEqual(sBase) &&
1108 this->fInfo.fCenterEnd == s.fInfo.fCenterEnd &&
1109 this->fInfo.fA == s.fInfo.fA &&
1110 this->fInfo.fB == s.fInfo.fB &&
1111 this->fInfo.fC == s.fInfo.fC &&
1112 this->fTLimit == s.fTLimit &&
1113 this->fIsFlipped == s.fIsFlipped);
1114 }
1115
1116 CircleOutside2PtConicalEffect(GrContext* ctx,
joshualittb2456052015-07-08 09:36:59 -07001117 GrShaderDataManager* shaderDataManager,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001118 const SkTwoPointConicalGradient& shader,
1119 const SkMatrix& matrix,
1120 SkShader::TileMode tm,
1121 const CircleConicalInfo& info)
joshualittb2456052015-07-08 09:36:59 -07001122 : INHERITED(ctx, shaderDataManager, shader, matrix, tm), fInfo(info) {
joshualitteb2a6762014-12-04 11:35:33 -08001123 this->initClassID<CircleOutside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001124 if (shader.getStartRadius() != shader.getEndRadius()) {
reed80ea19c2015-05-12 10:37:34 -07001125 fTLimit = shader.getStartRadius() / (shader.getStartRadius() - shader.getEndRadius());
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001126 } else {
1127 fTLimit = SK_ScalarMin;
1128 }
1129
1130 fIsFlipped = shader.isFlippedGrad();
1131 }
1132
joshualittb0a8a372014-09-23 09:50:21 -07001133 GR_DECLARE_FRAGMENT_PROCESSOR_TEST;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001134
1135 const CircleConicalInfo fInfo;
1136 SkScalar fTLimit;
1137 bool fIsFlipped;
1138
1139 typedef GrGradientEffect INHERITED;
1140};
1141
1142class GLCircleOutside2PtConicalEffect : public GrGLGradientEffect {
1143public:
joshualitteb2a6762014-12-04 11:35:33 -08001144 GLCircleOutside2PtConicalEffect(const GrProcessor&);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001145 virtual ~GLCircleOutside2PtConicalEffect() {}
1146
joshualitt15988992014-10-09 15:04:05 -07001147 virtual void emitCode(GrGLFPBuilder*,
joshualittb0a8a372014-09-23 09:50:21 -07001148 const GrFragmentProcessor&,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001149 const char* outputColor,
1150 const char* inputColor,
1151 const TransformedCoordsArray&,
mtklein36352bf2015-03-25 18:17:31 -07001152 const TextureSamplerArray&) override;
1153 void setData(const GrGLProgramDataManager&, const GrProcessor&) override;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001154
jvanverthcfc18862015-04-28 08:48:20 -07001155 static void GenKey(const GrProcessor&, const GrGLSLCaps& caps, GrProcessorKeyBuilder* b);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001156
1157protected:
1158 UniformHandle fCenterUni;
1159 UniformHandle fParamUni;
1160
1161 const char* fVSVaryingName;
1162 const char* fFSVaryingName;
1163
1164 bool fIsFlipped;
1165
1166 // @{
1167 /// Values last uploaded as uniforms
1168
1169 SkScalar fCachedCenterX;
1170 SkScalar fCachedCenterY;
1171 SkScalar fCachedA;
1172 SkScalar fCachedB;
1173 SkScalar fCachedC;
1174 SkScalar fCachedTLimit;
1175
1176 // @}
1177
1178private:
1179 typedef GrGLGradientEffect INHERITED;
1180
1181};
1182
jvanverthcfc18862015-04-28 08:48:20 -07001183void CircleOutside2PtConicalEffect::getGLProcessorKey(const GrGLSLCaps& caps,
joshualitteb2a6762014-12-04 11:35:33 -08001184 GrProcessorKeyBuilder* b) const {
1185 GLCircleOutside2PtConicalEffect::GenKey(*this, caps, b);
1186}
1187
1188GrGLFragmentProcessor* CircleOutside2PtConicalEffect::createGLInstance() const {
1189 return SkNEW_ARGS(GLCircleOutside2PtConicalEffect, (*this));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001190}
1191
joshualittb0a8a372014-09-23 09:50:21 -07001192GR_DEFINE_FRAGMENT_PROCESSOR_TEST(CircleOutside2PtConicalEffect);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001193
joshualitt01258472014-09-22 10:29:30 -07001194/*
1195 * All Two point conical gradient test create functions may occasionally create edge case shaders
1196 */
joshualitt0067ff52015-07-08 14:26:19 -07001197GrFragmentProcessor* CircleOutside2PtConicalEffect::TestCreate(GrProcessorTestData* d) {
1198 SkPoint center1 = {d->fRandom->nextUScalar1(), d->fRandom->nextUScalar1()};
1199 SkScalar radius1 = d->fRandom->nextUScalar1() + 0.0001f; // make sure radius1 != 0
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001200 SkPoint center2;
1201 SkScalar radius2;
1202 SkScalar diffLen;
1203 do {
joshualitt0067ff52015-07-08 14:26:19 -07001204 center2.set(d->fRandom->nextUScalar1(), d->fRandom->nextUScalar1());
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001205 // If the circles share a center than we can't be in the outside case
1206 } while (center1 == center2);
joshualitt01258472014-09-22 10:29:30 -07001207 SkPoint diff = center2 - center1;
1208 diffLen = diff.length();
1209 // Below makes sure that circle one is not contained within circle two
1210 // and have radius2 >= radius to match sorting on cpu side
joshualitt0067ff52015-07-08 14:26:19 -07001211 radius2 = radius1 + d->fRandom->nextRangeF(0.f, diffLen);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001212
1213 SkColor colors[kMaxRandomGradientColors];
1214 SkScalar stopsArray[kMaxRandomGradientColors];
1215 SkScalar* stops = stopsArray;
1216 SkShader::TileMode tm;
joshualitt0067ff52015-07-08 14:26:19 -07001217 int colorCount = RandomGradientParams(d->fRandom, colors, &stops, &tm);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001218 SkAutoTUnref<SkShader> shader(SkGradientShader::CreateTwoPointConical(center1, radius1,
1219 center2, radius2,
1220 colors, stops, colorCount,
1221 tm));
1222 SkPaint paint;
bsalomon83d081a2014-07-08 09:56:10 -07001223 GrColor paintColor;
joshualitt8ca93e72015-07-08 06:51:43 -07001224 GrFragmentProcessor* fp;
joshualitt0067ff52015-07-08 14:26:19 -07001225 SkAssertResult(shader->asFragmentProcessor(d->fContext, paint,
1226 GrTest::TestMatrix(d->fRandom), NULL,
1227 &paintColor, d->fShaderDataManager, &fp));
joshualitt8ca93e72015-07-08 06:51:43 -07001228 return fp;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001229}
1230
joshualitteb2a6762014-12-04 11:35:33 -08001231GLCircleOutside2PtConicalEffect::GLCircleOutside2PtConicalEffect(const GrProcessor& processor)
1232 : fVSVaryingName(NULL)
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001233 , fFSVaryingName(NULL)
1234 , fCachedCenterX(SK_ScalarMax)
1235 , fCachedCenterY(SK_ScalarMax)
1236 , fCachedA(SK_ScalarMax)
1237 , fCachedB(SK_ScalarMax)
1238 , fCachedC(SK_ScalarMax)
1239 , fCachedTLimit(SK_ScalarMax) {
joshualittb0a8a372014-09-23 09:50:21 -07001240 const CircleOutside2PtConicalEffect& data = processor.cast<CircleOutside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001241 fIsFlipped = data.isFlipped();
1242 }
1243
joshualitt15988992014-10-09 15:04:05 -07001244void GLCircleOutside2PtConicalEffect::emitCode(GrGLFPBuilder* builder,
joshualitt60030bc2014-11-25 14:21:55 -08001245 const GrFragmentProcessor& fp,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001246 const char* outputColor,
1247 const char* inputColor,
1248 const TransformedCoordsArray& coords,
1249 const TextureSamplerArray& samplers) {
joshualitteb2a6762014-12-04 11:35:33 -08001250 const CircleOutside2PtConicalEffect& ge = fp.cast<CircleOutside2PtConicalEffect>();
joshualitt60030bc2014-11-25 14:21:55 -08001251 this->emitUniforms(builder, ge);
joshualitt30ba4362014-08-21 20:18:45 -07001252 fCenterUni = builder->addUniform(GrGLProgramBuilder::kFragment_Visibility,
bsalomon422f56f2014-12-09 10:18:12 -08001253 kVec2f_GrSLType, kDefault_GrSLPrecision,
1254 "Conical2FSCenter");
joshualitt30ba4362014-08-21 20:18:45 -07001255 fParamUni = builder->addUniform(GrGLProgramBuilder::kFragment_Visibility,
bsalomon422f56f2014-12-09 10:18:12 -08001256 kVec4f_GrSLType, kDefault_GrSLPrecision,
1257 "Conical2FSParams");
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001258 SkString tName("t");
1259
1260 GrGLShaderVar center = builder->getUniformVariable(fCenterUni);
1261 // params.x = A
1262 // params.y = B
1263 // params.z = C
1264 GrGLShaderVar params = builder->getUniformVariable(fParamUni);
1265
1266 // if we have a vec3 from being in perspective, convert it to a vec2 first
egdaniel29bee0f2015-04-29 11:54:42 -07001267 GrGLFragmentBuilder* fsBuilder = builder->getFragmentShaderBuilder();
joshualitt30ba4362014-08-21 20:18:45 -07001268 SkString coords2DString = fsBuilder->ensureFSCoords2D(coords, 0);
skia.committer@gmail.comede0c5c2014-04-23 03:04:11 +00001269 const char* coords2D = coords2DString.c_str();
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001270
1271 // output will default to transparent black (we simply won't write anything
1272 // else to it if invalid, instead of discarding or returning prematurely)
joshualitt30ba4362014-08-21 20:18:45 -07001273 fsBuilder->codeAppendf("\t%s = vec4(0.0,0.0,0.0,0.0);\n", outputColor);
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001274
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001275 // p = coords2D
1276 // e = center end
1277 // r = radius end
1278 // A = dot(e, e) - r^2 + 2 * r - 1
1279 // B = (r -1) / A
1280 // C = 1 / A
1281 // d = dot(e, p) + B
1282 // t = d +/- sqrt(d^2 - A * dot(p, p) + C)
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001283
joshualitt30ba4362014-08-21 20:18:45 -07001284 fsBuilder->codeAppendf("\tfloat pDotp = dot(%s, %s);\n", coords2D, coords2D);
joshualitt49586be2014-09-16 08:21:41 -07001285 fsBuilder->codeAppendf("\tfloat d = dot(%s, %s) + %s.y;\n", coords2D, center.c_str(),
1286 params.c_str());
1287 fsBuilder->codeAppendf("\tfloat deter = d * d - %s.x * pDotp + %s.z;\n", params.c_str(),
1288 params.c_str());
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001289
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001290 // Must check to see if we flipped the circle order (to make sure start radius < end radius)
1291 // If so we must also flip sign on sqrt
1292 if (!fIsFlipped) {
joshualitt30ba4362014-08-21 20:18:45 -07001293 fsBuilder->codeAppendf("\tfloat %s = d + sqrt(deter);\n", tName.c_str());
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001294 } else {
joshualitt30ba4362014-08-21 20:18:45 -07001295 fsBuilder->codeAppendf("\tfloat %s = d - sqrt(deter);\n", tName.c_str());
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001296 }
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001297
joshualitt30ba4362014-08-21 20:18:45 -07001298 fsBuilder->codeAppendf("\tif (%s >= %s.w && deter >= 0.0) {\n", tName.c_str(), params.c_str());
1299 fsBuilder->codeAppend("\t\t");
joshualitt60030bc2014-11-25 14:21:55 -08001300 this->emitColor(builder, ge, tName.c_str(), outputColor, inputColor, samplers);
joshualitt30ba4362014-08-21 20:18:45 -07001301 fsBuilder->codeAppend("\t}\n");
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001302}
1303
kkinnunen7510b222014-07-30 00:04:16 -07001304void GLCircleOutside2PtConicalEffect::setData(const GrGLProgramDataManager& pdman,
joshualittb0a8a372014-09-23 09:50:21 -07001305 const GrProcessor& processor) {
1306 INHERITED::setData(pdman, processor);
1307 const CircleOutside2PtConicalEffect& data = processor.cast<CircleOutside2PtConicalEffect>();
commit-bot@chromium.org44d83c12014-04-21 13:10:25 +00001308 SkASSERT(data.isFlipped() == fIsFlipped);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001309 SkScalar centerX = data.centerX();
1310 SkScalar centerY = data.centerY();
1311 SkScalar A = data.A();
1312 SkScalar B = data.B();
1313 SkScalar C = data.C();
1314 SkScalar tLimit = data.tLimit();
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001315
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001316 if (fCachedCenterX != centerX || fCachedCenterY != centerY ||
1317 fCachedA != A || fCachedB != B || fCachedC != C || fCachedTLimit != tLimit) {
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001318
kkinnunen7510b222014-07-30 00:04:16 -07001319 pdman.set2f(fCenterUni, SkScalarToFloat(centerX), SkScalarToFloat(centerY));
1320 pdman.set4f(fParamUni, SkScalarToFloat(A), SkScalarToFloat(B), SkScalarToFloat(C),
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001321 SkScalarToFloat(tLimit));
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001322
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001323 fCachedCenterX = centerX;
1324 fCachedCenterY = centerY;
1325 fCachedA = A;
1326 fCachedB = B;
1327 fCachedC = C;
1328 fCachedTLimit = tLimit;
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001329 }
1330}
1331
joshualittb0a8a372014-09-23 09:50:21 -07001332void GLCircleOutside2PtConicalEffect::GenKey(const GrProcessor& processor,
jvanverthcfc18862015-04-28 08:48:20 -07001333 const GrGLSLCaps&, GrProcessorKeyBuilder* b) {
bsalomon63e99f72014-07-21 08:03:14 -07001334 uint32_t* key = b->add32n(2);
joshualittb0a8a372014-09-23 09:50:21 -07001335 key[0] = GenBaseGradientKey(processor);
1336 key[1] = processor.cast<CircleOutside2PtConicalEffect>().isFlipped();
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001337}
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +00001338
1339//////////////////////////////////////////////////////////////////////////////
1340
joshualittb0a8a372014-09-23 09:50:21 -07001341GrFragmentProcessor* Gr2PtConicalGradientEffect::Create(GrContext* ctx,
joshualittb2456052015-07-08 09:36:59 -07001342 GrShaderDataManager* shaderDataManager,
joshualittb0a8a372014-09-23 09:50:21 -07001343 const SkTwoPointConicalGradient& shader,
1344 SkShader::TileMode tm,
1345 const SkMatrix* localMatrix) {
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +00001346 SkMatrix matrix;
1347 if (!shader.getLocalMatrix().invert(&matrix)) {
1348 return NULL;
1349 }
commit-bot@chromium.org96fb7482014-05-09 20:28:11 +00001350 if (localMatrix) {
1351 SkMatrix inv;
1352 if (!localMatrix->invert(&inv)) {
1353 return NULL;
1354 }
1355 matrix.postConcat(inv);
1356 }
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +00001357
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001358 if (shader.getStartRadius() < kErrorTol) {
1359 SkScalar focalX;
1360 ConicalType type = set_matrix_focal_conical(shader, &matrix, &focalX);
1361 if (type == kInside_ConicalType) {
joshualittb2456052015-07-08 09:36:59 -07001362 return FocalInside2PtConicalEffect::Create(ctx, shaderDataManager, shader, matrix, tm,
1363 focalX);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001364 } else if(type == kEdge_ConicalType) {
1365 set_matrix_edge_conical(shader, &matrix);
joshualittb2456052015-07-08 09:36:59 -07001366 return Edge2PtConicalEffect::Create(ctx, shaderDataManager, shader, matrix, tm);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001367 } else {
joshualittb2456052015-07-08 09:36:59 -07001368 return FocalOutside2PtConicalEffect::Create(ctx, shaderDataManager, shader, matrix, tm,
1369 focalX);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001370 }
1371 }
1372
1373 CircleConicalInfo info;
1374 ConicalType type = set_matrix_circle_conical(shader, &matrix, &info);
1375
1376 if (type == kInside_ConicalType) {
joshualittb2456052015-07-08 09:36:59 -07001377 return CircleInside2PtConicalEffect::Create(ctx, shaderDataManager, shader, matrix, tm,
1378 info);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001379 } else if (type == kEdge_ConicalType) {
1380 set_matrix_edge_conical(shader, &matrix);
joshualittb2456052015-07-08 09:36:59 -07001381 return Edge2PtConicalEffect::Create(ctx, shaderDataManager, shader, matrix, tm);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001382 } else {
joshualittb2456052015-07-08 09:36:59 -07001383 return CircleOutside2PtConicalEffect::Create(ctx, shaderDataManager, shader, matrix, tm,
1384 info);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001385 }
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +00001386}
1387
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001388#endif