blob: c23e1e65ac140abdfe2854bd8bcb08da1d86f1d4 [file] [log] [blame]
Chris Daltonabed2672021-06-17 16:54:28 -06001/*
2 * Copyright 2021 Google LLC.
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
Chris Daltonc3176002021-07-23 15:33:09 -06008#include "src/gpu/effects/GrModulateAtlasCoverageEffect.h"
Chris Daltonabed2672021-06-17 16:54:28 -06009
10#include "src/gpu/GrDynamicAtlas.h"
11#include "src/gpu/effects/GrTextureEffect.h"
12
Chris Daltonc3176002021-07-23 15:33:09 -060013GrModulateAtlasCoverageEffect::GrModulateAtlasCoverageEffect(
14 Flags flags,
15 std::unique_ptr<GrFragmentProcessor> inputFP,
16 GrSurfaceProxyView atlasView,
17 const SkMatrix& devToAtlasMatrix,
18 const SkIRect& devIBounds)
19 : GrFragmentProcessor(kTessellate_GrModulateAtlasCoverageEffect_ClassID,
Chris Daltonabed2672021-06-17 16:54:28 -060020 kCompatibleWithCoverageAsAlpha_OptimizationFlag)
21 , fFlags(flags)
22 , fBounds((fFlags & Flags::kCheckBounds) ? devIBounds : SkIRect{0,0,0,0}) {
23 this->registerChild(std::move(inputFP));
24 this->registerChild(GrTextureEffect::Make(std::move(atlasView), kUnknown_SkAlphaType,
25 devToAtlasMatrix, GrSamplerState::Filter::kNearest),
26 SkSL::SampleUsage::Explicit());
27}
28
Chris Daltonc3176002021-07-23 15:33:09 -060029GrModulateAtlasCoverageEffect::GrModulateAtlasCoverageEffect(
30 const GrModulateAtlasCoverageEffect& that)
31 : GrFragmentProcessor(kTessellate_GrModulateAtlasCoverageEffect_ClassID,
Chris Daltonabed2672021-06-17 16:54:28 -060032 kCompatibleWithCoverageAsAlpha_OptimizationFlag)
33 , fFlags(that.fFlags)
34 , fBounds(that.fBounds) {
35 this->cloneAndRegisterAllChildProcessors(that);
36}
37
Chris Daltonc3176002021-07-23 15:33:09 -060038std::unique_ptr<GrGLSLFragmentProcessor> GrModulateAtlasCoverageEffect::onMakeProgramImpl() const {
Chris Daltonabed2672021-06-17 16:54:28 -060039 class Impl : public GrGLSLFragmentProcessor {
40 void emitCode(EmitArgs& args) override {
Chris Daltonc3176002021-07-23 15:33:09 -060041 auto fp = args.fFp.cast<GrModulateAtlasCoverageEffect>();
Chris Daltonabed2672021-06-17 16:54:28 -060042 auto f = args.fFragBuilder;
43 auto uniHandler = args.fUniformHandler;
44 SkString inputColor = this->invokeChild(0, args);
45 f->codeAppend("half coverage = 0;");
46 if (fp.fFlags & Flags::kCheckBounds) {
47 const char* boundsName;
48 fBoundsUniform = uniHandler->addUniform(&fp, kFragment_GrShaderFlag,
49 kFloat4_GrSLType, "bounds", &boundsName);
50 // Are we inside the path's valid atlas bounds?
51 f->codeAppendf("if (all(greaterThan(sk_FragCoord.xy, %s.xy)) && "
52 "all(lessThan(sk_FragCoord.xy, %s.zw))) ",
53 boundsName, boundsName);
54 }
55 f->codeAppendf("{");
56 SkString atlasCoverage = this->invokeChild(1, args, "sk_FragCoord.xy");
57 f->codeAppendf("coverage = %s.a;", atlasCoverage.c_str());
58 f->codeAppendf("}");
59 const char* coverageMaybeInvertName;
60 fCoverageMaybeInvertUniform = uniHandler->addUniform(&fp, kFragment_GrShaderFlag,
61 kHalf2_GrSLType, "coverageInvert",
62 &coverageMaybeInvertName);
63 // Invert coverage, if needed.
64 f->codeAppendf("coverage = coverage * %s.x + %s.y;",
65 coverageMaybeInvertName, coverageMaybeInvertName);
66 f->codeAppendf("return %s * coverage;", inputColor.c_str());
67 }
68 void onSetData(const GrGLSLProgramDataManager& pdman,
69 const GrFragmentProcessor& processor) override {
Chris Daltonc3176002021-07-23 15:33:09 -060070 auto fp = processor.cast<GrModulateAtlasCoverageEffect>();
Chris Daltonabed2672021-06-17 16:54:28 -060071 if (fp.fFlags & Flags::kCheckBounds) {
72 pdman.set4fv(fBoundsUniform, 1, SkRect::Make(fp.fBounds).asScalars());
73 }
74 if (fp.fFlags & Flags::kInvertCoverage) {
75 pdman.set2f(fCoverageMaybeInvertUniform, -1, 1); // -1*coverage + 1 = 1 - coverage.
76 } else {
77 pdman.set2f(fCoverageMaybeInvertUniform, 1, 0); // 1*coverage + 0 = coverage.
78 }
79 }
80 UniformHandle fBoundsUniform;
81 UniformHandle fCoverageMaybeInvertUniform;
82 };
83 return std::make_unique<Impl>();
84}