blob: 43858443786ef1904ff4e7dc89a805f0513bad2d [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"
Michael Ludwigfbe28592020-06-26 16:02:15 -040011#include "src/gpu/GrCoordTransform.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050012#include "src/gpu/GrShaderCaps.h"
13#include "src/gpu/gl/GrGLGpu.h"
John Rosasco24cbdab2019-09-25 14:14:35 -070014#ifdef SK_GL
Mike Kleinc0bd9f92019-04-23 12:05:21 -050015#include "src/gpu/gl/GrGLVaryingHandler.h"
John Rosasco24cbdab2019-09-25 14:14:35 -070016#endif
Mike Kleinc0bd9f92019-04-23 12:05:21 -050017#include "src/gpu/glsl/GrGLSLFragmentShaderBuilder.h"
18#include "src/gpu/glsl/GrGLSLPrimitiveProcessor.h"
19#include "src/gpu/glsl/GrGLSLUniformHandler.h"
20#include "src/gpu/glsl/GrGLSLVarying.h"
jvanverthcba99b82015-06-24 06:59:57 -070021
egdaniele659a582015-11-13 09:55:43 -080022class GrGLPathProcessor : public GrGLSLPrimitiveProcessor {
joshualitt102081a2015-09-11 11:52:17 -070023public:
Brian Osmancf860852018-10-31 14:04:39 -040024 GrGLPathProcessor() : fColor(SK_PMColor4fILLEGAL) {}
joshualitt102081a2015-09-11 11:52:17 -070025
26 static void GenKey(const GrPathProcessor& pathProc,
Brian Salomon94efbf52016-11-29 13:43:05 -050027 const GrShaderCaps&,
joshualitt102081a2015-09-11 11:52:17 -070028 GrProcessorKeyBuilder* b) {
Brian Salomonbfd51832017-01-04 13:22:08 -050029 b->add32(SkToInt(pathProc.viewMatrix().hasPerspective()));
joshualitt102081a2015-09-11 11:52:17 -070030 }
31
32 void emitCode(EmitArgs& args) override {
Chris Dalton60283612018-02-14 13:38:14 -070033 GrGLSLFPFragmentBuilder* fragBuilder = args.fFragBuilder;
joshualitt102081a2015-09-11 11:52:17 -070034 const GrPathProcessor& pathProc = args.fGP.cast<GrPathProcessor>();
35
cdaltonc08f1962016-02-12 12:14:06 -080036 if (!pathProc.viewMatrix().hasPerspective()) {
37 args.fVaryingHandler->setNoPerspective();
38 }
39
joshualitt102081a2015-09-11 11:52:17 -070040 // emit transforms
Brian Salomon510eb152020-02-27 13:52:16 -050041 this->emitTransforms(args.fVaryingHandler, args.fUniformHandler,
42 args.fFPCoordTransformHandler);
joshualitt102081a2015-09-11 11:52:17 -070043
44 // Setup uniform color
Brian Salomonbfd51832017-01-04 13:22:08 -050045 const char* stagedLocalVarName;
Ethan Nicholas16464c32020-04-06 13:53:05 -040046 fColorUniform = args.fUniformHandler->addUniform(nullptr,
47 kFragment_GrShaderFlag,
Ethan Nicholasf7b88202017-09-18 14:10:39 -040048 kHalf4_GrSLType,
Brian Salomonbfd51832017-01-04 13:22:08 -050049 "Color",
50 &stagedLocalVarName);
51 fragBuilder->codeAppendf("%s = %s;", args.fOutputColor, stagedLocalVarName);
joshualitt102081a2015-09-11 11:52:17 -070052
53 // setup constant solid coverage
Ethan Nicholasf7b88202017-09-18 14:10:39 -040054 fragBuilder->codeAppendf("%s = half4(1);", args.fOutputCoverage);
joshualitt102081a2015-09-11 11:52:17 -070055 }
56
Ethan Nicholasd4efe682019-08-29 16:10:13 -040057 SkString matrix_to_sksl(const SkMatrix& m) {
58 return SkStringPrintf("float3x3(%f, %f, %f, %f, %f, %f, %f, %f, %f)", m[0], m[1], m[2],
59 m[3], m[4], m[5], m[6], m[7], m[8]);
60 }
61
egdaniel0eafe792015-11-20 14:01:22 -080062 void emitTransforms(GrGLSLVaryingHandler* varyingHandler,
Brian Salomon510eb152020-02-27 13:52:16 -050063 GrGLSLUniformHandler* uniformHandler,
bsalomona624bf32016-09-20 09:12:47 -070064 FPCoordTransformHandler* transformHandler) {
Brian Salomon7d8b3972019-11-26 22:34:44 -050065 for (int i = 0; *transformHandler; ++*transformHandler, ++i) {
66 auto [coordTransform, fp] = transformHandler->get();
joshualitt102081a2015-09-11 11:52:17 -070067
Brian Salomon510eb152020-02-27 13:52:16 -050068 GrShaderVar fragmentVar;
Brian Salomon8d1dcd72020-03-20 21:06:41 -040069 GrShaderVar transformVar;
Brian Salomon510eb152020-02-27 13:52:16 -050070 if (fp.isSampledWithExplicitCoords()) {
71 if (coordTransform.isNoOp()) {
72 transformHandler->omitCoordsForCurrCoordTransform();
73 continue;
74 } else {
75 const char* name;
76 SkString strUniName;
77 strUniName.printf("CoordTransformMatrix_%d", i);
Brian Salomon8d1dcd72020-03-20 21:06:41 -040078 auto& uni = fUniformTransform.push_back();
79 if (coordTransform.matrix().isScaleTranslate()) {
80 uni.fType = kFloat4_GrSLType;
81 } else {
82 uni.fType = kFloat3x3_GrSLType;
83 }
84 uni.fHandle = uniformHandler
Ethan Nicholas16464c32020-04-06 13:53:05 -040085 ->addUniform(nullptr,
86 kFragment_GrShaderFlag,
Brian Salomon8d1dcd72020-03-20 21:06:41 -040087 uni.fType,
88 strUniName.c_str(),
89 &name)
90 .toIndex();
91 transformVar = uniformHandler->getUniformVariable(uni.fHandle);
Brian Salomon510eb152020-02-27 13:52:16 -050092 }
93 } else {
94 SkString strVaryingName;
95 strVaryingName.printf("TransformedCoord_%d", i);
Michael Ludwige88320b2020-06-24 09:04:56 -040096 GrSLType varyingType = coordTransform.matrix().hasPerspective() ? kFloat3_GrSLType
97 : kFloat2_GrSLType;
Brian Salomon510eb152020-02-27 13:52:16 -050098 GrGLSLVarying v(varyingType);
John Rosasco24cbdab2019-09-25 14:14:35 -070099#ifdef SK_GL
Brian Salomon510eb152020-02-27 13:52:16 -0500100 GrGLVaryingHandler* glVaryingHandler = (GrGLVaryingHandler*)varyingHandler;
101 fVaryingTransform.push_back().fHandle =
102 glVaryingHandler->addPathProcessingVarying(strVaryingName.c_str(), &v)
103 .toIndex();
John Rosasco24cbdab2019-09-25 14:14:35 -0700104#endif
Brian Salomon510eb152020-02-27 13:52:16 -0500105 fVaryingTransform.back().fType = varyingType;
Brian Salomon510eb152020-02-27 13:52:16 -0500106 fragmentVar = {SkString(v.fsIn()), varyingType};
107 }
Ethan Nicholasafe2c902020-04-28 13:55:02 -0400108 transformHandler->specifyCoordsForCurrCoordTransform(transformVar, fragmentVar);
joshualitt102081a2015-09-11 11:52:17 -0700109 }
110 }
111
egdaniel018fb622015-10-28 07:26:40 -0700112 void setData(const GrGLSLProgramDataManager& pd,
bsalomona624bf32016-09-20 09:12:47 -0700113 const GrPrimitiveProcessor& primProc,
Brian Salomonc241b582019-11-27 08:57:17 -0500114 const CoordTransformRange& transformRange) override {
joshualitt102081a2015-09-11 11:52:17 -0700115 const GrPathProcessor& pathProc = primProc.cast<GrPathProcessor>();
Brian Salomonbfd51832017-01-04 13:22:08 -0500116 if (pathProc.color() != fColor) {
Brian Osmancf860852018-10-31 14:04:39 -0400117 pd.set4fv(fColorUniform, 1, pathProc.color().vec());
joshualitt102081a2015-09-11 11:52:17 -0700118 fColor = pathProc.color();
119 }
joshualitt102081a2015-09-11 11:52:17 -0700120
Brian Salomon510eb152020-02-27 13:52:16 -0500121 int v = 0, u = 0;
Brian Salomonc241b582019-11-27 08:57:17 -0500122 for (auto [transform, fp] : transformRange) {
Brian Salomonb10a6622020-02-20 13:36:01 -0500123 if (fp.isSampledWithExplicitCoords()) {
Brian Salomon510eb152020-02-27 13:52:16 -0500124 if (transform.isNoOp()) {
125 continue;
126 }
127 if (fUniformTransform[u].fHandle.isValid()) {
128 SkMatrix m = GetTransformMatrix(transform, SkMatrix::I());
129 if (!SkMatrixPriv::CheapEqual(fUniformTransform[u].fCurrentValue, m)) {
130 fUniformTransform[u].fCurrentValue = m;
Brian Salomon8d1dcd72020-03-20 21:06:41 -0400131 if (fUniformTransform[u].fType == kFloat4_GrSLType) {
132 float values[4] = {m.getScaleX(), m.getTranslateX(),
133 m.getScaleY(), m.getTranslateY()};
134 pd.set4fv(fUniformTransform[u].fHandle.toIndex(), 1, values);
135 } else {
136 SkASSERT(fUniformTransform[u].fType == kFloat3x3_GrSLType);
137 pd.setSkMatrix(fUniformTransform[u].fHandle.toIndex(), m);
138 }
Brian Salomon510eb152020-02-27 13:52:16 -0500139 }
140 }
141 ++u;
Brian Salomonb10a6622020-02-20 13:36:01 -0500142 } else {
Brian Salomon510eb152020-02-27 13:52:16 -0500143 SkASSERT(fVaryingTransform[v].fHandle.isValid());
144 SkMatrix m = GetTransformMatrix(transform, pathProc.localMatrix());
145 if (!SkMatrixPriv::CheapEqual(fVaryingTransform[v].fCurrentValue, m)) {
146 fVaryingTransform[v].fCurrentValue = m;
Michael Ludwige88320b2020-06-24 09:04:56 -0400147 SkASSERT(fVaryingTransform[v].fType == kFloat2_GrSLType ||
148 fVaryingTransform[v].fType == kFloat3_GrSLType);
149 int components = fVaryingTransform[v].fType == kFloat2_GrSLType ? 2 : 3;
Brian Salomon510eb152020-02-27 13:52:16 -0500150 pd.setPathFragmentInputTransform(fVaryingTransform[v].fHandle, components, m);
151 }
152 ++v;
Brian Salomon7eabfe82019-12-02 14:20:20 -0500153 }
joshualitt102081a2015-09-11 11:52:17 -0700154 }
155 }
156
157private:
Brian Salomon510eb152020-02-27 13:52:16 -0500158 using VaryingHandle = GrGLSLProgramDataManager::VaryingHandle;
159
160 // Varying transforms are used for non-explicitly sampled FPs. We provide a matrix
161 // to GL as fixed function state and it uses it to compute a varying that we pick up
162 // in the FS as the output of the coord transform.
bsalomon790c90b2016-09-12 12:56:58 -0700163 struct TransformVarying {
Brian Salomon510eb152020-02-27 13:52:16 -0500164 VaryingHandle fHandle;
165 SkMatrix fCurrentValue = SkMatrix::InvalidMatrix();
166 GrSLType fType = kVoid_GrSLType;
167 };
168 // For explicitly sampled FPs we stick a uniform in the FS and apply it to the explicit coords
169 // to implement the CoordTransform.
170 struct TransformUniform {
171 UniformHandle fHandle;
172 SkMatrix fCurrentValue = SkMatrix::InvalidMatrix();
Brian Salomon8d1dcd72020-03-20 21:06:41 -0400173 GrSLType fType = kVoid_GrSLType;
egdanielfe8a8392016-05-09 10:22:19 -0700174 };
175
Brian Salomon510eb152020-02-27 13:52:16 -0500176 SkTArray<TransformVarying, true> fVaryingTransform;
177 SkTArray<TransformUniform, true> fUniformTransform;
egdanielfe8a8392016-05-09 10:22:19 -0700178
joshualitt102081a2015-09-11 11:52:17 -0700179 UniformHandle fColorUniform;
Brian Osmancf860852018-10-31 14:04:39 -0400180 SkPMColor4f fColor;
joshualitt102081a2015-09-11 11:52:17 -0700181
egdaniele659a582015-11-13 09:55:43 -0800182 typedef GrGLSLPrimitiveProcessor INHERITED;
joshualitt102081a2015-09-11 11:52:17 -0700183};
184
Brian Osmancf860852018-10-31 14:04:39 -0400185GrPathProcessor::GrPathProcessor(const SkPMColor4f& color,
joshualitt8072caa2015-02-12 14:20:52 -0800186 const SkMatrix& viewMatrix,
187 const SkMatrix& localMatrix)
Ethan Nicholasabff9562017-10-09 10:54:08 -0400188 : INHERITED(kGrPathProcessor_ClassID)
189 , fColor(color)
Brian Salomon92aee3d2016-12-21 09:20:25 -0500190 , fViewMatrix(viewMatrix)
Ethan Nicholasabff9562017-10-09 10:54:08 -0400191 , fLocalMatrix(localMatrix) {}
joshualitt8072caa2015-02-12 14:20:52 -0800192
Brian Salomon94efbf52016-11-29 13:43:05 -0500193void GrPathProcessor::getGLSLProcessorKey(const GrShaderCaps& caps,
egdaniel57d3b032015-11-13 11:57:27 -0800194 GrProcessorKeyBuilder* b) const {
joshualitt465283c2015-09-11 08:19:35 -0700195 GrGLPathProcessor::GenKey(*this, caps, b);
joshualitt8072caa2015-02-12 14:20:52 -0800196}
197
Brian Salomon94efbf52016-11-29 13:43:05 -0500198GrGLSLPrimitiveProcessor* GrPathProcessor::createGLSLInstance(const GrShaderCaps& caps) const {
jvanverth50530632015-04-27 10:36:27 -0700199 SkASSERT(caps.pathRenderingSupport());
joshualitt465283c2015-09-11 08:19:35 -0700200 return new GrGLPathProcessor();
joshualitt8072caa2015-02-12 14:20:52 -0800201}