1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker * Copyright 2021 Google LLC
3*c8dee2aaSAndroid Build Coastguard Worker *
4*c8dee2aaSAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker */
7*c8dee2aaSAndroid Build Coastguard Worker
8*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLFunctionDefinition.h"
9*c8dee2aaSAndroid Build Coastguard Worker
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkSpan.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkTypes.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "src/base/SkSafeMath.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLAnalysis.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLCompiler.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLContext.h"
16*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLDefines.h"
17*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLErrorReporter.h"
18*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLOperator.h"
19*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLProgramSettings.h"
20*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLBinaryExpression.h"
21*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLBlock.h"
22*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLExpression.h"
23*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLExpressionStatement.h"
24*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLFieldSymbol.h"
25*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLIRHelpers.h"
26*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLNop.h"
27*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLReturnStatement.h"
28*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLSwizzle.h"
29*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLSymbol.h"
30*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLSymbolTable.h" // IWYU pragma: keep
31*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLType.h"
32*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLVarDeclarations.h"
33*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLVariable.h"
34*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLVariableReference.h"
35*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/transform/SkSLProgramWriter.h"
36*c8dee2aaSAndroid Build Coastguard Worker
37*c8dee2aaSAndroid Build Coastguard Worker #include <algorithm>
38*c8dee2aaSAndroid Build Coastguard Worker #include <cstddef>
39*c8dee2aaSAndroid Build Coastguard Worker #include <forward_list>
40*c8dee2aaSAndroid Build Coastguard Worker
41*c8dee2aaSAndroid Build Coastguard Worker namespace SkSL {
42*c8dee2aaSAndroid Build Coastguard Worker
append_rtadjust_fixup_to_vertex_main(const Context & context,const FunctionDeclaration & decl,Block & body)43*c8dee2aaSAndroid Build Coastguard Worker static void append_rtadjust_fixup_to_vertex_main(const Context& context,
44*c8dee2aaSAndroid Build Coastguard Worker const FunctionDeclaration& decl,
45*c8dee2aaSAndroid Build Coastguard Worker Block& body) {
46*c8dee2aaSAndroid Build Coastguard Worker // If this program uses RTAdjust...
47*c8dee2aaSAndroid Build Coastguard Worker if (const SkSL::Symbol* rtAdjust = context.fSymbolTable->find(Compiler::RTADJUST_NAME)) {
48*c8dee2aaSAndroid Build Coastguard Worker // ...append a line to the end of the function body which fixes up sk_Position.
49*c8dee2aaSAndroid Build Coastguard Worker struct AppendRTAdjustFixupHelper : public IRHelpers {
50*c8dee2aaSAndroid Build Coastguard Worker AppendRTAdjustFixupHelper(const Context& ctx, const SkSL::Symbol* rtAdjust)
51*c8dee2aaSAndroid Build Coastguard Worker : IRHelpers(ctx)
52*c8dee2aaSAndroid Build Coastguard Worker , fRTAdjust(rtAdjust) {
53*c8dee2aaSAndroid Build Coastguard Worker fSkPositionField = &fContext.fSymbolTable->find(Compiler::POSITION_NAME)
54*c8dee2aaSAndroid Build Coastguard Worker ->as<FieldSymbol>();
55*c8dee2aaSAndroid Build Coastguard Worker }
56*c8dee2aaSAndroid Build Coastguard Worker
57*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> Pos() const {
58*c8dee2aaSAndroid Build Coastguard Worker return Field(&fSkPositionField->owner(), fSkPositionField->fieldIndex());
59*c8dee2aaSAndroid Build Coastguard Worker }
60*c8dee2aaSAndroid Build Coastguard Worker
61*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> Adjust() const {
62*c8dee2aaSAndroid Build Coastguard Worker return fRTAdjust->instantiate(fContext, Position());
63*c8dee2aaSAndroid Build Coastguard Worker }
64*c8dee2aaSAndroid Build Coastguard Worker
65*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Statement> makeFixupStmt() const {
66*c8dee2aaSAndroid Build Coastguard Worker // sk_Position = float4(sk_Position.xy * rtAdjust.xz + sk_Position.ww * rtAdjust.yw,
67*c8dee2aaSAndroid Build Coastguard Worker // 0,
68*c8dee2aaSAndroid Build Coastguard Worker // sk_Position.w);
69*c8dee2aaSAndroid Build Coastguard Worker return Assign(
70*c8dee2aaSAndroid Build Coastguard Worker Pos(),
71*c8dee2aaSAndroid Build Coastguard Worker CtorXYZW(Add(Mul(Swizzle(Pos(), {SwizzleComponent::X, SwizzleComponent::Y}),
72*c8dee2aaSAndroid Build Coastguard Worker Swizzle(Adjust(), {SwizzleComponent::X, SwizzleComponent::Z})),
73*c8dee2aaSAndroid Build Coastguard Worker Mul(Swizzle(Pos(), {SwizzleComponent::W, SwizzleComponent::W}),
74*c8dee2aaSAndroid Build Coastguard Worker Swizzle(Adjust(), {SwizzleComponent::Y, SwizzleComponent::W}))),
75*c8dee2aaSAndroid Build Coastguard Worker Float(0.0),
76*c8dee2aaSAndroid Build Coastguard Worker Swizzle(Pos(), {SwizzleComponent::W})));
77*c8dee2aaSAndroid Build Coastguard Worker }
78*c8dee2aaSAndroid Build Coastguard Worker
79*c8dee2aaSAndroid Build Coastguard Worker const FieldSymbol* fSkPositionField;
80*c8dee2aaSAndroid Build Coastguard Worker const SkSL::Symbol* fRTAdjust;
81*c8dee2aaSAndroid Build Coastguard Worker };
82*c8dee2aaSAndroid Build Coastguard Worker
83*c8dee2aaSAndroid Build Coastguard Worker AppendRTAdjustFixupHelper helper(context, rtAdjust);
84*c8dee2aaSAndroid Build Coastguard Worker body.children().push_back(helper.makeFixupStmt());
85*c8dee2aaSAndroid Build Coastguard Worker }
86*c8dee2aaSAndroid Build Coastguard Worker }
87*c8dee2aaSAndroid Build Coastguard Worker
Convert(const Context & context,Position pos,const FunctionDeclaration & function,std::unique_ptr<Statement> body)88*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<FunctionDefinition> FunctionDefinition::Convert(const Context& context,
89*c8dee2aaSAndroid Build Coastguard Worker Position pos,
90*c8dee2aaSAndroid Build Coastguard Worker const FunctionDeclaration& function,
91*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Statement> body) {
92*c8dee2aaSAndroid Build Coastguard Worker class Finalizer : public ProgramWriter {
93*c8dee2aaSAndroid Build Coastguard Worker public:
94*c8dee2aaSAndroid Build Coastguard Worker Finalizer(const Context& context, const FunctionDeclaration& function, Position pos)
95*c8dee2aaSAndroid Build Coastguard Worker : fContext(context)
96*c8dee2aaSAndroid Build Coastguard Worker , fFunction(function) {
97*c8dee2aaSAndroid Build Coastguard Worker // Function parameters count as local variables.
98*c8dee2aaSAndroid Build Coastguard Worker for (const Variable* var : function.parameters()) {
99*c8dee2aaSAndroid Build Coastguard Worker this->addLocalVariable(var, pos);
100*c8dee2aaSAndroid Build Coastguard Worker }
101*c8dee2aaSAndroid Build Coastguard Worker }
102*c8dee2aaSAndroid Build Coastguard Worker
103*c8dee2aaSAndroid Build Coastguard Worker ~Finalizer() override {
104*c8dee2aaSAndroid Build Coastguard Worker SkASSERT(fBreakableLevel == 0);
105*c8dee2aaSAndroid Build Coastguard Worker SkASSERT(fContinuableLevel == std::forward_list<int>{0});
106*c8dee2aaSAndroid Build Coastguard Worker }
107*c8dee2aaSAndroid Build Coastguard Worker
108*c8dee2aaSAndroid Build Coastguard Worker void addLocalVariable(const Variable* var, Position pos) {
109*c8dee2aaSAndroid Build Coastguard Worker if (var->type().isOrContainsUnsizedArray()) {
110*c8dee2aaSAndroid Build Coastguard Worker if (var->storage() != Variable::Storage::kParameter) {
111*c8dee2aaSAndroid Build Coastguard Worker fContext.fErrors->error(pos, "unsized arrays are not permitted here");
112*c8dee2aaSAndroid Build Coastguard Worker }
113*c8dee2aaSAndroid Build Coastguard Worker // Number of slots does not apply to unsized arrays since they are
114*c8dee2aaSAndroid Build Coastguard Worker // dynamically sized.
115*c8dee2aaSAndroid Build Coastguard Worker return;
116*c8dee2aaSAndroid Build Coastguard Worker }
117*c8dee2aaSAndroid Build Coastguard Worker // We count the number of slots used, but don't consider the precision of the base type.
118*c8dee2aaSAndroid Build Coastguard Worker // In practice, this reflects what GPUs actually do pretty well. (i.e., RelaxedPrecision
119*c8dee2aaSAndroid Build Coastguard Worker // math doesn't mean your variable takes less space.) We also don't attempt to reclaim
120*c8dee2aaSAndroid Build Coastguard Worker // slots at the end of a Block.
121*c8dee2aaSAndroid Build Coastguard Worker size_t prevSlotsUsed = fSlotsUsed;
122*c8dee2aaSAndroid Build Coastguard Worker fSlotsUsed = SkSafeMath::Add(fSlotsUsed, var->type().slotCount());
123*c8dee2aaSAndroid Build Coastguard Worker // To avoid overzealous error reporting, only trigger the error at the first
124*c8dee2aaSAndroid Build Coastguard Worker // place where the stack limit is exceeded.
125*c8dee2aaSAndroid Build Coastguard Worker if (prevSlotsUsed < kVariableSlotLimit && fSlotsUsed >= kVariableSlotLimit) {
126*c8dee2aaSAndroid Build Coastguard Worker fContext.fErrors->error(pos, "variable '" + std::string(var->name()) +
127*c8dee2aaSAndroid Build Coastguard Worker "' exceeds the stack size limit");
128*c8dee2aaSAndroid Build Coastguard Worker }
129*c8dee2aaSAndroid Build Coastguard Worker }
130*c8dee2aaSAndroid Build Coastguard Worker
131*c8dee2aaSAndroid Build Coastguard Worker void fuseVariableDeclarationsWithInitialization(std::unique_ptr<Statement>& stmt) {
132*c8dee2aaSAndroid Build Coastguard Worker switch (stmt->kind()) {
133*c8dee2aaSAndroid Build Coastguard Worker case Statement::Kind::kNop:
134*c8dee2aaSAndroid Build Coastguard Worker case Statement::Kind::kBlock:
135*c8dee2aaSAndroid Build Coastguard Worker // Blocks and no-ops are inert; it is safe to fuse a variable declaration with
136*c8dee2aaSAndroid Build Coastguard Worker // its initialization across a nop or an open-brace, so we don't null out
137*c8dee2aaSAndroid Build Coastguard Worker // `fUninitializedVarDecl` here.
138*c8dee2aaSAndroid Build Coastguard Worker break;
139*c8dee2aaSAndroid Build Coastguard Worker
140*c8dee2aaSAndroid Build Coastguard Worker case Statement::Kind::kVarDeclaration:
141*c8dee2aaSAndroid Build Coastguard Worker // Look for variable declarations without an initializer.
142*c8dee2aaSAndroid Build Coastguard Worker if (VarDeclaration& decl = stmt->as<VarDeclaration>(); !decl.value()) {
143*c8dee2aaSAndroid Build Coastguard Worker fUninitializedVarDecl = &decl;
144*c8dee2aaSAndroid Build Coastguard Worker break;
145*c8dee2aaSAndroid Build Coastguard Worker }
146*c8dee2aaSAndroid Build Coastguard Worker [[fallthrough]];
147*c8dee2aaSAndroid Build Coastguard Worker
148*c8dee2aaSAndroid Build Coastguard Worker default:
149*c8dee2aaSAndroid Build Coastguard Worker // We found an intervening statement; it's not safe to fuse a declaration
150*c8dee2aaSAndroid Build Coastguard Worker // with an initializer if we encounter any other code.
151*c8dee2aaSAndroid Build Coastguard Worker fUninitializedVarDecl = nullptr;
152*c8dee2aaSAndroid Build Coastguard Worker break;
153*c8dee2aaSAndroid Build Coastguard Worker
154*c8dee2aaSAndroid Build Coastguard Worker case Statement::Kind::kExpression: {
155*c8dee2aaSAndroid Build Coastguard Worker // We found an expression-statement. If there was a variable declaration
156*c8dee2aaSAndroid Build Coastguard Worker // immediately above it, it might be possible to fuse them.
157*c8dee2aaSAndroid Build Coastguard Worker if (fUninitializedVarDecl) {
158*c8dee2aaSAndroid Build Coastguard Worker VarDeclaration* vardecl = fUninitializedVarDecl;
159*c8dee2aaSAndroid Build Coastguard Worker fUninitializedVarDecl = nullptr;
160*c8dee2aaSAndroid Build Coastguard Worker
161*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression>& nextExpr = stmt->as<ExpressionStatement>()
162*c8dee2aaSAndroid Build Coastguard Worker .expression();
163*c8dee2aaSAndroid Build Coastguard Worker // This statement must be a binary-expression...
164*c8dee2aaSAndroid Build Coastguard Worker if (!nextExpr->is<BinaryExpression>()) {
165*c8dee2aaSAndroid Build Coastguard Worker break;
166*c8dee2aaSAndroid Build Coastguard Worker }
167*c8dee2aaSAndroid Build Coastguard Worker // ... performing simple `var = expr` assignment...
168*c8dee2aaSAndroid Build Coastguard Worker BinaryExpression& binaryExpr = nextExpr->as<BinaryExpression>();
169*c8dee2aaSAndroid Build Coastguard Worker if (binaryExpr.getOperator().kind() != OperatorKind::EQ) {
170*c8dee2aaSAndroid Build Coastguard Worker break;
171*c8dee2aaSAndroid Build Coastguard Worker }
172*c8dee2aaSAndroid Build Coastguard Worker // ... directly into the variable (not a field/swizzle)...
173*c8dee2aaSAndroid Build Coastguard Worker Expression& leftExpr = *binaryExpr.left();
174*c8dee2aaSAndroid Build Coastguard Worker if (!leftExpr.is<VariableReference>()) {
175*c8dee2aaSAndroid Build Coastguard Worker break;
176*c8dee2aaSAndroid Build Coastguard Worker }
177*c8dee2aaSAndroid Build Coastguard Worker // ... and it must be the same variable as our vardecl.
178*c8dee2aaSAndroid Build Coastguard Worker VariableReference& varRef = leftExpr.as<VariableReference>();
179*c8dee2aaSAndroid Build Coastguard Worker if (varRef.variable() != vardecl->var()) {
180*c8dee2aaSAndroid Build Coastguard Worker break;
181*c8dee2aaSAndroid Build Coastguard Worker }
182*c8dee2aaSAndroid Build Coastguard Worker // The init-expression must not reference the variable.
183*c8dee2aaSAndroid Build Coastguard Worker // `int x; x = x = 0;` is legal SkSL, but `int x = x = 0;` is not.
184*c8dee2aaSAndroid Build Coastguard Worker if (Analysis::ContainsVariable(*binaryExpr.right(), *varRef.variable())) {
185*c8dee2aaSAndroid Build Coastguard Worker break;
186*c8dee2aaSAndroid Build Coastguard Worker }
187*c8dee2aaSAndroid Build Coastguard Worker // We found a match! Move the init-expression directly onto the vardecl, and
188*c8dee2aaSAndroid Build Coastguard Worker // turn the assignment into a no-op.
189*c8dee2aaSAndroid Build Coastguard Worker vardecl->value() = std::move(binaryExpr.right());
190*c8dee2aaSAndroid Build Coastguard Worker
191*c8dee2aaSAndroid Build Coastguard Worker // Turn the expression-statement into a no-op.
192*c8dee2aaSAndroid Build Coastguard Worker stmt = Nop::Make();
193*c8dee2aaSAndroid Build Coastguard Worker }
194*c8dee2aaSAndroid Build Coastguard Worker break;
195*c8dee2aaSAndroid Build Coastguard Worker }
196*c8dee2aaSAndroid Build Coastguard Worker }
197*c8dee2aaSAndroid Build Coastguard Worker }
198*c8dee2aaSAndroid Build Coastguard Worker
199*c8dee2aaSAndroid Build Coastguard Worker bool functionReturnsValue() const {
200*c8dee2aaSAndroid Build Coastguard Worker return !fFunction.returnType().isVoid();
201*c8dee2aaSAndroid Build Coastguard Worker }
202*c8dee2aaSAndroid Build Coastguard Worker
203*c8dee2aaSAndroid Build Coastguard Worker bool visitExpressionPtr(std::unique_ptr<Expression>& expr) override {
204*c8dee2aaSAndroid Build Coastguard Worker // We don't need to scan expressions.
205*c8dee2aaSAndroid Build Coastguard Worker return false;
206*c8dee2aaSAndroid Build Coastguard Worker }
207*c8dee2aaSAndroid Build Coastguard Worker
208*c8dee2aaSAndroid Build Coastguard Worker bool visitStatementPtr(std::unique_ptr<Statement>& stmt) override {
209*c8dee2aaSAndroid Build Coastguard Worker // When the optimizer is on, we look for variable declarations that are immediately
210*c8dee2aaSAndroid Build Coastguard Worker // followed by an initialization expression, and fuse them into one statement.
211*c8dee2aaSAndroid Build Coastguard Worker // (e.g.: `int i; i = 1;` can become `int i = 1;`)
212*c8dee2aaSAndroid Build Coastguard Worker if (fContext.fConfig->fSettings.fOptimize) {
213*c8dee2aaSAndroid Build Coastguard Worker this->fuseVariableDeclarationsWithInitialization(stmt);
214*c8dee2aaSAndroid Build Coastguard Worker }
215*c8dee2aaSAndroid Build Coastguard Worker
216*c8dee2aaSAndroid Build Coastguard Worker // Perform error checking.
217*c8dee2aaSAndroid Build Coastguard Worker switch (stmt->kind()) {
218*c8dee2aaSAndroid Build Coastguard Worker case Statement::Kind::kVarDeclaration:
219*c8dee2aaSAndroid Build Coastguard Worker this->addLocalVariable(stmt->as<VarDeclaration>().var(), stmt->fPosition);
220*c8dee2aaSAndroid Build Coastguard Worker break;
221*c8dee2aaSAndroid Build Coastguard Worker
222*c8dee2aaSAndroid Build Coastguard Worker case Statement::Kind::kReturn: {
223*c8dee2aaSAndroid Build Coastguard Worker // Early returns from a vertex main() function will bypass sk_Position
224*c8dee2aaSAndroid Build Coastguard Worker // normalization, so SkASSERT that we aren't doing that. If this becomes an
225*c8dee2aaSAndroid Build Coastguard Worker // issue, we can add normalization before each return statement.
226*c8dee2aaSAndroid Build Coastguard Worker if (ProgramConfig::IsVertex(fContext.fConfig->fKind) && fFunction.isMain()) {
227*c8dee2aaSAndroid Build Coastguard Worker fContext.fErrors->error(
228*c8dee2aaSAndroid Build Coastguard Worker stmt->fPosition,
229*c8dee2aaSAndroid Build Coastguard Worker "early returns from vertex programs are not supported");
230*c8dee2aaSAndroid Build Coastguard Worker }
231*c8dee2aaSAndroid Build Coastguard Worker
232*c8dee2aaSAndroid Build Coastguard Worker // Verify that the return statement matches the function's return type.
233*c8dee2aaSAndroid Build Coastguard Worker ReturnStatement& returnStmt = stmt->as<ReturnStatement>();
234*c8dee2aaSAndroid Build Coastguard Worker if (returnStmt.expression()) {
235*c8dee2aaSAndroid Build Coastguard Worker if (this->functionReturnsValue()) {
236*c8dee2aaSAndroid Build Coastguard Worker // Coerce return expression to the function's return type.
237*c8dee2aaSAndroid Build Coastguard Worker returnStmt.setExpression(fFunction.returnType().coerceExpression(
238*c8dee2aaSAndroid Build Coastguard Worker std::move(returnStmt.expression()), fContext));
239*c8dee2aaSAndroid Build Coastguard Worker } else {
240*c8dee2aaSAndroid Build Coastguard Worker // Returning something from a function with a void return type.
241*c8dee2aaSAndroid Build Coastguard Worker fContext.fErrors->error(returnStmt.expression()->fPosition,
242*c8dee2aaSAndroid Build Coastguard Worker "may not return a value from a void function");
243*c8dee2aaSAndroid Build Coastguard Worker returnStmt.setExpression(nullptr);
244*c8dee2aaSAndroid Build Coastguard Worker }
245*c8dee2aaSAndroid Build Coastguard Worker } else {
246*c8dee2aaSAndroid Build Coastguard Worker if (this->functionReturnsValue()) {
247*c8dee2aaSAndroid Build Coastguard Worker // Returning nothing from a function with a non-void return type.
248*c8dee2aaSAndroid Build Coastguard Worker fContext.fErrors->error(returnStmt.fPosition,
249*c8dee2aaSAndroid Build Coastguard Worker "expected function to return '" +
250*c8dee2aaSAndroid Build Coastguard Worker fFunction.returnType().displayName() + "'");
251*c8dee2aaSAndroid Build Coastguard Worker }
252*c8dee2aaSAndroid Build Coastguard Worker }
253*c8dee2aaSAndroid Build Coastguard Worker break;
254*c8dee2aaSAndroid Build Coastguard Worker }
255*c8dee2aaSAndroid Build Coastguard Worker case Statement::Kind::kDo:
256*c8dee2aaSAndroid Build Coastguard Worker case Statement::Kind::kFor: {
257*c8dee2aaSAndroid Build Coastguard Worker ++fBreakableLevel;
258*c8dee2aaSAndroid Build Coastguard Worker ++fContinuableLevel.front();
259*c8dee2aaSAndroid Build Coastguard Worker bool result = INHERITED::visitStatementPtr(stmt);
260*c8dee2aaSAndroid Build Coastguard Worker --fContinuableLevel.front();
261*c8dee2aaSAndroid Build Coastguard Worker --fBreakableLevel;
262*c8dee2aaSAndroid Build Coastguard Worker return result;
263*c8dee2aaSAndroid Build Coastguard Worker }
264*c8dee2aaSAndroid Build Coastguard Worker case Statement::Kind::kSwitch: {
265*c8dee2aaSAndroid Build Coastguard Worker ++fBreakableLevel;
266*c8dee2aaSAndroid Build Coastguard Worker fContinuableLevel.push_front(0);
267*c8dee2aaSAndroid Build Coastguard Worker bool result = INHERITED::visitStatementPtr(stmt);
268*c8dee2aaSAndroid Build Coastguard Worker fContinuableLevel.pop_front();
269*c8dee2aaSAndroid Build Coastguard Worker --fBreakableLevel;
270*c8dee2aaSAndroid Build Coastguard Worker return result;
271*c8dee2aaSAndroid Build Coastguard Worker }
272*c8dee2aaSAndroid Build Coastguard Worker case Statement::Kind::kBreak:
273*c8dee2aaSAndroid Build Coastguard Worker if (fBreakableLevel == 0) {
274*c8dee2aaSAndroid Build Coastguard Worker fContext.fErrors->error(stmt->fPosition,
275*c8dee2aaSAndroid Build Coastguard Worker "break statement must be inside a loop or switch");
276*c8dee2aaSAndroid Build Coastguard Worker }
277*c8dee2aaSAndroid Build Coastguard Worker break;
278*c8dee2aaSAndroid Build Coastguard Worker
279*c8dee2aaSAndroid Build Coastguard Worker case Statement::Kind::kContinue:
280*c8dee2aaSAndroid Build Coastguard Worker if (fContinuableLevel.front() == 0) {
281*c8dee2aaSAndroid Build Coastguard Worker if (std::any_of(fContinuableLevel.begin(),
282*c8dee2aaSAndroid Build Coastguard Worker fContinuableLevel.end(),
283*c8dee2aaSAndroid Build Coastguard Worker [](int level) { return level > 0; })) {
284*c8dee2aaSAndroid Build Coastguard Worker fContext.fErrors->error(stmt->fPosition,
285*c8dee2aaSAndroid Build Coastguard Worker "continue statement cannot be used in a switch");
286*c8dee2aaSAndroid Build Coastguard Worker } else {
287*c8dee2aaSAndroid Build Coastguard Worker fContext.fErrors->error(stmt->fPosition,
288*c8dee2aaSAndroid Build Coastguard Worker "continue statement must be inside a loop");
289*c8dee2aaSAndroid Build Coastguard Worker }
290*c8dee2aaSAndroid Build Coastguard Worker }
291*c8dee2aaSAndroid Build Coastguard Worker break;
292*c8dee2aaSAndroid Build Coastguard Worker
293*c8dee2aaSAndroid Build Coastguard Worker default:
294*c8dee2aaSAndroid Build Coastguard Worker break;
295*c8dee2aaSAndroid Build Coastguard Worker }
296*c8dee2aaSAndroid Build Coastguard Worker return INHERITED::visitStatementPtr(stmt);
297*c8dee2aaSAndroid Build Coastguard Worker }
298*c8dee2aaSAndroid Build Coastguard Worker
299*c8dee2aaSAndroid Build Coastguard Worker private:
300*c8dee2aaSAndroid Build Coastguard Worker const Context& fContext;
301*c8dee2aaSAndroid Build Coastguard Worker const FunctionDeclaration& fFunction;
302*c8dee2aaSAndroid Build Coastguard Worker // how deeply nested we are in breakable constructs (for, do, switch).
303*c8dee2aaSAndroid Build Coastguard Worker int fBreakableLevel = 0;
304*c8dee2aaSAndroid Build Coastguard Worker // number of slots consumed by all variables declared in the function
305*c8dee2aaSAndroid Build Coastguard Worker size_t fSlotsUsed = 0;
306*c8dee2aaSAndroid Build Coastguard Worker // how deeply nested we are in continuable constructs (for, do).
307*c8dee2aaSAndroid Build Coastguard Worker // We keep a stack (via a forward_list) in order to disallow continue inside of switch.
308*c8dee2aaSAndroid Build Coastguard Worker std::forward_list<int> fContinuableLevel{0};
309*c8dee2aaSAndroid Build Coastguard Worker // We track uninitialized variable declarations, and if they are immediately assigned-to,
310*c8dee2aaSAndroid Build Coastguard Worker // we can move the assignment directly into the decl.
311*c8dee2aaSAndroid Build Coastguard Worker VarDeclaration* fUninitializedVarDecl = nullptr;
312*c8dee2aaSAndroid Build Coastguard Worker
313*c8dee2aaSAndroid Build Coastguard Worker using INHERITED = ProgramWriter;
314*c8dee2aaSAndroid Build Coastguard Worker };
315*c8dee2aaSAndroid Build Coastguard Worker
316*c8dee2aaSAndroid Build Coastguard Worker // We don't allow modules to define actual functions with intrinsic names. (Those should be
317*c8dee2aaSAndroid Build Coastguard Worker // reserved for actual intrinsics.)
318*c8dee2aaSAndroid Build Coastguard Worker if (function.isIntrinsic()) {
319*c8dee2aaSAndroid Build Coastguard Worker context.fErrors->error(pos, "intrinsic function '" + std::string(function.name()) +
320*c8dee2aaSAndroid Build Coastguard Worker "' should not have a definition");
321*c8dee2aaSAndroid Build Coastguard Worker return nullptr;
322*c8dee2aaSAndroid Build Coastguard Worker }
323*c8dee2aaSAndroid Build Coastguard Worker
324*c8dee2aaSAndroid Build Coastguard Worker // A function body must always be a braced block. (The parser should enforce this already, but
325*c8dee2aaSAndroid Build Coastguard Worker // we rely on it, so it's best to be certain.)
326*c8dee2aaSAndroid Build Coastguard Worker if (!body || !body->is<Block>() || !body->as<Block>().isScope()) {
327*c8dee2aaSAndroid Build Coastguard Worker context.fErrors->error(pos, "function body '" + function.description() +
328*c8dee2aaSAndroid Build Coastguard Worker "' must be a braced block");
329*c8dee2aaSAndroid Build Coastguard Worker return nullptr;
330*c8dee2aaSAndroid Build Coastguard Worker }
331*c8dee2aaSAndroid Build Coastguard Worker
332*c8dee2aaSAndroid Build Coastguard Worker // A function can't have more than one definition.
333*c8dee2aaSAndroid Build Coastguard Worker if (function.definition()) {
334*c8dee2aaSAndroid Build Coastguard Worker context.fErrors->error(pos, "function '" + function.description() +
335*c8dee2aaSAndroid Build Coastguard Worker "' was already defined");
336*c8dee2aaSAndroid Build Coastguard Worker return nullptr;
337*c8dee2aaSAndroid Build Coastguard Worker }
338*c8dee2aaSAndroid Build Coastguard Worker
339*c8dee2aaSAndroid Build Coastguard Worker // Run the function finalizer. This checks for illegal constructs and missing return statements,
340*c8dee2aaSAndroid Build Coastguard Worker // and also performs some simple code cleanup.
341*c8dee2aaSAndroid Build Coastguard Worker Finalizer(context, function, pos).visitStatementPtr(body);
342*c8dee2aaSAndroid Build Coastguard Worker if (function.isMain() && ProgramConfig::IsVertex(context.fConfig->fKind)) {
343*c8dee2aaSAndroid Build Coastguard Worker append_rtadjust_fixup_to_vertex_main(context, function, body->as<Block>());
344*c8dee2aaSAndroid Build Coastguard Worker }
345*c8dee2aaSAndroid Build Coastguard Worker
346*c8dee2aaSAndroid Build Coastguard Worker if (Analysis::CanExitWithoutReturningValue(function, *body)) {
347*c8dee2aaSAndroid Build Coastguard Worker context.fErrors->error(body->fPosition, "function '" + std::string(function.name()) +
348*c8dee2aaSAndroid Build Coastguard Worker "' can exit without returning a value");
349*c8dee2aaSAndroid Build Coastguard Worker }
350*c8dee2aaSAndroid Build Coastguard Worker
351*c8dee2aaSAndroid Build Coastguard Worker return FunctionDefinition::Make(context, pos, function, std::move(body));
352*c8dee2aaSAndroid Build Coastguard Worker }
353*c8dee2aaSAndroid Build Coastguard Worker
Make(const Context & context,Position pos,const FunctionDeclaration & function,std::unique_ptr<Statement> body)354*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<FunctionDefinition> FunctionDefinition::Make(const Context& context,
355*c8dee2aaSAndroid Build Coastguard Worker Position pos,
356*c8dee2aaSAndroid Build Coastguard Worker const FunctionDeclaration& function,
357*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Statement> body) {
358*c8dee2aaSAndroid Build Coastguard Worker SkASSERT(!function.isIntrinsic());
359*c8dee2aaSAndroid Build Coastguard Worker SkASSERT(body && body->as<Block>().isScope());
360*c8dee2aaSAndroid Build Coastguard Worker SkASSERT(!function.definition());
361*c8dee2aaSAndroid Build Coastguard Worker
362*c8dee2aaSAndroid Build Coastguard Worker return std::make_unique<FunctionDefinition>(pos, &function, std::move(body));
363*c8dee2aaSAndroid Build Coastguard Worker }
364*c8dee2aaSAndroid Build Coastguard Worker
365*c8dee2aaSAndroid Build Coastguard Worker } // namespace SkSL
366