blob: 0d63776976e52af2044fc11253550eb9647d34f5 [file] [log] [blame]
joshualitt8072caa2015-02-12 14:20:52 -08001/*
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
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "src/gpu/glsl/GrGLSLGeometryProcessor.h"
joshualitt8072caa2015-02-12 14:20:52 -08009
Mike Kleinc0bd9f92019-04-23 12:05:21 -050010#include "src/gpu/GrCoordTransform.h"
Ethan Nicholas58430122020-04-14 09:54:02 -040011#include "src/gpu/GrPipeline.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050012#include "src/gpu/glsl/GrGLSLFragmentShaderBuilder.h"
13#include "src/gpu/glsl/GrGLSLUniformHandler.h"
14#include "src/gpu/glsl/GrGLSLVarying.h"
15#include "src/gpu/glsl/GrGLSLVertexGeoBuilder.h"
joshualitt8072caa2015-02-12 14:20:52 -080016
Ethan Nicholasd3a95c22020-06-03 13:24:46 -040017#include <unordered_map>
18
egdaniele659a582015-11-13 09:55:43 -080019void GrGLSLGeometryProcessor::emitCode(EmitArgs& args) {
joshualitt8072caa2015-02-12 14:20:52 -080020 GrGPArgs gpArgs;
21 this->onEmitCode(args, &gpArgs);
Chris Daltonc17bf322017-10-24 10:59:03 -060022
Chris Dalton5a2f9622019-12-27 14:56:38 -070023 if (args.fGP.willUseTessellationShaders()) {
24 // Tessellation shaders are temporarily responsible for integrating their own code strings
25 // while we work out full support.
26 return;
27 }
28
Chris Daltonc17bf322017-10-24 10:59:03 -060029 GrGLSLVertexBuilder* vBuilder = args.fVertBuilder;
30 if (!args.fGP.willUseGeoShader()) {
31 // Emit the vertex position to the hardware in the normalized window coordinates it expects.
Chris Dalton23261772017-12-10 16:41:45 -070032 SkASSERT(kFloat2_GrSLType == gpArgs.fPositionVar.getType() ||
33 kFloat3_GrSLType == gpArgs.fPositionVar.getType());
Chris Daltonc17bf322017-10-24 10:59:03 -060034 vBuilder->emitNormalizedSkPosition(gpArgs.fPositionVar.c_str(), args.fRTAdjustName,
35 gpArgs.fPositionVar.getType());
Chris Dalton23261772017-12-10 16:41:45 -070036 if (kFloat2_GrSLType == gpArgs.fPositionVar.getType()) {
37 args.fVaryingHandler->setNoPerspective();
38 }
Chris Daltonc17bf322017-10-24 10:59:03 -060039 } else {
40 // Since we have a geometry shader, leave the vertex position in Skia device space for now.
41 // The geometry Shader will operate in device space, and then convert the final positions to
42 // normalized hardware window coordinates under the hood, once everything else has finished.
Chris Dalton23261772017-12-10 16:41:45 -070043 // The subclass must call setNoPerspective on the varying handler, if applicable.
Chris Daltonc17bf322017-10-24 10:59:03 -060044 vBuilder->codeAppendf("sk_Position = float4(%s", gpArgs.fPositionVar.c_str());
Chris Dalton23261772017-12-10 16:41:45 -070045 switch (gpArgs.fPositionVar.getType()) {
46 case kFloat_GrSLType:
John Stiles30212b72020-06-11 17:55:07 -040047 vBuilder->codeAppend(", 0");
48 [[fallthrough]];
Chris Dalton23261772017-12-10 16:41:45 -070049 case kFloat2_GrSLType:
John Stiles30212b72020-06-11 17:55:07 -040050 vBuilder->codeAppend(", 0");
51 [[fallthrough]];
Chris Dalton23261772017-12-10 16:41:45 -070052 case kFloat3_GrSLType:
John Stiles30212b72020-06-11 17:55:07 -040053 vBuilder->codeAppend(", 1");
54 [[fallthrough]];
Chris Dalton23261772017-12-10 16:41:45 -070055 case kFloat4_GrSLType:
56 vBuilder->codeAppend(");");
57 break;
58 default:
59 SK_ABORT("Invalid position var type");
60 break;
Chris Daltonc17bf322017-10-24 10:59:03 -060061 }
cdaltonc08f1962016-02-12 12:14:06 -080062 }
joshualitt8072caa2015-02-12 14:20:52 -080063}
64
egdaniel7ea439b2015-12-03 09:20:44 -080065void GrGLSLGeometryProcessor::emitTransforms(GrGLSLVertexBuilder* vb,
egdaniel0eafe792015-11-20 14:01:22 -080066 GrGLSLVaryingHandler* varyingHandler,
egdaniel7ea439b2015-12-03 09:20:44 -080067 GrGLSLUniformHandler* uniformHandler,
Brian Salomon04460cc2017-12-06 14:47:42 -050068 const GrShaderVar& localCoordsVar,
egdaniele659a582015-11-13 09:55:43 -080069 const SkMatrix& localMatrix,
bsalomona624bf32016-09-20 09:12:47 -070070 FPCoordTransformHandler* handler) {
Brian Salomon706851d2020-02-20 14:18:00 -050071 // We only require localCoordsVar to be valid if there is a coord transform that needs
72 // it. CTs on FPs called with explicit coords do not require a local coord.
73 auto getLocalCoords = [&localCoordsVar,
74 localCoords = SkString(),
75 localCoordLength = int()]() mutable {
76 if (localCoords.isEmpty()) {
77 localCoordLength = GrSLTypeVecLength(localCoordsVar.getType());
78 SkASSERT(GrSLTypeIsFloatType(localCoordsVar.getType()));
79 SkASSERT(localCoordLength == 2 || localCoordLength == 3);
80 if (localCoordLength == 3) {
81 localCoords = localCoordsVar.getName();
82 } else {
83 localCoords.printf("float3(%s, 1)", localCoordsVar.c_str());
84 }
85 }
86 return std::make_tuple(localCoords, localCoordLength);
87 };
Brian Salomon04460cc2017-12-06 14:47:42 -050088
Brian Salomon8d1dcd72020-03-20 21:06:41 -040089 GrShaderVar transformVar;
Brian Salomon7d8b3972019-11-26 22:34:44 -050090 for (int i = 0; *handler; ++*handler, ++i) {
91 auto [coordTransform, fp] = handler->get();
Brian Salomon706851d2020-02-20 14:18:00 -050092 // Add uniform for coord transform matrix.
Ethan Nicholas58430122020-04-14 09:54:02 -040093 SkString matrix;
Brian Salomon706851d2020-02-20 14:18:00 -050094 if (!fp.isSampledWithExplicitCoords() || !coordTransform.isNoOp()) {
Brian Salomon7eabfe82019-12-02 14:20:20 -050095 SkString strUniName;
96 strUniName.printf("CoordTransformMatrix_%d", i);
Brian Salomon2fade722020-03-20 12:35:32 -040097 auto flag = fp.isSampledWithExplicitCoords() ? kFragment_GrShaderFlag
98 : kVertex_GrShaderFlag;
Brian Salomon8d1dcd72020-03-20 21:06:41 -040099 auto& uni = fInstalledTransforms.push_back();
Brian Salomon14ed8852020-03-24 10:43:38 -0400100 if (fp.isSampledWithExplicitCoords() && coordTransform.matrix().isScaleTranslate()) {
Brian Salomon8d1dcd72020-03-20 21:06:41 -0400101 uni.fType = kFloat4_GrSLType;
102 } else {
103 uni.fType = kFloat3x3_GrSLType;
104 }
Ethan Nicholas58430122020-04-14 09:54:02 -0400105 const char* matrixName;
Brian Salomon8d1dcd72020-03-20 21:06:41 -0400106 uni.fHandle =
Ethan Nicholas16464c32020-04-06 13:53:05 -0400107 uniformHandler->addUniform(&fp, flag, uni.fType, strUniName.c_str(),
108 &matrixName);
Ethan Nicholas58430122020-04-14 09:54:02 -0400109 matrix = matrixName;
Brian Salomon8d1dcd72020-03-20 21:06:41 -0400110 transformVar = uniformHandler->getUniformVariable(uni.fHandle);
Brian Salomon706851d2020-02-20 14:18:00 -0500111 } else {
112 // Install a coord transform that will be skipped.
113 fInstalledTransforms.push_back();
114 handler->omitCoordsForCurrCoordTransform();
115 continue;
116 }
117
Brian Salomon8d1dcd72020-03-20 21:06:41 -0400118 GrShaderVar fsVar;
Brian Salomon706851d2020-02-20 14:18:00 -0500119 // Add varying if required and register varying and matrix uniform.
120 if (!fp.isSampledWithExplicitCoords()) {
121 auto [localCoordsStr, localCoordLength] = getLocalCoords();
122 GrGLSLVarying v(kFloat2_GrSLType);
Brian Salomon7eabfe82019-12-02 14:20:20 -0500123 if (localMatrix.hasPerspective() || coordTransform.matrix().hasPerspective() ||
Brian Salomon706851d2020-02-20 14:18:00 -0500124 localCoordLength == 3) {
125 v = GrGLSLVarying(kFloat3_GrSLType);
Ethan Nicholasd4efe682019-08-29 16:10:13 -0400126 }
Brian Salomon7eabfe82019-12-02 14:20:20 -0500127 SkString strVaryingName;
128 strVaryingName.printf("TransformedCoords_%d", i);
Brian Salomon706851d2020-02-20 14:18:00 -0500129 varyingHandler->addVarying(strVaryingName.c_str(), &v);
Brian Salomon7eabfe82019-12-02 14:20:20 -0500130
Brian Salomon8d1dcd72020-03-20 21:06:41 -0400131 SkASSERT(fInstalledTransforms.back().fType == kFloat3x3_GrSLType);
Ethan Nicholasbffad832020-05-05 14:31:55 -0400132 if (fp.sampleMatrix().fKind != SkSL::SampleMatrix::Kind::kConstantOrUniform) {
133 if (v.type() == kFloat2_GrSLType) {
134 vb->codeAppendf("%s = (%s * %s).xy;", v.vsOut(), matrix.c_str(),
135 localCoordsStr.c_str());
136 } else {
137 vb->codeAppendf("%s = %s * %s;", v.vsOut(), matrix.c_str(),
138 localCoordsStr.c_str());
139 }
Brian Salomon7eabfe82019-12-02 14:20:20 -0500140 }
Ben Wagnerdabd98c2020-03-25 15:17:18 -0400141 fsVar = GrShaderVar(SkString(v.fsIn()), v.type(), GrShaderVar::TypeModifier::In);
Ethan Nicholasafe2c902020-04-28 13:55:02 -0400142 fTransformInfos.push_back({ v.vsOut(), v.type(), matrix, localCoordsStr, &fp });
joshualitt8072caa2015-02-12 14:20:52 -0800143 }
Ethan Nicholasafe2c902020-04-28 13:55:02 -0400144 handler->specifyCoordsForCurrCoordTransform(transformVar, fsVar);
Ethan Nicholas58430122020-04-14 09:54:02 -0400145 }
146}
147
148void GrGLSLGeometryProcessor::emitTransformCode(GrGLSLVertexBuilder* vb,
149 GrGLSLUniformHandler* uniformHandler) {
Ethan Nicholasd3a95c22020-06-03 13:24:46 -0400150 std::unordered_map<const GrFragmentProcessor*, const char*> localCoordsMap;
Ethan Nicholas58430122020-04-14 09:54:02 -0400151 for (const auto& tr : fTransformInfos) {
152 switch (tr.fFP->sampleMatrix().fKind) {
Ethan Nicholasd3a95c22020-06-03 13:24:46 -0400153 case SkSL::SampleMatrix::Kind::kConstantOrUniform: {
154 SkString localCoords;
155 localCoordsMap.insert({ tr.fFP, tr.fName });
156 if (tr.fFP->sampleMatrix().fBase) {
157 SkASSERT(localCoordsMap[tr.fFP->sampleMatrix().fBase]);
158 localCoords = SkStringPrintf("float3(%s, 1)",
159 localCoordsMap[tr.fFP->sampleMatrix().fBase]);
160 } else {
161 localCoords = tr.fLocalCoords.c_str();
162 }
Ethan Nicholas58430122020-04-14 09:54:02 -0400163 vb->codeAppend("{\n");
164 uniformHandler->writeUniformMappings(tr.fFP->sampleMatrix().fOwner, vb);
165 if (tr.fType == kFloat2_GrSLType) {
166 vb->codeAppendf("%s = (%s * %s * %s).xy", tr.fName,
Ethan Nicholasafe2c902020-04-28 13:55:02 -0400167 tr.fFP->sampleMatrix().fExpression.c_str(), tr.fMatrix.c_str(),
Ethan Nicholasd3a95c22020-06-03 13:24:46 -0400168 localCoords.c_str());
Ethan Nicholas58430122020-04-14 09:54:02 -0400169 } else {
170 SkASSERT(tr.fType == kFloat3_GrSLType);
171 vb->codeAppendf("%s = %s * %s * %s", tr.fName,
Ethan Nicholasafe2c902020-04-28 13:55:02 -0400172 tr.fFP->sampleMatrix().fExpression.c_str(), tr.fMatrix.c_str(),
Ethan Nicholasd3a95c22020-06-03 13:24:46 -0400173 localCoords.c_str());
Ethan Nicholas58430122020-04-14 09:54:02 -0400174 }
175 vb->codeAppend(";\n");
176 vb->codeAppend("}\n");
John Stiles30212b72020-06-11 17:55:07 -0400177 break;
Ethan Nicholasd3a95c22020-06-03 13:24:46 -0400178 }
Ethan Nicholas58430122020-04-14 09:54:02 -0400179 default:
180 break;
181 }
joshualitt8072caa2015-02-12 14:20:52 -0800182 }
183}
184
bsalomona624bf32016-09-20 09:12:47 -0700185void GrGLSLGeometryProcessor::setTransformDataHelper(const SkMatrix& localMatrix,
186 const GrGLSLProgramDataManager& pdman,
Brian Salomonc241b582019-11-27 08:57:17 -0500187 const CoordTransformRange& transformRange) {
bsalomona624bf32016-09-20 09:12:47 -0700188 int i = 0;
Brian Salomonc241b582019-11-27 08:57:17 -0500189 for (auto [transform, fp] : transformRange) {
Brian Salomon7eabfe82019-12-02 14:20:20 -0500190 if (fInstalledTransforms[i].fHandle.isValid()) {
191 SkMatrix m;
Brian Salomonb10a6622020-02-20 13:36:01 -0500192 if (fp.isSampledWithExplicitCoords()) {
Brian Salomon7eabfe82019-12-02 14:20:20 -0500193 m = GetTransformMatrix(transform, SkMatrix::I());
Brian Salomonb10a6622020-02-20 13:36:01 -0500194 } else {
195 m = GetTransformMatrix(transform, localMatrix);
Brian Salomon7eabfe82019-12-02 14:20:20 -0500196 }
Mike Reed2c383152019-12-18 16:47:47 -0500197 if (!SkMatrixPriv::CheapEqual(fInstalledTransforms[i].fCurrentValue, m)) {
Brian Salomon8d1dcd72020-03-20 21:06:41 -0400198 if (fInstalledTransforms[i].fType == kFloat4_GrSLType) {
199 float values[4] = {m.getScaleX(), m.getTranslateX(),
200 m.getScaleY(), m.getTranslateY()};
Brian Salomon14ed8852020-03-24 10:43:38 -0400201 SkASSERT(m.isScaleTranslate());
Brian Salomon8d1dcd72020-03-20 21:06:41 -0400202 pdman.set4fv(fInstalledTransforms[i].fHandle.toIndex(), 1, values);
203 } else {
Brian Salomon14ed8852020-03-24 10:43:38 -0400204 SkASSERT(!m.isScaleTranslate() || !fp.isSampledWithExplicitCoords());
Brian Salomon8d1dcd72020-03-20 21:06:41 -0400205 SkASSERT(fInstalledTransforms[i].fType == kFloat3x3_GrSLType);
206 pdman.setSkMatrix(fInstalledTransforms[i].fHandle.toIndex(), m);
207 }
Brian Salomon7eabfe82019-12-02 14:20:20 -0500208 fInstalledTransforms[i].fCurrentValue = m;
209 }
bsalomona624bf32016-09-20 09:12:47 -0700210 }
211 ++i;
212 }
213 SkASSERT(i == fInstalledTransforms.count());
214}
215
Brian Salomon7f235432017-08-16 09:41:48 -0400216void GrGLSLGeometryProcessor::writeOutputPosition(GrGLSLVertexBuilder* vertBuilder,
217 GrGPArgs* gpArgs,
218 const char* posName) {
Ethan Nicholas8aa45692017-09-20 11:24:15 -0400219 gpArgs->fPositionVar.set(kFloat2_GrSLType, "pos2");
220 vertBuilder->codeAppendf("float2 %s = %s;", gpArgs->fPositionVar.c_str(), posName);
joshualitt5559ca22015-05-21 15:50:36 -0700221}
222
Brian Salomon7f235432017-08-16 09:41:48 -0400223void GrGLSLGeometryProcessor::writeOutputPosition(GrGLSLVertexBuilder* vertBuilder,
224 GrGLSLUniformHandler* uniformHandler,
225 GrGPArgs* gpArgs,
226 const char* posName,
227 const SkMatrix& mat,
228 UniformHandle* viewMatrixUniform) {
joshualitt8072caa2015-02-12 14:20:52 -0800229 if (mat.isIdentity()) {
Ethan Nicholas8aa45692017-09-20 11:24:15 -0400230 gpArgs->fPositionVar.set(kFloat2_GrSLType, "pos2");
231 vertBuilder->codeAppendf("float2 %s = %s;", gpArgs->fPositionVar.c_str(), posName);
joshualitt8072caa2015-02-12 14:20:52 -0800232 } else {
joshualitt5559ca22015-05-21 15:50:36 -0700233 const char* viewMatrixName;
Ethan Nicholas16464c32020-04-06 13:53:05 -0400234 *viewMatrixUniform = uniformHandler->addUniform(nullptr,
235 kVertex_GrShaderFlag,
Ethan Nicholas8aa45692017-09-20 11:24:15 -0400236 kFloat3x3_GrSLType,
egdaniel7ea439b2015-12-03 09:20:44 -0800237 "uViewM",
238 &viewMatrixName);
joshualitt5559ca22015-05-21 15:50:36 -0700239 if (!mat.hasPerspective()) {
Ethan Nicholas8aa45692017-09-20 11:24:15 -0400240 gpArgs->fPositionVar.set(kFloat2_GrSLType, "pos2");
241 vertBuilder->codeAppendf("float2 %s = (%s * float3(%s, 1)).xy;",
egdaniel4ca2e602015-11-18 08:01:26 -0800242 gpArgs->fPositionVar.c_str(), viewMatrixName, posName);
joshualitt5559ca22015-05-21 15:50:36 -0700243 } else {
Ethan Nicholas8aa45692017-09-20 11:24:15 -0400244 gpArgs->fPositionVar.set(kFloat3_GrSLType, "pos3");
245 vertBuilder->codeAppendf("float3 %s = %s * float3(%s, 1);",
egdaniel4ca2e602015-11-18 08:01:26 -0800246 gpArgs->fPositionVar.c_str(), viewMatrixName, posName);
joshualitt5559ca22015-05-21 15:50:36 -0700247 }
joshualitt8072caa2015-02-12 14:20:52 -0800248 }
249}