blob: 2a04e16611feb6e39a1a896961ad7002500e3bc1 [file] [log] [blame]
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +00001/*
2 * Copyright 2014 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
joshualitt30ba4362014-08-21 20:18:45 -07008#include "gl/builders/GrGLProgramBuilder.h"
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +00009#include "GrOvalEffect.h"
10
joshualittb0a8a372014-09-23 09:50:21 -070011#include "gl/GrGLProcessor.h"
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000012#include "gl/GrGLSL.h"
joshualittb0a8a372014-09-23 09:50:21 -070013#include "GrTBackendProcessorFactory.h"
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000014
15#include "SkRect.h"
16
17//////////////////////////////////////////////////////////////////////////////
18
19class GLCircleEffect;
20
joshualittb0a8a372014-09-23 09:50:21 -070021class CircleEffect : public GrFragmentProcessor {
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000022public:
joshualittb0a8a372014-09-23 09:50:21 -070023 static GrFragmentProcessor* Create(GrPrimitiveEdgeType, const SkPoint& center, SkScalar radius);
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000024
25 virtual ~CircleEffect() {};
26 static const char* Name() { return "Circle"; }
27
28 const SkPoint& getCenter() const { return fCenter; }
29 SkScalar getRadius() const { return fRadius; }
30
joshualittb0a8a372014-09-23 09:50:21 -070031 GrPrimitiveEdgeType getEdgeType() const { return fEdgeType; }
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000032
joshualittb0a8a372014-09-23 09:50:21 -070033 typedef GLCircleEffect GLProcessor;
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000034
joshualittb0a8a372014-09-23 09:50:21 -070035 virtual const GrBackendFragmentProcessorFactory& getFactory() const SK_OVERRIDE;
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000036
37private:
joshualittb0a8a372014-09-23 09:50:21 -070038 CircleEffect(GrPrimitiveEdgeType, const SkPoint& center, SkScalar radius);
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000039
joshualittb0a8a372014-09-23 09:50:21 -070040 virtual bool onIsEqual(const GrProcessor&) const SK_OVERRIDE;
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000041
egdaniel1a8ecdf2014-10-03 06:24:12 -070042 virtual void onComputeInvariantOutput(InvariantOutput* inout) const SK_OVERRIDE;
43
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000044 SkPoint fCenter;
45 SkScalar fRadius;
joshualittb0a8a372014-09-23 09:50:21 -070046 GrPrimitiveEdgeType fEdgeType;
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000047
joshualittb0a8a372014-09-23 09:50:21 -070048 GR_DECLARE_FRAGMENT_PROCESSOR_TEST;
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000049
joshualittb0a8a372014-09-23 09:50:21 -070050 typedef GrFragmentProcessor INHERITED;
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000051};
52
joshualittb0a8a372014-09-23 09:50:21 -070053GrFragmentProcessor* CircleEffect::Create(GrPrimitiveEdgeType edgeType, const SkPoint& center,
54 SkScalar radius) {
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000055 SkASSERT(radius >= 0);
bsalomon55fad7a2014-07-08 07:34:20 -070056 return SkNEW_ARGS(CircleEffect, (edgeType, center, radius));
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000057}
58
egdaniel1a8ecdf2014-10-03 06:24:12 -070059void CircleEffect::onComputeInvariantOutput(InvariantOutput* inout) const {
60 inout->fValidFlags = 0;
61 inout->fIsSingleComponent = false;
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000062}
63
joshualittb0a8a372014-09-23 09:50:21 -070064const GrBackendFragmentProcessorFactory& CircleEffect::getFactory() const {
65 return GrTBackendFragmentProcessorFactory<CircleEffect>::getInstance();
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000066}
67
joshualittb0a8a372014-09-23 09:50:21 -070068CircleEffect::CircleEffect(GrPrimitiveEdgeType edgeType, const SkPoint& c, SkScalar r)
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000069 : fCenter(c)
70 , fRadius(r)
71 , fEdgeType(edgeType) {
72 this->setWillReadFragmentPosition();
73}
74
joshualittb0a8a372014-09-23 09:50:21 -070075bool CircleEffect::onIsEqual(const GrProcessor& other) const {
joshualitt49586be2014-09-16 08:21:41 -070076 const CircleEffect& ce = other.cast<CircleEffect>();
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +000077 return fEdgeType == ce.fEdgeType && fCenter == ce.fCenter && fRadius == ce.fRadius;
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000078}
79
80//////////////////////////////////////////////////////////////////////////////
81
joshualittb0a8a372014-09-23 09:50:21 -070082GR_DEFINE_FRAGMENT_PROCESSOR_TEST(CircleEffect);
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000083
joshualittb0a8a372014-09-23 09:50:21 -070084GrFragmentProcessor* CircleEffect::TestCreate(SkRandom* random,
85 GrContext*,
86 const GrDrawTargetCaps& caps,
87 GrTexture*[]) {
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000088 SkPoint center;
89 center.fX = random->nextRangeScalar(0.f, 1000.f);
90 center.fY = random->nextRangeScalar(0.f, 1000.f);
91 SkScalar radius = random->nextRangeF(0.f, 1000.f);
joshualittb0a8a372014-09-23 09:50:21 -070092 GrPrimitiveEdgeType et;
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000093 do {
joshualittb0a8a372014-09-23 09:50:21 -070094 et = (GrPrimitiveEdgeType)random->nextULessThan(kGrProcessorEdgeTypeCnt);
95 } while (kHairlineAA_GrProcessorEdgeType == et);
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000096 return CircleEffect::Create(et, center, radius);
97}
98
99//////////////////////////////////////////////////////////////////////////////
100
joshualittb0a8a372014-09-23 09:50:21 -0700101class GLCircleEffect : public GrGLFragmentProcessor {
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000102public:
joshualittb0a8a372014-09-23 09:50:21 -0700103 GLCircleEffect(const GrBackendProcessorFactory&, const GrProcessor&);
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000104
joshualitt30ba4362014-08-21 20:18:45 -0700105 virtual void emitCode(GrGLProgramBuilder* builder,
joshualittb0a8a372014-09-23 09:50:21 -0700106 const GrFragmentProcessor& fp,
107 const GrProcessorKey& key,
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000108 const char* outputColor,
109 const char* inputColor,
110 const TransformedCoordsArray&,
111 const TextureSamplerArray&) SK_OVERRIDE;
112
joshualittb0a8a372014-09-23 09:50:21 -0700113 static inline void GenKey(const GrProcessor&, const GrGLCaps&, GrProcessorKeyBuilder*);
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000114
joshualittb0a8a372014-09-23 09:50:21 -0700115 virtual void setData(const GrGLProgramDataManager&, const GrProcessor&) SK_OVERRIDE;
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000116
117private:
kkinnunen7510b222014-07-30 00:04:16 -0700118 GrGLProgramDataManager::UniformHandle fCircleUniform;
119 SkPoint fPrevCenter;
120 SkScalar fPrevRadius;
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000121
joshualittb0a8a372014-09-23 09:50:21 -0700122 typedef GrGLFragmentProcessor INHERITED;
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000123};
124
joshualittb0a8a372014-09-23 09:50:21 -0700125GLCircleEffect::GLCircleEffect(const GrBackendProcessorFactory& factory,
126 const GrProcessor&)
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000127 : INHERITED (factory) {
128 fPrevRadius = -1.f;
129}
130
joshualitt30ba4362014-08-21 20:18:45 -0700131void GLCircleEffect::emitCode(GrGLProgramBuilder* builder,
joshualittb0a8a372014-09-23 09:50:21 -0700132 const GrFragmentProcessor& fp,
133 const GrProcessorKey& key,
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000134 const char* outputColor,
135 const char* inputColor,
136 const TransformedCoordsArray&,
137 const TextureSamplerArray& samplers) {
joshualittb0a8a372014-09-23 09:50:21 -0700138 const CircleEffect& ce = fp.cast<CircleEffect>();
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000139 const char *circleName;
commit-bot@chromium.org0a09d712014-04-09 21:26:11 +0000140 // The circle uniform is (center.x, center.y, radius + 0.5) for regular fills and
141 // (... ,radius - 0.5) for inverse fills.
joshualitt30ba4362014-08-21 20:18:45 -0700142 fCircleUniform = builder->addUniform(GrGLProgramBuilder::kFragment_Visibility,
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000143 kVec3f_GrSLType,
144 "circle",
145 &circleName);
joshualitt30ba4362014-08-21 20:18:45 -0700146
147 GrGLFragmentShaderBuilder* fsBuilder = builder->getFragmentShaderBuilder();
148 const char* fragmentPos = fsBuilder->fragmentPosition();
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000149
joshualittb0a8a372014-09-23 09:50:21 -0700150 SkASSERT(kHairlineAA_GrProcessorEdgeType != ce.getEdgeType());
151 if (GrProcessorEdgeTypeIsInverseFill(ce.getEdgeType())) {
joshualitt30ba4362014-08-21 20:18:45 -0700152 fsBuilder->codeAppendf("\t\tfloat d = length(%s.xy - %s.xy) - %s.z;\n",
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000153 circleName, fragmentPos, circleName);
154 } else {
joshualitt30ba4362014-08-21 20:18:45 -0700155 fsBuilder->codeAppendf("\t\tfloat d = %s.z - length(%s.xy - %s.xy);\n",
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000156 circleName, fragmentPos, circleName);
157 }
joshualittb0a8a372014-09-23 09:50:21 -0700158 if (GrProcessorEdgeTypeIsAA(ce.getEdgeType())) {
joshualitt30ba4362014-08-21 20:18:45 -0700159 fsBuilder->codeAppend("\t\td = clamp(d, 0.0, 1.0);\n");
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000160 } else {
joshualitt30ba4362014-08-21 20:18:45 -0700161 fsBuilder->codeAppend("\t\td = d > 0.5 ? 1.0 : 0.0;\n");
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000162 }
163
joshualitt30ba4362014-08-21 20:18:45 -0700164 fsBuilder->codeAppendf("\t\t%s = %s;\n", outputColor,
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000165 (GrGLSLExpr4(inputColor) * GrGLSLExpr1("d")).c_str());
166}
167
joshualittb0a8a372014-09-23 09:50:21 -0700168void GLCircleEffect::GenKey(const GrProcessor& processor, const GrGLCaps&,
169 GrProcessorKeyBuilder* b) {
170 const CircleEffect& ce = processor.cast<CircleEffect>();
bsalomon63e99f72014-07-21 08:03:14 -0700171 b->add32(ce.getEdgeType());
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000172}
173
joshualittb0a8a372014-09-23 09:50:21 -0700174void GLCircleEffect::setData(const GrGLProgramDataManager& pdman, const GrProcessor& processor) {
175 const CircleEffect& ce = processor.cast<CircleEffect>();
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000176 if (ce.getRadius() != fPrevRadius || ce.getCenter() != fPrevCenter) {
commit-bot@chromium.org0a09d712014-04-09 21:26:11 +0000177 SkScalar radius = ce.getRadius();
joshualittb0a8a372014-09-23 09:50:21 -0700178 if (GrProcessorEdgeTypeIsInverseFill(ce.getEdgeType())) {
commit-bot@chromium.org0a09d712014-04-09 21:26:11 +0000179 radius -= 0.5f;
180 } else {
181 radius += 0.5f;
182 }
kkinnunen7510b222014-07-30 00:04:16 -0700183 pdman.set3f(fCircleUniform, ce.getCenter().fX, ce.getCenter().fY, radius);
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000184 fPrevCenter = ce.getCenter();
185 fPrevRadius = ce.getRadius();
186 }
187}
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000188
189//////////////////////////////////////////////////////////////////////////////
190
191class GLEllipseEffect;
192
joshualittb0a8a372014-09-23 09:50:21 -0700193class EllipseEffect : public GrFragmentProcessor {
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000194public:
joshualittb0a8a372014-09-23 09:50:21 -0700195 static GrFragmentProcessor* Create(GrPrimitiveEdgeType, const SkPoint& center, SkScalar rx,
196 SkScalar ry);
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000197
198 virtual ~EllipseEffect() {};
199 static const char* Name() { return "Ellipse"; }
200
201 const SkPoint& getCenter() const { return fCenter; }
202 SkVector getRadii() const { return fRadii; }
203
joshualittb0a8a372014-09-23 09:50:21 -0700204 GrPrimitiveEdgeType getEdgeType() const { return fEdgeType; }
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000205
joshualittb0a8a372014-09-23 09:50:21 -0700206 typedef GLEllipseEffect GLProcessor;
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000207
joshualittb0a8a372014-09-23 09:50:21 -0700208 virtual const GrBackendFragmentProcessorFactory& getFactory() const SK_OVERRIDE;
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000209
210private:
joshualittb0a8a372014-09-23 09:50:21 -0700211 EllipseEffect(GrPrimitiveEdgeType, const SkPoint& center, SkScalar rx, SkScalar ry);
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000212
joshualittb0a8a372014-09-23 09:50:21 -0700213 virtual bool onIsEqual(const GrProcessor&) const SK_OVERRIDE;
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000214
egdaniel1a8ecdf2014-10-03 06:24:12 -0700215 virtual void onComputeInvariantOutput(InvariantOutput* inout) const SK_OVERRIDE;
216
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000217 SkPoint fCenter;
218 SkVector fRadii;
joshualittb0a8a372014-09-23 09:50:21 -0700219 GrPrimitiveEdgeType fEdgeType;
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000220
joshualittb0a8a372014-09-23 09:50:21 -0700221 GR_DECLARE_FRAGMENT_PROCESSOR_TEST;
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000222
joshualittb0a8a372014-09-23 09:50:21 -0700223 typedef GrFragmentProcessor INHERITED;
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000224};
225
joshualittb0a8a372014-09-23 09:50:21 -0700226GrFragmentProcessor* EllipseEffect::Create(GrPrimitiveEdgeType edgeType,
227 const SkPoint& center,
228 SkScalar rx,
229 SkScalar ry) {
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000230 SkASSERT(rx >= 0 && ry >= 0);
bsalomon55fad7a2014-07-08 07:34:20 -0700231 return SkNEW_ARGS(EllipseEffect, (edgeType, center, rx, ry));
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000232}
233
egdaniel1a8ecdf2014-10-03 06:24:12 -0700234void EllipseEffect::onComputeInvariantOutput(InvariantOutput* inout) const {
235 inout->fValidFlags = 0;
236 inout->fIsSingleComponent = false;
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000237}
238
joshualittb0a8a372014-09-23 09:50:21 -0700239const GrBackendFragmentProcessorFactory& EllipseEffect::getFactory() const {
240 return GrTBackendFragmentProcessorFactory<EllipseEffect>::getInstance();
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000241}
242
joshualittb0a8a372014-09-23 09:50:21 -0700243EllipseEffect::EllipseEffect(GrPrimitiveEdgeType edgeType, const SkPoint& c, SkScalar rx, SkScalar ry)
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000244 : fCenter(c)
245 , fRadii(SkVector::Make(rx, ry))
246 , fEdgeType(edgeType) {
247 this->setWillReadFragmentPosition();
248}
249
joshualittb0a8a372014-09-23 09:50:21 -0700250bool EllipseEffect::onIsEqual(const GrProcessor& other) const {
joshualitt49586be2014-09-16 08:21:41 -0700251 const EllipseEffect& ee = other.cast<EllipseEffect>();
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000252 return fEdgeType == ee.fEdgeType && fCenter == ee.fCenter && fRadii == ee.fRadii;
253}
254
255//////////////////////////////////////////////////////////////////////////////
256
joshualittb0a8a372014-09-23 09:50:21 -0700257GR_DEFINE_FRAGMENT_PROCESSOR_TEST(EllipseEffect);
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000258
joshualittb0a8a372014-09-23 09:50:21 -0700259GrFragmentProcessor* EllipseEffect::TestCreate(SkRandom* random,
260 GrContext*,
261 const GrDrawTargetCaps& caps,
262 GrTexture*[]) {
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000263 SkPoint center;
264 center.fX = random->nextRangeScalar(0.f, 1000.f);
265 center.fY = random->nextRangeScalar(0.f, 1000.f);
266 SkScalar rx = random->nextRangeF(0.f, 1000.f);
267 SkScalar ry = random->nextRangeF(0.f, 1000.f);
joshualittb0a8a372014-09-23 09:50:21 -0700268 GrPrimitiveEdgeType et;
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000269 do {
joshualittb0a8a372014-09-23 09:50:21 -0700270 et = (GrPrimitiveEdgeType)random->nextULessThan(kGrProcessorEdgeTypeCnt);
271 } while (kHairlineAA_GrProcessorEdgeType == et);
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000272 return EllipseEffect::Create(et, center, rx, ry);
273}
274
275//////////////////////////////////////////////////////////////////////////////
276
joshualittb0a8a372014-09-23 09:50:21 -0700277class GLEllipseEffect : public GrGLFragmentProcessor {
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000278public:
joshualittb0a8a372014-09-23 09:50:21 -0700279 GLEllipseEffect(const GrBackendProcessorFactory&, const GrProcessor&);
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000280
joshualitt30ba4362014-08-21 20:18:45 -0700281 virtual void emitCode(GrGLProgramBuilder* builder,
joshualittb0a8a372014-09-23 09:50:21 -0700282 const GrFragmentProcessor& fp,
283 const GrProcessorKey& key,
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000284 const char* outputColor,
285 const char* inputColor,
286 const TransformedCoordsArray&,
287 const TextureSamplerArray&) SK_OVERRIDE;
288
joshualittb0a8a372014-09-23 09:50:21 -0700289 static inline void GenKey(const GrProcessor&, const GrGLCaps&, GrProcessorKeyBuilder*);
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000290
joshualittb0a8a372014-09-23 09:50:21 -0700291 virtual void setData(const GrGLProgramDataManager&, const GrProcessor&) SK_OVERRIDE;
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000292
293private:
kkinnunen7510b222014-07-30 00:04:16 -0700294 GrGLProgramDataManager::UniformHandle fEllipseUniform;
295 SkPoint fPrevCenter;
296 SkVector fPrevRadii;
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000297
joshualittb0a8a372014-09-23 09:50:21 -0700298 typedef GrGLFragmentProcessor INHERITED;
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000299};
300
joshualittb0a8a372014-09-23 09:50:21 -0700301GLEllipseEffect::GLEllipseEffect(const GrBackendProcessorFactory& factory,
302 const GrProcessor& effect)
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000303 : INHERITED (factory) {
304 fPrevRadii.fX = -1.f;
305}
306
joshualitt30ba4362014-08-21 20:18:45 -0700307void GLEllipseEffect::emitCode(GrGLProgramBuilder* builder,
joshualittb0a8a372014-09-23 09:50:21 -0700308 const GrFragmentProcessor& fp,
309 const GrProcessorKey& key,
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000310 const char* outputColor,
311 const char* inputColor,
312 const TransformedCoordsArray&,
313 const TextureSamplerArray& samplers) {
joshualittb0a8a372014-09-23 09:50:21 -0700314 const EllipseEffect& ee = fp.cast<EllipseEffect>();
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000315 const char *ellipseName;
316 // The ellipse uniform is (center.x, center.y, 1 / rx^2, 1 / ry^2)
joshualitt30ba4362014-08-21 20:18:45 -0700317 fEllipseUniform = builder->addUniform(GrGLProgramBuilder::kFragment_Visibility,
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000318 kVec4f_GrSLType,
319 "ellipse",
320 &ellipseName);
joshualitt30ba4362014-08-21 20:18:45 -0700321
322 GrGLFragmentShaderBuilder* fsBuilder = builder->getFragmentShaderBuilder();
323 const char* fragmentPos = fsBuilder->fragmentPosition();
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000324
325 // d is the offset to the ellipse center
joshualitt30ba4362014-08-21 20:18:45 -0700326 fsBuilder->codeAppendf("\t\tvec2 d = %s.xy - %s.xy;\n", fragmentPos, ellipseName);
327 fsBuilder->codeAppendf("\t\tvec2 Z = d * %s.zw;\n", ellipseName);
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000328 // implicit is the evaluation of (x/rx)^2 + (y/ry)^2 - 1.
joshualitt30ba4362014-08-21 20:18:45 -0700329 fsBuilder->codeAppend("\t\tfloat implicit = dot(Z, d) - 1.0;\n");
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000330 // grad_dot is the squared length of the gradient of the implicit.
joshualitt30ba4362014-08-21 20:18:45 -0700331 fsBuilder->codeAppendf("\t\tfloat grad_dot = 4.0 * dot(Z, Z);\n");
commit-bot@chromium.org1b035d82014-04-09 17:11:09 +0000332 // avoid calling inversesqrt on zero.
joshualitt30ba4362014-08-21 20:18:45 -0700333 fsBuilder->codeAppend("\t\tgrad_dot = max(grad_dot, 1.0e-4);\n");
334 fsBuilder->codeAppendf("\t\tfloat approx_dist = implicit * inversesqrt(grad_dot);\n");
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000335
336 switch (ee.getEdgeType()) {
joshualittb0a8a372014-09-23 09:50:21 -0700337 case kFillAA_GrProcessorEdgeType:
joshualitt30ba4362014-08-21 20:18:45 -0700338 fsBuilder->codeAppend("\t\tfloat alpha = clamp(0.5 - approx_dist, 0.0, 1.0);\n");
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000339 break;
joshualittb0a8a372014-09-23 09:50:21 -0700340 case kInverseFillAA_GrProcessorEdgeType:
joshualitt30ba4362014-08-21 20:18:45 -0700341 fsBuilder->codeAppend("\t\tfloat alpha = clamp(0.5 + approx_dist, 0.0, 1.0);\n");
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000342 break;
joshualittb0a8a372014-09-23 09:50:21 -0700343 case kFillBW_GrProcessorEdgeType:
joshualitt30ba4362014-08-21 20:18:45 -0700344 fsBuilder->codeAppend("\t\tfloat alpha = approx_dist > 0.0 ? 0.0 : 1.0;\n");
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000345 break;
joshualittb0a8a372014-09-23 09:50:21 -0700346 case kInverseFillBW_GrProcessorEdgeType:
joshualitt30ba4362014-08-21 20:18:45 -0700347 fsBuilder->codeAppend("\t\tfloat alpha = approx_dist > 0.0 ? 1.0 : 0.0;\n");
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000348 break;
joshualittb0a8a372014-09-23 09:50:21 -0700349 case kHairlineAA_GrProcessorEdgeType:
commit-bot@chromium.org88cb22b2014-04-30 14:17:00 +0000350 SkFAIL("Hairline not expected here.");
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000351 }
352
joshualitt30ba4362014-08-21 20:18:45 -0700353 fsBuilder->codeAppendf("\t\t%s = %s;\n", outputColor,
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000354 (GrGLSLExpr4(inputColor) * GrGLSLExpr1("alpha")).c_str());
355}
356
joshualittb0a8a372014-09-23 09:50:21 -0700357void GLEllipseEffect::GenKey(const GrProcessor& effect, const GrGLCaps&,
358 GrProcessorKeyBuilder* b) {
joshualitt49586be2014-09-16 08:21:41 -0700359 const EllipseEffect& ee = effect.cast<EllipseEffect>();
bsalomon63e99f72014-07-21 08:03:14 -0700360 b->add32(ee.getEdgeType());
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000361}
362
joshualittb0a8a372014-09-23 09:50:21 -0700363void GLEllipseEffect::setData(const GrGLProgramDataManager& pdman, const GrProcessor& effect) {
joshualitt49586be2014-09-16 08:21:41 -0700364 const EllipseEffect& ee = effect.cast<EllipseEffect>();
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000365 if (ee.getRadii() != fPrevRadii || ee.getCenter() != fPrevCenter) {
366 SkScalar invRXSqd = 1.f / (ee.getRadii().fX * ee.getRadii().fX);
367 SkScalar invRYSqd = 1.f / (ee.getRadii().fY * ee.getRadii().fY);
kkinnunen7510b222014-07-30 00:04:16 -0700368 pdman.set4f(fEllipseUniform, ee.getCenter().fX, ee.getCenter().fY, invRXSqd, invRYSqd);
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000369 fPrevCenter = ee.getCenter();
370 fPrevRadii = ee.getRadii();
371 }
372}
373
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000374//////////////////////////////////////////////////////////////////////////////
375
joshualittb0a8a372014-09-23 09:50:21 -0700376GrFragmentProcessor* GrOvalEffect::Create(GrPrimitiveEdgeType edgeType, const SkRect& oval) {
377 if (kHairlineAA_GrProcessorEdgeType == edgeType) {
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000378 return NULL;
379 }
380 SkScalar w = oval.width();
381 SkScalar h = oval.height();
382 if (SkScalarNearlyEqual(w, h)) {
383 w /= 2;
384 return CircleEffect::Create(edgeType, SkPoint::Make(oval.fLeft + w, oval.fTop + w), w);
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000385 } else {
386 w /= 2;
387 h /= 2;
388 return EllipseEffect::Create(edgeType, SkPoint::Make(oval.fLeft + w, oval.fTop + h), w, h);
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000389 }
390
391 return NULL;
392}