blob: 7a39ab19bc9d70e7bbd64970ea5c1d299f9f7fa0 [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
11#include "SkSLErrorReporter.h"
12#include "ast/SkSLASTBinaryExpression.h"
13#include "ast/SkSLASTBlock.h"
14#include "ast/SkSLASTBreakStatement.h"
15#include "ast/SkSLASTCallSuffix.h"
16#include "ast/SkSLASTContinueStatement.h"
17#include "ast/SkSLASTDiscardStatement.h"
18#include "ast/SkSLASTDoStatement.h"
Ethan Nicholasaae47c82017-11-10 15:34:03 -050019#include "ast/SkSLASTEnum.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070020#include "ast/SkSLASTExpression.h"
21#include "ast/SkSLASTExpressionStatement.h"
22#include "ast/SkSLASTExtension.h"
23#include "ast/SkSLASTForStatement.h"
24#include "ast/SkSLASTFunction.h"
25#include "ast/SkSLASTIdentifier.h"
26#include "ast/SkSLASTIfStatement.h"
27#include "ast/SkSLASTInterfaceBlock.h"
ethannicholas5961bc92016-10-12 06:39:56 -070028#include "ast/SkSLASTModifiersDeclaration.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070029#include "ast/SkSLASTPrefixExpression.h"
30#include "ast/SkSLASTReturnStatement.h"
Ethan Nicholas762466e2017-06-29 10:03:38 -040031#include "ast/SkSLASTSection.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070032#include "ast/SkSLASTStatement.h"
33#include "ast/SkSLASTSuffixExpression.h"
Ethan Nicholasaf197692017-02-27 13:26:45 -050034#include "ast/SkSLASTSwitchStatement.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070035#include "ast/SkSLASTTernaryExpression.h"
36#include "ast/SkSLASTVarDeclaration.h"
37#include "ast/SkSLASTVarDeclarationStatement.h"
38#include "ast/SkSLASTWhileStatement.h"
39#include "ir/SkSLBlock.h"
40#include "ir/SkSLExpression.h"
41#include "ir/SkSLExtension.h"
42#include "ir/SkSLFunctionDefinition.h"
43#include "ir/SkSLInterfaceBlock.h"
44#include "ir/SkSLModifiers.h"
ethannicholas5961bc92016-10-12 06:39:56 -070045#include "ir/SkSLModifiersDeclaration.h"
Ethan Nicholas941e7e22016-12-12 15:33:30 -050046#include "ir/SkSLProgram.h"
Ethan Nicholas762466e2017-06-29 10:03:38 -040047#include "ir/SkSLSection.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070048#include "ir/SkSLSymbolTable.h"
49#include "ir/SkSLStatement.h"
50#include "ir/SkSLType.h"
51#include "ir/SkSLTypeReference.h"
ethannicholas22f939e2016-10-13 13:25:34 -070052#include "ir/SkSLVarDeclarations.h"
Ethan Nicholas8f7e28f2018-03-26 14:24:27 -040053#include "ir/SkSLVariableReference.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070054
55namespace SkSL {
56
Ethan Nicholascb0f4092019-04-19 11:26:50 -040057struct Swizzle;
58
ethannicholasb3058bd2016-07-01 08:22:01 -070059/**
Ethan Nicholas11d53972016-11-28 11:23:23 -050060 * Performs semantic analysis on an abstract syntax tree (AST) and produces the corresponding
ethannicholasb3058bd2016-07-01 08:22:01 -070061 * (unoptimized) intermediate representation (IR).
62 */
63class IRGenerator {
64public:
Ethan Nicholas11d53972016-11-28 11:23:23 -050065 IRGenerator(const Context* context, std::shared_ptr<SymbolTable> root,
ethannicholasd598f792016-07-25 10:08:54 -070066 ErrorReporter& errorReporter);
ethannicholasb3058bd2016-07-01 08:22:01 -070067
Robert Phillipsfe8da172018-01-24 14:52:02 +000068 void convertProgram(Program::Kind kind,
69 const char* text,
Ethan Nicholas5b5f0962017-09-11 13:50:14 -070070 size_t length,
Ethan Nicholas7da6dfa2017-06-21 11:25:18 -040071 SymbolTable& types,
Ethan Nicholas7da6dfa2017-06-21 11:25:18 -040072 std::vector<std::unique_ptr<ProgramElement>>* result);
ethannicholasb3058bd2016-07-01 08:22:01 -070073
Ethan Nicholas86a43402017-01-19 13:32:00 -050074 /**
75 * If both operands are compile-time constants and can be folded, returns an expression
76 * representing the folded value. Otherwise, returns null. Note that unlike most other functions
77 * here, null does not represent a compilation error.
78 */
79 std::unique_ptr<Expression> constantFold(const Expression& left,
80 Token::Kind op,
81 const Expression& right) const;
Ethan Nicholas00543112018-07-31 09:44:36 -040082
83 std::unique_ptr<Expression> getArg(int offset, String name) const;
84
Ethan Nicholas941e7e22016-12-12 15:33:30 -050085 Program::Inputs fInputs;
Ethan Nicholas762466e2017-06-29 10:03:38 -040086 const Program::Settings* fSettings;
Ethan Nicholas86a43402017-01-19 13:32:00 -050087 const Context& fContext;
Ethan Nicholas00543112018-07-31 09:44:36 -040088 Program::Kind fKind;
Ethan Nicholas941e7e22016-12-12 15:33:30 -050089
ethannicholasb3058bd2016-07-01 08:22:01 -070090private:
Ethan Nicholas3605ace2016-11-21 15:59:48 -050091 /**
Ethan Nicholas941e7e22016-12-12 15:33:30 -050092 * Prepare to compile a program. Resets state, pushes a new symbol table, and installs the
93 * settings.
Ethan Nicholas3605ace2016-11-21 15:59:48 -050094 */
Ethan Nicholas3c6ae622018-04-24 13:06:09 -040095 void start(const Program::Settings* settings,
96 std::vector<std::unique_ptr<ProgramElement>>* inherited);
Ethan Nicholas3605ace2016-11-21 15:59:48 -050097
98 /**
99 * Performs cleanup after compilation is complete.
100 */
101 void finish();
102
ethannicholasb3058bd2016-07-01 08:22:01 -0700103 void pushSymbolTable();
104 void popSymbolTable();
105
Ethan Nicholas7da6dfa2017-06-21 11:25:18 -0400106 std::unique_ptr<VarDeclarations> convertVarDeclarations(const ASTVarDeclarations& decl,
107 Variable::Storage storage);
Ethan Nicholasaae47c82017-11-10 15:34:03 -0500108 void convertFunction(const ASTFunction& f);
Ethan Nicholas7da6dfa2017-06-21 11:25:18 -0400109 std::unique_ptr<Statement> convertStatement(const ASTStatement& statement);
110 std::unique_ptr<Expression> convertExpression(const ASTExpression& expression);
111 std::unique_ptr<ModifiersDeclaration> convertModifiersDeclaration(
112 const ASTModifiersDeclaration& m);
113
ethannicholasd598f792016-07-25 10:08:54 -0700114 const Type* convertType(const ASTType& type);
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);
Ethan Nicholasdcba08e2017-08-02 10:52:54 -0400118 int callCost(const FunctionDeclaration& function,
119 const std::vector<std::unique_ptr<Expression>>& arguments);
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700120 std::unique_ptr<Expression> call(int offset, std::unique_ptr<Expression> function,
ethannicholasb3058bd2016-07-01 08:22:01 -0700121 std::vector<std::unique_ptr<Expression>> arguments);
Ethan Nicholasdcba08e2017-08-02 10:52:54 -0400122 int coercionCost(const Expression& expr, const Type& type);
ethannicholasd598f792016-07-25 10:08:54 -0700123 std::unique_ptr<Expression> coerce(std::unique_ptr<Expression> expr, const Type& type);
Ethan Nicholas26a9aad2018-03-27 14:10:52 -0400124 std::unique_ptr<Expression> convertAppend(int offset,
125 const std::vector<std::unique_ptr<ASTExpression>>& args);
ethannicholasb3058bd2016-07-01 08:22:01 -0700126 std::unique_ptr<Block> convertBlock(const ASTBlock& block);
127 std::unique_ptr<Statement> convertBreak(const ASTBreakStatement& b);
Ethan Nicholas84645e32017-02-09 13:57:14 -0500128 std::unique_ptr<Expression> convertNumberConstructor(
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700129 int offset,
Ethan Nicholas84645e32017-02-09 13:57:14 -0500130 const Type& type,
131 std::vector<std::unique_ptr<Expression>> params);
132 std::unique_ptr<Expression> convertCompoundConstructor(
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700133 int offset,
Ethan Nicholas84645e32017-02-09 13:57:14 -0500134 const Type& type,
135 std::vector<std::unique_ptr<Expression>> params);
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700136 std::unique_ptr<Expression> convertConstructor(int offset,
Ethan Nicholas11d53972016-11-28 11:23:23 -0500137 const Type& type,
ethannicholasb3058bd2016-07-01 08:22:01 -0700138 std::vector<std::unique_ptr<Expression>> params);
139 std::unique_ptr<Statement> convertContinue(const ASTContinueStatement& c);
140 std::unique_ptr<Statement> convertDiscard(const ASTDiscardStatement& d);
141 std::unique_ptr<Statement> convertDo(const ASTDoStatement& d);
Ethan Nicholasaf197692017-02-27 13:26:45 -0500142 std::unique_ptr<Statement> convertSwitch(const ASTSwitchStatement& s);
ethannicholasb3058bd2016-07-01 08:22:01 -0700143 std::unique_ptr<Expression> convertBinaryExpression(const ASTBinaryExpression& expression);
144 std::unique_ptr<Extension> convertExtension(const ASTExtension& e);
145 std::unique_ptr<Statement> convertExpressionStatement(const ASTExpressionStatement& s);
146 std::unique_ptr<Statement> convertFor(const ASTForStatement& f);
147 std::unique_ptr<Expression> convertIdentifier(const ASTIdentifier& identifier);
148 std::unique_ptr<Statement> convertIf(const ASTIfStatement& s);
149 std::unique_ptr<Expression> convertIndex(std::unique_ptr<Expression> base,
150 const ASTExpression& index);
151 std::unique_ptr<InterfaceBlock> convertInterfaceBlock(const ASTInterfaceBlock& s);
Ethan Nicholas11d53972016-11-28 11:23:23 -0500152 Modifiers convertModifiers(const Modifiers& m);
ethannicholasb3058bd2016-07-01 08:22:01 -0700153 std::unique_ptr<Expression> convertPrefixExpression(const ASTPrefixExpression& expression);
154 std::unique_ptr<Statement> convertReturn(const ASTReturnStatement& r);
Ethan Nicholas762466e2017-06-29 10:03:38 -0400155 std::unique_ptr<Section> convertSection(const ASTSection& e);
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700156 std::unique_ptr<Expression> getCap(int offset, String name);
ethannicholasb3058bd2016-07-01 08:22:01 -0700157 std::unique_ptr<Expression> convertSuffixExpression(const ASTSuffixExpression& expression);
Ethan Nicholasaae47c82017-11-10 15:34:03 -0500158 std::unique_ptr<Expression> convertTypeField(int offset, const Type& type,
159 StringFragment field);
Ethan Nicholas11d53972016-11-28 11:23:23 -0500160 std::unique_ptr<Expression> convertField(std::unique_ptr<Expression> base,
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700161 StringFragment field);
ethannicholasb3058bd2016-07-01 08:22:01 -0700162 std::unique_ptr<Expression> convertSwizzle(std::unique_ptr<Expression> base,
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700163 StringFragment fields);
ethannicholasb3058bd2016-07-01 08:22:01 -0700164 std::unique_ptr<Expression> convertTernaryExpression(const ASTTernaryExpression& expression);
165 std::unique_ptr<Statement> convertVarDeclarationStatement(const ASTVarDeclarationStatement& s);
166 std::unique_ptr<Statement> convertWhile(const ASTWhileStatement& w);
Ethan Nicholasaae47c82017-11-10 15:34:03 -0500167 void convertEnum(const ASTEnum& e);
168 std::unique_ptr<Block> applyInvocationIDWorkaround(std::unique_ptr<Block> main);
Robert Phillipsfe8da172018-01-24 14:52:02 +0000169 // returns a statement which converts sk_Position from device to normalized coordinates
170 std::unique_ptr<Statement> getNormalizeSkPositionCode();
ethannicholasb3058bd2016-07-01 08:22:01 -0700171
172 void checkValid(const Expression& expr);
Ethan Nicholas8f7e28f2018-03-26 14:24:27 -0400173 void setRefKind(const Expression& expr, VariableReference::RefKind kind);
Ethan Nicholasaae47c82017-11-10 15:34:03 -0500174 void getConstantInt(const Expression& value, int64_t* out);
Ethan Nicholascb0f4092019-04-19 11:26:50 -0400175 bool checkSwizzleWrite(const Swizzle& swizzle);
ethannicholasb3058bd2016-07-01 08:22:01 -0700176
ethannicholasd598f792016-07-25 10:08:54 -0700177 const FunctionDeclaration* fCurrentFunction;
Ethan Nicholas762466e2017-06-29 10:03:38 -0400178 std::unordered_map<String, Program::Settings::Value> fCapsMap;
179 std::shared_ptr<SymbolTable> fRootSymbolTable;
ethannicholasb3058bd2016-07-01 08:22:01 -0700180 std::shared_ptr<SymbolTable> fSymbolTable;
Ethan Nicholas762466e2017-06-29 10:03:38 -0400181 // holds extra temp variable declarations needed for the current function
182 std::vector<std::unique_ptr<Statement>> fExtraVars;
ethannicholas22f939e2016-10-13 13:25:34 -0700183 int fLoopLevel;
Ethan Nicholasaf197692017-02-27 13:26:45 -0500184 int fSwitchLevel;
Ethan Nicholas762466e2017-06-29 10:03:38 -0400185 // count of temporary variables we have created
186 int fTmpCount;
ethannicholasb3058bd2016-07-01 08:22:01 -0700187 ErrorReporter& fErrors;
Ethan Nicholas7da6dfa2017-06-21 11:25:18 -0400188 int fInvocations;
Ethan Nicholasaae47c82017-11-10 15:34:03 -0500189 std::vector<std::unique_ptr<ProgramElement>>* fProgramElements;
Ethan Nicholas3c6ae622018-04-24 13:06:09 -0400190 const Variable* fSkPerVertex = nullptr;
Robert Phillipsfe8da172018-01-24 14:52:02 +0000191 Variable* fRTAdjust;
192 Variable* fRTAdjustInterfaceBlock;
193 int fRTAdjustFieldIndex;
Ethan Nicholas00543112018-07-31 09:44:36 -0400194 bool fStarted = false;
ethannicholasb3058bd2016-07-01 08:22:01 -0700195
196 friend class AutoSymbolTable;
ethannicholas22f939e2016-10-13 13:25:34 -0700197 friend class AutoLoopLevel;
Ethan Nicholasaf197692017-02-27 13:26:45 -0500198 friend class AutoSwitchLevel;
ethannicholasb3058bd2016-07-01 08:22:01 -0700199 friend class Compiler;
200};
201
202}
203
204#endif