1 /*
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 
8 #include "effects/GrPorterDuffXferProcessor.h"
9 
10 #include "GrBlend.h"
11 #include "GrCaps.h"
12 #include "GrPipeline.h"
13 #include "GrProcessor.h"
14 #include "GrProcessorAnalysis.h"
15 #include "GrTypes.h"
16 #include "GrXferProcessor.h"
17 #include "SkTo.h"
18 #include "glsl/GrGLSLBlend.h"
19 #include "glsl/GrGLSLFragmentShaderBuilder.h"
20 #include "glsl/GrGLSLProgramDataManager.h"
21 #include "glsl/GrGLSLUniformHandler.h"
22 #include "glsl/GrGLSLXferProcessor.h"
23 
24 /**
25  * Wraps the shader outputs and HW blend state that comprise a Porter Duff blend mode with coverage.
26  */
27 class BlendFormula {
28 public:
29     /**
30      * Values the shader can write to primary and secondary outputs. These must all be modulated by
31      * coverage to support mixed samples. The XP will ignore the multiplies when not using coverage.
32      */
33     enum OutputType {
34         kNone_OutputType,        //<! 0
35         kCoverage_OutputType,    //<! inputCoverage
36         kModulate_OutputType,    //<! inputColor * inputCoverage
37         kSAModulate_OutputType,  //<! inputColor.a * inputCoverage
38         kISAModulate_OutputType, //<! (1 - inputColor.a) * inputCoverage
39         kISCModulate_OutputType, //<! (1 - inputColor) * inputCoverage
40 
41         kLast_OutputType = kISCModulate_OutputType
42     };
43 
44     BlendFormula() = default;
45 
BlendFormula(OutputType primaryOut,OutputType secondaryOut,GrBlendEquation equation,GrBlendCoeff srcCoeff,GrBlendCoeff dstCoeff)46     constexpr BlendFormula(OutputType primaryOut, OutputType secondaryOut, GrBlendEquation equation,
47                            GrBlendCoeff srcCoeff, GrBlendCoeff dstCoeff)
48             : fPrimaryOutputType(primaryOut)
49             , fSecondaryOutputType(secondaryOut)
50             , fBlendEquation(equation)
51             , fSrcCoeff(srcCoeff)
52             , fDstCoeff(dstCoeff)
53             , fProps(GetProperties(primaryOut, secondaryOut, equation, srcCoeff, dstCoeff)) {}
54 
operator =(const BlendFormula & other)55     BlendFormula& operator=(const BlendFormula& other) {
56         SkDEBUGCODE(other.validatePreoptimized());
57         fData = other.fData;
58         return *this;
59     }
60 
operator ==(const BlendFormula & other) const61     bool operator==(const BlendFormula& other) const {
62         SkDEBUGCODE(this->validatePreoptimized());
63         SkDEBUGCODE(other.validatePreoptimized());
64         return fData == other.fData;
65     }
66 
hasSecondaryOutput() const67     bool hasSecondaryOutput() const {
68         SkDEBUGCODE(this->validatePreoptimized());
69         return kNone_OutputType != fSecondaryOutputType;
70     }
modifiesDst() const71     bool modifiesDst() const {
72         SkDEBUGCODE(this->validatePreoptimized());
73         return SkToBool(fProps & kModifiesDst_Property);
74     }
usesDstColor() const75     bool usesDstColor() const {
76         SkDEBUGCODE(this->validatePreoptimized());
77         return SkToBool(fProps & kUsesDstColor_Property);
78     }
usesInputColor() const79     bool usesInputColor() const {
80         SkDEBUGCODE(this->validatePreoptimized());
81         return SkToBool(fProps & kUsesInputColor_Property);
82     }
canTweakAlphaForCoverage() const83     bool canTweakAlphaForCoverage() const {
84         SkDEBUGCODE(this->validatePreoptimized());
85         return SkToBool(fProps & kCanTweakAlphaForCoverage_Property);
86     }
87 
equation() const88     GrBlendEquation equation() const {
89         SkDEBUGCODE(this->validatePreoptimized());
90         return fBlendEquation;
91     }
92 
srcCoeff() const93     GrBlendCoeff srcCoeff() const {
94         SkDEBUGCODE(this->validatePreoptimized());
95         return fSrcCoeff;
96     }
97 
dstCoeff() const98     GrBlendCoeff dstCoeff() const {
99         SkDEBUGCODE(this->validatePreoptimized());
100         return fDstCoeff;
101     }
102 
primaryOutput() const103     OutputType primaryOutput() const {
104         SkDEBUGCODE(this->validatePreoptimized());
105         return fPrimaryOutputType;
106     }
107 
secondaryOutput() const108     OutputType secondaryOutput() const {
109         SkDEBUGCODE(this->validatePreoptimized());
110         return fSecondaryOutputType;
111     }
112 
113 private:
114     enum Properties {
115         kModifiesDst_Property              = 1,
116         kUsesDstColor_Property             = 1 << 1,
117         kUsesInputColor_Property           = 1 << 2,
118         kCanTweakAlphaForCoverage_Property = 1 << 3,
119 
120         kLast_Property = kCanTweakAlphaForCoverage_Property
121     };
GR_DECL_BITFIELD_OPS_FRIENDS(Properties)122     GR_DECL_BITFIELD_OPS_FRIENDS(Properties)
123 
124 #ifdef SK_DEBUG
125     void validatePreoptimized() const {
126         // The provided formula should already be optimized before a BlendFormula is constructed.
127         // Preferably these asserts would be done statically in the constexpr constructor, but this
128         // is not allowed in C++11.
129         SkASSERT((kNone_OutputType == fPrimaryOutputType) ==
130                  !GrBlendCoeffsUseSrcColor(fSrcCoeff, fDstCoeff));
131         SkASSERT(!GrBlendCoeffRefsSrc2(fSrcCoeff));
132         SkASSERT((kNone_OutputType == fSecondaryOutputType) == !GrBlendCoeffRefsSrc2(fDstCoeff));
133         SkASSERT(fPrimaryOutputType != fSecondaryOutputType ||
134                  kNone_OutputType == fPrimaryOutputType);
135         SkASSERT(kNone_OutputType != fPrimaryOutputType ||
136                  kNone_OutputType == fSecondaryOutputType);
137     }
138 #endif
139 
140     /**
141      * Deduce the properties of a BlendFormula.
142      */
143     static constexpr Properties GetProperties(OutputType PrimaryOut, OutputType SecondaryOut,
144                                               GrBlendEquation BlendEquation, GrBlendCoeff SrcCoeff,
145                                               GrBlendCoeff DstCoeff);
146 
147     union {
148         struct {
149             // We allot the enums one more bit than they require because MSVC seems to sign-extend
150             // them when the top bit is set. (This is in violation of the C++03 standard 9.6/4)
151             OutputType        fPrimaryOutputType    : 4;
152             OutputType        fSecondaryOutputType  : 4;
153             GrBlendEquation   fBlendEquation        : 6;
154             GrBlendCoeff      fSrcCoeff             : 6;
155             GrBlendCoeff      fDstCoeff             : 6;
156             Properties        fProps                : 32 - (4 + 4 + 6 + 6 + 6);
157         };
158         uint32_t fData;
159     };
160 
161     GR_STATIC_ASSERT(kLast_OutputType      < (1 << 3));
162     GR_STATIC_ASSERT(kLast_GrBlendEquation < (1 << 5));
163     GR_STATIC_ASSERT(kLast_GrBlendCoeff    < (1 << 5));
164     GR_STATIC_ASSERT(kLast_Property        < (1 << 6));
165 };
166 
167 GR_STATIC_ASSERT(4 == sizeof(BlendFormula));
168 
169 GR_MAKE_BITFIELD_OPS(BlendFormula::Properties);
170 
GetProperties(OutputType PrimaryOut,OutputType SecondaryOut,GrBlendEquation BlendEquation,GrBlendCoeff SrcCoeff,GrBlendCoeff DstCoeff)171 constexpr BlendFormula::Properties BlendFormula::GetProperties(OutputType PrimaryOut,
172                                                                OutputType SecondaryOut,
173                                                                GrBlendEquation BlendEquation,
174                                                                GrBlendCoeff SrcCoeff,
175                                                                GrBlendCoeff DstCoeff) {
176     return static_cast<Properties>(
177             (GrBlendModifiesDst(BlendEquation, SrcCoeff, DstCoeff) ? kModifiesDst_Property : 0) |
178             (GrBlendCoeffsUseDstColor(SrcCoeff, DstCoeff) ? kUsesDstColor_Property : 0) |
179             ((PrimaryOut >= kModulate_OutputType && GrBlendCoeffsUseSrcColor(SrcCoeff, DstCoeff)) ||
180                              (SecondaryOut >= kModulate_OutputType &&
181                               GrBlendCoeffRefsSrc2(DstCoeff))
182                      ? kUsesInputColor_Property
183                      : 0) |  // We assert later that SrcCoeff doesn't ref src2.
184             ((kModulate_OutputType == PrimaryOut || kNone_OutputType == PrimaryOut) &&
185                              kNone_OutputType == SecondaryOut &&
186                              GrBlendAllowsCoverageAsAlpha(BlendEquation, SrcCoeff, DstCoeff)
187                      ? kCanTweakAlphaForCoverage_Property
188                      : 0));
189 }
190 
191 /**
192  * When there is no coverage, or the blend mode can tweak alpha for coverage, we use the standard
193  * Porter Duff formula.
194  */
MakeCoeffFormula(GrBlendCoeff srcCoeff,GrBlendCoeff dstCoeff)195 static constexpr BlendFormula MakeCoeffFormula(GrBlendCoeff srcCoeff, GrBlendCoeff dstCoeff) {
196     // When the coeffs are (Zero, Zero) or (Zero, One) we set the primary output to none.
197     return (kZero_GrBlendCoeff == srcCoeff &&
198             (kZero_GrBlendCoeff == dstCoeff || kOne_GrBlendCoeff == dstCoeff))
199            ? BlendFormula(BlendFormula::kNone_OutputType, BlendFormula::kNone_OutputType,
200                           kAdd_GrBlendEquation, kZero_GrBlendCoeff, dstCoeff)
201            : BlendFormula(BlendFormula::kModulate_OutputType, BlendFormula::kNone_OutputType,
202                         kAdd_GrBlendEquation, srcCoeff, dstCoeff);
203 }
204 
205 /**
206  * Basic coeff formula similar to MakeCoeffFormula but we will make the src f*Sa. This is used in
207  * LCD dst-out.
208  */
MakeSAModulateFormula(GrBlendCoeff srcCoeff,GrBlendCoeff dstCoeff)209 static constexpr BlendFormula MakeSAModulateFormula(GrBlendCoeff srcCoeff, GrBlendCoeff dstCoeff) {
210     return BlendFormula(BlendFormula::kSAModulate_OutputType, BlendFormula::kNone_OutputType,
211                         kAdd_GrBlendEquation, srcCoeff, dstCoeff);
212 }
213 
214 /**
215  * When there is coverage, the equation with f=coverage is:
216  *
217  *   D' = f * (S * srcCoeff + D * dstCoeff) + (1-f) * D
218  *
219  * This can be rewritten as:
220  *
221  *   D' = f * S * srcCoeff + D * (1 - [f * (1 - dstCoeff)])
222  *
223  * To implement this formula, we output [f * (1 - dstCoeff)] for the secondary color and replace the
224  * HW dst coeff with IS2C.
225  *
226  * Xfer modes: dst-atop (Sa!=1)
227  */
MakeCoverageFormula(BlendFormula::OutputType oneMinusDstCoeffModulateOutput,GrBlendCoeff srcCoeff)228 static constexpr BlendFormula MakeCoverageFormula(
229         BlendFormula::OutputType oneMinusDstCoeffModulateOutput, GrBlendCoeff srcCoeff) {
230     return BlendFormula(BlendFormula::kModulate_OutputType, oneMinusDstCoeffModulateOutput,
231                         kAdd_GrBlendEquation, srcCoeff, kIS2C_GrBlendCoeff);
232 }
233 
234 /**
235  * When there is coverage and the src coeff is Zero, the equation with f=coverage becomes:
236  *
237  *   D' = f * D * dstCoeff + (1-f) * D
238  *
239  * This can be rewritten as:
240  *
241  *   D' = D - D * [f * (1 - dstCoeff)]
242  *
243  * To implement this formula, we output [f * (1 - dstCoeff)] for the primary color and use a reverse
244  * subtract HW blend equation with coeffs of (DC, One).
245  *
246  * Xfer modes: clear, dst-out (Sa=1), dst-in (Sa!=1), modulate (Sc!=1)
247  */
MakeCoverageSrcCoeffZeroFormula(BlendFormula::OutputType oneMinusDstCoeffModulateOutput)248 static constexpr BlendFormula MakeCoverageSrcCoeffZeroFormula(
249         BlendFormula::OutputType oneMinusDstCoeffModulateOutput) {
250     return BlendFormula(oneMinusDstCoeffModulateOutput, BlendFormula::kNone_OutputType,
251                         kReverseSubtract_GrBlendEquation, kDC_GrBlendCoeff, kOne_GrBlendCoeff);
252 }
253 
254 /**
255  * When there is coverage and the dst coeff is Zero, the equation with f=coverage becomes:
256  *
257  *   D' = f * S * srcCoeff + (1-f) * D
258  *
259  * To implement this formula, we output [f] for the secondary color and replace the HW dst coeff
260  * with IS2A. (Note that we can avoid dual source blending when Sa=1 by using ISA.)
261  *
262  * Xfer modes (Sa!=1): src, src-in, src-out
263  */
MakeCoverageDstCoeffZeroFormula(GrBlendCoeff srcCoeff)264 static constexpr BlendFormula MakeCoverageDstCoeffZeroFormula(GrBlendCoeff srcCoeff) {
265     return BlendFormula(BlendFormula::kModulate_OutputType, BlendFormula::kCoverage_OutputType,
266                         kAdd_GrBlendEquation, srcCoeff, kIS2A_GrBlendCoeff);
267 }
268 
269 // Older GCC won't like the constexpr arrays because of
270 // https://gcc.gnu.org/bugzilla/show_bug.cgi?id=61484.
271 // MSVC 2015 crashes with an internal compiler error.
272 #if !defined(__clang__) && ((defined(__GNUC__) && __GNUC__ < 5) || (defined(_MSC_VER) && _MSC_VER <= 1910))
273 #   define MAYBE_CONSTEXPR const
274 #else
275 #   define MAYBE_CONSTEXPR constexpr
276 #endif
277 
278 /**
279  * This table outlines the blend formulas we will use with each xfermode, with and without coverage,
280  * with and without an opaque input color. Optimization properties are deduced at compile time so we
281  * can make runtime decisions quickly. RGB coverage is not supported.
282  */
283 static MAYBE_CONSTEXPR BlendFormula gBlendTable[2][2][(int)SkBlendMode::kLastCoeffMode + 1] = {
284                      /*>> No coverage, input color unknown <<*/ {{
285 
286     /* clear */      MakeCoeffFormula(kZero_GrBlendCoeff, kZero_GrBlendCoeff),
287     /* src */        MakeCoeffFormula(kOne_GrBlendCoeff,  kZero_GrBlendCoeff),
288     /* dst */        MakeCoeffFormula(kZero_GrBlendCoeff, kOne_GrBlendCoeff),
289     /* src-over */   MakeCoeffFormula(kOne_GrBlendCoeff,  kISA_GrBlendCoeff),
290     /* dst-over */   MakeCoeffFormula(kIDA_GrBlendCoeff,  kOne_GrBlendCoeff),
291     /* src-in */     MakeCoeffFormula(kDA_GrBlendCoeff,   kZero_GrBlendCoeff),
292     /* dst-in */     MakeCoeffFormula(kZero_GrBlendCoeff, kSA_GrBlendCoeff),
293     /* src-out */    MakeCoeffFormula(kIDA_GrBlendCoeff,  kZero_GrBlendCoeff),
294     /* dst-out */    MakeCoeffFormula(kZero_GrBlendCoeff, kISA_GrBlendCoeff),
295     /* src-atop */   MakeCoeffFormula(kDA_GrBlendCoeff,   kISA_GrBlendCoeff),
296     /* dst-atop */   MakeCoeffFormula(kIDA_GrBlendCoeff,  kSA_GrBlendCoeff),
297     /* xor */        MakeCoeffFormula(kIDA_GrBlendCoeff,  kISA_GrBlendCoeff),
298     /* plus */       MakeCoeffFormula(kOne_GrBlendCoeff,  kOne_GrBlendCoeff),
299     /* modulate */   MakeCoeffFormula(kZero_GrBlendCoeff, kSC_GrBlendCoeff),
300     /* screen */     MakeCoeffFormula(kOne_GrBlendCoeff,  kISC_GrBlendCoeff),
301 
302                      }, /*>> Has coverage, input color unknown <<*/ {
303 
304     /* clear */      MakeCoverageSrcCoeffZeroFormula(BlendFormula::kCoverage_OutputType),
305     /* src */        MakeCoverageDstCoeffZeroFormula(kOne_GrBlendCoeff),
306     /* dst */        MakeCoeffFormula(kZero_GrBlendCoeff, kOne_GrBlendCoeff),
307     /* src-over */   MakeCoeffFormula(kOne_GrBlendCoeff,  kISA_GrBlendCoeff),
308     /* dst-over */   MakeCoeffFormula(kIDA_GrBlendCoeff,  kOne_GrBlendCoeff),
309     /* src-in */     MakeCoverageDstCoeffZeroFormula(kDA_GrBlendCoeff),
310     /* dst-in */     MakeCoverageSrcCoeffZeroFormula(BlendFormula::kISAModulate_OutputType),
311     /* src-out */    MakeCoverageDstCoeffZeroFormula(kIDA_GrBlendCoeff),
312     /* dst-out */    MakeCoeffFormula(kZero_GrBlendCoeff, kISA_GrBlendCoeff),
313     /* src-atop */   MakeCoeffFormula(kDA_GrBlendCoeff,   kISA_GrBlendCoeff),
314     /* dst-atop */   MakeCoverageFormula(BlendFormula::kISAModulate_OutputType, kIDA_GrBlendCoeff),
315     /* xor */        MakeCoeffFormula(kIDA_GrBlendCoeff,  kISA_GrBlendCoeff),
316     /* plus */       MakeCoeffFormula(kOne_GrBlendCoeff,  kOne_GrBlendCoeff),
317     /* modulate */   MakeCoverageSrcCoeffZeroFormula(BlendFormula::kISCModulate_OutputType),
318     /* screen */     MakeCoeffFormula(kOne_GrBlendCoeff,  kISC_GrBlendCoeff),
319 
320                      }}, /*>> No coverage, input color opaque <<*/ {{
321 
322     /* clear */      MakeCoeffFormula(kZero_GrBlendCoeff, kZero_GrBlendCoeff),
323     /* src */        MakeCoeffFormula(kOne_GrBlendCoeff,  kZero_GrBlendCoeff),
324     /* dst */        MakeCoeffFormula(kZero_GrBlendCoeff, kOne_GrBlendCoeff),
325     /* src-over */   MakeCoeffFormula(kOne_GrBlendCoeff,  kISA_GrBlendCoeff), // see comment below
326     /* dst-over */   MakeCoeffFormula(kIDA_GrBlendCoeff,  kOne_GrBlendCoeff),
327     /* src-in */     MakeCoeffFormula(kDA_GrBlendCoeff,   kZero_GrBlendCoeff),
328     /* dst-in */     MakeCoeffFormula(kZero_GrBlendCoeff, kOne_GrBlendCoeff),
329     /* src-out */    MakeCoeffFormula(kIDA_GrBlendCoeff,  kZero_GrBlendCoeff),
330     /* dst-out */    MakeCoeffFormula(kZero_GrBlendCoeff, kZero_GrBlendCoeff),
331     /* src-atop */   MakeCoeffFormula(kDA_GrBlendCoeff,   kZero_GrBlendCoeff),
332     /* dst-atop */   MakeCoeffFormula(kIDA_GrBlendCoeff,  kOne_GrBlendCoeff),
333     /* xor */        MakeCoeffFormula(kIDA_GrBlendCoeff,  kZero_GrBlendCoeff),
334     /* plus */       MakeCoeffFormula(kOne_GrBlendCoeff,  kOne_GrBlendCoeff),
335     /* modulate */   MakeCoeffFormula(kZero_GrBlendCoeff, kSC_GrBlendCoeff),
336     /* screen */     MakeCoeffFormula(kOne_GrBlendCoeff,  kISC_GrBlendCoeff),
337 
338                      }, /*>> Has coverage, input color opaque <<*/ {
339 
340     /* clear */      MakeCoverageSrcCoeffZeroFormula(BlendFormula::kCoverage_OutputType),
341     /* src */        MakeCoeffFormula(kOne_GrBlendCoeff,  kISA_GrBlendCoeff),
342     /* dst */        MakeCoeffFormula(kZero_GrBlendCoeff, kOne_GrBlendCoeff),
343     /* src-over */   MakeCoeffFormula(kOne_GrBlendCoeff,  kISA_GrBlendCoeff),
344     /* dst-over */   MakeCoeffFormula(kIDA_GrBlendCoeff,  kOne_GrBlendCoeff),
345     /* src-in */     MakeCoeffFormula(kDA_GrBlendCoeff,   kISA_GrBlendCoeff),
346     /* dst-in */     MakeCoeffFormula(kZero_GrBlendCoeff, kOne_GrBlendCoeff),
347     /* src-out */    MakeCoeffFormula(kIDA_GrBlendCoeff,  kISA_GrBlendCoeff),
348     /* dst-out */    MakeCoverageSrcCoeffZeroFormula(BlendFormula::kCoverage_OutputType),
349     /* src-atop */   MakeCoeffFormula(kDA_GrBlendCoeff,   kISA_GrBlendCoeff),
350     /* dst-atop */   MakeCoeffFormula(kIDA_GrBlendCoeff,  kOne_GrBlendCoeff),
351     /* xor */        MakeCoeffFormula(kIDA_GrBlendCoeff,  kISA_GrBlendCoeff),
352     /* plus */       MakeCoeffFormula(kOne_GrBlendCoeff,  kOne_GrBlendCoeff),
353     /* modulate */   MakeCoverageSrcCoeffZeroFormula(BlendFormula::kISCModulate_OutputType),
354     /* screen */     MakeCoeffFormula(kOne_GrBlendCoeff,  kISC_GrBlendCoeff),
355 }}};
356 // In the above table src-over is not optimized to src mode when the color is opaque because we
357 // found no advantage to doing so. Also, we are using a global src-over XP in most cases which is
358 // not specialized for opaque input. If the table were set to use the src formula then we'd have to
359 // change when we use this global XP to keep analysis and practice in sync.
360 
361 static MAYBE_CONSTEXPR BlendFormula gLCDBlendTable[(int)SkBlendMode::kLastCoeffMode + 1] = {
362     /* clear */      MakeCoverageSrcCoeffZeroFormula(BlendFormula::kCoverage_OutputType),
363     /* src */        MakeCoverageFormula(BlendFormula::kCoverage_OutputType, kOne_GrBlendCoeff),
364     /* dst */        MakeCoeffFormula(kZero_GrBlendCoeff, kOne_GrBlendCoeff),
365     /* src-over */   MakeCoverageFormula(BlendFormula::kSAModulate_OutputType, kOne_GrBlendCoeff),
366     /* dst-over */   MakeCoeffFormula(kIDA_GrBlendCoeff, kOne_GrBlendCoeff),
367     /* src-in */     MakeCoverageFormula(BlendFormula::kCoverage_OutputType, kDA_GrBlendCoeff),
368     /* dst-in */     MakeCoverageSrcCoeffZeroFormula(BlendFormula::kISAModulate_OutputType),
369     /* src-out */    MakeCoverageFormula(BlendFormula::kCoverage_OutputType, kIDA_GrBlendCoeff),
370     /* dst-out */    MakeSAModulateFormula(kZero_GrBlendCoeff, kISC_GrBlendCoeff),
371     /* src-atop */   MakeCoverageFormula(BlendFormula::kSAModulate_OutputType, kDA_GrBlendCoeff),
372     /* dst-atop */   MakeCoverageFormula(BlendFormula::kISAModulate_OutputType, kIDA_GrBlendCoeff),
373     /* xor */        MakeCoverageFormula(BlendFormula::kSAModulate_OutputType, kIDA_GrBlendCoeff),
374     /* plus */       MakeCoeffFormula(kOne_GrBlendCoeff, kOne_GrBlendCoeff),
375     /* modulate */   MakeCoverageSrcCoeffZeroFormula(BlendFormula::kISCModulate_OutputType),
376     /* screen */     MakeCoeffFormula(kOne_GrBlendCoeff, kISC_GrBlendCoeff),
377 };
378 
379 #undef MAYBE_CONSTEXPR
380 
get_blend_formula(bool isOpaque,bool hasCoverage,bool hasMixedSamples,SkBlendMode xfermode)381 static BlendFormula get_blend_formula(bool isOpaque,
382                                       bool hasCoverage,
383                                       bool hasMixedSamples,
384                                       SkBlendMode xfermode) {
385     SkASSERT((unsigned)xfermode <= (unsigned)SkBlendMode::kLastCoeffMode);
386     bool conflatesCoverage = hasCoverage || hasMixedSamples;
387     return gBlendTable[isOpaque][conflatesCoverage][(int)xfermode];
388 }
389 
get_lcd_blend_formula(SkBlendMode xfermode)390 static BlendFormula get_lcd_blend_formula(SkBlendMode xfermode) {
391     SkASSERT((unsigned)xfermode <= (unsigned)SkBlendMode::kLastCoeffMode);
392 
393     return gLCDBlendTable[(int)xfermode];
394 }
395 
396 ///////////////////////////////////////////////////////////////////////////////
397 
398 class PorterDuffXferProcessor : public GrXferProcessor {
399 public:
PorterDuffXferProcessor(BlendFormula blendFormula,GrProcessorAnalysisCoverage coverage)400     PorterDuffXferProcessor(BlendFormula blendFormula, GrProcessorAnalysisCoverage coverage)
401             : INHERITED(kPorterDuffXferProcessor_ClassID, false, false, coverage)
402             , fBlendFormula(blendFormula) {
403     }
404 
name() const405     const char* name() const override { return "Porter Duff"; }
406 
407     GrGLSLXferProcessor* createGLSLInstance() const override;
408 
getBlendFormula() const409     BlendFormula getBlendFormula() const { return fBlendFormula; }
410 
411 private:
412     void onGetGLSLProcessorKey(const GrShaderCaps& caps, GrProcessorKeyBuilder* b) const override;
413 
onHasSecondaryOutput() const414     bool onHasSecondaryOutput() const override { return fBlendFormula.hasSecondaryOutput(); }
415 
onGetBlendInfo(GrXferProcessor::BlendInfo * blendInfo) const416     void onGetBlendInfo(GrXferProcessor::BlendInfo* blendInfo) const override {
417         blendInfo->fEquation = fBlendFormula.equation();
418         blendInfo->fSrcBlend = fBlendFormula.srcCoeff();
419         blendInfo->fDstBlend = fBlendFormula.dstCoeff();
420         blendInfo->fWriteColor = fBlendFormula.modifiesDst();
421     }
422 
onIsEqual(const GrXferProcessor & xpBase) const423     bool onIsEqual(const GrXferProcessor& xpBase) const override {
424         const PorterDuffXferProcessor& xp = xpBase.cast<PorterDuffXferProcessor>();
425         return fBlendFormula == xp.fBlendFormula;
426     }
427 
428     const BlendFormula fBlendFormula;
429 
430     typedef GrXferProcessor INHERITED;
431 };
432 
433 ///////////////////////////////////////////////////////////////////////////////
434 
append_color_output(const PorterDuffXferProcessor & xp,GrGLSLXPFragmentBuilder * fragBuilder,BlendFormula::OutputType outputType,const char * output,const char * inColor,const char * inCoverage)435 static void append_color_output(const PorterDuffXferProcessor& xp,
436                                 GrGLSLXPFragmentBuilder* fragBuilder,
437                                 BlendFormula::OutputType outputType, const char* output,
438                                 const char* inColor, const char* inCoverage) {
439     SkASSERT(inCoverage);
440     SkASSERT(inColor);
441     switch (outputType) {
442         case BlendFormula::kNone_OutputType:
443             fragBuilder->codeAppendf("%s = half4(0.0);", output);
444             break;
445         case BlendFormula::kCoverage_OutputType:
446             // We can have a coverage formula while not reading coverage if there are mixed samples.
447             fragBuilder->codeAppendf("%s = %s;", output, inCoverage);
448             break;
449         case BlendFormula::kModulate_OutputType:
450             fragBuilder->codeAppendf("%s = %s * %s;", output, inColor, inCoverage);
451             break;
452         case BlendFormula::kSAModulate_OutputType:
453             fragBuilder->codeAppendf("%s = %s.a * %s;", output, inColor, inCoverage);
454             break;
455         case BlendFormula::kISAModulate_OutputType:
456             fragBuilder->codeAppendf("%s = (1.0 - %s.a) * %s;", output, inColor, inCoverage);
457             break;
458         case BlendFormula::kISCModulate_OutputType:
459             fragBuilder->codeAppendf("%s = (half4(1.0) - %s) * %s;", output, inColor, inCoverage);
460             break;
461         default:
462             SK_ABORT("Unsupported output type.");
463             break;
464     }
465 }
466 
467 class GLPorterDuffXferProcessor : public GrGLSLXferProcessor {
468 public:
GenKey(const GrProcessor & processor,GrProcessorKeyBuilder * b)469     static void GenKey(const GrProcessor& processor, GrProcessorKeyBuilder* b) {
470         const PorterDuffXferProcessor& xp = processor.cast<PorterDuffXferProcessor>();
471         b->add32(xp.getBlendFormula().primaryOutput() |
472                  (xp.getBlendFormula().secondaryOutput() << 3));
473         GR_STATIC_ASSERT(BlendFormula::kLast_OutputType < 8);
474     }
475 
476 private:
emitOutputsForBlendState(const EmitArgs & args)477     void emitOutputsForBlendState(const EmitArgs& args) override {
478         const PorterDuffXferProcessor& xp = args.fXP.cast<PorterDuffXferProcessor>();
479         GrGLSLXPFragmentBuilder* fragBuilder = args.fXPFragBuilder;
480 
481         BlendFormula blendFormula = xp.getBlendFormula();
482         if (blendFormula.hasSecondaryOutput()) {
483             append_color_output(xp, fragBuilder, blendFormula.secondaryOutput(),
484                                 args.fOutputSecondary, args.fInputColor, args.fInputCoverage);
485         }
486         append_color_output(xp, fragBuilder, blendFormula.primaryOutput(), args.fOutputPrimary,
487                             args.fInputColor, args.fInputCoverage);
488     }
489 
onSetData(const GrGLSLProgramDataManager &,const GrXferProcessor &)490     void onSetData(const GrGLSLProgramDataManager&, const GrXferProcessor&) override {}
491 
492     typedef GrGLSLXferProcessor INHERITED;
493 };
494 
495 ///////////////////////////////////////////////////////////////////////////////
496 
onGetGLSLProcessorKey(const GrShaderCaps &,GrProcessorKeyBuilder * b) const497 void PorterDuffXferProcessor::onGetGLSLProcessorKey(const GrShaderCaps&,
498                                                     GrProcessorKeyBuilder* b) const {
499     GLPorterDuffXferProcessor::GenKey(*this, b);
500 }
501 
createGLSLInstance() const502 GrGLSLXferProcessor* PorterDuffXferProcessor::createGLSLInstance() const {
503     return new GLPorterDuffXferProcessor;
504 }
505 
506 ///////////////////////////////////////////////////////////////////////////////
507 
508 class ShaderPDXferProcessor : public GrXferProcessor {
509 public:
ShaderPDXferProcessor(bool hasMixedSamples,SkBlendMode xfermode,GrProcessorAnalysisCoverage coverage)510     ShaderPDXferProcessor(bool hasMixedSamples, SkBlendMode xfermode,
511                           GrProcessorAnalysisCoverage coverage)
512             : INHERITED(kShaderPDXferProcessor_ClassID, true, hasMixedSamples, coverage)
513             , fXfermode(xfermode) {
514     }
515 
name() const516     const char* name() const override { return "Porter Duff Shader"; }
517 
518     GrGLSLXferProcessor* createGLSLInstance() const override;
519 
getXfermode() const520     SkBlendMode getXfermode() const { return fXfermode; }
521 
522 private:
523     void onGetGLSLProcessorKey(const GrShaderCaps& caps, GrProcessorKeyBuilder* b) const override;
524 
onIsEqual(const GrXferProcessor & xpBase) const525     bool onIsEqual(const GrXferProcessor& xpBase) const override {
526         const ShaderPDXferProcessor& xp = xpBase.cast<ShaderPDXferProcessor>();
527         return fXfermode == xp.fXfermode;
528     }
529 
530     const SkBlendMode fXfermode;
531 
532     typedef GrXferProcessor INHERITED;
533 };
534 
535 ///////////////////////////////////////////////////////////////////////////////
536 
537 class GLShaderPDXferProcessor : public GrGLSLXferProcessor {
538 public:
GenKey(const GrProcessor & processor,GrProcessorKeyBuilder * b)539     static void GenKey(const GrProcessor& processor, GrProcessorKeyBuilder* b) {
540         const ShaderPDXferProcessor& xp = processor.cast<ShaderPDXferProcessor>();
541         b->add32((int)xp.getXfermode());
542     }
543 
544 private:
emitBlendCodeForDstRead(GrGLSLXPFragmentBuilder * fragBuilder,GrGLSLUniformHandler * uniformHandler,const char * srcColor,const char * srcCoverage,const char * dstColor,const char * outColor,const char * outColorSecondary,const GrXferProcessor & proc)545     void emitBlendCodeForDstRead(GrGLSLXPFragmentBuilder* fragBuilder,
546                                  GrGLSLUniformHandler* uniformHandler,
547                                  const char* srcColor,
548                                  const char* srcCoverage,
549                                  const char* dstColor,
550                                  const char* outColor,
551                                  const char* outColorSecondary,
552                                  const GrXferProcessor& proc) override {
553         const ShaderPDXferProcessor& xp = proc.cast<ShaderPDXferProcessor>();
554 
555         GrGLSLBlend::AppendMode(fragBuilder, srcColor, dstColor, outColor, xp.getXfermode());
556 
557         // Apply coverage.
558         INHERITED::DefaultCoverageModulation(fragBuilder, srcCoverage, dstColor, outColor,
559                                              outColorSecondary, xp);
560     }
561 
onSetData(const GrGLSLProgramDataManager &,const GrXferProcessor &)562     void onSetData(const GrGLSLProgramDataManager&, const GrXferProcessor&) override {}
563 
564     typedef GrGLSLXferProcessor INHERITED;
565 };
566 
567 ///////////////////////////////////////////////////////////////////////////////
568 
onGetGLSLProcessorKey(const GrShaderCaps &,GrProcessorKeyBuilder * b) const569 void ShaderPDXferProcessor::onGetGLSLProcessorKey(const GrShaderCaps&,
570                                                   GrProcessorKeyBuilder* b) const {
571     GLShaderPDXferProcessor::GenKey(*this, b);
572 }
573 
createGLSLInstance() const574 GrGLSLXferProcessor* ShaderPDXferProcessor::createGLSLInstance() const {
575     return new GLShaderPDXferProcessor;
576 }
577 
578 ///////////////////////////////////////////////////////////////////////////////
579 
580 class PDLCDXferProcessor : public GrXferProcessor {
581 public:
582     static sk_sp<const GrXferProcessor> Make(SkBlendMode mode,
583                                              const GrProcessorAnalysisColor& inputColor);
584 
585     ~PDLCDXferProcessor() override;
586 
name() const587     const char* name() const override { return "Porter Duff LCD"; }
588 
589     GrGLSLXferProcessor* createGLSLInstance() const override;
590 
alpha() const591     float alpha() const { return fAlpha; }
592 
593 private:
594     PDLCDXferProcessor(const SkPMColor4f& blendConstant, float alpha);
595 
596     void onGetGLSLProcessorKey(const GrShaderCaps& caps, GrProcessorKeyBuilder* b) const override;
597 
onGetBlendInfo(GrXferProcessor::BlendInfo * blendInfo) const598     void onGetBlendInfo(GrXferProcessor::BlendInfo* blendInfo) const override {
599         blendInfo->fSrcBlend = kConstC_GrBlendCoeff;
600         blendInfo->fDstBlend = kISC_GrBlendCoeff;
601         blendInfo->fBlendConstant = fBlendConstant;
602     }
603 
onIsEqual(const GrXferProcessor & xpBase) const604     bool onIsEqual(const GrXferProcessor& xpBase) const override {
605         const PDLCDXferProcessor& xp = xpBase.cast<PDLCDXferProcessor>();
606         if (fBlendConstant != xp.fBlendConstant || fAlpha != xp.fAlpha) {
607             return false;
608         }
609         return true;
610     }
611 
612     SkPMColor4f fBlendConstant;
613     float fAlpha;
614 
615     typedef GrXferProcessor INHERITED;
616 };
617 
618 ///////////////////////////////////////////////////////////////////////////////
619 
620 class GLPDLCDXferProcessor : public GrGLSLXferProcessor {
621 public:
GLPDLCDXferProcessor(const GrProcessor &)622     GLPDLCDXferProcessor(const GrProcessor&) : fLastAlpha(SK_FloatNaN) {}
623 
~GLPDLCDXferProcessor()624     ~GLPDLCDXferProcessor() override {}
625 
GenKey(const GrProcessor & processor,const GrShaderCaps & caps,GrProcessorKeyBuilder * b)626     static void GenKey(const GrProcessor& processor, const GrShaderCaps& caps,
627                        GrProcessorKeyBuilder* b) {}
628 
629 private:
emitOutputsForBlendState(const EmitArgs & args)630     void emitOutputsForBlendState(const EmitArgs& args) override {
631         const char* alpha;
632         fAlphaUniform = args.fUniformHandler->addUniform(kFragment_GrShaderFlag, kHalf_GrSLType,
633                                                          "alpha", &alpha);
634         GrGLSLXPFragmentBuilder* fragBuilder = args.fXPFragBuilder;
635         // We want to force our primary output to be alpha * Coverage, where alpha is the alpha
636         // value of the src color. We know that there are no color stages (or we wouldn't have
637         // created this xp) and the r,g, and b channels of the op's input color are baked into the
638         // blend constant.
639         SkASSERT(args.fInputCoverage);
640         fragBuilder->codeAppendf("%s = %s * %s;", args.fOutputPrimary, alpha, args.fInputCoverage);
641     }
642 
onSetData(const GrGLSLProgramDataManager & pdm,const GrXferProcessor & xp)643     void onSetData(const GrGLSLProgramDataManager& pdm, const GrXferProcessor& xp) override {
644         float alpha = xp.cast<PDLCDXferProcessor>().alpha();
645         if (fLastAlpha != alpha) {
646             pdm.set1f(fAlphaUniform, alpha);
647             fLastAlpha = alpha;
648         }
649     }
650 
651     GrGLSLUniformHandler::UniformHandle fAlphaUniform;
652     float fLastAlpha;
653     typedef GrGLSLXferProcessor INHERITED;
654 };
655 
656 ///////////////////////////////////////////////////////////////////////////////
657 
PDLCDXferProcessor(const SkPMColor4f & blendConstant,float alpha)658 PDLCDXferProcessor::PDLCDXferProcessor(const SkPMColor4f& blendConstant, float alpha)
659     : INHERITED(kPDLCDXferProcessor_ClassID, false, false, GrProcessorAnalysisCoverage::kLCD)
660     , fBlendConstant(blendConstant)
661     , fAlpha(alpha) {
662 }
663 
Make(SkBlendMode mode,const GrProcessorAnalysisColor & color)664 sk_sp<const GrXferProcessor> PDLCDXferProcessor::Make(SkBlendMode mode,
665                                                       const GrProcessorAnalysisColor& color) {
666     if (SkBlendMode::kSrcOver != mode) {
667         return nullptr;
668     }
669     SkPMColor4f blendConstantPM;
670     if (!color.isConstant(&blendConstantPM)) {
671         return nullptr;
672     }
673     SkColor4f blendConstantUPM = blendConstantPM.unpremul();
674     float alpha = blendConstantUPM.fA;
675     blendConstantPM = { blendConstantUPM.fR, blendConstantUPM.fG, blendConstantUPM.fB, 1 };
676     return sk_sp<GrXferProcessor>(new PDLCDXferProcessor(blendConstantPM, alpha));
677 }
678 
~PDLCDXferProcessor()679 PDLCDXferProcessor::~PDLCDXferProcessor() {
680 }
681 
onGetGLSLProcessorKey(const GrShaderCaps & caps,GrProcessorKeyBuilder * b) const682 void PDLCDXferProcessor::onGetGLSLProcessorKey(const GrShaderCaps& caps,
683                                                GrProcessorKeyBuilder* b) const {
684     GLPDLCDXferProcessor::GenKey(*this, caps, b);
685 }
686 
createGLSLInstance() const687 GrGLSLXferProcessor* PDLCDXferProcessor::createGLSLInstance() const {
688     return new GLPDLCDXferProcessor(*this);
689 }
690 
691 ///////////////////////////////////////////////////////////////////////////////
692 
GrPorterDuffXPFactory(SkBlendMode xfermode)693 constexpr GrPorterDuffXPFactory::GrPorterDuffXPFactory(SkBlendMode xfermode)
694         : fBlendMode(xfermode) {}
695 
Get(SkBlendMode blendMode)696 const GrXPFactory* GrPorterDuffXPFactory::Get(SkBlendMode blendMode) {
697     SkASSERT((unsigned)blendMode <= (unsigned)SkBlendMode::kLastCoeffMode);
698 
699     // If these objects are constructed as static constexpr by cl.exe (2015 SP2) the vtables are
700     // null.
701 #ifdef SK_BUILD_FOR_WIN
702 #define _CONSTEXPR_
703 #else
704 #define _CONSTEXPR_ constexpr
705 #endif
706     static _CONSTEXPR_ const GrPorterDuffXPFactory gClearPDXPF(SkBlendMode::kClear);
707     static _CONSTEXPR_ const GrPorterDuffXPFactory gSrcPDXPF(SkBlendMode::kSrc);
708     static _CONSTEXPR_ const GrPorterDuffXPFactory gDstPDXPF(SkBlendMode::kDst);
709     static _CONSTEXPR_ const GrPorterDuffXPFactory gSrcOverPDXPF(SkBlendMode::kSrcOver);
710     static _CONSTEXPR_ const GrPorterDuffXPFactory gDstOverPDXPF(SkBlendMode::kDstOver);
711     static _CONSTEXPR_ const GrPorterDuffXPFactory gSrcInPDXPF(SkBlendMode::kSrcIn);
712     static _CONSTEXPR_ const GrPorterDuffXPFactory gDstInPDXPF(SkBlendMode::kDstIn);
713     static _CONSTEXPR_ const GrPorterDuffXPFactory gSrcOutPDXPF(SkBlendMode::kSrcOut);
714     static _CONSTEXPR_ const GrPorterDuffXPFactory gDstOutPDXPF(SkBlendMode::kDstOut);
715     static _CONSTEXPR_ const GrPorterDuffXPFactory gSrcATopPDXPF(SkBlendMode::kSrcATop);
716     static _CONSTEXPR_ const GrPorterDuffXPFactory gDstATopPDXPF(SkBlendMode::kDstATop);
717     static _CONSTEXPR_ const GrPorterDuffXPFactory gXorPDXPF(SkBlendMode::kXor);
718     static _CONSTEXPR_ const GrPorterDuffXPFactory gPlusPDXPF(SkBlendMode::kPlus);
719     static _CONSTEXPR_ const GrPorterDuffXPFactory gModulatePDXPF(SkBlendMode::kModulate);
720     static _CONSTEXPR_ const GrPorterDuffXPFactory gScreenPDXPF(SkBlendMode::kScreen);
721 #undef _CONSTEXPR_
722 
723     switch (blendMode) {
724         case SkBlendMode::kClear:
725             return &gClearPDXPF;
726         case SkBlendMode::kSrc:
727             return &gSrcPDXPF;
728         case SkBlendMode::kDst:
729             return &gDstPDXPF;
730         case SkBlendMode::kSrcOver:
731             return &gSrcOverPDXPF;
732         case SkBlendMode::kDstOver:
733             return &gDstOverPDXPF;
734         case SkBlendMode::kSrcIn:
735             return &gSrcInPDXPF;
736         case SkBlendMode::kDstIn:
737             return &gDstInPDXPF;
738         case SkBlendMode::kSrcOut:
739             return &gSrcOutPDXPF;
740         case SkBlendMode::kDstOut:
741             return &gDstOutPDXPF;
742         case SkBlendMode::kSrcATop:
743             return &gSrcATopPDXPF;
744         case SkBlendMode::kDstATop:
745             return &gDstATopPDXPF;
746         case SkBlendMode::kXor:
747             return &gXorPDXPF;
748         case SkBlendMode::kPlus:
749             return &gPlusPDXPF;
750         case SkBlendMode::kModulate:
751             return &gModulatePDXPF;
752         case SkBlendMode::kScreen:
753             return &gScreenPDXPF;
754         default:
755             SK_ABORT("Unexpected blend mode.");
756             return nullptr;
757     }
758 }
759 
makeXferProcessor(const GrProcessorAnalysisColor & color,GrProcessorAnalysisCoverage coverage,bool hasMixedSamples,const GrCaps & caps,GrClampType clampType) const760 sk_sp<const GrXferProcessor> GrPorterDuffXPFactory::makeXferProcessor(
761         const GrProcessorAnalysisColor& color, GrProcessorAnalysisCoverage coverage,
762         bool hasMixedSamples, const GrCaps& caps, GrClampType clampType) const {
763     BlendFormula blendFormula;
764     bool isLCD = coverage == GrProcessorAnalysisCoverage::kLCD;
765     if (isLCD) {
766         // See comment in MakeSrcOverXferProcessor about color.isOpaque here
767         if (SkBlendMode::kSrcOver == fBlendMode && color.isConstant() && /*color.isOpaque() &&*/
768             !caps.shaderCaps()->dualSourceBlendingSupport() &&
769             !caps.shaderCaps()->dstReadInShaderSupport()) {
770             // If we don't have dual source blending or in shader dst reads, we fall back to this
771             // trick for rendering SrcOver LCD text instead of doing a dst copy.
772             return PDLCDXferProcessor::Make(fBlendMode, color);
773         }
774         blendFormula = get_lcd_blend_formula(fBlendMode);
775     } else {
776         blendFormula =
777                 get_blend_formula(color.isOpaque(), GrProcessorAnalysisCoverage::kNone != coverage,
778                                   hasMixedSamples, fBlendMode);
779     }
780 
781     // Skia always saturates after the kPlus blend mode, so it requires shader-based blending when
782     // pixels aren't guaranteed to automatically be normalized (i.e. any floating point config).
783     if ((blendFormula.hasSecondaryOutput() && !caps.shaderCaps()->dualSourceBlendingSupport()) ||
784         (isLCD && (SkBlendMode::kSrcOver != fBlendMode /*|| !color.isOpaque()*/)) ||
785         (GrClampType::kAuto != clampType && SkBlendMode::kPlus == fBlendMode)) {
786         return sk_sp<const GrXferProcessor>(new ShaderPDXferProcessor(hasMixedSamples, fBlendMode,
787                                                                       coverage));
788     }
789     return sk_sp<const GrXferProcessor>(new PorterDuffXferProcessor(blendFormula, coverage));
790 }
791 
analysis_properties(const GrProcessorAnalysisColor & color,const GrProcessorAnalysisCoverage & coverage,const GrCaps & caps,GrClampType clampType,SkBlendMode mode)792 static inline GrXPFactory::AnalysisProperties analysis_properties(
793         const GrProcessorAnalysisColor& color, const GrProcessorAnalysisCoverage& coverage,
794         const GrCaps& caps, GrClampType clampType, SkBlendMode mode) {
795     using AnalysisProperties = GrXPFactory::AnalysisProperties;
796     AnalysisProperties props = AnalysisProperties::kNone;
797     bool hasCoverage = GrProcessorAnalysisCoverage::kNone != coverage;
798     bool isLCD = GrProcessorAnalysisCoverage::kLCD == coverage;
799     BlendFormula formula;
800     if (isLCD) {
801         formula = gLCDBlendTable[(int)mode];
802     } else {
803         formula = gBlendTable[color.isOpaque()][hasCoverage][(int)mode];
804     }
805 
806     if (formula.canTweakAlphaForCoverage() && !isLCD) {
807         props |= AnalysisProperties::kCompatibleWithAlphaAsCoverage;
808     }
809 
810     if (isLCD) {
811         // See comment in MakeSrcOverXferProcessor about color.isOpaque here
812         if (SkBlendMode::kSrcOver == mode && color.isConstant() && /*color.isOpaque() &&*/
813             !caps.shaderCaps()->dualSourceBlendingSupport() &&
814             !caps.shaderCaps()->dstReadInShaderSupport()) {
815             props |= AnalysisProperties::kIgnoresInputColor;
816         } else {
817             // For LCD blending, if the color is not opaque we must read the dst in shader even if
818             // we have dual source blending. The opaqueness check must be done after blending so for
819             // simplicity we only allow src-over to not take the dst read path (though src, src-in,
820             // and DstATop would also work). We also fall into the dst read case for src-over if we
821             // do not have dual source blending.
822             if (SkBlendMode::kSrcOver != mode ||
823                 /*!color.isOpaque() ||*/ // See comment in MakeSrcOverXferProcessor about isOpaque.
824                 (formula.hasSecondaryOutput() && !caps.shaderCaps()->dualSourceBlendingSupport())) {
825                 props |= AnalysisProperties::kReadsDstInShader;
826             }
827         }
828     } else {
829         // With dual-source blending we never need the destination color in the shader.
830         if (!caps.shaderCaps()->dualSourceBlendingSupport()) {
831             // Mixed samples implicity computes a fractional coverage from sample coverage. This
832             // could affect the formula used. However, we don't expect to have mixed samples without
833             // dual source blending.
834             SkASSERT(!caps.usesMixedSamples());
835             if (formula.hasSecondaryOutput()) {
836                 props |= AnalysisProperties::kReadsDstInShader;
837             }
838         }
839     }
840 
841     if (GrClampType::kAuto != clampType && SkBlendMode::kPlus == mode) {
842         props |= AnalysisProperties::kReadsDstInShader;
843     }
844 
845     if (!formula.modifiesDst() || !formula.usesInputColor()) {
846         props |= AnalysisProperties::kIgnoresInputColor;
847     }
848     return props;
849 }
850 
analysisProperties(const GrProcessorAnalysisColor & color,const GrProcessorAnalysisCoverage & coverage,const GrCaps & caps,GrClampType clampType) const851 GrXPFactory::AnalysisProperties GrPorterDuffXPFactory::analysisProperties(
852         const GrProcessorAnalysisColor& color,
853         const GrProcessorAnalysisCoverage& coverage,
854         const GrCaps& caps,
855         GrClampType clampType) const {
856     return analysis_properties(color, coverage, caps, clampType, fBlendMode);
857 }
858 
859 GR_DEFINE_XP_FACTORY_TEST(GrPorterDuffXPFactory);
860 
861 #if GR_TEST_UTILS
TestGet(GrProcessorTestData * d)862 const GrXPFactory* GrPorterDuffXPFactory::TestGet(GrProcessorTestData* d) {
863     SkBlendMode mode = SkBlendMode(d->fRandom->nextULessThan((int)SkBlendMode::kLastCoeffMode));
864     return GrPorterDuffXPFactory::Get(mode);
865 }
866 #endif
867 
TestGetXPOutputTypes(const GrXferProcessor * xp,int * outPrimary,int * outSecondary)868 void GrPorterDuffXPFactory::TestGetXPOutputTypes(const GrXferProcessor* xp,
869                                                  int* outPrimary,
870                                                  int* outSecondary) {
871     if (!!strcmp(xp->name(), "Porter Duff")) {
872         *outPrimary = *outSecondary = -1;
873         return;
874     }
875     BlendFormula blendFormula = static_cast<const PorterDuffXferProcessor*>(xp)->getBlendFormula();
876     *outPrimary = blendFormula.primaryOutput();
877     *outSecondary = blendFormula.secondaryOutput();
878 }
879 
880 ////////////////////////////////////////////////////////////////////////////////////////////////
881 // SrcOver Global functions
882 ////////////////////////////////////////////////////////////////////////////////////////////////
SimpleSrcOverXP()883 const GrXferProcessor& GrPorterDuffXPFactory::SimpleSrcOverXP() {
884     static BlendFormula gSrcOverBlendFormula =
885             MakeCoeffFormula(kOne_GrBlendCoeff, kISA_GrBlendCoeff);
886     static PorterDuffXferProcessor gSrcOverXP(gSrcOverBlendFormula,
887                                               GrProcessorAnalysisCoverage::kSingleChannel);
888     return gSrcOverXP;
889 }
890 
MakeSrcOverXferProcessor(const GrProcessorAnalysisColor & color,GrProcessorAnalysisCoverage coverage,bool hasMixedSamples,const GrCaps & caps)891 sk_sp<const GrXferProcessor> GrPorterDuffXPFactory::MakeSrcOverXferProcessor(
892         const GrProcessorAnalysisColor& color, GrProcessorAnalysisCoverage coverage,
893         bool hasMixedSamples, const GrCaps& caps) {
894     // We want to not make an xfer processor if possible. Thus for the simple case where we are not
895     // doing lcd blending we will just use our global SimpleSrcOverXP. This slightly differs from
896     // the general case where we convert a src-over blend that has solid coverage and an opaque
897     // color to src-mode, which allows disabling of blending.
898     if (coverage != GrProcessorAnalysisCoverage::kLCD) {
899         // We return nullptr here, which our caller interprets as meaning "use SimpleSrcOverXP".
900         // We don't simply return the address of that XP here because our caller would have to unref
901         // it and since it is a global object and GrProgramElement's ref-cnting system is not thread
902         // safe.
903         return nullptr;
904     }
905 
906     // Currently up the stack Skia is requiring that the dst is opaque or that the client has said
907     // the opaqueness doesn't matter. Thus for src-over we don't need to worry about the src color
908     // being opaque or not. This allows us to use faster code paths as well as avoid various bugs
909     // that occur with dst reads in the shader blending. For now we disable the check for
910     // opaqueness, but in the future we should pass down the knowledge about dst opaqueness and make
911     // the correct decision here.
912     //
913     // This also fixes a chrome bug on macs where we are getting random fuzziness when doing
914     // blending in the shader for non opaque sources.
915     if (color.isConstant() && /*color.isOpaque() &&*/
916         !caps.shaderCaps()->dualSourceBlendingSupport() &&
917         !caps.shaderCaps()->dstReadInShaderSupport()) {
918         // If we don't have dual source blending or in shader dst reads, we fall
919         // back to this trick for rendering SrcOver LCD text instead of doing a
920         // dst copy.
921         return PDLCDXferProcessor::Make(SkBlendMode::kSrcOver, color);
922     }
923 
924     BlendFormula blendFormula;
925     blendFormula = get_lcd_blend_formula(SkBlendMode::kSrcOver);
926     // See comment above regarding why the opaque check is commented out here.
927     if (/*!color.isOpaque() ||*/
928         (blendFormula.hasSecondaryOutput() && !caps.shaderCaps()->dualSourceBlendingSupport())) {
929         return sk_sp<GrXferProcessor>(
930                 new ShaderPDXferProcessor(hasMixedSamples, SkBlendMode::kSrcOver, coverage));
931     }
932     return sk_sp<GrXferProcessor>(new PorterDuffXferProcessor(blendFormula, coverage));
933 }
934 
MakeNoCoverageXP(SkBlendMode blendmode)935 sk_sp<const GrXferProcessor> GrPorterDuffXPFactory::MakeNoCoverageXP(SkBlendMode blendmode) {
936     BlendFormula formula = get_blend_formula(false, false, false, blendmode);
937     return sk_make_sp<PorterDuffXferProcessor>(formula, GrProcessorAnalysisCoverage::kNone);
938 }
939 
SrcOverAnalysisProperties(const GrProcessorAnalysisColor & color,const GrProcessorAnalysisCoverage & coverage,const GrCaps & caps,GrClampType clampType)940 GrXPFactory::AnalysisProperties GrPorterDuffXPFactory::SrcOverAnalysisProperties(
941         const GrProcessorAnalysisColor& color,
942         const GrProcessorAnalysisCoverage& coverage,
943         const GrCaps& caps,
944         GrClampType clampType) {
945     return analysis_properties(color, coverage, caps, clampType, SkBlendMode::kSrcOver);
946 }
947