blob: f5f2151fe3f1f5d1b5bc07cd998a29b178971b9f [file] [log] [blame]
ethannicholasb3058bd2016-07-01 08:22:01 -07001/*
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 */
Mike Klein6ad99092016-10-26 10:35:22 -04007
ethannicholasb3058bd2016-07-01 08:22:01 -07008#include "SkSLCompiler.h"
9
ethannicholas5961bc92016-10-12 06:39:56 -070010#include "ast/SkSLASTPrecision.h"
ethannicholas22f939e2016-10-13 13:25:34 -070011#include "SkSLCFGGenerator.h"
Ethan Nicholas941e7e22016-12-12 15:33:30 -050012#include "SkSLGLSLCodeGenerator.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070013#include "SkSLIRGenerator.h"
14#include "SkSLParser.h"
15#include "SkSLSPIRVCodeGenerator.h"
16#include "ir/SkSLExpression.h"
Ethan Nicholascb670962017-04-20 19:31:52 -040017#include "ir/SkSLExpressionStatement.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070018#include "ir/SkSLIntLiteral.h"
ethannicholas5961bc92016-10-12 06:39:56 -070019#include "ir/SkSLModifiersDeclaration.h"
Ethan Nicholascb670962017-04-20 19:31:52 -040020#include "ir/SkSLNop.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070021#include "ir/SkSLSymbolTable.h"
Ethan Nicholascb670962017-04-20 19:31:52 -040022#include "ir/SkSLTernaryExpression.h"
ethannicholasddb37d62016-10-20 09:54:00 -070023#include "ir/SkSLUnresolvedFunction.h"
ethannicholas22f939e2016-10-13 13:25:34 -070024#include "ir/SkSLVarDeclarations.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070025
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -040026#ifdef SK_ENABLE_SPIRV_VALIDATION
27#include "spirv-tools/libspirv.hpp"
28#endif
29
ethannicholasb3058bd2016-07-01 08:22:01 -070030#define STRINGIFY(x) #x
31
32// include the built-in shader symbols as static strings
33
ethannicholas5961bc92016-10-12 06:39:56 -070034static const char* SKSL_INCLUDE =
ethannicholasb3058bd2016-07-01 08:22:01 -070035#include "sksl.include"
36;
37
ethannicholas5961bc92016-10-12 06:39:56 -070038static const char* SKSL_VERT_INCLUDE =
ethannicholasb3058bd2016-07-01 08:22:01 -070039#include "sksl_vert.include"
40;
41
ethannicholas5961bc92016-10-12 06:39:56 -070042static const char* SKSL_FRAG_INCLUDE =
ethannicholasb3058bd2016-07-01 08:22:01 -070043#include "sksl_frag.include"
44;
45
Ethan Nicholas52cad152017-02-16 16:37:32 -050046static const char* SKSL_GEOM_INCLUDE =
47#include "sksl_geom.include"
48;
49
ethannicholasb3058bd2016-07-01 08:22:01 -070050namespace SkSL {
51
Mike Klein6ad99092016-10-26 10:35:22 -040052Compiler::Compiler()
ethannicholasb3058bd2016-07-01 08:22:01 -070053: fErrorCount(0) {
Ethan Nicholas8feeff92017-03-30 14:11:58 -040054 auto types = std::shared_ptr<SymbolTable>(new SymbolTable(this));
55 auto symbols = std::shared_ptr<SymbolTable>(new SymbolTable(types, this));
ethannicholasd598f792016-07-25 10:08:54 -070056 fIRGenerator = new IRGenerator(&fContext, symbols, *this);
ethannicholasb3058bd2016-07-01 08:22:01 -070057 fTypes = types;
ethannicholasd598f792016-07-25 10:08:54 -070058 #define ADD_TYPE(t) types->addWithoutOwnership(fContext.f ## t ## _Type->fName, \
59 fContext.f ## t ## _Type.get())
ethannicholasb3058bd2016-07-01 08:22:01 -070060 ADD_TYPE(Void);
61 ADD_TYPE(Float);
62 ADD_TYPE(Vec2);
63 ADD_TYPE(Vec3);
64 ADD_TYPE(Vec4);
65 ADD_TYPE(Double);
66 ADD_TYPE(DVec2);
67 ADD_TYPE(DVec3);
68 ADD_TYPE(DVec4);
69 ADD_TYPE(Int);
70 ADD_TYPE(IVec2);
71 ADD_TYPE(IVec3);
72 ADD_TYPE(IVec4);
73 ADD_TYPE(UInt);
74 ADD_TYPE(UVec2);
75 ADD_TYPE(UVec3);
76 ADD_TYPE(UVec4);
77 ADD_TYPE(Bool);
78 ADD_TYPE(BVec2);
79 ADD_TYPE(BVec3);
80 ADD_TYPE(BVec4);
81 ADD_TYPE(Mat2x2);
Ethan Nicholas0df1b042017-03-31 13:56:23 -040082 types->addWithoutOwnership(String("mat2x2"), fContext.fMat2x2_Type.get());
ethannicholasb3058bd2016-07-01 08:22:01 -070083 ADD_TYPE(Mat2x3);
84 ADD_TYPE(Mat2x4);
85 ADD_TYPE(Mat3x2);
86 ADD_TYPE(Mat3x3);
Ethan Nicholas0df1b042017-03-31 13:56:23 -040087 types->addWithoutOwnership(String("mat3x3"), fContext.fMat3x3_Type.get());
ethannicholasb3058bd2016-07-01 08:22:01 -070088 ADD_TYPE(Mat3x4);
89 ADD_TYPE(Mat4x2);
90 ADD_TYPE(Mat4x3);
91 ADD_TYPE(Mat4x4);
Ethan Nicholas0df1b042017-03-31 13:56:23 -040092 types->addWithoutOwnership(String("mat4x4"), fContext.fMat4x4_Type.get());
ethannicholasb3058bd2016-07-01 08:22:01 -070093 ADD_TYPE(GenType);
94 ADD_TYPE(GenDType);
95 ADD_TYPE(GenIType);
96 ADD_TYPE(GenUType);
97 ADD_TYPE(GenBType);
98 ADD_TYPE(Mat);
99 ADD_TYPE(Vec);
100 ADD_TYPE(GVec);
101 ADD_TYPE(GVec2);
102 ADD_TYPE(GVec3);
103 ADD_TYPE(GVec4);
104 ADD_TYPE(DVec);
105 ADD_TYPE(IVec);
106 ADD_TYPE(UVec);
107 ADD_TYPE(BVec);
108
109 ADD_TYPE(Sampler1D);
110 ADD_TYPE(Sampler2D);
111 ADD_TYPE(Sampler3D);
ethannicholas5961bc92016-10-12 06:39:56 -0700112 ADD_TYPE(SamplerExternalOES);
ethannicholasb3058bd2016-07-01 08:22:01 -0700113 ADD_TYPE(SamplerCube);
114 ADD_TYPE(Sampler2DRect);
115 ADD_TYPE(Sampler1DArray);
116 ADD_TYPE(Sampler2DArray);
117 ADD_TYPE(SamplerCubeArray);
118 ADD_TYPE(SamplerBuffer);
119 ADD_TYPE(Sampler2DMS);
120 ADD_TYPE(Sampler2DMSArray);
121
Brian Salomonbf7b6202016-11-11 16:08:03 -0500122 ADD_TYPE(ISampler2D);
123
Brian Salomon2a51de82016-11-16 12:06:01 -0500124 ADD_TYPE(Image2D);
125 ADD_TYPE(IImage2D);
126
Greg Daniel64773e62016-11-22 09:44:03 -0500127 ADD_TYPE(SubpassInput);
128 ADD_TYPE(SubpassInputMS);
129
ethannicholasb3058bd2016-07-01 08:22:01 -0700130 ADD_TYPE(GSampler1D);
131 ADD_TYPE(GSampler2D);
132 ADD_TYPE(GSampler3D);
133 ADD_TYPE(GSamplerCube);
134 ADD_TYPE(GSampler2DRect);
135 ADD_TYPE(GSampler1DArray);
136 ADD_TYPE(GSampler2DArray);
137 ADD_TYPE(GSamplerCubeArray);
138 ADD_TYPE(GSamplerBuffer);
139 ADD_TYPE(GSampler2DMS);
140 ADD_TYPE(GSampler2DMSArray);
141
142 ADD_TYPE(Sampler1DShadow);
143 ADD_TYPE(Sampler2DShadow);
144 ADD_TYPE(SamplerCubeShadow);
145 ADD_TYPE(Sampler2DRectShadow);
146 ADD_TYPE(Sampler1DArrayShadow);
147 ADD_TYPE(Sampler2DArrayShadow);
148 ADD_TYPE(SamplerCubeArrayShadow);
149 ADD_TYPE(GSampler2DArrayShadow);
150 ADD_TYPE(GSamplerCubeArrayShadow);
151
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400152 String skCapsName("sk_Caps");
153 Variable* skCaps = new Variable(Position(), Modifiers(), skCapsName,
Ethan Nicholas3605ace2016-11-21 15:59:48 -0500154 *fContext.fSkCaps_Type, Variable::kGlobal_Storage);
155 fIRGenerator->fSymbolTable->add(skCapsName, std::unique_ptr<Symbol>(skCaps));
156
ethannicholas5961bc92016-10-12 06:39:56 -0700157 Modifiers::Flag ignored1;
158 std::vector<std::unique_ptr<ProgramElement>> ignored2;
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400159 this->internalConvertProgram(String(SKSL_INCLUDE), &ignored1, &ignored2);
ethannicholasddb37d62016-10-20 09:54:00 -0700160 fIRGenerator->fSymbolTable->markAllFunctionsBuiltin();
ethannicholasb3058bd2016-07-01 08:22:01 -0700161 ASSERT(!fErrorCount);
162}
163
164Compiler::~Compiler() {
165 delete fIRGenerator;
166}
167
ethannicholas22f939e2016-10-13 13:25:34 -0700168// add the definition created by assigning to the lvalue to the definition set
Ethan Nicholas86a43402017-01-19 13:32:00 -0500169void Compiler::addDefinition(const Expression* lvalue, std::unique_ptr<Expression>* expr,
170 DefinitionMap* definitions) {
ethannicholas22f939e2016-10-13 13:25:34 -0700171 switch (lvalue->fKind) {
172 case Expression::kVariableReference_Kind: {
173 const Variable& var = ((VariableReference*) lvalue)->fVariable;
174 if (var.fStorage == Variable::kLocal_Storage) {
175 (*definitions)[&var] = expr;
176 }
177 break;
178 }
179 case Expression::kSwizzle_Kind:
180 // We consider the variable written to as long as at least some of its components have
181 // been written to. This will lead to some false negatives (we won't catch it if you
182 // write to foo.x and then read foo.y), but being stricter could lead to false positives
Mike Klein6ad99092016-10-26 10:35:22 -0400183 // (we write to foo.x, and then pass foo to a function which happens to only read foo.x,
184 // but since we pass foo as a whole it is flagged as an error) unless we perform a much
ethannicholas22f939e2016-10-13 13:25:34 -0700185 // more complicated whole-program analysis. This is probably good enough.
Mike Klein6ad99092016-10-26 10:35:22 -0400186 this->addDefinition(((Swizzle*) lvalue)->fBase.get(),
Ethan Nicholas86a43402017-01-19 13:32:00 -0500187 (std::unique_ptr<Expression>*) &fContext.fDefined_Expression,
ethannicholas22f939e2016-10-13 13:25:34 -0700188 definitions);
189 break;
190 case Expression::kIndex_Kind:
191 // see comments in Swizzle
Mike Klein6ad99092016-10-26 10:35:22 -0400192 this->addDefinition(((IndexExpression*) lvalue)->fBase.get(),
Ethan Nicholas86a43402017-01-19 13:32:00 -0500193 (std::unique_ptr<Expression>*) &fContext.fDefined_Expression,
ethannicholas22f939e2016-10-13 13:25:34 -0700194 definitions);
195 break;
196 case Expression::kFieldAccess_Kind:
197 // see comments in Swizzle
Mike Klein6ad99092016-10-26 10:35:22 -0400198 this->addDefinition(((FieldAccess*) lvalue)->fBase.get(),
Ethan Nicholas86a43402017-01-19 13:32:00 -0500199 (std::unique_ptr<Expression>*) &fContext.fDefined_Expression,
ethannicholas22f939e2016-10-13 13:25:34 -0700200 definitions);
201 break;
202 default:
203 // not an lvalue, can't happen
204 ASSERT(false);
205 }
206}
207
208// add local variables defined by this node to the set
Mike Klein6ad99092016-10-26 10:35:22 -0400209void Compiler::addDefinitions(const BasicBlock::Node& node,
Ethan Nicholas86a43402017-01-19 13:32:00 -0500210 DefinitionMap* definitions) {
ethannicholas22f939e2016-10-13 13:25:34 -0700211 switch (node.fKind) {
212 case BasicBlock::Node::kExpression_Kind: {
Ethan Nicholascb670962017-04-20 19:31:52 -0400213 ASSERT(node.expression());
214 const Expression* expr = (Expression*) node.expression()->get();
Ethan Nicholas86a43402017-01-19 13:32:00 -0500215 switch (expr->fKind) {
216 case Expression::kBinary_Kind: {
217 BinaryExpression* b = (BinaryExpression*) expr;
218 if (b->fOperator == Token::EQ) {
219 this->addDefinition(b->fLeft.get(), &b->fRight, definitions);
220 } else if (Token::IsAssignment(b->fOperator)) {
221 this->addDefinition(
222 b->fLeft.get(),
223 (std::unique_ptr<Expression>*) &fContext.fDefined_Expression,
224 definitions);
225
226 }
227 break;
ethannicholas22f939e2016-10-13 13:25:34 -0700228 }
Ethan Nicholas86a43402017-01-19 13:32:00 -0500229 case Expression::kPrefix_Kind: {
230 const PrefixExpression* p = (PrefixExpression*) expr;
231 if (p->fOperator == Token::MINUSMINUS || p->fOperator == Token::PLUSPLUS) {
232 this->addDefinition(
233 p->fOperand.get(),
234 (std::unique_ptr<Expression>*) &fContext.fDefined_Expression,
235 definitions);
236 }
237 break;
238 }
239 case Expression::kPostfix_Kind: {
240 const PostfixExpression* p = (PostfixExpression*) expr;
241 if (p->fOperator == Token::MINUSMINUS || p->fOperator == Token::PLUSPLUS) {
242 this->addDefinition(
243 p->fOperand.get(),
244 (std::unique_ptr<Expression>*) &fContext.fDefined_Expression,
245 definitions);
Ethan Nicholas86a43402017-01-19 13:32:00 -0500246 }
247 break;
248 }
Ethan Nicholascb670962017-04-20 19:31:52 -0400249 case Expression::kVariableReference_Kind: {
250 const VariableReference* v = (VariableReference*) expr;
251 if (v->fRefKind != VariableReference::kRead_RefKind) {
252 this->addDefinition(
253 v,
254 (std::unique_ptr<Expression>*) &fContext.fDefined_Expression,
255 definitions);
256 }
257 }
Ethan Nicholas86a43402017-01-19 13:32:00 -0500258 default:
259 break;
ethannicholas22f939e2016-10-13 13:25:34 -0700260 }
261 break;
262 }
263 case BasicBlock::Node::kStatement_Kind: {
Ethan Nicholascb670962017-04-20 19:31:52 -0400264 const Statement* stmt = (Statement*) node.statement()->get();
ethannicholas22f939e2016-10-13 13:25:34 -0700265 if (stmt->fKind == Statement::kVarDeclarations_Kind) {
Ethan Nicholas86a43402017-01-19 13:32:00 -0500266 VarDeclarationsStatement* vd = (VarDeclarationsStatement*) stmt;
Ethan Nicholascb670962017-04-20 19:31:52 -0400267 for (const auto& decl : vd->fDeclaration->fVars) {
268 if (decl->fValue) {
269 (*definitions)[decl->fVar] = &decl->fValue;
ethannicholas22f939e2016-10-13 13:25:34 -0700270 }
271 }
272 }
273 break;
274 }
275 }
276}
277
278void Compiler::scanCFG(CFG* cfg, BlockId blockId, std::set<BlockId>* workList) {
279 BasicBlock& block = cfg->fBlocks[blockId];
280
281 // compute definitions after this block
Ethan Nicholas86a43402017-01-19 13:32:00 -0500282 DefinitionMap after = block.fBefore;
ethannicholas22f939e2016-10-13 13:25:34 -0700283 for (const BasicBlock::Node& n : block.fNodes) {
284 this->addDefinitions(n, &after);
285 }
286
287 // propagate definitions to exits
288 for (BlockId exitId : block.fExits) {
289 BasicBlock& exit = cfg->fBlocks[exitId];
290 for (const auto& pair : after) {
Ethan Nicholas86a43402017-01-19 13:32:00 -0500291 std::unique_ptr<Expression>* e1 = pair.second;
292 auto found = exit.fBefore.find(pair.first);
293 if (found == exit.fBefore.end()) {
294 // exit has no definition for it, just copy it
295 workList->insert(exitId);
ethannicholas22f939e2016-10-13 13:25:34 -0700296 exit.fBefore[pair.first] = e1;
297 } else {
Ethan Nicholas86a43402017-01-19 13:32:00 -0500298 // exit has a (possibly different) value already defined
299 std::unique_ptr<Expression>* e2 = exit.fBefore[pair.first];
ethannicholas22f939e2016-10-13 13:25:34 -0700300 if (e1 != e2) {
301 // definition has changed, merge and add exit block to worklist
302 workList->insert(exitId);
Ethan Nicholasaf197692017-02-27 13:26:45 -0500303 if (e1 && e2) {
304 exit.fBefore[pair.first] =
Ethan Nicholas86a43402017-01-19 13:32:00 -0500305 (std::unique_ptr<Expression>*) &fContext.fDefined_Expression;
Ethan Nicholasaf197692017-02-27 13:26:45 -0500306 } else {
307 exit.fBefore[pair.first] = nullptr;
308 }
ethannicholas22f939e2016-10-13 13:25:34 -0700309 }
310 }
311 }
312 }
313}
314
315// returns a map which maps all local variables in the function to null, indicating that their value
316// is initially unknown
Ethan Nicholas86a43402017-01-19 13:32:00 -0500317static DefinitionMap compute_start_state(const CFG& cfg) {
318 DefinitionMap result;
Mike Klein6ad99092016-10-26 10:35:22 -0400319 for (const auto& block : cfg.fBlocks) {
320 for (const auto& node : block.fNodes) {
ethannicholas22f939e2016-10-13 13:25:34 -0700321 if (node.fKind == BasicBlock::Node::kStatement_Kind) {
Ethan Nicholascb670962017-04-20 19:31:52 -0400322 ASSERT(node.statement());
323 const Statement* s = node.statement()->get();
ethannicholas22f939e2016-10-13 13:25:34 -0700324 if (s->fKind == Statement::kVarDeclarations_Kind) {
325 const VarDeclarationsStatement* vd = (const VarDeclarationsStatement*) s;
Ethan Nicholascb670962017-04-20 19:31:52 -0400326 for (const auto& decl : vd->fDeclaration->fVars) {
327 result[decl->fVar] = nullptr;
Mike Klein6ad99092016-10-26 10:35:22 -0400328 }
ethannicholas22f939e2016-10-13 13:25:34 -0700329 }
330 }
331 }
332 }
333 return result;
334}
335
Ethan Nicholascb670962017-04-20 19:31:52 -0400336/**
337 * Returns true if assigning to this lvalue has no effect.
338 */
339static bool is_dead(const Expression& lvalue) {
340 switch (lvalue.fKind) {
341 case Expression::kVariableReference_Kind:
342 return ((VariableReference&) lvalue).fVariable.dead();
343 case Expression::kSwizzle_Kind:
344 return is_dead(*((Swizzle&) lvalue).fBase);
345 case Expression::kFieldAccess_Kind:
346 return is_dead(*((FieldAccess&) lvalue).fBase);
347 case Expression::kIndex_Kind: {
348 const IndexExpression& idx = (IndexExpression&) lvalue;
349 return is_dead(*idx.fBase) && !idx.fIndex->hasSideEffects();
350 }
351 default:
352 ABORT("invalid lvalue: %s\n", lvalue.description().c_str());
353 }
354}
ethannicholas22f939e2016-10-13 13:25:34 -0700355
Ethan Nicholascb670962017-04-20 19:31:52 -0400356/**
357 * Returns true if this is an assignment which can be collapsed down to just the right hand side due
358 * to a dead target and lack of side effects on the left hand side.
359 */
360static bool dead_assignment(const BinaryExpression& b) {
361 if (!Token::IsAssignment(b.fOperator)) {
362 return false;
363 }
364 return is_dead(*b.fLeft);
365}
366
367void Compiler::computeDataFlow(CFG* cfg) {
368 cfg->fBlocks[cfg->fStart].fBefore = compute_start_state(*cfg);
ethannicholas22f939e2016-10-13 13:25:34 -0700369 std::set<BlockId> workList;
Ethan Nicholascb670962017-04-20 19:31:52 -0400370 for (BlockId i = 0; i < cfg->fBlocks.size(); i++) {
ethannicholas22f939e2016-10-13 13:25:34 -0700371 workList.insert(i);
372 }
373 while (workList.size()) {
374 BlockId next = *workList.begin();
375 workList.erase(workList.begin());
Ethan Nicholascb670962017-04-20 19:31:52 -0400376 this->scanCFG(cfg, next, &workList);
ethannicholas22f939e2016-10-13 13:25:34 -0700377 }
Ethan Nicholascb670962017-04-20 19:31:52 -0400378}
379
380/**
381 * Attempts to replace the expression pointed to by iter with a new one (in both the CFG and the
382 * IR). If the expression can be cleanly removed, returns true and updates the iterator to point to
383 * the newly-inserted element. Otherwise updates only the IR and returns false (and the CFG will
384 * need to be regenerated).
385 */
386bool try_replace_expression(BasicBlock* b,
387 std::vector<BasicBlock::Node>::iterator* iter,
388 std::unique_ptr<Expression>* newExpression) {
389 std::unique_ptr<Expression>* target = (*iter)->expression();
390 if (!b->tryRemoveExpression(iter)) {
391 *target = std::move(*newExpression);
392 return false;
393 }
394 *target = std::move(*newExpression);
395 return b->tryInsertExpression(iter, target);
396}
397
398/**
399 * Returns true if the expression is a constant numeric literal with the specified value.
400 */
401bool is_constant(Expression& expr, double value) {
402 switch (expr.fKind) {
403 case Expression::kIntLiteral_Kind:
404 return ((IntLiteral&) expr).fValue == value;
405 case Expression::kFloatLiteral_Kind:
406 return ((FloatLiteral&) expr).fValue == value;
407 default:
408 return false;
409 }
410}
411
412/**
413 * Collapses the binary expression pointed to by iter down to just the right side (in both the IR
414 * and CFG structures).
415 */
416void delete_left(BasicBlock* b,
417 std::vector<BasicBlock::Node>::iterator* iter,
418 bool* outUpdated,
419 bool* outNeedsRescan) {
420 *outUpdated = true;
421 if (!try_replace_expression(b, iter, &((BinaryExpression&) **(*iter)->expression()).fRight)) {
422 *outNeedsRescan = true;
423 }
424}
425
426/**
427 * Collapses the binary expression pointed to by iter down to just the left side (in both the IR and
428 * CFG structures).
429 */
430void delete_right(BasicBlock* b,
431 std::vector<BasicBlock::Node>::iterator* iter,
432 bool* outUpdated,
433 bool* outNeedsRescan) {
434 *outUpdated = true;
435 if (!try_replace_expression(b, iter, &((BinaryExpression&) **(*iter)->expression()).fLeft)) {
436 *outNeedsRescan = true;
437 }
438}
439
440void Compiler::simplifyExpression(DefinitionMap& definitions,
441 BasicBlock& b,
442 std::vector<BasicBlock::Node>::iterator* iter,
443 std::unordered_set<const Variable*>* undefinedVariables,
444 bool* outUpdated,
445 bool* outNeedsRescan) {
446 Expression* expr = (*iter)->expression()->get();
447 ASSERT(expr);
448 if ((*iter)->fConstantPropagation) {
449 std::unique_ptr<Expression> optimized = expr->constantPropagate(*fIRGenerator, definitions);
450 if (optimized) {
451 if (!try_replace_expression(&b, iter, &optimized)) {
452 *outNeedsRescan = true;
453 }
454 ASSERT((*iter)->fKind == BasicBlock::Node::kExpression_Kind);
455 expr = (*iter)->expression()->get();
456 *outUpdated = true;
457 }
458 }
459 switch (expr->fKind) {
460 case Expression::kVariableReference_Kind: {
461 const Variable& var = ((VariableReference*) expr)->fVariable;
462 if (var.fStorage == Variable::kLocal_Storage && !definitions[&var] &&
463 (*undefinedVariables).find(&var) == (*undefinedVariables).end()) {
464 (*undefinedVariables).insert(&var);
465 this->error(expr->fPosition,
466 "'" + var.fName + "' has not been assigned");
467 }
468 break;
469 }
470 case Expression::kTernary_Kind: {
471 TernaryExpression* t = (TernaryExpression*) expr;
472 if (t->fTest->fKind == Expression::kBoolLiteral_Kind) {
473 // ternary has a constant test, replace it with either the true or
474 // false branch
475 if (((BoolLiteral&) *t->fTest).fValue) {
476 (*iter)->setExpression(std::move(t->fIfTrue));
477 } else {
478 (*iter)->setExpression(std::move(t->fIfFalse));
479 }
480 *outUpdated = true;
481 *outNeedsRescan = true;
482 }
483 break;
484 }
485 case Expression::kBinary_Kind: {
486 // collapse useless expressions like x * 1 or x + 0
487 BinaryExpression* bin = (BinaryExpression*) expr;
488 switch (bin->fOperator) {
489 case Token::STAR:
490 if (is_constant(*bin->fLeft, 1)) {
491 delete_left(&b, iter, outUpdated, outNeedsRescan);
492 }
493 else if (is_constant(*bin->fRight, 1)) {
494 delete_right(&b, iter, outUpdated, outNeedsRescan);
495 }
496 break;
Ethan Nicholas56e42712017-04-21 10:23:37 -0400497 case Token::PLUS:
Ethan Nicholascb670962017-04-20 19:31:52 -0400498 if (is_constant(*bin->fLeft, 0)) {
499 delete_left(&b, iter, outUpdated, outNeedsRescan);
500 }
Ethan Nicholas56e42712017-04-21 10:23:37 -0400501 if (is_constant(*bin->fRight, 0)) {
502 delete_right(&b, iter, outUpdated, outNeedsRescan);
503 }
504 break;
505 case Token::MINUS:
506 if (is_constant(*bin->fRight, 0)) {
Ethan Nicholascb670962017-04-20 19:31:52 -0400507 delete_right(&b, iter, outUpdated, outNeedsRescan);
508 }
509 break;
510 case Token::SLASH:
511 if (is_constant(*bin->fRight, 1)) {
512 delete_right(&b, iter, outUpdated, outNeedsRescan);
513 }
514 break;
515 default:
516 break;
517 }
518 }
519 default:
520 break;
521 }
522}
523
524void Compiler::simplifyStatement(DefinitionMap& definitions,
525 BasicBlock& b,
526 std::vector<BasicBlock::Node>::iterator* iter,
527 std::unordered_set<const Variable*>* undefinedVariables,
528 bool* outUpdated,
529 bool* outNeedsRescan) {
530 Statement* stmt = (*iter)->statement()->get();
531 switch (stmt->fKind) {
532 case Statement::kVarDeclarations_Kind: {
533 VarDeclarations& vd = *((VarDeclarationsStatement&) *stmt).fDeclaration;
534 for (auto varIter = vd.fVars.begin(); varIter != vd.fVars.end(); ) {
535 const auto& varDecl = **varIter;
536 if (varDecl.fVar->dead() &&
537 (!varDecl.fValue ||
538 !varDecl.fValue->hasSideEffects())) {
539 if (varDecl.fValue) {
540 ASSERT((*iter)->statement()->get() == stmt);
541 if (!b.tryRemoveExpressionBefore(iter, varDecl.fValue.get())) {
542 *outNeedsRescan = true;
543 }
544 }
545 varIter = vd.fVars.erase(varIter);
546 *outUpdated = true;
547 } else {
548 ++varIter;
549 }
550 }
551 if (vd.fVars.size() == 0) {
552 (*iter)->setStatement(std::unique_ptr<Statement>(new Nop()));
553 }
554 break;
555 }
556 case Statement::kIf_Kind: {
557 IfStatement& i = (IfStatement&) *stmt;
558 if (i.fIfFalse && i.fIfFalse->isEmpty()) {
559 // else block doesn't do anything, remove it
560 i.fIfFalse.reset();
561 *outUpdated = true;
562 *outNeedsRescan = true;
563 }
564 if (!i.fIfFalse && i.fIfTrue->isEmpty()) {
565 // if block doesn't do anything, no else block
566 if (i.fTest->hasSideEffects()) {
567 // test has side effects, keep it
568 (*iter)->setStatement(std::unique_ptr<Statement>(
569 new ExpressionStatement(std::move(i.fTest))));
570 } else {
571 // no if, no else, no test side effects, kill the whole if
572 // statement
573 (*iter)->setStatement(std::unique_ptr<Statement>(new Nop()));
574 }
575 *outUpdated = true;
576 *outNeedsRescan = true;
577 }
578 break;
579 }
580 case Statement::kExpression_Kind: {
581 ExpressionStatement& e = (ExpressionStatement&) *stmt;
582 ASSERT((*iter)->statement()->get() == &e);
583 if (e.fExpression->fKind == Expression::kBinary_Kind) {
584 BinaryExpression& bin = (BinaryExpression&) *e.fExpression;
585 if (dead_assignment(bin)) {
586 if (!b.tryRemoveExpressionBefore(iter, &bin)) {
587 *outNeedsRescan = true;
588 }
589 if (bin.fRight->hasSideEffects()) {
590 // still have to evaluate the right due to side effects,
591 // replace the binary expression with just the right side
592 e.fExpression = std::move(bin.fRight);
593 if (!b.tryInsertExpression(iter, &e.fExpression)) {
594 *outNeedsRescan = true;
595 }
596 } else {
597 // no side effects, kill the whole statement
598 ASSERT((*iter)->statement()->get() == stmt);
599 (*iter)->setStatement(std::unique_ptr<Statement>(new Nop()));
600 }
601 *outUpdated = true;
602 break;
603 }
604 }
605 if (!e.fExpression->hasSideEffects()) {
606 // Expression statement with no side effects, kill it
607 if (!b.tryRemoveExpressionBefore(iter, e.fExpression.get())) {
608 *outNeedsRescan = true;
609 }
610 ASSERT((*iter)->statement()->get() == stmt);
611 (*iter)->setStatement(std::unique_ptr<Statement>(new Nop()));
612 *outUpdated = true;
613 }
614 break;
615 }
616 default:
617 break;
618 }
619}
620
621void Compiler::scanCFG(FunctionDefinition& f) {
622 CFG cfg = CFGGenerator().getCFG(f);
623 this->computeDataFlow(&cfg);
ethannicholas22f939e2016-10-13 13:25:34 -0700624
625 // check for unreachable code
626 for (size_t i = 0; i < cfg.fBlocks.size(); i++) {
Mike Klein6ad99092016-10-26 10:35:22 -0400627 if (i != cfg.fStart && !cfg.fBlocks[i].fEntrances.size() &&
ethannicholas22f939e2016-10-13 13:25:34 -0700628 cfg.fBlocks[i].fNodes.size()) {
Ethan Nicholas86a43402017-01-19 13:32:00 -0500629 Position p;
630 switch (cfg.fBlocks[i].fNodes[0].fKind) {
631 case BasicBlock::Node::kStatement_Kind:
Ethan Nicholascb670962017-04-20 19:31:52 -0400632 p = (*cfg.fBlocks[i].fNodes[0].statement())->fPosition;
Ethan Nicholas86a43402017-01-19 13:32:00 -0500633 break;
634 case BasicBlock::Node::kExpression_Kind:
Ethan Nicholascb670962017-04-20 19:31:52 -0400635 p = (*cfg.fBlocks[i].fNodes[0].expression())->fPosition;
Ethan Nicholas86a43402017-01-19 13:32:00 -0500636 break;
637 }
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400638 this->error(p, String("unreachable"));
ethannicholas22f939e2016-10-13 13:25:34 -0700639 }
640 }
641 if (fErrorCount) {
642 return;
643 }
644
Ethan Nicholascb670962017-04-20 19:31:52 -0400645 // check for dead code & undefined variables, perform constant propagation
646 std::unordered_set<const Variable*> undefinedVariables;
647 bool updated;
648 bool needsRescan = false;
649 do {
650 if (needsRescan) {
651 cfg = CFGGenerator().getCFG(f);
652 this->computeDataFlow(&cfg);
653 needsRescan = false;
Ethan Nicholas113628d2017-02-02 16:11:39 -0500654 }
Ethan Nicholascb670962017-04-20 19:31:52 -0400655
656 updated = false;
657 for (BasicBlock& b : cfg.fBlocks) {
658 DefinitionMap definitions = b.fBefore;
659
660 for (auto iter = b.fNodes.begin(); iter != b.fNodes.end() && !needsRescan; ++iter) {
661 if (iter->fKind == BasicBlock::Node::kExpression_Kind) {
662 this->simplifyExpression(definitions, b, &iter, &undefinedVariables, &updated,
663 &needsRescan);
664 } else {
665 this->simplifyStatement(definitions, b, &iter, &undefinedVariables, &updated,
666 &needsRescan);
667 }
668 this->addDefinitions(*iter, &definitions);
669 }
670 }
671 } while (updated);
672 ASSERT(!needsRescan);
ethannicholas22f939e2016-10-13 13:25:34 -0700673
674 // check for missing return
675 if (f.fDeclaration.fReturnType != *fContext.fVoid_Type) {
676 if (cfg.fBlocks[cfg.fExit].fEntrances.size()) {
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400677 this->error(f.fPosition, String("function can exit without returning a value"));
ethannicholas22f939e2016-10-13 13:25:34 -0700678 }
679 }
680}
681
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400682void Compiler::internalConvertProgram(String text,
ethannicholas5961bc92016-10-12 06:39:56 -0700683 Modifiers::Flag* defaultPrecision,
ethannicholasb3058bd2016-07-01 08:22:01 -0700684 std::vector<std::unique_ptr<ProgramElement>>* result) {
685 Parser parser(text, *fTypes, *this);
686 std::vector<std::unique_ptr<ASTDeclaration>> parsed = parser.file();
687 if (fErrorCount) {
688 return;
689 }
ethannicholas5961bc92016-10-12 06:39:56 -0700690 *defaultPrecision = Modifiers::kHighp_Flag;
ethannicholasb3058bd2016-07-01 08:22:01 -0700691 for (size_t i = 0; i < parsed.size(); i++) {
692 ASTDeclaration& decl = *parsed[i];
693 switch (decl.fKind) {
694 case ASTDeclaration::kVar_Kind: {
ethannicholas14fe8cc2016-09-07 13:37:16 -0700695 std::unique_ptr<VarDeclarations> s = fIRGenerator->convertVarDeclarations(
Mike Klein6ad99092016-10-26 10:35:22 -0400696 (ASTVarDeclarations&) decl,
ethannicholasb3058bd2016-07-01 08:22:01 -0700697 Variable::kGlobal_Storage);
698 if (s) {
699 result->push_back(std::move(s));
700 }
701 break;
702 }
703 case ASTDeclaration::kFunction_Kind: {
704 std::unique_ptr<FunctionDefinition> f = fIRGenerator->convertFunction(
705 (ASTFunction&) decl);
ethannicholas22f939e2016-10-13 13:25:34 -0700706 if (!fErrorCount && f) {
707 this->scanCFG(*f);
ethannicholasb3058bd2016-07-01 08:22:01 -0700708 result->push_back(std::move(f));
709 }
710 break;
711 }
ethannicholas5961bc92016-10-12 06:39:56 -0700712 case ASTDeclaration::kModifiers_Kind: {
713 std::unique_ptr<ModifiersDeclaration> f = fIRGenerator->convertModifiersDeclaration(
714 (ASTModifiersDeclaration&) decl);
715 if (f) {
716 result->push_back(std::move(f));
717 }
718 break;
719 }
ethannicholasb3058bd2016-07-01 08:22:01 -0700720 case ASTDeclaration::kInterfaceBlock_Kind: {
721 std::unique_ptr<InterfaceBlock> i = fIRGenerator->convertInterfaceBlock(
722 (ASTInterfaceBlock&) decl);
723 if (i) {
724 result->push_back(std::move(i));
725 }
726 break;
727 }
728 case ASTDeclaration::kExtension_Kind: {
729 std::unique_ptr<Extension> e = fIRGenerator->convertExtension((ASTExtension&) decl);
730 if (e) {
731 result->push_back(std::move(e));
732 }
733 break;
734 }
ethannicholas5961bc92016-10-12 06:39:56 -0700735 case ASTDeclaration::kPrecision_Kind: {
736 *defaultPrecision = ((ASTPrecision&) decl).fPrecision;
737 break;
738 }
ethannicholasb3058bd2016-07-01 08:22:01 -0700739 default:
740 ABORT("unsupported declaration: %s\n", decl.description().c_str());
741 }
742 }
743}
744
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400745std::unique_ptr<Program> Compiler::convertProgram(Program::Kind kind, String text,
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500746 const Program::Settings& settings) {
ethannicholasb3058bd2016-07-01 08:22:01 -0700747 fErrorText = "";
748 fErrorCount = 0;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500749 fIRGenerator->start(&settings);
ethannicholasd598f792016-07-25 10:08:54 -0700750 std::vector<std::unique_ptr<ProgramElement>> elements;
ethannicholas5961bc92016-10-12 06:39:56 -0700751 Modifiers::Flag ignored;
ethannicholasb3058bd2016-07-01 08:22:01 -0700752 switch (kind) {
753 case Program::kVertex_Kind:
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400754 this->internalConvertProgram(String(SKSL_VERT_INCLUDE), &ignored, &elements);
ethannicholasb3058bd2016-07-01 08:22:01 -0700755 break;
756 case Program::kFragment_Kind:
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400757 this->internalConvertProgram(String(SKSL_FRAG_INCLUDE), &ignored, &elements);
ethannicholasb3058bd2016-07-01 08:22:01 -0700758 break;
Ethan Nicholas52cad152017-02-16 16:37:32 -0500759 case Program::kGeometry_Kind:
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400760 this->internalConvertProgram(String(SKSL_GEOM_INCLUDE), &ignored, &elements);
Ethan Nicholas52cad152017-02-16 16:37:32 -0500761 break;
ethannicholasb3058bd2016-07-01 08:22:01 -0700762 }
ethannicholasddb37d62016-10-20 09:54:00 -0700763 fIRGenerator->fSymbolTable->markAllFunctionsBuiltin();
ethannicholas5961bc92016-10-12 06:39:56 -0700764 Modifiers::Flag defaultPrecision;
765 this->internalConvertProgram(text, &defaultPrecision, &elements);
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500766 auto result = std::unique_ptr<Program>(new Program(kind, settings, defaultPrecision, &fContext,
767 std::move(elements),
768 fIRGenerator->fSymbolTable,
769 fIRGenerator->fInputs));
Ethan Nicholas3605ace2016-11-21 15:59:48 -0500770 fIRGenerator->finish();
ethannicholasb3058bd2016-07-01 08:22:01 -0700771 this->writeErrorCount();
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500772 if (fErrorCount) {
773 return nullptr;
774 }
775 return result;
776}
777
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400778bool Compiler::toSPIRV(const Program& program, OutputStream& out) {
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -0400779#ifdef SK_ENABLE_SPIRV_VALIDATION
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400780 StringStream buffer;
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -0400781 SPIRVCodeGenerator cg(&fContext, &program, this, &buffer);
782 bool result = cg.generateCode();
783 if (result) {
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -0400784 spvtools::SpirvTools tools(SPV_ENV_VULKAN_1_0);
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400785 ASSERT(0 == buffer.size() % 4);
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -0400786 auto dumpmsg = [](spv_message_level_t, const char*, const spv_position_t&, const char* m) {
787 SkDebugf("SPIR-V validation error: %s\n", m);
788 };
789 tools.SetMessageConsumer(dumpmsg);
790 // Verify that the SPIR-V we produced is valid. If this assert fails, check the logs prior
791 // to the failure to see the validation errors.
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400792 ASSERT_RESULT(tools.Validate((const uint32_t*) buffer.data(), buffer.size() / 4));
793 out.write(buffer.data(), buffer.size());
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -0400794 }
795#else
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500796 SPIRVCodeGenerator cg(&fContext, &program, this, &out);
797 bool result = cg.generateCode();
Ethan Nicholasa6ae1f72017-03-16 09:56:54 -0400798#endif
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500799 this->writeErrorCount();
Ethan Nicholasce33f102016-12-09 17:22:59 -0500800 return result;
801}
802
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400803bool Compiler::toSPIRV(const Program& program, String* out) {
804 StringStream buffer;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500805 bool result = this->toSPIRV(program, buffer);
806 if (result) {
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400807 *out = String(buffer.data(), buffer.size());
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500808 }
809 return result;
810}
811
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400812bool Compiler::toGLSL(const Program& program, OutputStream& out) {
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500813 GLSLCodeGenerator cg(&fContext, &program, this, &out);
814 bool result = cg.generateCode();
815 this->writeErrorCount();
816 return result;
817}
818
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400819bool Compiler::toGLSL(const Program& program, String* out) {
820 StringStream buffer;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500821 bool result = this->toGLSL(program, buffer);
822 if (result) {
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400823 *out = String(buffer.data(), buffer.size());
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500824 }
825 return result;
826}
827
828
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400829void Compiler::error(Position position, String msg) {
ethannicholasb3058bd2016-07-01 08:22:01 -0700830 fErrorCount++;
831 fErrorText += "error: " + position.description() + ": " + msg.c_str() + "\n";
832}
833
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400834String Compiler::errorText() {
835 String result = fErrorText;
ethannicholasb3058bd2016-07-01 08:22:01 -0700836 return result;
837}
838
839void Compiler::writeErrorCount() {
840 if (fErrorCount) {
841 fErrorText += to_string(fErrorCount) + " error";
842 if (fErrorCount > 1) {
843 fErrorText += "s";
844 }
845 fErrorText += "\n";
846 }
847}
848
ethannicholasb3058bd2016-07-01 08:22:01 -0700849} // namespace