blob: d956cd31b08d6d5c1fbc701c000f31e0c23ac993 [file] [log] [blame]
egdaniel87509242014-12-17 13:37:13 -08001
2/*
3 * Copyright 2014 Google Inc.
4 *
5 * Use of this source code is governed by a BSD-style license that can be
6 * found in the LICENSE file.
7 */
8
9#include "effects/GrCoverageSetOpXP.h"
10#include "GrColor.h"
11#include "GrDrawTargetCaps.h"
12#include "GrInvariantOutput.h"
13#include "GrProcessor.h"
14#include "GrProcOptInfo.h"
15#include "gl/GrGLXferProcessor.h"
16#include "gl/builders/GrGLFragmentShaderBuilder.h"
17#include "gl/builders/GrGLProgramBuilder.h"
18
19class GrGLCoverageSetOpXP : public GrGLXferProcessor {
20public:
21 GrGLCoverageSetOpXP(const GrProcessor&) {}
22
23 ~GrGLCoverageSetOpXP() SK_OVERRIDE {}
24
25 void emitCode(const EmitArgs& args) SK_OVERRIDE {
26 const GrCoverageSetOpXP& xp = args.fXP.cast<GrCoverageSetOpXP>();
27 GrGLFPFragmentBuilder* fsBuilder = args.fPB->getFragmentShaderBuilder();
28
29 if (xp.invertCoverage()) {
30 fsBuilder->codeAppendf("%s = 1.0 - %s;", args.fOutputPrimary, args.fInputCoverage);
31 } else {
32 fsBuilder->codeAppendf("%s = %s;", args.fOutputPrimary, args.fInputCoverage);
33 }
34 }
35
36 void setData(const GrGLProgramDataManager&, const GrXferProcessor&) SK_OVERRIDE {};
37
38 static void GenKey(const GrProcessor& processor, const GrGLCaps& caps,
39 GrProcessorKeyBuilder* b) {
40 const GrCoverageSetOpXP& xp = processor.cast<GrCoverageSetOpXP>();
41 uint32_t key = xp.invertCoverage() ? 0x0 : 0x1;
42 b->add32(key);
43 };
44
45private:
46 typedef GrGLXferProcessor INHERITED;
47};
48
49///////////////////////////////////////////////////////////////////////////////
50
51GrCoverageSetOpXP::GrCoverageSetOpXP(SkRegion::Op regionOp, bool invertCoverage)
52 : fRegionOp(regionOp)
53 , fInvertCoverage(invertCoverage) {
54 this->initClassID<GrCoverageSetOpXP>();
55}
56
57GrCoverageSetOpXP::~GrCoverageSetOpXP() {
58}
59
60void GrCoverageSetOpXP::getGLProcessorKey(const GrGLCaps& caps, GrProcessorKeyBuilder* b) const {
61 GrGLCoverageSetOpXP::GenKey(*this, caps, b);
62}
63
64GrGLXferProcessor* GrCoverageSetOpXP::createGLInstance() const {
65 return SkNEW_ARGS(GrGLCoverageSetOpXP, (*this));
66}
67
68GrXferProcessor::OptFlags
69GrCoverageSetOpXP::getOptimizations(const GrProcOptInfo& colorPOI,
70 const GrProcOptInfo& coveragePOI,
71 bool colorWriteDisabled,
72 bool doesStencilWrite,
73 GrColor* color,
74 const GrDrawTargetCaps& caps) {
75 // We never look at the color input
76 return GrXferProcessor::kIgnoreColor_OptFlag;
77}
78
79void GrCoverageSetOpXP::getBlendInfo(GrXferProcessor::BlendInfo* blendInfo) const {
80 switch (fRegionOp) {
81 case SkRegion::kReplace_Op:
82 blendInfo->fSrcBlend = kOne_GrBlendCoeff;
83 blendInfo->fDstBlend = kZero_GrBlendCoeff;
84 break;
85 case SkRegion::kIntersect_Op:
86 blendInfo->fSrcBlend = kDC_GrBlendCoeff;
87 blendInfo->fDstBlend = kZero_GrBlendCoeff;
88 break;
89 case SkRegion::kUnion_Op:
90 blendInfo->fSrcBlend = kOne_GrBlendCoeff;
91 blendInfo->fDstBlend = kISC_GrBlendCoeff;
92 break;
93 case SkRegion::kXOR_Op:
94 blendInfo->fSrcBlend = kIDC_GrBlendCoeff;
95 blendInfo->fDstBlend = kISC_GrBlendCoeff;
96 break;
97 case SkRegion::kDifference_Op:
98 blendInfo->fSrcBlend = kZero_GrBlendCoeff;
99 blendInfo->fDstBlend = kISC_GrBlendCoeff;
100 break;
101 case SkRegion::kReverseDifference_Op:
102 blendInfo->fSrcBlend = kIDC_GrBlendCoeff;
103 blendInfo->fDstBlend = kZero_GrBlendCoeff;
104 break;
105 }
106 blendInfo->fBlendConstant = 0;
107}
108
109///////////////////////////////////////////////////////////////////////////////
110
111GrCoverageSetOpXPFactory::GrCoverageSetOpXPFactory(SkRegion::Op regionOp, bool invertCoverage)
112 : fRegionOp(regionOp)
113 , fInvertCoverage(invertCoverage) {
114 this->initClassID<GrCoverageSetOpXPFactory>();
115}
116
117GrXPFactory* GrCoverageSetOpXPFactory::Create(SkRegion::Op regionOp, bool invertCoverage) {
118 switch (regionOp) {
119 case SkRegion::kReplace_Op: {
120 if (invertCoverage) {
121 static GrCoverageSetOpXPFactory gReplaceCDXPFI(regionOp, invertCoverage);
122 return SkRef(&gReplaceCDXPFI);
123 } else {
124 static GrCoverageSetOpXPFactory gReplaceCDXPF(regionOp, invertCoverage);
125 return SkRef(&gReplaceCDXPF);
126 }
127 break;
128 }
129 case SkRegion::kIntersect_Op: {
130 if (invertCoverage) {
131 static GrCoverageSetOpXPFactory gIntersectCDXPFI(regionOp, invertCoverage);
132 return SkRef(&gIntersectCDXPFI);
133 } else {
134 static GrCoverageSetOpXPFactory gIntersectCDXPF(regionOp, invertCoverage);
135 return SkRef(&gIntersectCDXPF);
136 }
137 break;
138 }
139 case SkRegion::kUnion_Op: {
140 if (invertCoverage) {
141 static GrCoverageSetOpXPFactory gUnionCDXPFI(regionOp, invertCoverage);
142 return SkRef(&gUnionCDXPFI);
143 } else {
144 static GrCoverageSetOpXPFactory gUnionCDXPF(regionOp, invertCoverage);
145 return SkRef(&gUnionCDXPF);
146 }
147 break;
148 }
149 case SkRegion::kXOR_Op: {
150 if (invertCoverage) {
151 static GrCoverageSetOpXPFactory gXORCDXPFI(regionOp, invertCoverage);
152 return SkRef(&gXORCDXPFI);
153 } else {
154 static GrCoverageSetOpXPFactory gXORCDXPF(regionOp, invertCoverage);
155 return SkRef(&gXORCDXPF);
156 }
157 break;
158 }
159 case SkRegion::kDifference_Op: {
160 if (invertCoverage) {
161 static GrCoverageSetOpXPFactory gDifferenceCDXPFI(regionOp, invertCoverage);
162 return SkRef(&gDifferenceCDXPFI);
163 } else {
164 static GrCoverageSetOpXPFactory gDifferenceCDXPF(regionOp, invertCoverage);
165 return SkRef(&gDifferenceCDXPF);
166 }
167 break;
168 }
169 case SkRegion::kReverseDifference_Op: {
170 if (invertCoverage) {
171 static GrCoverageSetOpXPFactory gRevDiffCDXPFI(regionOp, invertCoverage);
172 return SkRef(&gRevDiffCDXPFI);
173 } else {
174 static GrCoverageSetOpXPFactory gRevDiffCDXPF(regionOp, invertCoverage);
175 return SkRef(&gRevDiffCDXPF);
176 }
177 break;
178 }
179 default:
180 return NULL;
181 }
182}
183
184GrXferProcessor* GrCoverageSetOpXPFactory::createXferProcessor(const GrProcOptInfo& /* colorPOI*/,
185 const GrProcOptInfo& covPOI) const {
186 return GrCoverageSetOpXP::Create(fRegionOp, fInvertCoverage);
187}
188
egdaniel9e4ecdc2014-12-18 12:44:55 -0800189void GrCoverageSetOpXPFactory::getInvariantOutput(const GrProcOptInfo& colorPOI,
190 const GrProcOptInfo& coveragePOI,
191 bool colorWriteDisabled,
192 GrXPFactory::InvariantOutput* output) const {
193 if (SkRegion::kReplace_Op == fRegionOp) {
194 if (coveragePOI.isSolidWhite()) {
195 output->fBlendedColor = GrColor_WHITE;
196 output->fBlendedColorFlags = kRGBA_GrColorComponentFlags;
egdaniel87509242014-12-17 13:37:13 -0800197 } else {
egdaniel9e4ecdc2014-12-18 12:44:55 -0800198 output->fBlendedColorFlags = 0;
egdaniel87509242014-12-17 13:37:13 -0800199 }
egdaniel9e4ecdc2014-12-18 12:44:55 -0800200
201 if (coveragePOI.readsDst()) {
202 output->fWillBlendWithDst = true;
203 } else {
204 output->fWillBlendWithDst = false;
205 }
206 } else {
207 output->fBlendedColorFlags = 0;
208 output->fWillBlendWithDst = true;
egdaniel87509242014-12-17 13:37:13 -0800209 }
egdaniel87509242014-12-17 13:37:13 -0800210}
211
212GR_DEFINE_XP_FACTORY_TEST(GrCoverageSetOpXPFactory);
213
214GrXPFactory* GrCoverageSetOpXPFactory::TestCreate(SkRandom* random,
215 GrContext*,
216 const GrDrawTargetCaps&,
217 GrTexture*[]) {
218 SkRegion::Op regionOp = SkRegion::Op(random->nextULessThan(SkRegion::kLastOp + 1));
219 bool invertCoverage = random->nextBool();
220 return GrCoverageSetOpXPFactory::Create(regionOp, invertCoverage);
221}
222