1 /*
2 * Copyright 2013 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 "GrPathProcessor.h"
9 
10 #include "GrShaderCaps.h"
11 #include "SkTo.h"
12 #include "gl/GrGLGpu.h"
13 #include "gl/GrGLVaryingHandler.h"
14 #include "glsl/GrGLSLFragmentShaderBuilder.h"
15 #include "glsl/GrGLSLPrimitiveProcessor.h"
16 #include "glsl/GrGLSLUniformHandler.h"
17 #include "glsl/GrGLSLVarying.h"
18 
19 class GrGLPathProcessor : public GrGLSLPrimitiveProcessor {
20 public:
GrGLPathProcessor()21     GrGLPathProcessor() : fColor(SK_PMColor4fILLEGAL) {}
22 
GenKey(const GrPathProcessor & pathProc,const GrShaderCaps &,GrProcessorKeyBuilder * b)23     static void GenKey(const GrPathProcessor& pathProc,
24                        const GrShaderCaps&,
25                        GrProcessorKeyBuilder* b) {
26         b->add32(SkToInt(pathProc.viewMatrix().hasPerspective()));
27     }
28 
emitCode(EmitArgs & args)29     void emitCode(EmitArgs& args) override {
30         GrGLSLFPFragmentBuilder* fragBuilder = args.fFragBuilder;
31         const GrPathProcessor& pathProc = args.fGP.cast<GrPathProcessor>();
32 
33         if (!pathProc.viewMatrix().hasPerspective()) {
34             args.fVaryingHandler->setNoPerspective();
35         }
36 
37         // emit transforms
38         this->emitTransforms(args.fVaryingHandler, args.fFPCoordTransformHandler);
39 
40         // Setup uniform color
41         const char* stagedLocalVarName;
42         fColorUniform = args.fUniformHandler->addUniform(kFragment_GrShaderFlag,
43                                                          kHalf4_GrSLType,
44                                                          "Color",
45                                                          &stagedLocalVarName);
46         fragBuilder->codeAppendf("%s = %s;", args.fOutputColor, stagedLocalVarName);
47 
48         // setup constant solid coverage
49         fragBuilder->codeAppendf("%s = half4(1);", args.fOutputCoverage);
50     }
51 
emitTransforms(GrGLSLVaryingHandler * varyingHandler,FPCoordTransformHandler * transformHandler)52     void emitTransforms(GrGLSLVaryingHandler* varyingHandler,
53                         FPCoordTransformHandler* transformHandler) {
54         int i = 0;
55         while (const GrCoordTransform* coordTransform = transformHandler->nextCoordTransform()) {
56             GrSLType varyingType =
57                     coordTransform->getMatrix().hasPerspective() ? kHalf3_GrSLType
58                                                                  : kHalf2_GrSLType;
59 
60             SkString strVaryingName;
61             strVaryingName.printf("TransformedCoord_%d", i);
62             GrGLSLVarying v(varyingType);
63             GrGLVaryingHandler* glVaryingHandler = (GrGLVaryingHandler*) varyingHandler;
64             fInstalledTransforms.push_back().fHandle =
65                     glVaryingHandler->addPathProcessingVarying(strVaryingName.c_str(),
66                                                                &v).toIndex();
67             fInstalledTransforms.back().fType = varyingType;
68 
69             transformHandler->specifyCoordsForCurrCoordTransform(SkString(v.fsIn()), varyingType);
70             ++i;
71         }
72     }
73 
setData(const GrGLSLProgramDataManager & pd,const GrPrimitiveProcessor & primProc,FPCoordTransformIter && transformIter)74     void setData(const GrGLSLProgramDataManager& pd,
75                  const GrPrimitiveProcessor& primProc,
76                  FPCoordTransformIter&& transformIter) override {
77         const GrPathProcessor& pathProc = primProc.cast<GrPathProcessor>();
78         if (pathProc.color() != fColor) {
79             pd.set4fv(fColorUniform, 1, pathProc.color().vec());
80             fColor = pathProc.color();
81         }
82 
83         int t = 0;
84         while (const GrCoordTransform* coordTransform = transformIter.next()) {
85             SkASSERT(fInstalledTransforms[t].fHandle.isValid());
86             const SkMatrix& m = GetTransformMatrix(pathProc.localMatrix(), *coordTransform);
87             if (fInstalledTransforms[t].fCurrentValue.cheapEqualTo(m)) {
88                 continue;
89             }
90             fInstalledTransforms[t].fCurrentValue = m;
91 
92             SkASSERT(fInstalledTransforms[t].fType == kHalf2_GrSLType ||
93                      fInstalledTransforms[t].fType == kHalf3_GrSLType);
94             unsigned components = fInstalledTransforms[t].fType == kHalf2_GrSLType ? 2 : 3;
95             pd.setPathFragmentInputTransform(fInstalledTransforms[t].fHandle, components, m);
96             ++t;
97         }
98     }
99 
100 private:
101     typedef GrGLSLProgramDataManager::VaryingHandle VaryingHandle;
102     struct TransformVarying {
103         VaryingHandle  fHandle;
104         SkMatrix       fCurrentValue = SkMatrix::InvalidMatrix();
105         GrSLType       fType = kVoid_GrSLType;
106     };
107 
108     SkTArray<TransformVarying, true> fInstalledTransforms;
109 
110     UniformHandle fColorUniform;
111     SkPMColor4f fColor;
112 
113     typedef GrGLSLPrimitiveProcessor INHERITED;
114 };
115 
GrPathProcessor(const SkPMColor4f & color,const SkMatrix & viewMatrix,const SkMatrix & localMatrix)116 GrPathProcessor::GrPathProcessor(const SkPMColor4f& color,
117                                  const SkMatrix& viewMatrix,
118                                  const SkMatrix& localMatrix)
119         : INHERITED(kGrPathProcessor_ClassID)
120         , fColor(color)
121         , fViewMatrix(viewMatrix)
122         , fLocalMatrix(localMatrix) {}
123 
getGLSLProcessorKey(const GrShaderCaps & caps,GrProcessorKeyBuilder * b) const124 void GrPathProcessor::getGLSLProcessorKey(const GrShaderCaps& caps,
125                                           GrProcessorKeyBuilder* b) const {
126     GrGLPathProcessor::GenKey(*this, caps, b);
127 }
128 
createGLSLInstance(const GrShaderCaps & caps) const129 GrGLSLPrimitiveProcessor* GrPathProcessor::createGLSLInstance(const GrShaderCaps& caps) const {
130     SkASSERT(caps.pathRenderingSupport());
131     return new GrGLPathProcessor();
132 }
133