blob: e42b335e42bea13fbba4dcfcb4b14a7dc83ee4ed [file] [log] [blame]
Ethan Nicholas762466e2017-06-29 10:03:38 -04001/*
2 * Copyright 2016 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/sksl/SkSLCPPCodeGenerator.h"
Ethan Nicholas762466e2017-06-29 10:03:38 -04009
Mike Kleinc0bd9f92019-04-23 12:05:21 -050010#include "src/sksl/SkSLCPPUniformCTypes.h"
11#include "src/sksl/SkSLCompiler.h"
12#include "src/sksl/SkSLHCodeGenerator.h"
Ethan Nicholas762466e2017-06-29 10:03:38 -040013
Michael Ludwig92e4c7f2018-08-30 16:08:18 -040014#include <algorithm>
15
Ethan Nicholas762466e2017-06-29 10:03:38 -040016namespace SkSL {
17
18static bool needs_uniform_var(const Variable& var) {
Ethan Nicholas5f9836e2017-12-20 15:16:33 -050019 return (var.fModifiers.fFlags & Modifiers::kUniform_Flag) &&
20 var.fType.kind() != Type::kSampler_Kind;
Ethan Nicholas762466e2017-06-29 10:03:38 -040021}
22
23CPPCodeGenerator::CPPCodeGenerator(const Context* context, const Program* program,
24 ErrorReporter* errors, String name, OutputStream* out)
25: INHERITED(context, program, errors, out)
26, fName(std::move(name))
27, fFullName(String::printf("Gr%s", fName.c_str()))
Ethan Nicholasd4efe682019-08-29 16:10:13 -040028, fSectionAndParameterHelper(program, *errors) {
Ethan Nicholas762466e2017-06-29 10:03:38 -040029 fLineEnding = "\\n";
Ethan Nicholas13863662019-07-29 13:05:15 -040030 fTextureFunctionOverride = "sample";
Ethan Nicholas762466e2017-06-29 10:03:38 -040031}
32
33void CPPCodeGenerator::writef(const char* s, va_list va) {
34 static constexpr int BUFFER_SIZE = 1024;
Ethan Nicholas9fb036f2017-07-05 16:19:09 -040035 va_list copy;
36 va_copy(copy, va);
Ethan Nicholas762466e2017-06-29 10:03:38 -040037 char buffer[BUFFER_SIZE];
38 int length = vsnprintf(buffer, BUFFER_SIZE, s, va);
39 if (length < BUFFER_SIZE) {
40 fOut->write(buffer, length);
41 } else {
42 std::unique_ptr<char[]> heap(new char[length + 1]);
Ethan Nicholas9fb036f2017-07-05 16:19:09 -040043 vsprintf(heap.get(), s, copy);
Ethan Nicholas762466e2017-06-29 10:03:38 -040044 fOut->write(heap.get(), length);
45 }
z102.zhangd74f2c82018-08-10 09:08:47 +080046 va_end(copy);
Ethan Nicholas762466e2017-06-29 10:03:38 -040047}
48
49void CPPCodeGenerator::writef(const char* s, ...) {
50 va_list va;
51 va_start(va, s);
52 this->writef(s, va);
53 va_end(va);
54}
55
56void CPPCodeGenerator::writeHeader() {
57}
58
Ethan Nicholasf7b88202017-09-18 14:10:39 -040059bool CPPCodeGenerator::usesPrecisionModifiers() const {
60 return false;
Ethan Nicholas762466e2017-06-29 10:03:38 -040061}
62
Ethan Nicholasf7b88202017-09-18 14:10:39 -040063String CPPCodeGenerator::getTypeName(const Type& type) {
64 return type.name();
Ethan Nicholas5af9ea32017-07-28 15:19:46 -040065}
Ethan Nicholasf7b88202017-09-18 14:10:39 -040066
Ethan Nicholas762466e2017-06-29 10:03:38 -040067void CPPCodeGenerator::writeBinaryExpression(const BinaryExpression& b,
68 Precedence parentPrecedence) {
69 if (b.fOperator == Token::PERCENT) {
70 // need to use "%%" instead of "%" b/c the code will be inside of a printf
71 Precedence precedence = GetBinaryPrecedence(b.fOperator);
72 if (precedence >= parentPrecedence) {
73 this->write("(");
74 }
75 this->writeExpression(*b.fLeft, precedence);
76 this->write(" %% ");
77 this->writeExpression(*b.fRight, precedence);
78 if (precedence >= parentPrecedence) {
79 this->write(")");
80 }
Ethan Nicholasee1c8a72019-02-22 10:50:47 -050081 } else if (b.fLeft->fKind == Expression::kNullLiteral_Kind ||
82 b.fRight->fKind == Expression::kNullLiteral_Kind) {
83 const Variable* var;
84 if (b.fLeft->fKind != Expression::kNullLiteral_Kind) {
85 SkASSERT(b.fLeft->fKind == Expression::kVariableReference_Kind);
86 var = &((VariableReference&) *b.fLeft).fVariable;
87 } else {
88 SkASSERT(b.fRight->fKind == Expression::kVariableReference_Kind);
89 var = &((VariableReference&) *b.fRight).fVariable;
90 }
91 SkASSERT(var->fType.kind() == Type::kNullable_Kind &&
92 var->fType.componentType() == *fContext.fFragmentProcessor_Type);
93 this->write("%s");
94 const char* op;
95 switch (b.fOperator) {
96 case Token::EQEQ:
97 op = "<";
98 break;
99 case Token::NEQ:
100 op = ">=";
101 break;
102 default:
103 SkASSERT(false);
104 }
Ethan Nicholasbcd51e82019-04-09 10:40:41 -0400105 fFormatArgs.push_back("_outer." + String(var->fName) + "_index " + op + " 0 ? \"true\" "
Ethan Nicholasee1c8a72019-02-22 10:50:47 -0500106 ": \"false\"");
Ethan Nicholas762466e2017-06-29 10:03:38 -0400107 } else {
108 INHERITED::writeBinaryExpression(b, parentPrecedence);
109 }
110}
111
112void CPPCodeGenerator::writeIndexExpression(const IndexExpression& i) {
113 const Expression& base = *i.fBase;
114 if (base.fKind == Expression::kVariableReference_Kind) {
115 int builtin = ((VariableReference&) base).fVariable.fModifiers.fLayout.fBuiltin;
116 if (SK_TRANSFORMEDCOORDS2D_BUILTIN == builtin) {
117 this->write("%s");
118 if (i.fIndex->fKind != Expression::kIntLiteral_Kind) {
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700119 fErrors.error(i.fIndex->fOffset,
Ethan Nicholas762466e2017-06-29 10:03:38 -0400120 "index into sk_TransformedCoords2D must be an integer literal");
121 return;
122 }
123 int64_t index = ((IntLiteral&) *i.fIndex).fValue;
124 String name = "sk_TransformedCoords2D_" + to_string(index);
Brian Salomonbf5c0c02019-11-11 14:55:28 -0500125 fFormatArgs.push_back(name + ".c_str()");
Ethan Nicholas762466e2017-06-29 10:03:38 -0400126 if (fWrittenTransformedCoords.find(index) == fWrittenTransformedCoords.end()) {
Michael Ludwig1fc5fbd2018-09-07 13:13:06 -0400127 addExtraEmitCodeLine("SkString " + name +
128 " = fragBuilder->ensureCoords2D(args.fTransformedCoords[" +
Ethan Nicholasd4efe682019-08-29 16:10:13 -0400129 to_string(index) + "].fVaryingPoint);");
Ethan Nicholas762466e2017-06-29 10:03:38 -0400130 fWrittenTransformedCoords.insert(index);
131 }
132 return;
133 } else if (SK_TEXTURESAMPLERS_BUILTIN == builtin) {
134 this->write("%s");
135 if (i.fIndex->fKind != Expression::kIntLiteral_Kind) {
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700136 fErrors.error(i.fIndex->fOffset,
Ethan Nicholas762466e2017-06-29 10:03:38 -0400137 "index into sk_TextureSamplers must be an integer literal");
138 return;
139 }
140 int64_t index = ((IntLiteral&) *i.fIndex).fValue;
141 fFormatArgs.push_back(" fragBuilder->getProgramBuilder()->samplerVariable("
Stephen Whited523a062019-06-19 13:12:46 -0400142 "args.fTexSamplers[" + to_string(index) + "])");
Ethan Nicholas762466e2017-06-29 10:03:38 -0400143 return;
144 }
145 }
146 INHERITED::writeIndexExpression(i);
147}
148
Ethan Nicholasf7b88202017-09-18 14:10:39 -0400149static String default_value(const Type& type) {
Ethan Nicholase9d172a2017-11-20 12:12:24 -0500150 if (type.fName == "bool") {
151 return "false";
152 }
Ethan Nicholasf7b88202017-09-18 14:10:39 -0400153 switch (type.kind()) {
154 case Type::kScalar_Kind: return "0";
155 case Type::kVector_Kind: return type.name() + "(0)";
156 case Type::kMatrix_Kind: return type.name() + "(1)";
157 default: ABORT("unsupported default_value type\n");
158 }
Ethan Nicholas762466e2017-06-29 10:03:38 -0400159}
160
Ethan Nicholase9d172a2017-11-20 12:12:24 -0500161static String default_value(const Variable& var) {
Brian Osman495993a2018-10-16 15:45:55 -0400162 if (var.fModifiers.fLayout.fCType == SkSL::Layout::CType::kSkPMColor4f) {
Brian Osmanf28e55d2018-10-03 16:35:54 -0400163 return "{SK_FloatNaN, SK_FloatNaN, SK_FloatNaN, SK_FloatNaN}";
Ethan Nicholase9d172a2017-11-20 12:12:24 -0500164 }
165 return default_value(var.fType);
166}
167
Ethan Nicholas762466e2017-06-29 10:03:38 -0400168static bool is_private(const Variable& var) {
169 return !(var.fModifiers.fFlags & Modifiers::kUniform_Flag) &&
170 !(var.fModifiers.fFlags & Modifiers::kIn_Flag) &&
171 var.fStorage == Variable::kGlobal_Storage &&
172 var.fModifiers.fLayout.fBuiltin == -1;
173}
174
Michael Ludwiga4275592018-08-31 10:52:47 -0400175static bool is_uniform_in(const Variable& var) {
176 return (var.fModifiers.fFlags & Modifiers::kUniform_Flag) &&
177 (var.fModifiers.fFlags & Modifiers::kIn_Flag) &&
178 var.fType.kind() != Type::kSampler_Kind;
179}
180
Ethan Nicholasd608c092017-10-26 09:30:08 -0400181void CPPCodeGenerator::writeRuntimeValue(const Type& type, const Layout& layout,
182 const String& cppCode) {
Ethan Nicholasf7b88202017-09-18 14:10:39 -0400183 if (type.isFloat()) {
Ethan Nicholas762466e2017-06-29 10:03:38 -0400184 this->write("%f");
185 fFormatArgs.push_back(cppCode);
186 } else if (type == *fContext.fInt_Type) {
187 this->write("%d");
188 fFormatArgs.push_back(cppCode);
189 } else if (type == *fContext.fBool_Type) {
190 this->write("%s");
191 fFormatArgs.push_back("(" + cppCode + " ? \"true\" : \"false\")");
Ethan Nicholasf7b88202017-09-18 14:10:39 -0400192 } else if (type == *fContext.fFloat2_Type || type == *fContext.fHalf2_Type) {
193 this->write(type.name() + "(%f, %f)");
Ethan Nicholas762466e2017-06-29 10:03:38 -0400194 fFormatArgs.push_back(cppCode + ".fX");
195 fFormatArgs.push_back(cppCode + ".fY");
Ethan Nicholas82399462017-10-16 12:35:44 -0400196 } else if (type == *fContext.fFloat4_Type || type == *fContext.fHalf4_Type) {
197 this->write(type.name() + "(%f, %f, %f, %f)");
Ethan Nicholas78aceb22018-08-31 16:13:58 -0400198 switch (layout.fCType) {
199 case Layout::CType::kSkPMColor:
200 fFormatArgs.push_back("SkGetPackedR32(" + cppCode + ") / 255.0");
201 fFormatArgs.push_back("SkGetPackedG32(" + cppCode + ") / 255.0");
202 fFormatArgs.push_back("SkGetPackedB32(" + cppCode + ") / 255.0");
203 fFormatArgs.push_back("SkGetPackedA32(" + cppCode + ") / 255.0");
204 break;
Brian Osmanf28e55d2018-10-03 16:35:54 -0400205 case Layout::CType::kSkPMColor4f:
206 fFormatArgs.push_back(cppCode + ".fR");
207 fFormatArgs.push_back(cppCode + ".fG");
208 fFormatArgs.push_back(cppCode + ".fB");
209 fFormatArgs.push_back(cppCode + ".fA");
210 break;
Mike Reedf79aacb2020-01-22 18:05:16 +0000211 case Layout::CType::kSkVector4:
212 fFormatArgs.push_back(cppCode + ".fData[0]");
213 fFormatArgs.push_back(cppCode + ".fData[1]");
214 fFormatArgs.push_back(cppCode + ".fData[2]");
215 fFormatArgs.push_back(cppCode + ".fData[3]");
Brian Salomoneca66b32019-06-01 11:18:15 -0400216 break;
Ethan Nicholas78aceb22018-08-31 16:13:58 -0400217 case Layout::CType::kSkRect: // fall through
218 case Layout::CType::kDefault:
219 fFormatArgs.push_back(cppCode + ".left()");
220 fFormatArgs.push_back(cppCode + ".top()");
221 fFormatArgs.push_back(cppCode + ".right()");
222 fFormatArgs.push_back(cppCode + ".bottom()");
223 break;
224 default:
225 SkASSERT(false);
Ethan Nicholasd608c092017-10-26 09:30:08 -0400226 }
Ethan Nicholasaae47c82017-11-10 15:34:03 -0500227 } else if (type.kind() == Type::kEnum_Kind) {
228 this->write("%d");
229 fFormatArgs.push_back("(int) " + cppCode);
Ruiqi Maob609e6d2018-07-17 10:19:38 -0400230 } else if (type == *fContext.fInt4_Type ||
231 type == *fContext.fShort4_Type ||
232 type == *fContext.fByte4_Type) {
Ethan Nicholas2d5f9b32017-12-13 14:36:14 -0500233 this->write(type.name() + "(%d, %d, %d, %d)");
234 fFormatArgs.push_back(cppCode + ".left()");
235 fFormatArgs.push_back(cppCode + ".top()");
236 fFormatArgs.push_back(cppCode + ".right()");
237 fFormatArgs.push_back(cppCode + ".bottom()");
Ethan Nicholas762466e2017-06-29 10:03:38 -0400238 } else {
Ethan Nicholas82399462017-10-16 12:35:44 -0400239 printf("unsupported runtime value type '%s'\n", String(type.fName).c_str());
Ethan Nicholasd9d33c32018-06-12 11:05:59 -0400240 SkASSERT(false);
Ethan Nicholas762466e2017-06-29 10:03:38 -0400241 }
242}
243
244void CPPCodeGenerator::writeVarInitializer(const Variable& var, const Expression& value) {
245 if (is_private(var)) {
Ethan Nicholasd608c092017-10-26 09:30:08 -0400246 this->writeRuntimeValue(var.fType, var.fModifiers.fLayout, var.fName);
Ethan Nicholas762466e2017-06-29 10:03:38 -0400247 } else {
248 this->writeExpression(value, kTopLevel_Precedence);
249 }
250}
251
Ethan Nicholasceb4d482017-07-10 15:40:20 -0400252String CPPCodeGenerator::getSamplerHandle(const Variable& var) {
253 int samplerCount = 0;
Ethan Nicholas68990be2017-07-13 09:36:52 -0400254 for (const auto param : fSectionAndParameterHelper.getParameters()) {
Ethan Nicholasceb4d482017-07-10 15:40:20 -0400255 if (&var == param) {
256 return "args.fTexSamplers[" + to_string(samplerCount) + "]";
257 }
258 if (param->fType.kind() == Type::kSampler_Kind) {
259 ++samplerCount;
260 }
261 }
262 ABORT("should have found sampler in parameters\n");
263}
264
Ethan Nicholasdcba08e2017-08-02 10:52:54 -0400265void CPPCodeGenerator::writeIntLiteral(const IntLiteral& i) {
266 this->write(to_string((int32_t) i.fValue));
267}
268
Ethan Nicholas82399462017-10-16 12:35:44 -0400269void CPPCodeGenerator::writeSwizzle(const Swizzle& swizzle) {
270 if (fCPPMode) {
Ethan Nicholasd9d33c32018-06-12 11:05:59 -0400271 SkASSERT(swizzle.fComponents.size() == 1); // no support for multiple swizzle components yet
Ethan Nicholas82399462017-10-16 12:35:44 -0400272 this->writeExpression(*swizzle.fBase, kPostfix_Precedence);
273 switch (swizzle.fComponents[0]) {
274 case 0: this->write(".left()"); break;
275 case 1: this->write(".top()"); break;
276 case 2: this->write(".right()"); break;
277 case 3: this->write(".bottom()"); break;
278 }
279 } else {
280 INHERITED::writeSwizzle(swizzle);
281 }
282}
283
Ethan Nicholas762466e2017-06-29 10:03:38 -0400284void CPPCodeGenerator::writeVariableReference(const VariableReference& ref) {
Ethan Nicholas82399462017-10-16 12:35:44 -0400285 if (fCPPMode) {
286 this->write(ref.fVariable.fName);
287 return;
288 }
Ethan Nicholas762466e2017-06-29 10:03:38 -0400289 switch (ref.fVariable.fModifiers.fLayout.fBuiltin) {
290 case SK_INCOLOR_BUILTIN:
291 this->write("%s");
Michael Ludwig231de032018-08-30 14:33:01 -0400292 // EmitArgs.fInputColor is automatically set to half4(1) if
293 // no input was specified
294 fFormatArgs.push_back(String("args.fInputColor"));
Ethan Nicholas762466e2017-06-29 10:03:38 -0400295 break;
296 case SK_OUTCOLOR_BUILTIN:
297 this->write("%s");
298 fFormatArgs.push_back(String("args.fOutputColor"));
299 break;
Ethan Nicholascd700e92018-08-24 16:43:57 -0400300 case SK_WIDTH_BUILTIN:
301 this->write("sk_Width");
302 break;
303 case SK_HEIGHT_BUILTIN:
304 this->write("sk_Height");
305 break;
Ethan Nicholas762466e2017-06-29 10:03:38 -0400306 default:
307 if (ref.fVariable.fType.kind() == Type::kSampler_Kind) {
Ethan Nicholasceb4d482017-07-10 15:40:20 -0400308 this->write("%s");
309 fFormatArgs.push_back("fragBuilder->getProgramBuilder()->samplerVariable(" +
Stephen Whited523a062019-06-19 13:12:46 -0400310 this->getSamplerHandle(ref.fVariable) + ")");
Ethan Nicholasceb4d482017-07-10 15:40:20 -0400311 return;
Ethan Nicholas762466e2017-06-29 10:03:38 -0400312 }
313 if (ref.fVariable.fModifiers.fFlags & Modifiers::kUniform_Flag) {
314 this->write("%s");
315 String name = ref.fVariable.fName;
Brian Osman1cb41712017-10-19 12:54:52 -0400316 String var = String::printf("args.fUniformHandler->getUniformCStr(%sVar)",
317 HCodeGenerator::FieldName(name.c_str()).c_str());
Ethan Nicholas762466e2017-06-29 10:03:38 -0400318 String code;
Ethan Nicholasfc994162019-06-06 10:04:27 -0400319 if (ref.fVariable.fModifiers.fLayout.fWhen.fLength) {
Ethan Nicholas762466e2017-06-29 10:03:38 -0400320 code = String::printf("%sVar.isValid() ? %s : \"%s\"",
321 HCodeGenerator::FieldName(name.c_str()).c_str(),
322 var.c_str(),
Ethan Nicholasf7b88202017-09-18 14:10:39 -0400323 default_value(ref.fVariable.fType).c_str());
Ethan Nicholas762466e2017-06-29 10:03:38 -0400324 } else {
325 code = var;
326 }
327 fFormatArgs.push_back(code);
328 } else if (SectionAndParameterHelper::IsParameter(ref.fVariable)) {
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700329 String name(ref.fVariable.fName);
Ethan Nicholasd608c092017-10-26 09:30:08 -0400330 this->writeRuntimeValue(ref.fVariable.fType, ref.fVariable.fModifiers.fLayout,
Ethan Nicholasbcd51e82019-04-09 10:40:41 -0400331 String::printf("_outer.%s", name.c_str()).c_str());
Ethan Nicholas762466e2017-06-29 10:03:38 -0400332 } else {
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700333 this->write(ref.fVariable.fName);
Ethan Nicholas762466e2017-06-29 10:03:38 -0400334 }
335 }
336}
337
Ethan Nicholas6e1cbc02017-07-14 10:12:15 -0400338void CPPCodeGenerator::writeIfStatement(const IfStatement& s) {
339 if (s.fIsStatic) {
340 this->write("@");
341 }
342 INHERITED::writeIfStatement(s);
343}
344
Ethan Nicholasf1b14642018-08-09 16:18:07 -0400345void CPPCodeGenerator::writeReturnStatement(const ReturnStatement& s) {
346 if (fInMain) {
347 fErrors.error(s.fOffset, "fragmentProcessor main() may not contain return statements");
348 }
349 INHERITED::writeReturnStatement(s);
350}
351
Ethan Nicholas6e1cbc02017-07-14 10:12:15 -0400352void CPPCodeGenerator::writeSwitchStatement(const SwitchStatement& s) {
353 if (s.fIsStatic) {
354 this->write("@");
355 }
356 INHERITED::writeSwitchStatement(s);
357}
358
Michael Ludwig9094f2c2018-09-07 13:44:21 -0400359void CPPCodeGenerator::writeFieldAccess(const FieldAccess& access) {
360 if (access.fBase->fType.name() == "fragmentProcessor") {
361 // Special field access on fragment processors are converted into function calls on
362 // GrFragmentProcessor's getters.
363 if (access.fBase->fKind != Expression::kVariableReference_Kind) {
364 fErrors.error(access.fBase->fOffset, "fragmentProcessor must be a reference\n");
365 return;
366 }
367
368 const Type::Field& field = fContext.fFragmentProcessor_Type->fields()[access.fFieldIndex];
Ethan Nicholasee1c8a72019-02-22 10:50:47 -0500369 const Variable& var = ((const VariableReference&) *access.fBase).fVariable;
Ethan Nicholasbcd51e82019-04-09 10:40:41 -0400370 String cppAccess = String::printf("_outer.childProcessor(_outer.%s_index).%s()",
Ethan Nicholasee1c8a72019-02-22 10:50:47 -0500371 String(var.fName).c_str(),
372 String(field.fName).c_str());
Michael Ludwig9094f2c2018-09-07 13:44:21 -0400373
374 if (fCPPMode) {
375 this->write(cppAccess.c_str());
376 } else {
377 writeRuntimeValue(*field.fType, Layout(), cppAccess);
378 }
379 return;
380 }
381 INHERITED::writeFieldAccess(access);
382}
383
Ethan Nicholasee1c8a72019-02-22 10:50:47 -0500384int CPPCodeGenerator::getChildFPIndex(const Variable& var) const {
Michael Ludwig9094f2c2018-09-07 13:44:21 -0400385 int index = 0;
386 bool found = false;
387 for (const auto& p : fProgram) {
388 if (ProgramElement::kVar_Kind == p.fKind) {
389 const VarDeclarations& decls = (const VarDeclarations&) p;
390 for (const auto& raw : decls.fVars) {
391 const VarDeclaration& decl = (VarDeclaration&) *raw;
Ethan Nicholasee1c8a72019-02-22 10:50:47 -0500392 if (decl.fVar == &var) {
Michael Ludwig9094f2c2018-09-07 13:44:21 -0400393 found = true;
Ethan Nicholasee1c8a72019-02-22 10:50:47 -0500394 } else if (decl.fVar->fType.nonnullable() == *fContext.fFragmentProcessor_Type) {
Michael Ludwig9094f2c2018-09-07 13:44:21 -0400395 ++index;
396 }
397 }
398 }
399 if (found) {
400 break;
401 }
402 }
403 SkASSERT(found);
404 return index;
405}
406
Ethan Nicholasceb4d482017-07-10 15:40:20 -0400407void CPPCodeGenerator::writeFunctionCall(const FunctionCall& c) {
Ethan Nicholas13863662019-07-29 13:05:15 -0400408 if (c.fFunction.fBuiltin && c.fFunction.fName == "sample" &&
409 c.fArguments[0]->fType.kind() != Type::Kind::kSampler_Kind) {
Michael Ludwig92e4c7f2018-08-30 16:08:18 -0400410 // Sanity checks that are detected by function definition in sksl_fp.inc
Ethan Nicholasd4efe682019-08-29 16:10:13 -0400411 SkASSERT(c.fArguments.size() >= 1 && c.fArguments.size() <= 3);
Florin Malita390f9bd2019-03-04 12:25:57 -0500412 SkASSERT("fragmentProcessor" == c.fArguments[0]->fType.name() ||
413 "fragmentProcessor?" == c.fArguments[0]->fType.name());
Michael Ludwig92e4c7f2018-08-30 16:08:18 -0400414
415 // Actually fail during compilation if arguments with valid types are
Ethan Nicholas13863662019-07-29 13:05:15 -0400416 // provided that are not variable references, since sample() is a
Michael Ludwig92e4c7f2018-08-30 16:08:18 -0400417 // special function that impacts code emission.
418 if (c.fArguments[0]->fKind != Expression::kVariableReference_Kind) {
419 fErrors.error(c.fArguments[0]->fOffset,
Ethan Nicholas13863662019-07-29 13:05:15 -0400420 "sample()'s fragmentProcessor argument must be a variable reference\n");
Michael Ludwig92e4c7f2018-08-30 16:08:18 -0400421 return;
422 }
Ethan Nicholasee1c8a72019-02-22 10:50:47 -0500423 const Variable& child = ((const VariableReference&) *c.fArguments[0]).fVariable;
Michael Ludwig92e4c7f2018-08-30 16:08:18 -0400424
Michael Ludwig1fc5fbd2018-09-07 13:13:06 -0400425 // Start a new extra emit code section so that the emitted child processor can depend on
426 // sksl variables defined in earlier sksl code.
427 this->newExtraEmitCodeBlock();
Michael Ludwig92e4c7f2018-08-30 16:08:18 -0400428
Michael Ludwig1fc5fbd2018-09-07 13:13:06 -0400429 // Set to the empty string when no input color parameter should be emitted, which means this
430 // must be properly formatted with a prefixed comma when the parameter should be inserted
Ethan Nicholasc6dce5a2019-07-24 16:51:36 -0400431 // into the invokeChild() parameter list.
Michael Ludwig92e4c7f2018-08-30 16:08:18 -0400432 String inputArg;
Ethan Nicholasd4efe682019-08-29 16:10:13 -0400433 if (c.fArguments.size() > 1 && c.fArguments[1]->fType.name() == "half4") {
Ethan Nicholasc6dce5a2019-07-24 16:51:36 -0400434 // Use the invokeChild() variant that accepts an input color, so convert the 2nd
Michael Ludwig1fc5fbd2018-09-07 13:13:06 -0400435 // argument's expression into C++ code that produces sksl stored in an SkString.
Ethan Nicholasd4efe682019-08-29 16:10:13 -0400436 String inputName = "_input" + to_string(c.fOffset);
Michael Ludwig1fc5fbd2018-09-07 13:13:06 -0400437 addExtraEmitCodeLine(convertSKSLExpressionToCPP(*c.fArguments[1], inputName));
Michael Ludwig92e4c7f2018-08-30 16:08:18 -0400438
Ethan Nicholasc6dce5a2019-07-24 16:51:36 -0400439 // invokeChild() needs a char*
Michael Ludwig92e4c7f2018-08-30 16:08:18 -0400440 inputArg = ", " + inputName + ".c_str()";
441 }
442
Ethan Nicholasd4efe682019-08-29 16:10:13 -0400443 bool hasCoords = c.fArguments.back()->fType.name() == "float2";
444
Michael Ludwig92e4c7f2018-08-30 16:08:18 -0400445 // Write the output handling after the possible input handling
Ethan Nicholas13863662019-07-29 13:05:15 -0400446 String childName = "_sample" + to_string(c.fOffset);
Ethan Nicholas6ad52892019-05-03 13:13:42 +0000447 addExtraEmitCodeLine("SkString " + childName + "(\"" + childName + "\");");
Ethan Nicholasd4efe682019-08-29 16:10:13 -0400448 String coordsName;
449 if (hasCoords) {
450 coordsName = "_coords" + to_string(c.fOffset);
451 addExtraEmitCodeLine(convertSKSLExpressionToCPP(*c.fArguments.back(), coordsName));
452 }
Ethan Nicholasee1c8a72019-02-22 10:50:47 -0500453 if (c.fArguments[0]->fType.kind() == Type::kNullable_Kind) {
Ethan Nicholasbcd51e82019-04-09 10:40:41 -0400454 addExtraEmitCodeLine("if (_outer." + String(child.fName) + "_index >= 0) {\n ");
Ethan Nicholasee1c8a72019-02-22 10:50:47 -0500455 }
Ethan Nicholasd4efe682019-08-29 16:10:13 -0400456 if (hasCoords) {
457 addExtraEmitCodeLine("this->invokeChild(_outer." + String(child.fName) + "_index" +
458 inputArg + ", &" + childName + ", args, " + coordsName +
459 ".c_str());");
460 } else {
461 addExtraEmitCodeLine("this->invokeChild(_outer." + String(child.fName) + "_index" +
462 inputArg + ", &" + childName + ", args);");
463 }
464
Ethan Nicholasee1c8a72019-02-22 10:50:47 -0500465 if (c.fArguments[0]->fType.kind() == Type::kNullable_Kind) {
Ethan Nicholas6ad52892019-05-03 13:13:42 +0000466 // Null FPs are not emitted, but their output can still be referenced in dependent
467 // expressions - thus we always declare the variable.
468 // Note: this is essentially dead code required to satisfy the compiler, because
469 // 'process' function calls should always be guarded at a higher level, in the .fp
470 // source.
471 addExtraEmitCodeLine(
472 "} else {"
473 " fragBuilder->codeAppendf(\"half4 %s;\", " + childName + ".c_str());"
474 "}");
Ethan Nicholasee1c8a72019-02-22 10:50:47 -0500475 }
Ethan Nicholas6ad52892019-05-03 13:13:42 +0000476 this->write("%s");
477 fFormatArgs.push_back(childName + ".c_str()");
Ethan Nicholasc9472af2017-10-10 16:30:21 -0400478 return;
479 }
Ethan Nicholas095f5b42019-08-30 11:51:41 -0400480 if (c.fFunction.fBuiltin) {
481 INHERITED::writeFunctionCall(c);
482 } else {
483 this->write("%s");
484 fFormatArgs.push_back((String(c.fFunction.fName) + "_name.c_str()").c_str());
485 this->write("(");
486 const char* separator = "";
487 for (const auto& arg : c.fArguments) {
488 this->write(separator);
489 separator = ", ";
490 this->writeExpression(*arg, kSequence_Precedence);
491 }
492 this->write(")");
493 }
Ethan Nicholas13863662019-07-29 13:05:15 -0400494 if (c.fFunction.fBuiltin && c.fFunction.fName == "sample") {
Ethan Nicholasceb4d482017-07-10 15:40:20 -0400495 this->write(".%s");
Ethan Nicholasd9d33c32018-06-12 11:05:59 -0400496 SkASSERT(c.fArguments.size() >= 1);
497 SkASSERT(c.fArguments[0]->fKind == Expression::kVariableReference_Kind);
Ethan Nicholasceb4d482017-07-10 15:40:20 -0400498 String sampler = this->getSamplerHandle(((VariableReference&) *c.fArguments[0]).fVariable);
499 fFormatArgs.push_back("fragBuilder->getProgramBuilder()->samplerSwizzle(" + sampler +
Greg Daniel369ee6b2019-12-02 15:30:02 -0500500 ").asString().c_str()");
Ethan Nicholasceb4d482017-07-10 15:40:20 -0400501 }
502}
503
Ethan Nicholas095f5b42019-08-30 11:51:41 -0400504static const char* glsltype_string(const Context& context, const Type& type) {
505 if (type == *context.fFloat_Type) {
506 return "kFloat_GrSLType";
507 } else if (type == *context.fHalf_Type) {
508 return "kHalf_GrSLType";
509 } else if (type == *context.fFloat2_Type) {
510 return "kFloat2_GrSLType";
511 } else if (type == *context.fHalf2_Type) {
512 return "kHalf2_GrSLType";
Ethan Nicholas8ae1b562019-12-17 15:18:02 -0500513 } else if (type == *context.fFloat3_Type) {
514 return "kFloat3_GrSLType";
515 } else if (type == *context.fHalf3_Type) {
516 return "kHalf3_GrSLType";
Ethan Nicholas095f5b42019-08-30 11:51:41 -0400517 } else if (type == *context.fFloat4_Type) {
518 return "kFloat4_GrSLType";
519 } else if (type == *context.fHalf4_Type) {
520 return "kHalf4_GrSLType";
521 } else if (type == *context.fFloat4x4_Type) {
522 return "kFloat4x4_GrSLType";
523 } else if (type == *context.fHalf4x4_Type) {
524 return "kHalf4x4_GrSLType";
525 } else if (type == *context.fVoid_Type) {
526 return "kVoid_GrSLType";
Ethan Nicholas8ae1b562019-12-17 15:18:02 -0500527 } else if (type.kind() == Type::kEnum_Kind) {
528 return "int";
Ethan Nicholas095f5b42019-08-30 11:51:41 -0400529 }
530 SkASSERT(false);
531 return nullptr;
532}
533
Ethan Nicholas762466e2017-06-29 10:03:38 -0400534void CPPCodeGenerator::writeFunction(const FunctionDefinition& f) {
Ethan Nicholas095f5b42019-08-30 11:51:41 -0400535 const FunctionDeclaration& decl = f.fDeclaration;
536 fFunctionHeader = "";
537 OutputStream* oldOut = fOut;
538 StringStream buffer;
539 fOut = &buffer;
540 if (decl.fName == "main") {
Ethan Nicholasf1b14642018-08-09 16:18:07 -0400541 fInMain = true;
Ethan Nicholas762466e2017-06-29 10:03:38 -0400542 for (const auto& s : ((Block&) *f.fBody).fStatements) {
543 this->writeStatement(*s);
544 this->writeLine();
545 }
Ethan Nicholasf1b14642018-08-09 16:18:07 -0400546 fInMain = false;
Ethan Nicholas762466e2017-06-29 10:03:38 -0400547
548 fOut = oldOut;
549 this->write(fFunctionHeader);
550 this->write(buffer.str());
551 } else {
Ethan Nicholas095f5b42019-08-30 11:51:41 -0400552 this->addExtraEmitCodeLine("SkString " + decl.fName + "_name;");
553 String args = "const GrShaderVar " + decl.fName + "_args[] = { ";
554 const char* separator = "";
555 for (const auto& param : decl.fParameters) {
556 args += String(separator) + "GrShaderVar(\"" + param->fName + "\", " +
557 glsltype_string(fContext, param->fType) + ")";
558 separator = ", ";
559 }
560 args += "};";
561 this->addExtraEmitCodeLine(args.c_str());
562 for (const auto& s : ((Block&) *f.fBody).fStatements) {
563 this->writeStatement(*s);
564 this->writeLine();
565 }
566
567 fOut = oldOut;
568 String emit = "fragBuilder->emitFunction(";
569 emit += glsltype_string(fContext, decl.fReturnType);
570 emit += ", \"" + decl.fName + "\"";
571 emit += ", " + to_string((int64_t) decl.fParameters.size());
572 emit += ", " + decl.fName + "_args";
573 emit += ", \"" + buffer.str() + "\"";
574 emit += ", &" + decl.fName + "_name);";
575 this->addExtraEmitCodeLine(emit.c_str());
Ethan Nicholas762466e2017-06-29 10:03:38 -0400576 }
577}
578
579void CPPCodeGenerator::writeSetting(const Setting& s) {
580 static constexpr const char* kPrefix = "sk_Args.";
581 if (!strncmp(s.fName.c_str(), kPrefix, strlen(kPrefix))) {
582 const char* name = s.fName.c_str() + strlen(kPrefix);
Ethan Nicholasd608c092017-10-26 09:30:08 -0400583 this->writeRuntimeValue(s.fType, Layout(), HCodeGenerator::FieldName(name).c_str());
Ethan Nicholas762466e2017-06-29 10:03:38 -0400584 } else {
585 this->write(s.fName.c_str());
586 }
587}
588
Ethan Nicholasf57c0d62017-07-31 11:18:22 -0400589bool CPPCodeGenerator::writeSection(const char* name, const char* prefix) {
Ethan Nicholas68990be2017-07-13 09:36:52 -0400590 const Section* s = fSectionAndParameterHelper.getSection(name);
591 if (s) {
592 this->writef("%s%s", prefix, s->fText.c_str());
Ethan Nicholasf57c0d62017-07-31 11:18:22 -0400593 return true;
Ethan Nicholas762466e2017-06-29 10:03:38 -0400594 }
Ethan Nicholasf57c0d62017-07-31 11:18:22 -0400595 return false;
Ethan Nicholas762466e2017-06-29 10:03:38 -0400596}
597
598void CPPCodeGenerator::writeProgramElement(const ProgramElement& p) {
599 if (p.fKind == ProgramElement::kSection_Kind) {
600 return;
601 }
602 if (p.fKind == ProgramElement::kVar_Kind) {
603 const VarDeclarations& decls = (const VarDeclarations&) p;
604 if (!decls.fVars.size()) {
605 return;
606 }
Ethan Nicholas82a62d22017-11-07 14:42:10 +0000607 const Variable& var = *((VarDeclaration&) *decls.fVars[0]).fVar;
Ethan Nicholas762466e2017-06-29 10:03:38 -0400608 if (var.fModifiers.fFlags & (Modifiers::kIn_Flag | Modifiers::kUniform_Flag) ||
609 -1 != var.fModifiers.fLayout.fBuiltin) {
610 return;
611 }
612 }
613 INHERITED::writeProgramElement(p);
614}
615
616void CPPCodeGenerator::addUniform(const Variable& var) {
617 if (!needs_uniform_var(var)) {
618 return;
619 }
Ethan Nicholasfc994162019-06-06 10:04:27 -0400620 if (var.fModifiers.fLayout.fWhen.fLength) {
621 this->writef(" if (%s) {\n ", String(var.fModifiers.fLayout.fWhen).c_str());
Ethan Nicholas762466e2017-06-29 10:03:38 -0400622 }
Ethan Nicholas095f5b42019-08-30 11:51:41 -0400623 const char* type = glsltype_string(fContext, var.fType);
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700624 String name(var.fName);
Ethan Nicholas762466e2017-06-29 10:03:38 -0400625 this->writef(" %sVar = args.fUniformHandler->addUniform(kFragment_GrShaderFlag, %s, "
Ethan Nicholas858fecc2019-03-07 13:19:18 -0500626 "\"%s\");\n", HCodeGenerator::FieldName(name.c_str()).c_str(), type,
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700627 name.c_str());
Ethan Nicholasfc994162019-06-06 10:04:27 -0400628 if (var.fModifiers.fLayout.fWhen.fLength) {
Ethan Nicholas762466e2017-06-29 10:03:38 -0400629 this->write(" }\n");
630 }
631}
632
Ethan Nicholascd700e92018-08-24 16:43:57 -0400633void CPPCodeGenerator::writeInputVars() {
634}
635
Ethan Nicholas762466e2017-06-29 10:03:38 -0400636void CPPCodeGenerator::writePrivateVars() {
Ethan Nicholas3c6ae622018-04-24 13:06:09 -0400637 for (const auto& p : fProgram) {
638 if (ProgramElement::kVar_Kind == p.fKind) {
639 const VarDeclarations& decls = (const VarDeclarations&) p;
640 for (const auto& raw : decls.fVars) {
Ethan Nicholas82a62d22017-11-07 14:42:10 +0000641 VarDeclaration& decl = (VarDeclaration&) *raw;
642 if (is_private(*decl.fVar)) {
643 if (decl.fVar->fType == *fContext.fFragmentProcessor_Type) {
644 fErrors.error(decl.fOffset,
Ethan Nicholasc9472af2017-10-10 16:30:21 -0400645 "fragmentProcessor variables must be declared 'in'");
646 return;
647 }
Ethan Nicholase9d172a2017-11-20 12:12:24 -0500648 this->writef("%s %s = %s;\n",
Ethan Nicholas82a62d22017-11-07 14:42:10 +0000649 HCodeGenerator::FieldType(fContext, decl.fVar->fType,
650 decl.fVar->fModifiers.fLayout).c_str(),
Ethan Nicholase9d172a2017-11-20 12:12:24 -0500651 String(decl.fVar->fName).c_str(),
652 default_value(*decl.fVar).c_str());
Michael Ludwiga4275592018-08-31 10:52:47 -0400653 } else if (decl.fVar->fModifiers.fLayout.fFlags & Layout::kTracked_Flag) {
654 // An auto-tracked uniform in variable, so add a field to hold onto the prior
655 // state. Note that tracked variables must be uniform in's and that is validated
656 // before writePrivateVars() is called.
657 const UniformCTypeMapper* mapper = UniformCTypeMapper::Get(fContext, *decl.fVar);
658 SkASSERT(mapper && mapper->supportsTracking());
659
660 String name = HCodeGenerator::FieldName(String(decl.fVar->fName).c_str());
661 // The member statement is different if the mapper reports a default value
662 if (mapper->defaultValue().size() > 0) {
663 this->writef("%s %sPrev = %s;\n",
Ethan Nicholas78aceb22018-08-31 16:13:58 -0400664 Layout::CTypeToStr(mapper->ctype()), name.c_str(),
Michael Ludwiga4275592018-08-31 10:52:47 -0400665 mapper->defaultValue().c_str());
666 } else {
667 this->writef("%s %sPrev;\n",
Ethan Nicholas78aceb22018-08-31 16:13:58 -0400668 Layout::CTypeToStr(mapper->ctype()), name.c_str());
Michael Ludwiga4275592018-08-31 10:52:47 -0400669 }
Ethan Nicholas762466e2017-06-29 10:03:38 -0400670 }
671 }
672 }
673 }
674}
675
676void CPPCodeGenerator::writePrivateVarValues() {
Ethan Nicholas3c6ae622018-04-24 13:06:09 -0400677 for (const auto& p : fProgram) {
678 if (ProgramElement::kVar_Kind == p.fKind) {
679 const VarDeclarations& decls = (const VarDeclarations&) p;
680 for (const auto& raw : decls.fVars) {
Ethan Nicholas82a62d22017-11-07 14:42:10 +0000681 VarDeclaration& decl = (VarDeclaration&) *raw;
682 if (is_private(*decl.fVar) && decl.fValue) {
683 this->writef("%s = ", String(decl.fVar->fName).c_str());
Ethan Nicholas82399462017-10-16 12:35:44 -0400684 fCPPMode = true;
Ethan Nicholas82a62d22017-11-07 14:42:10 +0000685 this->writeExpression(*decl.fValue, kAssignment_Precedence);
Ethan Nicholas82399462017-10-16 12:35:44 -0400686 fCPPMode = false;
687 this->write(";\n");
Ethan Nicholas762466e2017-06-29 10:03:38 -0400688 }
689 }
690 }
691 }
692}
693
Ethan Nicholas82399462017-10-16 12:35:44 -0400694static bool is_accessible(const Variable& var) {
Ethan Nicholasee1c8a72019-02-22 10:50:47 -0500695 const Type& type = var.fType.nonnullable();
696 return Type::kSampler_Kind != type.kind() &&
697 Type::kOther_Kind != type.kind();
Ethan Nicholas82399462017-10-16 12:35:44 -0400698}
699
Michael Ludwig1fc5fbd2018-09-07 13:13:06 -0400700void CPPCodeGenerator::newExtraEmitCodeBlock() {
701 // This should only be called when emitting SKSL for emitCode(), which can be detected if the
702 // cpp buffer is not null, and the cpp buffer is not the current output.
703 SkASSERT(fCPPBuffer && fCPPBuffer != fOut);
704
705 // Start a new block as an empty string
706 fExtraEmitCodeBlocks.push_back("");
707 // Mark its location in the output buffer, uses ${\d} for the token since ${} will not occur in
708 // valid sksl and makes detection trivial.
709 this->writef("${%zu}", fExtraEmitCodeBlocks.size() - 1);
710}
711
712void CPPCodeGenerator::addExtraEmitCodeLine(const String& toAppend) {
713 SkASSERT(fExtraEmitCodeBlocks.size() > 0);
714 String& currentBlock = fExtraEmitCodeBlocks[fExtraEmitCodeBlocks.size() - 1];
715 // Automatically add indentation and newline
716 currentBlock += " " + toAppend + "\n";
717}
718
719void CPPCodeGenerator::flushEmittedCode() {
Michael Ludwig92e4c7f2018-08-30 16:08:18 -0400720 if (fCPPBuffer == nullptr) {
721 // Not actually within writeEmitCode() so nothing to flush
722 return;
723 }
724
725 StringStream* skslBuffer = static_cast<StringStream*>(fOut);
726
727 String sksl = skslBuffer->str();
Michael Ludwig1fc5fbd2018-09-07 13:13:06 -0400728 // Empty the accumulation buffer since its current contents are consumed.
Michael Ludwig92e4c7f2018-08-30 16:08:18 -0400729 skslBuffer->reset();
730
Michael Ludwig1fc5fbd2018-09-07 13:13:06 -0400731 // Switch to the cpp buffer
Michael Ludwigd0440192018-09-07 14:24:52 +0000732 fOut = fCPPBuffer;
Michael Ludwig1fc5fbd2018-09-07 13:13:06 -0400733
734 // Iterate through the sksl, keeping track of where the last statement ended (e.g. the latest
735 // encountered ';', '{', or '}'). If an extra emit code block token is encountered then the
736 // code from 0 to last statement end is sent to writeCodeAppend, the extra code block is
737 // appended to the cpp buffer, and then the sksl string is trimmed to start where the last
738 // statement left off (minus the encountered token).
739 size_t i = 0;
740 int flushPoint = -1;
741 int tokenStart = -1;
742 while (i < sksl.size()) {
743 if (tokenStart >= 0) {
744 // Looking for the end of the token
745 if (sksl[i] == '}') {
746 // Must append the sksl from 0 to flushPoint (inclusive) then the extra code
747 // accumulated in the block with index parsed from chars [tokenStart+2, i-1]
748 String toFlush = String(sksl.c_str(), flushPoint + 1);
749 // writeCodeAppend automatically removes the format args that it consumed, so
750 // fFormatArgs will be in a valid state for any future sksl
751 this->writeCodeAppend(toFlush);
752
753 int codeBlock = stoi(String(sksl.c_str() + tokenStart + 2, i - tokenStart - 2));
754 SkASSERT(codeBlock < (int) fExtraEmitCodeBlocks.size());
755 if (fExtraEmitCodeBlocks[codeBlock].size() > 0) {
756 this->write(fExtraEmitCodeBlocks[codeBlock].c_str());
757 }
758
759 // Now reset the sksl buffer to start after the flush point, but remove the token.
760 String compacted = String(sksl.c_str() + flushPoint + 1,
761 tokenStart - flushPoint - 1);
762 if (i < sksl.size() - 1) {
763 compacted += String(sksl.c_str() + i + 1, sksl.size() - i - 1);
764 }
765 sksl = compacted;
766
767 // And reset iteration
768 i = -1;
769 flushPoint = -1;
770 tokenStart = -1;
771 }
772 } else {
773 // Looking for the start of extra emit block tokens, and tracking when statements end
774 if (sksl[i] == ';' || sksl[i] == '{' || sksl[i] == '}') {
775 flushPoint = i;
776 } else if (i < sksl.size() - 1 && sksl[i] == '$' && sksl[i + 1] == '{') {
777 // found an extra emit code block token
778 tokenStart = i++;
779 }
780 }
781 i++;
Michael Ludwigd0440192018-09-07 14:24:52 +0000782 }
Michael Ludwig1fc5fbd2018-09-07 13:13:06 -0400783
784 // Once we've gone through the sksl string to this point, there are no remaining extra emit
785 // code blocks to interleave, so append the remainder as usual.
Michael Ludwig92e4c7f2018-08-30 16:08:18 -0400786 this->writeCodeAppend(sksl);
787
Michael Ludwig1fc5fbd2018-09-07 13:13:06 -0400788 // After appending, switch back to the emptied sksl buffer and reset the extra code blocks
Michael Ludwig92e4c7f2018-08-30 16:08:18 -0400789 fOut = skslBuffer;
Michael Ludwig1fc5fbd2018-09-07 13:13:06 -0400790 fExtraEmitCodeBlocks.clear();
Michael Ludwig92e4c7f2018-08-30 16:08:18 -0400791}
792
Ethan Nicholas5b6e6272017-10-13 13:11:06 -0400793void CPPCodeGenerator::writeCodeAppend(const String& code) {
794 // codeAppendf can only handle appending 1024 bytes at a time, so we need to break the string
795 // into chunks. Unfortunately we can't tell exactly how long the string is going to end up,
796 // because printf escape sequences get replaced by strings of unknown length, but keeping the
797 // format string below 512 bytes is probably safe.
798 static constexpr size_t maxChunkSize = 512;
799 size_t start = 0;
800 size_t index = 0;
801 size_t argStart = 0;
802 size_t argCount;
803 while (index < code.size()) {
804 argCount = 0;
805 this->write(" fragBuilder->codeAppendf(\"");
806 while (index < code.size() && index < start + maxChunkSize) {
807 if ('%' == code[index]) {
808 if (index == start + maxChunkSize - 1 || index == code.size() - 1) {
809 break;
810 }
811 if (code[index + 1] != '%') {
812 ++argCount;
813 }
Ethan Nicholasef0c9fd2017-10-30 10:04:14 -0400814 } else if ('\\' == code[index] && index == start + maxChunkSize - 1) {
815 // avoid splitting an escape sequence that happens to fall across a chunk boundary
816 break;
Ethan Nicholas5b6e6272017-10-13 13:11:06 -0400817 }
818 ++index;
819 }
820 fOut->write(code.c_str() + start, index - start);
821 this->write("\"");
822 for (size_t i = argStart; i < argStart + argCount; ++i) {
823 this->writef(", %s", fFormatArgs[i].c_str());
824 }
825 this->write(");\n");
826 argStart += argCount;
827 start = index;
828 }
Michael Ludwig92e4c7f2018-08-30 16:08:18 -0400829
830 // argStart is equal to the number of fFormatArgs that were consumed
831 // so they should be removed from the list
832 if (argStart > 0) {
833 fFormatArgs.erase(fFormatArgs.begin(), fFormatArgs.begin() + argStart);
834 }
835}
836
837String CPPCodeGenerator::convertSKSLExpressionToCPP(const Expression& e,
838 const String& cppVar) {
839 // To do this conversion, we temporarily switch the sksl output stream
840 // to an empty stringstream and reset the format args to empty.
841 OutputStream* oldSKSL = fOut;
842 StringStream exprBuffer;
843 fOut = &exprBuffer;
844
845 std::vector<String> oldArgs(fFormatArgs);
846 fFormatArgs.clear();
847
848 // Convert the argument expression into a format string and args
849 this->writeExpression(e, Precedence::kTopLevel_Precedence);
850 std::vector<String> newArgs(fFormatArgs);
Michael Ludwig1fc5fbd2018-09-07 13:13:06 -0400851 String expr = exprBuffer.str();
Michael Ludwig92e4c7f2018-08-30 16:08:18 -0400852
853 // After generating, restore the original output stream and format args
854 fFormatArgs = oldArgs;
855 fOut = oldSKSL;
856
Michael Ludwig1fc5fbd2018-09-07 13:13:06 -0400857 // The sksl written to exprBuffer is not processed by flushEmittedCode(), so any extra emit code
858 // block tokens won't get handled. So we need to strip them from the expression and stick them
859 // to the end of the original sksl stream.
860 String exprFormat = "";
861 int tokenStart = -1;
862 for (size_t i = 0; i < expr.size(); i++) {
863 if (tokenStart >= 0) {
864 if (expr[i] == '}') {
865 // End of the token, so append the token to fOut
866 fOut->write(expr.c_str() + tokenStart, i - tokenStart + 1);
867 tokenStart = -1;
868 }
869 } else {
870 if (i < expr.size() - 1 && expr[i] == '$' && expr[i + 1] == '{') {
871 tokenStart = i++;
872 } else {
873 exprFormat += expr[i];
874 }
875 }
876 }
877
Michael Ludwig92e4c7f2018-08-30 16:08:18 -0400878 // Now build the final C++ code snippet from the format string and args
879 String cppExpr;
880 if (newArgs.size() == 0) {
881 // This was a static expression, so we can simplify the input
882 // color declaration in the emitted code to just a static string
Michael Ludwig1fc5fbd2018-09-07 13:13:06 -0400883 cppExpr = "SkString " + cppVar + "(\"" + exprFormat + "\");";
Michael Ludwig92e4c7f2018-08-30 16:08:18 -0400884 } else {
885 // String formatting must occur dynamically, so have the C++ declaration
886 // use SkStringPrintf with the format args that were accumulated
887 // when the expression was written.
888 cppExpr = "SkString " + cppVar + " = SkStringPrintf(\"" + exprFormat + "\"";
889 for (size_t i = 0; i < newArgs.size(); i++) {
890 cppExpr += ", " + newArgs[i];
891 }
Michael Ludwig1fc5fbd2018-09-07 13:13:06 -0400892 cppExpr += ");";
Michael Ludwig92e4c7f2018-08-30 16:08:18 -0400893 }
894 return cppExpr;
Ethan Nicholas5b6e6272017-10-13 13:11:06 -0400895}
896
Ethan Nicholas762466e2017-06-29 10:03:38 -0400897bool CPPCodeGenerator::writeEmitCode(std::vector<const Variable*>& uniforms) {
898 this->write(" void emitCode(EmitArgs& args) override {\n"
899 " GrGLSLFPFragmentBuilder* fragBuilder = args.fFragBuilder;\n");
900 this->writef(" const %s& _outer = args.fFp.cast<%s>();\n"
901 " (void) _outer;\n",
902 fFullName.c_str(), fFullName.c_str());
Ethan Nicholas3c6ae622018-04-24 13:06:09 -0400903 for (const auto& p : fProgram) {
904 if (ProgramElement::kVar_Kind == p.fKind) {
905 const VarDeclarations& decls = (const VarDeclarations&) p;
906 for (const auto& raw : decls.fVars) {
Ethan Nicholas82a62d22017-11-07 14:42:10 +0000907 VarDeclaration& decl = (VarDeclaration&) *raw;
908 String nameString(decl.fVar->fName);
Ethan Nicholas82399462017-10-16 12:35:44 -0400909 const char* name = nameString.c_str();
Ethan Nicholas82a62d22017-11-07 14:42:10 +0000910 if (SectionAndParameterHelper::IsParameter(*decl.fVar) &&
911 is_accessible(*decl.fVar)) {
Ethan Nicholasbcd51e82019-04-09 10:40:41 -0400912 this->writef(" auto %s = _outer.%s;\n"
Ethan Nicholas82399462017-10-16 12:35:44 -0400913 " (void) %s;\n",
914 name, name, name);
915 }
916 }
917 }
918 }
Ethan Nicholas762466e2017-06-29 10:03:38 -0400919 this->writePrivateVarValues();
920 for (const auto u : uniforms) {
921 this->addUniform(*u);
Ethan Nicholas762466e2017-06-29 10:03:38 -0400922 }
923 this->writeSection(EMIT_CODE_SECTION);
Michael Ludwig92e4c7f2018-08-30 16:08:18 -0400924
925 // Save original buffer as the CPP buffer for flushEmittedCode()
926 fCPPBuffer = fOut;
927 StringStream skslBuffer;
928 fOut = &skslBuffer;
929
Michael Ludwig1fc5fbd2018-09-07 13:13:06 -0400930 this->newExtraEmitCodeBlock();
Ethan Nicholas762466e2017-06-29 10:03:38 -0400931 bool result = INHERITED::generateCode();
Michael Ludwig1fc5fbd2018-09-07 13:13:06 -0400932 this->flushEmittedCode();
Michael Ludwig92e4c7f2018-08-30 16:08:18 -0400933
934 // Then restore the original CPP buffer and close the function
935 fOut = fCPPBuffer;
936 fCPPBuffer = nullptr;
Ethan Nicholas5b6e6272017-10-13 13:11:06 -0400937 this->write(" }\n");
Ethan Nicholas762466e2017-06-29 10:03:38 -0400938 return result;
939}
940
941void CPPCodeGenerator::writeSetData(std::vector<const Variable*>& uniforms) {
942 const char* fullName = fFullName.c_str();
Ethan Nicholas68990be2017-07-13 09:36:52 -0400943 const Section* section = fSectionAndParameterHelper.getSection(SET_DATA_SECTION);
944 const char* pdman = section ? section->fArgument.c_str() : "pdman";
Ethan Nicholas762466e2017-06-29 10:03:38 -0400945 this->writef(" void onSetData(const GrGLSLProgramDataManager& %s, "
946 "const GrFragmentProcessor& _proc) override {\n",
947 pdman);
948 bool wroteProcessor = false;
949 for (const auto u : uniforms) {
Michael Ludwiga4275592018-08-31 10:52:47 -0400950 if (is_uniform_in(*u)) {
Ethan Nicholas762466e2017-06-29 10:03:38 -0400951 if (!wroteProcessor) {
952 this->writef(" const %s& _outer = _proc.cast<%s>();\n", fullName, fullName);
953 wroteProcessor = true;
954 this->writef(" {\n");
955 }
Michael Ludwiga4275592018-08-31 10:52:47 -0400956
957 const UniformCTypeMapper* mapper = UniformCTypeMapper::Get(fContext, *u);
958 SkASSERT(mapper);
959
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700960 String nameString(u->fName);
961 const char* name = nameString.c_str();
Michael Ludwiga4275592018-08-31 10:52:47 -0400962
963 // Switches for setData behavior in the generated code
964 bool conditionalUniform = u->fModifiers.fLayout.fWhen != "";
965 bool isTracked = u->fModifiers.fLayout.fFlags & Layout::kTracked_Flag;
966 bool needsValueDeclaration = isTracked || !mapper->canInlineUniformValue();
967
968 String uniformName = HCodeGenerator::FieldName(name) + "Var";
969
970 String indent = " "; // 8 by default, 12 when nested for conditional uniforms
971 if (conditionalUniform) {
972 // Add a pre-check to make sure the uniform was emitted
973 // before trying to send any data to the GPU
974 this->writef(" if (%s.isValid()) {\n", uniformName.c_str());
975 indent += " ";
976 }
977
978 String valueVar = "";
979 if (needsValueDeclaration) {
980 valueVar.appendf("%sValue", name);
981 // Use AccessType since that will match the return type of _outer's public API.
982 String valueType = HCodeGenerator::AccessType(fContext, u->fType,
983 u->fModifiers.fLayout);
Ethan Nicholasbcd51e82019-04-09 10:40:41 -0400984 this->writef("%s%s %s = _outer.%s;\n",
Michael Ludwiga4275592018-08-31 10:52:47 -0400985 indent.c_str(), valueType.c_str(), valueVar.c_str(), name);
Ethan Nicholas762466e2017-06-29 10:03:38 -0400986 } else {
Michael Ludwiga4275592018-08-31 10:52:47 -0400987 // Not tracked and the mapper only needs to use the value once
988 // so send it a safe expression instead of the variable name
Ethan Nicholasbcd51e82019-04-09 10:40:41 -0400989 valueVar.appendf("(_outer.%s)", name);
Michael Ludwiga4275592018-08-31 10:52:47 -0400990 }
991
992 if (isTracked) {
993 SkASSERT(mapper->supportsTracking());
994
995 String prevVar = HCodeGenerator::FieldName(name) + "Prev";
996 this->writef("%sif (%s) {\n"
997 "%s %s;\n"
998 "%s %s;\n"
999 "%s}\n", indent.c_str(),
1000 mapper->dirtyExpression(valueVar, prevVar).c_str(), indent.c_str(),
1001 mapper->saveState(valueVar, prevVar).c_str(), indent.c_str(),
1002 mapper->setUniform(pdman, uniformName, valueVar).c_str(), indent.c_str());
1003 } else {
1004 this->writef("%s%s;\n", indent.c_str(),
1005 mapper->setUniform(pdman, uniformName, valueVar).c_str());
1006 }
1007
1008 if (conditionalUniform) {
1009 // Close the earlier precheck block
1010 this->writef(" }\n");
Ethan Nicholas762466e2017-06-29 10:03:38 -04001011 }
1012 }
1013 }
1014 if (wroteProcessor) {
1015 this->writef(" }\n");
1016 }
Ethan Nicholas68990be2017-07-13 09:36:52 -04001017 if (section) {
Ethan Nicholas2d5f9b32017-12-13 14:36:14 -05001018 int samplerIndex = 0;
Ethan Nicholas3c6ae622018-04-24 13:06:09 -04001019 for (const auto& p : fProgram) {
1020 if (ProgramElement::kVar_Kind == p.fKind) {
1021 const VarDeclarations& decls = (const VarDeclarations&) p;
1022 for (const auto& raw : decls.fVars) {
Ethan Nicholas82a62d22017-11-07 14:42:10 +00001023 VarDeclaration& decl = (VarDeclaration&) *raw;
1024 String nameString(decl.fVar->fName);
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001025 const char* name = nameString.c_str();
Ethan Nicholas2d5f9b32017-12-13 14:36:14 -05001026 if (decl.fVar->fType.kind() == Type::kSampler_Kind) {
Robert Phillipsbd99c0c2019-12-12 13:26:58 +00001027 this->writef(" const GrSurfaceProxyView& %sView = "
1028 "_outer.textureSampler(%d).view();\n",
Ethan Nicholas2d5f9b32017-12-13 14:36:14 -05001029 name, samplerIndex);
Robert Phillipsbd99c0c2019-12-12 13:26:58 +00001030 this->writef(" GrTexture& %s = *%sView.proxy()->peekTexture();\n",
Ethan Nicholas2d5f9b32017-12-13 14:36:14 -05001031 name, name);
1032 this->writef(" (void) %s;\n", name);
1033 ++samplerIndex;
1034 } else if (needs_uniform_var(*decl.fVar)) {
Ethan Nicholas762466e2017-06-29 10:03:38 -04001035 this->writef(" UniformHandle& %s = %sVar;\n"
1036 " (void) %s;\n",
1037 name, HCodeGenerator::FieldName(name).c_str(), name);
Ethan Nicholas82a62d22017-11-07 14:42:10 +00001038 } else if (SectionAndParameterHelper::IsParameter(*decl.fVar) &&
1039 decl.fVar->fType != *fContext.fFragmentProcessor_Type) {
Ethan Nicholas762466e2017-06-29 10:03:38 -04001040 if (!wroteProcessor) {
1041 this->writef(" const %s& _outer = _proc.cast<%s>();\n", fullName,
1042 fullName);
1043 wroteProcessor = true;
1044 }
Ethan Nicholasbcd51e82019-04-09 10:40:41 -04001045 this->writef(" auto %s = _outer.%s;\n"
Ethan Nicholas762466e2017-06-29 10:03:38 -04001046 " (void) %s;\n",
1047 name, name, name);
1048 }
1049 }
1050 }
1051 }
1052 this->writeSection(SET_DATA_SECTION);
1053 }
1054 this->write(" }\n");
1055}
1056
Brian Salomonf7dcd762018-07-30 14:48:15 -04001057void CPPCodeGenerator::writeOnTextureSampler() {
1058 bool foundSampler = false;
1059 for (const auto& param : fSectionAndParameterHelper.getParameters()) {
1060 if (param->fType.kind() == Type::kSampler_Kind) {
1061 if (!foundSampler) {
1062 this->writef(
1063 "const GrFragmentProcessor::TextureSampler& %s::onTextureSampler(int "
1064 "index) const {\n",
1065 fFullName.c_str());
1066 this->writef(" return IthTextureSampler(index, %s",
1067 HCodeGenerator::FieldName(String(param->fName).c_str()).c_str());
1068 foundSampler = true;
1069 } else {
1070 this->writef(", %s",
1071 HCodeGenerator::FieldName(String(param->fName).c_str()).c_str());
1072 }
1073 }
1074 }
1075 if (foundSampler) {
1076 this->write(");\n}\n");
1077 }
1078}
1079
Ethan Nicholasf57c0d62017-07-31 11:18:22 -04001080void CPPCodeGenerator::writeClone() {
1081 if (!this->writeSection(CLONE_SECTION)) {
1082 if (fSectionAndParameterHelper.getSection(FIELDS_SECTION)) {
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001083 fErrors.error(0, "fragment processors with custom @fields must also have a custom"
1084 "@clone");
Ethan Nicholasf57c0d62017-07-31 11:18:22 -04001085 }
1086 this->writef("%s::%s(const %s& src)\n"
Ethan Nicholasabff9562017-10-09 10:54:08 -04001087 ": INHERITED(k%s_ClassID, src.optimizationFlags())", fFullName.c_str(),
1088 fFullName.c_str(), fFullName.c_str(), fFullName.c_str());
Ethan Nicholasbcd51e82019-04-09 10:40:41 -04001089 const auto transforms = fSectionAndParameterHelper.getSections(COORD_TRANSFORM_SECTION);
1090 for (size_t i = 0; i < transforms.size(); ++i) {
1091 const Section& s = *transforms[i];
1092 String fieldName = HCodeGenerator::CoordTransformName(s.fArgument, i);
1093 this->writef("\n, %s(src.%s)", fieldName.c_str(), fieldName.c_str());
1094 }
Ethan Nicholasf57c0d62017-07-31 11:18:22 -04001095 for (const auto& param : fSectionAndParameterHelper.getParameters()) {
Robert Phillipsbce7d862019-02-21 22:53:57 +00001096 String fieldName = HCodeGenerator::FieldName(String(param->fName).c_str());
Ethan Nicholasee1c8a72019-02-22 10:50:47 -05001097 if (param->fType.nonnullable() == *fContext.fFragmentProcessor_Type) {
1098 this->writef("\n, %s_index(src.%s_index)",
1099 fieldName.c_str(),
1100 fieldName.c_str());
1101 } else {
1102 this->writef("\n, %s(src.%s)",
1103 fieldName.c_str(),
1104 fieldName.c_str());
1105 }
Ethan Nicholasf57c0d62017-07-31 11:18:22 -04001106 }
Ethan Nicholasabff9562017-10-09 10:54:08 -04001107 this->writef(" {\n");
Brian Salomonf7dcd762018-07-30 14:48:15 -04001108 int samplerCount = 0;
Ethan Nicholasf57c0d62017-07-31 11:18:22 -04001109 for (const auto& param : fSectionAndParameterHelper.getParameters()) {
1110 if (param->fType.kind() == Type::kSampler_Kind) {
Brian Salomonf7dcd762018-07-30 14:48:15 -04001111 ++samplerCount;
Ethan Nicholasee1c8a72019-02-22 10:50:47 -05001112 } else if (param->fType.nonnullable() == *fContext.fFragmentProcessor_Type) {
1113 String fieldName = HCodeGenerator::FieldName(String(param->fName).c_str());
1114 if (param->fType.kind() == Type::kNullable_Kind) {
1115 this->writef(" if (%s_index >= 0) {\n ", fieldName.c_str());
1116 }
1117 this->writef(" this->registerChildProcessor(src.childProcessor(%s_index)."
1118 "clone());\n", fieldName.c_str());
1119 if (param->fType.kind() == Type::kNullable_Kind) {
1120 this->writef(" }\n");
1121 }
Ethan Nicholasf57c0d62017-07-31 11:18:22 -04001122 }
1123 }
Brian Salomonf7dcd762018-07-30 14:48:15 -04001124 if (samplerCount) {
1125 this->writef(" this->setTextureSamplerCnt(%d);", samplerCount);
1126 }
Ethan Nicholas929a6812018-08-06 14:56:59 -04001127 for (size_t i = 0; i < transforms.size(); ++i) {
1128 const Section& s = *transforms[i];
1129 String fieldName = HCodeGenerator::CoordTransformName(s.fArgument, i);
1130 this->writef(" this->addCoordTransform(&%s);\n", fieldName.c_str());
Ethan Nicholasf57c0d62017-07-31 11:18:22 -04001131 }
1132 this->write("}\n");
Brian Salomonaff329b2017-08-11 09:40:37 -04001133 this->writef("std::unique_ptr<GrFragmentProcessor> %s::clone() const {\n",
1134 fFullName.c_str());
1135 this->writef(" return std::unique_ptr<GrFragmentProcessor>(new %s(*this));\n",
1136 fFullName.c_str());
Ethan Nicholasf57c0d62017-07-31 11:18:22 -04001137 this->write("}\n");
1138 }
1139}
1140
Ethan Nicholas762466e2017-06-29 10:03:38 -04001141void CPPCodeGenerator::writeTest() {
Ethan Nicholas68990be2017-07-13 09:36:52 -04001142 const Section* test = fSectionAndParameterHelper.getSection(TEST_CODE_SECTION);
1143 if (test) {
Brian Salomonaff329b2017-08-11 09:40:37 -04001144 this->writef(
1145 "GR_DEFINE_FRAGMENT_PROCESSOR_TEST(%s);\n"
1146 "#if GR_TEST_UTILS\n"
1147 "std::unique_ptr<GrFragmentProcessor> %s::TestCreate(GrProcessorTestData* %s) {\n",
1148 fFullName.c_str(),
1149 fFullName.c_str(),
1150 test->fArgument.c_str());
Ethan Nicholas68990be2017-07-13 09:36:52 -04001151 this->writeSection(TEST_CODE_SECTION);
1152 this->write("}\n"
1153 "#endif\n");
Ethan Nicholas762466e2017-06-29 10:03:38 -04001154 }
Ethan Nicholas762466e2017-06-29 10:03:38 -04001155}
1156
1157void CPPCodeGenerator::writeGetKey() {
1158 this->writef("void %s::onGetGLSLProcessorKey(const GrShaderCaps& caps, "
1159 "GrProcessorKeyBuilder* b) const {\n",
1160 fFullName.c_str());
Ethan Nicholascab767f2019-07-01 13:32:07 -04001161 for (const auto& p : fProgram) {
1162 if (ProgramElement::kVar_Kind == p.fKind) {
1163 const VarDeclarations& decls = (const VarDeclarations&) p;
1164 for (const auto& raw : decls.fVars) {
1165 const VarDeclaration& decl = (VarDeclaration&) *raw;
1166 const Variable& var = *decl.fVar;
1167 String nameString(var.fName);
1168 const char* name = nameString.c_str();
1169 if (var.fModifiers.fLayout.fKey != Layout::kNo_Key &&
1170 (var.fModifiers.fFlags & Modifiers::kUniform_Flag)) {
1171 fErrors.error(var.fOffset,
1172 "layout(key) may not be specified on uniforms");
Ethan Nicholasbcd51e82019-04-09 10:40:41 -04001173 }
Ethan Nicholascab767f2019-07-01 13:32:07 -04001174 switch (var.fModifiers.fLayout.fKey) {
1175 case Layout::kKey_Key:
1176 if (is_private(var)) {
1177 this->writef("%s %s =",
1178 HCodeGenerator::FieldType(fContext, var.fType,
1179 var.fModifiers.fLayout).c_str(),
1180 String(var.fName).c_str());
1181 if (decl.fValue) {
1182 fCPPMode = true;
1183 this->writeExpression(*decl.fValue, kAssignment_Precedence);
1184 fCPPMode = false;
1185 } else {
1186 this->writef("%s", default_value(var).c_str());
1187 }
1188 this->write(";\n");
1189 }
1190 if (var.fModifiers.fLayout.fWhen.fLength) {
1191 this->writef("if (%s) {", String(var.fModifiers.fLayout.fWhen).c_str());
1192 }
1193 if (var.fType == *fContext.fFloat4x4_Type) {
1194 ABORT("no automatic key handling for float4x4\n");
1195 } else if (var.fType == *fContext.fFloat2_Type) {
1196 this->writef(" b->add32(%s.fX);\n",
1197 HCodeGenerator::FieldName(name).c_str());
1198 this->writef(" b->add32(%s.fY);\n",
1199 HCodeGenerator::FieldName(name).c_str());
1200 } else if (var.fType == *fContext.fFloat4_Type) {
1201 this->writef(" b->add32(%s.x());\n",
1202 HCodeGenerator::FieldName(name).c_str());
1203 this->writef(" b->add32(%s.y());\n",
1204 HCodeGenerator::FieldName(name).c_str());
1205 this->writef(" b->add32(%s.width());\n",
1206 HCodeGenerator::FieldName(name).c_str());
1207 this->writef(" b->add32(%s.height());\n",
1208 HCodeGenerator::FieldName(name).c_str());
1209 } else if (var.fType == *fContext.fHalf4_Type) {
1210 this->writef(" uint16_t red = SkFloatToHalf(%s.fR);\n",
1211 HCodeGenerator::FieldName(name).c_str());
1212 this->writef(" uint16_t green = SkFloatToHalf(%s.fG);\n",
1213 HCodeGenerator::FieldName(name).c_str());
1214 this->writef(" uint16_t blue = SkFloatToHalf(%s.fB);\n",
1215 HCodeGenerator::FieldName(name).c_str());
1216 this->writef(" uint16_t alpha = SkFloatToHalf(%s.fA);\n",
1217 HCodeGenerator::FieldName(name).c_str());
1218 this->write(" b->add32(((uint32_t)red << 16) | green);\n");
1219 this->write(" b->add32(((uint32_t)blue << 16) | alpha);\n");
1220 } else {
1221 this->writef(" b->add32((int32_t) %s);\n",
1222 HCodeGenerator::FieldName(name).c_str());
1223 }
1224 if (var.fModifiers.fLayout.fWhen.fLength) {
1225 this->write("}");
1226 }
1227 break;
1228 case Layout::kIdentity_Key:
1229 if (var.fType.kind() != Type::kMatrix_Kind) {
1230 fErrors.error(var.fOffset,
1231 "layout(key=identity) requires matrix type");
1232 }
1233 this->writef(" b->add32(%s.isIdentity() ? 1 : 0);\n",
1234 HCodeGenerator::FieldName(name).c_str());
1235 break;
1236 case Layout::kNo_Key:
1237 break;
Ethan Nicholas762466e2017-06-29 10:03:38 -04001238 }
Ethan Nicholascab767f2019-07-01 13:32:07 -04001239 }
Ethan Nicholas762466e2017-06-29 10:03:38 -04001240 }
1241 }
1242 this->write("}\n");
1243}
1244
1245bool CPPCodeGenerator::generateCode() {
1246 std::vector<const Variable*> uniforms;
Ethan Nicholas3c6ae622018-04-24 13:06:09 -04001247 for (const auto& p : fProgram) {
1248 if (ProgramElement::kVar_Kind == p.fKind) {
1249 const VarDeclarations& decls = (const VarDeclarations&) p;
1250 for (const auto& raw : decls.fVars) {
Ethan Nicholas82a62d22017-11-07 14:42:10 +00001251 VarDeclaration& decl = (VarDeclaration&) *raw;
1252 if ((decl.fVar->fModifiers.fFlags & Modifiers::kUniform_Flag) &&
1253 decl.fVar->fType.kind() != Type::kSampler_Kind) {
1254 uniforms.push_back(decl.fVar);
Ethan Nicholas762466e2017-06-29 10:03:38 -04001255 }
Michael Ludwiga4275592018-08-31 10:52:47 -04001256
1257 if (is_uniform_in(*decl.fVar)) {
1258 // Validate the "uniform in" declarations to make sure they are fully supported,
1259 // instead of generating surprising C++
1260 const UniformCTypeMapper* mapper =
1261 UniformCTypeMapper::Get(fContext, *decl.fVar);
1262 if (mapper == nullptr) {
1263 fErrors.error(decl.fOffset, String(decl.fVar->fName)
1264 + "'s type is not supported for use as a 'uniform in'");
1265 return false;
1266 }
1267 if (decl.fVar->fModifiers.fLayout.fFlags & Layout::kTracked_Flag) {
1268 if (!mapper->supportsTracking()) {
1269 fErrors.error(decl.fOffset, String(decl.fVar->fName)
1270 + "'s type does not support state tracking");
1271 return false;
1272 }
1273 }
1274
1275 } else {
1276 // If it's not a uniform_in, it's an error to be tracked
1277 if (decl.fVar->fModifiers.fLayout.fFlags & Layout::kTracked_Flag) {
1278 fErrors.error(decl.fOffset, "Non-'in uniforms' cannot be tracked");
1279 return false;
1280 }
1281 }
Ethan Nicholas762466e2017-06-29 10:03:38 -04001282 }
1283 }
1284 }
1285 const char* baseName = fName.c_str();
1286 const char* fullName = fFullName.c_str();
Ethan Nicholas130fb3f2018-02-01 12:14:34 -05001287 this->writef("%s\n", HCodeGenerator::GetHeader(fProgram, fErrors).c_str());
Ethan Nicholas762466e2017-06-29 10:03:38 -04001288 this->writef(kFragmentProcessorHeader, fullName);
Mike Kleinc0bd9f92019-04-23 12:05:21 -05001289 this->writef("#include \"%s.h\"\n\n", fullName);
Ethan Nicholas9fb036f2017-07-05 16:19:09 -04001290 this->writeSection(CPP_SECTION);
Mike Kleinc0bd9f92019-04-23 12:05:21 -05001291 this->writef("#include \"include/gpu/GrTexture.h\"\n"
1292 "#include \"src/gpu/glsl/GrGLSLFragmentProcessor.h\"\n"
1293 "#include \"src/gpu/glsl/GrGLSLFragmentShaderBuilder.h\"\n"
1294 "#include \"src/gpu/glsl/GrGLSLProgramBuilder.h\"\n"
1295 "#include \"src/sksl/SkSLCPP.h\"\n"
1296 "#include \"src/sksl/SkSLUtil.h\"\n"
Ethan Nicholas762466e2017-06-29 10:03:38 -04001297 "class GrGLSL%s : public GrGLSLFragmentProcessor {\n"
1298 "public:\n"
1299 " GrGLSL%s() {}\n",
Ethan Nicholas9fb036f2017-07-05 16:19:09 -04001300 baseName, baseName);
Ethan Nicholas762466e2017-06-29 10:03:38 -04001301 bool result = this->writeEmitCode(uniforms);
1302 this->write("private:\n");
1303 this->writeSetData(uniforms);
1304 this->writePrivateVars();
1305 for (const auto& u : uniforms) {
Ethan Nicholas762466e2017-06-29 10:03:38 -04001306 if (needs_uniform_var(*u) && !(u->fModifiers.fFlags & Modifiers::kIn_Flag)) {
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001307 this->writef(" UniformHandle %sVar;\n",
1308 HCodeGenerator::FieldName(String(u->fName).c_str()).c_str());
Ethan Nicholas762466e2017-06-29 10:03:38 -04001309 }
1310 }
Ethan Nicholas68990be2017-07-13 09:36:52 -04001311 for (const auto& param : fSectionAndParameterHelper.getParameters()) {
Ethan Nicholas762466e2017-06-29 10:03:38 -04001312 if (needs_uniform_var(*param)) {
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001313 this->writef(" UniformHandle %sVar;\n",
1314 HCodeGenerator::FieldName(String(param->fName).c_str()).c_str());
Ethan Nicholas762466e2017-06-29 10:03:38 -04001315 }
1316 }
Ethan Nicholas762466e2017-06-29 10:03:38 -04001317 this->writef("};\n"
1318 "GrGLSLFragmentProcessor* %s::onCreateGLSLInstance() const {\n"
1319 " return new GrGLSL%s();\n"
1320 "}\n",
1321 fullName, baseName);
1322 this->writeGetKey();
1323 this->writef("bool %s::onIsEqual(const GrFragmentProcessor& other) const {\n"
1324 " const %s& that = other.cast<%s>();\n"
1325 " (void) that;\n",
1326 fullName, fullName, fullName);
Ethan Nicholas68990be2017-07-13 09:36:52 -04001327 for (const auto& param : fSectionAndParameterHelper.getParameters()) {
Ethan Nicholasee1c8a72019-02-22 10:50:47 -05001328 if (param->fType.nonnullable() == *fContext.fFragmentProcessor_Type) {
Ethan Nicholasc9472af2017-10-10 16:30:21 -04001329 continue;
1330 }
Ethan Nicholas5b5f0962017-09-11 13:50:14 -07001331 String nameString(param->fName);
1332 const char* name = nameString.c_str();
Ethan Nicholas762466e2017-06-29 10:03:38 -04001333 this->writef(" if (%s != that.%s) return false;\n",
1334 HCodeGenerator::FieldName(name).c_str(),
1335 HCodeGenerator::FieldName(name).c_str());
1336 }
1337 this->write(" return true;\n"
1338 "}\n");
Ethan Nicholasf57c0d62017-07-31 11:18:22 -04001339 this->writeClone();
Brian Salomonf7dcd762018-07-30 14:48:15 -04001340 this->writeOnTextureSampler();
Ethan Nicholas762466e2017-06-29 10:03:38 -04001341 this->writeTest();
Ethan Nicholas9fb036f2017-07-05 16:19:09 -04001342 this->writeSection(CPP_END_SECTION);
Greg Daniel3e8c3452018-04-06 10:37:55 -04001343
Ethan Nicholas762466e2017-06-29 10:03:38 -04001344 result &= 0 == fErrors.errorCount();
1345 return result;
1346}
1347
1348} // namespace