blob: f7b3eb89b28af3247d65440c5523816a6102a2d8 [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
joshualitt30ba4362014-08-21 20:18:45 -070014#include "gl/builders/GrGLProgramBuilder.h"
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +000015// For brevity
kkinnunen7510b222014-07-30 00:04:16 -070016typedef GrGLProgramDataManager::UniformHandle UniformHandle;
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +000017
commit-bot@chromium.org80894672014-04-22 21:24:22 +000018static const SkScalar kErrorTol = 0.00001f;
egdaniel8405ef92014-06-09 11:57:28 -070019static const SkScalar kEdgeErrorTol = 5.f * kErrorTol;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +000020
21/**
22 * We have three general cases for 2pt conical gradients. First we always assume that
23 * the start radius <= end radius. Our first case (kInside_) is when the start circle
24 * is completely enclosed by the end circle. The second case (kOutside_) is the case
25 * when the start circle is either completely outside the end circle or the circles
26 * overlap. The final case (kEdge_) is when the start circle is inside the end one,
27 * but the two are just barely touching at 1 point along their edges.
28 */
29enum ConicalType {
30 kInside_ConicalType,
31 kOutside_ConicalType,
32 kEdge_ConicalType,
33};
34
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000035//////////////////////////////////////////////////////////////////////////////
36
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +000037static void set_matrix_edge_conical(const SkTwoPointConicalGradient& shader,
38 SkMatrix* invLMatrix) {
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000039 // Inverse of the current local matrix is passed in then,
40 // translate to center1, rotate so center2 is on x axis.
41 const SkPoint& center1 = shader.getStartCenter();
42 const SkPoint& center2 = shader.getEndCenter();
43
44 invLMatrix->postTranslate(-center1.fX, -center1.fY);
45
46 SkPoint diff = center2 - center1;
47 SkScalar diffLen = diff.length();
48 if (0 != diffLen) {
49 SkScalar invDiffLen = SkScalarInvert(diffLen);
50 SkMatrix rot;
51 rot.setSinCos(-SkScalarMul(invDiffLen, diff.fY),
52 SkScalarMul(invDiffLen, diff.fX));
53 invLMatrix->postConcat(rot);
54 }
55}
56
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +000057class Edge2PtConicalEffect : public GrGradientEffect {
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +000058public:
59
joshualittb0a8a372014-09-23 09:50:21 -070060 static GrFragmentProcessor* Create(GrContext* ctx,
61 const SkTwoPointConicalGradient& shader,
62 const SkMatrix& matrix,
63 SkShader::TileMode tm) {
bsalomon55fad7a2014-07-08 07:34:20 -070064 return SkNEW_ARGS(Edge2PtConicalEffect, (ctx, shader, matrix, tm));
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000065 }
66
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +000067 virtual ~Edge2PtConicalEffect() {}
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000068
mtklein36352bf2015-03-25 18:17:31 -070069 const char* name() const override {
joshualitteb2a6762014-12-04 11:35:33 -080070 return "Two-Point Conical Gradient Edge Touching";
71 }
72
jvanverthcfc18862015-04-28 08:48:20 -070073 void getGLProcessorKey(const GrGLSLCaps&, GrProcessorKeyBuilder*) const override;
joshualitteb2a6762014-12-04 11:35:33 -080074
mtklein36352bf2015-03-25 18:17:31 -070075 GrGLFragmentProcessor* createGLInstance() const override;
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000076
77 // The radial gradient parameters can collapse to a linear (instead of quadratic) equation.
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000078 SkScalar center() const { return fCenterX1; }
79 SkScalar diffRadius() const { return fDiffRadius; }
80 SkScalar radius() const { return fRadius0; }
81
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000082private:
mtklein36352bf2015-03-25 18:17:31 -070083 bool onIsEqual(const GrFragmentProcessor& sBase) const override {
joshualitt49586be2014-09-16 08:21:41 -070084 const Edge2PtConicalEffect& s = sBase.cast<Edge2PtConicalEffect>();
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000085 return (INHERITED::onIsEqual(sBase) &&
86 this->fCenterX1 == s.fCenterX1 &&
87 this->fRadius0 == s.fRadius0 &&
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +000088 this->fDiffRadius == s.fDiffRadius);
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000089 }
90
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +000091 Edge2PtConicalEffect(GrContext* ctx,
92 const SkTwoPointConicalGradient& shader,
93 const SkMatrix& matrix,
94 SkShader::TileMode tm)
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000095 : INHERITED(ctx, shader, matrix, tm),
96 fCenterX1(shader.getCenterX1()),
97 fRadius0(shader.getStartRadius()),
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +000098 fDiffRadius(shader.getDiffRadius()){
joshualitteb2a6762014-12-04 11:35:33 -080099 this->initClassID<Edge2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000100 // We should only be calling this shader if we are degenerate case with touching circles
egdaniel8405ef92014-06-09 11:57:28 -0700101 // When deciding if we are in edge case, we scaled by the end radius for cases when the
joshualitt01258472014-09-22 10:29:30 -0700102 // start radius was close to zero, otherwise we scaled by the start radius. In addition
103 // Our test for the edge case in set_matrix_circle_conical has a higher tolerance so we
104 // need the sqrt value below
105 SkASSERT(SkScalarAbs(SkScalarAbs(fDiffRadius) - fCenterX1) <
106 (fRadius0 < kErrorTol ? shader.getEndRadius() * kEdgeErrorTol :
107 fRadius0 * sqrt(kEdgeErrorTol)));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000108
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +0000109 // We pass the linear part of the quadratic as a varying.
110 // float b = -2.0 * (fCenterX1 * x + fRadius0 * fDiffRadius * z)
111 fBTransform = this->getCoordTransform();
112 SkMatrix& bMatrix = *fBTransform.accessMatrix();
113 SkScalar r0dr = SkScalarMul(fRadius0, fDiffRadius);
114 bMatrix[SkMatrix::kMScaleX] = -2 * (SkScalarMul(fCenterX1, bMatrix[SkMatrix::kMScaleX]) +
115 SkScalarMul(r0dr, bMatrix[SkMatrix::kMPersp0]));
116 bMatrix[SkMatrix::kMSkewX] = -2 * (SkScalarMul(fCenterX1, bMatrix[SkMatrix::kMSkewX]) +
117 SkScalarMul(r0dr, bMatrix[SkMatrix::kMPersp1]));
118 bMatrix[SkMatrix::kMTransX] = -2 * (SkScalarMul(fCenterX1, bMatrix[SkMatrix::kMTransX]) +
119 SkScalarMul(r0dr, bMatrix[SkMatrix::kMPersp2]));
120 this->addCoordTransform(&fBTransform);
121 }
122
joshualittb0a8a372014-09-23 09:50:21 -0700123 GR_DECLARE_FRAGMENT_PROCESSOR_TEST;
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +0000124
125 // @{
126 // Cache of values - these can change arbitrarily, EXCEPT
127 // we shouldn't change between degenerate and non-degenerate?!
128
129 GrCoordTransform fBTransform;
130 SkScalar fCenterX1;
131 SkScalar fRadius0;
132 SkScalar fDiffRadius;
133
134 // @}
135
136 typedef GrGradientEffect INHERITED;
137};
138
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000139class GLEdge2PtConicalEffect : public GrGLGradientEffect {
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +0000140public:
joshualitteb2a6762014-12-04 11:35:33 -0800141 GLEdge2PtConicalEffect(const GrProcessor&);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000142 virtual ~GLEdge2PtConicalEffect() { }
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000143
joshualitt15988992014-10-09 15:04:05 -0700144 virtual void emitCode(GrGLFPBuilder*,
joshualittb0a8a372014-09-23 09:50:21 -0700145 const GrFragmentProcessor&,
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000146 const char* outputColor,
147 const char* inputColor,
148 const TransformedCoordsArray&,
mtklein36352bf2015-03-25 18:17:31 -0700149 const TextureSamplerArray&) override;
150 void setData(const GrGLProgramDataManager&, const GrProcessor&) override;
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000151
jvanverthcfc18862015-04-28 08:48:20 -0700152 static void GenKey(const GrProcessor&, const GrGLSLCaps& caps, GrProcessorKeyBuilder* b);
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000153
154protected:
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000155 UniformHandle fParamUni;
156
157 const char* fVSVaryingName;
158 const char* fFSVaryingName;
159
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000160 // @{
161 /// Values last uploaded as uniforms
162
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000163 SkScalar fCachedRadius;
164 SkScalar fCachedDiffRadius;
165
166 // @}
167
168private:
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000169 typedef GrGLGradientEffect INHERITED;
170
171};
172
jvanverthcfc18862015-04-28 08:48:20 -0700173void Edge2PtConicalEffect::getGLProcessorKey(const GrGLSLCaps& caps,
joshualitteb2a6762014-12-04 11:35:33 -0800174 GrProcessorKeyBuilder* b) const {
175 GLEdge2PtConicalEffect::GenKey(*this, caps, b);
176}
177
178GrGLFragmentProcessor* Edge2PtConicalEffect::createGLInstance() const {
179 return SkNEW_ARGS(GLEdge2PtConicalEffect, (*this));
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000180}
skia.committer@gmail.com221b9112014-04-04 03:04:32 +0000181
joshualittb0a8a372014-09-23 09:50:21 -0700182GR_DEFINE_FRAGMENT_PROCESSOR_TEST(Edge2PtConicalEffect);
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000183
joshualitt01258472014-09-22 10:29:30 -0700184/*
185 * All Two point conical gradient test create functions may occasionally create edge case shaders
186 */
joshualittb0a8a372014-09-23 09:50:21 -0700187GrFragmentProcessor* Edge2PtConicalEffect::TestCreate(SkRandom* random,
188 GrContext* context,
189 const GrDrawTargetCaps&,
190 GrTexture**) {
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000191 SkPoint center1 = {random->nextUScalar1(), random->nextUScalar1()};
192 SkScalar radius1 = random->nextUScalar1();
193 SkPoint center2;
194 SkScalar radius2;
195 do {
196 center2.set(random->nextUScalar1(), random->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;
211 int colorCount = RandomGradientParams(random, colors, &stops, &tm);
212 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;
joshualitt5531d512014-12-17 15:50:11 -0800219 SkAssertResult(shader->asFragmentProcessor(context, paint,
joshualitt4eaf9ce2015-04-28 13:31:18 -0700220 GrTest::TestMatrix(random), NULL,
joshualitt5531d512014-12-17 15:50:11 -0800221 &paintColor, &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,
382 const SkTwoPointConicalGradient& shader,
383 const SkMatrix& matrix,
384 SkShader::TileMode tm,
385 SkScalar focalX) {
bsalomon55fad7a2014-07-08 07:34:20 -0700386 return SkNEW_ARGS(FocalOutside2PtConicalEffect, (ctx, shader, matrix, tm, focalX));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000387 }
388
389 virtual ~FocalOutside2PtConicalEffect() { }
390
mtklein36352bf2015-03-25 18:17:31 -0700391 const char* name() const override {
joshualitteb2a6762014-12-04 11:35:33 -0800392 return "Two-Point Conical Gradient Focal Outside";
393 }
394
jvanverthcfc18862015-04-28 08:48:20 -0700395 void getGLProcessorKey(const GrGLSLCaps&, GrProcessorKeyBuilder*) const override;
joshualitteb2a6762014-12-04 11:35:33 -0800396
mtklein36352bf2015-03-25 18:17:31 -0700397 GrGLFragmentProcessor* createGLInstance() const override;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000398
399 bool isFlipped() const { return fIsFlipped; }
400 SkScalar focal() const { return fFocalX; }
401
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000402private:
mtklein36352bf2015-03-25 18:17:31 -0700403 bool onIsEqual(const GrFragmentProcessor& sBase) const override {
joshualitt49586be2014-09-16 08:21:41 -0700404 const FocalOutside2PtConicalEffect& s = sBase.cast<FocalOutside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000405 return (INHERITED::onIsEqual(sBase) &&
406 this->fFocalX == s.fFocalX &&
407 this->fIsFlipped == s.fIsFlipped);
408 }
409
410 FocalOutside2PtConicalEffect(GrContext* ctx,
411 const SkTwoPointConicalGradient& shader,
412 const SkMatrix& matrix,
413 SkShader::TileMode tm,
414 SkScalar focalX)
joshualitteb2a6762014-12-04 11:35:33 -0800415 : INHERITED(ctx, shader, matrix, tm), fFocalX(focalX), fIsFlipped(shader.isFlippedGrad()) {
416 this->initClassID<FocalOutside2PtConicalEffect>();
417 }
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000418
joshualittb0a8a372014-09-23 09:50:21 -0700419 GR_DECLARE_FRAGMENT_PROCESSOR_TEST;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000420
421 SkScalar fFocalX;
422 bool fIsFlipped;
423
424 typedef GrGradientEffect INHERITED;
425};
426
427class GLFocalOutside2PtConicalEffect : public GrGLGradientEffect {
428public:
joshualitteb2a6762014-12-04 11:35:33 -0800429 GLFocalOutside2PtConicalEffect(const GrProcessor&);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000430 virtual ~GLFocalOutside2PtConicalEffect() { }
431
joshualitt15988992014-10-09 15:04:05 -0700432 virtual void emitCode(GrGLFPBuilder*,
joshualittb0a8a372014-09-23 09:50:21 -0700433 const GrFragmentProcessor&,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000434 const char* outputColor,
435 const char* inputColor,
436 const TransformedCoordsArray&,
mtklein36352bf2015-03-25 18:17:31 -0700437 const TextureSamplerArray&) override;
438 void setData(const GrGLProgramDataManager&, const GrProcessor&) override;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000439
jvanverthcfc18862015-04-28 08:48:20 -0700440 static void GenKey(const GrProcessor&, const GrGLSLCaps& caps, GrProcessorKeyBuilder* b);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000441
442protected:
443 UniformHandle fParamUni;
444
445 const char* fVSVaryingName;
446 const char* fFSVaryingName;
447
448 bool fIsFlipped;
449
450 // @{
451 /// Values last uploaded as uniforms
452
453 SkScalar fCachedFocal;
454
455 // @}
456
457private:
458 typedef GrGLGradientEffect INHERITED;
459
460};
461
jvanverthcfc18862015-04-28 08:48:20 -0700462void FocalOutside2PtConicalEffect::getGLProcessorKey(const GrGLSLCaps& caps,
joshualitteb2a6762014-12-04 11:35:33 -0800463 GrProcessorKeyBuilder* b) const {
464 GLFocalOutside2PtConicalEffect::GenKey(*this, caps, b);
465}
466
467GrGLFragmentProcessor* FocalOutside2PtConicalEffect::createGLInstance() const {
468 return SkNEW_ARGS(GLFocalOutside2PtConicalEffect, (*this));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000469}
470
joshualittb0a8a372014-09-23 09:50:21 -0700471GR_DEFINE_FRAGMENT_PROCESSOR_TEST(FocalOutside2PtConicalEffect);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000472
joshualitt01258472014-09-22 10:29:30 -0700473/*
474 * All Two point conical gradient test create functions may occasionally create edge case shaders
475 */
joshualittb0a8a372014-09-23 09:50:21 -0700476GrFragmentProcessor* FocalOutside2PtConicalEffect::TestCreate(SkRandom* random,
477 GrContext* context,
478 const GrDrawTargetCaps&,
479 GrTexture**) {
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000480 SkPoint center1 = {random->nextUScalar1(), random->nextUScalar1()};
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000481 SkScalar radius1 = 0.f;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000482 SkPoint center2;
483 SkScalar radius2;
484 do {
485 center2.set(random->nextUScalar1(), random->nextUScalar1());
skia.committer@gmail.comede0c5c2014-04-23 03:04:11 +0000486 // 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 +0000487 } while (center1 == center2);
488 SkPoint diff = center2 - center1;
489 SkScalar diffLen = diff.length();
490 // Below makes sure that the focal point is not contained within circle two
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000491 radius2 = random->nextRangeF(0.f, diffLen);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000492
493 SkColor colors[kMaxRandomGradientColors];
494 SkScalar stopsArray[kMaxRandomGradientColors];
495 SkScalar* stops = stopsArray;
496 SkShader::TileMode tm;
497 int colorCount = RandomGradientParams(random, colors, &stops, &tm);
498 SkAutoTUnref<SkShader> shader(SkGradientShader::CreateTwoPointConical(center1, radius1,
499 center2, radius2,
500 colors, stops, colorCount,
501 tm));
502 SkPaint paint;
joshualittb0a8a372014-09-23 09:50:21 -0700503 GrFragmentProcessor* effect;
bsalomon83d081a2014-07-08 09:56:10 -0700504 GrColor paintColor;
joshualitt5531d512014-12-17 15:50:11 -0800505
506 SkAssertResult(shader->asFragmentProcessor(context, paint,
joshualitt4eaf9ce2015-04-28 13:31:18 -0700507 GrTest::TestMatrix(random), NULL,
joshualitt5531d512014-12-17 15:50:11 -0800508 &paintColor, &effect));
dandov9de5b512014-06-10 14:38:28 -0700509 return effect;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000510}
511
joshualitteb2a6762014-12-04 11:35:33 -0800512GLFocalOutside2PtConicalEffect::GLFocalOutside2PtConicalEffect(const GrProcessor& processor)
513 : fVSVaryingName(NULL)
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000514 , fFSVaryingName(NULL)
515 , fCachedFocal(SK_ScalarMax) {
joshualittb0a8a372014-09-23 09:50:21 -0700516 const FocalOutside2PtConicalEffect& data = processor.cast<FocalOutside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000517 fIsFlipped = data.isFlipped();
518}
519
joshualitt15988992014-10-09 15:04:05 -0700520void GLFocalOutside2PtConicalEffect::emitCode(GrGLFPBuilder* builder,
joshualitt60030bc2014-11-25 14:21:55 -0800521 const GrFragmentProcessor& fp,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000522 const char* outputColor,
523 const char* inputColor,
524 const TransformedCoordsArray& coords,
525 const TextureSamplerArray& samplers) {
joshualitteb2a6762014-12-04 11:35:33 -0800526 const FocalOutside2PtConicalEffect& ge = fp.cast<FocalOutside2PtConicalEffect>();
joshualitt60030bc2014-11-25 14:21:55 -0800527 this->emitUniforms(builder, ge);
joshualitt30ba4362014-08-21 20:18:45 -0700528 fParamUni = builder->addUniformArray(GrGLProgramBuilder::kFragment_Visibility,
bsalomon422f56f2014-12-09 10:18:12 -0800529 kFloat_GrSLType, kDefault_GrSLPrecision,
530 "Conical2FSParams", 2);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000531 SkString tName("t");
532 SkString p0; // focalX
533 SkString p1; // 1 - focalX * focalX
534
535 builder->getUniformVariable(fParamUni).appendArrayAccess(0, &p0);
536 builder->getUniformVariable(fParamUni).appendArrayAccess(1, &p1);
537
538 // if we have a vec3 from being in perspective, convert it to a vec2 first
egdaniel29bee0f2015-04-29 11:54:42 -0700539 GrGLFragmentBuilder* fsBuilder = builder->getFragmentShaderBuilder();
joshualitt30ba4362014-08-21 20:18:45 -0700540 SkString coords2DString = fsBuilder->ensureFSCoords2D(coords, 0);
skia.committer@gmail.comede0c5c2014-04-23 03:04:11 +0000541 const char* coords2D = coords2DString.c_str();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000542
543 // t = p.x * focal.x +/- sqrt(p.x^2 + (1 - focal.x^2) * p.y^2)
544
545 // output will default to transparent black (we simply won't write anything
546 // else to it if invalid, instead of discarding or returning prematurely)
joshualitt30ba4362014-08-21 20:18:45 -0700547 fsBuilder->codeAppendf("\t%s = vec4(0.0,0.0,0.0,0.0);\n", outputColor);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000548
joshualitt30ba4362014-08-21 20:18:45 -0700549 fsBuilder->codeAppendf("\tfloat xs = %s.x * %s.x;\n", coords2D, coords2D);
550 fsBuilder->codeAppendf("\tfloat ys = %s.y * %s.y;\n", coords2D, coords2D);
551 fsBuilder->codeAppendf("\tfloat d = xs + %s * ys;\n", p1.c_str());
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000552
553 // Must check to see if we flipped the circle order (to make sure start radius < end radius)
554 // If so we must also flip sign on sqrt
555 if (!fIsFlipped) {
joshualitt30ba4362014-08-21 20:18:45 -0700556 fsBuilder->codeAppendf("\tfloat %s = %s.x * %s + sqrt(d);\n", tName.c_str(),
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000557 coords2D, p0.c_str());
558 } else {
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 }
562
joshualitt30ba4362014-08-21 20:18:45 -0700563 fsBuilder->codeAppendf("\tif (%s >= 0.0 && d >= 0.0) {\n", tName.c_str());
564 fsBuilder->codeAppend("\t\t");
joshualitt60030bc2014-11-25 14:21:55 -0800565 this->emitColor(builder, ge, tName.c_str(), outputColor, inputColor, samplers);
joshualitt30ba4362014-08-21 20:18:45 -0700566 fsBuilder->codeAppend("\t}\n");
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000567}
568
kkinnunen7510b222014-07-30 00:04:16 -0700569void GLFocalOutside2PtConicalEffect::setData(const GrGLProgramDataManager& pdman,
joshualittb0a8a372014-09-23 09:50:21 -0700570 const GrProcessor& processor) {
571 INHERITED::setData(pdman, processor);
572 const FocalOutside2PtConicalEffect& data = processor.cast<FocalOutside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000573 SkASSERT(data.isFlipped() == fIsFlipped);
574 SkScalar focal = data.focal();
575
576 if (fCachedFocal != focal) {
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000577 SkScalar oneMinus2F = 1.f - SkScalarMul(focal, focal);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000578
579 float values[2] = {
580 SkScalarToFloat(focal),
581 SkScalarToFloat(oneMinus2F),
582 };
583
kkinnunen7510b222014-07-30 00:04:16 -0700584 pdman.set1fv(fParamUni, 2, values);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000585 fCachedFocal = focal;
586 }
587}
588
joshualittb0a8a372014-09-23 09:50:21 -0700589void GLFocalOutside2PtConicalEffect::GenKey(const GrProcessor& processor,
jvanverthcfc18862015-04-28 08:48:20 -0700590 const GrGLSLCaps&, GrProcessorKeyBuilder* b) {
bsalomon63e99f72014-07-21 08:03:14 -0700591 uint32_t* key = b->add32n(2);
joshualittb0a8a372014-09-23 09:50:21 -0700592 key[0] = GenBaseGradientKey(processor);
593 key[1] = processor.cast<FocalOutside2PtConicalEffect>().isFlipped();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000594}
595
596//////////////////////////////////////////////////////////////////////////////
597
598class GLFocalInside2PtConicalEffect;
599
600class FocalInside2PtConicalEffect : public GrGradientEffect {
601public:
602
joshualittb0a8a372014-09-23 09:50:21 -0700603 static GrFragmentProcessor* Create(GrContext* ctx,
604 const SkTwoPointConicalGradient& shader,
605 const SkMatrix& matrix,
606 SkShader::TileMode tm,
607 SkScalar focalX) {
bsalomon55fad7a2014-07-08 07:34:20 -0700608 return SkNEW_ARGS(FocalInside2PtConicalEffect, (ctx, shader, matrix, tm, focalX));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000609 }
610
611 virtual ~FocalInside2PtConicalEffect() {}
612
mtklein36352bf2015-03-25 18:17:31 -0700613 const char* name() const override {
joshualitteb2a6762014-12-04 11:35:33 -0800614 return "Two-Point Conical Gradient Focal Inside";
615 }
616
jvanverthcfc18862015-04-28 08:48:20 -0700617 void getGLProcessorKey(const GrGLSLCaps&, GrProcessorKeyBuilder*) const override;
joshualitteb2a6762014-12-04 11:35:33 -0800618
mtklein36352bf2015-03-25 18:17:31 -0700619 GrGLFragmentProcessor* createGLInstance() const override;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000620
621 SkScalar focal() const { return fFocalX; }
622
joshualittb0a8a372014-09-23 09:50:21 -0700623 typedef GLFocalInside2PtConicalEffect GLProcessor;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000624
625private:
mtklein36352bf2015-03-25 18:17:31 -0700626 bool onIsEqual(const GrFragmentProcessor& sBase) const override {
joshualitt49586be2014-09-16 08:21:41 -0700627 const FocalInside2PtConicalEffect& s = sBase.cast<FocalInside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000628 return (INHERITED::onIsEqual(sBase) &&
629 this->fFocalX == s.fFocalX);
630 }
631
632 FocalInside2PtConicalEffect(GrContext* ctx,
633 const SkTwoPointConicalGradient& shader,
634 const SkMatrix& matrix,
635 SkShader::TileMode tm,
636 SkScalar focalX)
joshualitteb2a6762014-12-04 11:35:33 -0800637 : INHERITED(ctx, shader, matrix, tm), fFocalX(focalX) {
638 this->initClassID<FocalInside2PtConicalEffect>();
639 }
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000640
joshualittb0a8a372014-09-23 09:50:21 -0700641 GR_DECLARE_FRAGMENT_PROCESSOR_TEST;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000642
643 SkScalar fFocalX;
644
645 typedef GrGradientEffect INHERITED;
646};
647
648class GLFocalInside2PtConicalEffect : public GrGLGradientEffect {
649public:
joshualitteb2a6762014-12-04 11:35:33 -0800650 GLFocalInside2PtConicalEffect(const GrProcessor&);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000651 virtual ~GLFocalInside2PtConicalEffect() {}
652
joshualitt15988992014-10-09 15:04:05 -0700653 virtual void emitCode(GrGLFPBuilder*,
joshualittb0a8a372014-09-23 09:50:21 -0700654 const GrFragmentProcessor&,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000655 const char* outputColor,
656 const char* inputColor,
657 const TransformedCoordsArray&,
mtklein36352bf2015-03-25 18:17:31 -0700658 const TextureSamplerArray&) override;
659 void setData(const GrGLProgramDataManager&, const GrProcessor&) override;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000660
jvanverthcfc18862015-04-28 08:48:20 -0700661 static void GenKey(const GrProcessor&, const GrGLSLCaps& caps, GrProcessorKeyBuilder* b);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000662
663protected:
664 UniformHandle fFocalUni;
665
666 const char* fVSVaryingName;
667 const char* fFSVaryingName;
668
669 // @{
670 /// Values last uploaded as uniforms
671
672 SkScalar fCachedFocal;
673
674 // @}
675
676private:
677 typedef GrGLGradientEffect INHERITED;
678
679};
680
jvanverthcfc18862015-04-28 08:48:20 -0700681void FocalInside2PtConicalEffect::getGLProcessorKey(const GrGLSLCaps& caps,
joshualitteb2a6762014-12-04 11:35:33 -0800682 GrProcessorKeyBuilder* b) const {
683 GLFocalInside2PtConicalEffect::GenKey(*this, caps, b);
684}
685
686GrGLFragmentProcessor* FocalInside2PtConicalEffect::createGLInstance() const {
687 return SkNEW_ARGS(GLFocalInside2PtConicalEffect, (*this));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000688}
689
joshualittb0a8a372014-09-23 09:50:21 -0700690GR_DEFINE_FRAGMENT_PROCESSOR_TEST(FocalInside2PtConicalEffect);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000691
joshualitt01258472014-09-22 10:29:30 -0700692/*
693 * All Two point conical gradient test create functions may occasionally create edge case shaders
694 */
joshualittb0a8a372014-09-23 09:50:21 -0700695GrFragmentProcessor* FocalInside2PtConicalEffect::TestCreate(SkRandom* random,
696 GrContext* context,
697 const GrDrawTargetCaps&,
698 GrTexture**) {
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000699 SkPoint center1 = {random->nextUScalar1(), random->nextUScalar1()};
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000700 SkScalar radius1 = 0.f;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000701 SkPoint center2;
702 SkScalar radius2;
703 do {
704 center2.set(random->nextUScalar1(), random->nextUScalar1());
705 // Below makes sure radius2 is larger enouch such that the focal point
706 // is inside the end circle
707 SkScalar increase = random->nextUScalar1();
708 SkPoint diff = center2 - center1;
709 SkScalar diffLen = diff.length();
710 radius2 = diffLen + increase;
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000711 // If the circles are identical the factory will give us an empty shader.
712 } while (radius1 == radius2 && center1 == center2);
713
714 SkColor colors[kMaxRandomGradientColors];
715 SkScalar stopsArray[kMaxRandomGradientColors];
716 SkScalar* stops = stopsArray;
717 SkShader::TileMode tm;
718 int colorCount = RandomGradientParams(random, colors, &stops, &tm);
719 SkAutoTUnref<SkShader> shader(SkGradientShader::CreateTwoPointConical(center1, radius1,
720 center2, radius2,
721 colors, stops, colorCount,
722 tm));
723 SkPaint paint;
bsalomon83d081a2014-07-08 09:56:10 -0700724 GrColor paintColor;
joshualittb0a8a372014-09-23 09:50:21 -0700725 GrFragmentProcessor* fp;
joshualitt5531d512014-12-17 15:50:11 -0800726 SkAssertResult(shader->asFragmentProcessor(context, paint,
joshualitt4eaf9ce2015-04-28 13:31:18 -0700727 GrTest::TestMatrix(random), NULL,
joshualitt5531d512014-12-17 15:50:11 -0800728 &paintColor, &fp));
joshualittb0a8a372014-09-23 09:50:21 -0700729 return fp;
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000730}
731
joshualitteb2a6762014-12-04 11:35:33 -0800732GLFocalInside2PtConicalEffect::GLFocalInside2PtConicalEffect(const GrProcessor&)
733 : fVSVaryingName(NULL)
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000734 , fFSVaryingName(NULL)
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000735 , fCachedFocal(SK_ScalarMax) {}
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000736
joshualitt15988992014-10-09 15:04:05 -0700737void GLFocalInside2PtConicalEffect::emitCode(GrGLFPBuilder* builder,
joshualitt60030bc2014-11-25 14:21:55 -0800738 const GrFragmentProcessor& fp,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000739 const char* outputColor,
740 const char* inputColor,
741 const TransformedCoordsArray& coords,
742 const TextureSamplerArray& samplers) {
joshualitteb2a6762014-12-04 11:35:33 -0800743 const FocalInside2PtConicalEffect& ge = fp.cast<FocalInside2PtConicalEffect>();
joshualitt60030bc2014-11-25 14:21:55 -0800744 this->emitUniforms(builder, ge);
joshualitt30ba4362014-08-21 20:18:45 -0700745 fFocalUni = builder->addUniform(GrGLProgramBuilder::kFragment_Visibility,
bsalomon422f56f2014-12-09 10:18:12 -0800746 kFloat_GrSLType, kDefault_GrSLPrecision,
747 "Conical2FSParams");
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000748 SkString tName("t");
749
750 // this is the distance along x-axis from the end center to focal point in
751 // transformed coordinates
752 GrGLShaderVar focal = builder->getUniformVariable(fFocalUni);
753
754 // if we have a vec3 from being in perspective, convert it to a vec2 first
egdaniel29bee0f2015-04-29 11:54:42 -0700755 GrGLFragmentBuilder* fsBuilder = builder->getFragmentShaderBuilder();
joshualitt30ba4362014-08-21 20:18:45 -0700756 SkString coords2DString = fsBuilder->ensureFSCoords2D(coords, 0);
skia.committer@gmail.comede0c5c2014-04-23 03:04:11 +0000757 const char* coords2D = coords2DString.c_str();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000758
759 // t = p.x * focalX + length(p)
joshualitt30ba4362014-08-21 20:18:45 -0700760 fsBuilder->codeAppendf("\tfloat %s = %s.x * %s + length(%s);\n", tName.c_str(),
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000761 coords2D, focal.c_str(), coords2D);
762
joshualitt60030bc2014-11-25 14:21:55 -0800763 this->emitColor(builder, ge, tName.c_str(), outputColor, inputColor, samplers);
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000764}
765
kkinnunen7510b222014-07-30 00:04:16 -0700766void GLFocalInside2PtConicalEffect::setData(const GrGLProgramDataManager& pdman,
joshualittb0a8a372014-09-23 09:50:21 -0700767 const GrProcessor& processor) {
768 INHERITED::setData(pdman, processor);
769 const FocalInside2PtConicalEffect& data = processor.cast<FocalInside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000770 SkScalar focal = data.focal();
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000771
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000772 if (fCachedFocal != focal) {
kkinnunen7510b222014-07-30 00:04:16 -0700773 pdman.set1f(fFocalUni, SkScalarToFloat(focal));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000774 fCachedFocal = focal;
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000775 }
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000776}
777
joshualittb0a8a372014-09-23 09:50:21 -0700778void GLFocalInside2PtConicalEffect::GenKey(const GrProcessor& processor,
jvanverthcfc18862015-04-28 08:48:20 -0700779 const GrGLSLCaps&, GrProcessorKeyBuilder* b) {
joshualittb0a8a372014-09-23 09:50:21 -0700780 b->add32(GenBaseGradientKey(processor));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000781}
782
783//////////////////////////////////////////////////////////////////////////////
784// Circle Conical Gradients
785//////////////////////////////////////////////////////////////////////////////
786
787struct CircleConicalInfo {
788 SkPoint fCenterEnd;
789 SkScalar fA;
790 SkScalar fB;
791 SkScalar fC;
792};
793
794// Returns focal distance along x-axis in transformed coords
795static ConicalType set_matrix_circle_conical(const SkTwoPointConicalGradient& shader,
796 SkMatrix* invLMatrix, CircleConicalInfo* info) {
797 // Inverse of the current local matrix is passed in then,
798 // translate and scale such that start circle is on the origin and has radius 1
799 const SkPoint& centerStart = shader.getStartCenter();
800 const SkPoint& centerEnd = shader.getEndCenter();
801 SkScalar radiusStart = shader.getStartRadius();
802 SkScalar radiusEnd = shader.getEndRadius();
803
804 SkMatrix matrix;
805
806 matrix.setTranslate(-centerStart.fX, -centerStart.fY);
807
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000808 SkScalar invStartRad = 1.f / radiusStart;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000809 matrix.postScale(invStartRad, invStartRad);
810
811 radiusEnd /= radiusStart;
812
813 SkPoint centerEndTrans;
814 matrix.mapPoints(&centerEndTrans, &centerEnd, 1);
815
816 SkScalar A = centerEndTrans.fX * centerEndTrans.fX + centerEndTrans.fY * centerEndTrans.fY
817 - radiusEnd * radiusEnd + 2 * radiusEnd - 1;
818
819 // Check to see if start circle is inside end circle with edges touching.
820 // If touching we return that it is of kEdge_ConicalType, and leave the matrix setting
egdaniel8405ef92014-06-09 11:57:28 -0700821 // to the edge shader. kEdgeErrorTol = 5 * kErrorTol was picked after manual testing
822 // so that C = 1 / A is stable, and the linear approximation used in the Edge shader is
823 // still accurate.
824 if (SkScalarAbs(A) < kEdgeErrorTol) {
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000825 return kEdge_ConicalType;
826 }
827
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000828 SkScalar C = 1.f / A;
829 SkScalar B = (radiusEnd - 1.f) * C;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000830
831 matrix.postScale(C, C);
832
833 invLMatrix->postConcat(matrix);
834
835 info->fCenterEnd = centerEndTrans;
836 info->fA = A;
837 info->fB = B;
838 info->fC = C;
839
840 // 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 +0000841 if (A < 0.f) {
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000842 return kInside_ConicalType;
843 }
844 return kOutside_ConicalType;
845}
846
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000847class CircleInside2PtConicalEffect : public GrGradientEffect {
848public:
849
joshualittb0a8a372014-09-23 09:50:21 -0700850 static GrFragmentProcessor* Create(GrContext* ctx,
851 const SkTwoPointConicalGradient& shader,
852 const SkMatrix& matrix,
853 SkShader::TileMode tm,
854 const CircleConicalInfo& info) {
bsalomon55fad7a2014-07-08 07:34:20 -0700855 return SkNEW_ARGS(CircleInside2PtConicalEffect, (ctx, shader, matrix, tm, info));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000856 }
857
858 virtual ~CircleInside2PtConicalEffect() {}
859
mtklein36352bf2015-03-25 18:17:31 -0700860 const char* name() const override { return "Two-Point Conical Gradient Inside"; }
joshualitteb2a6762014-12-04 11:35:33 -0800861
jvanverthcfc18862015-04-28 08:48:20 -0700862 virtual void getGLProcessorKey(const GrGLSLCaps& caps,
mtklein36352bf2015-03-25 18:17:31 -0700863 GrProcessorKeyBuilder* b) const override;
joshualitteb2a6762014-12-04 11:35:33 -0800864
mtklein36352bf2015-03-25 18:17:31 -0700865 GrGLFragmentProcessor* createGLInstance() const override;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000866
867 SkScalar centerX() const { return fInfo.fCenterEnd.fX; }
868 SkScalar centerY() const { return fInfo.fCenterEnd.fY; }
869 SkScalar A() const { return fInfo.fA; }
870 SkScalar B() const { return fInfo.fB; }
871 SkScalar C() const { return fInfo.fC; }
872
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000873private:
mtklein36352bf2015-03-25 18:17:31 -0700874 bool onIsEqual(const GrFragmentProcessor& sBase) const override {
joshualitt49586be2014-09-16 08:21:41 -0700875 const CircleInside2PtConicalEffect& s = sBase.cast<CircleInside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000876 return (INHERITED::onIsEqual(sBase) &&
877 this->fInfo.fCenterEnd == s.fInfo.fCenterEnd &&
878 this->fInfo.fA == s.fInfo.fA &&
879 this->fInfo.fB == s.fInfo.fB &&
880 this->fInfo.fC == s.fInfo.fC);
881 }
882
883 CircleInside2PtConicalEffect(GrContext* ctx,
884 const SkTwoPointConicalGradient& shader,
885 const SkMatrix& matrix,
886 SkShader::TileMode tm,
887 const CircleConicalInfo& info)
joshualitteb2a6762014-12-04 11:35:33 -0800888 : INHERITED(ctx, shader, matrix, tm), fInfo(info) {
889 this->initClassID<CircleInside2PtConicalEffect>();
890 }
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000891
joshualittb0a8a372014-09-23 09:50:21 -0700892 GR_DECLARE_FRAGMENT_PROCESSOR_TEST;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000893
894 const CircleConicalInfo fInfo;
895
896 typedef GrGradientEffect INHERITED;
897};
898
899class GLCircleInside2PtConicalEffect : public GrGLGradientEffect {
900public:
joshualitteb2a6762014-12-04 11:35:33 -0800901 GLCircleInside2PtConicalEffect(const GrProcessor&);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000902 virtual ~GLCircleInside2PtConicalEffect() {}
903
joshualitt15988992014-10-09 15:04:05 -0700904 virtual void emitCode(GrGLFPBuilder*,
joshualittb0a8a372014-09-23 09:50:21 -0700905 const GrFragmentProcessor&,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000906 const char* outputColor,
907 const char* inputColor,
908 const TransformedCoordsArray&,
mtklein36352bf2015-03-25 18:17:31 -0700909 const TextureSamplerArray&) override;
910 void setData(const GrGLProgramDataManager&, const GrProcessor&) override;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000911
jvanverthcfc18862015-04-28 08:48:20 -0700912 static void GenKey(const GrProcessor&, const GrGLSLCaps& caps, GrProcessorKeyBuilder* b);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000913
914protected:
915 UniformHandle fCenterUni;
916 UniformHandle fParamUni;
917
918 const char* fVSVaryingName;
919 const char* fFSVaryingName;
920
921 // @{
922 /// Values last uploaded as uniforms
923
924 SkScalar fCachedCenterX;
925 SkScalar fCachedCenterY;
926 SkScalar fCachedA;
927 SkScalar fCachedB;
928 SkScalar fCachedC;
929
930 // @}
931
932private:
933 typedef GrGLGradientEffect INHERITED;
934
935};
936
jvanverthcfc18862015-04-28 08:48:20 -0700937void CircleInside2PtConicalEffect::getGLProcessorKey(const GrGLSLCaps& caps,
joshualitteb2a6762014-12-04 11:35:33 -0800938 GrProcessorKeyBuilder* b) const {
939 GLCircleInside2PtConicalEffect::GenKey(*this, caps, b);
940}
941
942GrGLFragmentProcessor* CircleInside2PtConicalEffect::createGLInstance() const {
943 return SkNEW_ARGS(GLCircleInside2PtConicalEffect, (*this));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000944}
945
joshualittb0a8a372014-09-23 09:50:21 -0700946GR_DEFINE_FRAGMENT_PROCESSOR_TEST(CircleInside2PtConicalEffect);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000947
joshualitt01258472014-09-22 10:29:30 -0700948/*
949 * All Two point conical gradient test create functions may occasionally create edge case shaders
950 */
joshualittb0a8a372014-09-23 09:50:21 -0700951GrFragmentProcessor* CircleInside2PtConicalEffect::TestCreate(SkRandom* random,
952 GrContext* context,
953 const GrDrawTargetCaps&,
954 GrTexture**) {
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000955 SkPoint center1 = {random->nextUScalar1(), random->nextUScalar1()};
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000956 SkScalar radius1 = random->nextUScalar1() + 0.0001f; // make sure radius1 != 0
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000957 SkPoint center2;
958 SkScalar radius2;
959 do {
960 center2.set(random->nextUScalar1(), random->nextUScalar1());
961 // Below makes sure that circle one is contained within circle two
962 SkScalar increase = random->nextUScalar1();
963 SkPoint diff = center2 - center1;
964 SkScalar diffLen = diff.length();
965 radius2 = radius1 + diffLen + increase;
966 // If the circles are identical the factory will give us an empty shader.
967 } while (radius1 == radius2 && center1 == center2);
968
969 SkColor colors[kMaxRandomGradientColors];
970 SkScalar stopsArray[kMaxRandomGradientColors];
971 SkScalar* stops = stopsArray;
972 SkShader::TileMode tm;
973 int colorCount = RandomGradientParams(random, colors, &stops, &tm);
974 SkAutoTUnref<SkShader> shader(SkGradientShader::CreateTwoPointConical(center1, radius1,
975 center2, radius2,
976 colors, stops, colorCount,
977 tm));
978 SkPaint paint;
bsalomon83d081a2014-07-08 09:56:10 -0700979 GrColor paintColor;
joshualittb0a8a372014-09-23 09:50:21 -0700980 GrFragmentProcessor* processor;
joshualitt5531d512014-12-17 15:50:11 -0800981 SkAssertResult(shader->asFragmentProcessor(context, paint,
joshualitt4eaf9ce2015-04-28 13:31:18 -0700982 GrTest::TestMatrix(random), NULL,
joshualitt5531d512014-12-17 15:50:11 -0800983 &paintColor, &processor));
joshualittb0a8a372014-09-23 09:50:21 -0700984 return processor;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000985}
986
joshualitteb2a6762014-12-04 11:35:33 -0800987GLCircleInside2PtConicalEffect::GLCircleInside2PtConicalEffect(const GrProcessor& processor)
988 : fVSVaryingName(NULL)
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000989 , fFSVaryingName(NULL)
990 , fCachedCenterX(SK_ScalarMax)
991 , fCachedCenterY(SK_ScalarMax)
992 , fCachedA(SK_ScalarMax)
993 , fCachedB(SK_ScalarMax)
994 , fCachedC(SK_ScalarMax) {}
995
joshualitt15988992014-10-09 15:04:05 -0700996void GLCircleInside2PtConicalEffect::emitCode(GrGLFPBuilder* builder,
joshualitt60030bc2014-11-25 14:21:55 -0800997 const GrFragmentProcessor& fp,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000998 const char* outputColor,
999 const char* inputColor,
1000 const TransformedCoordsArray& coords,
1001 const TextureSamplerArray& samplers) {
joshualitteb2a6762014-12-04 11:35:33 -08001002 const CircleInside2PtConicalEffect& ge = fp.cast<CircleInside2PtConicalEffect>();
joshualitt60030bc2014-11-25 14:21:55 -08001003 this->emitUniforms(builder, ge);
joshualitt30ba4362014-08-21 20:18:45 -07001004 fCenterUni = builder->addUniform(GrGLProgramBuilder::kFragment_Visibility,
bsalomon422f56f2014-12-09 10:18:12 -08001005 kVec2f_GrSLType, kDefault_GrSLPrecision,
1006 "Conical2FSCenter");
joshualitt30ba4362014-08-21 20:18:45 -07001007 fParamUni = builder->addUniform(GrGLProgramBuilder::kFragment_Visibility,
bsalomon422f56f2014-12-09 10:18:12 -08001008 kVec3f_GrSLType, kDefault_GrSLPrecision,
1009 "Conical2FSParams");
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001010 SkString tName("t");
1011
1012 GrGLShaderVar center = builder->getUniformVariable(fCenterUni);
1013 // params.x = A
1014 // params.y = B
1015 // params.z = C
1016 GrGLShaderVar params = builder->getUniformVariable(fParamUni);
1017
1018 // if we have a vec3 from being in perspective, convert it to a vec2 first
egdaniel29bee0f2015-04-29 11:54:42 -07001019 GrGLFragmentBuilder* fsBuilder = builder->getFragmentShaderBuilder();
joshualitt30ba4362014-08-21 20:18:45 -07001020 SkString coords2DString = fsBuilder->ensureFSCoords2D(coords, 0);
skia.committer@gmail.comede0c5c2014-04-23 03:04:11 +00001021 const char* coords2D = coords2DString.c_str();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001022
1023 // p = coords2D
1024 // e = center end
1025 // r = radius end
1026 // A = dot(e, e) - r^2 + 2 * r - 1
1027 // B = (r -1) / A
1028 // C = 1 / A
1029 // d = dot(e, p) + B
1030 // t = d +/- sqrt(d^2 - A * dot(p, p) + C)
joshualitt30ba4362014-08-21 20:18:45 -07001031 fsBuilder->codeAppendf("\tfloat pDotp = dot(%s, %s);\n", coords2D, coords2D);
joshualitt49586be2014-09-16 08:21:41 -07001032 fsBuilder->codeAppendf("\tfloat d = dot(%s, %s) + %s.y;\n", coords2D, center.c_str(),
1033 params.c_str());
joshualitt30ba4362014-08-21 20:18:45 -07001034 fsBuilder->codeAppendf("\tfloat %s = d + sqrt(d * d - %s.x * pDotp + %s.z);\n",
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001035 tName.c_str(), params.c_str(), params.c_str());
1036
joshualitt60030bc2014-11-25 14:21:55 -08001037 this->emitColor(builder, ge, tName.c_str(), outputColor, inputColor, samplers);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001038}
1039
kkinnunen7510b222014-07-30 00:04:16 -07001040void GLCircleInside2PtConicalEffect::setData(const GrGLProgramDataManager& pdman,
joshualittb0a8a372014-09-23 09:50:21 -07001041 const GrProcessor& processor) {
1042 INHERITED::setData(pdman, processor);
1043 const CircleInside2PtConicalEffect& data = processor.cast<CircleInside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001044 SkScalar centerX = data.centerX();
1045 SkScalar centerY = data.centerY();
1046 SkScalar A = data.A();
1047 SkScalar B = data.B();
1048 SkScalar C = data.C();
1049
1050 if (fCachedCenterX != centerX || fCachedCenterY != centerY ||
1051 fCachedA != A || fCachedB != B || fCachedC != C) {
1052
kkinnunen7510b222014-07-30 00:04:16 -07001053 pdman.set2f(fCenterUni, SkScalarToFloat(centerX), SkScalarToFloat(centerY));
1054 pdman.set3f(fParamUni, SkScalarToFloat(A), SkScalarToFloat(B), SkScalarToFloat(C));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001055
1056 fCachedCenterX = centerX;
1057 fCachedCenterY = centerY;
1058 fCachedA = A;
1059 fCachedB = B;
1060 fCachedC = C;
1061 }
1062}
1063
joshualittb0a8a372014-09-23 09:50:21 -07001064void GLCircleInside2PtConicalEffect::GenKey(const GrProcessor& processor,
jvanverthcfc18862015-04-28 08:48:20 -07001065 const GrGLSLCaps&, GrProcessorKeyBuilder* b) {
joshualittb0a8a372014-09-23 09:50:21 -07001066 b->add32(GenBaseGradientKey(processor));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001067}
1068
1069//////////////////////////////////////////////////////////////////////////////
1070
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001071class CircleOutside2PtConicalEffect : public GrGradientEffect {
1072public:
1073
joshualittb0a8a372014-09-23 09:50:21 -07001074 static GrFragmentProcessor* Create(GrContext* ctx,
1075 const SkTwoPointConicalGradient& shader,
1076 const SkMatrix& matrix,
1077 SkShader::TileMode tm,
1078 const CircleConicalInfo& info) {
bsalomon55fad7a2014-07-08 07:34:20 -07001079 return SkNEW_ARGS(CircleOutside2PtConicalEffect, (ctx, shader, matrix, tm, info));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001080 }
1081
1082 virtual ~CircleOutside2PtConicalEffect() {}
1083
mtklein36352bf2015-03-25 18:17:31 -07001084 const char* name() const override { return "Two-Point Conical Gradient Outside"; }
joshualitteb2a6762014-12-04 11:35:33 -08001085
jvanverthcfc18862015-04-28 08:48:20 -07001086 void getGLProcessorKey(const GrGLSLCaps&, GrProcessorKeyBuilder*) const override;
joshualitteb2a6762014-12-04 11:35:33 -08001087
mtklein36352bf2015-03-25 18:17:31 -07001088 GrGLFragmentProcessor* createGLInstance() const override;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001089
1090 SkScalar centerX() const { return fInfo.fCenterEnd.fX; }
1091 SkScalar centerY() const { return fInfo.fCenterEnd.fY; }
1092 SkScalar A() const { return fInfo.fA; }
1093 SkScalar B() const { return fInfo.fB; }
1094 SkScalar C() const { return fInfo.fC; }
1095 SkScalar tLimit() const { return fTLimit; }
1096 bool isFlipped() const { return fIsFlipped; }
1097
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001098private:
mtklein36352bf2015-03-25 18:17:31 -07001099 bool onIsEqual(const GrFragmentProcessor& sBase) const override {
joshualitt49586be2014-09-16 08:21:41 -07001100 const CircleOutside2PtConicalEffect& s = sBase.cast<CircleOutside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001101 return (INHERITED::onIsEqual(sBase) &&
1102 this->fInfo.fCenterEnd == s.fInfo.fCenterEnd &&
1103 this->fInfo.fA == s.fInfo.fA &&
1104 this->fInfo.fB == s.fInfo.fB &&
1105 this->fInfo.fC == s.fInfo.fC &&
1106 this->fTLimit == s.fTLimit &&
1107 this->fIsFlipped == s.fIsFlipped);
1108 }
1109
1110 CircleOutside2PtConicalEffect(GrContext* ctx,
1111 const SkTwoPointConicalGradient& shader,
1112 const SkMatrix& matrix,
1113 SkShader::TileMode tm,
1114 const CircleConicalInfo& info)
1115 : INHERITED(ctx, shader, matrix, tm), fInfo(info) {
joshualitteb2a6762014-12-04 11:35:33 -08001116 this->initClassID<CircleOutside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001117 if (shader.getStartRadius() != shader.getEndRadius()) {
reed80ea19c2015-05-12 10:37:34 -07001118 fTLimit = shader.getStartRadius() / (shader.getStartRadius() - shader.getEndRadius());
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001119 } else {
1120 fTLimit = SK_ScalarMin;
1121 }
1122
1123 fIsFlipped = shader.isFlippedGrad();
1124 }
1125
joshualittb0a8a372014-09-23 09:50:21 -07001126 GR_DECLARE_FRAGMENT_PROCESSOR_TEST;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001127
1128 const CircleConicalInfo fInfo;
1129 SkScalar fTLimit;
1130 bool fIsFlipped;
1131
1132 typedef GrGradientEffect INHERITED;
1133};
1134
1135class GLCircleOutside2PtConicalEffect : public GrGLGradientEffect {
1136public:
joshualitteb2a6762014-12-04 11:35:33 -08001137 GLCircleOutside2PtConicalEffect(const GrProcessor&);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001138 virtual ~GLCircleOutside2PtConicalEffect() {}
1139
joshualitt15988992014-10-09 15:04:05 -07001140 virtual void emitCode(GrGLFPBuilder*,
joshualittb0a8a372014-09-23 09:50:21 -07001141 const GrFragmentProcessor&,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001142 const char* outputColor,
1143 const char* inputColor,
1144 const TransformedCoordsArray&,
mtklein36352bf2015-03-25 18:17:31 -07001145 const TextureSamplerArray&) override;
1146 void setData(const GrGLProgramDataManager&, const GrProcessor&) override;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001147
jvanverthcfc18862015-04-28 08:48:20 -07001148 static void GenKey(const GrProcessor&, const GrGLSLCaps& caps, GrProcessorKeyBuilder* b);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001149
1150protected:
1151 UniformHandle fCenterUni;
1152 UniformHandle fParamUni;
1153
1154 const char* fVSVaryingName;
1155 const char* fFSVaryingName;
1156
1157 bool fIsFlipped;
1158
1159 // @{
1160 /// Values last uploaded as uniforms
1161
1162 SkScalar fCachedCenterX;
1163 SkScalar fCachedCenterY;
1164 SkScalar fCachedA;
1165 SkScalar fCachedB;
1166 SkScalar fCachedC;
1167 SkScalar fCachedTLimit;
1168
1169 // @}
1170
1171private:
1172 typedef GrGLGradientEffect INHERITED;
1173
1174};
1175
jvanverthcfc18862015-04-28 08:48:20 -07001176void CircleOutside2PtConicalEffect::getGLProcessorKey(const GrGLSLCaps& caps,
joshualitteb2a6762014-12-04 11:35:33 -08001177 GrProcessorKeyBuilder* b) const {
1178 GLCircleOutside2PtConicalEffect::GenKey(*this, caps, b);
1179}
1180
1181GrGLFragmentProcessor* CircleOutside2PtConicalEffect::createGLInstance() const {
1182 return SkNEW_ARGS(GLCircleOutside2PtConicalEffect, (*this));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001183}
1184
joshualittb0a8a372014-09-23 09:50:21 -07001185GR_DEFINE_FRAGMENT_PROCESSOR_TEST(CircleOutside2PtConicalEffect);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001186
joshualitt01258472014-09-22 10:29:30 -07001187/*
1188 * All Two point conical gradient test create functions may occasionally create edge case shaders
1189 */
joshualittb0a8a372014-09-23 09:50:21 -07001190GrFragmentProcessor* CircleOutside2PtConicalEffect::TestCreate(SkRandom* random,
1191 GrContext* context,
1192 const GrDrawTargetCaps&,
1193 GrTexture**) {
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001194 SkPoint center1 = {random->nextUScalar1(), random->nextUScalar1()};
commit-bot@chromium.org80894672014-04-22 21:24:22 +00001195 SkScalar radius1 = random->nextUScalar1() + 0.0001f; // make sure radius1 != 0
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001196 SkPoint center2;
1197 SkScalar radius2;
1198 SkScalar diffLen;
1199 do {
1200 center2.set(random->nextUScalar1(), random->nextUScalar1());
1201 // If the circles share a center than we can't be in the outside case
1202 } while (center1 == center2);
joshualitt01258472014-09-22 10:29:30 -07001203 SkPoint diff = center2 - center1;
1204 diffLen = diff.length();
1205 // Below makes sure that circle one is not contained within circle two
1206 // and have radius2 >= radius to match sorting on cpu side
1207 radius2 = radius1 + random->nextRangeF(0.f, diffLen);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001208
1209 SkColor colors[kMaxRandomGradientColors];
1210 SkScalar stopsArray[kMaxRandomGradientColors];
1211 SkScalar* stops = stopsArray;
1212 SkShader::TileMode tm;
1213 int colorCount = RandomGradientParams(random, colors, &stops, &tm);
1214 SkAutoTUnref<SkShader> shader(SkGradientShader::CreateTwoPointConical(center1, radius1,
1215 center2, radius2,
1216 colors, stops, colorCount,
1217 tm));
1218 SkPaint paint;
bsalomon83d081a2014-07-08 09:56:10 -07001219 GrColor paintColor;
joshualittb0a8a372014-09-23 09:50:21 -07001220 GrFragmentProcessor* processor;
joshualitt5531d512014-12-17 15:50:11 -08001221
1222 SkAssertResult(shader->asFragmentProcessor(context, paint,
joshualitt4eaf9ce2015-04-28 13:31:18 -07001223 GrTest::TestMatrix(random), NULL,
joshualitt5531d512014-12-17 15:50:11 -08001224 &paintColor, &processor));
joshualittb0a8a372014-09-23 09:50:21 -07001225 return processor;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001226}
1227
joshualitteb2a6762014-12-04 11:35:33 -08001228GLCircleOutside2PtConicalEffect::GLCircleOutside2PtConicalEffect(const GrProcessor& processor)
1229 : fVSVaryingName(NULL)
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001230 , fFSVaryingName(NULL)
1231 , fCachedCenterX(SK_ScalarMax)
1232 , fCachedCenterY(SK_ScalarMax)
1233 , fCachedA(SK_ScalarMax)
1234 , fCachedB(SK_ScalarMax)
1235 , fCachedC(SK_ScalarMax)
1236 , fCachedTLimit(SK_ScalarMax) {
joshualittb0a8a372014-09-23 09:50:21 -07001237 const CircleOutside2PtConicalEffect& data = processor.cast<CircleOutside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001238 fIsFlipped = data.isFlipped();
1239 }
1240
joshualitt15988992014-10-09 15:04:05 -07001241void GLCircleOutside2PtConicalEffect::emitCode(GrGLFPBuilder* builder,
joshualitt60030bc2014-11-25 14:21:55 -08001242 const GrFragmentProcessor& fp,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001243 const char* outputColor,
1244 const char* inputColor,
1245 const TransformedCoordsArray& coords,
1246 const TextureSamplerArray& samplers) {
joshualitteb2a6762014-12-04 11:35:33 -08001247 const CircleOutside2PtConicalEffect& ge = fp.cast<CircleOutside2PtConicalEffect>();
joshualitt60030bc2014-11-25 14:21:55 -08001248 this->emitUniforms(builder, ge);
joshualitt30ba4362014-08-21 20:18:45 -07001249 fCenterUni = builder->addUniform(GrGLProgramBuilder::kFragment_Visibility,
bsalomon422f56f2014-12-09 10:18:12 -08001250 kVec2f_GrSLType, kDefault_GrSLPrecision,
1251 "Conical2FSCenter");
joshualitt30ba4362014-08-21 20:18:45 -07001252 fParamUni = builder->addUniform(GrGLProgramBuilder::kFragment_Visibility,
bsalomon422f56f2014-12-09 10:18:12 -08001253 kVec4f_GrSLType, kDefault_GrSLPrecision,
1254 "Conical2FSParams");
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001255 SkString tName("t");
1256
1257 GrGLShaderVar center = builder->getUniformVariable(fCenterUni);
1258 // params.x = A
1259 // params.y = B
1260 // params.z = C
1261 GrGLShaderVar params = builder->getUniformVariable(fParamUni);
1262
1263 // if we have a vec3 from being in perspective, convert it to a vec2 first
egdaniel29bee0f2015-04-29 11:54:42 -07001264 GrGLFragmentBuilder* fsBuilder = builder->getFragmentShaderBuilder();
joshualitt30ba4362014-08-21 20:18:45 -07001265 SkString coords2DString = fsBuilder->ensureFSCoords2D(coords, 0);
skia.committer@gmail.comede0c5c2014-04-23 03:04:11 +00001266 const char* coords2D = coords2DString.c_str();
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001267
1268 // output will default to transparent black (we simply won't write anything
1269 // else to it if invalid, instead of discarding or returning prematurely)
joshualitt30ba4362014-08-21 20:18:45 -07001270 fsBuilder->codeAppendf("\t%s = vec4(0.0,0.0,0.0,0.0);\n", outputColor);
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001271
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001272 // p = coords2D
1273 // e = center end
1274 // r = radius end
1275 // A = dot(e, e) - r^2 + 2 * r - 1
1276 // B = (r -1) / A
1277 // C = 1 / A
1278 // d = dot(e, p) + B
1279 // t = d +/- sqrt(d^2 - A * dot(p, p) + C)
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001280
joshualitt30ba4362014-08-21 20:18:45 -07001281 fsBuilder->codeAppendf("\tfloat pDotp = dot(%s, %s);\n", coords2D, coords2D);
joshualitt49586be2014-09-16 08:21:41 -07001282 fsBuilder->codeAppendf("\tfloat d = dot(%s, %s) + %s.y;\n", coords2D, center.c_str(),
1283 params.c_str());
1284 fsBuilder->codeAppendf("\tfloat deter = d * d - %s.x * pDotp + %s.z;\n", params.c_str(),
1285 params.c_str());
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001286
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001287 // Must check to see if we flipped the circle order (to make sure start radius < end radius)
1288 // If so we must also flip sign on sqrt
1289 if (!fIsFlipped) {
joshualitt30ba4362014-08-21 20:18:45 -07001290 fsBuilder->codeAppendf("\tfloat %s = d + sqrt(deter);\n", tName.c_str());
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001291 } else {
joshualitt30ba4362014-08-21 20:18:45 -07001292 fsBuilder->codeAppendf("\tfloat %s = d - sqrt(deter);\n", tName.c_str());
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001293 }
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001294
joshualitt30ba4362014-08-21 20:18:45 -07001295 fsBuilder->codeAppendf("\tif (%s >= %s.w && deter >= 0.0) {\n", tName.c_str(), params.c_str());
1296 fsBuilder->codeAppend("\t\t");
joshualitt60030bc2014-11-25 14:21:55 -08001297 this->emitColor(builder, ge, tName.c_str(), outputColor, inputColor, samplers);
joshualitt30ba4362014-08-21 20:18:45 -07001298 fsBuilder->codeAppend("\t}\n");
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001299}
1300
kkinnunen7510b222014-07-30 00:04:16 -07001301void GLCircleOutside2PtConicalEffect::setData(const GrGLProgramDataManager& pdman,
joshualittb0a8a372014-09-23 09:50:21 -07001302 const GrProcessor& processor) {
1303 INHERITED::setData(pdman, processor);
1304 const CircleOutside2PtConicalEffect& data = processor.cast<CircleOutside2PtConicalEffect>();
commit-bot@chromium.org44d83c12014-04-21 13:10:25 +00001305 SkASSERT(data.isFlipped() == fIsFlipped);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001306 SkScalar centerX = data.centerX();
1307 SkScalar centerY = data.centerY();
1308 SkScalar A = data.A();
1309 SkScalar B = data.B();
1310 SkScalar C = data.C();
1311 SkScalar tLimit = data.tLimit();
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001312
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001313 if (fCachedCenterX != centerX || fCachedCenterY != centerY ||
1314 fCachedA != A || fCachedB != B || fCachedC != C || fCachedTLimit != tLimit) {
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001315
kkinnunen7510b222014-07-30 00:04:16 -07001316 pdman.set2f(fCenterUni, SkScalarToFloat(centerX), SkScalarToFloat(centerY));
1317 pdman.set4f(fParamUni, SkScalarToFloat(A), SkScalarToFloat(B), SkScalarToFloat(C),
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001318 SkScalarToFloat(tLimit));
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001319
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001320 fCachedCenterX = centerX;
1321 fCachedCenterY = centerY;
1322 fCachedA = A;
1323 fCachedB = B;
1324 fCachedC = C;
1325 fCachedTLimit = tLimit;
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001326 }
1327}
1328
joshualittb0a8a372014-09-23 09:50:21 -07001329void GLCircleOutside2PtConicalEffect::GenKey(const GrProcessor& processor,
jvanverthcfc18862015-04-28 08:48:20 -07001330 const GrGLSLCaps&, GrProcessorKeyBuilder* b) {
bsalomon63e99f72014-07-21 08:03:14 -07001331 uint32_t* key = b->add32n(2);
joshualittb0a8a372014-09-23 09:50:21 -07001332 key[0] = GenBaseGradientKey(processor);
1333 key[1] = processor.cast<CircleOutside2PtConicalEffect>().isFlipped();
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001334}
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +00001335
1336//////////////////////////////////////////////////////////////////////////////
1337
joshualittb0a8a372014-09-23 09:50:21 -07001338GrFragmentProcessor* Gr2PtConicalGradientEffect::Create(GrContext* ctx,
1339 const SkTwoPointConicalGradient& shader,
1340 SkShader::TileMode tm,
1341 const SkMatrix* localMatrix) {
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +00001342 SkMatrix matrix;
1343 if (!shader.getLocalMatrix().invert(&matrix)) {
1344 return NULL;
1345 }
commit-bot@chromium.org96fb7482014-05-09 20:28:11 +00001346 if (localMatrix) {
1347 SkMatrix inv;
1348 if (!localMatrix->invert(&inv)) {
1349 return NULL;
1350 }
1351 matrix.postConcat(inv);
1352 }
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +00001353
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001354 if (shader.getStartRadius() < kErrorTol) {
1355 SkScalar focalX;
1356 ConicalType type = set_matrix_focal_conical(shader, &matrix, &focalX);
1357 if (type == kInside_ConicalType) {
1358 return FocalInside2PtConicalEffect::Create(ctx, shader, matrix, tm, focalX);
1359 } else if(type == kEdge_ConicalType) {
1360 set_matrix_edge_conical(shader, &matrix);
1361 return Edge2PtConicalEffect::Create(ctx, shader, matrix, tm);
1362 } else {
1363 return FocalOutside2PtConicalEffect::Create(ctx, shader, matrix, tm, focalX);
1364 }
1365 }
1366
1367 CircleConicalInfo info;
1368 ConicalType type = set_matrix_circle_conical(shader, &matrix, &info);
1369
1370 if (type == kInside_ConicalType) {
1371 return CircleInside2PtConicalEffect::Create(ctx, shader, matrix, tm, info);
1372 } else if (type == kEdge_ConicalType) {
1373 set_matrix_edge_conical(shader, &matrix);
1374 return Edge2PtConicalEffect::Create(ctx, shader, matrix, tm);
1375 } else {
1376 return CircleOutside2PtConicalEffect::Create(ctx, shader, matrix, tm, info);
1377 }
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +00001378}
1379
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001380#endif