blob: 09dd82ffb7a2f30f38f4268a97875a8b94016753 [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
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "src/gpu/effects/GrRRectEffect.h"
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +00009
Mike Kleinc0bd9f92019-04-23 12:05:21 -050010#include "src/core/SkRRectPriv.h"
11#include "src/core/SkTLazy.h"
12#include "src/gpu/GrFragmentProcessor.h"
13#include "src/gpu/GrShaderCaps.h"
14#include "src/gpu/effects/GrConvexPolyEffect.h"
15#include "src/gpu/effects/GrOvalEffect.h"
16#include "src/gpu/glsl/GrGLSLFragmentProcessor.h"
17#include "src/gpu/glsl/GrGLSLFragmentShaderBuilder.h"
18#include "src/gpu/glsl/GrGLSLProgramDataManager.h"
19#include "src/gpu/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).
John Stiles72e57642020-06-24 10:42:35 -040048 static GrFPResult Make(std::unique_ptr<GrFragmentProcessor>, GrClipEdgeType,
John Stiles851b90e2020-06-17 13:53:37 -040049 uint32_t circularCornerFlags, 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 Salomonaff329b2017-08-11 09:40:37 -040055 std::unique_ptr<GrFragmentProcessor> clone() const override;
Brian Salomon1a2a7ab2017-07-26 13:11:51 -040056
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
Ethan Nicholas0f3c7322017-11-09 14:51:17 -050061 GrClipEdgeType getEdgeType() const { return fEdgeType; }
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +000062
commit-bot@chromium.orge5280892014-02-21 17:52:29 +000063private:
John Stiles56bf90b2020-06-10 17:25:58 -040064 CircularRRectEffect(std::unique_ptr<GrFragmentProcessor> inputFP,
65 GrClipEdgeType, uint32_t circularCornerFlags, const SkRRect&);
John Stilesc0d5d9c2020-06-15 15:57:25 -040066 CircularRRectEffect(const CircularRRectEffect& that);
commit-bot@chromium.orge5280892014-02-21 17:52:29 +000067
Brian Salomon3176e862021-08-09 11:23:04 -040068 std::unique_ptr<ProgramImpl> onMakeProgramImpl() const override;
wangyixb1daa862015-08-18 11:29:31 -070069
Brian Salomon13b28732021-08-06 15:33:58 -040070 void onAddToKey(const GrShaderCaps&, GrProcessorKeyBuilder*) const override;
wangyix4b3050b2015-08-04 07:59:37 -070071
mtklein36352bf2015-03-25 18:17:31 -070072 bool onIsEqual(const GrFragmentProcessor& other) const override;
commit-bot@chromium.orge5280892014-02-21 17:52:29 +000073
John Stiles56bf90b2020-06-10 17:25:58 -040074 SkRRect fRRect;
Ethan Nicholas0f3c7322017-11-09 14:51:17 -050075 GrClipEdgeType fEdgeType;
John Stiles56bf90b2020-06-10 17:25:58 -040076 uint32_t fCircularCornerFlags;
commit-bot@chromium.orge5280892014-02-21 17:52:29 +000077
Brian Salomon0c26a9d2017-07-06 10:09:38 -040078 GR_DECLARE_FRAGMENT_PROCESSOR_TEST
commit-bot@chromium.orge5280892014-02-21 17:52:29 +000079
John Stiles7571f9e2020-09-02 22:42:33 -040080 using INHERITED = GrFragmentProcessor;
commit-bot@chromium.orge5280892014-02-21 17:52:29 +000081};
82
John Stiles72e57642020-06-24 10:42:35 -040083GrFPResult CircularRRectEffect::Make(std::unique_ptr<GrFragmentProcessor> inputFP,
84 GrClipEdgeType edgeType,
85 uint32_t circularCornerFlags, const SkRRect& rrect) {
Ethan Nicholas1706f842017-11-10 11:58:19 -050086 if (GrClipEdgeType::kFillAA != edgeType && GrClipEdgeType::kInverseFillAA != edgeType) {
John Stiles72e57642020-06-24 10:42:35 -040087 return GrFPFailure(std::move(inputFP));
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +000088 }
John Stiles72e57642020-06-24 10:42:35 -040089 return GrFPSuccess(std::unique_ptr<GrFragmentProcessor>(
John Stiles851b90e2020-06-17 13:53:37 -040090 new CircularRRectEffect(std::move(inputFP), edgeType, circularCornerFlags, rrect)));
commit-bot@chromium.orge5280892014-02-21 17:52:29 +000091}
92
John Stiles56bf90b2020-06-10 17:25:58 -040093CircularRRectEffect::CircularRRectEffect(std::unique_ptr<GrFragmentProcessor> inputFP,
Brian Osman8f391372021-04-29 15:27:34 -040094 GrClipEdgeType edgeType,
95 uint32_t circularCornerFlags,
joshualittb0a8a372014-09-23 09:50:21 -070096 const SkRRect& rrect)
Brian Osman8f391372021-04-29 15:27:34 -040097 : INHERITED(kCircularRRectEffect_ClassID,
98 ProcessorOptimizationFlags(inputFP.get()) &
99 kCompatibleWithCoverageAsAlpha_OptimizationFlag)
Brian Salomon587e08f2017-01-27 10:59:27 -0500100 , fRRect(rrect)
101 , fEdgeType(edgeType)
102 , fCircularCornerFlags(circularCornerFlags) {
Brian Osman54867de2020-07-10 14:22:57 -0400103 this->registerChild(std::move(inputFP));
John Stilesc0d5d9c2020-06-15 15:57:25 -0400104}
105
106CircularRRectEffect::CircularRRectEffect(const CircularRRectEffect& that)
John Stiles307f8f52021-08-09 15:36:59 -0400107 : INHERITED(that)
John Stilesc0d5d9c2020-06-15 15:57:25 -0400108 , fRRect(that.fRRect)
109 , fEdgeType(that.fEdgeType)
John Stiles307f8f52021-08-09 15:36:59 -0400110 , fCircularCornerFlags(that.fCircularCornerFlags) {}
commit-bot@chromium.orge5280892014-02-21 17:52:29 +0000111
Brian Salomonaff329b2017-08-11 09:40:37 -0400112std::unique_ptr<GrFragmentProcessor> CircularRRectEffect::clone() const {
John Stilesc0d5d9c2020-06-15 15:57:25 -0400113 return std::unique_ptr<GrFragmentProcessor>(new CircularRRectEffect(*this));
Brian Salomon1a2a7ab2017-07-26 13:11:51 -0400114}
115
bsalomon0e08fc12014-10-15 08:19:04 -0700116bool CircularRRectEffect::onIsEqual(const GrFragmentProcessor& other) const {
joshualitt49586be2014-09-16 08:21:41 -0700117 const CircularRRectEffect& crre = other.cast<CircularRRectEffect>();
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000118 // The corner flags are derived from fRRect, so no need to check them.
119 return fEdgeType == crre.fEdgeType && fRRect == crre.fRRect;
commit-bot@chromium.orge5280892014-02-21 17:52:29 +0000120}
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000121
122//////////////////////////////////////////////////////////////////////////////
commit-bot@chromium.orge5280892014-02-21 17:52:29 +0000123
joshualittb0a8a372014-09-23 09:50:21 -0700124GR_DEFINE_FRAGMENT_PROCESSOR_TEST(CircularRRectEffect);
commit-bot@chromium.orge5280892014-02-21 17:52:29 +0000125
Hal Canary6f6961e2017-01-31 13:50:44 -0500126#if GR_TEST_UTILS
Brian Salomonaff329b2017-08-11 09:40:37 -0400127std::unique_ptr<GrFragmentProcessor> CircularRRectEffect::TestCreate(GrProcessorTestData* d) {
joshualitt0067ff52015-07-08 14:26:19 -0700128 SkScalar w = d->fRandom->nextRangeScalar(20.f, 1000.f);
129 SkScalar h = d->fRandom->nextRangeScalar(20.f, 1000.f);
130 SkScalar r = d->fRandom->nextRangeF(kRadiusMin, 9.f);
commit-bot@chromium.orge5280892014-02-21 17:52:29 +0000131 SkRRect rrect;
132 rrect.setRectXY(SkRect::MakeWH(w, h), r, r);
John Stiles6609cb62020-07-17 14:52:12 -0400133 std::unique_ptr<GrFragmentProcessor> fp = d->inputFP();
John Stiles851b90e2020-06-17 13:53:37 -0400134 bool success;
commit-bot@chromium.orgcabf4b22014-03-05 18:27:43 +0000135 do {
Ethan Nicholas0f3c7322017-11-09 14:51:17 -0500136 GrClipEdgeType et =
Ethan Nicholas1706f842017-11-10 11:58:19 -0500137 (GrClipEdgeType)d->fRandom->nextULessThan(kGrClipEdgeTypeCnt);
John Stiles6609cb62020-07-17 14:52:12 -0400138 std::tie(success, fp) = GrRRectEffect::Make(std::move(fp), et, rrect,
John Stiles851b90e2020-06-17 13:53:37 -0400139 *d->caps()->shaderCaps());
140 } while (!success);
joshualittb0a8a372014-09-23 09:50:21 -0700141 return fp;
commit-bot@chromium.orge5280892014-02-21 17:52:29 +0000142}
Hal Canary6f6961e2017-01-31 13:50:44 -0500143#endif
commit-bot@chromium.orge5280892014-02-21 17:52:29 +0000144
145//////////////////////////////////////////////////////////////////////////////
146
Brian Salomon3176e862021-08-09 11:23:04 -0400147class GLCircularRRectEffect : public GrFragmentProcessor::ProgramImpl {
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000148public:
Brian Salomon0a241ce2017-12-15 11:31:05 -0500149 GLCircularRRectEffect() = default;
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000150
John Stiles1cf2c8d2020-08-13 22:58:04 -0400151 void emitCode(EmitArgs&) override;
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000152
Brian Salomon94efbf52016-11-29 13:43:05 -0500153 static inline void GenKey(const GrProcessor&, const GrShaderCaps&, GrProcessorKeyBuilder*);
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000154
wangyixb1daa862015-08-18 11:29:31 -0700155protected:
Brian Salomonab015ef2017-04-04 10:15:51 -0400156 void onSetData(const GrGLSLProgramDataManager&, const GrFragmentProcessor&) override;
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000157
158private:
egdaniel018fb622015-10-28 07:26:40 -0700159 GrGLSLProgramDataManager::UniformHandle fInnerRectUniform;
160 GrGLSLProgramDataManager::UniformHandle fRadiusPlusHalfUniform;
robertphillipsbf536af2016-02-04 06:11:53 -0800161 SkRRect fPrevRRect;
Brian Salomon3176e862021-08-09 11:23:04 -0400162 using INHERITED = ProgramImpl;
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000163};
164
wangyix7c157a92015-07-22 15:08:53 -0700165void GLCircularRRectEffect::emitCode(EmitArgs& args) {
166 const CircularRRectEffect& crre = args.fFp.cast<CircularRRectEffect>();
egdaniel7ea439b2015-12-03 09:20:44 -0800167 GrGLSLUniformHandler* uniformHandler = args.fUniformHandler;
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000168 const char *rectName;
169 const char *radiusPlusHalfName;
commit-bot@chromium.orgc5c748c2014-03-11 15:54:51 +0000170 // The inner rect is the rrect bounds inset by the radius. Its left, top, right, and bottom
171 // edges correspond to components x, y, z, and w, respectively. When a side of the rrect has
172 // only rectangular corners, that side's value corresponds to the rect edge's value outset by
173 // half a pixel.
Ethan Nicholas16464c32020-04-06 13:53:05 -0400174 fInnerRectUniform = uniformHandler->addUniform(&crre, kFragment_GrShaderFlag, kFloat4_GrSLType,
Ethan Nicholasf7b88202017-09-18 14:10:39 -0400175 "innerRect", &rectName);
bsalomoncd074912015-12-09 10:33:51 -0800176 // x is (r + .5) and y is 1/(r + .5)
Ethan Nicholas16464c32020-04-06 13:53:05 -0400177 fRadiusPlusHalfUniform = uniformHandler->addUniform(&crre, kFragment_GrShaderFlag,
178 kHalf2_GrSLType, "radiusPlusHalf",
179 &radiusPlusHalfName);
joshualitt30ba4362014-08-21 20:18:45 -0700180
Chris Dalton47c8ed32017-11-15 18:27:09 -0700181 // If we're on a device where float != fp32 then the length calculation could overflow.
bsalomoncd074912015-12-09 10:33:51 -0800182 SkString clampedCircleDistance;
Chris Dalton47c8ed32017-11-15 18:27:09 -0700183 if (!args.fShaderCaps->floatIs32Bits()) {
Ethan Nicholase1f55022019-02-05 17:17:40 -0500184 clampedCircleDistance.printf("saturate(%s.x * (1.0 - length(dxy * %s.y)))",
bsalomoncd074912015-12-09 10:33:51 -0800185 radiusPlusHalfName, radiusPlusHalfName);
186 } else {
Ethan Nicholase1f55022019-02-05 17:17:40 -0500187 clampedCircleDistance.printf("saturate(%s.x - length(dxy))", radiusPlusHalfName);
bsalomoncd074912015-12-09 10:33:51 -0800188 }
189
cdalton85285412016-02-18 12:37:07 -0800190 GrGLSLFPFragmentBuilder* fragBuilder = args.fFragBuilder;
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000191 // At each quarter-circle corner we compute a vector that is the offset of the fragment position
192 // from the circle center. The vector is pinned in x and y to be in the quarter-plane relevant
193 // to that corner. This means that points near the interior near the rrect top edge will have
194 // a vector that points straight up for both the TL left and TR corners. Computing an
195 // alpha from this vector at either the TR or TL corner will give the correct result. Similarly,
196 // fragments near the other three edges will get the correct AA. Fragments in the interior of
197 // the rrect will have a (0,0) vector at all four corners. So long as the radius > 0.5 they will
198 // correctly produce an alpha value of 1 at all four corners. We take the min of all the alphas.
199 // The code below is a simplified version of the above that performs maxs on the vector
200 // components before computing distances and alpha values so that only one distance computation
201 // need be computed to determine the min alpha.
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000202 //
203 // For the cases where one half of the rrect is rectangular we drop one of the x or y
204 // computations, compute a separate rect edge alpha for the rect side, and mul the two computed
205 // alphas together.
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000206 switch (crre.getCircularCornerFlags()) {
207 case CircularRRectEffect::kAll_CornerFlags:
John Stiles56bf90b2020-06-10 17:25:58 -0400208 fragBuilder->codeAppendf("float2 dxy0 = %s.LT - sk_FragCoord.xy;", rectName);
209 fragBuilder->codeAppendf("float2 dxy1 = sk_FragCoord.xy - %s.RB;", rectName);
Chris Dalton47c8ed32017-11-15 18:27:09 -0700210 fragBuilder->codeAppend("float2 dxy = max(max(dxy0, dxy1), 0.0);");
Ethan Nicholase1f55022019-02-05 17:17:40 -0500211 fragBuilder->codeAppendf("half alpha = half(%s);", clampedCircleDistance.c_str());
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000212 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000213 case CircularRRectEffect::kTopLeft_CornerFlag:
John Stiles56bf90b2020-06-10 17:25:58 -0400214 fragBuilder->codeAppendf("float2 dxy = max(%s.LT - sk_FragCoord.xy, 0.0);",
Ethan Nicholas38657112017-02-09 17:01:22 -0500215 rectName);
John Stiles56bf90b2020-06-10 17:25:58 -0400216 fragBuilder->codeAppendf("half rightAlpha = half(saturate(%s.R - sk_FragCoord.x));",
Ethan Nicholas38657112017-02-09 17:01:22 -0500217 rectName);
John Stiles56bf90b2020-06-10 17:25:58 -0400218 fragBuilder->codeAppendf("half bottomAlpha = half(saturate(%s.B - sk_FragCoord.y));",
Ethan Nicholas38657112017-02-09 17:01:22 -0500219 rectName);
Ethan Nicholase1f55022019-02-05 17:17:40 -0500220 fragBuilder->codeAppendf("half alpha = bottomAlpha * rightAlpha * half(%s);",
bsalomoncd074912015-12-09 10:33:51 -0800221 clampedCircleDistance.c_str());
commit-bot@chromium.orgc5c748c2014-03-11 15:54:51 +0000222 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000223 case CircularRRectEffect::kTopRight_CornerFlag:
John Stiles56bf90b2020-06-10 17:25:58 -0400224 fragBuilder->codeAppendf("float2 dxy = max(float2(sk_FragCoord.x - %s.R, "
225 "%s.T - sk_FragCoord.y), 0.0);",
Ethan Nicholas38657112017-02-09 17:01:22 -0500226 rectName, rectName);
John Stiles56bf90b2020-06-10 17:25:58 -0400227 fragBuilder->codeAppendf("half leftAlpha = half(saturate(sk_FragCoord.x - %s.L));",
Ethan Nicholas38657112017-02-09 17:01:22 -0500228 rectName);
John Stiles56bf90b2020-06-10 17:25:58 -0400229 fragBuilder->codeAppendf("half bottomAlpha = half(saturate(%s.B - sk_FragCoord.y));",
Ethan Nicholas38657112017-02-09 17:01:22 -0500230 rectName);
Ethan Nicholase1f55022019-02-05 17:17:40 -0500231 fragBuilder->codeAppendf("half alpha = bottomAlpha * leftAlpha * half(%s);",
bsalomoncd074912015-12-09 10:33:51 -0800232 clampedCircleDistance.c_str());
commit-bot@chromium.orgc5c748c2014-03-11 15:54:51 +0000233 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000234 case CircularRRectEffect::kBottomRight_CornerFlag:
John Stiles56bf90b2020-06-10 17:25:58 -0400235 fragBuilder->codeAppendf("float2 dxy = max(sk_FragCoord.xy - %s.RB, 0.0);",
Ethan Nicholas38657112017-02-09 17:01:22 -0500236 rectName);
John Stiles56bf90b2020-06-10 17:25:58 -0400237 fragBuilder->codeAppendf("half leftAlpha = half(saturate(sk_FragCoord.x - %s.L));",
Ethan Nicholas38657112017-02-09 17:01:22 -0500238 rectName);
John Stiles56bf90b2020-06-10 17:25:58 -0400239 fragBuilder->codeAppendf("half topAlpha = half(saturate(sk_FragCoord.y - %s.T));",
Ethan Nicholas38657112017-02-09 17:01:22 -0500240 rectName);
Ethan Nicholase1f55022019-02-05 17:17:40 -0500241 fragBuilder->codeAppendf("half alpha = topAlpha * leftAlpha * half(%s);",
bsalomoncd074912015-12-09 10:33:51 -0800242 clampedCircleDistance.c_str());
commit-bot@chromium.orgc5c748c2014-03-11 15:54:51 +0000243 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000244 case CircularRRectEffect::kBottomLeft_CornerFlag:
John Stiles56bf90b2020-06-10 17:25:58 -0400245 fragBuilder->codeAppendf("float2 dxy = max(float2(%s.L - sk_FragCoord.x, "
246 "sk_FragCoord.y - %s.B), 0.0);",
Ethan Nicholas38657112017-02-09 17:01:22 -0500247 rectName, rectName);
John Stiles56bf90b2020-06-10 17:25:58 -0400248 fragBuilder->codeAppendf("half rightAlpha = half(saturate(%s.R - sk_FragCoord.x));",
Ethan Nicholas38657112017-02-09 17:01:22 -0500249 rectName);
John Stiles56bf90b2020-06-10 17:25:58 -0400250 fragBuilder->codeAppendf("half topAlpha = half(saturate(sk_FragCoord.y - %s.T));",
Ethan Nicholas38657112017-02-09 17:01:22 -0500251 rectName);
Ethan Nicholase1f55022019-02-05 17:17:40 -0500252 fragBuilder->codeAppendf("half alpha = topAlpha * rightAlpha * half(%s);",
bsalomoncd074912015-12-09 10:33:51 -0800253 clampedCircleDistance.c_str());
commit-bot@chromium.orgc5c748c2014-03-11 15:54:51 +0000254 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000255 case CircularRRectEffect::kLeft_CornerFlags:
John Stiles56bf90b2020-06-10 17:25:58 -0400256 fragBuilder->codeAppendf("float2 dxy0 = %s.LT - sk_FragCoord.xy;", rectName);
257 fragBuilder->codeAppendf("float dy1 = sk_FragCoord.y - %s.B;", rectName);
Chris Dalton47c8ed32017-11-15 18:27:09 -0700258 fragBuilder->codeAppend("float2 dxy = max(float2(dxy0.x, max(dxy0.y, dy1)), 0.0);");
John Stiles56bf90b2020-06-10 17:25:58 -0400259 fragBuilder->codeAppendf("half rightAlpha = half(saturate(%s.R - sk_FragCoord.x));",
Ethan Nicholas38657112017-02-09 17:01:22 -0500260 rectName);
Ethan Nicholase1f55022019-02-05 17:17:40 -0500261 fragBuilder->codeAppendf("half alpha = rightAlpha * half(%s);",
bsalomoncd074912015-12-09 10:33:51 -0800262 clampedCircleDistance.c_str());
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000263 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000264 case CircularRRectEffect::kTop_CornerFlags:
John Stiles56bf90b2020-06-10 17:25:58 -0400265 fragBuilder->codeAppendf("float2 dxy0 = %s.LT - sk_FragCoord.xy;", rectName);
266 fragBuilder->codeAppendf("float dx1 = sk_FragCoord.x - %s.R;", rectName);
Chris Dalton47c8ed32017-11-15 18:27:09 -0700267 fragBuilder->codeAppend("float2 dxy = max(float2(max(dxy0.x, dx1), dxy0.y), 0.0);");
John Stiles56bf90b2020-06-10 17:25:58 -0400268 fragBuilder->codeAppendf("half bottomAlpha = half(saturate(%s.B - sk_FragCoord.y));",
Ethan Nicholas38657112017-02-09 17:01:22 -0500269 rectName);
Ethan Nicholase1f55022019-02-05 17:17:40 -0500270 fragBuilder->codeAppendf("half alpha = bottomAlpha * half(%s);",
bsalomoncd074912015-12-09 10:33:51 -0800271 clampedCircleDistance.c_str());
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000272 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000273 case CircularRRectEffect::kRight_CornerFlags:
John Stiles56bf90b2020-06-10 17:25:58 -0400274 fragBuilder->codeAppendf("float dy0 = %s.T - sk_FragCoord.y;", rectName);
275 fragBuilder->codeAppendf("float2 dxy1 = sk_FragCoord.xy - %s.RB;", rectName);
Chris Dalton47c8ed32017-11-15 18:27:09 -0700276 fragBuilder->codeAppend("float2 dxy = max(float2(dxy1.x, max(dy0, dxy1.y)), 0.0);");
John Stiles56bf90b2020-06-10 17:25:58 -0400277 fragBuilder->codeAppendf("half leftAlpha = half(saturate(sk_FragCoord.x - %s.L));",
Ethan Nicholas38657112017-02-09 17:01:22 -0500278 rectName);
Ethan Nicholase1f55022019-02-05 17:17:40 -0500279 fragBuilder->codeAppendf("half alpha = leftAlpha * half(%s);",
bsalomoncd074912015-12-09 10:33:51 -0800280 clampedCircleDistance.c_str());
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000281 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000282 case CircularRRectEffect::kBottom_CornerFlags:
John Stiles56bf90b2020-06-10 17:25:58 -0400283 fragBuilder->codeAppendf("float dx0 = %s.L - sk_FragCoord.x;", rectName);
284 fragBuilder->codeAppendf("float2 dxy1 = sk_FragCoord.xy - %s.RB;", rectName);
Chris Dalton47c8ed32017-11-15 18:27:09 -0700285 fragBuilder->codeAppend("float2 dxy = max(float2(max(dx0, dxy1.x), dxy1.y), 0.0);");
John Stiles56bf90b2020-06-10 17:25:58 -0400286 fragBuilder->codeAppendf("half topAlpha = half(saturate(sk_FragCoord.y - %s.T));",
Ethan Nicholas38657112017-02-09 17:01:22 -0500287 rectName);
Ethan Nicholase1f55022019-02-05 17:17:40 -0500288 fragBuilder->codeAppendf("half alpha = topAlpha * half(%s);",
bsalomoncd074912015-12-09 10:33:51 -0800289 clampedCircleDistance.c_str());
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000290 break;
291 }
skia.committer@gmail.com06acb582014-03-06 03:02:32 +0000292
Ethan Nicholas1706f842017-11-10 11:58:19 -0500293 if (GrClipEdgeType::kInverseFillAA == crre.getEdgeType()) {
bsalomoncd074912015-12-09 10:33:51 -0800294 fragBuilder->codeAppend("alpha = 1.0 - alpha;");
commit-bot@chromium.orgfbde87f2014-03-04 16:25:34 +0000295 }
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000296
Brian Osman6b5dbb42020-07-15 15:31:05 -0400297 SkString inputSample = this->invokeChild(/*childIndex=*/0, args);
John Stiles56bf90b2020-06-10 17:25:58 -0400298
John Stiles4bdc1212020-12-14 18:04:13 -0500299 fragBuilder->codeAppendf("return %s * alpha;", inputSample.c_str());
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000300}
301
Brian Salomon94efbf52016-11-29 13:43:05 -0500302void GLCircularRRectEffect::GenKey(const GrProcessor& processor, const GrShaderCaps&,
joshualittb0a8a372014-09-23 09:50:21 -0700303 GrProcessorKeyBuilder* b) {
304 const CircularRRectEffect& crre = processor.cast<CircularRRectEffect>();
Brian Salomon4dea72a2019-12-18 10:43:10 -0500305 static_assert(kGrClipEdgeTypeCnt <= 8);
Ethan Nicholas1706f842017-11-10 11:58:19 -0500306 b->add32((crre.getCircularCornerFlags() << 3) | (int) crre.getEdgeType());
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000307}
308
egdaniel018fb622015-10-28 07:26:40 -0700309void GLCircularRRectEffect::onSetData(const GrGLSLProgramDataManager& pdman,
Brian Salomonab015ef2017-04-04 10:15:51 -0400310 const GrFragmentProcessor& processor) {
joshualittb0a8a372014-09-23 09:50:21 -0700311 const CircularRRectEffect& crre = processor.cast<CircularRRectEffect>();
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000312 const SkRRect& rrect = crre.getRRect();
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000313 if (rrect != fPrevRRect) {
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000314 SkRect rect = rrect.getBounds();
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000315 SkScalar radius = 0;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000316 switch (crre.getCircularCornerFlags()) {
317 case CircularRRectEffect::kAll_CornerFlags:
Mike Reed242135a2018-02-22 13:41:39 -0500318 SkASSERT(SkRRectPriv::IsSimpleCircular(rrect));
319 radius = SkRRectPriv::GetSimpleRadii(rrect).fX;
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000320 SkASSERT(radius >= kRadiusMin);
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000321 rect.inset(radius, radius);
322 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000323 case CircularRRectEffect::kTopLeft_CornerFlag:
commit-bot@chromium.orgc5c748c2014-03-11 15:54:51 +0000324 radius = rrect.radii(SkRRect::kUpperLeft_Corner).fX;
325 rect.fLeft += radius;
326 rect.fTop += radius;
327 rect.fRight += 0.5f;
328 rect.fBottom += 0.5f;
329 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000330 case CircularRRectEffect::kTopRight_CornerFlag:
commit-bot@chromium.orgc5c748c2014-03-11 15:54:51 +0000331 radius = rrect.radii(SkRRect::kUpperRight_Corner).fX;
bsalomon@google.comde9f2512014-03-11 17:09:17 +0000332 rect.fLeft -= 0.5f;
commit-bot@chromium.orgc5c748c2014-03-11 15:54:51 +0000333 rect.fTop += radius;
334 rect.fRight -= radius;
335 rect.fBottom += 0.5f;
336 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000337 case CircularRRectEffect::kBottomRight_CornerFlag:
commit-bot@chromium.orgc5c748c2014-03-11 15:54:51 +0000338 radius = rrect.radii(SkRRect::kLowerRight_Corner).fX;
bsalomon@google.comde9f2512014-03-11 17:09:17 +0000339 rect.fLeft -= 0.5f;
340 rect.fTop -= 0.5f;
commit-bot@chromium.orgc5c748c2014-03-11 15:54:51 +0000341 rect.fRight -= radius;
342 rect.fBottom -= radius;
343 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000344 case CircularRRectEffect::kBottomLeft_CornerFlag:
commit-bot@chromium.orgc5c748c2014-03-11 15:54:51 +0000345 radius = rrect.radii(SkRRect::kLowerLeft_Corner).fX;
346 rect.fLeft += radius;
bsalomon@google.comde9f2512014-03-11 17:09:17 +0000347 rect.fTop -= 0.5f;
348 rect.fRight += 0.5f;
commit-bot@chromium.orgc5c748c2014-03-11 15:54:51 +0000349 rect.fBottom -= radius;
350 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000351 case CircularRRectEffect::kLeft_CornerFlags:
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000352 radius = rrect.radii(SkRRect::kUpperLeft_Corner).fX;
353 rect.fLeft += radius;
354 rect.fTop += radius;
355 rect.fRight += 0.5f;
356 rect.fBottom -= radius;
357 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000358 case CircularRRectEffect::kTop_CornerFlags:
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000359 radius = rrect.radii(SkRRect::kUpperLeft_Corner).fX;
360 rect.fLeft += radius;
361 rect.fTop += radius;
362 rect.fRight -= radius;
363 rect.fBottom += 0.5f;
364 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000365 case CircularRRectEffect::kRight_CornerFlags:
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000366 radius = rrect.radii(SkRRect::kUpperRight_Corner).fX;
367 rect.fLeft -= 0.5f;
368 rect.fTop += radius;
369 rect.fRight -= radius;
370 rect.fBottom -= radius;
371 break;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000372 case CircularRRectEffect::kBottom_CornerFlags:
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000373 radius = rrect.radii(SkRRect::kLowerLeft_Corner).fX;
374 rect.fLeft += radius;
375 rect.fTop -= 0.5f;
376 rect.fRight -= radius;
377 rect.fBottom -= radius;
378 break;
commit-bot@chromium.orgdd584222014-03-10 16:08:22 +0000379 default:
Ben Wagnerb4aab9a2017-08-16 10:53:04 -0400380 SK_ABORT("Should have been one of the above cases.");
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000381 }
kkinnunen7510b222014-07-30 00:04:16 -0700382 pdman.set4f(fInnerRectUniform, rect.fLeft, rect.fTop, rect.fRight, rect.fBottom);
bsalomoncd074912015-12-09 10:33:51 -0800383 radius += 0.5f;
384 pdman.set2f(fRadiusPlusHalfUniform, radius, 1.f / radius);
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000385 fPrevRRect = rrect;
386 }
387}
388
joshualitteb2a6762014-12-04 11:35:33 -0800389////////////////////////////////////////////////////////////////////////////////////////////////////
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000390
Brian Salomon13b28732021-08-06 15:33:58 -0400391void CircularRRectEffect::onAddToKey(const GrShaderCaps& caps, GrProcessorKeyBuilder* b) const {
joshualitteb2a6762014-12-04 11:35:33 -0800392 GLCircularRRectEffect::GenKey(*this, caps, b);
393}
394
Brian Salomon3176e862021-08-09 11:23:04 -0400395std::unique_ptr<GrFragmentProcessor::ProgramImpl> CircularRRectEffect::onMakeProgramImpl() const {
Brian Salomon18ab2032021-02-23 10:07:05 -0500396 return std::make_unique<GLCircularRRectEffect>();
joshualitteb2a6762014-12-04 11:35:33 -0800397}
398
399//////////////////////////////////////////////////////////////////////////////
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000400
joshualittb0a8a372014-09-23 09:50:21 -0700401class EllipticalRRectEffect : public GrFragmentProcessor {
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000402public:
John Stiles72e57642020-06-24 10:42:35 -0400403 static GrFPResult Make(std::unique_ptr<GrFragmentProcessor>, GrClipEdgeType, const SkRRect&);
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000404
Brian Salomond3b65972017-03-22 12:05:03 -0400405 ~EllipticalRRectEffect() override {}
joshualitteb2a6762014-12-04 11:35:33 -0800406
mtklein36352bf2015-03-25 18:17:31 -0700407 const char* name() const override { return "EllipticalRRect"; }
joshualitteb2a6762014-12-04 11:35:33 -0800408
Brian Salomonaff329b2017-08-11 09:40:37 -0400409 std::unique_ptr<GrFragmentProcessor> clone() const override;
Brian Salomon1a2a7ab2017-07-26 13:11:51 -0400410
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000411 const SkRRect& getRRect() const { return fRRect; }
412
Ethan Nicholas0f3c7322017-11-09 14:51:17 -0500413 GrClipEdgeType getEdgeType() const { return fEdgeType; }
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000414
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000415private:
John Stilesc0d5d9c2020-06-15 15:57:25 -0400416 EllipticalRRectEffect(std::unique_ptr<GrFragmentProcessor>, GrClipEdgeType, const SkRRect&);
417 EllipticalRRectEffect(const EllipticalRRectEffect& that);
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000418
Brian Salomon3176e862021-08-09 11:23:04 -0400419 std::unique_ptr<ProgramImpl> onMakeProgramImpl() const override;
wangyixb1daa862015-08-18 11:29:31 -0700420
Brian Salomon13b28732021-08-06 15:33:58 -0400421 void onAddToKey(const GrShaderCaps&, GrProcessorKeyBuilder*) const override;
wangyix4b3050b2015-08-04 07:59:37 -0700422
mtklein36352bf2015-03-25 18:17:31 -0700423 bool onIsEqual(const GrFragmentProcessor& other) const override;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000424
Brian Salomon587e08f2017-01-27 10:59:27 -0500425 SkRRect fRRect;
Ethan Nicholas0f3c7322017-11-09 14:51:17 -0500426 GrClipEdgeType fEdgeType;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000427
Brian Salomon0c26a9d2017-07-06 10:09:38 -0400428 GR_DECLARE_FRAGMENT_PROCESSOR_TEST
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000429
John Stiles7571f9e2020-09-02 22:42:33 -0400430 using INHERITED = GrFragmentProcessor;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000431};
432
John Stiles72e57642020-06-24 10:42:35 -0400433GrFPResult EllipticalRRectEffect::Make(std::unique_ptr<GrFragmentProcessor> inputFP,
434 GrClipEdgeType edgeType, const SkRRect& rrect) {
Ethan Nicholas1706f842017-11-10 11:58:19 -0500435 if (GrClipEdgeType::kFillAA != edgeType && GrClipEdgeType::kInverseFillAA != edgeType) {
John Stiles72e57642020-06-24 10:42:35 -0400436 return GrFPFailure(std::move(inputFP));
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000437 }
John Stiles72e57642020-06-24 10:42:35 -0400438 return GrFPSuccess(std::unique_ptr<GrFragmentProcessor>(
John Stiles851b90e2020-06-17 13:53:37 -0400439 new EllipticalRRectEffect(std::move(inputFP), edgeType, rrect)));
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000440}
441
John Stilesc0d5d9c2020-06-15 15:57:25 -0400442EllipticalRRectEffect::EllipticalRRectEffect(std::unique_ptr<GrFragmentProcessor> inputFP,
Brian Osman8f391372021-04-29 15:27:34 -0400443 GrClipEdgeType edgeType,
444 const SkRRect& rrect)
445 : INHERITED(kEllipticalRRectEffect_ClassID,
446 ProcessorOptimizationFlags(inputFP.get()) &
447 kCompatibleWithCoverageAsAlpha_OptimizationFlag)
Brian Salomonf3b995b2017-02-15 10:22:23 -0500448 , fRRect(rrect)
449 , fEdgeType(edgeType) {
Brian Osman54867de2020-07-10 14:22:57 -0400450 this->registerChild(std::move(inputFP));
John Stilesc0d5d9c2020-06-15 15:57:25 -0400451}
452
453EllipticalRRectEffect::EllipticalRRectEffect(const EllipticalRRectEffect& that)
John Stiles307f8f52021-08-09 15:36:59 -0400454 : INHERITED(that)
John Stilesc0d5d9c2020-06-15 15:57:25 -0400455 , fRRect(that.fRRect)
John Stiles307f8f52021-08-09 15:36:59 -0400456 , fEdgeType(that.fEdgeType) {}
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000457
Brian Salomonaff329b2017-08-11 09:40:37 -0400458std::unique_ptr<GrFragmentProcessor> EllipticalRRectEffect::clone() const {
John Stilesc0d5d9c2020-06-15 15:57:25 -0400459 return std::unique_ptr<GrFragmentProcessor>(new EllipticalRRectEffect(*this));
Brian Salomon1a2a7ab2017-07-26 13:11:51 -0400460}
461
bsalomon0e08fc12014-10-15 08:19:04 -0700462bool EllipticalRRectEffect::onIsEqual(const GrFragmentProcessor& other) const {
joshualitt49586be2014-09-16 08:21:41 -0700463 const EllipticalRRectEffect& erre = other.cast<EllipticalRRectEffect>();
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000464 return fEdgeType == erre.fEdgeType && fRRect == erre.fRRect;
465}
466
467//////////////////////////////////////////////////////////////////////////////
468
joshualittb0a8a372014-09-23 09:50:21 -0700469GR_DEFINE_FRAGMENT_PROCESSOR_TEST(EllipticalRRectEffect);
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000470
Hal Canary6f6961e2017-01-31 13:50:44 -0500471#if GR_TEST_UTILS
Brian Salomonaff329b2017-08-11 09:40:37 -0400472std::unique_ptr<GrFragmentProcessor> EllipticalRRectEffect::TestCreate(GrProcessorTestData* d) {
joshualitt0067ff52015-07-08 14:26:19 -0700473 SkScalar w = d->fRandom->nextRangeScalar(20.f, 1000.f);
474 SkScalar h = d->fRandom->nextRangeScalar(20.f, 1000.f);
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000475 SkVector r[4];
joshualitt0067ff52015-07-08 14:26:19 -0700476 r[SkRRect::kUpperLeft_Corner].fX = d->fRandom->nextRangeF(kRadiusMin, 9.f);
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000477 // ensure at least one corner really is elliptical
478 do {
joshualitt0067ff52015-07-08 14:26:19 -0700479 r[SkRRect::kUpperLeft_Corner].fY = d->fRandom->nextRangeF(kRadiusMin, 9.f);
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000480 } while (r[SkRRect::kUpperLeft_Corner].fY == r[SkRRect::kUpperLeft_Corner].fX);
481
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000482 SkRRect rrect;
joshualitt0067ff52015-07-08 14:26:19 -0700483 if (d->fRandom->nextBool()) {
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000484 // half the time create a four-radii rrect.
joshualitt0067ff52015-07-08 14:26:19 -0700485 r[SkRRect::kLowerRight_Corner].fX = d->fRandom->nextRangeF(kRadiusMin, 9.f);
486 r[SkRRect::kLowerRight_Corner].fY = d->fRandom->nextRangeF(kRadiusMin, 9.f);
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000487
488 r[SkRRect::kUpperRight_Corner].fX = r[SkRRect::kLowerRight_Corner].fX;
489 r[SkRRect::kUpperRight_Corner].fY = r[SkRRect::kUpperLeft_Corner].fY;
490
491 r[SkRRect::kLowerLeft_Corner].fX = r[SkRRect::kUpperLeft_Corner].fX;
492 r[SkRRect::kLowerLeft_Corner].fY = r[SkRRect::kLowerRight_Corner].fY;
493
494 rrect.setRectRadii(SkRect::MakeWH(w, h), r);
495 } else {
496 rrect.setRectXY(SkRect::MakeWH(w, h), r[SkRRect::kUpperLeft_Corner].fX,
497 r[SkRRect::kUpperLeft_Corner].fY);
498 }
John Stiles6609cb62020-07-17 14:52:12 -0400499 std::unique_ptr<GrFragmentProcessor> fp = d->inputFP();
John Stiles851b90e2020-06-17 13:53:37 -0400500 bool success;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000501 do {
Ethan Nicholas1706f842017-11-10 11:58:19 -0500502 GrClipEdgeType et = (GrClipEdgeType)d->fRandom->nextULessThan(kGrClipEdgeTypeCnt);
John Stiles6609cb62020-07-17 14:52:12 -0400503 std::tie(success, fp) = GrRRectEffect::Make(std::move(fp), et, rrect,
John Stiles851b90e2020-06-17 13:53:37 -0400504 *d->caps()->shaderCaps());
505 } while (!success);
joshualittb0a8a372014-09-23 09:50:21 -0700506 return fp;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000507}
Hal Canary6f6961e2017-01-31 13:50:44 -0500508#endif
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000509
510//////////////////////////////////////////////////////////////////////////////
511
Brian Salomon3176e862021-08-09 11:23:04 -0400512class GLEllipticalRRectEffect : public GrFragmentProcessor::ProgramImpl {
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000513public:
Brian Salomon0a241ce2017-12-15 11:31:05 -0500514 GLEllipticalRRectEffect() = default;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000515
robertphillips9cdb9922016-02-03 12:25:40 -0800516 void emitCode(EmitArgs&) override;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000517
Brian Salomon94efbf52016-11-29 13:43:05 -0500518 static inline void GenKey(const GrProcessor&, const GrShaderCaps&, GrProcessorKeyBuilder*);
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000519
wangyixb1daa862015-08-18 11:29:31 -0700520protected:
Brian Salomonab015ef2017-04-04 10:15:51 -0400521 void onSetData(const GrGLSLProgramDataManager&, const GrFragmentProcessor&) override;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000522
523private:
egdaniel018fb622015-10-28 07:26:40 -0700524 GrGLSLProgramDataManager::UniformHandle fInnerRectUniform;
525 GrGLSLProgramDataManager::UniformHandle fInvRadiiSqdUniform;
bsalomone87256c42015-12-09 17:14:40 -0800526 GrGLSLProgramDataManager::UniformHandle fScaleUniform;
527 SkRRect fPrevRRect;
Brian Salomon3176e862021-08-09 11:23:04 -0400528 using INHERITED = ProgramImpl;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000529};
530
wangyix7c157a92015-07-22 15:08:53 -0700531void GLEllipticalRRectEffect::emitCode(EmitArgs& args) {
532 const EllipticalRRectEffect& erre = args.fFp.cast<EllipticalRRectEffect>();
egdaniel7ea439b2015-12-03 09:20:44 -0800533 GrGLSLUniformHandler* uniformHandler = args.fUniformHandler;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000534 const char *rectName;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000535 // The inner rect is the rrect bounds inset by the x/y radii
Ethan Nicholas16464c32020-04-06 13:53:05 -0400536 fInnerRectUniform = uniformHandler->addUniform(&erre, kFragment_GrShaderFlag, kFloat4_GrSLType,
Ethan Nicholasf7b88202017-09-18 14:10:39 -0400537 "innerRect", &rectName);
joshualitt30ba4362014-08-21 20:18:45 -0700538
cdalton85285412016-02-18 12:37:07 -0800539 GrGLSLFPFragmentBuilder* fragBuilder = args.fFragBuilder;
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000540 // At each quarter-ellipse corner we compute a vector that is the offset of the fragment pos
541 // to the ellipse center. The vector is pinned in x and y to be in the quarter-plane relevant
542 // to that corner. This means that points near the interior near the rrect top edge will have
543 // a vector that points straight up for both the TL left and TR corners. Computing an
544 // alpha from this vector at either the TR or TL corner will give the correct result. Similarly,
545 // fragments near the other three edges will get the correct AA. Fragments in the interior of
546 // the rrect will have a (0,0) vector at all four corners. So long as the radii > 0.5 they will
547 // 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 -0700548 //
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000549 // The code below is a simplified version of the above that performs maxs on the vector
550 // components before computing distances and alpha values so that only one distance computation
551 // need be computed to determine the min alpha.
John Stilesc0d5d9c2020-06-15 15:57:25 -0400552 fragBuilder->codeAppendf("float2 dxy0 = %s.LT - sk_FragCoord.xy;", rectName);
553 fragBuilder->codeAppendf("float2 dxy1 = sk_FragCoord.xy - %s.RB;", rectName);
bsalomone87256c42015-12-09 17:14:40 -0800554
Chris Dalton47c8ed32017-11-15 18:27:09 -0700555 // If we're on a device where float != fp32 then we'll do the distance computation in a space
bsalomone87256c42015-12-09 17:14:40 -0800556 // that is normalized by the largest radius. The scale uniform will be scale, 1/scale. The
557 // radii uniform values are already in this normalized space.
558 const char* scaleName = nullptr;
Chris Dalton47c8ed32017-11-15 18:27:09 -0700559 if (!args.fShaderCaps->floatIs32Bits()) {
Ethan Nicholas16464c32020-04-06 13:53:05 -0400560 fScaleUniform = uniformHandler->addUniform(&erre, kFragment_GrShaderFlag, kHalf2_GrSLType,
561 "scale", &scaleName);
bsalomone87256c42015-12-09 17:14:40 -0800562 }
563
bsalomonc41f4d62015-08-03 14:23:03 -0700564 // The uniforms with the inv squared radii are highp to prevent underflow.
commit-bot@chromium.org9615d5f2014-03-20 21:39:03 +0000565 switch (erre.getRRect().getType()) {
566 case SkRRect::kSimple_Type: {
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000567 const char *invRadiiXYSqdName;
Ethan Nicholas16464c32020-04-06 13:53:05 -0400568 fInvRadiiSqdUniform = uniformHandler->addUniform(&erre,
569 kFragment_GrShaderFlag,
Ethan Nicholas10e93b62019-03-20 10:46:14 -0400570 kFloat2_GrSLType,
cdalton5e58cee2016-02-11 12:49:47 -0800571 "invRadiiXY",
572 &invRadiiXYSqdName);
Chris Dalton47c8ed32017-11-15 18:27:09 -0700573 fragBuilder->codeAppend("float2 dxy = max(max(dxy0, dxy1), 0.0);");
bsalomone87256c42015-12-09 17:14:40 -0800574 if (scaleName) {
575 fragBuilder->codeAppendf("dxy *= %s.y;", scaleName);
576 }
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000577 // Z is the x/y offsets divided by squared radii.
Chris Dalton47c8ed32017-11-15 18:27:09 -0700578 fragBuilder->codeAppendf("float2 Z = dxy * %s.xy;", invRadiiXYSqdName);
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000579 break;
580 }
commit-bot@chromium.org9615d5f2014-03-20 21:39:03 +0000581 case SkRRect::kNinePatch_Type: {
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000582 const char *invRadiiLTRBSqdName;
Ethan Nicholas16464c32020-04-06 13:53:05 -0400583 fInvRadiiSqdUniform = uniformHandler->addUniform(&erre,
584 kFragment_GrShaderFlag,
Ethan Nicholas10e93b62019-03-20 10:46:14 -0400585 kFloat4_GrSLType,
cdalton5e58cee2016-02-11 12:49:47 -0800586 "invRadiiLTRB",
587 &invRadiiLTRBSqdName);
bsalomone87256c42015-12-09 17:14:40 -0800588 if (scaleName) {
589 fragBuilder->codeAppendf("dxy0 *= %s.y;", scaleName);
590 fragBuilder->codeAppendf("dxy1 *= %s.y;", scaleName);
591 }
Chris Dalton47c8ed32017-11-15 18:27:09 -0700592 fragBuilder->codeAppend("float2 dxy = max(max(dxy0, dxy1), 0.0);");
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000593 // Z is the x/y offsets divided by squared radii. We only care about the (at most) one
594 // corner where both the x and y offsets are positive, hence the maxes. (The inverse
595 // squared radii will always be positive.)
Chris Dalton47c8ed32017-11-15 18:27:09 -0700596 fragBuilder->codeAppendf("float2 Z = max(max(dxy0 * %s.xy, dxy1 * %s.zw), 0.0);",
egdaniel4ca2e602015-11-18 08:01:26 -0800597 invRadiiLTRBSqdName, invRadiiLTRBSqdName);
bsalomone87256c42015-12-09 17:14:40 -0800598
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000599 break;
commit-bot@chromium.org9615d5f2014-03-20 21:39:03 +0000600 }
601 default:
Ben Wagnerb4aab9a2017-08-16 10:53:04 -0400602 SK_ABORT("RRect should always be simple or nine-patch.");
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000603 }
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000604 // implicit is the evaluation of (x/a)^2 + (y/b)^2 - 1.
Ethan Nicholase1f55022019-02-05 17:17:40 -0500605 fragBuilder->codeAppend("half implicit = half(dot(Z, dxy) - 1.0);");
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000606 // grad_dot is the squared length of the gradient of the implicit.
Ethan Nicholase1f55022019-02-05 17:17:40 -0500607 fragBuilder->codeAppend("half grad_dot = half(4.0 * dot(Z, Z));");
commit-bot@chromium.org1b035d82014-04-09 17:11:09 +0000608 // avoid calling inversesqrt on zero.
bsalomone87256c42015-12-09 17:14:40 -0800609 fragBuilder->codeAppend("grad_dot = max(grad_dot, 1.0e-4);");
Ethan Nicholase1f55022019-02-05 17:17:40 -0500610 fragBuilder->codeAppend("half approx_dist = implicit * half(inversesqrt(grad_dot));");
bsalomone87256c42015-12-09 17:14:40 -0800611 if (scaleName) {
612 fragBuilder->codeAppendf("approx_dist *= %s.x;", scaleName);
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000613 }
614
Ethan Nicholas1706f842017-11-10 11:58:19 -0500615 if (GrClipEdgeType::kFillAA == erre.getEdgeType()) {
Ethan Nicholasf7b88202017-09-18 14:10:39 -0400616 fragBuilder->codeAppend("half alpha = clamp(0.5 - approx_dist, 0.0, 1.0);");
bsalomone87256c42015-12-09 17:14:40 -0800617 } else {
Ethan Nicholasf7b88202017-09-18 14:10:39 -0400618 fragBuilder->codeAppend("half alpha = clamp(0.5 + approx_dist, 0.0, 1.0);");
bsalomone87256c42015-12-09 17:14:40 -0800619 }
620
Brian Osman6b5dbb42020-07-15 15:31:05 -0400621 SkString inputSample = this->invokeChild(/*childIndex=*/0, args);
John Stilesc0d5d9c2020-06-15 15:57:25 -0400622
John Stiles4bdc1212020-12-14 18:04:13 -0500623 fragBuilder->codeAppendf("return %s * alpha;", inputSample.c_str());
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000624}
625
Brian Salomon94efbf52016-11-29 13:43:05 -0500626void GLEllipticalRRectEffect::GenKey(const GrProcessor& effect, const GrShaderCaps&,
joshualittb0a8a372014-09-23 09:50:21 -0700627 GrProcessorKeyBuilder* b) {
joshualitt49586be2014-09-16 08:21:41 -0700628 const EllipticalRRectEffect& erre = effect.cast<EllipticalRRectEffect>();
Brian Salomon4dea72a2019-12-18 10:43:10 -0500629 static_assert((int)GrClipEdgeType::kLast < (1 << 3));
Ethan Nicholas1706f842017-11-10 11:58:19 -0500630 b->add32(erre.getRRect().getType() | (int) erre.getEdgeType() << 3);
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000631}
632
egdaniel018fb622015-10-28 07:26:40 -0700633void GLEllipticalRRectEffect::onSetData(const GrGLSLProgramDataManager& pdman,
Brian Salomonab015ef2017-04-04 10:15:51 -0400634 const GrFragmentProcessor& effect) {
joshualitt49586be2014-09-16 08:21:41 -0700635 const EllipticalRRectEffect& erre = effect.cast<EllipticalRRectEffect>();
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000636 const SkRRect& rrect = erre.getRRect();
bsalomone87256c42015-12-09 17:14:40 -0800637 // If we're using a scale factor to work around precision issues, choose the largest radius
638 // 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 +0000639 if (rrect != fPrevRRect) {
640 SkRect rect = rrect.getBounds();
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000641 const SkVector& r0 = rrect.radii(SkRRect::kUpperLeft_Corner);
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000642 SkASSERT(r0.fX >= kRadiusMin);
643 SkASSERT(r0.fY >= kRadiusMin);
commit-bot@chromium.org9615d5f2014-03-20 21:39:03 +0000644 switch (erre.getRRect().getType()) {
645 case SkRRect::kSimple_Type:
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000646 rect.inset(r0.fX, r0.fY);
bsalomone87256c42015-12-09 17:14:40 -0800647 if (fScaleUniform.isValid()) {
648 if (r0.fX > r0.fY) {
649 pdman.set2f(fInvRadiiSqdUniform, 1.f, (r0.fX * r0.fX) / (r0.fY * r0.fY));
650 pdman.set2f(fScaleUniform, r0.fX, 1.f / r0.fX);
651 } else {
652 pdman.set2f(fInvRadiiSqdUniform, (r0.fY * r0.fY) / (r0.fX * r0.fX), 1.f);
653 pdman.set2f(fScaleUniform, r0.fY, 1.f / r0.fY);
654 }
655 } else {
656 pdman.set2f(fInvRadiiSqdUniform, 1.f / (r0.fX * r0.fX),
657 1.f / (r0.fY * r0.fY));
658 }
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000659 break;
commit-bot@chromium.org9615d5f2014-03-20 21:39:03 +0000660 case SkRRect::kNinePatch_Type: {
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000661 const SkVector& r1 = rrect.radii(SkRRect::kLowerRight_Corner);
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000662 SkASSERT(r1.fX >= kRadiusMin);
663 SkASSERT(r1.fY >= kRadiusMin);
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000664 rect.fLeft += r0.fX;
665 rect.fTop += r0.fY;
666 rect.fRight -= r1.fX;
667 rect.fBottom -= r1.fY;
bsalomone87256c42015-12-09 17:14:40 -0800668 if (fScaleUniform.isValid()) {
Brian Osman788b9162020-02-07 10:36:46 -0500669 float scale = std::max(std::max(r0.fX, r0.fY), std::max(r1.fX, r1.fY));
bsalomone87256c42015-12-09 17:14:40 -0800670 float scaleSqd = scale * scale;
671 pdman.set4f(fInvRadiiSqdUniform, scaleSqd / (r0.fX * r0.fX),
672 scaleSqd / (r0.fY * r0.fY),
673 scaleSqd / (r1.fX * r1.fX),
674 scaleSqd / (r1.fY * r1.fY));
675 pdman.set2f(fScaleUniform, scale, 1.f / scale);
676 } else {
677 pdman.set4f(fInvRadiiSqdUniform, 1.f / (r0.fX * r0.fX),
678 1.f / (r0.fY * r0.fY),
679 1.f / (r1.fX * r1.fX),
680 1.f / (r1.fY * r1.fY));
681 }
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000682 break;
683 }
commit-bot@chromium.org9615d5f2014-03-20 21:39:03 +0000684 default:
Ben Wagnerb4aab9a2017-08-16 10:53:04 -0400685 SK_ABORT("RRect should always be simple or nine-patch.");
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000686 }
kkinnunen7510b222014-07-30 00:04:16 -0700687 pdman.set4f(fInnerRectUniform, rect.fLeft, rect.fTop, rect.fRight, rect.fBottom);
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000688 fPrevRRect = rrect;
689 }
690}
691
joshualitteb2a6762014-12-04 11:35:33 -0800692////////////////////////////////////////////////////////////////////////////////////////////////////
693
Brian Salomon13b28732021-08-06 15:33:58 -0400694void EllipticalRRectEffect::onAddToKey(const GrShaderCaps& caps, GrProcessorKeyBuilder* b) const {
joshualitteb2a6762014-12-04 11:35:33 -0800695 GLEllipticalRRectEffect::GenKey(*this, caps, b);
696}
697
Brian Salomon3176e862021-08-09 11:23:04 -0400698std::unique_ptr<GrFragmentProcessor::ProgramImpl> EllipticalRRectEffect::onMakeProgramImpl() const {
Brian Salomon18ab2032021-02-23 10:07:05 -0500699 return std::make_unique<GLEllipticalRRectEffect>();
joshualitteb2a6762014-12-04 11:35:33 -0800700}
701
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000702//////////////////////////////////////////////////////////////////////////////
703
John Stiles72e57642020-06-24 10:42:35 -0400704GrFPResult GrRRectEffect::Make(std::unique_ptr<GrFragmentProcessor> inputFP,
705 GrClipEdgeType edgeType, const SkRRect& rrect,
706 const GrShaderCaps& caps) {
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000707 if (rrect.isRect()) {
Brian Osmanb384eb22021-06-24 15:34:08 -0400708 auto fp = GrFragmentProcessor::Rect(std::move(inputFP), edgeType, rrect.getBounds());
Brian Salomon066f4112021-02-23 10:55:22 -0500709 return GrFPSuccess(std::move(fp));
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000710 }
711
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000712 if (rrect.isOval()) {
John Stiles851b90e2020-06-17 13:53:37 -0400713 return GrOvalEffect::Make(std::move(inputFP), edgeType, rrect.getBounds(), caps);
commit-bot@chromium.org3eedb802014-03-28 15:58:31 +0000714 }
715
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000716 if (rrect.isSimple()) {
Mike Reed242135a2018-02-22 13:41:39 -0500717 if (SkRRectPriv::GetSimpleRadii(rrect).fX < kRadiusMin ||
718 SkRRectPriv::GetSimpleRadii(rrect).fY < kRadiusMin) {
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000719 // In this case the corners are extremely close to rectangular and we collapse the
720 // clip to a rectangular clip.
Brian Osmanb384eb22021-06-24 15:34:08 -0400721 auto fp = GrFragmentProcessor::Rect(std::move(inputFP), edgeType, rrect.getBounds());
Brian Salomon066f4112021-02-23 10:55:22 -0500722 return GrFPSuccess(std::move(fp));
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000723 }
Mike Reed242135a2018-02-22 13:41:39 -0500724 if (SkRRectPriv::GetSimpleRadii(rrect).fX == SkRRectPriv::GetSimpleRadii(rrect).fY) {
John Stiles851b90e2020-06-17 13:53:37 -0400725 return CircularRRectEffect::Make(std::move(inputFP), edgeType,
John Stiles56bf90b2020-06-10 17:25:58 -0400726 CircularRRectEffect::kAll_CornerFlags, rrect);
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000727 } else {
John Stiles851b90e2020-06-17 13:53:37 -0400728 return EllipticalRRectEffect::Make(std::move(inputFP), edgeType, rrect);
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000729 }
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000730 }
731
732 if (rrect.isComplex() || rrect.isNinePatch()) {
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000733 // Check for the "tab" cases - two adjacent circular corners and two square corners.
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000734 SkScalar circularRadius = 0;
735 uint32_t cornerFlags = 0;
736
737 SkVector radii[4];
738 bool squashedRadii = false;
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000739 for (int c = 0; c < 4; ++c) {
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000740 radii[c] = rrect.radii((SkRRect::Corner)c);
741 SkASSERT((0 == radii[c].fX) == (0 == radii[c].fY));
742 if (0 == radii[c].fX) {
743 // The corner is square, so no need to squash or flag as circular.
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000744 continue;
745 }
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000746 if (radii[c].fX < kRadiusMin || radii[c].fY < kRadiusMin) {
747 radii[c].set(0, 0);
748 squashedRadii = true;
749 continue;
750 }
751 if (radii[c].fX != radii[c].fY) {
bsalomon@google.com44a435b2014-03-13 19:20:32 +0000752 cornerFlags = ~0U;
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000753 break;
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000754 }
commit-bot@chromium.orgdd584222014-03-10 16:08:22 +0000755 if (!cornerFlags) {
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000756 circularRadius = radii[c].fX;
commit-bot@chromium.orgdd584222014-03-10 16:08:22 +0000757 cornerFlags = 1 << c;
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000758 } else {
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000759 if (radii[c].fX != circularRadius) {
bsalomon@google.com44a435b2014-03-13 19:20:32 +0000760 cornerFlags = ~0U;
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000761 break;
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000762 }
commit-bot@chromium.orgdd584222014-03-10 16:08:22 +0000763 cornerFlags |= 1 << c;
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000764 }
765 }
766
commit-bot@chromium.orgdd584222014-03-10 16:08:22 +0000767 switch (cornerFlags) {
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000768 case CircularRRectEffect::kAll_CornerFlags:
769 // This rrect should have been caught in the simple case above. Though, it would
770 // be correctly handled in the fallthrough code.
771 SkASSERT(false);
John Stiles30212b72020-06-11 17:55:07 -0400772 [[fallthrough]];
commit-bot@chromium.org4355f212014-03-12 15:32:50 +0000773 case CircularRRectEffect::kTopLeft_CornerFlag:
774 case CircularRRectEffect::kTopRight_CornerFlag:
775 case CircularRRectEffect::kBottomRight_CornerFlag:
776 case CircularRRectEffect::kBottomLeft_CornerFlag:
777 case CircularRRectEffect::kLeft_CornerFlags:
778 case CircularRRectEffect::kTop_CornerFlags:
779 case CircularRRectEffect::kRight_CornerFlags:
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000780 case CircularRRectEffect::kBottom_CornerFlags: {
781 SkTCopyOnFirstWrite<SkRRect> rr(rrect);
782 if (squashedRadii) {
783 rr.writable()->setRectRadii(rrect.getBounds(), radii);
784 }
John Stiles851b90e2020-06-17 13:53:37 -0400785 return CircularRRectEffect::Make(std::move(inputFP), edgeType, cornerFlags, *rr);
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000786 }
John Stilesf08a82b2020-06-16 16:34:12 -0400787 case CircularRRectEffect::kNone_CornerFlags: {
Brian Osmanb384eb22021-06-24 15:34:08 -0400788 auto fp =
789 GrFragmentProcessor::Rect(std::move(inputFP), edgeType, rrect.getBounds());
Brian Salomon066f4112021-02-23 10:55:22 -0500790 return GrFPSuccess(std::move(fp));
John Stilesf08a82b2020-06-16 16:34:12 -0400791 }
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000792 default: {
Brian Salomon34653652021-07-26 13:30:06 -0400793 const SkVector ul = rrect.radii(SkRRect::kUpperLeft_Corner);
794 const SkVector lr = rrect.radii(SkRRect::kLowerRight_Corner);
795 if (rrect.isNinePatch() &&
796 ul.fX >= kRadiusMin &&
797 ul.fY >= kRadiusMin &&
798 lr.fX >= kRadiusMin &&
799 lr.fY >= kRadiusMin) {
John Stiles851b90e2020-06-17 13:53:37 -0400800 return EllipticalRRectEffect::Make(std::move(inputFP), edgeType, rrect);
commit-bot@chromium.orgfa5edbe2014-03-13 18:01:05 +0000801 }
John Stiles72e57642020-06-24 10:42:35 -0400802 return GrFPFailure(std::move(inputFP));
commit-bot@chromium.org2a8be902014-03-24 19:24:59 +0000803 }
commit-bot@chromium.orgcb3672e2014-02-21 22:41:56 +0000804 }
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000805 }
John Stiles72e57642020-06-24 10:42:35 -0400806 return GrFPFailure(std::move(inputFP));
commit-bot@chromium.orgc2f78242014-02-19 15:18:05 +0000807}