blob: 357e39c3adc6fee44097071b5dcae4eb6cd23311 [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,
62 const SkTwoPointConicalGradient& shader,
63 const SkMatrix& matrix,
64 SkShader::TileMode tm) {
bsalomon55fad7a2014-07-08 07:34:20 -070065 return SkNEW_ARGS(Edge2PtConicalEffect, (ctx, shader, matrix, tm));
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000066 }
67
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +000068 virtual ~Edge2PtConicalEffect() {}
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000069
mtklein36352bf2015-03-25 18:17:31 -070070 const char* name() const override {
joshualitteb2a6762014-12-04 11:35:33 -080071 return "Two-Point Conical Gradient Edge Touching";
72 }
73
jvanverthcfc18862015-04-28 08:48:20 -070074 void getGLProcessorKey(const GrGLSLCaps&, GrProcessorKeyBuilder*) const override;
joshualitteb2a6762014-12-04 11:35:33 -080075
mtklein36352bf2015-03-25 18:17:31 -070076 GrGLFragmentProcessor* createGLInstance() const override;
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000077
78 // The radial gradient parameters can collapse to a linear (instead of quadratic) equation.
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000079 SkScalar center() const { return fCenterX1; }
80 SkScalar diffRadius() const { return fDiffRadius; }
81 SkScalar radius() const { return fRadius0; }
82
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000083private:
mtklein36352bf2015-03-25 18:17:31 -070084 bool onIsEqual(const GrFragmentProcessor& sBase) const override {
joshualitt49586be2014-09-16 08:21:41 -070085 const Edge2PtConicalEffect& s = sBase.cast<Edge2PtConicalEffect>();
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000086 return (INHERITED::onIsEqual(sBase) &&
87 this->fCenterX1 == s.fCenterX1 &&
88 this->fRadius0 == s.fRadius0 &&
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +000089 this->fDiffRadius == s.fDiffRadius);
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000090 }
91
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +000092 Edge2PtConicalEffect(GrContext* ctx,
93 const SkTwoPointConicalGradient& shader,
94 const SkMatrix& matrix,
95 SkShader::TileMode tm)
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +000096 : INHERITED(ctx, shader, matrix, tm),
97 fCenterX1(shader.getCenterX1()),
98 fRadius0(shader.getStartRadius()),
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +000099 fDiffRadius(shader.getDiffRadius()){
joshualitteb2a6762014-12-04 11:35:33 -0800100 this->initClassID<Edge2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000101 // We should only be calling this shader if we are degenerate case with touching circles
egdaniel8405ef92014-06-09 11:57:28 -0700102 // When deciding if we are in edge case, we scaled by the end radius for cases when the
joshualitt01258472014-09-22 10:29:30 -0700103 // start radius was close to zero, otherwise we scaled by the start radius. In addition
104 // Our test for the edge case in set_matrix_circle_conical has a higher tolerance so we
105 // need the sqrt value below
106 SkASSERT(SkScalarAbs(SkScalarAbs(fDiffRadius) - fCenterX1) <
107 (fRadius0 < kErrorTol ? shader.getEndRadius() * kEdgeErrorTol :
108 fRadius0 * sqrt(kEdgeErrorTol)));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000109
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +0000110 // We pass the linear part of the quadratic as a varying.
111 // float b = -2.0 * (fCenterX1 * x + fRadius0 * fDiffRadius * z)
112 fBTransform = this->getCoordTransform();
113 SkMatrix& bMatrix = *fBTransform.accessMatrix();
114 SkScalar r0dr = SkScalarMul(fRadius0, fDiffRadius);
115 bMatrix[SkMatrix::kMScaleX] = -2 * (SkScalarMul(fCenterX1, bMatrix[SkMatrix::kMScaleX]) +
116 SkScalarMul(r0dr, bMatrix[SkMatrix::kMPersp0]));
117 bMatrix[SkMatrix::kMSkewX] = -2 * (SkScalarMul(fCenterX1, bMatrix[SkMatrix::kMSkewX]) +
118 SkScalarMul(r0dr, bMatrix[SkMatrix::kMPersp1]));
119 bMatrix[SkMatrix::kMTransX] = -2 * (SkScalarMul(fCenterX1, bMatrix[SkMatrix::kMTransX]) +
120 SkScalarMul(r0dr, bMatrix[SkMatrix::kMPersp2]));
121 this->addCoordTransform(&fBTransform);
122 }
123
joshualittb0a8a372014-09-23 09:50:21 -0700124 GR_DECLARE_FRAGMENT_PROCESSOR_TEST;
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +0000125
126 // @{
127 // Cache of values - these can change arbitrarily, EXCEPT
128 // we shouldn't change between degenerate and non-degenerate?!
129
130 GrCoordTransform fBTransform;
131 SkScalar fCenterX1;
132 SkScalar fRadius0;
133 SkScalar fDiffRadius;
134
135 // @}
136
137 typedef GrGradientEffect INHERITED;
138};
139
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000140class GLEdge2PtConicalEffect : public GrGLGradientEffect {
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +0000141public:
joshualitteb2a6762014-12-04 11:35:33 -0800142 GLEdge2PtConicalEffect(const GrProcessor&);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000143 virtual ~GLEdge2PtConicalEffect() { }
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000144
joshualitt15988992014-10-09 15:04:05 -0700145 virtual void emitCode(GrGLFPBuilder*,
joshualittb0a8a372014-09-23 09:50:21 -0700146 const GrFragmentProcessor&,
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000147 const char* outputColor,
148 const char* inputColor,
149 const TransformedCoordsArray&,
mtklein36352bf2015-03-25 18:17:31 -0700150 const TextureSamplerArray&) override;
151 void setData(const GrGLProgramDataManager&, const GrProcessor&) override;
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000152
jvanverthcfc18862015-04-28 08:48:20 -0700153 static void GenKey(const GrProcessor&, const GrGLSLCaps& caps, GrProcessorKeyBuilder* b);
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000154
155protected:
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000156 UniformHandle fParamUni;
157
158 const char* fVSVaryingName;
159 const char* fFSVaryingName;
160
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000161 // @{
162 /// Values last uploaded as uniforms
163
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000164 SkScalar fCachedRadius;
165 SkScalar fCachedDiffRadius;
166
167 // @}
168
169private:
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000170 typedef GrGLGradientEffect INHERITED;
171
172};
173
jvanverthcfc18862015-04-28 08:48:20 -0700174void Edge2PtConicalEffect::getGLProcessorKey(const GrGLSLCaps& caps,
joshualitteb2a6762014-12-04 11:35:33 -0800175 GrProcessorKeyBuilder* b) const {
176 GLEdge2PtConicalEffect::GenKey(*this, caps, b);
177}
178
179GrGLFragmentProcessor* Edge2PtConicalEffect::createGLInstance() const {
180 return SkNEW_ARGS(GLEdge2PtConicalEffect, (*this));
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000181}
skia.committer@gmail.com221b9112014-04-04 03:04:32 +0000182
joshualittb0a8a372014-09-23 09:50:21 -0700183GR_DEFINE_FRAGMENT_PROCESSOR_TEST(Edge2PtConicalEffect);
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000184
joshualitt01258472014-09-22 10:29:30 -0700185/*
186 * All Two point conical gradient test create functions may occasionally create edge case shaders
187 */
joshualittb0a8a372014-09-23 09:50:21 -0700188GrFragmentProcessor* Edge2PtConicalEffect::TestCreate(SkRandom* random,
189 GrContext* context,
bsalomon4b91f762015-05-19 09:29:46 -0700190 const GrCaps&,
joshualittb0a8a372014-09-23 09:50:21 -0700191 GrTexture**) {
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000192 SkPoint center1 = {random->nextUScalar1(), random->nextUScalar1()};
193 SkScalar radius1 = random->nextUScalar1();
194 SkPoint center2;
195 SkScalar radius2;
196 do {
197 center2.set(random->nextUScalar1(), random->nextUScalar1());
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000198 // If the circles are identical the factory will give us an empty shader.
199 // This will happen if we pick identical centers
200 } while (center1 == center2);
201
202 // Below makes sure that circle one is contained within circle two
203 // and both circles are touching on an edge
204 SkPoint diff = center2 - center1;
205 SkScalar diffLen = diff.length();
206 radius2 = radius1 + diffLen;
207
208 SkColor colors[kMaxRandomGradientColors];
209 SkScalar stopsArray[kMaxRandomGradientColors];
210 SkScalar* stops = stopsArray;
211 SkShader::TileMode tm;
212 int colorCount = RandomGradientParams(random, colors, &stops, &tm);
213 SkAutoTUnref<SkShader> shader(SkGradientShader::CreateTwoPointConical(center1, radius1,
214 center2, radius2,
215 colors, stops, colorCount,
216 tm));
217 SkPaint paint;
joshualittb0a8a372014-09-23 09:50:21 -0700218 GrFragmentProcessor* fp;
bsalomon83d081a2014-07-08 09:56:10 -0700219 GrColor paintColor;
joshualitt8ca93e72015-07-08 06:51:43 -0700220 GrPaint grPaint;
joshualitt5531d512014-12-17 15:50:11 -0800221 SkAssertResult(shader->asFragmentProcessor(context, paint,
joshualitt4eaf9ce2015-04-28 13:31:18 -0700222 GrTest::TestMatrix(random), NULL,
joshualitt8ca93e72015-07-08 06:51:43 -0700223 &paintColor, grPaint.getShaderDataManager(), &fp));
joshualittb0a8a372014-09-23 09:50:21 -0700224 return fp;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000225}
226
joshualitteb2a6762014-12-04 11:35:33 -0800227GLEdge2PtConicalEffect::GLEdge2PtConicalEffect(const GrProcessor&)
228 : fVSVaryingName(NULL)
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000229 , fFSVaryingName(NULL)
230 , fCachedRadius(-SK_ScalarMax)
231 , fCachedDiffRadius(-SK_ScalarMax) {}
232
joshualitt15988992014-10-09 15:04:05 -0700233void GLEdge2PtConicalEffect::emitCode(GrGLFPBuilder* builder,
joshualitt60030bc2014-11-25 14:21:55 -0800234 const GrFragmentProcessor& fp,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000235 const char* outputColor,
236 const char* inputColor,
237 const TransformedCoordsArray& coords,
238 const TextureSamplerArray& samplers) {
joshualitteb2a6762014-12-04 11:35:33 -0800239 const Edge2PtConicalEffect& ge = fp.cast<Edge2PtConicalEffect>();
joshualitt60030bc2014-11-25 14:21:55 -0800240 this->emitUniforms(builder, ge);
joshualitt30ba4362014-08-21 20:18:45 -0700241 fParamUni = builder->addUniformArray(GrGLProgramBuilder::kFragment_Visibility,
bsalomon422f56f2014-12-09 10:18:12 -0800242 kFloat_GrSLType, kDefault_GrSLPrecision,
243 "Conical2FSParams", 3);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000244
245 SkString cName("c");
246 SkString tName("t");
247 SkString p0; // start radius
248 SkString p1; // start radius squared
249 SkString p2; // difference in radii (r1 - r0)
250
251 builder->getUniformVariable(fParamUni).appendArrayAccess(0, &p0);
252 builder->getUniformVariable(fParamUni).appendArrayAccess(1, &p1);
253 builder->getUniformVariable(fParamUni).appendArrayAccess(2, &p2);
254
255 // We interpolate the linear component in coords[1].
joshualitt23e280d2014-09-18 12:26:38 -0700256 SkASSERT(coords[0].getType() == coords[1].getType());
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000257 const char* coords2D;
258 SkString bVar;
egdaniel29bee0f2015-04-29 11:54:42 -0700259 GrGLFragmentBuilder* fsBuilder = builder->getFragmentShaderBuilder();
joshualitt23e280d2014-09-18 12:26:38 -0700260 if (kVec3f_GrSLType == coords[0].getType()) {
joshualitt30ba4362014-08-21 20:18:45 -0700261 fsBuilder->codeAppendf("\tvec3 interpolants = vec3(%s.xy / %s.z, %s.x / %s.z);\n",
joshualitt49586be2014-09-16 08:21:41 -0700262 coords[0].c_str(), coords[0].c_str(), coords[1].c_str(),
263 coords[1].c_str());
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000264 coords2D = "interpolants.xy";
265 bVar = "interpolants.z";
266 } else {
267 coords2D = coords[0].c_str();
268 bVar.printf("%s.x", coords[1].c_str());
269 }
270
271 // output will default to transparent black (we simply won't write anything
272 // else to it if invalid, instead of discarding or returning prematurely)
joshualitt30ba4362014-08-21 20:18:45 -0700273 fsBuilder->codeAppendf("\t%s = vec4(0.0,0.0,0.0,0.0);\n", outputColor);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000274
275 // c = (x^2)+(y^2) - params[1]
joshualitt30ba4362014-08-21 20:18:45 -0700276 fsBuilder->codeAppendf("\tfloat %s = dot(%s, %s) - %s;\n",
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000277 cName.c_str(), coords2D, coords2D, p1.c_str());
278
279 // linear case: t = -c/b
joshualitt30ba4362014-08-21 20:18:45 -0700280 fsBuilder->codeAppendf("\tfloat %s = -(%s / %s);\n", tName.c_str(),
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000281 cName.c_str(), bVar.c_str());
282
283 // if r(t) > 0, then t will be the x coordinate
joshualitt30ba4362014-08-21 20:18:45 -0700284 fsBuilder->codeAppendf("\tif (%s * %s + %s > 0.0) {\n", tName.c_str(),
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000285 p2.c_str(), p0.c_str());
joshualitt30ba4362014-08-21 20:18:45 -0700286 fsBuilder->codeAppend("\t");
joshualitt60030bc2014-11-25 14:21:55 -0800287 this->emitColor(builder, ge, tName.c_str(), outputColor, inputColor, samplers);
joshualitt30ba4362014-08-21 20:18:45 -0700288 fsBuilder->codeAppend("\t}\n");
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000289}
290
kkinnunen7510b222014-07-30 00:04:16 -0700291void GLEdge2PtConicalEffect::setData(const GrGLProgramDataManager& pdman,
joshualittb0a8a372014-09-23 09:50:21 -0700292 const GrProcessor& processor) {
293 INHERITED::setData(pdman, processor);
294 const Edge2PtConicalEffect& data = processor.cast<Edge2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000295 SkScalar radius0 = data.radius();
296 SkScalar diffRadius = data.diffRadius();
297
298 if (fCachedRadius != radius0 ||
299 fCachedDiffRadius != diffRadius) {
300
301 float values[3] = {
302 SkScalarToFloat(radius0),
303 SkScalarToFloat(SkScalarMul(radius0, radius0)),
304 SkScalarToFloat(diffRadius)
305 };
306
kkinnunen7510b222014-07-30 00:04:16 -0700307 pdman.set1fv(fParamUni, 3, values);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000308 fCachedRadius = radius0;
309 fCachedDiffRadius = diffRadius;
310 }
311}
312
joshualittb0a8a372014-09-23 09:50:21 -0700313void GLEdge2PtConicalEffect::GenKey(const GrProcessor& processor,
jvanverthcfc18862015-04-28 08:48:20 -0700314 const GrGLSLCaps&, GrProcessorKeyBuilder* b) {
joshualittb0a8a372014-09-23 09:50:21 -0700315 b->add32(GenBaseGradientKey(processor));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000316}
317
318//////////////////////////////////////////////////////////////////////////////
319// Focal Conical Gradients
320//////////////////////////////////////////////////////////////////////////////
321
322static ConicalType set_matrix_focal_conical(const SkTwoPointConicalGradient& shader,
323 SkMatrix* invLMatrix, SkScalar* focalX) {
324 // Inverse of the current local matrix is passed in then,
325 // translate, scale, and rotate such that endCircle is unit circle on x-axis,
326 // and focal point is at the origin.
327 ConicalType conicalType;
328 const SkPoint& focal = shader.getStartCenter();
329 const SkPoint& centerEnd = shader.getEndCenter();
330 SkScalar radius = shader.getEndRadius();
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000331 SkScalar invRadius = 1.f / radius;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000332
333 SkMatrix matrix;
334
335 matrix.setTranslate(-centerEnd.fX, -centerEnd.fY);
336 matrix.postScale(invRadius, invRadius);
337
338 SkPoint focalTrans;
339 matrix.mapPoints(&focalTrans, &focal, 1);
340 *focalX = focalTrans.length();
341
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000342 if (0.f != *focalX) {
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000343 SkScalar invFocalX = SkScalarInvert(*focalX);
344 SkMatrix rot;
345 rot.setSinCos(-SkScalarMul(invFocalX, focalTrans.fY),
346 SkScalarMul(invFocalX, focalTrans.fX));
347 matrix.postConcat(rot);
348 }
349
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000350 matrix.postTranslate(-(*focalX), 0.f);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000351
352 // If the focal point is touching the edge of the circle it will
353 // cause a degenerate case that must be handled separately
egdaniel8405ef92014-06-09 11:57:28 -0700354 // kEdgeErrorTol = 5 * kErrorTol was picked after manual testing the
355 // stability trade off versus the linear approx used in the Edge Shader
356 if (SkScalarAbs(1.f - (*focalX)) < kEdgeErrorTol) {
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000357 return kEdge_ConicalType;
358 }
359
360 // Scale factor 1 / (1 - focalX * focalX)
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000361 SkScalar oneMinusF2 = 1.f - SkScalarMul(*focalX, *focalX);
reed80ea19c2015-05-12 10:37:34 -0700362 SkScalar s = SkScalarInvert(oneMinusF2);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000363
364
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000365 if (s >= 0.f) {
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000366 conicalType = kInside_ConicalType;
367 matrix.postScale(s, s * SkScalarSqrt(oneMinusF2));
368 } else {
369 conicalType = kOutside_ConicalType;
370 matrix.postScale(s, s);
371 }
372
373 invLMatrix->postConcat(matrix);
374
375 return conicalType;
376}
377
378//////////////////////////////////////////////////////////////////////////////
379
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000380class FocalOutside2PtConicalEffect : public GrGradientEffect {
381public:
382
joshualittb0a8a372014-09-23 09:50:21 -0700383 static GrFragmentProcessor* Create(GrContext* ctx,
384 const SkTwoPointConicalGradient& shader,
385 const SkMatrix& matrix,
386 SkShader::TileMode tm,
387 SkScalar focalX) {
bsalomon55fad7a2014-07-08 07:34:20 -0700388 return SkNEW_ARGS(FocalOutside2PtConicalEffect, (ctx, shader, matrix, tm, 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,
413 const SkTwoPointConicalGradient& shader,
414 const SkMatrix& matrix,
415 SkShader::TileMode tm,
416 SkScalar focalX)
joshualitteb2a6762014-12-04 11:35:33 -0800417 : INHERITED(ctx, shader, matrix, tm), fFocalX(focalX), fIsFlipped(shader.isFlippedGrad()) {
418 this->initClassID<FocalOutside2PtConicalEffect>();
419 }
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000420
joshualittb0a8a372014-09-23 09:50:21 -0700421 GR_DECLARE_FRAGMENT_PROCESSOR_TEST;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000422
423 SkScalar fFocalX;
424 bool fIsFlipped;
425
426 typedef GrGradientEffect INHERITED;
427};
428
429class GLFocalOutside2PtConicalEffect : public GrGLGradientEffect {
430public:
joshualitteb2a6762014-12-04 11:35:33 -0800431 GLFocalOutside2PtConicalEffect(const GrProcessor&);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000432 virtual ~GLFocalOutside2PtConicalEffect() { }
433
joshualitt15988992014-10-09 15:04:05 -0700434 virtual void emitCode(GrGLFPBuilder*,
joshualittb0a8a372014-09-23 09:50:21 -0700435 const GrFragmentProcessor&,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000436 const char* outputColor,
437 const char* inputColor,
438 const TransformedCoordsArray&,
mtklein36352bf2015-03-25 18:17:31 -0700439 const TextureSamplerArray&) override;
440 void setData(const GrGLProgramDataManager&, const GrProcessor&) override;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000441
jvanverthcfc18862015-04-28 08:48:20 -0700442 static void GenKey(const GrProcessor&, const GrGLSLCaps& caps, GrProcessorKeyBuilder* b);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000443
444protected:
445 UniformHandle fParamUni;
446
447 const char* fVSVaryingName;
448 const char* fFSVaryingName;
449
450 bool fIsFlipped;
451
452 // @{
453 /// Values last uploaded as uniforms
454
455 SkScalar fCachedFocal;
456
457 // @}
458
459private:
460 typedef GrGLGradientEffect INHERITED;
461
462};
463
jvanverthcfc18862015-04-28 08:48:20 -0700464void FocalOutside2PtConicalEffect::getGLProcessorKey(const GrGLSLCaps& caps,
joshualitteb2a6762014-12-04 11:35:33 -0800465 GrProcessorKeyBuilder* b) const {
466 GLFocalOutside2PtConicalEffect::GenKey(*this, caps, b);
467}
468
469GrGLFragmentProcessor* FocalOutside2PtConicalEffect::createGLInstance() const {
470 return SkNEW_ARGS(GLFocalOutside2PtConicalEffect, (*this));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000471}
472
joshualittb0a8a372014-09-23 09:50:21 -0700473GR_DEFINE_FRAGMENT_PROCESSOR_TEST(FocalOutside2PtConicalEffect);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000474
joshualitt01258472014-09-22 10:29:30 -0700475/*
476 * All Two point conical gradient test create functions may occasionally create edge case shaders
477 */
joshualittb0a8a372014-09-23 09:50:21 -0700478GrFragmentProcessor* FocalOutside2PtConicalEffect::TestCreate(SkRandom* random,
479 GrContext* context,
bsalomon4b91f762015-05-19 09:29:46 -0700480 const GrCaps&,
joshualittb0a8a372014-09-23 09:50:21 -0700481 GrTexture**) {
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000482 SkPoint center1 = {random->nextUScalar1(), random->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 {
487 center2.set(random->nextUScalar1(), random->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
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000493 radius2 = random->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;
499 int colorCount = RandomGradientParams(random, colors, &stops, &tm);
500 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;
joshualitt5531d512014-12-17 15:50:11 -0800508 SkAssertResult(shader->asFragmentProcessor(context, paint,
joshualitt4eaf9ce2015-04-28 13:31:18 -0700509 GrTest::TestMatrix(random), NULL,
joshualitt8ca93e72015-07-08 06:51:43 -0700510 &paintColor, grPaint.getShaderDataManager(),
511 &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,
607 const SkTwoPointConicalGradient& shader,
608 const SkMatrix& matrix,
609 SkShader::TileMode tm,
610 SkScalar focalX) {
bsalomon55fad7a2014-07-08 07:34:20 -0700611 return SkNEW_ARGS(FocalInside2PtConicalEffect, (ctx, shader, matrix, tm, focalX));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000612 }
613
614 virtual ~FocalInside2PtConicalEffect() {}
615
mtklein36352bf2015-03-25 18:17:31 -0700616 const char* name() const override {
joshualitteb2a6762014-12-04 11:35:33 -0800617 return "Two-Point Conical Gradient Focal Inside";
618 }
619
jvanverthcfc18862015-04-28 08:48:20 -0700620 void getGLProcessorKey(const GrGLSLCaps&, GrProcessorKeyBuilder*) const override;
joshualitteb2a6762014-12-04 11:35:33 -0800621
mtklein36352bf2015-03-25 18:17:31 -0700622 GrGLFragmentProcessor* createGLInstance() const override;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000623
624 SkScalar focal() const { return fFocalX; }
625
joshualittb0a8a372014-09-23 09:50:21 -0700626 typedef GLFocalInside2PtConicalEffect GLProcessor;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000627
628private:
mtklein36352bf2015-03-25 18:17:31 -0700629 bool onIsEqual(const GrFragmentProcessor& sBase) const override {
joshualitt49586be2014-09-16 08:21:41 -0700630 const FocalInside2PtConicalEffect& s = sBase.cast<FocalInside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000631 return (INHERITED::onIsEqual(sBase) &&
632 this->fFocalX == s.fFocalX);
633 }
634
635 FocalInside2PtConicalEffect(GrContext* ctx,
636 const SkTwoPointConicalGradient& shader,
637 const SkMatrix& matrix,
638 SkShader::TileMode tm,
639 SkScalar focalX)
joshualitteb2a6762014-12-04 11:35:33 -0800640 : INHERITED(ctx, shader, matrix, tm), fFocalX(focalX) {
641 this->initClassID<FocalInside2PtConicalEffect>();
642 }
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000643
joshualittb0a8a372014-09-23 09:50:21 -0700644 GR_DECLARE_FRAGMENT_PROCESSOR_TEST;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000645
646 SkScalar fFocalX;
647
648 typedef GrGradientEffect INHERITED;
649};
650
651class GLFocalInside2PtConicalEffect : public GrGLGradientEffect {
652public:
joshualitteb2a6762014-12-04 11:35:33 -0800653 GLFocalInside2PtConicalEffect(const GrProcessor&);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000654 virtual ~GLFocalInside2PtConicalEffect() {}
655
joshualitt15988992014-10-09 15:04:05 -0700656 virtual void emitCode(GrGLFPBuilder*,
joshualittb0a8a372014-09-23 09:50:21 -0700657 const GrFragmentProcessor&,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000658 const char* outputColor,
659 const char* inputColor,
660 const TransformedCoordsArray&,
mtklein36352bf2015-03-25 18:17:31 -0700661 const TextureSamplerArray&) override;
662 void setData(const GrGLProgramDataManager&, const GrProcessor&) override;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000663
jvanverthcfc18862015-04-28 08:48:20 -0700664 static void GenKey(const GrProcessor&, const GrGLSLCaps& caps, GrProcessorKeyBuilder* b);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000665
666protected:
667 UniformHandle fFocalUni;
668
669 const char* fVSVaryingName;
670 const char* fFSVaryingName;
671
672 // @{
673 /// Values last uploaded as uniforms
674
675 SkScalar fCachedFocal;
676
677 // @}
678
679private:
680 typedef GrGLGradientEffect INHERITED;
681
682};
683
jvanverthcfc18862015-04-28 08:48:20 -0700684void FocalInside2PtConicalEffect::getGLProcessorKey(const GrGLSLCaps& caps,
joshualitteb2a6762014-12-04 11:35:33 -0800685 GrProcessorKeyBuilder* b) const {
686 GLFocalInside2PtConicalEffect::GenKey(*this, caps, b);
687}
688
689GrGLFragmentProcessor* FocalInside2PtConicalEffect::createGLInstance() const {
690 return SkNEW_ARGS(GLFocalInside2PtConicalEffect, (*this));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000691}
692
joshualittb0a8a372014-09-23 09:50:21 -0700693GR_DEFINE_FRAGMENT_PROCESSOR_TEST(FocalInside2PtConicalEffect);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000694
joshualitt01258472014-09-22 10:29:30 -0700695/*
696 * All Two point conical gradient test create functions may occasionally create edge case shaders
697 */
joshualittb0a8a372014-09-23 09:50:21 -0700698GrFragmentProcessor* FocalInside2PtConicalEffect::TestCreate(SkRandom* random,
699 GrContext* context,
bsalomon4b91f762015-05-19 09:29:46 -0700700 const GrCaps&,
joshualittb0a8a372014-09-23 09:50:21 -0700701 GrTexture**) {
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000702 SkPoint center1 = {random->nextUScalar1(), random->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 {
707 center2.set(random->nextUScalar1(), random->nextUScalar1());
708 // Below makes sure radius2 is larger enouch such that the focal point
709 // is inside the end circle
710 SkScalar increase = random->nextUScalar1();
711 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;
721 int colorCount = RandomGradientParams(random, colors, &stops, &tm);
722 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;
joshualitt8ca93e72015-07-08 06:51:43 -0700729 GrPaint grPaint;
joshualitt5531d512014-12-17 15:50:11 -0800730 SkAssertResult(shader->asFragmentProcessor(context, paint,
joshualitt4eaf9ce2015-04-28 13:31:18 -0700731 GrTest::TestMatrix(random), NULL,
joshualitt8ca93e72015-07-08 06:51:43 -0700732 &paintColor, grPaint.getShaderDataManager(), &fp));
joshualittb0a8a372014-09-23 09:50:21 -0700733 return fp;
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000734}
735
joshualitteb2a6762014-12-04 11:35:33 -0800736GLFocalInside2PtConicalEffect::GLFocalInside2PtConicalEffect(const GrProcessor&)
737 : fVSVaryingName(NULL)
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000738 , fFSVaryingName(NULL)
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000739 , fCachedFocal(SK_ScalarMax) {}
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000740
joshualitt15988992014-10-09 15:04:05 -0700741void GLFocalInside2PtConicalEffect::emitCode(GrGLFPBuilder* builder,
joshualitt60030bc2014-11-25 14:21:55 -0800742 const GrFragmentProcessor& fp,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000743 const char* outputColor,
744 const char* inputColor,
745 const TransformedCoordsArray& coords,
746 const TextureSamplerArray& samplers) {
joshualitteb2a6762014-12-04 11:35:33 -0800747 const FocalInside2PtConicalEffect& ge = fp.cast<FocalInside2PtConicalEffect>();
joshualitt60030bc2014-11-25 14:21:55 -0800748 this->emitUniforms(builder, ge);
joshualitt30ba4362014-08-21 20:18:45 -0700749 fFocalUni = builder->addUniform(GrGLProgramBuilder::kFragment_Visibility,
bsalomon422f56f2014-12-09 10:18:12 -0800750 kFloat_GrSLType, kDefault_GrSLPrecision,
751 "Conical2FSParams");
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000752 SkString tName("t");
753
754 // this is the distance along x-axis from the end center to focal point in
755 // transformed coordinates
756 GrGLShaderVar focal = builder->getUniformVariable(fFocalUni);
757
758 // if we have a vec3 from being in perspective, convert it to a vec2 first
egdaniel29bee0f2015-04-29 11:54:42 -0700759 GrGLFragmentBuilder* fsBuilder = builder->getFragmentShaderBuilder();
joshualitt30ba4362014-08-21 20:18:45 -0700760 SkString coords2DString = fsBuilder->ensureFSCoords2D(coords, 0);
skia.committer@gmail.comede0c5c2014-04-23 03:04:11 +0000761 const char* coords2D = coords2DString.c_str();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000762
763 // t = p.x * focalX + length(p)
joshualitt30ba4362014-08-21 20:18:45 -0700764 fsBuilder->codeAppendf("\tfloat %s = %s.x * %s + length(%s);\n", tName.c_str(),
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000765 coords2D, focal.c_str(), coords2D);
766
joshualitt60030bc2014-11-25 14:21:55 -0800767 this->emitColor(builder, ge, tName.c_str(), outputColor, inputColor, samplers);
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000768}
769
kkinnunen7510b222014-07-30 00:04:16 -0700770void GLFocalInside2PtConicalEffect::setData(const GrGLProgramDataManager& pdman,
joshualittb0a8a372014-09-23 09:50:21 -0700771 const GrProcessor& processor) {
772 INHERITED::setData(pdman, processor);
773 const FocalInside2PtConicalEffect& data = processor.cast<FocalInside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000774 SkScalar focal = data.focal();
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000775
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000776 if (fCachedFocal != focal) {
kkinnunen7510b222014-07-30 00:04:16 -0700777 pdman.set1f(fFocalUni, SkScalarToFloat(focal));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000778 fCachedFocal = focal;
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +0000779 }
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000780}
781
joshualittb0a8a372014-09-23 09:50:21 -0700782void GLFocalInside2PtConicalEffect::GenKey(const GrProcessor& processor,
jvanverthcfc18862015-04-28 08:48:20 -0700783 const GrGLSLCaps&, GrProcessorKeyBuilder* b) {
joshualittb0a8a372014-09-23 09:50:21 -0700784 b->add32(GenBaseGradientKey(processor));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000785}
786
787//////////////////////////////////////////////////////////////////////////////
788// Circle Conical Gradients
789//////////////////////////////////////////////////////////////////////////////
790
791struct CircleConicalInfo {
792 SkPoint fCenterEnd;
793 SkScalar fA;
794 SkScalar fB;
795 SkScalar fC;
796};
797
798// Returns focal distance along x-axis in transformed coords
799static ConicalType set_matrix_circle_conical(const SkTwoPointConicalGradient& shader,
800 SkMatrix* invLMatrix, CircleConicalInfo* info) {
801 // Inverse of the current local matrix is passed in then,
802 // translate and scale such that start circle is on the origin and has radius 1
803 const SkPoint& centerStart = shader.getStartCenter();
804 const SkPoint& centerEnd = shader.getEndCenter();
805 SkScalar radiusStart = shader.getStartRadius();
806 SkScalar radiusEnd = shader.getEndRadius();
807
808 SkMatrix matrix;
809
810 matrix.setTranslate(-centerStart.fX, -centerStart.fY);
811
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000812 SkScalar invStartRad = 1.f / radiusStart;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000813 matrix.postScale(invStartRad, invStartRad);
814
815 radiusEnd /= radiusStart;
816
817 SkPoint centerEndTrans;
818 matrix.mapPoints(&centerEndTrans, &centerEnd, 1);
819
820 SkScalar A = centerEndTrans.fX * centerEndTrans.fX + centerEndTrans.fY * centerEndTrans.fY
821 - radiusEnd * radiusEnd + 2 * radiusEnd - 1;
822
823 // Check to see if start circle is inside end circle with edges touching.
824 // If touching we return that it is of kEdge_ConicalType, and leave the matrix setting
egdaniel8405ef92014-06-09 11:57:28 -0700825 // to the edge shader. kEdgeErrorTol = 5 * kErrorTol was picked after manual testing
826 // so that C = 1 / A is stable, and the linear approximation used in the Edge shader is
827 // still accurate.
828 if (SkScalarAbs(A) < kEdgeErrorTol) {
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000829 return kEdge_ConicalType;
830 }
831
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000832 SkScalar C = 1.f / A;
833 SkScalar B = (radiusEnd - 1.f) * C;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000834
835 matrix.postScale(C, C);
836
837 invLMatrix->postConcat(matrix);
838
839 info->fCenterEnd = centerEndTrans;
840 info->fA = A;
841 info->fB = B;
842 info->fC = C;
843
844 // 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 +0000845 if (A < 0.f) {
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000846 return kInside_ConicalType;
847 }
848 return kOutside_ConicalType;
849}
850
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000851class CircleInside2PtConicalEffect : public GrGradientEffect {
852public:
853
joshualittb0a8a372014-09-23 09:50:21 -0700854 static GrFragmentProcessor* Create(GrContext* ctx,
855 const SkTwoPointConicalGradient& shader,
856 const SkMatrix& matrix,
857 SkShader::TileMode tm,
858 const CircleConicalInfo& info) {
bsalomon55fad7a2014-07-08 07:34:20 -0700859 return SkNEW_ARGS(CircleInside2PtConicalEffect, (ctx, shader, matrix, tm, info));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000860 }
861
862 virtual ~CircleInside2PtConicalEffect() {}
863
mtklein36352bf2015-03-25 18:17:31 -0700864 const char* name() const override { return "Two-Point Conical Gradient Inside"; }
joshualitteb2a6762014-12-04 11:35:33 -0800865
jvanverthcfc18862015-04-28 08:48:20 -0700866 virtual void getGLProcessorKey(const GrGLSLCaps& caps,
mtklein36352bf2015-03-25 18:17:31 -0700867 GrProcessorKeyBuilder* b) const override;
joshualitteb2a6762014-12-04 11:35:33 -0800868
mtklein36352bf2015-03-25 18:17:31 -0700869 GrGLFragmentProcessor* createGLInstance() const override;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000870
871 SkScalar centerX() const { return fInfo.fCenterEnd.fX; }
872 SkScalar centerY() const { return fInfo.fCenterEnd.fY; }
873 SkScalar A() const { return fInfo.fA; }
874 SkScalar B() const { return fInfo.fB; }
875 SkScalar C() const { return fInfo.fC; }
876
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000877private:
mtklein36352bf2015-03-25 18:17:31 -0700878 bool onIsEqual(const GrFragmentProcessor& sBase) const override {
joshualitt49586be2014-09-16 08:21:41 -0700879 const CircleInside2PtConicalEffect& s = sBase.cast<CircleInside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000880 return (INHERITED::onIsEqual(sBase) &&
881 this->fInfo.fCenterEnd == s.fInfo.fCenterEnd &&
882 this->fInfo.fA == s.fInfo.fA &&
883 this->fInfo.fB == s.fInfo.fB &&
884 this->fInfo.fC == s.fInfo.fC);
885 }
886
887 CircleInside2PtConicalEffect(GrContext* ctx,
888 const SkTwoPointConicalGradient& shader,
889 const SkMatrix& matrix,
890 SkShader::TileMode tm,
891 const CircleConicalInfo& info)
joshualitteb2a6762014-12-04 11:35:33 -0800892 : INHERITED(ctx, shader, matrix, tm), fInfo(info) {
893 this->initClassID<CircleInside2PtConicalEffect>();
894 }
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000895
joshualittb0a8a372014-09-23 09:50:21 -0700896 GR_DECLARE_FRAGMENT_PROCESSOR_TEST;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000897
898 const CircleConicalInfo fInfo;
899
900 typedef GrGradientEffect INHERITED;
901};
902
903class GLCircleInside2PtConicalEffect : public GrGLGradientEffect {
904public:
joshualitteb2a6762014-12-04 11:35:33 -0800905 GLCircleInside2PtConicalEffect(const GrProcessor&);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000906 virtual ~GLCircleInside2PtConicalEffect() {}
907
joshualitt15988992014-10-09 15:04:05 -0700908 virtual void emitCode(GrGLFPBuilder*,
joshualittb0a8a372014-09-23 09:50:21 -0700909 const GrFragmentProcessor&,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000910 const char* outputColor,
911 const char* inputColor,
912 const TransformedCoordsArray&,
mtklein36352bf2015-03-25 18:17:31 -0700913 const TextureSamplerArray&) override;
914 void setData(const GrGLProgramDataManager&, const GrProcessor&) override;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000915
jvanverthcfc18862015-04-28 08:48:20 -0700916 static void GenKey(const GrProcessor&, const GrGLSLCaps& caps, GrProcessorKeyBuilder* b);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000917
918protected:
919 UniformHandle fCenterUni;
920 UniformHandle fParamUni;
921
922 const char* fVSVaryingName;
923 const char* fFSVaryingName;
924
925 // @{
926 /// Values last uploaded as uniforms
927
928 SkScalar fCachedCenterX;
929 SkScalar fCachedCenterY;
930 SkScalar fCachedA;
931 SkScalar fCachedB;
932 SkScalar fCachedC;
933
934 // @}
935
936private:
937 typedef GrGLGradientEffect INHERITED;
938
939};
940
jvanverthcfc18862015-04-28 08:48:20 -0700941void CircleInside2PtConicalEffect::getGLProcessorKey(const GrGLSLCaps& caps,
joshualitteb2a6762014-12-04 11:35:33 -0800942 GrProcessorKeyBuilder* b) const {
943 GLCircleInside2PtConicalEffect::GenKey(*this, caps, b);
944}
945
946GrGLFragmentProcessor* CircleInside2PtConicalEffect::createGLInstance() const {
947 return SkNEW_ARGS(GLCircleInside2PtConicalEffect, (*this));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000948}
949
joshualittb0a8a372014-09-23 09:50:21 -0700950GR_DEFINE_FRAGMENT_PROCESSOR_TEST(CircleInside2PtConicalEffect);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000951
joshualitt01258472014-09-22 10:29:30 -0700952/*
953 * All Two point conical gradient test create functions may occasionally create edge case shaders
954 */
joshualittb0a8a372014-09-23 09:50:21 -0700955GrFragmentProcessor* CircleInside2PtConicalEffect::TestCreate(SkRandom* random,
956 GrContext* context,
bsalomon4b91f762015-05-19 09:29:46 -0700957 const GrCaps&,
joshualittb0a8a372014-09-23 09:50:21 -0700958 GrTexture**) {
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000959 SkPoint center1 = {random->nextUScalar1(), random->nextUScalar1()};
commit-bot@chromium.org80894672014-04-22 21:24:22 +0000960 SkScalar radius1 = random->nextUScalar1() + 0.0001f; // make sure radius1 != 0
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000961 SkPoint center2;
962 SkScalar radius2;
963 do {
964 center2.set(random->nextUScalar1(), random->nextUScalar1());
965 // Below makes sure that circle one is contained within circle two
966 SkScalar increase = random->nextUScalar1();
967 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;
977 int colorCount = RandomGradientParams(random, colors, &stops, &tm);
978 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;
985 GrPaint grPaint;
joshualitt5531d512014-12-17 15:50:11 -0800986 SkAssertResult(shader->asFragmentProcessor(context, paint,
joshualitt4eaf9ce2015-04-28 13:31:18 -0700987 GrTest::TestMatrix(random), NULL,
joshualitt8ca93e72015-07-08 06:51:43 -0700988 &paintColor, grPaint.getShaderDataManager(), &fp));
989 return fp;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000990}
991
joshualitteb2a6762014-12-04 11:35:33 -0800992GLCircleInside2PtConicalEffect::GLCircleInside2PtConicalEffect(const GrProcessor& processor)
993 : fVSVaryingName(NULL)
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +0000994 , fFSVaryingName(NULL)
995 , fCachedCenterX(SK_ScalarMax)
996 , fCachedCenterY(SK_ScalarMax)
997 , fCachedA(SK_ScalarMax)
998 , fCachedB(SK_ScalarMax)
999 , fCachedC(SK_ScalarMax) {}
1000
joshualitt15988992014-10-09 15:04:05 -07001001void GLCircleInside2PtConicalEffect::emitCode(GrGLFPBuilder* builder,
joshualitt60030bc2014-11-25 14:21:55 -08001002 const GrFragmentProcessor& fp,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001003 const char* outputColor,
1004 const char* inputColor,
1005 const TransformedCoordsArray& coords,
1006 const TextureSamplerArray& samplers) {
joshualitteb2a6762014-12-04 11:35:33 -08001007 const CircleInside2PtConicalEffect& ge = fp.cast<CircleInside2PtConicalEffect>();
joshualitt60030bc2014-11-25 14:21:55 -08001008 this->emitUniforms(builder, ge);
joshualitt30ba4362014-08-21 20:18:45 -07001009 fCenterUni = builder->addUniform(GrGLProgramBuilder::kFragment_Visibility,
bsalomon422f56f2014-12-09 10:18:12 -08001010 kVec2f_GrSLType, kDefault_GrSLPrecision,
1011 "Conical2FSCenter");
joshualitt30ba4362014-08-21 20:18:45 -07001012 fParamUni = builder->addUniform(GrGLProgramBuilder::kFragment_Visibility,
bsalomon422f56f2014-12-09 10:18:12 -08001013 kVec3f_GrSLType, kDefault_GrSLPrecision,
1014 "Conical2FSParams");
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001015 SkString tName("t");
1016
1017 GrGLShaderVar center = builder->getUniformVariable(fCenterUni);
1018 // params.x = A
1019 // params.y = B
1020 // params.z = C
1021 GrGLShaderVar params = builder->getUniformVariable(fParamUni);
1022
1023 // if we have a vec3 from being in perspective, convert it to a vec2 first
egdaniel29bee0f2015-04-29 11:54:42 -07001024 GrGLFragmentBuilder* fsBuilder = builder->getFragmentShaderBuilder();
joshualitt30ba4362014-08-21 20:18:45 -07001025 SkString coords2DString = fsBuilder->ensureFSCoords2D(coords, 0);
skia.committer@gmail.comede0c5c2014-04-23 03:04:11 +00001026 const char* coords2D = coords2DString.c_str();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001027
1028 // p = coords2D
1029 // e = center end
1030 // r = radius end
1031 // A = dot(e, e) - r^2 + 2 * r - 1
1032 // B = (r -1) / A
1033 // C = 1 / A
1034 // d = dot(e, p) + B
1035 // t = d +/- sqrt(d^2 - A * dot(p, p) + C)
joshualitt30ba4362014-08-21 20:18:45 -07001036 fsBuilder->codeAppendf("\tfloat pDotp = dot(%s, %s);\n", coords2D, coords2D);
joshualitt49586be2014-09-16 08:21:41 -07001037 fsBuilder->codeAppendf("\tfloat d = dot(%s, %s) + %s.y;\n", coords2D, center.c_str(),
1038 params.c_str());
joshualitt30ba4362014-08-21 20:18:45 -07001039 fsBuilder->codeAppendf("\tfloat %s = d + sqrt(d * d - %s.x * pDotp + %s.z);\n",
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001040 tName.c_str(), params.c_str(), params.c_str());
1041
joshualitt60030bc2014-11-25 14:21:55 -08001042 this->emitColor(builder, ge, tName.c_str(), outputColor, inputColor, samplers);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001043}
1044
kkinnunen7510b222014-07-30 00:04:16 -07001045void GLCircleInside2PtConicalEffect::setData(const GrGLProgramDataManager& pdman,
joshualittb0a8a372014-09-23 09:50:21 -07001046 const GrProcessor& processor) {
1047 INHERITED::setData(pdman, processor);
1048 const CircleInside2PtConicalEffect& data = processor.cast<CircleInside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001049 SkScalar centerX = data.centerX();
1050 SkScalar centerY = data.centerY();
1051 SkScalar A = data.A();
1052 SkScalar B = data.B();
1053 SkScalar C = data.C();
1054
1055 if (fCachedCenterX != centerX || fCachedCenterY != centerY ||
1056 fCachedA != A || fCachedB != B || fCachedC != C) {
1057
kkinnunen7510b222014-07-30 00:04:16 -07001058 pdman.set2f(fCenterUni, SkScalarToFloat(centerX), SkScalarToFloat(centerY));
1059 pdman.set3f(fParamUni, SkScalarToFloat(A), SkScalarToFloat(B), SkScalarToFloat(C));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001060
1061 fCachedCenterX = centerX;
1062 fCachedCenterY = centerY;
1063 fCachedA = A;
1064 fCachedB = B;
1065 fCachedC = C;
1066 }
1067}
1068
joshualittb0a8a372014-09-23 09:50:21 -07001069void GLCircleInside2PtConicalEffect::GenKey(const GrProcessor& processor,
jvanverthcfc18862015-04-28 08:48:20 -07001070 const GrGLSLCaps&, GrProcessorKeyBuilder* b) {
joshualittb0a8a372014-09-23 09:50:21 -07001071 b->add32(GenBaseGradientKey(processor));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001072}
1073
1074//////////////////////////////////////////////////////////////////////////////
1075
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001076class CircleOutside2PtConicalEffect : public GrGradientEffect {
1077public:
1078
joshualittb0a8a372014-09-23 09:50:21 -07001079 static GrFragmentProcessor* Create(GrContext* ctx,
1080 const SkTwoPointConicalGradient& shader,
1081 const SkMatrix& matrix,
1082 SkShader::TileMode tm,
1083 const CircleConicalInfo& info) {
bsalomon55fad7a2014-07-08 07:34:20 -07001084 return SkNEW_ARGS(CircleOutside2PtConicalEffect, (ctx, shader, matrix, tm, info));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001085 }
1086
1087 virtual ~CircleOutside2PtConicalEffect() {}
1088
mtklein36352bf2015-03-25 18:17:31 -07001089 const char* name() const override { return "Two-Point Conical Gradient Outside"; }
joshualitteb2a6762014-12-04 11:35:33 -08001090
jvanverthcfc18862015-04-28 08:48:20 -07001091 void getGLProcessorKey(const GrGLSLCaps&, GrProcessorKeyBuilder*) const override;
joshualitteb2a6762014-12-04 11:35:33 -08001092
mtklein36352bf2015-03-25 18:17:31 -07001093 GrGLFragmentProcessor* createGLInstance() const override;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001094
1095 SkScalar centerX() const { return fInfo.fCenterEnd.fX; }
1096 SkScalar centerY() const { return fInfo.fCenterEnd.fY; }
1097 SkScalar A() const { return fInfo.fA; }
1098 SkScalar B() const { return fInfo.fB; }
1099 SkScalar C() const { return fInfo.fC; }
1100 SkScalar tLimit() const { return fTLimit; }
1101 bool isFlipped() const { return fIsFlipped; }
1102
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001103private:
mtklein36352bf2015-03-25 18:17:31 -07001104 bool onIsEqual(const GrFragmentProcessor& sBase) const override {
joshualitt49586be2014-09-16 08:21:41 -07001105 const CircleOutside2PtConicalEffect& s = sBase.cast<CircleOutside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001106 return (INHERITED::onIsEqual(sBase) &&
1107 this->fInfo.fCenterEnd == s.fInfo.fCenterEnd &&
1108 this->fInfo.fA == s.fInfo.fA &&
1109 this->fInfo.fB == s.fInfo.fB &&
1110 this->fInfo.fC == s.fInfo.fC &&
1111 this->fTLimit == s.fTLimit &&
1112 this->fIsFlipped == s.fIsFlipped);
1113 }
1114
1115 CircleOutside2PtConicalEffect(GrContext* ctx,
1116 const SkTwoPointConicalGradient& shader,
1117 const SkMatrix& matrix,
1118 SkShader::TileMode tm,
1119 const CircleConicalInfo& info)
1120 : INHERITED(ctx, shader, matrix, tm), fInfo(info) {
joshualitteb2a6762014-12-04 11:35:33 -08001121 this->initClassID<CircleOutside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001122 if (shader.getStartRadius() != shader.getEndRadius()) {
reed80ea19c2015-05-12 10:37:34 -07001123 fTLimit = shader.getStartRadius() / (shader.getStartRadius() - shader.getEndRadius());
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001124 } else {
1125 fTLimit = SK_ScalarMin;
1126 }
1127
1128 fIsFlipped = shader.isFlippedGrad();
1129 }
1130
joshualittb0a8a372014-09-23 09:50:21 -07001131 GR_DECLARE_FRAGMENT_PROCESSOR_TEST;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001132
1133 const CircleConicalInfo fInfo;
1134 SkScalar fTLimit;
1135 bool fIsFlipped;
1136
1137 typedef GrGradientEffect INHERITED;
1138};
1139
1140class GLCircleOutside2PtConicalEffect : public GrGLGradientEffect {
1141public:
joshualitteb2a6762014-12-04 11:35:33 -08001142 GLCircleOutside2PtConicalEffect(const GrProcessor&);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001143 virtual ~GLCircleOutside2PtConicalEffect() {}
1144
joshualitt15988992014-10-09 15:04:05 -07001145 virtual void emitCode(GrGLFPBuilder*,
joshualittb0a8a372014-09-23 09:50:21 -07001146 const GrFragmentProcessor&,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001147 const char* outputColor,
1148 const char* inputColor,
1149 const TransformedCoordsArray&,
mtklein36352bf2015-03-25 18:17:31 -07001150 const TextureSamplerArray&) override;
1151 void setData(const GrGLProgramDataManager&, const GrProcessor&) override;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001152
jvanverthcfc18862015-04-28 08:48:20 -07001153 static void GenKey(const GrProcessor&, const GrGLSLCaps& caps, GrProcessorKeyBuilder* b);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001154
1155protected:
1156 UniformHandle fCenterUni;
1157 UniformHandle fParamUni;
1158
1159 const char* fVSVaryingName;
1160 const char* fFSVaryingName;
1161
1162 bool fIsFlipped;
1163
1164 // @{
1165 /// Values last uploaded as uniforms
1166
1167 SkScalar fCachedCenterX;
1168 SkScalar fCachedCenterY;
1169 SkScalar fCachedA;
1170 SkScalar fCachedB;
1171 SkScalar fCachedC;
1172 SkScalar fCachedTLimit;
1173
1174 // @}
1175
1176private:
1177 typedef GrGLGradientEffect INHERITED;
1178
1179};
1180
jvanverthcfc18862015-04-28 08:48:20 -07001181void CircleOutside2PtConicalEffect::getGLProcessorKey(const GrGLSLCaps& caps,
joshualitteb2a6762014-12-04 11:35:33 -08001182 GrProcessorKeyBuilder* b) const {
1183 GLCircleOutside2PtConicalEffect::GenKey(*this, caps, b);
1184}
1185
1186GrGLFragmentProcessor* CircleOutside2PtConicalEffect::createGLInstance() const {
1187 return SkNEW_ARGS(GLCircleOutside2PtConicalEffect, (*this));
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001188}
1189
joshualittb0a8a372014-09-23 09:50:21 -07001190GR_DEFINE_FRAGMENT_PROCESSOR_TEST(CircleOutside2PtConicalEffect);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001191
joshualitt01258472014-09-22 10:29:30 -07001192/*
1193 * All Two point conical gradient test create functions may occasionally create edge case shaders
1194 */
joshualittb0a8a372014-09-23 09:50:21 -07001195GrFragmentProcessor* CircleOutside2PtConicalEffect::TestCreate(SkRandom* random,
1196 GrContext* context,
bsalomon4b91f762015-05-19 09:29:46 -07001197 const GrCaps&,
joshualittb0a8a372014-09-23 09:50:21 -07001198 GrTexture**) {
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001199 SkPoint center1 = {random->nextUScalar1(), random->nextUScalar1()};
commit-bot@chromium.org80894672014-04-22 21:24:22 +00001200 SkScalar radius1 = random->nextUScalar1() + 0.0001f; // make sure radius1 != 0
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001201 SkPoint center2;
1202 SkScalar radius2;
1203 SkScalar diffLen;
1204 do {
1205 center2.set(random->nextUScalar1(), random->nextUScalar1());
1206 // If the circles share a center than we can't be in the outside case
1207 } while (center1 == center2);
joshualitt01258472014-09-22 10:29:30 -07001208 SkPoint diff = center2 - center1;
1209 diffLen = diff.length();
1210 // Below makes sure that circle one is not contained within circle two
1211 // and have radius2 >= radius to match sorting on cpu side
1212 radius2 = radius1 + random->nextRangeF(0.f, diffLen);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001213
1214 SkColor colors[kMaxRandomGradientColors];
1215 SkScalar stopsArray[kMaxRandomGradientColors];
1216 SkScalar* stops = stopsArray;
1217 SkShader::TileMode tm;
1218 int colorCount = RandomGradientParams(random, colors, &stops, &tm);
1219 SkAutoTUnref<SkShader> shader(SkGradientShader::CreateTwoPointConical(center1, radius1,
1220 center2, radius2,
1221 colors, stops, colorCount,
1222 tm));
1223 SkPaint paint;
bsalomon83d081a2014-07-08 09:56:10 -07001224 GrColor paintColor;
joshualitt8ca93e72015-07-08 06:51:43 -07001225 GrFragmentProcessor* fp;
1226 GrPaint grPaint;
joshualitt5531d512014-12-17 15:50:11 -08001227 SkAssertResult(shader->asFragmentProcessor(context, paint,
joshualitt4eaf9ce2015-04-28 13:31:18 -07001228 GrTest::TestMatrix(random), NULL,
joshualitt8ca93e72015-07-08 06:51:43 -07001229 &paintColor, grPaint.getShaderDataManager(), &fp));
1230 return fp;
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001231}
1232
joshualitteb2a6762014-12-04 11:35:33 -08001233GLCircleOutside2PtConicalEffect::GLCircleOutside2PtConicalEffect(const GrProcessor& processor)
1234 : fVSVaryingName(NULL)
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001235 , fFSVaryingName(NULL)
1236 , fCachedCenterX(SK_ScalarMax)
1237 , fCachedCenterY(SK_ScalarMax)
1238 , fCachedA(SK_ScalarMax)
1239 , fCachedB(SK_ScalarMax)
1240 , fCachedC(SK_ScalarMax)
1241 , fCachedTLimit(SK_ScalarMax) {
joshualittb0a8a372014-09-23 09:50:21 -07001242 const CircleOutside2PtConicalEffect& data = processor.cast<CircleOutside2PtConicalEffect>();
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001243 fIsFlipped = data.isFlipped();
1244 }
1245
joshualitt15988992014-10-09 15:04:05 -07001246void GLCircleOutside2PtConicalEffect::emitCode(GrGLFPBuilder* builder,
joshualitt60030bc2014-11-25 14:21:55 -08001247 const GrFragmentProcessor& fp,
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001248 const char* outputColor,
1249 const char* inputColor,
1250 const TransformedCoordsArray& coords,
1251 const TextureSamplerArray& samplers) {
joshualitteb2a6762014-12-04 11:35:33 -08001252 const CircleOutside2PtConicalEffect& ge = fp.cast<CircleOutside2PtConicalEffect>();
joshualitt60030bc2014-11-25 14:21:55 -08001253 this->emitUniforms(builder, ge);
joshualitt30ba4362014-08-21 20:18:45 -07001254 fCenterUni = builder->addUniform(GrGLProgramBuilder::kFragment_Visibility,
bsalomon422f56f2014-12-09 10:18:12 -08001255 kVec2f_GrSLType, kDefault_GrSLPrecision,
1256 "Conical2FSCenter");
joshualitt30ba4362014-08-21 20:18:45 -07001257 fParamUni = builder->addUniform(GrGLProgramBuilder::kFragment_Visibility,
bsalomon422f56f2014-12-09 10:18:12 -08001258 kVec4f_GrSLType, kDefault_GrSLPrecision,
1259 "Conical2FSParams");
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001260 SkString tName("t");
1261
1262 GrGLShaderVar center = builder->getUniformVariable(fCenterUni);
1263 // params.x = A
1264 // params.y = B
1265 // params.z = C
1266 GrGLShaderVar params = builder->getUniformVariable(fParamUni);
1267
1268 // if we have a vec3 from being in perspective, convert it to a vec2 first
egdaniel29bee0f2015-04-29 11:54:42 -07001269 GrGLFragmentBuilder* fsBuilder = builder->getFragmentShaderBuilder();
joshualitt30ba4362014-08-21 20:18:45 -07001270 SkString coords2DString = fsBuilder->ensureFSCoords2D(coords, 0);
skia.committer@gmail.comede0c5c2014-04-23 03:04:11 +00001271 const char* coords2D = coords2DString.c_str();
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001272
1273 // output will default to transparent black (we simply won't write anything
1274 // else to it if invalid, instead of discarding or returning prematurely)
joshualitt30ba4362014-08-21 20:18:45 -07001275 fsBuilder->codeAppendf("\t%s = vec4(0.0,0.0,0.0,0.0);\n", outputColor);
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001276
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001277 // p = coords2D
1278 // e = center end
1279 // r = radius end
1280 // A = dot(e, e) - r^2 + 2 * r - 1
1281 // B = (r -1) / A
1282 // C = 1 / A
1283 // d = dot(e, p) + B
1284 // t = d +/- sqrt(d^2 - A * dot(p, p) + C)
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001285
joshualitt30ba4362014-08-21 20:18:45 -07001286 fsBuilder->codeAppendf("\tfloat pDotp = dot(%s, %s);\n", coords2D, coords2D);
joshualitt49586be2014-09-16 08:21:41 -07001287 fsBuilder->codeAppendf("\tfloat d = dot(%s, %s) + %s.y;\n", coords2D, center.c_str(),
1288 params.c_str());
1289 fsBuilder->codeAppendf("\tfloat deter = d * d - %s.x * pDotp + %s.z;\n", params.c_str(),
1290 params.c_str());
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001291
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001292 // Must check to see if we flipped the circle order (to make sure start radius < end radius)
1293 // If so we must also flip sign on sqrt
1294 if (!fIsFlipped) {
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 } else {
joshualitt30ba4362014-08-21 20:18:45 -07001297 fsBuilder->codeAppendf("\tfloat %s = d - sqrt(deter);\n", tName.c_str());
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001298 }
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001299
joshualitt30ba4362014-08-21 20:18:45 -07001300 fsBuilder->codeAppendf("\tif (%s >= %s.w && deter >= 0.0) {\n", tName.c_str(), params.c_str());
1301 fsBuilder->codeAppend("\t\t");
joshualitt60030bc2014-11-25 14:21:55 -08001302 this->emitColor(builder, ge, tName.c_str(), outputColor, inputColor, samplers);
joshualitt30ba4362014-08-21 20:18:45 -07001303 fsBuilder->codeAppend("\t}\n");
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001304}
1305
kkinnunen7510b222014-07-30 00:04:16 -07001306void GLCircleOutside2PtConicalEffect::setData(const GrGLProgramDataManager& pdman,
joshualittb0a8a372014-09-23 09:50:21 -07001307 const GrProcessor& processor) {
1308 INHERITED::setData(pdman, processor);
1309 const CircleOutside2PtConicalEffect& data = processor.cast<CircleOutside2PtConicalEffect>();
commit-bot@chromium.org44d83c12014-04-21 13:10:25 +00001310 SkASSERT(data.isFlipped() == fIsFlipped);
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001311 SkScalar centerX = data.centerX();
1312 SkScalar centerY = data.centerY();
1313 SkScalar A = data.A();
1314 SkScalar B = data.B();
1315 SkScalar C = data.C();
1316 SkScalar tLimit = data.tLimit();
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001317
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001318 if (fCachedCenterX != centerX || fCachedCenterY != centerY ||
1319 fCachedA != A || fCachedB != B || fCachedC != C || fCachedTLimit != tLimit) {
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001320
kkinnunen7510b222014-07-30 00:04:16 -07001321 pdman.set2f(fCenterUni, SkScalarToFloat(centerX), SkScalarToFloat(centerY));
1322 pdman.set4f(fParamUni, SkScalarToFloat(A), SkScalarToFloat(B), SkScalarToFloat(C),
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001323 SkScalarToFloat(tLimit));
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001324
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001325 fCachedCenterX = centerX;
1326 fCachedCenterY = centerY;
1327 fCachedA = A;
1328 fCachedB = B;
1329 fCachedC = C;
1330 fCachedTLimit = tLimit;
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001331 }
1332}
1333
joshualittb0a8a372014-09-23 09:50:21 -07001334void GLCircleOutside2PtConicalEffect::GenKey(const GrProcessor& processor,
jvanverthcfc18862015-04-28 08:48:20 -07001335 const GrGLSLCaps&, GrProcessorKeyBuilder* b) {
bsalomon63e99f72014-07-21 08:03:14 -07001336 uint32_t* key = b->add32n(2);
joshualittb0a8a372014-09-23 09:50:21 -07001337 key[0] = GenBaseGradientKey(processor);
1338 key[1] = processor.cast<CircleOutside2PtConicalEffect>().isFlipped();
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001339}
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +00001340
1341//////////////////////////////////////////////////////////////////////////////
1342
joshualittb0a8a372014-09-23 09:50:21 -07001343GrFragmentProcessor* Gr2PtConicalGradientEffect::Create(GrContext* ctx,
1344 const SkTwoPointConicalGradient& shader,
1345 SkShader::TileMode tm,
1346 const SkMatrix* localMatrix) {
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +00001347 SkMatrix matrix;
1348 if (!shader.getLocalMatrix().invert(&matrix)) {
1349 return NULL;
1350 }
commit-bot@chromium.org96fb7482014-05-09 20:28:11 +00001351 if (localMatrix) {
1352 SkMatrix inv;
1353 if (!localMatrix->invert(&inv)) {
1354 return NULL;
1355 }
1356 matrix.postConcat(inv);
1357 }
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +00001358
commit-bot@chromium.orgc8379d72014-04-22 20:56:37 +00001359 if (shader.getStartRadius() < kErrorTol) {
1360 SkScalar focalX;
1361 ConicalType type = set_matrix_focal_conical(shader, &matrix, &focalX);
1362 if (type == kInside_ConicalType) {
1363 return FocalInside2PtConicalEffect::Create(ctx, shader, matrix, tm, focalX);
1364 } else if(type == kEdge_ConicalType) {
1365 set_matrix_edge_conical(shader, &matrix);
1366 return Edge2PtConicalEffect::Create(ctx, shader, matrix, tm);
1367 } else {
1368 return FocalOutside2PtConicalEffect::Create(ctx, shader, matrix, tm, focalX);
1369 }
1370 }
1371
1372 CircleConicalInfo info;
1373 ConicalType type = set_matrix_circle_conical(shader, &matrix, &info);
1374
1375 if (type == kInside_ConicalType) {
1376 return CircleInside2PtConicalEffect::Create(ctx, shader, matrix, tm, info);
1377 } else if (type == kEdge_ConicalType) {
1378 set_matrix_edge_conical(shader, &matrix);
1379 return Edge2PtConicalEffect::Create(ctx, shader, matrix, tm);
1380 } else {
1381 return CircleOutside2PtConicalEffect::Create(ctx, shader, matrix, tm, info);
1382 }
commit-bot@chromium.org2af1a2d2014-04-04 13:50:50 +00001383}
1384
commit-bot@chromium.orgaa64fbf2014-04-03 14:59:19 +00001385#endif