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/SkSLForStatement.h"
9*c8dee2aaSAndroid Build Coastguard Worker
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkTypes.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLAnalysis.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLBuiltinTypes.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLContext.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLDefines.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLErrorReporter.h"
16*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLProgramSettings.h"
17*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/analysis/SkSLProgramVisitor.h"
18*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLBlock.h"
19*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLExpressionStatement.h"
20*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLNop.h"
21*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLType.h"
22*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLVarDeclarations.h"
23*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLVariable.h"
24*c8dee2aaSAndroid Build Coastguard Worker
25*c8dee2aaSAndroid Build Coastguard Worker namespace SkSL {
26*c8dee2aaSAndroid Build Coastguard Worker
is_vardecl_block_initializer(const Statement * stmt)27*c8dee2aaSAndroid Build Coastguard Worker static bool is_vardecl_block_initializer(const Statement* stmt) {
28*c8dee2aaSAndroid Build Coastguard Worker if (!stmt) {
29*c8dee2aaSAndroid Build Coastguard Worker return false;
30*c8dee2aaSAndroid Build Coastguard Worker }
31*c8dee2aaSAndroid Build Coastguard Worker if (!stmt->is<SkSL::Block>()) {
32*c8dee2aaSAndroid Build Coastguard Worker return false;
33*c8dee2aaSAndroid Build Coastguard Worker }
34*c8dee2aaSAndroid Build Coastguard Worker const SkSL::Block& b = stmt->as<SkSL::Block>();
35*c8dee2aaSAndroid Build Coastguard Worker if (b.isScope()) {
36*c8dee2aaSAndroid Build Coastguard Worker return false;
37*c8dee2aaSAndroid Build Coastguard Worker }
38*c8dee2aaSAndroid Build Coastguard Worker for (const auto& child : b.children()) {
39*c8dee2aaSAndroid Build Coastguard Worker if (!child->is<SkSL::VarDeclaration>()) {
40*c8dee2aaSAndroid Build Coastguard Worker return false;
41*c8dee2aaSAndroid Build Coastguard Worker }
42*c8dee2aaSAndroid Build Coastguard Worker }
43*c8dee2aaSAndroid Build Coastguard Worker return true;
44*c8dee2aaSAndroid Build Coastguard Worker }
45*c8dee2aaSAndroid Build Coastguard Worker
is_simple_initializer(const Statement * stmt)46*c8dee2aaSAndroid Build Coastguard Worker static bool is_simple_initializer(const Statement* stmt) {
47*c8dee2aaSAndroid Build Coastguard Worker return !stmt || stmt->isEmpty() || stmt->is<SkSL::VarDeclaration>() ||
48*c8dee2aaSAndroid Build Coastguard Worker stmt->is<SkSL::ExpressionStatement>();
49*c8dee2aaSAndroid Build Coastguard Worker }
50*c8dee2aaSAndroid Build Coastguard Worker
description() const51*c8dee2aaSAndroid Build Coastguard Worker std::string ForStatement::description() const {
52*c8dee2aaSAndroid Build Coastguard Worker std::string result("for (");
53*c8dee2aaSAndroid Build Coastguard Worker if (this->initializer()) {
54*c8dee2aaSAndroid Build Coastguard Worker result += this->initializer()->description();
55*c8dee2aaSAndroid Build Coastguard Worker } else {
56*c8dee2aaSAndroid Build Coastguard Worker result += ";";
57*c8dee2aaSAndroid Build Coastguard Worker }
58*c8dee2aaSAndroid Build Coastguard Worker result += " ";
59*c8dee2aaSAndroid Build Coastguard Worker if (this->test()) {
60*c8dee2aaSAndroid Build Coastguard Worker result += this->test()->description();
61*c8dee2aaSAndroid Build Coastguard Worker }
62*c8dee2aaSAndroid Build Coastguard Worker result += "; ";
63*c8dee2aaSAndroid Build Coastguard Worker if (this->next()) {
64*c8dee2aaSAndroid Build Coastguard Worker result += this->next()->description();
65*c8dee2aaSAndroid Build Coastguard Worker }
66*c8dee2aaSAndroid Build Coastguard Worker result += ") " + this->statement()->description();
67*c8dee2aaSAndroid Build Coastguard Worker return result;
68*c8dee2aaSAndroid Build Coastguard Worker }
69*c8dee2aaSAndroid Build Coastguard Worker
hoist_vardecl_symbols_into_outer_scope(const Context & context,const Block & initBlock,SymbolTable * innerSymbols,SymbolTable * hoistedSymbols)70*c8dee2aaSAndroid Build Coastguard Worker static void hoist_vardecl_symbols_into_outer_scope(const Context& context,
71*c8dee2aaSAndroid Build Coastguard Worker const Block& initBlock,
72*c8dee2aaSAndroid Build Coastguard Worker SymbolTable* innerSymbols,
73*c8dee2aaSAndroid Build Coastguard Worker SymbolTable* hoistedSymbols) {
74*c8dee2aaSAndroid Build Coastguard Worker class SymbolHoister : public ProgramVisitor {
75*c8dee2aaSAndroid Build Coastguard Worker public:
76*c8dee2aaSAndroid Build Coastguard Worker SymbolHoister(const Context& ctx, SymbolTable* innerSym, SymbolTable* hoistSym)
77*c8dee2aaSAndroid Build Coastguard Worker : fContext(ctx)
78*c8dee2aaSAndroid Build Coastguard Worker , fInnerSymbols(innerSym)
79*c8dee2aaSAndroid Build Coastguard Worker , fHoistedSymbols(hoistSym) {}
80*c8dee2aaSAndroid Build Coastguard Worker
81*c8dee2aaSAndroid Build Coastguard Worker bool visitStatement(const Statement& stmt) override {
82*c8dee2aaSAndroid Build Coastguard Worker if (stmt.is<VarDeclaration>()) {
83*c8dee2aaSAndroid Build Coastguard Worker // Hoist the variable's symbol outside of the initializer block's symbol table, and
84*c8dee2aaSAndroid Build Coastguard Worker // into the outer symbol table. If the initializer's symbol table originally had
85*c8dee2aaSAndroid Build Coastguard Worker // ownership, transfer it. (If the variable was owned elsewhere, it can keep its
86*c8dee2aaSAndroid Build Coastguard Worker // current owner.)
87*c8dee2aaSAndroid Build Coastguard Worker Variable* var = stmt.as<VarDeclaration>().var();
88*c8dee2aaSAndroid Build Coastguard Worker fInnerSymbols->moveSymbolTo(fHoistedSymbols, var, fContext);
89*c8dee2aaSAndroid Build Coastguard Worker return false;
90*c8dee2aaSAndroid Build Coastguard Worker }
91*c8dee2aaSAndroid Build Coastguard Worker return ProgramVisitor::visitStatement(stmt);
92*c8dee2aaSAndroid Build Coastguard Worker }
93*c8dee2aaSAndroid Build Coastguard Worker
94*c8dee2aaSAndroid Build Coastguard Worker const Context& fContext;
95*c8dee2aaSAndroid Build Coastguard Worker SymbolTable* fInnerSymbols;
96*c8dee2aaSAndroid Build Coastguard Worker SymbolTable* fHoistedSymbols;
97*c8dee2aaSAndroid Build Coastguard Worker };
98*c8dee2aaSAndroid Build Coastguard Worker
99*c8dee2aaSAndroid Build Coastguard Worker SymbolHoister{context, innerSymbols, hoistedSymbols}.visitStatement(initBlock);
100*c8dee2aaSAndroid Build Coastguard Worker }
101*c8dee2aaSAndroid Build Coastguard Worker
Convert(const Context & context,Position pos,ForLoopPositions positions,std::unique_ptr<Statement> initializer,std::unique_ptr<Expression> test,std::unique_ptr<Expression> next,std::unique_ptr<Statement> statement,std::unique_ptr<SymbolTable> symbolTable)102*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Statement> ForStatement::Convert(const Context& context,
103*c8dee2aaSAndroid Build Coastguard Worker Position pos,
104*c8dee2aaSAndroid Build Coastguard Worker ForLoopPositions positions,
105*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Statement> initializer,
106*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> test,
107*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> next,
108*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Statement> statement,
109*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<SymbolTable> symbolTable) {
110*c8dee2aaSAndroid Build Coastguard Worker bool isSimpleInitializer = is_simple_initializer(initializer.get());
111*c8dee2aaSAndroid Build Coastguard Worker bool isVardeclBlockInitializer = !isSimpleInitializer &&
112*c8dee2aaSAndroid Build Coastguard Worker is_vardecl_block_initializer(initializer.get());
113*c8dee2aaSAndroid Build Coastguard Worker
114*c8dee2aaSAndroid Build Coastguard Worker if (!isSimpleInitializer && !isVardeclBlockInitializer) {
115*c8dee2aaSAndroid Build Coastguard Worker context.fErrors->error(initializer->fPosition, "invalid for loop initializer");
116*c8dee2aaSAndroid Build Coastguard Worker return nullptr;
117*c8dee2aaSAndroid Build Coastguard Worker }
118*c8dee2aaSAndroid Build Coastguard Worker
119*c8dee2aaSAndroid Build Coastguard Worker if (test) {
120*c8dee2aaSAndroid Build Coastguard Worker test = context.fTypes.fBool->coerceExpression(std::move(test), context);
121*c8dee2aaSAndroid Build Coastguard Worker if (!test) {
122*c8dee2aaSAndroid Build Coastguard Worker return nullptr;
123*c8dee2aaSAndroid Build Coastguard Worker }
124*c8dee2aaSAndroid Build Coastguard Worker }
125*c8dee2aaSAndroid Build Coastguard Worker
126*c8dee2aaSAndroid Build Coastguard Worker // The type of the next-expression doesn't matter, but it needs to be a complete expression.
127*c8dee2aaSAndroid Build Coastguard Worker // Report an error on intermediate expressions like FunctionReference or TypeReference.
128*c8dee2aaSAndroid Build Coastguard Worker if (next && next->isIncomplete(context)) {
129*c8dee2aaSAndroid Build Coastguard Worker return nullptr;
130*c8dee2aaSAndroid Build Coastguard Worker }
131*c8dee2aaSAndroid Build Coastguard Worker
132*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<LoopUnrollInfo> unrollInfo;
133*c8dee2aaSAndroid Build Coastguard Worker if (context.fConfig->strictES2Mode()) {
134*c8dee2aaSAndroid Build Coastguard Worker // In strict-ES2, loops must be unrollable or it's an error.
135*c8dee2aaSAndroid Build Coastguard Worker unrollInfo = Analysis::GetLoopUnrollInfo(context, pos, positions, initializer.get(), &test,
136*c8dee2aaSAndroid Build Coastguard Worker next.get(), statement.get(), context.fErrors);
137*c8dee2aaSAndroid Build Coastguard Worker if (!unrollInfo) {
138*c8dee2aaSAndroid Build Coastguard Worker return nullptr;
139*c8dee2aaSAndroid Build Coastguard Worker }
140*c8dee2aaSAndroid Build Coastguard Worker } else {
141*c8dee2aaSAndroid Build Coastguard Worker // In ES3, loops don't have to be unrollable, but we can use the unroll information for
142*c8dee2aaSAndroid Build Coastguard Worker // optimization purposes.
143*c8dee2aaSAndroid Build Coastguard Worker unrollInfo = Analysis::GetLoopUnrollInfo(context, pos, positions, initializer.get(), &test,
144*c8dee2aaSAndroid Build Coastguard Worker next.get(), statement.get(), /*errors=*/nullptr);
145*c8dee2aaSAndroid Build Coastguard Worker }
146*c8dee2aaSAndroid Build Coastguard Worker
147*c8dee2aaSAndroid Build Coastguard Worker if (Analysis::DetectVarDeclarationWithoutScope(*statement, context.fErrors)) {
148*c8dee2aaSAndroid Build Coastguard Worker return nullptr;
149*c8dee2aaSAndroid Build Coastguard Worker }
150*c8dee2aaSAndroid Build Coastguard Worker
151*c8dee2aaSAndroid Build Coastguard Worker if (isVardeclBlockInitializer) {
152*c8dee2aaSAndroid Build Coastguard Worker // If the initializer statement of a for loop contains multiple variables, this causes
153*c8dee2aaSAndroid Build Coastguard Worker // difficulties for several of our backends; e.g. Metal doesn't have a way to express arrays
154*c8dee2aaSAndroid Build Coastguard Worker // of different size in the same decl-stmt, because the array-size is part of the type. It's
155*c8dee2aaSAndroid Build Coastguard Worker // conceptually equivalent to synthesize a scope, declare the variables, and then emit a for
156*c8dee2aaSAndroid Build Coastguard Worker // statement with an empty init-stmt. (Note that we can't just do this transformation
157*c8dee2aaSAndroid Build Coastguard Worker // unilaterally for all for-statements, because the resulting for loop isn't ES2-compliant.)
158*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<SymbolTable> hoistedSymbols = symbolTable->insertNewParent();
159*c8dee2aaSAndroid Build Coastguard Worker hoist_vardecl_symbols_into_outer_scope(context, initializer->as<Block>(),
160*c8dee2aaSAndroid Build Coastguard Worker symbolTable.get(), hoistedSymbols.get());
161*c8dee2aaSAndroid Build Coastguard Worker StatementArray scope;
162*c8dee2aaSAndroid Build Coastguard Worker scope.push_back(std::move(initializer));
163*c8dee2aaSAndroid Build Coastguard Worker scope.push_back(ForStatement::Make(context,
164*c8dee2aaSAndroid Build Coastguard Worker pos,
165*c8dee2aaSAndroid Build Coastguard Worker positions,
166*c8dee2aaSAndroid Build Coastguard Worker /*initializer=*/nullptr,
167*c8dee2aaSAndroid Build Coastguard Worker std::move(test),
168*c8dee2aaSAndroid Build Coastguard Worker std::move(next),
169*c8dee2aaSAndroid Build Coastguard Worker std::move(statement),
170*c8dee2aaSAndroid Build Coastguard Worker std::move(unrollInfo),
171*c8dee2aaSAndroid Build Coastguard Worker std::move(symbolTable)));
172*c8dee2aaSAndroid Build Coastguard Worker return Block::Make(pos,
173*c8dee2aaSAndroid Build Coastguard Worker std::move(scope),
174*c8dee2aaSAndroid Build Coastguard Worker Block::Kind::kBracedScope,
175*c8dee2aaSAndroid Build Coastguard Worker std::move(hoistedSymbols));
176*c8dee2aaSAndroid Build Coastguard Worker }
177*c8dee2aaSAndroid Build Coastguard Worker
178*c8dee2aaSAndroid Build Coastguard Worker return ForStatement::Make(context,
179*c8dee2aaSAndroid Build Coastguard Worker pos,
180*c8dee2aaSAndroid Build Coastguard Worker positions,
181*c8dee2aaSAndroid Build Coastguard Worker std::move(initializer),
182*c8dee2aaSAndroid Build Coastguard Worker std::move(test),
183*c8dee2aaSAndroid Build Coastguard Worker std::move(next),
184*c8dee2aaSAndroid Build Coastguard Worker std::move(statement),
185*c8dee2aaSAndroid Build Coastguard Worker std::move(unrollInfo),
186*c8dee2aaSAndroid Build Coastguard Worker std::move(symbolTable));
187*c8dee2aaSAndroid Build Coastguard Worker }
188*c8dee2aaSAndroid Build Coastguard Worker
ConvertWhile(const Context & context,Position pos,std::unique_ptr<Expression> test,std::unique_ptr<Statement> statement)189*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Statement> ForStatement::ConvertWhile(const Context& context,
190*c8dee2aaSAndroid Build Coastguard Worker Position pos,
191*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> test,
192*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Statement> statement) {
193*c8dee2aaSAndroid Build Coastguard Worker if (context.fConfig->strictES2Mode()) {
194*c8dee2aaSAndroid Build Coastguard Worker context.fErrors->error(pos, "while loops are not supported");
195*c8dee2aaSAndroid Build Coastguard Worker return nullptr;
196*c8dee2aaSAndroid Build Coastguard Worker }
197*c8dee2aaSAndroid Build Coastguard Worker return ForStatement::Convert(context,
198*c8dee2aaSAndroid Build Coastguard Worker pos,
199*c8dee2aaSAndroid Build Coastguard Worker ForLoopPositions(),
200*c8dee2aaSAndroid Build Coastguard Worker /*initializer=*/nullptr,
201*c8dee2aaSAndroid Build Coastguard Worker std::move(test),
202*c8dee2aaSAndroid Build Coastguard Worker /*next=*/nullptr,
203*c8dee2aaSAndroid Build Coastguard Worker std::move(statement),
204*c8dee2aaSAndroid Build Coastguard Worker /*symbolTable=*/nullptr);
205*c8dee2aaSAndroid Build Coastguard Worker }
206*c8dee2aaSAndroid Build Coastguard Worker
Make(const Context & context,Position pos,ForLoopPositions positions,std::unique_ptr<Statement> initializer,std::unique_ptr<Expression> test,std::unique_ptr<Expression> next,std::unique_ptr<Statement> statement,std::unique_ptr<LoopUnrollInfo> unrollInfo,std::unique_ptr<SymbolTable> symbolTable)207*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Statement> ForStatement::Make(const Context& context,
208*c8dee2aaSAndroid Build Coastguard Worker Position pos,
209*c8dee2aaSAndroid Build Coastguard Worker ForLoopPositions positions,
210*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Statement> initializer,
211*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> test,
212*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> next,
213*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Statement> statement,
214*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<LoopUnrollInfo> unrollInfo,
215*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<SymbolTable> symbolTable) {
216*c8dee2aaSAndroid Build Coastguard Worker SkASSERT(is_simple_initializer(initializer.get()) ||
217*c8dee2aaSAndroid Build Coastguard Worker is_vardecl_block_initializer(initializer.get()));
218*c8dee2aaSAndroid Build Coastguard Worker SkASSERT(!test || test->type().matches(*context.fTypes.fBool));
219*c8dee2aaSAndroid Build Coastguard Worker SkASSERT(!Analysis::DetectVarDeclarationWithoutScope(*statement));
220*c8dee2aaSAndroid Build Coastguard Worker SkASSERT(unrollInfo || !context.fConfig->strictES2Mode());
221*c8dee2aaSAndroid Build Coastguard Worker
222*c8dee2aaSAndroid Build Coastguard Worker // Unrollable loops are easy to optimize because we know initializer, test and next don't have
223*c8dee2aaSAndroid Build Coastguard Worker // interesting side effects.
224*c8dee2aaSAndroid Build Coastguard Worker if (unrollInfo) {
225*c8dee2aaSAndroid Build Coastguard Worker // A zero-iteration unrollable loop can be replaced with Nop.
226*c8dee2aaSAndroid Build Coastguard Worker // An unrollable loop with an empty body can be replaced with Nop.
227*c8dee2aaSAndroid Build Coastguard Worker if (unrollInfo->fCount <= 0 || statement->isEmpty()) {
228*c8dee2aaSAndroid Build Coastguard Worker return Nop::Make();
229*c8dee2aaSAndroid Build Coastguard Worker }
230*c8dee2aaSAndroid Build Coastguard Worker }
231*c8dee2aaSAndroid Build Coastguard Worker
232*c8dee2aaSAndroid Build Coastguard Worker return std::make_unique<ForStatement>(pos,
233*c8dee2aaSAndroid Build Coastguard Worker positions,
234*c8dee2aaSAndroid Build Coastguard Worker std::move(initializer),
235*c8dee2aaSAndroid Build Coastguard Worker std::move(test),
236*c8dee2aaSAndroid Build Coastguard Worker std::move(next),
237*c8dee2aaSAndroid Build Coastguard Worker std::move(statement),
238*c8dee2aaSAndroid Build Coastguard Worker std::move(unrollInfo),
239*c8dee2aaSAndroid Build Coastguard Worker std::move(symbolTable));
240*c8dee2aaSAndroid Build Coastguard Worker }
241*c8dee2aaSAndroid Build Coastguard Worker
242*c8dee2aaSAndroid Build Coastguard Worker } // namespace SkSL
243