blob: 289bb58cc7f51e974fe7ca8cd083647800e9188d [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 Nicholasdaed2592021-03-04 14:30:25 -050014#include "include/private/SkSLModifiers.h"
Ethan Nicholas24c17722021-03-09 13:10:59 -050015#include "include/private/SkSLStatement.h"
Ethan Nicholasfc994162019-06-06 10:04:27 -040016#include "src/sksl/SkSLASTFile.h"
17#include "src/sksl/SkSLASTNode.h"
John Stiles45990502021-02-16 10:55:27 -050018#include "src/sksl/SkSLOperators.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050019#include "src/sksl/ir/SkSLBlock.h"
20#include "src/sksl/ir/SkSLExpression.h"
21#include "src/sksl/ir/SkSLExtension.h"
22#include "src/sksl/ir/SkSLFunctionDefinition.h"
23#include "src/sksl/ir/SkSLInterfaceBlock.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050024#include "src/sksl/ir/SkSLModifiersDeclaration.h"
25#include "src/sksl/ir/SkSLProgram.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050026#include "src/sksl/ir/SkSLSymbolTable.h"
27#include "src/sksl/ir/SkSLType.h"
28#include "src/sksl/ir/SkSLTypeReference.h"
29#include "src/sksl/ir/SkSLVarDeclarations.h"
30#include "src/sksl/ir/SkSLVariableReference.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070031
32namespace SkSL {
33
Ethan Nicholas95046142021-01-07 10:57:27 -050034namespace dsl {
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -050035 class DSLCore;
Ethan Nicholas1ff76092021-01-28 10:02:43 -050036 class DSLFunction;
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -050037 class DSLVar;
Ethan Nicholas95046142021-01-07 10:57:27 -050038 class DSLWriter;
39}
40
Brian Osmanbe0b3b72021-01-06 14:27:35 -050041class ExternalFunction;
Ethan Nicholas1e9f7f32020-10-08 05:28:32 -040042class FunctionCall;
John Stilesdc75a972020-11-25 16:24:55 -050043class StructDefinition;
Brian Osman3d87e9f2020-10-08 11:50:22 -040044struct ParsedModule;
Ethan Nicholascb0f4092019-04-19 11:26:50 -040045struct Swizzle;
46
ethannicholasb3058bd2016-07-01 08:22:01 -070047/**
John Stiles810c8cf2020-08-26 19:46:27 -040048 * Intrinsics are passed between the Compiler and the IRGenerator using IRIntrinsicMaps.
49 */
Brian Osman2b469eb2020-09-21 11:32:10 -040050class IRIntrinsicMap {
51public:
52 IRIntrinsicMap(IRIntrinsicMap* parent) : fParent(parent) {}
53
54 void insertOrDie(String key, std::unique_ptr<ProgramElement> element) {
55 SkASSERT(fIntrinsics.find(key) == fIntrinsics.end());
56 fIntrinsics[key] = Intrinsic{std::move(element), false};
57 }
58
Brian Osmanafa18ee2020-10-07 17:47:45 -040059 const ProgramElement* find(const String& key) {
60 auto iter = fIntrinsics.find(key);
61 if (iter == fIntrinsics.end()) {
62 return fParent ? fParent->find(key) : nullptr;
63 }
64 return iter->second.fIntrinsic.get();
65 }
66
Brian Osman2b469eb2020-09-21 11:32:10 -040067 // Only returns an intrinsic that isn't already marked as included, and then marks it.
Brian Osman00a8b5b2020-10-02 09:06:04 -040068 const ProgramElement* findAndInclude(const String& key) {
Brian Osman2b469eb2020-09-21 11:32:10 -040069 auto iter = fIntrinsics.find(key);
70 if (iter == fIntrinsics.end()) {
71 return fParent ? fParent->findAndInclude(key) : nullptr;
72 }
73 if (iter->second.fAlreadyIncluded) {
74 return nullptr;
75 }
76 iter->second.fAlreadyIncluded = true;
77 return iter->second.fIntrinsic.get();
78 }
79
80 void resetAlreadyIncluded() {
81 for (auto& pair : fIntrinsics) {
82 pair.second.fAlreadyIncluded = false;
83 }
84 if (fParent) {
85 fParent->resetAlreadyIncluded();
86 }
87 }
88
89private:
90 struct Intrinsic {
91 std::unique_ptr<ProgramElement> fIntrinsic;
92 bool fAlreadyIncluded = false;
93 };
94
95 std::unordered_map<String, Intrinsic> fIntrinsics;
96 IRIntrinsicMap* fParent = nullptr;
John Stiles810c8cf2020-08-26 19:46:27 -040097};
John Stiles810c8cf2020-08-26 19:46:27 -040098
99/**
Ethan Nicholas11d53972016-11-28 11:23:23 -0500100 * Performs semantic analysis on an abstract syntax tree (AST) and produces the corresponding
ethannicholasb3058bd2016-07-01 08:22:01 -0700101 * (unoptimized) intermediate representation (IR).
102 */
103class IRGenerator {
104public:
John Stilesc1a98b82021-02-24 13:35:02 -0500105 IRGenerator(const Context* context);
ethannicholasb3058bd2016-07-01 08:22:01 -0700106
Brian Osman88cda172020-10-09 12:05:16 -0400107 struct IRBundle {
108 std::vector<std::unique_ptr<ProgramElement>> fElements;
Brian Osman133724c2020-10-28 14:14:39 -0400109 std::vector<const ProgramElement*> fSharedElements;
Brian Osman88cda172020-10-09 12:05:16 -0400110 std::shared_ptr<SymbolTable> fSymbolTable;
111 Program::Inputs fInputs;
112 };
113
114 /**
John Stilesaecf8d52021-05-14 12:15:01 -0400115 * If externalFunctions is supplied, those values are registered in the symbol table of the
Brian Osman88cda172020-10-09 12:05:16 -0400116 * Program, but ownership is *not* transferred. It is up to the caller to keep them alive.
117 */
Brian Osmanbe0b3b72021-01-06 14:27:35 -0500118 IRBundle convertProgram(
Brian Osmanbe0b3b72021-01-06 14:27:35 -0500119 const ParsedModule& base,
120 bool isBuiltinCode,
Ethan Nicholas6823b502021-06-15 11:42:07 -0400121 skstd::string_view text);
ethannicholasb3058bd2016-07-01 08:22:01 -0700122
John Stilesd1204642021-02-17 16:30:02 -0500123 const Program::Settings& settings() const { return fContext.fConfig->fSettings; }
124 ProgramKind programKind() const { return fContext.fConfig->fKind; }
Brian Osman88cda172020-10-09 12:05:16 -0400125
Ethan Nicholas39f6da42021-08-23 13:10:07 -0400126 ErrorReporter& errorReporter() const { return *fContext.fErrors; }
John Stilesdc8ec312021-01-11 11:05:21 -0500127
Ethan Nicholasba9a04f2020-11-06 09:28:04 -0500128 std::shared_ptr<SymbolTable>& symbolTable() {
129 return fSymbolTable;
130 }
131
132 void setSymbolTable(std::shared_ptr<SymbolTable>& symbolTable) {
133 fSymbolTable = symbolTable;
134 }
135
Ethan Nicholas371f6e12021-05-04 14:30:02 -0400136 static void CheckModifiers(const Context& context,
Ethan Nicholas89cfde12021-09-27 11:20:34 -0400137 int line,
Ethan Nicholas371f6e12021-05-04 14:30:02 -0400138 const Modifiers& modifiers,
139 int permittedModifierFlags,
140 int permittedLayoutFlags);
141
Ethan Nicholas89cfde12021-09-27 11:20:34 -0400142 std::unique_ptr<Expression> convertIdentifier(int line, skstd::string_view identifier);
Ethan Nicholas722cb672021-05-06 10:47:06 -0400143
Ethan Nicholas494eb3e2021-08-27 19:17:01 -0400144 bool haveRTAdjustInterfaceBlock() { return fRTAdjustInterfaceBlock != nullptr; }
145
146 int getRTAdjustFieldIndex() { return fRTAdjustFieldIndex; }
147
Ethan Nicholas86a43402017-01-19 13:32:00 -0500148 const Context& fContext;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500149
ethannicholasb3058bd2016-07-01 08:22:01 -0700150private:
Ethan Nicholas8b3dd342021-03-23 12:32:56 -0400151 void start(const ParsedModule& base,
152 bool isBuiltinCode,
Ethan Nicholas8b3dd342021-03-23 12:32:56 -0400153 std::vector<std::unique_ptr<ProgramElement>>* elements,
154 std::vector<const ProgramElement*>* sharedElements);
155
156 IRGenerator::IRBundle finish();
157
Ethan Nicholas89cfde12021-09-27 11:20:34 -0400158 void checkVarDeclaration(int line,
Brian Osmana654faa2021-02-26 11:52:59 -0500159 const Modifiers& modifiers,
160 const Type* baseType,
Ethan Nicholas489e5522021-01-20 10:53:11 -0500161 Variable::Storage storage);
Ethan Nicholas89cfde12021-09-27 11:20:34 -0400162 std::unique_ptr<Variable> convertVar(int line, const Modifiers& modifiers,
Ethan Nicholas962dec42021-06-10 13:06:39 -0400163 const Type* baseType, skstd::string_view name,
164 bool isArray, std::unique_ptr<Expression> arraySize,
Ethan Nicholasbd974002021-02-22 16:20:06 -0500165 Variable::Storage storage);
166 std::unique_ptr<Statement> convertVarDeclaration(std::unique_ptr<Variable> var,
Ethan Nicholasdd2fdea2021-07-20 15:23:04 -0400167 std::unique_ptr<Expression> value,
168 bool addToSymbolTable = true);
Ethan Nicholas89cfde12021-09-27 11:20:34 -0400169 std::unique_ptr<Statement> convertVarDeclaration(int line, const Modifiers& modifiers,
Ethan Nicholas962dec42021-06-10 13:06:39 -0400170 const Type* baseType, skstd::string_view name,
Ethan Nicholas489e5522021-01-20 10:53:11 -0500171 bool isArray,
172 std::unique_ptr<Expression> arraySize,
173 std::unique_ptr<Expression> value,
174 Variable::Storage storage);
John Stiles8f2a0cf2020-10-13 12:48:21 -0400175 StatementArray convertVarDeclarations(const ASTNode& decl, Variable::Storage storage);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400176 void convertFunction(const ASTNode& f);
177 std::unique_ptr<Statement> convertStatement(const ASTNode& statement);
178 std::unique_ptr<Expression> convertExpression(const ASTNode& expression);
179 std::unique_ptr<ModifiersDeclaration> convertModifiersDeclaration(const ASTNode& m);
Ethan Nicholas7da6dfa2017-06-21 11:25:18 -0400180
Brian Osmand8070392020-09-09 15:50:02 -0400181 const Type* convertType(const ASTNode& type, bool allowVoid = false);
Ethan Nicholas89cfde12021-09-27 11:20:34 -0400182 std::unique_ptr<Expression> call(int line,
Ethan Nicholas9ead3df2021-01-06 12:10:48 -0500183 std::unique_ptr<Expression> function,
184 ExpressionArray arguments);
Ethan Nicholas89cfde12021-09-27 11:20:34 -0400185 std::unique_ptr<Expression> call(int line,
Ethan Nicholas11d53972016-11-28 11:23:23 -0500186 const FunctionDeclaration& function,
John Stiles8e3b6be2020-10-13 11:14:08 -0400187 ExpressionArray arguments);
Brian Osman0acb5b52020-09-02 13:45:47 -0400188 CoercionCost callCost(const FunctionDeclaration& function,
Brian Osman3099f792021-09-01 13:12:16 -0400189 const ExpressionArray& arguments) const;
190 const FunctionDeclaration* findBestFunctionForCall(
191 const std::vector<const FunctionDeclaration*>& functions,
192 const ExpressionArray& arguments) const;
Ethan Nicholas9ead3df2021-01-06 12:10:48 -0500193 CoercionCost coercionCost(const Expression& expr, const Type& type);
Ethan Nicholas89cfde12021-09-27 11:20:34 -0400194 int convertArraySize(const Type& type, int line, const ASTNode& s);
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500195 bool containsConstantZero(Expression& expr);
John Stiles45990502021-02-16 10:55:27 -0500196 bool dividesByZero(Operator op, Expression& right);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400197 std::unique_ptr<Block> convertBlock(const ASTNode& block);
198 std::unique_ptr<Statement> convertBreak(const ASTNode& b);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400199 std::unique_ptr<Statement> convertContinue(const ASTNode& c);
200 std::unique_ptr<Statement> convertDiscard(const ASTNode& d);
201 std::unique_ptr<Statement> convertDo(const ASTNode& d);
202 std::unique_ptr<Statement> convertSwitch(const ASTNode& s);
203 std::unique_ptr<Expression> convertBinaryExpression(const ASTNode& expression);
Ethan Nicholas89cfde12021-09-27 11:20:34 -0400204 std::unique_ptr<Extension> convertExtension(int line, skstd::string_view name);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400205 std::unique_ptr<Statement> convertExpressionStatement(const ASTNode& s);
Ethan Nicholas9ead3df2021-01-06 12:10:48 -0500206 std::unique_ptr<Expression> convertField(std::unique_ptr<Expression> base,
Ethan Nicholas962dec42021-06-10 13:06:39 -0400207 skstd::string_view field);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400208 std::unique_ptr<Statement> convertFor(const ASTNode& f);
209 std::unique_ptr<Expression> convertIdentifier(const ASTNode& identifier);
210 std::unique_ptr<Statement> convertIf(const ASTNode& s);
Ethan Nicholas494eb3e2021-08-27 19:17:01 -0400211 void scanInterfaceBlock(SkSL::InterfaceBlock& intf);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400212 std::unique_ptr<InterfaceBlock> convertInterfaceBlock(const ASTNode& s);
Ethan Nicholas11d53972016-11-28 11:23:23 -0500213 Modifiers convertModifiers(const Modifiers& m);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400214 std::unique_ptr<Expression> convertPrefixExpression(const ASTNode& expression);
Ethan Nicholas89cfde12021-09-27 11:20:34 -0400215 std::unique_ptr<Statement> convertReturn(int line, std::unique_ptr<Expression> result);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400216 std::unique_ptr<Statement> convertReturn(const ASTNode& r);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400217 std::unique_ptr<Expression> convertCallExpression(const ASTNode& expression);
218 std::unique_ptr<Expression> convertFieldExpression(const ASTNode& expression);
219 std::unique_ptr<Expression> convertIndexExpression(const ASTNode& expression);
220 std::unique_ptr<Expression> convertPostfixExpression(const ASTNode& expression);
John Stilesdc75a972020-11-25 16:24:55 -0500221 std::unique_ptr<StructDefinition> convertStructDefinition(const ASTNode& expression);
Ethan Nicholasd2e09602021-06-10 11:21:59 -0400222 std::unique_ptr<Expression> convertSwizzle(std::unique_ptr<Expression> base,
Ethan Nicholas962dec42021-06-10 13:06:39 -0400223 skstd::string_view fields);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400224 std::unique_ptr<Expression> convertTernaryExpression(const ASTNode& expression);
225 std::unique_ptr<Statement> convertVarDeclarationStatement(const ASTNode& s);
226 std::unique_ptr<Statement> convertWhile(const ASTNode& w);
John Stiles7bd70332020-11-30 17:04:09 -0500227 void convertGlobalVarDeclarations(const ASTNode& decl);
John Stiles3b204892021-08-27 17:35:35 -0400228 /** Appends sk_Position fixup to the bottom of main() if this is a vertex program. */
229 void appendRTAdjustFixupToVertexMain(const FunctionDeclaration& decl, Block* body);
ethannicholasb3058bd2016-07-01 08:22:01 -0700230
John Stiles403a3632020-08-20 12:11:48 -0400231 bool setRefKind(Expression& expr, VariableReference::RefKind kind);
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400232 void copyIntrinsicIfNeeded(const FunctionDeclaration& function);
Brian Osman9496fe52020-11-18 14:48:19 -0500233 void findAndDeclareBuiltinVariables();
ethannicholasb3058bd2016-07-01 08:22:01 -0700234
Brian Osman818fd6d2020-12-30 15:06:22 -0500235 // Runtime effects (and the interpreter, which uses the same CPU runtime) require adherence to
236 // the strict rules from The OpenGL ES Shading Language Version 1.00. (Including Appendix A).
237 bool strictES2Mode() const {
John Stilesca107c92021-02-19 09:54:44 -0500238 return fContext.fConfig->strictES2Mode();
Brian Osman818fd6d2020-12-30 15:06:22 -0500239 }
240
John Stilesbb2ef922021-07-26 08:32:07 -0400241 bool isRuntimeEffect() const {
John Stilesaddccaf2021-08-02 19:03:30 -0400242 return ProgramConfig::IsRuntimeEffect(fContext.fConfig->fKind);
John Stilesbb2ef922021-07-26 08:32:07 -0400243 }
244
John Stilesc1a98b82021-02-24 13:35:02 -0500245 const ShaderCapsClass& caps() const {
246 return fContext.fCaps;
247 }
248
John Stilesf2872e62021-05-04 11:38:43 -0400249 ModifiersPool& modifiersPool() const {
John Stiles10d39d92021-05-04 16:13:14 -0400250 return *fContext.fModifiersPool;
John Stilesf2872e62021-05-04 11:38:43 -0400251 }
252
Brian Osman88cda172020-10-09 12:05:16 -0400253 Program::Inputs fInputs;
Brian Osman88cda172020-10-09 12:05:16 -0400254
Ethan Nicholasfc994162019-06-06 10:04:27 -0400255 std::unique_ptr<ASTFile> fFile;
John Stilesdbd4e6f2021-02-16 13:29:15 -0500256
Brian Osman88cda172020-10-09 12:05:16 -0400257 std::shared_ptr<SymbolTable> fSymbolTable = nullptr;
John Stiles810c8cf2020-08-26 19:46:27 -0400258 // Symbols which have definitions in the include files.
259 IRIntrinsicMap* fIntrinsics = nullptr;
Brian Osman02bc5222021-01-28 11:00:20 -0500260 std::unordered_set<const Type*> fDefinedStructs;
Brian Osman88cda172020-10-09 12:05:16 -0400261 std::vector<std::unique_ptr<ProgramElement>>* fProgramElements = nullptr;
Brian Osman133724c2020-10-28 14:14:39 -0400262 std::vector<const ProgramElement*>* fSharedElements = nullptr;
Brian Osman88cda172020-10-09 12:05:16 -0400263 const Variable* fRTAdjust = nullptr;
264 const Variable* fRTAdjustInterfaceBlock = nullptr;
Robert Phillipsfe8da172018-01-24 14:52:02 +0000265 int fRTAdjustFieldIndex;
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400266 // true if we are currently processing one of the built-in SkSL include files
Ethan Nicholasba9a04f2020-11-06 09:28:04 -0500267 bool fIsBuiltinCode = false;
ethannicholasb3058bd2016-07-01 08:22:01 -0700268
269 friend class AutoSymbolTable;
ethannicholas22f939e2016-10-13 13:25:34 -0700270 friend class AutoLoopLevel;
Ethan Nicholasaf197692017-02-27 13:26:45 -0500271 friend class AutoSwitchLevel;
John Stilesd1c4dac2020-08-11 18:50:50 -0400272 friend class AutoDisableInline;
ethannicholasb3058bd2016-07-01 08:22:01 -0700273 friend class Compiler;
Ethan Nicholasdd2fdea2021-07-20 15:23:04 -0400274 friend class DSLParser;
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500275 friend class dsl::DSLCore;
Ethan Nicholas1ff76092021-01-28 10:02:43 -0500276 friend class dsl::DSLFunction;
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500277 friend class dsl::DSLVar;
Ethan Nicholas95046142021-01-07 10:57:27 -0500278 friend class dsl::DSLWriter;
ethannicholasb3058bd2016-07-01 08:22:01 -0700279};
280
John Stilesa6841be2020-08-06 14:11:56 -0400281} // namespace SkSL
ethannicholasb3058bd2016-07-01 08:22:01 -0700282
283#endif