blob: 79b3c57e2e4ae625fdd44be38339212e11f84dcb [file] [log] [blame]
Ethan Nicholas297d6ef2017-12-20 12:00:11 -05001/*
Ethan Nicholas130fb3f2018-02-01 12:14:34 -05002 * Copyright 2018 Google Inc.
Ethan Nicholas297d6ef2017-12-20 12:00:11 -05003 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
Ethan Nicholas130fb3f2018-02-01 12:14:34 -05008/**************************************************************************************************
9 *** This file was autogenerated from GrRRectBlurEffect.fp; do not modify.
10 **************************************************************************************************/
Ethan Nicholas297d6ef2017-12-20 12:00:11 -050011#include "GrRRectBlurEffect.h"
Ethan Nicholas297d6ef2017-12-20 12:00:11 -050012
John Stiles4ca88842020-06-08 17:20:01 -040013std::unique_ptr<GrFragmentProcessor> GrRRectBlurEffect::Make(
14 std::unique_ptr<GrFragmentProcessor> inputFP,
15 GrRecordingContext* context,
16 float sigma,
17 float xformedSigma,
18 const SkRRect& srcRRect,
19 const SkRRect& devRRect) {
Mike Reed242135a2018-02-22 13:41:39 -050020 SkASSERT(!SkRRectPriv::IsCircle(devRRect) &&
21 !devRRect.isRect()); // Should've been caught up-stream
Ethan Nicholas297d6ef2017-12-20 12:00:11 -050022
23 // TODO: loosen this up
Mike Reed242135a2018-02-22 13:41:39 -050024 if (!SkRRectPriv::IsSimpleCircular(devRRect)) {
Ethan Nicholas297d6ef2017-12-20 12:00:11 -050025 return nullptr;
26 }
27
28 // Make sure we can successfully ninepatch this rrect -- the blur sigma has to be
29 // sufficiently small relative to both the size of the corner radius and the
30 // width (and height) of the rrect.
Mike Kleind6ab77a2019-03-21 08:18:24 -050031 SkRRect rrectToDraw;
Brian Salomon9f2b86c2019-10-22 10:37:46 -040032 SkISize dimensions;
Mike Reed28d47732018-03-05 16:56:52 -050033 SkScalar ignored[kSkBlurRRectMaxDivisions];
Mike Kleind6ab77a2019-03-21 08:18:24 -050034 int ignoredSize;
Ethan Nicholas297d6ef2017-12-20 12:00:11 -050035 uint32_t ignored32;
36
Mike Kleind6ab77a2019-03-21 08:18:24 -050037 bool ninePatchable = SkComputeBlurredRRectParams(
Brian Salomon9f2b86c2019-10-22 10:37:46 -040038 srcRRect, devRRect, SkRect::MakeEmpty(), sigma, xformedSigma, &rrectToDraw, &dimensions,
Mike Kleind6ab77a2019-03-21 08:18:24 -050039 ignored, ignored, ignored, ignored, &ignoredSize, &ignoredSize, &ignored32);
Ethan Nicholas297d6ef2017-12-20 12:00:11 -050040 if (!ninePatchable) {
41 return nullptr;
42 }
43
Brian Salomon71f6cfd2020-06-17 17:14:12 -040044 std::unique_ptr<GrFragmentProcessor> maskFP =
45 find_or_create_rrect_blur_mask_fp(context, rrectToDraw, dimensions, xformedSigma);
46 if (!maskFP) {
Ethan Nicholas297d6ef2017-12-20 12:00:11 -050047 return nullptr;
48 }
49
Mike Reed242135a2018-02-22 13:41:39 -050050 return std::unique_ptr<GrFragmentProcessor>(
John Stiles4ca88842020-06-08 17:20:01 -040051 new GrRRectBlurEffect(std::move(inputFP), xformedSigma, devRRect.getBounds(),
Brian Salomon71f6cfd2020-06-17 17:14:12 -040052 SkRRectPriv::GetSimpleRadii(devRRect).fX, std::move(maskFP)));
Ethan Nicholas297d6ef2017-12-20 12:00:11 -050053}
Greg Daniel456f9b52020-03-05 19:14:18 +000054#include "src/gpu/GrTexture.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050055#include "src/gpu/glsl/GrGLSLFragmentProcessor.h"
56#include "src/gpu/glsl/GrGLSLFragmentShaderBuilder.h"
57#include "src/gpu/glsl/GrGLSLProgramBuilder.h"
58#include "src/sksl/SkSLCPP.h"
59#include "src/sksl/SkSLUtil.h"
Ethan Nicholas297d6ef2017-12-20 12:00:11 -050060class GrGLSLRRectBlurEffect : public GrGLSLFragmentProcessor {
61public:
62 GrGLSLRRectBlurEffect() {}
63 void emitCode(EmitArgs& args) override {
64 GrGLSLFPFragmentBuilder* fragBuilder = args.fFragBuilder;
Mike Kleind6ab77a2019-03-21 08:18:24 -050065 const GrRRectBlurEffect& _outer = args.fFp.cast<GrRRectBlurEffect>();
Ethan Nicholas297d6ef2017-12-20 12:00:11 -050066 (void)_outer;
Ethan Nicholasbcd51e82019-04-09 10:40:41 -040067 auto sigma = _outer.sigma;
Ethan Nicholas297d6ef2017-12-20 12:00:11 -050068 (void)sigma;
Ethan Nicholasbcd51e82019-04-09 10:40:41 -040069 auto rect = _outer.rect;
Ethan Nicholas297d6ef2017-12-20 12:00:11 -050070 (void)rect;
Ethan Nicholasbcd51e82019-04-09 10:40:41 -040071 auto cornerRadius = _outer.cornerRadius;
Ethan Nicholas297d6ef2017-12-20 12:00:11 -050072 (void)cornerRadius;
Ethan Nicholas16464c32020-04-06 13:53:05 -040073 cornerRadiusVar = args.fUniformHandler->addUniform(&_outer, kFragment_GrShaderFlag,
74 kHalf_GrSLType, "cornerRadius");
75 proxyRectVar = args.fUniformHandler->addUniform(&_outer, kFragment_GrShaderFlag,
76 kFloat4_GrSLType, "proxyRect");
77 blurRadiusVar = args.fUniformHandler->addUniform(&_outer, kFragment_GrShaderFlag,
78 kHalf_GrSLType, "blurRadius");
Ethan Nicholas297d6ef2017-12-20 12:00:11 -050079 fragBuilder->codeAppendf(
John Stiles50819422020-06-18 13:00:38 -040080 R"SkSL(half2 translatedFragPos = half2(sk_FragCoord.xy - %s.xy);
81half2 proxyCenter = half2((%s.zw - %s.xy) * 0.5);
82half edgeSize = (2.0 * %s + %s) + 0.5;
83translatedFragPos -= proxyCenter;
84half2 fragDirection = sign(translatedFragPos);
85translatedFragPos = abs(translatedFragPos);
86translatedFragPos -= proxyCenter - edgeSize;
87translatedFragPos = max(translatedFragPos, 0.0);
88translatedFragPos *= fragDirection;
89translatedFragPos += half2(edgeSize);
90half2 proxyDims = half2(2.0 * edgeSize);
91half2 texCoord = translatedFragPos / proxyDims;)SkSL",
Ethan Nicholasbcd51e82019-04-09 10:40:41 -040092 args.fUniformHandler->getUniformCStr(proxyRectVar),
John Stilesa2d46a12020-06-11 14:12:44 -040093 args.fUniformHandler->getUniformCStr(proxyRectVar),
94 args.fUniformHandler->getUniformCStr(proxyRectVar),
Ethan Nicholasbcd51e82019-04-09 10:40:41 -040095 args.fUniformHandler->getUniformCStr(blurRadiusVar),
John Stilesa2d46a12020-06-11 14:12:44 -040096 args.fUniformHandler->getUniformCStr(cornerRadiusVar));
John Stiles6609cb62020-07-17 14:52:12 -040097 SkString _sample9554 = this->invokeChild(0, args);
John Stiles50819422020-06-18 13:00:38 -040098 fragBuilder->codeAppendf(
99 R"SkSL(
100half4 inputColor = %s;)SkSL",
John Stiles6609cb62020-07-17 14:52:12 -0400101 _sample9554.c_str());
102 SkString _coords9602("float2(texCoord)");
103 SkString _sample9602 = this->invokeChild(1, args, _coords9602.c_str());
John Stiles50819422020-06-18 13:00:38 -0400104 fragBuilder->codeAppendf(
105 R"SkSL(
106%s = inputColor * %s;
107)SkSL",
John Stiles6609cb62020-07-17 14:52:12 -0400108 args.fOutputColor, _sample9602.c_str());
Ethan Nicholas297d6ef2017-12-20 12:00:11 -0500109 }
110
111private:
112 void onSetData(const GrGLSLProgramDataManager& pdman,
Mike Kleind6ab77a2019-03-21 08:18:24 -0500113 const GrFragmentProcessor& _proc) override {
Ethan Nicholas297d6ef2017-12-20 12:00:11 -0500114 const GrRRectBlurEffect& _outer = _proc.cast<GrRRectBlurEffect>();
Ethan Nicholasbcd51e82019-04-09 10:40:41 -0400115 { pdman.set1f(cornerRadiusVar, (_outer.cornerRadius)); }
116 auto sigma = _outer.sigma;
Ethan Nicholas297d6ef2017-12-20 12:00:11 -0500117 (void)sigma;
Ethan Nicholasbcd51e82019-04-09 10:40:41 -0400118 auto rect = _outer.rect;
Ethan Nicholas297d6ef2017-12-20 12:00:11 -0500119 (void)rect;
Ethan Nicholasbcd51e82019-04-09 10:40:41 -0400120 UniformHandle& cornerRadius = cornerRadiusVar;
Ethan Nicholas297d6ef2017-12-20 12:00:11 -0500121 (void)cornerRadius;
Ethan Nicholasbcd51e82019-04-09 10:40:41 -0400122 UniformHandle& proxyRect = proxyRectVar;
Ethan Nicholas297d6ef2017-12-20 12:00:11 -0500123 (void)proxyRect;
Ethan Nicholasbcd51e82019-04-09 10:40:41 -0400124 UniformHandle& blurRadius = blurRadiusVar;
Ethan Nicholas297d6ef2017-12-20 12:00:11 -0500125 (void)blurRadius;
126
127 float blurRadiusValue = 3.f * SkScalarCeilToScalar(sigma - 1 / 6.0f);
128 pdman.set1f(blurRadius, blurRadiusValue);
129
130 SkRect outset = rect;
131 outset.outset(blurRadiusValue, blurRadiusValue);
132 pdman.set4f(proxyRect, outset.fLeft, outset.fTop, outset.fRight, outset.fBottom);
133 }
Ethan Nicholasbcd51e82019-04-09 10:40:41 -0400134 UniformHandle proxyRectVar;
135 UniformHandle blurRadiusVar;
136 UniformHandle cornerRadiusVar;
Ethan Nicholas297d6ef2017-12-20 12:00:11 -0500137};
138GrGLSLFragmentProcessor* GrRRectBlurEffect::onCreateGLSLInstance() const {
139 return new GrGLSLRRectBlurEffect();
140}
Mike Kleind6ab77a2019-03-21 08:18:24 -0500141void GrRRectBlurEffect::onGetGLSLProcessorKey(const GrShaderCaps& caps,
Ethan Nicholas297d6ef2017-12-20 12:00:11 -0500142 GrProcessorKeyBuilder* b) const {}
143bool GrRRectBlurEffect::onIsEqual(const GrFragmentProcessor& other) const {
144 const GrRRectBlurEffect& that = other.cast<GrRRectBlurEffect>();
145 (void)that;
Ethan Nicholasbcd51e82019-04-09 10:40:41 -0400146 if (sigma != that.sigma) return false;
147 if (rect != that.rect) return false;
148 if (cornerRadius != that.cornerRadius) return false;
Ethan Nicholas297d6ef2017-12-20 12:00:11 -0500149 return true;
150}
151GrRRectBlurEffect::GrRRectBlurEffect(const GrRRectBlurEffect& src)
152 : INHERITED(kGrRRectBlurEffect_ClassID, src.optimizationFlags())
Ethan Nicholasbcd51e82019-04-09 10:40:41 -0400153 , sigma(src.sigma)
154 , rect(src.rect)
Brian Salomon71f6cfd2020-06-17 17:14:12 -0400155 , cornerRadius(src.cornerRadius) {
Brian Osman12c5d292020-07-13 16:11:35 -0400156 this->cloneAndRegisterAllChildProcessors(src);
Ethan Nicholas297d6ef2017-12-20 12:00:11 -0500157}
158std::unique_ptr<GrFragmentProcessor> GrRRectBlurEffect::clone() const {
159 return std::unique_ptr<GrFragmentProcessor>(new GrRRectBlurEffect(*this));
160}
161GR_DEFINE_FRAGMENT_PROCESSOR_TEST(GrRRectBlurEffect);
162#if GR_TEST_UTILS
163std::unique_ptr<GrFragmentProcessor> GrRRectBlurEffect::TestCreate(GrProcessorTestData* d) {
Mike Kleind6ab77a2019-03-21 08:18:24 -0500164 SkScalar w = d->fRandom->nextRangeScalar(100.f, 1000.f);
165 SkScalar h = d->fRandom->nextRangeScalar(100.f, 1000.f);
166 SkScalar r = d->fRandom->nextRangeF(1.f, 9.f);
Ethan Nicholas297d6ef2017-12-20 12:00:11 -0500167 SkScalar sigma = d->fRandom->nextRangeF(1.f, 10.f);
Mike Kleind6ab77a2019-03-21 08:18:24 -0500168 SkRRect rrect;
Ethan Nicholas297d6ef2017-12-20 12:00:11 -0500169 rrect.setRectXY(SkRect::MakeWH(w, h), r, r);
John Stiles6609cb62020-07-17 14:52:12 -0400170 return GrRRectBlurEffect::Make(d->inputFP(), d->context(), sigma, sigma, rrect, rrect);
Ethan Nicholas297d6ef2017-12-20 12:00:11 -0500171}
172#endif