blob: b0a449aceac23532d3f476aaefdfbea6a8a56d8a [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"
19#include "ast/SkSLASTExpression.h"
20#include "ast/SkSLASTExpressionStatement.h"
21#include "ast/SkSLASTExtension.h"
22#include "ast/SkSLASTForStatement.h"
23#include "ast/SkSLASTFunction.h"
24#include "ast/SkSLASTIdentifier.h"
25#include "ast/SkSLASTIfStatement.h"
26#include "ast/SkSLASTInterfaceBlock.h"
ethannicholas5961bc92016-10-12 06:39:56 -070027#include "ast/SkSLASTModifiersDeclaration.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070028#include "ast/SkSLASTPrefixExpression.h"
29#include "ast/SkSLASTReturnStatement.h"
30#include "ast/SkSLASTStatement.h"
31#include "ast/SkSLASTSuffixExpression.h"
Ethan Nicholasaf197692017-02-27 13:26:45 -050032#include "ast/SkSLASTSwitchStatement.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070033#include "ast/SkSLASTTernaryExpression.h"
34#include "ast/SkSLASTVarDeclaration.h"
35#include "ast/SkSLASTVarDeclarationStatement.h"
36#include "ast/SkSLASTWhileStatement.h"
37#include "ir/SkSLBlock.h"
38#include "ir/SkSLExpression.h"
39#include "ir/SkSLExtension.h"
40#include "ir/SkSLFunctionDefinition.h"
41#include "ir/SkSLInterfaceBlock.h"
42#include "ir/SkSLModifiers.h"
ethannicholas5961bc92016-10-12 06:39:56 -070043#include "ir/SkSLModifiersDeclaration.h"
Ethan Nicholas941e7e22016-12-12 15:33:30 -050044#include "ir/SkSLProgram.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070045#include "ir/SkSLSymbolTable.h"
46#include "ir/SkSLStatement.h"
47#include "ir/SkSLType.h"
48#include "ir/SkSLTypeReference.h"
ethannicholas22f939e2016-10-13 13:25:34 -070049#include "ir/SkSLVarDeclarations.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070050
51namespace SkSL {
52
Ethan Nicholas3605ace2016-11-21 15:59:48 -050053struct CapValue {
54 CapValue()
55 : fKind(kInt_Kind)
56 , fValue(-1) {
57 ASSERT(false);
58 }
59
60 CapValue(bool b)
61 : fKind(kBool_Kind)
62 , fValue(b) {}
63
64 CapValue(int i)
65 : fKind(kInt_Kind)
66 , fValue(i) {}
67
68 enum {
69 kBool_Kind,
70 kInt_Kind,
71 } fKind;
72 int fValue;
73};
74
ethannicholasb3058bd2016-07-01 08:22:01 -070075/**
Ethan Nicholas11d53972016-11-28 11:23:23 -050076 * Performs semantic analysis on an abstract syntax tree (AST) and produces the corresponding
ethannicholasb3058bd2016-07-01 08:22:01 -070077 * (unoptimized) intermediate representation (IR).
78 */
79class IRGenerator {
80public:
Ethan Nicholas11d53972016-11-28 11:23:23 -050081 IRGenerator(const Context* context, std::shared_ptr<SymbolTable> root,
ethannicholasd598f792016-07-25 10:08:54 -070082 ErrorReporter& errorReporter);
ethannicholasb3058bd2016-07-01 08:22:01 -070083
Ethan Nicholas11d53972016-11-28 11:23:23 -050084 std::unique_ptr<VarDeclarations> convertVarDeclarations(const ASTVarDeclarations& decl,
ethannicholas14fe8cc2016-09-07 13:37:16 -070085 Variable::Storage storage);
ethannicholasb3058bd2016-07-01 08:22:01 -070086 std::unique_ptr<FunctionDefinition> convertFunction(const ASTFunction& f);
87 std::unique_ptr<Statement> convertStatement(const ASTStatement& statement);
88 std::unique_ptr<Expression> convertExpression(const ASTExpression& expression);
ethannicholas5961bc92016-10-12 06:39:56 -070089 std::unique_ptr<ModifiersDeclaration> convertModifiersDeclaration(
90 const ASTModifiersDeclaration& m);
ethannicholasb3058bd2016-07-01 08:22:01 -070091
Ethan Nicholas86a43402017-01-19 13:32:00 -050092 /**
93 * If both operands are compile-time constants and can be folded, returns an expression
94 * representing the folded value. Otherwise, returns null. Note that unlike most other functions
95 * here, null does not represent a compilation error.
96 */
97 std::unique_ptr<Expression> constantFold(const Expression& left,
98 Token::Kind op,
99 const Expression& right) const;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500100 Program::Inputs fInputs;
Ethan Nicholas86a43402017-01-19 13:32:00 -0500101 const Context& fContext;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500102
ethannicholasb3058bd2016-07-01 08:22:01 -0700103private:
Ethan Nicholas3605ace2016-11-21 15:59:48 -0500104 /**
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500105 * Prepare to compile a program. Resets state, pushes a new symbol table, and installs the
106 * settings.
Ethan Nicholas3605ace2016-11-21 15:59:48 -0500107 */
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500108 void start(const Program::Settings* settings);
Ethan Nicholas3605ace2016-11-21 15:59:48 -0500109
110 /**
111 * Performs cleanup after compilation is complete.
112 */
113 void finish();
114
ethannicholasb3058bd2016-07-01 08:22:01 -0700115 void pushSymbolTable();
116 void popSymbolTable();
117
ethannicholasd598f792016-07-25 10:08:54 -0700118 const Type* convertType(const ASTType& type);
Ethan Nicholas11d53972016-11-28 11:23:23 -0500119 std::unique_ptr<Expression> call(Position position,
120 const FunctionDeclaration& function,
ethannicholasb3058bd2016-07-01 08:22:01 -0700121 std::vector<std::unique_ptr<Expression>> arguments);
Ethan Nicholas11d53972016-11-28 11:23:23 -0500122 bool determineCallCost(const FunctionDeclaration& function,
ethannicholasb3058bd2016-07-01 08:22:01 -0700123 const std::vector<std::unique_ptr<Expression>>& arguments,
124 int* outCost);
Ethan Nicholas11d53972016-11-28 11:23:23 -0500125 std::unique_ptr<Expression> call(Position position, std::unique_ptr<Expression> function,
ethannicholasb3058bd2016-07-01 08:22:01 -0700126 std::vector<std::unique_ptr<Expression>> arguments);
ethannicholasd598f792016-07-25 10:08:54 -0700127 std::unique_ptr<Expression> coerce(std::unique_ptr<Expression> expr, const Type& type);
ethannicholasb3058bd2016-07-01 08:22:01 -0700128 std::unique_ptr<Block> convertBlock(const ASTBlock& block);
129 std::unique_ptr<Statement> convertBreak(const ASTBreakStatement& b);
Ethan Nicholas84645e32017-02-09 13:57:14 -0500130 std::unique_ptr<Expression> convertNumberConstructor(
131 Position position,
132 const Type& type,
133 std::vector<std::unique_ptr<Expression>> params);
134 std::unique_ptr<Expression> convertCompoundConstructor(
135 Position position,
136 const Type& type,
137 std::vector<std::unique_ptr<Expression>> params);
Ethan Nicholas11d53972016-11-28 11:23:23 -0500138 std::unique_ptr<Expression> convertConstructor(Position position,
139 const Type& type,
ethannicholasb3058bd2016-07-01 08:22:01 -0700140 std::vector<std::unique_ptr<Expression>> params);
141 std::unique_ptr<Statement> convertContinue(const ASTContinueStatement& c);
142 std::unique_ptr<Statement> convertDiscard(const ASTDiscardStatement& d);
143 std::unique_ptr<Statement> convertDo(const ASTDoStatement& d);
Ethan Nicholasaf197692017-02-27 13:26:45 -0500144 std::unique_ptr<Statement> convertSwitch(const ASTSwitchStatement& s);
ethannicholasb3058bd2016-07-01 08:22:01 -0700145 std::unique_ptr<Expression> convertBinaryExpression(const ASTBinaryExpression& expression);
146 std::unique_ptr<Extension> convertExtension(const ASTExtension& e);
147 std::unique_ptr<Statement> convertExpressionStatement(const ASTExpressionStatement& s);
148 std::unique_ptr<Statement> convertFor(const ASTForStatement& f);
149 std::unique_ptr<Expression> convertIdentifier(const ASTIdentifier& identifier);
150 std::unique_ptr<Statement> convertIf(const ASTIfStatement& s);
151 std::unique_ptr<Expression> convertIndex(std::unique_ptr<Expression> base,
152 const ASTExpression& index);
153 std::unique_ptr<InterfaceBlock> convertInterfaceBlock(const ASTInterfaceBlock& s);
Ethan Nicholas11d53972016-11-28 11:23:23 -0500154 Modifiers convertModifiers(const Modifiers& m);
ethannicholasb3058bd2016-07-01 08:22:01 -0700155 std::unique_ptr<Expression> convertPrefixExpression(const ASTPrefixExpression& expression);
156 std::unique_ptr<Statement> convertReturn(const ASTReturnStatement& r);
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400157 std::unique_ptr<Expression> getCap(Position position, String name);
ethannicholasb3058bd2016-07-01 08:22:01 -0700158 std::unique_ptr<Expression> convertSuffixExpression(const ASTSuffixExpression& expression);
Ethan Nicholas11d53972016-11-28 11:23:23 -0500159 std::unique_ptr<Expression> convertField(std::unique_ptr<Expression> base,
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400160 const String& field);
ethannicholasb3058bd2016-07-01 08:22:01 -0700161 std::unique_ptr<Expression> convertSwizzle(std::unique_ptr<Expression> base,
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400162 const String& fields);
ethannicholasb3058bd2016-07-01 08:22:01 -0700163 std::unique_ptr<Expression> convertTernaryExpression(const ASTTernaryExpression& expression);
164 std::unique_ptr<Statement> convertVarDeclarationStatement(const ASTVarDeclarationStatement& s);
165 std::unique_ptr<Statement> convertWhile(const ASTWhileStatement& w);
166
167 void checkValid(const Expression& expr);
Ethan Nicholas86a43402017-01-19 13:32:00 -0500168 void markWrittenTo(const Expression& expr, bool readWrite);
ethannicholasb3058bd2016-07-01 08:22:01 -0700169
ethannicholasd598f792016-07-25 10:08:54 -0700170 const FunctionDeclaration* fCurrentFunction;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500171 const Program::Settings* fSettings;
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400172 std::unordered_map<String, CapValue> fCapsMap;
ethannicholasb3058bd2016-07-01 08:22:01 -0700173 std::shared_ptr<SymbolTable> fSymbolTable;
ethannicholas22f939e2016-10-13 13:25:34 -0700174 int fLoopLevel;
Ethan Nicholasaf197692017-02-27 13:26:45 -0500175 int fSwitchLevel;
ethannicholasb3058bd2016-07-01 08:22:01 -0700176 ErrorReporter& fErrors;
177
178 friend class AutoSymbolTable;
ethannicholas22f939e2016-10-13 13:25:34 -0700179 friend class AutoLoopLevel;
Ethan Nicholasaf197692017-02-27 13:26:45 -0500180 friend class AutoSwitchLevel;
ethannicholasb3058bd2016-07-01 08:22:01 -0700181 friend class Compiler;
182};
183
184}
185
186#endif