blob: 26b96ad8ecf3ac8d42b7d54c97ec02183643f0a0 [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"
egdaniel605dd0f2014-11-12 08:35:25 -080010#include "GrInvariantOutput.h"
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
bsalomon0e08fc12014-10-15 08:19:04 -070040 virtual bool onIsEqual(const GrFragmentProcessor&) const SK_OVERRIDE;
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000041
egdaniel605dd0f2014-11-12 08:35:25 -080042 virtual void onComputeInvariantOutput(GrInvariantOutput* inout) const SK_OVERRIDE;
egdaniel1a8ecdf2014-10-03 06:24:12 -070043
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
egdaniel605dd0f2014-11-12 08:35:25 -080059void CircleEffect::onComputeInvariantOutput(GrInvariantOutput* inout) const {
egdanielccb2e382014-10-13 12:53:46 -070060 inout->mulByUnknownAlpha();
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000061}
62
joshualittb0a8a372014-09-23 09:50:21 -070063const GrBackendFragmentProcessorFactory& CircleEffect::getFactory() const {
64 return GrTBackendFragmentProcessorFactory<CircleEffect>::getInstance();
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000065}
66
joshualittb0a8a372014-09-23 09:50:21 -070067CircleEffect::CircleEffect(GrPrimitiveEdgeType edgeType, const SkPoint& c, SkScalar r)
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000068 : fCenter(c)
69 , fRadius(r)
70 , fEdgeType(edgeType) {
71 this->setWillReadFragmentPosition();
72}
73
bsalomon0e08fc12014-10-15 08:19:04 -070074bool CircleEffect::onIsEqual(const GrFragmentProcessor& other) const {
joshualitt49586be2014-09-16 08:21:41 -070075 const CircleEffect& ce = other.cast<CircleEffect>();
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +000076 return fEdgeType == ce.fEdgeType && fCenter == ce.fCenter && fRadius == ce.fRadius;
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000077}
78
79//////////////////////////////////////////////////////////////////////////////
80
joshualittb0a8a372014-09-23 09:50:21 -070081GR_DEFINE_FRAGMENT_PROCESSOR_TEST(CircleEffect);
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000082
joshualittb0a8a372014-09-23 09:50:21 -070083GrFragmentProcessor* CircleEffect::TestCreate(SkRandom* random,
84 GrContext*,
85 const GrDrawTargetCaps& caps,
86 GrTexture*[]) {
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000087 SkPoint center;
88 center.fX = random->nextRangeScalar(0.f, 1000.f);
89 center.fY = random->nextRangeScalar(0.f, 1000.f);
90 SkScalar radius = random->nextRangeF(0.f, 1000.f);
joshualittb0a8a372014-09-23 09:50:21 -070091 GrPrimitiveEdgeType et;
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000092 do {
joshualittb0a8a372014-09-23 09:50:21 -070093 et = (GrPrimitiveEdgeType)random->nextULessThan(kGrProcessorEdgeTypeCnt);
94 } while (kHairlineAA_GrProcessorEdgeType == et);
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000095 return CircleEffect::Create(et, center, radius);
96}
97
98//////////////////////////////////////////////////////////////////////////////
99
joshualittb0a8a372014-09-23 09:50:21 -0700100class GLCircleEffect : public GrGLFragmentProcessor {
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000101public:
joshualittb0a8a372014-09-23 09:50:21 -0700102 GLCircleEffect(const GrBackendProcessorFactory&, const GrProcessor&);
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000103
joshualitt15988992014-10-09 15:04:05 -0700104 virtual void emitCode(GrGLFPBuilder* builder,
joshualittb0a8a372014-09-23 09:50:21 -0700105 const GrFragmentProcessor& fp,
106 const GrProcessorKey& key,
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000107 const char* outputColor,
108 const char* inputColor,
109 const TransformedCoordsArray&,
110 const TextureSamplerArray&) SK_OVERRIDE;
111
joshualittb0a8a372014-09-23 09:50:21 -0700112 static inline void GenKey(const GrProcessor&, const GrGLCaps&, GrProcessorKeyBuilder*);
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000113
joshualittb0a8a372014-09-23 09:50:21 -0700114 virtual void setData(const GrGLProgramDataManager&, const GrProcessor&) SK_OVERRIDE;
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000115
116private:
kkinnunen7510b222014-07-30 00:04:16 -0700117 GrGLProgramDataManager::UniformHandle fCircleUniform;
118 SkPoint fPrevCenter;
119 SkScalar fPrevRadius;
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000120
joshualittb0a8a372014-09-23 09:50:21 -0700121 typedef GrGLFragmentProcessor INHERITED;
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000122};
123
joshualittb0a8a372014-09-23 09:50:21 -0700124GLCircleEffect::GLCircleEffect(const GrBackendProcessorFactory& factory,
125 const GrProcessor&)
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000126 : INHERITED (factory) {
127 fPrevRadius = -1.f;
128}
129
joshualitt15988992014-10-09 15:04:05 -0700130void GLCircleEffect::emitCode(GrGLFPBuilder* builder,
joshualittb0a8a372014-09-23 09:50:21 -0700131 const GrFragmentProcessor& fp,
132 const GrProcessorKey& key,
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000133 const char* outputColor,
134 const char* inputColor,
135 const TransformedCoordsArray&,
136 const TextureSamplerArray& samplers) {
joshualittb0a8a372014-09-23 09:50:21 -0700137 const CircleEffect& ce = fp.cast<CircleEffect>();
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000138 const char *circleName;
commit-bot@chromium.org0a09d712014-04-09 21:26:11 +0000139 // The circle uniform is (center.x, center.y, radius + 0.5) for regular fills and
140 // (... ,radius - 0.5) for inverse fills.
joshualitt30ba4362014-08-21 20:18:45 -0700141 fCircleUniform = builder->addUniform(GrGLProgramBuilder::kFragment_Visibility,
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000142 kVec3f_GrSLType,
143 "circle",
144 &circleName);
joshualitt30ba4362014-08-21 20:18:45 -0700145
joshualitt15988992014-10-09 15:04:05 -0700146 GrGLFPFragmentBuilder* fsBuilder = builder->getFragmentShaderBuilder();
joshualitt30ba4362014-08-21 20:18:45 -0700147 const char* fragmentPos = fsBuilder->fragmentPosition();
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000148
joshualittb0a8a372014-09-23 09:50:21 -0700149 SkASSERT(kHairlineAA_GrProcessorEdgeType != ce.getEdgeType());
150 if (GrProcessorEdgeTypeIsInverseFill(ce.getEdgeType())) {
joshualitt30ba4362014-08-21 20:18:45 -0700151 fsBuilder->codeAppendf("\t\tfloat d = length(%s.xy - %s.xy) - %s.z;\n",
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000152 circleName, fragmentPos, circleName);
153 } else {
joshualitt30ba4362014-08-21 20:18:45 -0700154 fsBuilder->codeAppendf("\t\tfloat d = %s.z - length(%s.xy - %s.xy);\n",
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000155 circleName, fragmentPos, circleName);
156 }
joshualittb0a8a372014-09-23 09:50:21 -0700157 if (GrProcessorEdgeTypeIsAA(ce.getEdgeType())) {
joshualitt30ba4362014-08-21 20:18:45 -0700158 fsBuilder->codeAppend("\t\td = clamp(d, 0.0, 1.0);\n");
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000159 } else {
joshualitt30ba4362014-08-21 20:18:45 -0700160 fsBuilder->codeAppend("\t\td = d > 0.5 ? 1.0 : 0.0;\n");
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000161 }
162
joshualitt30ba4362014-08-21 20:18:45 -0700163 fsBuilder->codeAppendf("\t\t%s = %s;\n", outputColor,
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000164 (GrGLSLExpr4(inputColor) * GrGLSLExpr1("d")).c_str());
165}
166
joshualittb0a8a372014-09-23 09:50:21 -0700167void GLCircleEffect::GenKey(const GrProcessor& processor, const GrGLCaps&,
168 GrProcessorKeyBuilder* b) {
169 const CircleEffect& ce = processor.cast<CircleEffect>();
bsalomon63e99f72014-07-21 08:03:14 -0700170 b->add32(ce.getEdgeType());
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000171}
172
joshualittb0a8a372014-09-23 09:50:21 -0700173void GLCircleEffect::setData(const GrGLProgramDataManager& pdman, const GrProcessor& processor) {
174 const CircleEffect& ce = processor.cast<CircleEffect>();
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000175 if (ce.getRadius() != fPrevRadius || ce.getCenter() != fPrevCenter) {
commit-bot@chromium.org0a09d712014-04-09 21:26:11 +0000176 SkScalar radius = ce.getRadius();
joshualittb0a8a372014-09-23 09:50:21 -0700177 if (GrProcessorEdgeTypeIsInverseFill(ce.getEdgeType())) {
commit-bot@chromium.org0a09d712014-04-09 21:26:11 +0000178 radius -= 0.5f;
179 } else {
180 radius += 0.5f;
181 }
kkinnunen7510b222014-07-30 00:04:16 -0700182 pdman.set3f(fCircleUniform, ce.getCenter().fX, ce.getCenter().fY, radius);
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000183 fPrevCenter = ce.getCenter();
184 fPrevRadius = ce.getRadius();
185 }
186}
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000187
188//////////////////////////////////////////////////////////////////////////////
189
190class GLEllipseEffect;
191
joshualittb0a8a372014-09-23 09:50:21 -0700192class EllipseEffect : public GrFragmentProcessor {
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000193public:
joshualittb0a8a372014-09-23 09:50:21 -0700194 static GrFragmentProcessor* Create(GrPrimitiveEdgeType, const SkPoint& center, SkScalar rx,
195 SkScalar ry);
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000196
197 virtual ~EllipseEffect() {};
198 static const char* Name() { return "Ellipse"; }
199
200 const SkPoint& getCenter() const { return fCenter; }
201 SkVector getRadii() const { return fRadii; }
202
joshualittb0a8a372014-09-23 09:50:21 -0700203 GrPrimitiveEdgeType getEdgeType() const { return fEdgeType; }
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000204
joshualittb0a8a372014-09-23 09:50:21 -0700205 typedef GLEllipseEffect GLProcessor;
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000206
joshualittb0a8a372014-09-23 09:50:21 -0700207 virtual const GrBackendFragmentProcessorFactory& getFactory() const SK_OVERRIDE;
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000208
209private:
joshualittb0a8a372014-09-23 09:50:21 -0700210 EllipseEffect(GrPrimitiveEdgeType, const SkPoint& center, SkScalar rx, SkScalar ry);
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000211
bsalomon0e08fc12014-10-15 08:19:04 -0700212 virtual bool onIsEqual(const GrFragmentProcessor&) const SK_OVERRIDE;
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000213
egdaniel605dd0f2014-11-12 08:35:25 -0800214 virtual void onComputeInvariantOutput(GrInvariantOutput* inout) const SK_OVERRIDE;
egdaniel1a8ecdf2014-10-03 06:24:12 -0700215
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000216 SkPoint fCenter;
217 SkVector fRadii;
joshualittb0a8a372014-09-23 09:50:21 -0700218 GrPrimitiveEdgeType fEdgeType;
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000219
joshualittb0a8a372014-09-23 09:50:21 -0700220 GR_DECLARE_FRAGMENT_PROCESSOR_TEST;
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000221
joshualittb0a8a372014-09-23 09:50:21 -0700222 typedef GrFragmentProcessor INHERITED;
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000223};
224
joshualittb0a8a372014-09-23 09:50:21 -0700225GrFragmentProcessor* EllipseEffect::Create(GrPrimitiveEdgeType edgeType,
226 const SkPoint& center,
227 SkScalar rx,
228 SkScalar ry) {
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000229 SkASSERT(rx >= 0 && ry >= 0);
bsalomon55fad7a2014-07-08 07:34:20 -0700230 return SkNEW_ARGS(EllipseEffect, (edgeType, center, rx, ry));
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000231}
232
egdaniel605dd0f2014-11-12 08:35:25 -0800233void EllipseEffect::onComputeInvariantOutput(GrInvariantOutput* inout) const {
egdanielccb2e382014-10-13 12:53:46 -0700234 inout->mulByUnknownAlpha();
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000235}
236
joshualittb0a8a372014-09-23 09:50:21 -0700237const GrBackendFragmentProcessorFactory& EllipseEffect::getFactory() const {
238 return GrTBackendFragmentProcessorFactory<EllipseEffect>::getInstance();
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000239}
240
joshualittb0a8a372014-09-23 09:50:21 -0700241EllipseEffect::EllipseEffect(GrPrimitiveEdgeType edgeType, const SkPoint& c, SkScalar rx, SkScalar ry)
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000242 : fCenter(c)
243 , fRadii(SkVector::Make(rx, ry))
244 , fEdgeType(edgeType) {
245 this->setWillReadFragmentPosition();
246}
247
bsalomon0e08fc12014-10-15 08:19:04 -0700248bool EllipseEffect::onIsEqual(const GrFragmentProcessor& other) const {
joshualitt49586be2014-09-16 08:21:41 -0700249 const EllipseEffect& ee = other.cast<EllipseEffect>();
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000250 return fEdgeType == ee.fEdgeType && fCenter == ee.fCenter && fRadii == ee.fRadii;
251}
252
253//////////////////////////////////////////////////////////////////////////////
254
joshualittb0a8a372014-09-23 09:50:21 -0700255GR_DEFINE_FRAGMENT_PROCESSOR_TEST(EllipseEffect);
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000256
joshualittb0a8a372014-09-23 09:50:21 -0700257GrFragmentProcessor* EllipseEffect::TestCreate(SkRandom* random,
258 GrContext*,
259 const GrDrawTargetCaps& caps,
260 GrTexture*[]) {
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000261 SkPoint center;
262 center.fX = random->nextRangeScalar(0.f, 1000.f);
263 center.fY = random->nextRangeScalar(0.f, 1000.f);
264 SkScalar rx = random->nextRangeF(0.f, 1000.f);
265 SkScalar ry = random->nextRangeF(0.f, 1000.f);
joshualittb0a8a372014-09-23 09:50:21 -0700266 GrPrimitiveEdgeType et;
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000267 do {
joshualittb0a8a372014-09-23 09:50:21 -0700268 et = (GrPrimitiveEdgeType)random->nextULessThan(kGrProcessorEdgeTypeCnt);
269 } while (kHairlineAA_GrProcessorEdgeType == et);
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000270 return EllipseEffect::Create(et, center, rx, ry);
271}
272
273//////////////////////////////////////////////////////////////////////////////
274
joshualittb0a8a372014-09-23 09:50:21 -0700275class GLEllipseEffect : public GrGLFragmentProcessor {
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000276public:
joshualittb0a8a372014-09-23 09:50:21 -0700277 GLEllipseEffect(const GrBackendProcessorFactory&, const GrProcessor&);
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000278
joshualitt15988992014-10-09 15:04:05 -0700279 virtual void emitCode(GrGLFPBuilder* builder,
joshualittb0a8a372014-09-23 09:50:21 -0700280 const GrFragmentProcessor& fp,
281 const GrProcessorKey& key,
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000282 const char* outputColor,
283 const char* inputColor,
284 const TransformedCoordsArray&,
285 const TextureSamplerArray&) SK_OVERRIDE;
286
joshualittb0a8a372014-09-23 09:50:21 -0700287 static inline void GenKey(const GrProcessor&, const GrGLCaps&, GrProcessorKeyBuilder*);
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000288
joshualittb0a8a372014-09-23 09:50:21 -0700289 virtual void setData(const GrGLProgramDataManager&, const GrProcessor&) SK_OVERRIDE;
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000290
291private:
kkinnunen7510b222014-07-30 00:04:16 -0700292 GrGLProgramDataManager::UniformHandle fEllipseUniform;
293 SkPoint fPrevCenter;
294 SkVector fPrevRadii;
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000295
joshualittb0a8a372014-09-23 09:50:21 -0700296 typedef GrGLFragmentProcessor INHERITED;
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000297};
298
joshualittb0a8a372014-09-23 09:50:21 -0700299GLEllipseEffect::GLEllipseEffect(const GrBackendProcessorFactory& factory,
300 const GrProcessor& effect)
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000301 : INHERITED (factory) {
302 fPrevRadii.fX = -1.f;
303}
304
joshualitt15988992014-10-09 15:04:05 -0700305void GLEllipseEffect::emitCode(GrGLFPBuilder* builder,
joshualittb0a8a372014-09-23 09:50:21 -0700306 const GrFragmentProcessor& fp,
307 const GrProcessorKey& key,
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000308 const char* outputColor,
309 const char* inputColor,
310 const TransformedCoordsArray&,
311 const TextureSamplerArray& samplers) {
joshualittb0a8a372014-09-23 09:50:21 -0700312 const EllipseEffect& ee = fp.cast<EllipseEffect>();
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000313 const char *ellipseName;
314 // The ellipse uniform is (center.x, center.y, 1 / rx^2, 1 / ry^2)
joshualitt30ba4362014-08-21 20:18:45 -0700315 fEllipseUniform = builder->addUniform(GrGLProgramBuilder::kFragment_Visibility,
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000316 kVec4f_GrSLType,
317 "ellipse",
318 &ellipseName);
joshualitt30ba4362014-08-21 20:18:45 -0700319
joshualitt15988992014-10-09 15:04:05 -0700320 GrGLFPFragmentBuilder* fsBuilder = builder->getFragmentShaderBuilder();
joshualitt30ba4362014-08-21 20:18:45 -0700321 const char* fragmentPos = fsBuilder->fragmentPosition();
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000322
323 // d is the offset to the ellipse center
joshualitt30ba4362014-08-21 20:18:45 -0700324 fsBuilder->codeAppendf("\t\tvec2 d = %s.xy - %s.xy;\n", fragmentPos, ellipseName);
325 fsBuilder->codeAppendf("\t\tvec2 Z = d * %s.zw;\n", ellipseName);
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000326 // implicit is the evaluation of (x/rx)^2 + (y/ry)^2 - 1.
joshualitt30ba4362014-08-21 20:18:45 -0700327 fsBuilder->codeAppend("\t\tfloat implicit = dot(Z, d) - 1.0;\n");
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000328 // grad_dot is the squared length of the gradient of the implicit.
joshualitt30ba4362014-08-21 20:18:45 -0700329 fsBuilder->codeAppendf("\t\tfloat grad_dot = 4.0 * dot(Z, Z);\n");
commit-bot@chromium.org1b035d82014-04-09 17:11:09 +0000330 // avoid calling inversesqrt on zero.
joshualitt30ba4362014-08-21 20:18:45 -0700331 fsBuilder->codeAppend("\t\tgrad_dot = max(grad_dot, 1.0e-4);\n");
332 fsBuilder->codeAppendf("\t\tfloat approx_dist = implicit * inversesqrt(grad_dot);\n");
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000333
334 switch (ee.getEdgeType()) {
joshualittb0a8a372014-09-23 09:50:21 -0700335 case kFillAA_GrProcessorEdgeType:
joshualitt30ba4362014-08-21 20:18:45 -0700336 fsBuilder->codeAppend("\t\tfloat alpha = clamp(0.5 - approx_dist, 0.0, 1.0);\n");
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000337 break;
joshualittb0a8a372014-09-23 09:50:21 -0700338 case kInverseFillAA_GrProcessorEdgeType:
joshualitt30ba4362014-08-21 20:18:45 -0700339 fsBuilder->codeAppend("\t\tfloat alpha = clamp(0.5 + approx_dist, 0.0, 1.0);\n");
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000340 break;
joshualittb0a8a372014-09-23 09:50:21 -0700341 case kFillBW_GrProcessorEdgeType:
joshualitt30ba4362014-08-21 20:18:45 -0700342 fsBuilder->codeAppend("\t\tfloat alpha = approx_dist > 0.0 ? 0.0 : 1.0;\n");
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000343 break;
joshualittb0a8a372014-09-23 09:50:21 -0700344 case kInverseFillBW_GrProcessorEdgeType:
joshualitt30ba4362014-08-21 20:18:45 -0700345 fsBuilder->codeAppend("\t\tfloat alpha = approx_dist > 0.0 ? 1.0 : 0.0;\n");
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000346 break;
joshualittb0a8a372014-09-23 09:50:21 -0700347 case kHairlineAA_GrProcessorEdgeType:
commit-bot@chromium.org88cb22b2014-04-30 14:17:00 +0000348 SkFAIL("Hairline not expected here.");
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000349 }
350
joshualitt30ba4362014-08-21 20:18:45 -0700351 fsBuilder->codeAppendf("\t\t%s = %s;\n", outputColor,
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000352 (GrGLSLExpr4(inputColor) * GrGLSLExpr1("alpha")).c_str());
353}
354
joshualittb0a8a372014-09-23 09:50:21 -0700355void GLEllipseEffect::GenKey(const GrProcessor& effect, const GrGLCaps&,
356 GrProcessorKeyBuilder* b) {
joshualitt49586be2014-09-16 08:21:41 -0700357 const EllipseEffect& ee = effect.cast<EllipseEffect>();
bsalomon63e99f72014-07-21 08:03:14 -0700358 b->add32(ee.getEdgeType());
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000359}
360
joshualittb0a8a372014-09-23 09:50:21 -0700361void GLEllipseEffect::setData(const GrGLProgramDataManager& pdman, const GrProcessor& effect) {
joshualitt49586be2014-09-16 08:21:41 -0700362 const EllipseEffect& ee = effect.cast<EllipseEffect>();
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000363 if (ee.getRadii() != fPrevRadii || ee.getCenter() != fPrevCenter) {
364 SkScalar invRXSqd = 1.f / (ee.getRadii().fX * ee.getRadii().fX);
365 SkScalar invRYSqd = 1.f / (ee.getRadii().fY * ee.getRadii().fY);
kkinnunen7510b222014-07-30 00:04:16 -0700366 pdman.set4f(fEllipseUniform, ee.getCenter().fX, ee.getCenter().fY, invRXSqd, invRYSqd);
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000367 fPrevCenter = ee.getCenter();
368 fPrevRadii = ee.getRadii();
369 }
370}
371
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000372//////////////////////////////////////////////////////////////////////////////
373
joshualittb0a8a372014-09-23 09:50:21 -0700374GrFragmentProcessor* GrOvalEffect::Create(GrPrimitiveEdgeType edgeType, const SkRect& oval) {
375 if (kHairlineAA_GrProcessorEdgeType == edgeType) {
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000376 return NULL;
377 }
378 SkScalar w = oval.width();
379 SkScalar h = oval.height();
380 if (SkScalarNearlyEqual(w, h)) {
381 w /= 2;
382 return CircleEffect::Create(edgeType, SkPoint::Make(oval.fLeft + w, oval.fTop + w), w);
commit-bot@chromium.orgd0a50292014-04-02 15:00:39 +0000383 } else {
384 w /= 2;
385 h /= 2;
386 return EllipseEffect::Create(edgeType, SkPoint::Make(oval.fLeft + w, oval.fTop + h), w, h);
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000387 }
388
389 return NULL;
390}