blob: 89e4a7a5a16e1a9dce8cf88ac1982dd5ff2f2844 [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 */
Ethan Nicholas11d53972016-11-28 11:23:23 -05007
ethannicholasb3058bd2016-07-01 08:22:01 -07008#ifndef SKSL_IRGENERATOR
9#define SKSL_IRGENERATOR
10
John Stilesddefaee2020-08-11 15:13:26 -040011#include <unordered_map>
John Stilesb8e010c2020-08-11 18:05:39 -040012#include <unordered_set>
Ethan Nicholasdb80f692019-11-22 14:06:12 -050013
Ethan Nicholasfc994162019-06-06 10:04:27 -040014#include "src/sksl/SkSLASTFile.h"
15#include "src/sksl/SkSLASTNode.h"
Mike Klein4b432fa2019-06-06 11:44:05 -050016#include "src/sksl/SkSLErrorReporter.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050017#include "src/sksl/ir/SkSLBlock.h"
18#include "src/sksl/ir/SkSLExpression.h"
19#include "src/sksl/ir/SkSLExtension.h"
20#include "src/sksl/ir/SkSLFunctionDefinition.h"
21#include "src/sksl/ir/SkSLInterfaceBlock.h"
22#include "src/sksl/ir/SkSLModifiers.h"
23#include "src/sksl/ir/SkSLModifiersDeclaration.h"
24#include "src/sksl/ir/SkSLProgram.h"
25#include "src/sksl/ir/SkSLSection.h"
26#include "src/sksl/ir/SkSLStatement.h"
27#include "src/sksl/ir/SkSLSymbolTable.h"
28#include "src/sksl/ir/SkSLType.h"
29#include "src/sksl/ir/SkSLTypeReference.h"
30#include "src/sksl/ir/SkSLVarDeclarations.h"
31#include "src/sksl/ir/SkSLVariableReference.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070032
33namespace SkSL {
34
Ethan Nicholascb0f4092019-04-19 11:26:50 -040035struct Swizzle;
John Stilesea9ab822020-08-31 09:55:04 -040036struct FunctionCall;
Ethan Nicholascb0f4092019-04-19 11:26:50 -040037
ethannicholasb3058bd2016-07-01 08:22:01 -070038/**
John Stiles810c8cf2020-08-26 19:46:27 -040039 * Intrinsics are passed between the Compiler and the IRGenerator using IRIntrinsicMaps.
40 */
41struct IRIntrinsic {
42 std::unique_ptr<ProgramElement> fIntrinsic;
43 bool fAlreadyIncluded = false;
44};
45using IRIntrinsicMap = std::unordered_map<String, IRIntrinsic>;
46
47/**
Ethan Nicholas11d53972016-11-28 11:23:23 -050048 * Performs semantic analysis on an abstract syntax tree (AST) and produces the corresponding
ethannicholasb3058bd2016-07-01 08:22:01 -070049 * (unoptimized) intermediate representation (IR).
50 */
51class IRGenerator {
52public:
Ethan Nicholas11d53972016-11-28 11:23:23 -050053 IRGenerator(const Context* context, std::shared_ptr<SymbolTable> root,
ethannicholasd598f792016-07-25 10:08:54 -070054 ErrorReporter& errorReporter);
ethannicholasb3058bd2016-07-01 08:22:01 -070055
Robert Phillipsfe8da172018-01-24 14:52:02 +000056 void convertProgram(Program::Kind kind,
57 const char* text,
Ethan Nicholas5b5f0962017-09-11 13:50:14 -070058 size_t length,
Ethan Nicholas7da6dfa2017-06-21 11:25:18 -040059 std::vector<std::unique_ptr<ProgramElement>>* result);
ethannicholasb3058bd2016-07-01 08:22:01 -070060
Ethan Nicholas86a43402017-01-19 13:32:00 -050061 /**
62 * If both operands are compile-time constants and can be folded, returns an expression
63 * representing the folded value. Otherwise, returns null. Note that unlike most other functions
64 * here, null does not represent a compilation error.
65 */
66 std::unique_ptr<Expression> constantFold(const Expression& left,
67 Token::Kind op,
68 const Expression& right) const;
Ethan Nicholas00543112018-07-31 09:44:36 -040069
Ethan Nicholas941e7e22016-12-12 15:33:30 -050070 Program::Inputs fInputs;
Ethan Nicholas762466e2017-06-29 10:03:38 -040071 const Program::Settings* fSettings;
Ethan Nicholas86a43402017-01-19 13:32:00 -050072 const Context& fContext;
Ethan Nicholas00543112018-07-31 09:44:36 -040073 Program::Kind fKind;
Ethan Nicholas941e7e22016-12-12 15:33:30 -050074
ethannicholasb3058bd2016-07-01 08:22:01 -070075private:
Ethan Nicholas3605ace2016-11-21 15:59:48 -050076 /**
Ethan Nicholas941e7e22016-12-12 15:33:30 -050077 * Prepare to compile a program. Resets state, pushes a new symbol table, and installs the
78 * settings.
Ethan Nicholas3605ace2016-11-21 15:59:48 -050079 */
Ethan Nicholas3c6ae622018-04-24 13:06:09 -040080 void start(const Program::Settings* settings,
Ethan Nicholasc18bb512020-07-28 14:46:53 -040081 std::vector<std::unique_ptr<ProgramElement>>* inherited,
82 bool isBuiltinCode = false);
Ethan Nicholas3605ace2016-11-21 15:59:48 -050083
84 /**
85 * Performs cleanup after compilation is complete.
86 */
87 void finish();
88
ethannicholasb3058bd2016-07-01 08:22:01 -070089 void pushSymbolTable();
90 void popSymbolTable();
91
Ethan Nicholas63d7ee32020-08-17 10:57:12 -040092 void checkModifiers(int offset, const Modifiers& modifiers, int permitted);
Ethan Nicholasfc994162019-06-06 10:04:27 -040093 std::unique_ptr<VarDeclarations> convertVarDeclarations(const ASTNode& decl,
Ethan Nicholas7da6dfa2017-06-21 11:25:18 -040094 Variable::Storage storage);
Ethan Nicholasfc994162019-06-06 10:04:27 -040095 void convertFunction(const ASTNode& f);
Ethan Nicholas70728ef2020-05-28 07:09:00 -040096 std::unique_ptr<Statement> convertSingleStatement(const ASTNode& statement);
Ethan Nicholasfc994162019-06-06 10:04:27 -040097 std::unique_ptr<Statement> convertStatement(const ASTNode& statement);
98 std::unique_ptr<Expression> convertExpression(const ASTNode& expression);
99 std::unique_ptr<ModifiersDeclaration> convertModifiersDeclaration(const ASTNode& m);
Ethan Nicholas7da6dfa2017-06-21 11:25:18 -0400100
Ethan Nicholasfc994162019-06-06 10:04:27 -0400101 const Type* convertType(const ASTNode& type);
John Stilesddefaee2020-08-11 15:13:26 -0400102 std::unique_ptr<Expression> inlineExpression(
103 int offset,
104 std::unordered_map<const Variable*, const Variable*>* varMap,
105 const Expression& expression);
106 std::unique_ptr<Statement> inlineStatement(
107 int offset,
108 std::unordered_map<const Variable*, const Variable*>* varMap,
John Stilesea9ab822020-08-31 09:55:04 -0400109 SymbolTable* symbolTableForStatement,
John Stilesddefaee2020-08-11 15:13:26 -0400110 const Variable* returnVar,
111 bool haveEarlyReturns,
112 const Statement& statement);
John Stilesea9ab822020-08-31 09:55:04 -0400113 std::unique_ptr<Expression> inlineCall(std::unique_ptr<FunctionCall> call,
114 SymbolTable* symbolTableForCall);
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700115 std::unique_ptr<Expression> call(int offset,
Ethan Nicholas11d53972016-11-28 11:23:23 -0500116 const FunctionDeclaration& function,
ethannicholasb3058bd2016-07-01 08:22:01 -0700117 std::vector<std::unique_ptr<Expression>> arguments);
John Stilesea9ab822020-08-31 09:55:04 -0400118 bool isSafeToInline(const FunctionCall& function, int inlineThreshold);
Ethan Nicholasdcba08e2017-08-02 10:52:54 -0400119 int callCost(const FunctionDeclaration& function,
120 const std::vector<std::unique_ptr<Expression>>& arguments);
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700121 std::unique_ptr<Expression> call(int offset, std::unique_ptr<Expression> function,
ethannicholasb3058bd2016-07-01 08:22:01 -0700122 std::vector<std::unique_ptr<Expression>> arguments);
Ethan Nicholasdcba08e2017-08-02 10:52:54 -0400123 int coercionCost(const Expression& expr, const Type& type);
ethannicholasd598f792016-07-25 10:08:54 -0700124 std::unique_ptr<Expression> coerce(std::unique_ptr<Expression> expr, const Type& type);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400125 std::unique_ptr<Block> convertBlock(const ASTNode& block);
126 std::unique_ptr<Statement> convertBreak(const ASTNode& b);
Ethan Nicholas84645e32017-02-09 13:57:14 -0500127 std::unique_ptr<Expression> convertNumberConstructor(
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700128 int offset,
Ethan Nicholas84645e32017-02-09 13:57:14 -0500129 const Type& type,
130 std::vector<std::unique_ptr<Expression>> params);
131 std::unique_ptr<Expression> convertCompoundConstructor(
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700132 int offset,
Ethan Nicholas84645e32017-02-09 13:57:14 -0500133 const Type& type,
134 std::vector<std::unique_ptr<Expression>> params);
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700135 std::unique_ptr<Expression> convertConstructor(int offset,
Ethan Nicholas11d53972016-11-28 11:23:23 -0500136 const Type& type,
ethannicholasb3058bd2016-07-01 08:22:01 -0700137 std::vector<std::unique_ptr<Expression>> params);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400138 std::unique_ptr<Statement> convertContinue(const ASTNode& c);
139 std::unique_ptr<Statement> convertDiscard(const ASTNode& d);
140 std::unique_ptr<Statement> convertDo(const ASTNode& d);
141 std::unique_ptr<Statement> convertSwitch(const ASTNode& s);
142 std::unique_ptr<Expression> convertBinaryExpression(const ASTNode& expression);
143 std::unique_ptr<Extension> convertExtension(int offset, StringFragment name);
144 std::unique_ptr<Statement> convertExpressionStatement(const ASTNode& s);
145 std::unique_ptr<Statement> convertFor(const ASTNode& f);
146 std::unique_ptr<Expression> convertIdentifier(const ASTNode& identifier);
147 std::unique_ptr<Statement> convertIf(const ASTNode& s);
ethannicholasb3058bd2016-07-01 08:22:01 -0700148 std::unique_ptr<Expression> convertIndex(std::unique_ptr<Expression> base,
Ethan Nicholasfc994162019-06-06 10:04:27 -0400149 const ASTNode& index);
150 std::unique_ptr<InterfaceBlock> convertInterfaceBlock(const ASTNode& s);
Ethan Nicholas11d53972016-11-28 11:23:23 -0500151 Modifiers convertModifiers(const Modifiers& m);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400152 std::unique_ptr<Expression> convertPrefixExpression(const ASTNode& expression);
153 std::unique_ptr<Statement> convertReturn(const ASTNode& r);
154 std::unique_ptr<Section> convertSection(const ASTNode& e);
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700155 std::unique_ptr<Expression> getCap(int offset, String name);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400156 std::unique_ptr<Expression> convertCallExpression(const ASTNode& expression);
157 std::unique_ptr<Expression> convertFieldExpression(const ASTNode& expression);
158 std::unique_ptr<Expression> convertIndexExpression(const ASTNode& expression);
159 std::unique_ptr<Expression> convertPostfixExpression(const ASTNode& expression);
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400160 std::unique_ptr<Expression> findEnumRef(int offset,
161 const Type& type,
162 StringFragment field,
163 std::vector<std::unique_ptr<ProgramElement>>& elements);
Ethan Nicholasaae47c82017-11-10 15:34:03 -0500164 std::unique_ptr<Expression> convertTypeField(int offset, const Type& type,
165 StringFragment field);
Ethan Nicholas11d53972016-11-28 11:23:23 -0500166 std::unique_ptr<Expression> convertField(std::unique_ptr<Expression> base,
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700167 StringFragment field);
ethannicholasb3058bd2016-07-01 08:22:01 -0700168 std::unique_ptr<Expression> convertSwizzle(std::unique_ptr<Expression> base,
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700169 StringFragment fields);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400170 std::unique_ptr<Expression> convertTernaryExpression(const ASTNode& expression);
171 std::unique_ptr<Statement> convertVarDeclarationStatement(const ASTNode& s);
172 std::unique_ptr<Statement> convertWhile(const ASTNode& w);
173 void convertEnum(const ASTNode& e);
Ethan Nicholasaae47c82017-11-10 15:34:03 -0500174 std::unique_ptr<Block> applyInvocationIDWorkaround(std::unique_ptr<Block> main);
Robert Phillipsfe8da172018-01-24 14:52:02 +0000175 // returns a statement which converts sk_Position from device to normalized coordinates
176 std::unique_ptr<Statement> getNormalizeSkPositionCode();
ethannicholasb3058bd2016-07-01 08:22:01 -0700177
178 void checkValid(const Expression& expr);
John Stiles403a3632020-08-20 12:11:48 -0400179 bool setRefKind(Expression& expr, VariableReference::RefKind kind);
Brian Osman3e3db6c2020-08-14 09:42:12 -0400180 bool getConstantInt(const Expression& value, int64_t* out);
Ethan Nicholascb0f4092019-04-19 11:26:50 -0400181 bool checkSwizzleWrite(const Swizzle& swizzle);
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400182 void copyIntrinsicIfNeeded(const FunctionDeclaration& function);
ethannicholasb3058bd2016-07-01 08:22:01 -0700183
Ethan Nicholasfc994162019-06-06 10:04:27 -0400184 std::unique_ptr<ASTFile> fFile;
ethannicholasd598f792016-07-25 10:08:54 -0700185 const FunctionDeclaration* fCurrentFunction;
Ethan Nicholas762466e2017-06-29 10:03:38 -0400186 std::unordered_map<String, Program::Settings::Value> fCapsMap;
187 std::shared_ptr<SymbolTable> fRootSymbolTable;
ethannicholasb3058bd2016-07-01 08:22:01 -0700188 std::shared_ptr<SymbolTable> fSymbolTable;
Ethan Nicholas70728ef2020-05-28 07:09:00 -0400189 // additional statements that need to be inserted before the one that convertStatement is
190 // currently working on
191 std::vector<std::unique_ptr<Statement>> fExtraStatements;
John Stiles810c8cf2020-08-26 19:46:27 -0400192 // Symbols which have definitions in the include files.
193 IRIntrinsicMap* fIntrinsics = nullptr;
John Stilesb8e010c2020-08-11 18:05:39 -0400194 std::unordered_set<const FunctionDeclaration*> fReferencedIntrinsics;
ethannicholas22f939e2016-10-13 13:25:34 -0700195 int fLoopLevel;
Ethan Nicholasaf197692017-02-27 13:26:45 -0500196 int fSwitchLevel;
ethannicholasb3058bd2016-07-01 08:22:01 -0700197 ErrorReporter& fErrors;
Ethan Nicholas7da6dfa2017-06-21 11:25:18 -0400198 int fInvocations;
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400199 std::vector<std::unique_ptr<ProgramElement>>* fInherited;
Ethan Nicholasaae47c82017-11-10 15:34:03 -0500200 std::vector<std::unique_ptr<ProgramElement>>* fProgramElements;
Ethan Nicholas3c6ae622018-04-24 13:06:09 -0400201 const Variable* fSkPerVertex = nullptr;
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400202 const Variable* fRTAdjust;
203 const Variable* fRTAdjustInterfaceBlock;
Robert Phillipsfe8da172018-01-24 14:52:02 +0000204 int fRTAdjustFieldIndex;
Michael Ludwig9861b7c2020-06-23 18:37:17 -0400205 int fInlineVarCounter;
Ethan Nicholas70728ef2020-05-28 07:09:00 -0400206 bool fCanInline = true;
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400207 // true if we are currently processing one of the built-in SkSL include files
208 bool fIsBuiltinCode;
ethannicholasb3058bd2016-07-01 08:22:01 -0700209
210 friend class AutoSymbolTable;
ethannicholas22f939e2016-10-13 13:25:34 -0700211 friend class AutoLoopLevel;
Ethan Nicholasaf197692017-02-27 13:26:45 -0500212 friend class AutoSwitchLevel;
John Stilesd1c4dac2020-08-11 18:50:50 -0400213 friend class AutoDisableInline;
ethannicholasb3058bd2016-07-01 08:22:01 -0700214 friend class Compiler;
215};
216
John Stilesa6841be2020-08-06 14:11:56 -0400217} // namespace SkSL
ethannicholasb3058bd2016-07-01 08:22:01 -0700218
219#endif