blob: 61c84d23649abeee562e91fc5029fb7cab85a389 [file] [log] [blame]
jvanverth50530632015-04-27 10:36:27 -07001/*
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
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "src/gpu/GrPathProcessor.h"
Hal Canaryc640d0d2018-06-13 09:59:02 -04009
Mike Kleinc0bd9f92019-04-23 12:05:21 -050010#include "include/private/SkTo.h"
11#include "src/gpu/GrShaderCaps.h"
12#include "src/gpu/gl/GrGLGpu.h"
John Rosasco24cbdab2019-09-25 14:14:35 -070013#ifdef SK_GL
Mike Kleinc0bd9f92019-04-23 12:05:21 -050014#include "src/gpu/gl/GrGLVaryingHandler.h"
John Rosasco24cbdab2019-09-25 14:14:35 -070015#endif
Mike Kleinc0bd9f92019-04-23 12:05:21 -050016#include "src/gpu/glsl/GrGLSLFragmentShaderBuilder.h"
17#include "src/gpu/glsl/GrGLSLPrimitiveProcessor.h"
18#include "src/gpu/glsl/GrGLSLUniformHandler.h"
19#include "src/gpu/glsl/GrGLSLVarying.h"
jvanverthcba99b82015-06-24 06:59:57 -070020
egdaniele659a582015-11-13 09:55:43 -080021class GrGLPathProcessor : public GrGLSLPrimitiveProcessor {
joshualitt102081a2015-09-11 11:52:17 -070022public:
Brian Osmancf860852018-10-31 14:04:39 -040023 GrGLPathProcessor() : fColor(SK_PMColor4fILLEGAL) {}
joshualitt102081a2015-09-11 11:52:17 -070024
25 static void GenKey(const GrPathProcessor& pathProc,
Brian Salomon94efbf52016-11-29 13:43:05 -050026 const GrShaderCaps&,
joshualitt102081a2015-09-11 11:52:17 -070027 GrProcessorKeyBuilder* b) {
Brian Salomonbfd51832017-01-04 13:22:08 -050028 b->add32(SkToInt(pathProc.viewMatrix().hasPerspective()));
joshualitt102081a2015-09-11 11:52:17 -070029 }
30
31 void emitCode(EmitArgs& args) override {
Chris Dalton60283612018-02-14 13:38:14 -070032 GrGLSLFPFragmentBuilder* fragBuilder = args.fFragBuilder;
joshualitt102081a2015-09-11 11:52:17 -070033 const GrPathProcessor& pathProc = args.fGP.cast<GrPathProcessor>();
34
cdaltonc08f1962016-02-12 12:14:06 -080035 if (!pathProc.viewMatrix().hasPerspective()) {
36 args.fVaryingHandler->setNoPerspective();
37 }
38
joshualitt102081a2015-09-11 11:52:17 -070039 // emit transforms
Brian Salomon510eb152020-02-27 13:52:16 -050040 this->emitTransforms(args.fVaryingHandler, args.fUniformHandler,
41 args.fFPCoordTransformHandler);
joshualitt102081a2015-09-11 11:52:17 -070042
43 // Setup uniform color
Brian Salomonbfd51832017-01-04 13:22:08 -050044 const char* stagedLocalVarName;
45 fColorUniform = args.fUniformHandler->addUniform(kFragment_GrShaderFlag,
Ethan Nicholasf7b88202017-09-18 14:10:39 -040046 kHalf4_GrSLType,
Brian Salomonbfd51832017-01-04 13:22:08 -050047 "Color",
48 &stagedLocalVarName);
49 fragBuilder->codeAppendf("%s = %s;", args.fOutputColor, stagedLocalVarName);
joshualitt102081a2015-09-11 11:52:17 -070050
51 // setup constant solid coverage
Ethan Nicholasf7b88202017-09-18 14:10:39 -040052 fragBuilder->codeAppendf("%s = half4(1);", args.fOutputCoverage);
joshualitt102081a2015-09-11 11:52:17 -070053 }
54
Ethan Nicholasd4efe682019-08-29 16:10:13 -040055 SkString matrix_to_sksl(const SkMatrix& m) {
56 return SkStringPrintf("float3x3(%f, %f, %f, %f, %f, %f, %f, %f, %f)", m[0], m[1], m[2],
57 m[3], m[4], m[5], m[6], m[7], m[8]);
58 }
59
egdaniel0eafe792015-11-20 14:01:22 -080060 void emitTransforms(GrGLSLVaryingHandler* varyingHandler,
Brian Salomon510eb152020-02-27 13:52:16 -050061 GrGLSLUniformHandler* uniformHandler,
bsalomona624bf32016-09-20 09:12:47 -070062 FPCoordTransformHandler* transformHandler) {
Brian Salomon7d8b3972019-11-26 22:34:44 -050063 for (int i = 0; *transformHandler; ++*transformHandler, ++i) {
64 auto [coordTransform, fp] = transformHandler->get();
bsalomona624bf32016-09-20 09:12:47 -070065 GrSLType varyingType =
Brian Salomon7d8b3972019-11-26 22:34:44 -050066 coordTransform.matrix().hasPerspective() ? kHalf3_GrSLType : kHalf2_GrSLType;
joshualitt102081a2015-09-11 11:52:17 -070067
Brian Salomon510eb152020-02-27 13:52:16 -050068 SkString matrix;
69 UniformHandle uniformHandle;
70 GrShaderVar fragmentVar;
71 if (fp.isSampledWithExplicitCoords()) {
72 if (coordTransform.isNoOp()) {
73 transformHandler->omitCoordsForCurrCoordTransform();
74 continue;
75 } else {
76 const char* name;
77 SkString strUniName;
78 strUniName.printf("CoordTransformMatrix_%d", i);
79 fUniformTransform.push_back().fHandle =
80 uniformHandler
81 ->addUniform(kFragment_GrShaderFlag,
82 kFloat3x3_GrSLType,
83 strUniName.c_str(),
84 &name)
85 .toIndex();
86 uniformHandle = fUniformTransform.back().fHandle;
87 matrix = name;
88 }
89 } else {
90 SkString strVaryingName;
91 strVaryingName.printf("TransformedCoord_%d", i);
92 GrGLSLVarying v(varyingType);
John Rosasco24cbdab2019-09-25 14:14:35 -070093#ifdef SK_GL
Brian Salomon510eb152020-02-27 13:52:16 -050094 GrGLVaryingHandler* glVaryingHandler = (GrGLVaryingHandler*)varyingHandler;
95 fVaryingTransform.push_back().fHandle =
96 glVaryingHandler->addPathProcessingVarying(strVaryingName.c_str(), &v)
97 .toIndex();
John Rosasco24cbdab2019-09-25 14:14:35 -070098#endif
Brian Salomon510eb152020-02-27 13:52:16 -050099 fVaryingTransform.back().fType = varyingType;
100 matrix = matrix_to_sksl(coordTransform.matrix());
101 fragmentVar = {SkString(v.fsIn()), varyingType};
102 }
103 transformHandler->specifyCoordsForCurrCoordTransform(matrix, uniformHandle,
104 fragmentVar);
joshualitt102081a2015-09-11 11:52:17 -0700105 }
106 }
107
egdaniel018fb622015-10-28 07:26:40 -0700108 void setData(const GrGLSLProgramDataManager& pd,
bsalomona624bf32016-09-20 09:12:47 -0700109 const GrPrimitiveProcessor& primProc,
Brian Salomonc241b582019-11-27 08:57:17 -0500110 const CoordTransformRange& transformRange) override {
joshualitt102081a2015-09-11 11:52:17 -0700111 const GrPathProcessor& pathProc = primProc.cast<GrPathProcessor>();
Brian Salomonbfd51832017-01-04 13:22:08 -0500112 if (pathProc.color() != fColor) {
Brian Osmancf860852018-10-31 14:04:39 -0400113 pd.set4fv(fColorUniform, 1, pathProc.color().vec());
joshualitt102081a2015-09-11 11:52:17 -0700114 fColor = pathProc.color();
115 }
joshualitt102081a2015-09-11 11:52:17 -0700116
Brian Salomon510eb152020-02-27 13:52:16 -0500117 int v = 0, u = 0;
Brian Salomonc241b582019-11-27 08:57:17 -0500118 for (auto [transform, fp] : transformRange) {
Brian Salomonb10a6622020-02-20 13:36:01 -0500119 if (fp.isSampledWithExplicitCoords()) {
Brian Salomon510eb152020-02-27 13:52:16 -0500120 if (transform.isNoOp()) {
121 continue;
122 }
123 if (fUniformTransform[u].fHandle.isValid()) {
124 SkMatrix m = GetTransformMatrix(transform, SkMatrix::I());
125 if (!SkMatrixPriv::CheapEqual(fUniformTransform[u].fCurrentValue, m)) {
126 fUniformTransform[u].fCurrentValue = m;
127 pd.setSkMatrix(fUniformTransform[u].fHandle.toIndex(), m);
128 }
129 }
130 ++u;
Brian Salomonb10a6622020-02-20 13:36:01 -0500131 } else {
Brian Salomon510eb152020-02-27 13:52:16 -0500132 SkASSERT(fVaryingTransform[v].fHandle.isValid());
133 SkMatrix m = GetTransformMatrix(transform, pathProc.localMatrix());
134 if (!SkMatrixPriv::CheapEqual(fVaryingTransform[v].fCurrentValue, m)) {
135 fVaryingTransform[v].fCurrentValue = m;
136 SkASSERT(fVaryingTransform[v].fType == kHalf2_GrSLType ||
137 fVaryingTransform[v].fType == kHalf3_GrSLType);
138 int components = fVaryingTransform[v].fType == kHalf2_GrSLType ? 2 : 3;
139 pd.setPathFragmentInputTransform(fVaryingTransform[v].fHandle, components, m);
140 }
141 ++v;
Brian Salomon7eabfe82019-12-02 14:20:20 -0500142 }
joshualitt102081a2015-09-11 11:52:17 -0700143 }
144 }
145
146private:
Brian Salomon510eb152020-02-27 13:52:16 -0500147 using VaryingHandle = GrGLSLProgramDataManager::VaryingHandle;
148
149 // Varying transforms are used for non-explicitly sampled FPs. We provide a matrix
150 // to GL as fixed function state and it uses it to compute a varying that we pick up
151 // in the FS as the output of the coord transform.
bsalomon790c90b2016-09-12 12:56:58 -0700152 struct TransformVarying {
Brian Salomon510eb152020-02-27 13:52:16 -0500153 VaryingHandle fHandle;
154 SkMatrix fCurrentValue = SkMatrix::InvalidMatrix();
155 GrSLType fType = kVoid_GrSLType;
156 };
157 // For explicitly sampled FPs we stick a uniform in the FS and apply it to the explicit coords
158 // to implement the CoordTransform.
159 struct TransformUniform {
160 UniformHandle fHandle;
161 SkMatrix fCurrentValue = SkMatrix::InvalidMatrix();
egdanielfe8a8392016-05-09 10:22:19 -0700162 };
163
Brian Salomon510eb152020-02-27 13:52:16 -0500164 SkTArray<TransformVarying, true> fVaryingTransform;
165 SkTArray<TransformUniform, true> fUniformTransform;
egdanielfe8a8392016-05-09 10:22:19 -0700166
joshualitt102081a2015-09-11 11:52:17 -0700167 UniformHandle fColorUniform;
Brian Osmancf860852018-10-31 14:04:39 -0400168 SkPMColor4f fColor;
joshualitt102081a2015-09-11 11:52:17 -0700169
egdaniele659a582015-11-13 09:55:43 -0800170 typedef GrGLSLPrimitiveProcessor INHERITED;
joshualitt102081a2015-09-11 11:52:17 -0700171};
172
Brian Osmancf860852018-10-31 14:04:39 -0400173GrPathProcessor::GrPathProcessor(const SkPMColor4f& color,
joshualitt8072caa2015-02-12 14:20:52 -0800174 const SkMatrix& viewMatrix,
175 const SkMatrix& localMatrix)
Ethan Nicholasabff9562017-10-09 10:54:08 -0400176 : INHERITED(kGrPathProcessor_ClassID)
177 , fColor(color)
Brian Salomon92aee3d2016-12-21 09:20:25 -0500178 , fViewMatrix(viewMatrix)
Ethan Nicholasabff9562017-10-09 10:54:08 -0400179 , fLocalMatrix(localMatrix) {}
joshualitt8072caa2015-02-12 14:20:52 -0800180
Brian Salomon94efbf52016-11-29 13:43:05 -0500181void GrPathProcessor::getGLSLProcessorKey(const GrShaderCaps& caps,
egdaniel57d3b032015-11-13 11:57:27 -0800182 GrProcessorKeyBuilder* b) const {
joshualitt465283c2015-09-11 08:19:35 -0700183 GrGLPathProcessor::GenKey(*this, caps, b);
joshualitt8072caa2015-02-12 14:20:52 -0800184}
185
Brian Salomon94efbf52016-11-29 13:43:05 -0500186GrGLSLPrimitiveProcessor* GrPathProcessor::createGLSLInstance(const GrShaderCaps& caps) const {
jvanverth50530632015-04-27 10:36:27 -0700187 SkASSERT(caps.pathRenderingSupport());
joshualitt465283c2015-09-11 08:19:35 -0700188 return new GrGLPathProcessor();
joshualitt8072caa2015-02-12 14:20:52 -0800189}