blob: b36574077e0d7f66a28cac541c7ecf4cf8377afb [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();
joshualitt102081a2015-09-11 11:52:17 -070065
Brian Salomon510eb152020-02-27 13:52:16 -050066 SkString matrix;
Brian Salomon510eb152020-02-27 13:52:16 -050067 GrShaderVar fragmentVar;
Brian Salomon8d1dcd72020-03-20 21:06:41 -040068 GrShaderVar transformVar;
Brian Salomon510eb152020-02-27 13:52:16 -050069 if (fp.isSampledWithExplicitCoords()) {
70 if (coordTransform.isNoOp()) {
71 transformHandler->omitCoordsForCurrCoordTransform();
72 continue;
73 } else {
74 const char* name;
75 SkString strUniName;
76 strUniName.printf("CoordTransformMatrix_%d", i);
Brian Salomon8d1dcd72020-03-20 21:06:41 -040077 auto& uni = fUniformTransform.push_back();
78 if (coordTransform.matrix().isScaleTranslate()) {
79 uni.fType = kFloat4_GrSLType;
80 } else {
81 uni.fType = kFloat3x3_GrSLType;
82 }
83 uni.fHandle = uniformHandler
84 ->addUniform(kFragment_GrShaderFlag,
85 uni.fType,
86 strUniName.c_str(),
87 &name)
88 .toIndex();
89 transformVar = uniformHandler->getUniformVariable(uni.fHandle);
Brian Salomon510eb152020-02-27 13:52:16 -050090 }
91 } else {
92 SkString strVaryingName;
93 strVaryingName.printf("TransformedCoord_%d", i);
Brian Salomon8d1dcd72020-03-20 21:06:41 -040094 GrSLType varyingType = coordTransform.matrix().hasPerspective() ? kHalf3_GrSLType
95 : kHalf2_GrSLType;
Brian Salomon510eb152020-02-27 13:52:16 -050096 GrGLSLVarying v(varyingType);
John Rosasco24cbdab2019-09-25 14:14:35 -070097#ifdef SK_GL
Brian Salomon510eb152020-02-27 13:52:16 -050098 GrGLVaryingHandler* glVaryingHandler = (GrGLVaryingHandler*)varyingHandler;
99 fVaryingTransform.push_back().fHandle =
100 glVaryingHandler->addPathProcessingVarying(strVaryingName.c_str(), &v)
101 .toIndex();
John Rosasco24cbdab2019-09-25 14:14:35 -0700102#endif
Brian Salomon510eb152020-02-27 13:52:16 -0500103 fVaryingTransform.back().fType = varyingType;
104 matrix = matrix_to_sksl(coordTransform.matrix());
105 fragmentVar = {SkString(v.fsIn()), varyingType};
106 }
Brian Salomon8d1dcd72020-03-20 21:06:41 -0400107 transformHandler->specifyCoordsForCurrCoordTransform(transformVar, fragmentVar);
joshualitt102081a2015-09-11 11:52:17 -0700108 }
109 }
110
egdaniel018fb622015-10-28 07:26:40 -0700111 void setData(const GrGLSLProgramDataManager& pd,
bsalomona624bf32016-09-20 09:12:47 -0700112 const GrPrimitiveProcessor& primProc,
Brian Salomonc241b582019-11-27 08:57:17 -0500113 const CoordTransformRange& transformRange) override {
joshualitt102081a2015-09-11 11:52:17 -0700114 const GrPathProcessor& pathProc = primProc.cast<GrPathProcessor>();
Brian Salomonbfd51832017-01-04 13:22:08 -0500115 if (pathProc.color() != fColor) {
Brian Osmancf860852018-10-31 14:04:39 -0400116 pd.set4fv(fColorUniform, 1, pathProc.color().vec());
joshualitt102081a2015-09-11 11:52:17 -0700117 fColor = pathProc.color();
118 }
joshualitt102081a2015-09-11 11:52:17 -0700119
Brian Salomon510eb152020-02-27 13:52:16 -0500120 int v = 0, u = 0;
Brian Salomonc241b582019-11-27 08:57:17 -0500121 for (auto [transform, fp] : transformRange) {
Brian Salomonb10a6622020-02-20 13:36:01 -0500122 if (fp.isSampledWithExplicitCoords()) {
Brian Salomon510eb152020-02-27 13:52:16 -0500123 if (transform.isNoOp()) {
124 continue;
125 }
126 if (fUniformTransform[u].fHandle.isValid()) {
127 SkMatrix m = GetTransformMatrix(transform, SkMatrix::I());
128 if (!SkMatrixPriv::CheapEqual(fUniformTransform[u].fCurrentValue, m)) {
129 fUniformTransform[u].fCurrentValue = m;
Brian Salomon8d1dcd72020-03-20 21:06:41 -0400130 if (fUniformTransform[u].fType == kFloat4_GrSLType) {
131 float values[4] = {m.getScaleX(), m.getTranslateX(),
132 m.getScaleY(), m.getTranslateY()};
133 pd.set4fv(fUniformTransform[u].fHandle.toIndex(), 1, values);
134 } else {
135 SkASSERT(fUniformTransform[u].fType == kFloat3x3_GrSLType);
136 pd.setSkMatrix(fUniformTransform[u].fHandle.toIndex(), m);
137 }
Brian Salomon510eb152020-02-27 13:52:16 -0500138 }
139 }
140 ++u;
Brian Salomonb10a6622020-02-20 13:36:01 -0500141 } else {
Brian Salomon510eb152020-02-27 13:52:16 -0500142 SkASSERT(fVaryingTransform[v].fHandle.isValid());
143 SkMatrix m = GetTransformMatrix(transform, pathProc.localMatrix());
144 if (!SkMatrixPriv::CheapEqual(fVaryingTransform[v].fCurrentValue, m)) {
145 fVaryingTransform[v].fCurrentValue = m;
146 SkASSERT(fVaryingTransform[v].fType == kHalf2_GrSLType ||
147 fVaryingTransform[v].fType == kHalf3_GrSLType);
148 int components = fVaryingTransform[v].fType == kHalf2_GrSLType ? 2 : 3;
149 pd.setPathFragmentInputTransform(fVaryingTransform[v].fHandle, components, m);
150 }
151 ++v;
Brian Salomon7eabfe82019-12-02 14:20:20 -0500152 }
joshualitt102081a2015-09-11 11:52:17 -0700153 }
154 }
155
156private:
Brian Salomon510eb152020-02-27 13:52:16 -0500157 using VaryingHandle = GrGLSLProgramDataManager::VaryingHandle;
158
159 // Varying transforms are used for non-explicitly sampled FPs. We provide a matrix
160 // to GL as fixed function state and it uses it to compute a varying that we pick up
161 // in the FS as the output of the coord transform.
bsalomon790c90b2016-09-12 12:56:58 -0700162 struct TransformVarying {
Brian Salomon510eb152020-02-27 13:52:16 -0500163 VaryingHandle fHandle;
164 SkMatrix fCurrentValue = SkMatrix::InvalidMatrix();
165 GrSLType fType = kVoid_GrSLType;
166 };
167 // For explicitly sampled FPs we stick a uniform in the FS and apply it to the explicit coords
168 // to implement the CoordTransform.
169 struct TransformUniform {
170 UniformHandle fHandle;
171 SkMatrix fCurrentValue = SkMatrix::InvalidMatrix();
Brian Salomon8d1dcd72020-03-20 21:06:41 -0400172 GrSLType fType = kVoid_GrSLType;
egdanielfe8a8392016-05-09 10:22:19 -0700173 };
174
Brian Salomon510eb152020-02-27 13:52:16 -0500175 SkTArray<TransformVarying, true> fVaryingTransform;
176 SkTArray<TransformUniform, true> fUniformTransform;
egdanielfe8a8392016-05-09 10:22:19 -0700177
joshualitt102081a2015-09-11 11:52:17 -0700178 UniformHandle fColorUniform;
Brian Osmancf860852018-10-31 14:04:39 -0400179 SkPMColor4f fColor;
joshualitt102081a2015-09-11 11:52:17 -0700180
egdaniele659a582015-11-13 09:55:43 -0800181 typedef GrGLSLPrimitiveProcessor INHERITED;
joshualitt102081a2015-09-11 11:52:17 -0700182};
183
Brian Osmancf860852018-10-31 14:04:39 -0400184GrPathProcessor::GrPathProcessor(const SkPMColor4f& color,
joshualitt8072caa2015-02-12 14:20:52 -0800185 const SkMatrix& viewMatrix,
186 const SkMatrix& localMatrix)
Ethan Nicholasabff9562017-10-09 10:54:08 -0400187 : INHERITED(kGrPathProcessor_ClassID)
188 , fColor(color)
Brian Salomon92aee3d2016-12-21 09:20:25 -0500189 , fViewMatrix(viewMatrix)
Ethan Nicholasabff9562017-10-09 10:54:08 -0400190 , fLocalMatrix(localMatrix) {}
joshualitt8072caa2015-02-12 14:20:52 -0800191
Brian Salomon94efbf52016-11-29 13:43:05 -0500192void GrPathProcessor::getGLSLProcessorKey(const GrShaderCaps& caps,
egdaniel57d3b032015-11-13 11:57:27 -0800193 GrProcessorKeyBuilder* b) const {
joshualitt465283c2015-09-11 08:19:35 -0700194 GrGLPathProcessor::GenKey(*this, caps, b);
joshualitt8072caa2015-02-12 14:20:52 -0800195}
196
Brian Salomon94efbf52016-11-29 13:43:05 -0500197GrGLSLPrimitiveProcessor* GrPathProcessor::createGLSLInstance(const GrShaderCaps& caps) const {
jvanverth50530632015-04-27 10:36:27 -0700198 SkASSERT(caps.pathRenderingSupport());
joshualitt465283c2015-09-11 08:19:35 -0700199 return new GrGLPathProcessor();
joshualitt8072caa2015-02-12 14:20:52 -0800200}