blob: 8dde914bb4042e3fb537c8086aef58c064cf064b [file] [log] [blame]
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +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
8#include "GrRRectEffect.h"
9
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +000010#include "GrConvexPolyEffect.h"
joshualitteb2a6762014-12-04 11:35:33 -080011#include "GrFragmentProcessor.h"
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000012#include "GrOvalEffect.h"
Brian Salomon94efbf52016-11-29 13:43:05 -050013#include "GrShaderCaps.h"
commit-bot@chromium.orge5280892014-02-21 17:52:29 +000014#include "SkRRect.h"
egdaniel7ea439b2015-12-03 09:20:44 -080015#include "SkTLazy.h"
egdaniel64c47282015-11-13 06:54:19 -080016#include "glsl/GrGLSLFragmentProcessor.h"
egdaniel2d721d32015-11-11 13:06:05 -080017#include "glsl/GrGLSLFragmentShaderBuilder.h"
egdaniel018fb622015-10-28 07:26:40 -070018#include "glsl/GrGLSLProgramDataManager.h"
egdaniel7ea439b2015-12-03 09:20:44 -080019#include "glsl/GrGLSLUniformHandler.h"
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +000020
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +000021// The effects defined here only handle rrect radii >= kRadiusMin.
22static const SkScalar kRadiusMin = SK_ScalarHalf;
23
24//////////////////////////////////////////////////////////////////////////////
25
joshualittb0a8a372014-09-23 09:50:21 -070026class CircularRRectEffect : public GrFragmentProcessor {
commit-bot@chromium.orge5280892014-02-21 17:52:29 +000027public:
skia.committer@gmail.comf1f66c02014-03-05 03:02:06 +000028
commit-bot@chromium.orgdd584222014-03-10 16:08:22 +000029 enum CornerFlags {
30 kTopLeft_CornerFlag = (1 << SkRRect::kUpperLeft_Corner),
31 kTopRight_CornerFlag = (1 << SkRRect::kUpperRight_Corner),
32 kBottomRight_CornerFlag = (1 << SkRRect::kLowerRight_Corner),
33 kBottomLeft_CornerFlag = (1 << SkRRect::kLowerLeft_Corner),
34
35 kLeft_CornerFlags = kTopLeft_CornerFlag | kBottomLeft_CornerFlag,
36 kTop_CornerFlags = kTopLeft_CornerFlag | kTopRight_CornerFlag,
37 kRight_CornerFlags = kTopRight_CornerFlag | kBottomRight_CornerFlag,
38 kBottom_CornerFlags = kBottomLeft_CornerFlag | kBottomRight_CornerFlag,
39
40 kAll_CornerFlags = kTopLeft_CornerFlag | kTopRight_CornerFlag |
41 kBottomLeft_CornerFlag | kBottomRight_CornerFlag,
42
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +000043 kNone_CornerFlags = 0
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +000044 };
45
commit-bot@chromium.orgdd584222014-03-10 16:08:22 +000046 // The flags are used to indicate which corners are circluar (unflagged corners are assumed to
47 // be square).
bungeman06ca8ec2016-06-09 08:01:03 -070048 static sk_sp<GrFragmentProcessor> Make(GrPrimitiveEdgeType, uint32_t circularCornerFlags,
49 const SkRRect&);
commit-bot@chromium.orge5280892014-02-21 17:52:29 +000050
Brian Salomond3b65972017-03-22 12:05:03 -040051 ~CircularRRectEffect() override {}
joshualitteb2a6762014-12-04 11:35:33 -080052
mtklein36352bf2015-03-25 18:17:31 -070053 const char* name() const override { return "CircularRRect"; }
joshualitteb2a6762014-12-04 11:35:33 -080054
Brian Salomon1a2a7ab2017-07-26 13:11:51 -040055 sk_sp<GrFragmentProcessor> clone() const override;
56
commit-bot@chromium.orge5280892014-02-21 17:52:29 +000057 const SkRRect& getRRect() const { return fRRect; }
58
commit-bot@chromium.orgdd584222014-03-10 16:08:22 +000059 uint32_t getCircularCornerFlags() const { return fCircularCornerFlags; }
skia.committer@gmail.com06acb582014-03-06 03:02:32 +000060
joshualittb0a8a372014-09-23 09:50:21 -070061 GrPrimitiveEdgeType getEdgeType() const { return fEdgeType; }
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +000062
commit-bot@chromium.orge5280892014-02-21 17:52:29 +000063private:
joshualittb0a8a372014-09-23 09:50:21 -070064 CircularRRectEffect(GrPrimitiveEdgeType, uint32_t circularCornerFlags, const SkRRect&);
commit-bot@chromium.orge5280892014-02-21 17:52:29 +000065
egdaniel57d3b032015-11-13 11:57:27 -080066 GrGLSLFragmentProcessor* onCreateGLSLInstance() const override;
wangyixb1daa862015-08-18 11:29:31 -070067
Brian Salomon94efbf52016-11-29 13:43:05 -050068 void onGetGLSLProcessorKey(const GrShaderCaps&, GrProcessorKeyBuilder*) const override;
wangyix4b3050b2015-08-04 07:59:37 -070069
mtklein36352bf2015-03-25 18:17:31 -070070 bool onIsEqual(const GrFragmentProcessor& other) const override;
commit-bot@chromium.orge5280892014-02-21 17:52:29 +000071
joshualittb0a8a372014-09-23 09:50:21 -070072 SkRRect fRRect;
73 GrPrimitiveEdgeType fEdgeType;
74 uint32_t fCircularCornerFlags;
commit-bot@chromium.orge5280892014-02-21 17:52:29 +000075
Brian Salomon0c26a9d2017-07-06 10:09:38 -040076 GR_DECLARE_FRAGMENT_PROCESSOR_TEST
commit-bot@chromium.orge5280892014-02-21 17:52:29 +000077
joshualittb0a8a372014-09-23 09:50:21 -070078 typedef GrFragmentProcessor INHERITED;
commit-bot@chromium.orge5280892014-02-21 17:52:29 +000079};
80
bungeman06ca8ec2016-06-09 08:01:03 -070081sk_sp<GrFragmentProcessor> CircularRRectEffect::Make(GrPrimitiveEdgeType edgeType,
82 uint32_t circularCornerFlags,
83 const SkRRect& rrect) {
joshualittb0a8a372014-09-23 09:50:21 -070084 if (kFillAA_GrProcessorEdgeType != edgeType && kInverseFillAA_GrProcessorEdgeType != edgeType) {
halcanary96fcdcc2015-08-27 07:41:13 -070085 return nullptr;
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000086 }
bungeman06ca8ec2016-06-09 08:01:03 -070087 return sk_sp<GrFragmentProcessor>(
88 new CircularRRectEffect(edgeType, circularCornerFlags, rrect));
commit-bot@chromium.orge5280892014-02-21 17:52:29 +000089}
90
joshualittb0a8a372014-09-23 09:50:21 -070091CircularRRectEffect::CircularRRectEffect(GrPrimitiveEdgeType edgeType, uint32_t circularCornerFlags,
92 const SkRRect& rrect)
Brian Salomonf3b995b2017-02-15 10:22:23 -050093 : INHERITED(kCompatibleWithCoverageAsAlpha_OptimizationFlag)
Brian Salomon587e08f2017-01-27 10:59:27 -050094 , fRRect(rrect)
95 , fEdgeType(edgeType)
96 , fCircularCornerFlags(circularCornerFlags) {
joshualitteb2a6762014-12-04 11:35:33 -080097 this->initClassID<CircularRRectEffect>();
commit-bot@chromium.orge5280892014-02-21 17:52:29 +000098}
99
Brian Salomon1a2a7ab2017-07-26 13:11:51 -0400100sk_sp<GrFragmentProcessor> CircularRRectEffect::clone() const {
101 return sk_sp<GrFragmentProcessor>(
102 new CircularRRectEffect(fEdgeType, fCircularCornerFlags, fRRect));
103}
104
bsalomon0e08fc12014-10-15 08:19:04 -0700105bool CircularRRectEffect::onIsEqual(const GrFragmentProcessor& other) const {
joshualitt49586be2014-09-16 08:21:41 -0700106 const CircularRRectEffect& crre = other.cast<CircularRRectEffect>();
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000107 // The corner flags are derived from fRRect, so no need to check them.
108 return fEdgeType == crre.fEdgeType && fRRect == crre.fRRect;
commit-bot@chromium.orge5280892014-02-21 17:52:29 +0000109}
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000110
111//////////////////////////////////////////////////////////////////////////////
commit-bot@chromium.orge5280892014-02-21 17:52:29 +0000112
joshualittb0a8a372014-09-23 09:50:21 -0700113GR_DEFINE_FRAGMENT_PROCESSOR_TEST(CircularRRectEffect);
commit-bot@chromium.orge5280892014-02-21 17:52:29 +0000114
Hal Canary6f6961e2017-01-31 13:50:44 -0500115#if GR_TEST_UTILS
bungeman06ca8ec2016-06-09 08:01:03 -0700116sk_sp<GrFragmentProcessor> CircularRRectEffect::TestCreate(GrProcessorTestData* d) {
joshualitt0067ff52015-07-08 14:26:19 -0700117 SkScalar w = d->fRandom->nextRangeScalar(20.f, 1000.f);
118 SkScalar h = d->fRandom->nextRangeScalar(20.f, 1000.f);
119 SkScalar r = d->fRandom->nextRangeF(kRadiusMin, 9.f);
commit-bot@chromium.orge5280892014-02-21 17:52:29 +0000120 SkRRect rrect;
121 rrect.setRectXY(SkRect::MakeWH(w, h), r, r);
bungeman06ca8ec2016-06-09 08:01:03 -0700122 sk_sp<GrFragmentProcessor> fp;
commit-bot@chromium.orgcabf4b22014-03-05 18:27:43 +0000123 do {
joshualittb0a8a372014-09-23 09:50:21 -0700124 GrPrimitiveEdgeType et =
joshualitt0067ff52015-07-08 14:26:19 -0700125 (GrPrimitiveEdgeType)d->fRandom->nextULessThan(kGrProcessorEdgeTypeCnt);
bungeman06ca8ec2016-06-09 08:01:03 -0700126 fp = GrRRectEffect::Make(et, rrect);
halcanary96fcdcc2015-08-27 07:41:13 -0700127 } while (nullptr == fp);
joshualittb0a8a372014-09-23 09:50:21 -0700128 return fp;
commit-bot@chromium.orge5280892014-02-21 17:52:29 +0000129}
Hal Canary6f6961e2017-01-31 13:50:44 -0500130#endif
commit-bot@chromium.orge5280892014-02-21 17:52:29 +0000131
132//////////////////////////////////////////////////////////////////////////////
133
egdaniel64c47282015-11-13 06:54:19 -0800134class GLCircularRRectEffect : public GrGLSLFragmentProcessor {
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000135public:
robertphillips9cdb9922016-02-03 12:25:40 -0800136 GLCircularRRectEffect() {
137 fPrevRRect.setEmpty();
138 }
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000139
wangyix7c157a92015-07-22 15:08:53 -0700140 virtual void emitCode(EmitArgs&) override;
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000141
Brian Salomon94efbf52016-11-29 13:43:05 -0500142 static inline void GenKey(const GrProcessor&, const GrShaderCaps&, GrProcessorKeyBuilder*);
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000143
wangyixb1daa862015-08-18 11:29:31 -0700144protected:
Brian Salomonab015ef2017-04-04 10:15:51 -0400145 void onSetData(const GrGLSLProgramDataManager&, const GrFragmentProcessor&) override;
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000146
147private:
egdaniel018fb622015-10-28 07:26:40 -0700148 GrGLSLProgramDataManager::UniformHandle fInnerRectUniform;
149 GrGLSLProgramDataManager::UniformHandle fRadiusPlusHalfUniform;
robertphillipsbf536af2016-02-04 06:11:53 -0800150 SkRRect fPrevRRect;
egdaniel64c47282015-11-13 06:54:19 -0800151 typedef GrGLSLFragmentProcessor INHERITED;
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000152};
153
wangyix7c157a92015-07-22 15:08:53 -0700154void GLCircularRRectEffect::emitCode(EmitArgs& args) {
155 const CircularRRectEffect& crre = args.fFp.cast<CircularRRectEffect>();
egdaniel7ea439b2015-12-03 09:20:44 -0800156 GrGLSLUniformHandler* uniformHandler = args.fUniformHandler;
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000157 const char *rectName;
158 const char *radiusPlusHalfName;
commit-bot@chromium.orgc5c748c2014-03-11 15:54:51 +0000159 // The inner rect is the rrect bounds inset by the radius. Its left, top, right, and bottom
160 // edges correspond to components x, y, z, and w, respectively. When a side of the rrect has
161 // only rectangular corners, that side's value corresponds to the rect edge's value outset by
162 // half a pixel.
cdalton5e58cee2016-02-11 12:49:47 -0800163 fInnerRectUniform = uniformHandler->addUniform(kFragment_GrShaderFlag,
egdaniel7ea439b2015-12-03 09:20:44 -0800164 kVec4f_GrSLType, kDefault_GrSLPrecision,
165 "innerRect",
166 &rectName);
bsalomoncd074912015-12-09 10:33:51 -0800167 // x is (r + .5) and y is 1/(r + .5)
cdalton5e58cee2016-02-11 12:49:47 -0800168 fRadiusPlusHalfUniform = uniformHandler->addUniform(kFragment_GrShaderFlag,
bsalomoncd074912015-12-09 10:33:51 -0800169 kVec2f_GrSLType, kDefault_GrSLPrecision,
egdaniel7ea439b2015-12-03 09:20:44 -0800170 "radiusPlusHalf",
171 &radiusPlusHalfName);
joshualitt30ba4362014-08-21 20:18:45 -0700172
bsalomoncd074912015-12-09 10:33:51 -0800173 // If we're on a device with a "real" mediump then the length calculation could overflow.
174 SkString clampedCircleDistance;
Brian Salomon1edc5b92016-11-29 13:43:46 -0500175 if (args.fShaderCaps->floatPrecisionVaries()) {
bsalomoncd074912015-12-09 10:33:51 -0800176 clampedCircleDistance.printf("clamp(%s.x * (1.0 - length(dxy * %s.y)), 0.0, 1.0);",
177 radiusPlusHalfName, radiusPlusHalfName);
178 } else {
179 clampedCircleDistance.printf("clamp(%s.x - length(dxy), 0.0, 1.0);", radiusPlusHalfName);
180 }
181
cdalton85285412016-02-18 12:37:07 -0800182 GrGLSLFPFragmentBuilder* fragBuilder = args.fFragBuilder;
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000183 // At each quarter-circle corner we compute a vector that is the offset of the fragment position
184 // from the circle center. The vector is pinned in x and y to be in the quarter-plane relevant
185 // to that corner. This means that points near the interior near the rrect top edge will have
186 // a vector that points straight up for both the TL left and TR corners. Computing an
187 // alpha from this vector at either the TR or TL corner will give the correct result. Similarly,
188 // fragments near the other three edges will get the correct AA. Fragments in the interior of
189 // the rrect will have a (0,0) vector at all four corners. So long as the radius > 0.5 they will
190 // correctly produce an alpha value of 1 at all four corners. We take the min of all the alphas.
191 // The code below is a simplified version of the above that performs maxs on the vector
192 // components before computing distances and alpha values so that only one distance computation
193 // need be computed to determine the min alpha.
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000194 //
195 // For the cases where one half of the rrect is rectangular we drop one of the x or y
196 // computations, compute a separate rect edge alpha for the rect side, and mul the two computed
197 // alphas together.
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000198 switch (crre.getCircularCornerFlags()) {
199 case CircularRRectEffect::kAll_CornerFlags:
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400200 fragBuilder->codeAppendf("float2 dxy0 = %s.xy - sk_FragCoord.xy;", rectName);
201 fragBuilder->codeAppendf("float2 dxy1 = sk_FragCoord.xy - %s.zw;", rectName);
202 fragBuilder->codeAppend("float2 dxy = max(max(dxy0, dxy1), 0.0);");
bsalomoncd074912015-12-09 10:33:51 -0800203 fragBuilder->codeAppendf("float alpha = %s;", clampedCircleDistance.c_str());
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000204 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000205 case CircularRRectEffect::kTopLeft_CornerFlag:
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400206 fragBuilder->codeAppendf("float2 dxy = max(%s.xy - sk_FragCoord.xy, 0.0);",
Ethan Nicholas38657112017-02-09 17:01:22 -0500207 rectName);
208 fragBuilder->codeAppendf("float rightAlpha = clamp(%s.z - sk_FragCoord.x, 0.0, 1.0);",
209 rectName);
210 fragBuilder->codeAppendf("float bottomAlpha = clamp(%s.w - sk_FragCoord.y, 0.0, 1.0);",
211 rectName);
bsalomoncd074912015-12-09 10:33:51 -0800212 fragBuilder->codeAppendf("float alpha = bottomAlpha * rightAlpha * %s;",
213 clampedCircleDistance.c_str());
commit-bot@chromium.orgc5c748c2014-03-11 15:54:51 +0000214 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000215 case CircularRRectEffect::kTopRight_CornerFlag:
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400216 fragBuilder->codeAppendf("float2 dxy = max(float2(sk_FragCoord.x - %s.z, "
Ethan Nicholas38657112017-02-09 17:01:22 -0500217 "%s.y - sk_FragCoord.y), 0.0);",
218 rectName, rectName);
219 fragBuilder->codeAppendf("float leftAlpha = clamp(sk_FragCoord.x - %s.x, 0.0, 1.0);",
220 rectName);
221 fragBuilder->codeAppendf("float bottomAlpha = clamp(%s.w - sk_FragCoord.y, 0.0, 1.0);",
222 rectName);
bsalomoncd074912015-12-09 10:33:51 -0800223 fragBuilder->codeAppendf("float alpha = bottomAlpha * leftAlpha * %s;",
224 clampedCircleDistance.c_str());
commit-bot@chromium.orgc5c748c2014-03-11 15:54:51 +0000225 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000226 case CircularRRectEffect::kBottomRight_CornerFlag:
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400227 fragBuilder->codeAppendf("float2 dxy = max(sk_FragCoord.xy - %s.zw, 0.0);",
Ethan Nicholas38657112017-02-09 17:01:22 -0500228 rectName);
229 fragBuilder->codeAppendf("float leftAlpha = clamp(sk_FragCoord.x - %s.x, 0.0, 1.0);",
230 rectName);
231 fragBuilder->codeAppendf("float topAlpha = clamp(sk_FragCoord.y - %s.y, 0.0, 1.0);",
232 rectName);
bsalomoncd074912015-12-09 10:33:51 -0800233 fragBuilder->codeAppendf("float alpha = topAlpha * leftAlpha * %s;",
234 clampedCircleDistance.c_str());
commit-bot@chromium.orgc5c748c2014-03-11 15:54:51 +0000235 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000236 case CircularRRectEffect::kBottomLeft_CornerFlag:
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400237 fragBuilder->codeAppendf("float2 dxy = max(float2(%s.x - sk_FragCoord.x, sk_FragCoord.y - "
Ethan Nicholas38657112017-02-09 17:01:22 -0500238 "%s.w), 0.0);",
239 rectName, rectName);
240 fragBuilder->codeAppendf("float rightAlpha = clamp(%s.z - sk_FragCoord.x, 0.0, 1.0);",
241 rectName);
242 fragBuilder->codeAppendf("float topAlpha = clamp(sk_FragCoord.y - %s.y, 0.0, 1.0);",
243 rectName);
bsalomoncd074912015-12-09 10:33:51 -0800244 fragBuilder->codeAppendf("float alpha = topAlpha * rightAlpha * %s;",
245 clampedCircleDistance.c_str());
commit-bot@chromium.orgc5c748c2014-03-11 15:54:51 +0000246 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000247 case CircularRRectEffect::kLeft_CornerFlags:
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400248 fragBuilder->codeAppendf("float2 dxy0 = %s.xy - sk_FragCoord.xy;", rectName);
Ethan Nicholas38657112017-02-09 17:01:22 -0500249 fragBuilder->codeAppendf("float dy1 = sk_FragCoord.y - %s.w;", rectName);
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400250 fragBuilder->codeAppend("float2 dxy = max(float2(dxy0.x, max(dxy0.y, dy1)), 0.0);");
Ethan Nicholas38657112017-02-09 17:01:22 -0500251 fragBuilder->codeAppendf("float rightAlpha = clamp(%s.z - sk_FragCoord.x, 0.0, 1.0);",
252 rectName);
bsalomoncd074912015-12-09 10:33:51 -0800253 fragBuilder->codeAppendf("float alpha = rightAlpha * %s;",
254 clampedCircleDistance.c_str());
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000255 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000256 case CircularRRectEffect::kTop_CornerFlags:
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400257 fragBuilder->codeAppendf("float2 dxy0 = %s.xy - sk_FragCoord.xy;", rectName);
Ethan Nicholas38657112017-02-09 17:01:22 -0500258 fragBuilder->codeAppendf("float dx1 = sk_FragCoord.x - %s.z;", rectName);
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400259 fragBuilder->codeAppend("float2 dxy = max(float2(max(dxy0.x, dx1), dxy0.y), 0.0);");
Ethan Nicholas38657112017-02-09 17:01:22 -0500260 fragBuilder->codeAppendf("float bottomAlpha = clamp(%s.w - sk_FragCoord.y, 0.0, 1.0);",
261 rectName);
bsalomoncd074912015-12-09 10:33:51 -0800262 fragBuilder->codeAppendf("float alpha = bottomAlpha * %s;",
263 clampedCircleDistance.c_str());
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000264 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000265 case CircularRRectEffect::kRight_CornerFlags:
Ethan Nicholas38657112017-02-09 17:01:22 -0500266 fragBuilder->codeAppendf("float dy0 = %s.y - sk_FragCoord.y;", rectName);
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400267 fragBuilder->codeAppendf("float2 dxy1 = sk_FragCoord.xy - %s.zw;", rectName);
268 fragBuilder->codeAppend("float2 dxy = max(float2(dxy1.x, max(dy0, dxy1.y)), 0.0);");
Ethan Nicholas38657112017-02-09 17:01:22 -0500269 fragBuilder->codeAppendf("float leftAlpha = clamp(sk_FragCoord.x - %s.x, 0.0, 1.0);",
270 rectName);
bsalomoncd074912015-12-09 10:33:51 -0800271 fragBuilder->codeAppendf("float alpha = leftAlpha * %s;",
272 clampedCircleDistance.c_str());
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000273 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000274 case CircularRRectEffect::kBottom_CornerFlags:
Ethan Nicholas38657112017-02-09 17:01:22 -0500275 fragBuilder->codeAppendf("float dx0 = %s.x - sk_FragCoord.x;", rectName);
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400276 fragBuilder->codeAppendf("float2 dxy1 = sk_FragCoord.xy - %s.zw;", rectName);
277 fragBuilder->codeAppend("float2 dxy = max(float2(max(dx0, dxy1.x), dxy1.y), 0.0);");
Ethan Nicholas38657112017-02-09 17:01:22 -0500278 fragBuilder->codeAppendf("float topAlpha = clamp(sk_FragCoord.y - %s.y, 0.0, 1.0);",
279 rectName);
bsalomoncd074912015-12-09 10:33:51 -0800280 fragBuilder->codeAppendf("float alpha = topAlpha * %s;",
281 clampedCircleDistance.c_str());
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000282 break;
283 }
skia.committer@gmail.com06acb582014-03-06 03:02:32 +0000284
joshualittb0a8a372014-09-23 09:50:21 -0700285 if (kInverseFillAA_GrProcessorEdgeType == crre.getEdgeType()) {
bsalomoncd074912015-12-09 10:33:51 -0800286 fragBuilder->codeAppend("alpha = 1.0 - alpha;");
commit-bot@chromium.orgfbde87f2014-03-04 16:25:34 +0000287 }
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000288
Ethan Nicholas2983f402017-05-08 09:36:08 -0400289 fragBuilder->codeAppendf("%s = %s * alpha;", args.fOutputColor, args.fInputColor);
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000290}
291
Brian Salomon94efbf52016-11-29 13:43:05 -0500292void GLCircularRRectEffect::GenKey(const GrProcessor& processor, const GrShaderCaps&,
joshualittb0a8a372014-09-23 09:50:21 -0700293 GrProcessorKeyBuilder* b) {
294 const CircularRRectEffect& crre = processor.cast<CircularRRectEffect>();
295 GR_STATIC_ASSERT(kGrProcessorEdgeTypeCnt <= 8);
bsalomon63e99f72014-07-21 08:03:14 -0700296 b->add32((crre.getCircularCornerFlags() << 3) | crre.getEdgeType());
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000297}
298
egdaniel018fb622015-10-28 07:26:40 -0700299void GLCircularRRectEffect::onSetData(const GrGLSLProgramDataManager& pdman,
Brian Salomonab015ef2017-04-04 10:15:51 -0400300 const GrFragmentProcessor& processor) {
joshualittb0a8a372014-09-23 09:50:21 -0700301 const CircularRRectEffect& crre = processor.cast<CircularRRectEffect>();
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000302 const SkRRect& rrect = crre.getRRect();
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000303 if (rrect != fPrevRRect) {
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000304 SkRect rect = rrect.getBounds();
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000305 SkScalar radius = 0;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000306 switch (crre.getCircularCornerFlags()) {
307 case CircularRRectEffect::kAll_CornerFlags:
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000308 SkASSERT(rrect.isSimpleCircular());
309 radius = rrect.getSimpleRadii().fX;
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000310 SkASSERT(radius >= kRadiusMin);
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000311 rect.inset(radius, radius);
312 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000313 case CircularRRectEffect::kTopLeft_CornerFlag:
commit-bot@chromium.orgc5c748c2014-03-11 15:54:51 +0000314 radius = rrect.radii(SkRRect::kUpperLeft_Corner).fX;
315 rect.fLeft += radius;
316 rect.fTop += radius;
317 rect.fRight += 0.5f;
318 rect.fBottom += 0.5f;
319 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000320 case CircularRRectEffect::kTopRight_CornerFlag:
commit-bot@chromium.orgc5c748c2014-03-11 15:54:51 +0000321 radius = rrect.radii(SkRRect::kUpperRight_Corner).fX;
bsalomon@google.comde9f2512014-03-11 17:09:17 +0000322 rect.fLeft -= 0.5f;
commit-bot@chromium.orgc5c748c2014-03-11 15:54:51 +0000323 rect.fTop += radius;
324 rect.fRight -= radius;
325 rect.fBottom += 0.5f;
326 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000327 case CircularRRectEffect::kBottomRight_CornerFlag:
commit-bot@chromium.orgc5c748c2014-03-11 15:54:51 +0000328 radius = rrect.radii(SkRRect::kLowerRight_Corner).fX;
bsalomon@google.comde9f2512014-03-11 17:09:17 +0000329 rect.fLeft -= 0.5f;
330 rect.fTop -= 0.5f;
commit-bot@chromium.orgc5c748c2014-03-11 15:54:51 +0000331 rect.fRight -= radius;
332 rect.fBottom -= radius;
333 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000334 case CircularRRectEffect::kBottomLeft_CornerFlag:
commit-bot@chromium.orgc5c748c2014-03-11 15:54:51 +0000335 radius = rrect.radii(SkRRect::kLowerLeft_Corner).fX;
336 rect.fLeft += radius;
bsalomon@google.comde9f2512014-03-11 17:09:17 +0000337 rect.fTop -= 0.5f;
338 rect.fRight += 0.5f;
commit-bot@chromium.orgc5c748c2014-03-11 15:54:51 +0000339 rect.fBottom -= radius;
340 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000341 case CircularRRectEffect::kLeft_CornerFlags:
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000342 radius = rrect.radii(SkRRect::kUpperLeft_Corner).fX;
343 rect.fLeft += radius;
344 rect.fTop += radius;
345 rect.fRight += 0.5f;
346 rect.fBottom -= radius;
347 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000348 case CircularRRectEffect::kTop_CornerFlags:
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000349 radius = rrect.radii(SkRRect::kUpperLeft_Corner).fX;
350 rect.fLeft += radius;
351 rect.fTop += radius;
352 rect.fRight -= radius;
353 rect.fBottom += 0.5f;
354 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000355 case CircularRRectEffect::kRight_CornerFlags:
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000356 radius = rrect.radii(SkRRect::kUpperRight_Corner).fX;
357 rect.fLeft -= 0.5f;
358 rect.fTop += radius;
359 rect.fRight -= radius;
360 rect.fBottom -= radius;
361 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000362 case CircularRRectEffect::kBottom_CornerFlags:
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000363 radius = rrect.radii(SkRRect::kLowerLeft_Corner).fX;
364 rect.fLeft += radius;
365 rect.fTop -= 0.5f;
366 rect.fRight -= radius;
367 rect.fBottom -= radius;
368 break;
commit-bot@chromium.orgdd584222014-03-10 16:08:22 +0000369 default:
commit-bot@chromium.org88cb22b2014-04-30 14:17:00 +0000370 SkFAIL("Should have been one of the above cases.");
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000371 }
kkinnunen7510b222014-07-30 00:04:16 -0700372 pdman.set4f(fInnerRectUniform, rect.fLeft, rect.fTop, rect.fRight, rect.fBottom);
bsalomoncd074912015-12-09 10:33:51 -0800373 radius += 0.5f;
374 pdman.set2f(fRadiusPlusHalfUniform, radius, 1.f / radius);
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000375 fPrevRRect = rrect;
376 }
377}
378
joshualitteb2a6762014-12-04 11:35:33 -0800379////////////////////////////////////////////////////////////////////////////////////////////////////
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000380
Brian Salomon94efbf52016-11-29 13:43:05 -0500381void CircularRRectEffect::onGetGLSLProcessorKey(const GrShaderCaps& caps,
egdaniel57d3b032015-11-13 11:57:27 -0800382 GrProcessorKeyBuilder* b) const {
joshualitteb2a6762014-12-04 11:35:33 -0800383 GLCircularRRectEffect::GenKey(*this, caps, b);
384}
385
egdaniel57d3b032015-11-13 11:57:27 -0800386GrGLSLFragmentProcessor* CircularRRectEffect::onCreateGLSLInstance() const {
robertphillips9cdb9922016-02-03 12:25:40 -0800387 return new GLCircularRRectEffect;
joshualitteb2a6762014-12-04 11:35:33 -0800388}
389
390//////////////////////////////////////////////////////////////////////////////
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000391
joshualittb0a8a372014-09-23 09:50:21 -0700392class EllipticalRRectEffect : public GrFragmentProcessor {
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000393public:
bungeman06ca8ec2016-06-09 08:01:03 -0700394 static sk_sp<GrFragmentProcessor> Make(GrPrimitiveEdgeType, const SkRRect&);
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000395
Brian Salomond3b65972017-03-22 12:05:03 -0400396 ~EllipticalRRectEffect() override {}
joshualitteb2a6762014-12-04 11:35:33 -0800397
mtklein36352bf2015-03-25 18:17:31 -0700398 const char* name() const override { return "EllipticalRRect"; }
joshualitteb2a6762014-12-04 11:35:33 -0800399
Brian Salomon1a2a7ab2017-07-26 13:11:51 -0400400 sk_sp<GrFragmentProcessor> clone() const override;
401
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000402 const SkRRect& getRRect() const { return fRRect; }
403
joshualittb0a8a372014-09-23 09:50:21 -0700404 GrPrimitiveEdgeType getEdgeType() const { return fEdgeType; }
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000405
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000406private:
joshualittb0a8a372014-09-23 09:50:21 -0700407 EllipticalRRectEffect(GrPrimitiveEdgeType, const SkRRect&);
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000408
egdaniel57d3b032015-11-13 11:57:27 -0800409 GrGLSLFragmentProcessor* onCreateGLSLInstance() const override;
wangyixb1daa862015-08-18 11:29:31 -0700410
Brian Salomon94efbf52016-11-29 13:43:05 -0500411 void onGetGLSLProcessorKey(const GrShaderCaps&, GrProcessorKeyBuilder*) const override;
wangyix4b3050b2015-08-04 07:59:37 -0700412
mtklein36352bf2015-03-25 18:17:31 -0700413 bool onIsEqual(const GrFragmentProcessor& other) const override;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000414
Brian Salomon587e08f2017-01-27 10:59:27 -0500415 SkRRect fRRect;
416 GrPrimitiveEdgeType fEdgeType;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000417
Brian Salomon0c26a9d2017-07-06 10:09:38 -0400418 GR_DECLARE_FRAGMENT_PROCESSOR_TEST
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000419
joshualittb0a8a372014-09-23 09:50:21 -0700420 typedef GrFragmentProcessor INHERITED;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000421};
422
bungeman06ca8ec2016-06-09 08:01:03 -0700423sk_sp<GrFragmentProcessor>
424EllipticalRRectEffect::Make(GrPrimitiveEdgeType edgeType, const SkRRect& rrect) {
joshualittb0a8a372014-09-23 09:50:21 -0700425 if (kFillAA_GrProcessorEdgeType != edgeType && kInverseFillAA_GrProcessorEdgeType != edgeType) {
halcanary96fcdcc2015-08-27 07:41:13 -0700426 return nullptr;
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000427 }
bungeman06ca8ec2016-06-09 08:01:03 -0700428 return sk_sp<GrFragmentProcessor>(new EllipticalRRectEffect(edgeType, rrect));
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000429}
430
joshualittb0a8a372014-09-23 09:50:21 -0700431EllipticalRRectEffect::EllipticalRRectEffect(GrPrimitiveEdgeType edgeType, const SkRRect& rrect)
Brian Salomonf3b995b2017-02-15 10:22:23 -0500432 : INHERITED(kCompatibleWithCoverageAsAlpha_OptimizationFlag)
433 , fRRect(rrect)
434 , fEdgeType(edgeType) {
joshualitteb2a6762014-12-04 11:35:33 -0800435 this->initClassID<EllipticalRRectEffect>();
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000436}
437
Brian Salomon1a2a7ab2017-07-26 13:11:51 -0400438sk_sp<GrFragmentProcessor> EllipticalRRectEffect::clone() const {
439 return sk_sp<GrFragmentProcessor>(new EllipticalRRectEffect(fEdgeType, fRRect));
440}
441
bsalomon0e08fc12014-10-15 08:19:04 -0700442bool EllipticalRRectEffect::onIsEqual(const GrFragmentProcessor& other) const {
joshualitt49586be2014-09-16 08:21:41 -0700443 const EllipticalRRectEffect& erre = other.cast<EllipticalRRectEffect>();
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000444 return fEdgeType == erre.fEdgeType && fRRect == erre.fRRect;
445}
446
447//////////////////////////////////////////////////////////////////////////////
448
joshualittb0a8a372014-09-23 09:50:21 -0700449GR_DEFINE_FRAGMENT_PROCESSOR_TEST(EllipticalRRectEffect);
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000450
Hal Canary6f6961e2017-01-31 13:50:44 -0500451#if GR_TEST_UTILS
bungeman06ca8ec2016-06-09 08:01:03 -0700452sk_sp<GrFragmentProcessor> EllipticalRRectEffect::TestCreate(GrProcessorTestData* d) {
joshualitt0067ff52015-07-08 14:26:19 -0700453 SkScalar w = d->fRandom->nextRangeScalar(20.f, 1000.f);
454 SkScalar h = d->fRandom->nextRangeScalar(20.f, 1000.f);
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000455 SkVector r[4];
joshualitt0067ff52015-07-08 14:26:19 -0700456 r[SkRRect::kUpperLeft_Corner].fX = d->fRandom->nextRangeF(kRadiusMin, 9.f);
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000457 // ensure at least one corner really is elliptical
458 do {
joshualitt0067ff52015-07-08 14:26:19 -0700459 r[SkRRect::kUpperLeft_Corner].fY = d->fRandom->nextRangeF(kRadiusMin, 9.f);
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000460 } while (r[SkRRect::kUpperLeft_Corner].fY == r[SkRRect::kUpperLeft_Corner].fX);
461
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000462 SkRRect rrect;
joshualitt0067ff52015-07-08 14:26:19 -0700463 if (d->fRandom->nextBool()) {
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000464 // half the time create a four-radii rrect.
joshualitt0067ff52015-07-08 14:26:19 -0700465 r[SkRRect::kLowerRight_Corner].fX = d->fRandom->nextRangeF(kRadiusMin, 9.f);
466 r[SkRRect::kLowerRight_Corner].fY = d->fRandom->nextRangeF(kRadiusMin, 9.f);
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000467
468 r[SkRRect::kUpperRight_Corner].fX = r[SkRRect::kLowerRight_Corner].fX;
469 r[SkRRect::kUpperRight_Corner].fY = r[SkRRect::kUpperLeft_Corner].fY;
470
471 r[SkRRect::kLowerLeft_Corner].fX = r[SkRRect::kUpperLeft_Corner].fX;
472 r[SkRRect::kLowerLeft_Corner].fY = r[SkRRect::kLowerRight_Corner].fY;
473
474 rrect.setRectRadii(SkRect::MakeWH(w, h), r);
475 } else {
476 rrect.setRectXY(SkRect::MakeWH(w, h), r[SkRRect::kUpperLeft_Corner].fX,
477 r[SkRRect::kUpperLeft_Corner].fY);
478 }
bungeman06ca8ec2016-06-09 08:01:03 -0700479 sk_sp<GrFragmentProcessor> fp;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000480 do {
joshualitt0067ff52015-07-08 14:26:19 -0700481 GrPrimitiveEdgeType et =
482 (GrPrimitiveEdgeType)d->fRandom->nextULessThan(kGrProcessorEdgeTypeCnt);
bungeman06ca8ec2016-06-09 08:01:03 -0700483 fp = GrRRectEffect::Make(et, rrect);
halcanary96fcdcc2015-08-27 07:41:13 -0700484 } while (nullptr == fp);
joshualittb0a8a372014-09-23 09:50:21 -0700485 return fp;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000486}
Hal Canary6f6961e2017-01-31 13:50:44 -0500487#endif
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000488
489//////////////////////////////////////////////////////////////////////////////
490
egdaniel64c47282015-11-13 06:54:19 -0800491class GLEllipticalRRectEffect : public GrGLSLFragmentProcessor {
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000492public:
robertphillips9cdb9922016-02-03 12:25:40 -0800493 GLEllipticalRRectEffect() {
494 fPrevRRect.setEmpty();
495 }
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000496
robertphillips9cdb9922016-02-03 12:25:40 -0800497 void emitCode(EmitArgs&) override;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000498
Brian Salomon94efbf52016-11-29 13:43:05 -0500499 static inline void GenKey(const GrProcessor&, const GrShaderCaps&, GrProcessorKeyBuilder*);
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000500
wangyixb1daa862015-08-18 11:29:31 -0700501protected:
Brian Salomonab015ef2017-04-04 10:15:51 -0400502 void onSetData(const GrGLSLProgramDataManager&, const GrFragmentProcessor&) override;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000503
504private:
egdaniel018fb622015-10-28 07:26:40 -0700505 GrGLSLProgramDataManager::UniformHandle fInnerRectUniform;
506 GrGLSLProgramDataManager::UniformHandle fInvRadiiSqdUniform;
bsalomone87256c42015-12-09 17:14:40 -0800507 GrGLSLProgramDataManager::UniformHandle fScaleUniform;
508 SkRRect fPrevRRect;
egdaniel64c47282015-11-13 06:54:19 -0800509 typedef GrGLSLFragmentProcessor INHERITED;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000510};
511
wangyix7c157a92015-07-22 15:08:53 -0700512void GLEllipticalRRectEffect::emitCode(EmitArgs& args) {
513 const EllipticalRRectEffect& erre = args.fFp.cast<EllipticalRRectEffect>();
egdaniel7ea439b2015-12-03 09:20:44 -0800514 GrGLSLUniformHandler* uniformHandler = args.fUniformHandler;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000515 const char *rectName;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000516 // The inner rect is the rrect bounds inset by the x/y radii
cdalton5e58cee2016-02-11 12:49:47 -0800517 fInnerRectUniform = uniformHandler->addUniform(kFragment_GrShaderFlag,
egdaniel7ea439b2015-12-03 09:20:44 -0800518 kVec4f_GrSLType, kDefault_GrSLPrecision,
519 "innerRect",
520 &rectName);
joshualitt30ba4362014-08-21 20:18:45 -0700521
cdalton85285412016-02-18 12:37:07 -0800522 GrGLSLFPFragmentBuilder* fragBuilder = args.fFragBuilder;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000523 // At each quarter-ellipse corner we compute a vector that is the offset of the fragment pos
524 // to the ellipse center. The vector is pinned in x and y to be in the quarter-plane relevant
525 // to that corner. This means that points near the interior near the rrect top edge will have
526 // a vector that points straight up for both the TL left and TR corners. Computing an
527 // alpha from this vector at either the TR or TL corner will give the correct result. Similarly,
528 // fragments near the other three edges will get the correct AA. Fragments in the interior of
529 // the rrect will have a (0,0) vector at all four corners. So long as the radii > 0.5 they will
530 // correctly produce an alpha value of 1 at all four corners. We take the min of all the alphas.
bsalomonc41f4d62015-08-03 14:23:03 -0700531 //
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000532 // The code below is a simplified version of the above that performs maxs on the vector
533 // components before computing distances and alpha values so that only one distance computation
534 // need be computed to determine the min alpha.
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400535 fragBuilder->codeAppendf("float2 dxy0 = %s.xy - sk_FragCoord.xy;", rectName);
536 fragBuilder->codeAppendf("float2 dxy1 = sk_FragCoord.xy - %s.zw;", rectName);
bsalomone87256c42015-12-09 17:14:40 -0800537
538 // If we're on a device with a "real" mediump then we'll do the distance computation in a space
539 // that is normalized by the largest radius. The scale uniform will be scale, 1/scale. The
540 // radii uniform values are already in this normalized space.
541 const char* scaleName = nullptr;
Brian Salomon1edc5b92016-11-29 13:43:46 -0500542 if (args.fShaderCaps->floatPrecisionVaries()) {
cdalton5e58cee2016-02-11 12:49:47 -0800543 fScaleUniform = uniformHandler->addUniform(kFragment_GrShaderFlag,
bsalomone87256c42015-12-09 17:14:40 -0800544 kVec2f_GrSLType, kDefault_GrSLPrecision,
545 "scale", &scaleName);
546 }
547
bsalomonc41f4d62015-08-03 14:23:03 -0700548 // The uniforms with the inv squared radii are highp to prevent underflow.
commit-bot@chromium.org9615d5f2014-03-20 21:39:03 +0000549 switch (erre.getRRect().getType()) {
550 case SkRRect::kSimple_Type: {
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000551 const char *invRadiiXYSqdName;
cdalton5e58cee2016-02-11 12:49:47 -0800552 fInvRadiiSqdUniform = uniformHandler->addUniform(kFragment_GrShaderFlag,
553 kVec2f_GrSLType,
554 kDefault_GrSLPrecision,
555 "invRadiiXY",
556 &invRadiiXYSqdName);
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400557 fragBuilder->codeAppend("float2 dxy = max(max(dxy0, dxy1), 0.0);");
bsalomone87256c42015-12-09 17:14:40 -0800558 if (scaleName) {
559 fragBuilder->codeAppendf("dxy *= %s.y;", scaleName);
560 }
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000561 // Z is the x/y offsets divided by squared radii.
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400562 fragBuilder->codeAppendf("float2 Z = dxy * %s.xy;", invRadiiXYSqdName);
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000563 break;
564 }
commit-bot@chromium.org9615d5f2014-03-20 21:39:03 +0000565 case SkRRect::kNinePatch_Type: {
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000566 const char *invRadiiLTRBSqdName;
cdalton5e58cee2016-02-11 12:49:47 -0800567 fInvRadiiSqdUniform = uniformHandler->addUniform(kFragment_GrShaderFlag,
568 kVec4f_GrSLType,
569 kDefault_GrSLPrecision,
570 "invRadiiLTRB",
571 &invRadiiLTRBSqdName);
bsalomone87256c42015-12-09 17:14:40 -0800572 if (scaleName) {
573 fragBuilder->codeAppendf("dxy0 *= %s.y;", scaleName);
574 fragBuilder->codeAppendf("dxy1 *= %s.y;", scaleName);
575 }
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400576 fragBuilder->codeAppend("float2 dxy = max(max(dxy0, dxy1), 0.0);");
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000577 // Z is the x/y offsets divided by squared radii. We only care about the (at most) one
578 // corner where both the x and y offsets are positive, hence the maxes. (The inverse
579 // squared radii will always be positive.)
Ethan Nicholas5af9ea32017-07-28 15:19:46 -0400580 fragBuilder->codeAppendf("float2 Z = max(max(dxy0 * %s.xy, dxy1 * %s.zw), 0.0);",
egdaniel4ca2e602015-11-18 08:01:26 -0800581 invRadiiLTRBSqdName, invRadiiLTRBSqdName);
bsalomone87256c42015-12-09 17:14:40 -0800582
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000583 break;
commit-bot@chromium.org9615d5f2014-03-20 21:39:03 +0000584 }
585 default:
commit-bot@chromium.org88cb22b2014-04-30 14:17:00 +0000586 SkFAIL("RRect should always be simple or nine-patch.");
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000587 }
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000588 // implicit is the evaluation of (x/a)^2 + (y/b)^2 - 1.
bsalomone87256c42015-12-09 17:14:40 -0800589 fragBuilder->codeAppend("float implicit = dot(Z, dxy) - 1.0;");
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000590 // grad_dot is the squared length of the gradient of the implicit.
bsalomone87256c42015-12-09 17:14:40 -0800591 fragBuilder->codeAppend("float grad_dot = 4.0 * dot(Z, Z);");
commit-bot@chromium.org1b035d82014-04-09 17:11:09 +0000592 // avoid calling inversesqrt on zero.
bsalomone87256c42015-12-09 17:14:40 -0800593 fragBuilder->codeAppend("grad_dot = max(grad_dot, 1.0e-4);");
594 fragBuilder->codeAppend("float approx_dist = implicit * inversesqrt(grad_dot);");
595 if (scaleName) {
596 fragBuilder->codeAppendf("approx_dist *= %s.x;", scaleName);
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000597 }
598
bsalomone87256c42015-12-09 17:14:40 -0800599 if (kFillAA_GrProcessorEdgeType == erre.getEdgeType()) {
600 fragBuilder->codeAppend("float alpha = clamp(0.5 - approx_dist, 0.0, 1.0);");
601 } else {
602 fragBuilder->codeAppend("float alpha = clamp(0.5 + approx_dist, 0.0, 1.0);");
603 }
604
Ethan Nicholas2983f402017-05-08 09:36:08 -0400605 fragBuilder->codeAppendf("%s = %s * alpha;", args.fOutputColor, args.fInputColor);
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000606}
607
Brian Salomon94efbf52016-11-29 13:43:05 -0500608void GLEllipticalRRectEffect::GenKey(const GrProcessor& effect, const GrShaderCaps&,
joshualittb0a8a372014-09-23 09:50:21 -0700609 GrProcessorKeyBuilder* b) {
joshualitt49586be2014-09-16 08:21:41 -0700610 const EllipticalRRectEffect& erre = effect.cast<EllipticalRRectEffect>();
joshualittb0a8a372014-09-23 09:50:21 -0700611 GR_STATIC_ASSERT(kLast_GrProcessorEdgeType < (1 << 3));
bsalomon63e99f72014-07-21 08:03:14 -0700612 b->add32(erre.getRRect().getType() | erre.getEdgeType() << 3);
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000613}
614
egdaniel018fb622015-10-28 07:26:40 -0700615void GLEllipticalRRectEffect::onSetData(const GrGLSLProgramDataManager& pdman,
Brian Salomonab015ef2017-04-04 10:15:51 -0400616 const GrFragmentProcessor& effect) {
joshualitt49586be2014-09-16 08:21:41 -0700617 const EllipticalRRectEffect& erre = effect.cast<EllipticalRRectEffect>();
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000618 const SkRRect& rrect = erre.getRRect();
bsalomone87256c42015-12-09 17:14:40 -0800619 // If we're using a scale factor to work around precision issues, choose the largest radius
620 // as the scale factor. The inv radii need to be pre-adjusted by the scale factor.
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000621 if (rrect != fPrevRRect) {
622 SkRect rect = rrect.getBounds();
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000623 const SkVector& r0 = rrect.radii(SkRRect::kUpperLeft_Corner);
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000624 SkASSERT(r0.fX >= kRadiusMin);
625 SkASSERT(r0.fY >= kRadiusMin);
commit-bot@chromium.org9615d5f2014-03-20 21:39:03 +0000626 switch (erre.getRRect().getType()) {
627 case SkRRect::kSimple_Type:
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000628 rect.inset(r0.fX, r0.fY);
bsalomone87256c42015-12-09 17:14:40 -0800629 if (fScaleUniform.isValid()) {
630 if (r0.fX > r0.fY) {
631 pdman.set2f(fInvRadiiSqdUniform, 1.f, (r0.fX * r0.fX) / (r0.fY * r0.fY));
632 pdman.set2f(fScaleUniform, r0.fX, 1.f / r0.fX);
633 } else {
634 pdman.set2f(fInvRadiiSqdUniform, (r0.fY * r0.fY) / (r0.fX * r0.fX), 1.f);
635 pdman.set2f(fScaleUniform, r0.fY, 1.f / r0.fY);
636 }
637 } else {
638 pdman.set2f(fInvRadiiSqdUniform, 1.f / (r0.fX * r0.fX),
639 1.f / (r0.fY * r0.fY));
640 }
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000641 break;
commit-bot@chromium.org9615d5f2014-03-20 21:39:03 +0000642 case SkRRect::kNinePatch_Type: {
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000643 const SkVector& r1 = rrect.radii(SkRRect::kLowerRight_Corner);
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000644 SkASSERT(r1.fX >= kRadiusMin);
645 SkASSERT(r1.fY >= kRadiusMin);
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000646 rect.fLeft += r0.fX;
647 rect.fTop += r0.fY;
648 rect.fRight -= r1.fX;
649 rect.fBottom -= r1.fY;
bsalomone87256c42015-12-09 17:14:40 -0800650 if (fScaleUniform.isValid()) {
651 float scale = SkTMax(SkTMax(r0.fX, r0.fY), SkTMax(r1.fX, r1.fY));
652 float scaleSqd = scale * scale;
653 pdman.set4f(fInvRadiiSqdUniform, scaleSqd / (r0.fX * r0.fX),
654 scaleSqd / (r0.fY * r0.fY),
655 scaleSqd / (r1.fX * r1.fX),
656 scaleSqd / (r1.fY * r1.fY));
657 pdman.set2f(fScaleUniform, scale, 1.f / scale);
658 } else {
659 pdman.set4f(fInvRadiiSqdUniform, 1.f / (r0.fX * r0.fX),
660 1.f / (r0.fY * r0.fY),
661 1.f / (r1.fX * r1.fX),
662 1.f / (r1.fY * r1.fY));
663 }
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000664 break;
665 }
commit-bot@chromium.org9615d5f2014-03-20 21:39:03 +0000666 default:
commit-bot@chromium.org88cb22b2014-04-30 14:17:00 +0000667 SkFAIL("RRect should always be simple or nine-patch.");
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000668 }
kkinnunen7510b222014-07-30 00:04:16 -0700669 pdman.set4f(fInnerRectUniform, rect.fLeft, rect.fTop, rect.fRight, rect.fBottom);
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000670 fPrevRRect = rrect;
671 }
672}
673
joshualitteb2a6762014-12-04 11:35:33 -0800674////////////////////////////////////////////////////////////////////////////////////////////////////
675
Brian Salomon94efbf52016-11-29 13:43:05 -0500676void EllipticalRRectEffect::onGetGLSLProcessorKey(const GrShaderCaps& caps,
egdaniel57d3b032015-11-13 11:57:27 -0800677 GrProcessorKeyBuilder* b) const {
joshualitteb2a6762014-12-04 11:35:33 -0800678 GLEllipticalRRectEffect::GenKey(*this, caps, b);
679}
680
egdaniel57d3b032015-11-13 11:57:27 -0800681GrGLSLFragmentProcessor* EllipticalRRectEffect::onCreateGLSLInstance() const {
robertphillips9cdb9922016-02-03 12:25:40 -0800682 return new GLEllipticalRRectEffect;
joshualitteb2a6762014-12-04 11:35:33 -0800683}
684
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000685//////////////////////////////////////////////////////////////////////////////
686
bungeman06ca8ec2016-06-09 08:01:03 -0700687sk_sp<GrFragmentProcessor> GrRRectEffect::Make(GrPrimitiveEdgeType edgeType, const SkRRect& rrect) {
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000688 if (rrect.isRect()) {
bungeman06ca8ec2016-06-09 08:01:03 -0700689 return GrConvexPolyEffect::Make(edgeType, rrect.getBounds());
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000690 }
691
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000692 if (rrect.isOval()) {
bungeman06ca8ec2016-06-09 08:01:03 -0700693 return GrOvalEffect::Make(edgeType, rrect.getBounds());
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000694 }
695
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000696 if (rrect.isSimple()) {
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000697 if (rrect.getSimpleRadii().fX < kRadiusMin || rrect.getSimpleRadii().fY < kRadiusMin) {
698 // In this case the corners are extremely close to rectangular and we collapse the
699 // clip to a rectangular clip.
bungeman06ca8ec2016-06-09 08:01:03 -0700700 return GrConvexPolyEffect::Make(edgeType, rrect.getBounds());
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000701 }
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000702 if (rrect.getSimpleRadii().fX == rrect.getSimpleRadii().fY) {
bungeman06ca8ec2016-06-09 08:01:03 -0700703 return CircularRRectEffect::Make(edgeType, CircularRRectEffect::kAll_CornerFlags,
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000704 rrect);
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000705 } else {
bungeman06ca8ec2016-06-09 08:01:03 -0700706 return EllipticalRRectEffect::Make(edgeType, rrect);
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000707 }
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000708 }
709
710 if (rrect.isComplex() || rrect.isNinePatch()) {
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000711 // Check for the "tab" cases - two adjacent circular corners and two square corners.
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000712 SkScalar circularRadius = 0;
713 uint32_t cornerFlags = 0;
714
715 SkVector radii[4];
716 bool squashedRadii = false;
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000717 for (int c = 0; c < 4; ++c) {
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000718 radii[c] = rrect.radii((SkRRect::Corner)c);
719 SkASSERT((0 == radii[c].fX) == (0 == radii[c].fY));
720 if (0 == radii[c].fX) {
721 // The corner is square, so no need to squash or flag as circular.
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000722 continue;
723 }
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000724 if (radii[c].fX < kRadiusMin || radii[c].fY < kRadiusMin) {
725 radii[c].set(0, 0);
726 squashedRadii = true;
727 continue;
728 }
729 if (radii[c].fX != radii[c].fY) {
bsalomon@google.com44a435b2014-03-13 19:20:32 +0000730 cornerFlags = ~0U;
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000731 break;
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000732 }
commit-bot@chromium.orgdd584222014-03-10 16:08:22 +0000733 if (!cornerFlags) {
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000734 circularRadius = radii[c].fX;
commit-bot@chromium.orgdd584222014-03-10 16:08:22 +0000735 cornerFlags = 1 << c;
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000736 } else {
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000737 if (radii[c].fX != circularRadius) {
bsalomon@google.com44a435b2014-03-13 19:20:32 +0000738 cornerFlags = ~0U;
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000739 break;
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000740 }
commit-bot@chromium.orgdd584222014-03-10 16:08:22 +0000741 cornerFlags |= 1 << c;
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000742 }
743 }
744
commit-bot@chromium.orgdd584222014-03-10 16:08:22 +0000745 switch (cornerFlags) {
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000746 case CircularRRectEffect::kAll_CornerFlags:
747 // This rrect should have been caught in the simple case above. Though, it would
748 // be correctly handled in the fallthrough code.
749 SkASSERT(false);
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000750 case CircularRRectEffect::kTopLeft_CornerFlag:
751 case CircularRRectEffect::kTopRight_CornerFlag:
752 case CircularRRectEffect::kBottomRight_CornerFlag:
753 case CircularRRectEffect::kBottomLeft_CornerFlag:
754 case CircularRRectEffect::kLeft_CornerFlags:
755 case CircularRRectEffect::kTop_CornerFlags:
756 case CircularRRectEffect::kRight_CornerFlags:
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000757 case CircularRRectEffect::kBottom_CornerFlags: {
758 SkTCopyOnFirstWrite<SkRRect> rr(rrect);
759 if (squashedRadii) {
760 rr.writable()->setRectRadii(rrect.getBounds(), radii);
761 }
bungeman06ca8ec2016-06-09 08:01:03 -0700762 return CircularRRectEffect::Make(edgeType, cornerFlags, *rr);
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000763 }
764 case CircularRRectEffect::kNone_CornerFlags:
bungeman06ca8ec2016-06-09 08:01:03 -0700765 return GrConvexPolyEffect::Make(edgeType, rrect.getBounds());
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000766 default: {
767 if (squashedRadii) {
768 // If we got here then we squashed some but not all the radii to zero. (If all
769 // had been squashed cornerFlags would be 0.) The elliptical effect doesn't
770 // support some rounded and some square corners.
halcanary96fcdcc2015-08-27 07:41:13 -0700771 return nullptr;
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000772 }
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000773 if (rrect.isNinePatch()) {
bungeman06ca8ec2016-06-09 08:01:03 -0700774 return EllipticalRRectEffect::Make(edgeType, rrect);
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000775 }
halcanary96fcdcc2015-08-27 07:41:13 -0700776 return nullptr;
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000777 }
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000778 }
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000779 }
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000780
halcanary96fcdcc2015-08-27 07:41:13 -0700781 return nullptr;
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000782}