blob: abb787c7038352a140bbc3ec6750219cfa624d48 [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
Ethan Nicholasfc994162019-06-06 10:04:27 -040011#include "src/sksl/SkSLASTFile.h"
12#include "src/sksl/SkSLASTNode.h"
Mike Klein4b432fa2019-06-06 11:44:05 -050013#include "src/sksl/SkSLErrorReporter.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050014#include "src/sksl/ir/SkSLBlock.h"
15#include "src/sksl/ir/SkSLExpression.h"
16#include "src/sksl/ir/SkSLExtension.h"
17#include "src/sksl/ir/SkSLFunctionDefinition.h"
18#include "src/sksl/ir/SkSLInterfaceBlock.h"
19#include "src/sksl/ir/SkSLModifiers.h"
20#include "src/sksl/ir/SkSLModifiersDeclaration.h"
21#include "src/sksl/ir/SkSLProgram.h"
22#include "src/sksl/ir/SkSLSection.h"
23#include "src/sksl/ir/SkSLStatement.h"
24#include "src/sksl/ir/SkSLSymbolTable.h"
25#include "src/sksl/ir/SkSLType.h"
26#include "src/sksl/ir/SkSLTypeReference.h"
27#include "src/sksl/ir/SkSLVarDeclarations.h"
28#include "src/sksl/ir/SkSLVariableReference.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070029
30namespace SkSL {
31
Ethan Nicholascb0f4092019-04-19 11:26:50 -040032struct Swizzle;
33
ethannicholasb3058bd2016-07-01 08:22:01 -070034/**
Ethan Nicholas11d53972016-11-28 11:23:23 -050035 * Performs semantic analysis on an abstract syntax tree (AST) and produces the corresponding
ethannicholasb3058bd2016-07-01 08:22:01 -070036 * (unoptimized) intermediate representation (IR).
37 */
38class IRGenerator {
39public:
Ethan Nicholas11d53972016-11-28 11:23:23 -050040 IRGenerator(const Context* context, std::shared_ptr<SymbolTable> root,
ethannicholasd598f792016-07-25 10:08:54 -070041 ErrorReporter& errorReporter);
ethannicholasb3058bd2016-07-01 08:22:01 -070042
Robert Phillipsfe8da172018-01-24 14:52:02 +000043 void convertProgram(Program::Kind kind,
44 const char* text,
Ethan Nicholas5b5f0962017-09-11 13:50:14 -070045 size_t length,
Ethan Nicholas7da6dfa2017-06-21 11:25:18 -040046 SymbolTable& types,
Ethan Nicholas7da6dfa2017-06-21 11:25:18 -040047 std::vector<std::unique_ptr<ProgramElement>>* result);
ethannicholasb3058bd2016-07-01 08:22:01 -070048
Ethan Nicholas86a43402017-01-19 13:32:00 -050049 /**
50 * If both operands are compile-time constants and can be folded, returns an expression
51 * representing the folded value. Otherwise, returns null. Note that unlike most other functions
52 * here, null does not represent a compilation error.
53 */
54 std::unique_ptr<Expression> constantFold(const Expression& left,
55 Token::Kind op,
56 const Expression& right) const;
Ethan Nicholas00543112018-07-31 09:44:36 -040057
58 std::unique_ptr<Expression> getArg(int offset, String name) const;
59
Ethan Nicholas941e7e22016-12-12 15:33:30 -050060 Program::Inputs fInputs;
Ethan Nicholas762466e2017-06-29 10:03:38 -040061 const Program::Settings* fSettings;
Ethan Nicholas86a43402017-01-19 13:32:00 -050062 const Context& fContext;
Ethan Nicholas00543112018-07-31 09:44:36 -040063 Program::Kind fKind;
Ethan Nicholas941e7e22016-12-12 15:33:30 -050064
ethannicholasb3058bd2016-07-01 08:22:01 -070065private:
Ethan Nicholas3605ace2016-11-21 15:59:48 -050066 /**
Ethan Nicholas941e7e22016-12-12 15:33:30 -050067 * Prepare to compile a program. Resets state, pushes a new symbol table, and installs the
68 * settings.
Ethan Nicholas3605ace2016-11-21 15:59:48 -050069 */
Ethan Nicholas3c6ae622018-04-24 13:06:09 -040070 void start(const Program::Settings* settings,
71 std::vector<std::unique_ptr<ProgramElement>>* inherited);
Ethan Nicholas3605ace2016-11-21 15:59:48 -050072
73 /**
74 * Performs cleanup after compilation is complete.
75 */
76 void finish();
77
ethannicholasb3058bd2016-07-01 08:22:01 -070078 void pushSymbolTable();
79 void popSymbolTable();
80
Ethan Nicholasfc994162019-06-06 10:04:27 -040081 std::unique_ptr<VarDeclarations> convertVarDeclarations(const ASTNode& decl,
Ethan Nicholas7da6dfa2017-06-21 11:25:18 -040082 Variable::Storage storage);
Ethan Nicholasfc994162019-06-06 10:04:27 -040083 void convertFunction(const ASTNode& f);
84 std::unique_ptr<Statement> convertStatement(const ASTNode& statement);
85 std::unique_ptr<Expression> convertExpression(const ASTNode& expression);
86 std::unique_ptr<ModifiersDeclaration> convertModifiersDeclaration(const ASTNode& m);
Ethan Nicholas7da6dfa2017-06-21 11:25:18 -040087
Ethan Nicholasfc994162019-06-06 10:04:27 -040088 const Type* convertType(const ASTNode& type);
Ethan Nicholas5b5f0962017-09-11 13:50:14 -070089 std::unique_ptr<Expression> call(int offset,
Ethan Nicholas11d53972016-11-28 11:23:23 -050090 const FunctionDeclaration& function,
ethannicholasb3058bd2016-07-01 08:22:01 -070091 std::vector<std::unique_ptr<Expression>> arguments);
Ethan Nicholasdcba08e2017-08-02 10:52:54 -040092 int callCost(const FunctionDeclaration& function,
93 const std::vector<std::unique_ptr<Expression>>& arguments);
Ethan Nicholas5b5f0962017-09-11 13:50:14 -070094 std::unique_ptr<Expression> call(int offset, std::unique_ptr<Expression> function,
ethannicholasb3058bd2016-07-01 08:22:01 -070095 std::vector<std::unique_ptr<Expression>> arguments);
Ethan Nicholasdcba08e2017-08-02 10:52:54 -040096 int coercionCost(const Expression& expr, const Type& type);
ethannicholasd598f792016-07-25 10:08:54 -070097 std::unique_ptr<Expression> coerce(std::unique_ptr<Expression> expr, const Type& type);
Ethan Nicholasfc994162019-06-06 10:04:27 -040098 std::unique_ptr<Block> convertBlock(const ASTNode& block);
99 std::unique_ptr<Statement> convertBreak(const ASTNode& b);
Ethan Nicholas84645e32017-02-09 13:57:14 -0500100 std::unique_ptr<Expression> convertNumberConstructor(
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700101 int offset,
Ethan Nicholas84645e32017-02-09 13:57:14 -0500102 const Type& type,
103 std::vector<std::unique_ptr<Expression>> params);
104 std::unique_ptr<Expression> convertCompoundConstructor(
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700105 int offset,
Ethan Nicholas84645e32017-02-09 13:57:14 -0500106 const Type& type,
107 std::vector<std::unique_ptr<Expression>> params);
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700108 std::unique_ptr<Expression> convertConstructor(int offset,
Ethan Nicholas11d53972016-11-28 11:23:23 -0500109 const Type& type,
ethannicholasb3058bd2016-07-01 08:22:01 -0700110 std::vector<std::unique_ptr<Expression>> params);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400111 std::unique_ptr<Statement> convertContinue(const ASTNode& c);
112 std::unique_ptr<Statement> convertDiscard(const ASTNode& d);
113 std::unique_ptr<Statement> convertDo(const ASTNode& d);
114 std::unique_ptr<Statement> convertSwitch(const ASTNode& s);
115 std::unique_ptr<Expression> convertBinaryExpression(const ASTNode& expression);
116 std::unique_ptr<Extension> convertExtension(int offset, StringFragment name);
117 std::unique_ptr<Statement> convertExpressionStatement(const ASTNode& s);
118 std::unique_ptr<Statement> convertFor(const ASTNode& f);
119 std::unique_ptr<Expression> convertIdentifier(const ASTNode& identifier);
120 std::unique_ptr<Statement> convertIf(const ASTNode& s);
ethannicholasb3058bd2016-07-01 08:22:01 -0700121 std::unique_ptr<Expression> convertIndex(std::unique_ptr<Expression> base,
Ethan Nicholasfc994162019-06-06 10:04:27 -0400122 const ASTNode& index);
123 std::unique_ptr<InterfaceBlock> convertInterfaceBlock(const ASTNode& s);
Ethan Nicholas11d53972016-11-28 11:23:23 -0500124 Modifiers convertModifiers(const Modifiers& m);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400125 std::unique_ptr<Expression> convertPrefixExpression(const ASTNode& expression);
126 std::unique_ptr<Statement> convertReturn(const ASTNode& r);
127 std::unique_ptr<Section> convertSection(const ASTNode& e);
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700128 std::unique_ptr<Expression> getCap(int offset, String name);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400129 std::unique_ptr<Expression> convertCallExpression(const ASTNode& expression);
130 std::unique_ptr<Expression> convertFieldExpression(const ASTNode& expression);
131 std::unique_ptr<Expression> convertIndexExpression(const ASTNode& expression);
132 std::unique_ptr<Expression> convertPostfixExpression(const ASTNode& expression);
Ethan Nicholasaae47c82017-11-10 15:34:03 -0500133 std::unique_ptr<Expression> convertTypeField(int offset, const Type& type,
134 StringFragment field);
Ethan Nicholas11d53972016-11-28 11:23:23 -0500135 std::unique_ptr<Expression> convertField(std::unique_ptr<Expression> base,
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700136 StringFragment field);
ethannicholasb3058bd2016-07-01 08:22:01 -0700137 std::unique_ptr<Expression> convertSwizzle(std::unique_ptr<Expression> base,
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700138 StringFragment fields);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400139 std::unique_ptr<Expression> convertTernaryExpression(const ASTNode& expression);
140 std::unique_ptr<Statement> convertVarDeclarationStatement(const ASTNode& s);
141 std::unique_ptr<Statement> convertWhile(const ASTNode& w);
142 void convertEnum(const ASTNode& e);
Ethan Nicholasaae47c82017-11-10 15:34:03 -0500143 std::unique_ptr<Block> applyInvocationIDWorkaround(std::unique_ptr<Block> main);
Robert Phillipsfe8da172018-01-24 14:52:02 +0000144 // returns a statement which converts sk_Position from device to normalized coordinates
145 std::unique_ptr<Statement> getNormalizeSkPositionCode();
Chris Daltonb0fd4b12019-10-29 13:41:22 -0600146 void removeSampleMask(std::vector<std::unique_ptr<ProgramElement>>* out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700147
148 void checkValid(const Expression& expr);
Ethan Nicholas8f7e28f2018-03-26 14:24:27 -0400149 void setRefKind(const Expression& expr, VariableReference::RefKind kind);
Ethan Nicholasaae47c82017-11-10 15:34:03 -0500150 void getConstantInt(const Expression& value, int64_t* out);
Ethan Nicholascb0f4092019-04-19 11:26:50 -0400151 bool checkSwizzleWrite(const Swizzle& swizzle);
ethannicholasb3058bd2016-07-01 08:22:01 -0700152
Ethan Nicholasfc994162019-06-06 10:04:27 -0400153 std::unique_ptr<ASTFile> fFile;
ethannicholasd598f792016-07-25 10:08:54 -0700154 const FunctionDeclaration* fCurrentFunction;
Ethan Nicholas762466e2017-06-29 10:03:38 -0400155 std::unordered_map<String, Program::Settings::Value> fCapsMap;
156 std::shared_ptr<SymbolTable> fRootSymbolTable;
ethannicholasb3058bd2016-07-01 08:22:01 -0700157 std::shared_ptr<SymbolTable> fSymbolTable;
Ethan Nicholas762466e2017-06-29 10:03:38 -0400158 // holds extra temp variable declarations needed for the current function
159 std::vector<std::unique_ptr<Statement>> fExtraVars;
ethannicholas22f939e2016-10-13 13:25:34 -0700160 int fLoopLevel;
Ethan Nicholasaf197692017-02-27 13:26:45 -0500161 int fSwitchLevel;
Ethan Nicholas762466e2017-06-29 10:03:38 -0400162 // count of temporary variables we have created
163 int fTmpCount;
ethannicholasb3058bd2016-07-01 08:22:01 -0700164 ErrorReporter& fErrors;
Ethan Nicholas7da6dfa2017-06-21 11:25:18 -0400165 int fInvocations;
Ethan Nicholasaae47c82017-11-10 15:34:03 -0500166 std::vector<std::unique_ptr<ProgramElement>>* fProgramElements;
Ethan Nicholas3c6ae622018-04-24 13:06:09 -0400167 const Variable* fSkPerVertex = nullptr;
Robert Phillipsfe8da172018-01-24 14:52:02 +0000168 Variable* fRTAdjust;
169 Variable* fRTAdjustInterfaceBlock;
170 int fRTAdjustFieldIndex;
Ethan Nicholas00543112018-07-31 09:44:36 -0400171 bool fStarted = false;
Chris Daltonb0fd4b12019-10-29 13:41:22 -0600172 bool fUsesSampleMask = false;
ethannicholasb3058bd2016-07-01 08:22:01 -0700173
174 friend class AutoSymbolTable;
ethannicholas22f939e2016-10-13 13:25:34 -0700175 friend class AutoLoopLevel;
Ethan Nicholasaf197692017-02-27 13:26:45 -0500176 friend class AutoSwitchLevel;
ethannicholasb3058bd2016-07-01 08:22:01 -0700177 friend class Compiler;
178};
179
180}
181
182#endif