xref: /aosp_15_r20/external/skia/src/gpu/ganesh/effects/GrCoverageSetOpXP.cpp (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker  * Copyright 2014 Google Inc.
3*c8dee2aaSAndroid Build Coastguard Worker  *
4*c8dee2aaSAndroid Build Coastguard Worker  * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker  * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker  */
7*c8dee2aaSAndroid Build Coastguard Worker 
8*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/ganesh/effects/GrCoverageSetOpXP.h"
9*c8dee2aaSAndroid Build Coastguard Worker 
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/SkColorData.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkAssert.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "src/base/SkRandom.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/Blend.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/KeyBuilder.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/ganesh/GrXferProcessor.h"
16*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/ganesh/glsl/GrGLSLFragmentShaderBuilder.h"
17*c8dee2aaSAndroid Build Coastguard Worker 
18*c8dee2aaSAndroid Build Coastguard Worker #include <memory>
19*c8dee2aaSAndroid Build Coastguard Worker 
20*c8dee2aaSAndroid Build Coastguard Worker enum class GrClampType;
21*c8dee2aaSAndroid Build Coastguard Worker struct GrShaderCaps;
22*c8dee2aaSAndroid Build Coastguard Worker 
23*c8dee2aaSAndroid Build Coastguard Worker class CoverageSetOpXP : public GrXferProcessor {
24*c8dee2aaSAndroid Build Coastguard Worker public:
CoverageSetOpXP(SkRegion::Op regionOp,bool invertCoverage)25*c8dee2aaSAndroid Build Coastguard Worker     CoverageSetOpXP(SkRegion::Op regionOp, bool invertCoverage)
26*c8dee2aaSAndroid Build Coastguard Worker             : INHERITED(kCoverageSetOpXP_ClassID)
27*c8dee2aaSAndroid Build Coastguard Worker             , fRegionOp(regionOp)
28*c8dee2aaSAndroid Build Coastguard Worker             , fInvertCoverage(invertCoverage) {}
29*c8dee2aaSAndroid Build Coastguard Worker 
name() const30*c8dee2aaSAndroid Build Coastguard Worker     const char* name() const override { return "Coverage Set Op"; }
31*c8dee2aaSAndroid Build Coastguard Worker 
32*c8dee2aaSAndroid Build Coastguard Worker     std::unique_ptr<ProgramImpl> makeProgramImpl() const override;
33*c8dee2aaSAndroid Build Coastguard Worker 
34*c8dee2aaSAndroid Build Coastguard Worker private:
35*c8dee2aaSAndroid Build Coastguard Worker     void onAddToKey(const GrShaderCaps&, skgpu::KeyBuilder*) const override;
36*c8dee2aaSAndroid Build Coastguard Worker 
37*c8dee2aaSAndroid Build Coastguard Worker     void onGetBlendInfo(skgpu::BlendInfo* blendInfo) const override;
38*c8dee2aaSAndroid Build Coastguard Worker 
onIsEqual(const GrXferProcessor & xpBase) const39*c8dee2aaSAndroid Build Coastguard Worker     bool onIsEqual(const GrXferProcessor& xpBase) const override {
40*c8dee2aaSAndroid Build Coastguard Worker         const CoverageSetOpXP& xp = xpBase.cast<CoverageSetOpXP>();
41*c8dee2aaSAndroid Build Coastguard Worker         return (fRegionOp == xp.fRegionOp &&
42*c8dee2aaSAndroid Build Coastguard Worker                 fInvertCoverage == xp.fInvertCoverage);
43*c8dee2aaSAndroid Build Coastguard Worker     }
44*c8dee2aaSAndroid Build Coastguard Worker 
45*c8dee2aaSAndroid Build Coastguard Worker     SkRegion::Op fRegionOp;
46*c8dee2aaSAndroid Build Coastguard Worker     bool         fInvertCoverage;
47*c8dee2aaSAndroid Build Coastguard Worker 
48*c8dee2aaSAndroid Build Coastguard Worker     using INHERITED = GrXferProcessor;
49*c8dee2aaSAndroid Build Coastguard Worker };
50*c8dee2aaSAndroid Build Coastguard Worker 
onAddToKey(const GrShaderCaps & caps,skgpu::KeyBuilder * b) const51*c8dee2aaSAndroid Build Coastguard Worker void CoverageSetOpXP::onAddToKey(const GrShaderCaps& caps, skgpu::KeyBuilder* b) const {
52*c8dee2aaSAndroid Build Coastguard Worker     b->addBool(fInvertCoverage, "invert coverage");
53*c8dee2aaSAndroid Build Coastguard Worker }
54*c8dee2aaSAndroid Build Coastguard Worker 
makeProgramImpl() const55*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<GrXferProcessor::ProgramImpl> CoverageSetOpXP::makeProgramImpl() const {
56*c8dee2aaSAndroid Build Coastguard Worker     class Impl : public ProgramImpl {
57*c8dee2aaSAndroid Build Coastguard Worker     private:
58*c8dee2aaSAndroid Build Coastguard Worker         void emitOutputsForBlendState(const EmitArgs& args) override {
59*c8dee2aaSAndroid Build Coastguard Worker             const CoverageSetOpXP& xp = args.fXP.cast<CoverageSetOpXP>();
60*c8dee2aaSAndroid Build Coastguard Worker             GrGLSLXPFragmentBuilder* fb = args.fXPFragBuilder;
61*c8dee2aaSAndroid Build Coastguard Worker             if (xp.fInvertCoverage) {
62*c8dee2aaSAndroid Build Coastguard Worker                 fb->codeAppendf("%s = 1.0 - %s;", args.fOutputPrimary, args.fInputCoverage);
63*c8dee2aaSAndroid Build Coastguard Worker             } else {
64*c8dee2aaSAndroid Build Coastguard Worker                 fb->codeAppendf("%s = %s;", args.fOutputPrimary, args.fInputCoverage);
65*c8dee2aaSAndroid Build Coastguard Worker             }
66*c8dee2aaSAndroid Build Coastguard Worker         }
67*c8dee2aaSAndroid Build Coastguard Worker     };
68*c8dee2aaSAndroid Build Coastguard Worker     return std::make_unique<Impl>();
69*c8dee2aaSAndroid Build Coastguard Worker }
70*c8dee2aaSAndroid Build Coastguard Worker 
onGetBlendInfo(skgpu::BlendInfo * blendInfo) const71*c8dee2aaSAndroid Build Coastguard Worker void CoverageSetOpXP::onGetBlendInfo(skgpu::BlendInfo* blendInfo) const {
72*c8dee2aaSAndroid Build Coastguard Worker     switch (fRegionOp) {
73*c8dee2aaSAndroid Build Coastguard Worker         case SkRegion::kReplace_Op:
74*c8dee2aaSAndroid Build Coastguard Worker             blendInfo->fSrcBlend = skgpu::BlendCoeff::kOne;
75*c8dee2aaSAndroid Build Coastguard Worker             blendInfo->fDstBlend = skgpu::BlendCoeff::kZero;
76*c8dee2aaSAndroid Build Coastguard Worker             break;
77*c8dee2aaSAndroid Build Coastguard Worker         case SkRegion::kIntersect_Op:
78*c8dee2aaSAndroid Build Coastguard Worker             blendInfo->fSrcBlend = skgpu::BlendCoeff::kDC;
79*c8dee2aaSAndroid Build Coastguard Worker             blendInfo->fDstBlend = skgpu::BlendCoeff::kZero;
80*c8dee2aaSAndroid Build Coastguard Worker             break;
81*c8dee2aaSAndroid Build Coastguard Worker         case SkRegion::kUnion_Op:
82*c8dee2aaSAndroid Build Coastguard Worker             blendInfo->fSrcBlend = skgpu::BlendCoeff::kOne;
83*c8dee2aaSAndroid Build Coastguard Worker             blendInfo->fDstBlend = skgpu::BlendCoeff::kISC;
84*c8dee2aaSAndroid Build Coastguard Worker             break;
85*c8dee2aaSAndroid Build Coastguard Worker         case SkRegion::kXOR_Op:
86*c8dee2aaSAndroid Build Coastguard Worker             blendInfo->fSrcBlend = skgpu::BlendCoeff::kIDC;
87*c8dee2aaSAndroid Build Coastguard Worker             blendInfo->fDstBlend = skgpu::BlendCoeff::kISC;
88*c8dee2aaSAndroid Build Coastguard Worker             break;
89*c8dee2aaSAndroid Build Coastguard Worker         case SkRegion::kDifference_Op:
90*c8dee2aaSAndroid Build Coastguard Worker             blendInfo->fSrcBlend = skgpu::BlendCoeff::kZero;
91*c8dee2aaSAndroid Build Coastguard Worker             blendInfo->fDstBlend = skgpu::BlendCoeff::kISC;
92*c8dee2aaSAndroid Build Coastguard Worker             break;
93*c8dee2aaSAndroid Build Coastguard Worker         case SkRegion::kReverseDifference_Op:
94*c8dee2aaSAndroid Build Coastguard Worker             blendInfo->fSrcBlend = skgpu::BlendCoeff::kIDC;
95*c8dee2aaSAndroid Build Coastguard Worker             blendInfo->fDstBlend = skgpu::BlendCoeff::kZero;
96*c8dee2aaSAndroid Build Coastguard Worker             break;
97*c8dee2aaSAndroid Build Coastguard Worker     }
98*c8dee2aaSAndroid Build Coastguard Worker     blendInfo->fBlendConstant = SK_PMColor4fTRANSPARENT;
99*c8dee2aaSAndroid Build Coastguard Worker }
100*c8dee2aaSAndroid Build Coastguard Worker 
101*c8dee2aaSAndroid Build Coastguard Worker ///////////////////////////////////////////////////////////////////////////////
102*c8dee2aaSAndroid Build Coastguard Worker 
GrCoverageSetOpXPFactory(SkRegion::Op regionOp,bool invertCoverage)103*c8dee2aaSAndroid Build Coastguard Worker constexpr GrCoverageSetOpXPFactory::GrCoverageSetOpXPFactory(SkRegion::Op regionOp,
104*c8dee2aaSAndroid Build Coastguard Worker                                                              bool invertCoverage)
105*c8dee2aaSAndroid Build Coastguard Worker         : fRegionOp(regionOp), fInvertCoverage(invertCoverage) {}
106*c8dee2aaSAndroid Build Coastguard Worker 
Get(SkRegion::Op regionOp,bool invertCoverage)107*c8dee2aaSAndroid Build Coastguard Worker const GrXPFactory* GrCoverageSetOpXPFactory::Get(SkRegion::Op regionOp, bool invertCoverage) {
108*c8dee2aaSAndroid Build Coastguard Worker     switch (regionOp) {
109*c8dee2aaSAndroid Build Coastguard Worker         case SkRegion::kReplace_Op: {
110*c8dee2aaSAndroid Build Coastguard Worker             if (invertCoverage) {
111*c8dee2aaSAndroid Build Coastguard Worker                 static constexpr const GrCoverageSetOpXPFactory gReplaceCDXPFI(
112*c8dee2aaSAndroid Build Coastguard Worker                         SkRegion::kReplace_Op, true);
113*c8dee2aaSAndroid Build Coastguard Worker                 return &gReplaceCDXPFI;
114*c8dee2aaSAndroid Build Coastguard Worker             } else {
115*c8dee2aaSAndroid Build Coastguard Worker                 static constexpr const GrCoverageSetOpXPFactory gReplaceCDXPF(SkRegion::kReplace_Op,
116*c8dee2aaSAndroid Build Coastguard Worker                                                                               false);
117*c8dee2aaSAndroid Build Coastguard Worker                 return &gReplaceCDXPF;
118*c8dee2aaSAndroid Build Coastguard Worker             }
119*c8dee2aaSAndroid Build Coastguard Worker         }
120*c8dee2aaSAndroid Build Coastguard Worker         case SkRegion::kIntersect_Op: {
121*c8dee2aaSAndroid Build Coastguard Worker             if (invertCoverage) {
122*c8dee2aaSAndroid Build Coastguard Worker                 static constexpr const GrCoverageSetOpXPFactory gIntersectCDXPFI(
123*c8dee2aaSAndroid Build Coastguard Worker                         SkRegion::kIntersect_Op, true);
124*c8dee2aaSAndroid Build Coastguard Worker                 return &gIntersectCDXPFI;
125*c8dee2aaSAndroid Build Coastguard Worker             } else {
126*c8dee2aaSAndroid Build Coastguard Worker                 static constexpr const GrCoverageSetOpXPFactory gIntersectCDXPF(
127*c8dee2aaSAndroid Build Coastguard Worker                         SkRegion::kIntersect_Op, false);
128*c8dee2aaSAndroid Build Coastguard Worker                 return &gIntersectCDXPF;
129*c8dee2aaSAndroid Build Coastguard Worker             }
130*c8dee2aaSAndroid Build Coastguard Worker         }
131*c8dee2aaSAndroid Build Coastguard Worker         case SkRegion::kUnion_Op: {
132*c8dee2aaSAndroid Build Coastguard Worker             if (invertCoverage) {
133*c8dee2aaSAndroid Build Coastguard Worker                 static constexpr const GrCoverageSetOpXPFactory gUnionCDXPFI(SkRegion::kUnion_Op,
134*c8dee2aaSAndroid Build Coastguard Worker                                                                              true);
135*c8dee2aaSAndroid Build Coastguard Worker                 return &gUnionCDXPFI;
136*c8dee2aaSAndroid Build Coastguard Worker             } else {
137*c8dee2aaSAndroid Build Coastguard Worker                 static constexpr const GrCoverageSetOpXPFactory gUnionCDXPF(SkRegion::kUnion_Op,
138*c8dee2aaSAndroid Build Coastguard Worker                                                                             false);
139*c8dee2aaSAndroid Build Coastguard Worker                 return &gUnionCDXPF;
140*c8dee2aaSAndroid Build Coastguard Worker             }
141*c8dee2aaSAndroid Build Coastguard Worker         }
142*c8dee2aaSAndroid Build Coastguard Worker         case SkRegion::kXOR_Op: {
143*c8dee2aaSAndroid Build Coastguard Worker             if (invertCoverage) {
144*c8dee2aaSAndroid Build Coastguard Worker                 static constexpr const GrCoverageSetOpXPFactory gXORCDXPFI(SkRegion::kXOR_Op, true);
145*c8dee2aaSAndroid Build Coastguard Worker                 return &gXORCDXPFI;
146*c8dee2aaSAndroid Build Coastguard Worker             } else {
147*c8dee2aaSAndroid Build Coastguard Worker                 static constexpr const GrCoverageSetOpXPFactory gXORCDXPF(SkRegion::kXOR_Op, false);
148*c8dee2aaSAndroid Build Coastguard Worker                 return &gXORCDXPF;
149*c8dee2aaSAndroid Build Coastguard Worker             }
150*c8dee2aaSAndroid Build Coastguard Worker         }
151*c8dee2aaSAndroid Build Coastguard Worker         case SkRegion::kDifference_Op: {
152*c8dee2aaSAndroid Build Coastguard Worker             if (invertCoverage) {
153*c8dee2aaSAndroid Build Coastguard Worker                 static constexpr const GrCoverageSetOpXPFactory gDifferenceCDXPFI(
154*c8dee2aaSAndroid Build Coastguard Worker                         SkRegion::kDifference_Op, true);
155*c8dee2aaSAndroid Build Coastguard Worker                 return &gDifferenceCDXPFI;
156*c8dee2aaSAndroid Build Coastguard Worker             } else {
157*c8dee2aaSAndroid Build Coastguard Worker                 static constexpr const GrCoverageSetOpXPFactory gDifferenceCDXPF(
158*c8dee2aaSAndroid Build Coastguard Worker                         SkRegion::kDifference_Op, false);
159*c8dee2aaSAndroid Build Coastguard Worker                 return &gDifferenceCDXPF;
160*c8dee2aaSAndroid Build Coastguard Worker             }
161*c8dee2aaSAndroid Build Coastguard Worker         }
162*c8dee2aaSAndroid Build Coastguard Worker         case SkRegion::kReverseDifference_Op: {
163*c8dee2aaSAndroid Build Coastguard Worker             if (invertCoverage) {
164*c8dee2aaSAndroid Build Coastguard Worker                 static constexpr const GrCoverageSetOpXPFactory gRevDiffCDXPFI(
165*c8dee2aaSAndroid Build Coastguard Worker                         SkRegion::kReverseDifference_Op, true);
166*c8dee2aaSAndroid Build Coastguard Worker                 return &gRevDiffCDXPFI;
167*c8dee2aaSAndroid Build Coastguard Worker             } else {
168*c8dee2aaSAndroid Build Coastguard Worker                 static constexpr const GrCoverageSetOpXPFactory gRevDiffCDXPF(
169*c8dee2aaSAndroid Build Coastguard Worker                         SkRegion::kReverseDifference_Op, false);
170*c8dee2aaSAndroid Build Coastguard Worker                 return &gRevDiffCDXPF;
171*c8dee2aaSAndroid Build Coastguard Worker             }
172*c8dee2aaSAndroid Build Coastguard Worker         }
173*c8dee2aaSAndroid Build Coastguard Worker     }
174*c8dee2aaSAndroid Build Coastguard Worker     SK_ABORT("Unknown region op.");
175*c8dee2aaSAndroid Build Coastguard Worker }
176*c8dee2aaSAndroid Build Coastguard Worker 
makeXferProcessor(const GrProcessorAnalysisColor &,GrProcessorAnalysisCoverage,const GrCaps & caps,GrClampType) const177*c8dee2aaSAndroid Build Coastguard Worker sk_sp<const GrXferProcessor> GrCoverageSetOpXPFactory::makeXferProcessor(
178*c8dee2aaSAndroid Build Coastguard Worker         const GrProcessorAnalysisColor&,
179*c8dee2aaSAndroid Build Coastguard Worker         GrProcessorAnalysisCoverage,
180*c8dee2aaSAndroid Build Coastguard Worker         const GrCaps& caps,
181*c8dee2aaSAndroid Build Coastguard Worker         GrClampType) const {
182*c8dee2aaSAndroid Build Coastguard Worker     return sk_sp<GrXferProcessor>(new CoverageSetOpXP(fRegionOp, fInvertCoverage));
183*c8dee2aaSAndroid Build Coastguard Worker }
184*c8dee2aaSAndroid Build Coastguard Worker 
GR_DEFINE_XP_FACTORY_TEST(GrCoverageSetOpXPFactory) const185*c8dee2aaSAndroid Build Coastguard Worker GR_DEFINE_XP_FACTORY_TEST(GrCoverageSetOpXPFactory)
186*c8dee2aaSAndroid Build Coastguard Worker 
187*c8dee2aaSAndroid Build Coastguard Worker #if defined(GPU_TEST_UTILS)
188*c8dee2aaSAndroid Build Coastguard Worker const GrXPFactory* GrCoverageSetOpXPFactory::TestGet(GrProcessorTestData* d) {
189*c8dee2aaSAndroid Build Coastguard Worker     SkRegion::Op regionOp = SkRegion::Op(d->fRandom->nextULessThan(SkRegion::kLastOp + 1));
190*c8dee2aaSAndroid Build Coastguard Worker     bool invertCoverage = d->fRandom->nextBool();
191*c8dee2aaSAndroid Build Coastguard Worker     return GrCoverageSetOpXPFactory::Get(regionOp, invertCoverage);
192*c8dee2aaSAndroid Build Coastguard Worker }
193*c8dee2aaSAndroid Build Coastguard Worker #endif
194