blob: 39561c25a6c2a2dd98f71a48e85609d4e9072b6c [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 Nicholas95046142021-01-07 10:57:27 -050035namespace dsl {
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -050036 class DSLCore;
Ethan Nicholas1ff76092021-01-28 10:02:43 -050037 class DSLFunction;
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -050038 class DSLVar;
Ethan Nicholas95046142021-01-07 10:57:27 -050039 class DSLWriter;
40}
41
Brian Osmanbe0b3b72021-01-06 14:27:35 -050042class ExternalFunction;
Ethan Nicholas95046142021-01-07 10:57:27 -050043class ExternalValue;
Ethan Nicholas1e9f7f32020-10-08 05:28:32 -040044class FunctionCall;
John Stilesdc75a972020-11-25 16:24:55 -050045class StructDefinition;
Brian Osman3d87e9f2020-10-08 11:50:22 -040046struct ParsedModule;
Ethan Nicholascb0f4092019-04-19 11:26:50 -040047struct Swizzle;
48
ethannicholasb3058bd2016-07-01 08:22:01 -070049/**
John Stiles810c8cf2020-08-26 19:46:27 -040050 * Intrinsics are passed between the Compiler and the IRGenerator using IRIntrinsicMaps.
51 */
Brian Osman2b469eb2020-09-21 11:32:10 -040052class IRIntrinsicMap {
53public:
54 IRIntrinsicMap(IRIntrinsicMap* parent) : fParent(parent) {}
55
56 void insertOrDie(String key, std::unique_ptr<ProgramElement> element) {
57 SkASSERT(fIntrinsics.find(key) == fIntrinsics.end());
58 fIntrinsics[key] = Intrinsic{std::move(element), false};
59 }
60
Brian Osmanafa18ee2020-10-07 17:47:45 -040061 const ProgramElement* find(const String& key) {
62 auto iter = fIntrinsics.find(key);
63 if (iter == fIntrinsics.end()) {
64 return fParent ? fParent->find(key) : nullptr;
65 }
66 return iter->second.fIntrinsic.get();
67 }
68
Brian Osman2b469eb2020-09-21 11:32:10 -040069 // Only returns an intrinsic that isn't already marked as included, and then marks it.
Brian Osman00a8b5b2020-10-02 09:06:04 -040070 const ProgramElement* findAndInclude(const String& key) {
Brian Osman2b469eb2020-09-21 11:32:10 -040071 auto iter = fIntrinsics.find(key);
72 if (iter == fIntrinsics.end()) {
73 return fParent ? fParent->findAndInclude(key) : nullptr;
74 }
75 if (iter->second.fAlreadyIncluded) {
76 return nullptr;
77 }
78 iter->second.fAlreadyIncluded = true;
79 return iter->second.fIntrinsic.get();
80 }
81
82 void resetAlreadyIncluded() {
83 for (auto& pair : fIntrinsics) {
84 pair.second.fAlreadyIncluded = false;
85 }
86 if (fParent) {
87 fParent->resetAlreadyIncluded();
88 }
89 }
90
91private:
92 struct Intrinsic {
93 std::unique_ptr<ProgramElement> fIntrinsic;
94 bool fAlreadyIncluded = false;
95 };
96
97 std::unordered_map<String, Intrinsic> fIntrinsics;
98 IRIntrinsicMap* fParent = nullptr;
John Stiles810c8cf2020-08-26 19:46:27 -040099};
John Stiles810c8cf2020-08-26 19:46:27 -0400100
101/**
Ethan Nicholas11d53972016-11-28 11:23:23 -0500102 * Performs semantic analysis on an abstract syntax tree (AST) and produces the corresponding
ethannicholasb3058bd2016-07-01 08:22:01 -0700103 * (unoptimized) intermediate representation (IR).
104 */
105class IRGenerator {
106public:
Brian Osman0006ad02020-11-18 15:38:39 -0500107 IRGenerator(const Context* context,
John Stilesb30151e2021-01-11 16:13:08 -0500108 const ShaderCapsClass* caps);
ethannicholasb3058bd2016-07-01 08:22:01 -0700109
Brian Osman88cda172020-10-09 12:05:16 -0400110 struct IRBundle {
111 std::vector<std::unique_ptr<ProgramElement>> fElements;
Brian Osman133724c2020-10-28 14:14:39 -0400112 std::vector<const ProgramElement*> fSharedElements;
Brian Osman88cda172020-10-09 12:05:16 -0400113 std::unique_ptr<ModifiersPool> fModifiers;
114 std::shared_ptr<SymbolTable> fSymbolTable;
115 Program::Inputs fInputs;
116 };
117
118 /**
Brian Osmanbe0b3b72021-01-06 14:27:35 -0500119 * If externalFuncs is supplied, those values are registered in the symbol table of the
Brian Osman88cda172020-10-09 12:05:16 -0400120 * Program, but ownership is *not* transferred. It is up to the caller to keep them alive.
121 */
Brian Osmanbe0b3b72021-01-06 14:27:35 -0500122 IRBundle convertProgram(
123 Program::Kind kind,
124 const Program::Settings* settings,
125 const ParsedModule& base,
126 bool isBuiltinCode,
127 const char* text,
128 size_t length,
129 const std::vector<std::unique_ptr<ExternalFunction>>* externalFunctions);
ethannicholasb3058bd2016-07-01 08:22:01 -0700130
Ethan Nicholas01ec7e82020-10-08 12:10:12 -0400131 // both of these functions return null and report an error if the setting does not exist
132 const Type* typeForSetting(int offset, String name) const;
133 std::unique_ptr<Expression> valueForSetting(int offset, String name) const;
134
Brian Osman88cda172020-10-09 12:05:16 -0400135 const Program::Settings* settings() const { return fSettings; }
136
John Stilesb30151e2021-01-11 16:13:08 -0500137 ErrorReporter& errorReporter() const { return fContext.fErrors; }
John Stilesdc8ec312021-01-11 11:05:21 -0500138
Ethan Nicholasba9a04f2020-11-06 09:28:04 -0500139 std::shared_ptr<SymbolTable>& symbolTable() {
140 return fSymbolTable;
141 }
142
143 void setSymbolTable(std::shared_ptr<SymbolTable>& symbolTable) {
144 fSymbolTable = symbolTable;
145 }
146
147 void pushSymbolTable();
148 void popSymbolTable();
149
Ethan Nicholas86a43402017-01-19 13:32:00 -0500150 const Context& fContext;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500151
ethannicholasb3058bd2016-07-01 08:22:01 -0700152private:
Ethan Nicholas3605ace2016-11-21 15:59:48 -0500153 /**
Ethan Nicholas041fd0a2020-10-07 16:42:04 -0400154 * Relinquishes ownership of the Modifiers that have been collected so far and returns them.
155 */
156 std::unique_ptr<ModifiersPool> releaseModifiers();
157
Ethan Nicholas63d7ee32020-08-17 10:57:12 -0400158 void checkModifiers(int offset, const Modifiers& modifiers, int permitted);
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500159 void checkVarDeclaration(int offset, const Modifiers& modifiers, const Type* baseType,
Ethan Nicholas489e5522021-01-20 10:53:11 -0500160 Variable::Storage storage);
161 std::unique_ptr<Statement> convertVarDeclaration(int offset, const Modifiers& modifiers,
162 const Type* baseType, StringFragment name,
163 bool isArray,
164 std::unique_ptr<Expression> arraySize,
165 std::unique_ptr<Expression> value,
166 Variable::Storage storage);
John Stiles8f2a0cf2020-10-13 12:48:21 -0400167 StatementArray convertVarDeclarations(const ASTNode& decl, Variable::Storage storage);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400168 void convertFunction(const ASTNode& f);
Ethan Nicholas70728ef2020-05-28 07:09:00 -0400169 std::unique_ptr<Statement> convertSingleStatement(const ASTNode& statement);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400170 std::unique_ptr<Statement> convertStatement(const ASTNode& statement);
171 std::unique_ptr<Expression> convertExpression(const ASTNode& expression);
172 std::unique_ptr<ModifiersDeclaration> convertModifiersDeclaration(const ASTNode& m);
Ethan Nicholas7da6dfa2017-06-21 11:25:18 -0400173
Brian Osmand8070392020-09-09 15:50:02 -0400174 const Type* convertType(const ASTNode& type, bool allowVoid = false);
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700175 std::unique_ptr<Expression> call(int offset,
Ethan Nicholas9ead3df2021-01-06 12:10:48 -0500176 std::unique_ptr<Expression> function,
177 ExpressionArray arguments);
178 std::unique_ptr<Expression> call(int offset,
Ethan Nicholas11d53972016-11-28 11:23:23 -0500179 const FunctionDeclaration& function,
John Stiles8e3b6be2020-10-13 11:14:08 -0400180 ExpressionArray arguments);
Brian Osman0acb5b52020-09-02 13:45:47 -0400181 CoercionCost callCost(const FunctionDeclaration& function,
John Stiles8e3b6be2020-10-13 11:14:08 -0400182 const ExpressionArray& arguments);
Ethan Nicholasdcd2f862020-12-17 23:24:25 +0000183 std::unique_ptr<Expression> coerce(std::unique_ptr<Expression> expr, const Type& type);
Ethan Nicholas9ead3df2021-01-06 12:10:48 -0500184 CoercionCost coercionCost(const Expression& expr, const Type& type);
John Stiles80b02af2021-02-12 17:07:51 -0500185 int convertArraySize(const Type& type, int offset, const ASTNode& s);
186 int convertArraySize(const Type& type, std::unique_ptr<Expression> s);
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500187 bool containsConstantZero(Expression& expr);
188 bool dividesByZero(Token::Kind op, Expression& right);
Ethan Nicholas9ead3df2021-01-06 12:10:48 -0500189 std::unique_ptr<Expression> convertBinaryExpression(std::unique_ptr<Expression> left,
190 Token::Kind op,
191 std::unique_ptr<Expression> right);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400192 std::unique_ptr<Block> convertBlock(const ASTNode& block);
193 std::unique_ptr<Statement> convertBreak(const ASTNode& b);
John Stiles248f57b2021-02-03 15:11:18 -0500194 std::unique_ptr<Expression> convertArrayConstructor(int offset,
195 const Type& type,
196 ExpressionArray args);
John Stiles53f0ddf2021-01-05 18:47:09 -0500197 std::unique_ptr<Expression> convertScalarConstructor(int offset,
John Stiles8e3b6be2020-10-13 11:14:08 -0400198 const Type& type,
199 ExpressionArray params);
200 std::unique_ptr<Expression> convertCompoundConstructor(int offset,
201 const Type& type,
202 ExpressionArray params);
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700203 std::unique_ptr<Expression> convertConstructor(int offset,
Ethan Nicholas11d53972016-11-28 11:23:23 -0500204 const Type& type,
John Stiles8e3b6be2020-10-13 11:14:08 -0400205 ExpressionArray params);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400206 std::unique_ptr<Statement> convertContinue(const ASTNode& c);
207 std::unique_ptr<Statement> convertDiscard(const ASTNode& d);
Ethan Nicholas2ed0d942021-01-20 07:51:23 -0500208 std::unique_ptr<Statement> convertDo(std::unique_ptr<Statement> stmt,
209 std::unique_ptr<Expression> test);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400210 std::unique_ptr<Statement> convertDo(const ASTNode& d);
Ethan Nicholascfefec02021-02-09 15:22:57 -0500211 std::unique_ptr<Statement> convertSwitch(int offset,
212 bool isStatic,
213 std::unique_ptr<Expression> value,
214 ExpressionArray caseValues,
215 SkTArray<StatementArray> caseStatements,
216 std::shared_ptr<SymbolTable> symbolTable);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400217 std::unique_ptr<Statement> convertSwitch(const ASTNode& s);
218 std::unique_ptr<Expression> convertBinaryExpression(const ASTNode& expression);
219 std::unique_ptr<Extension> convertExtension(int offset, StringFragment name);
220 std::unique_ptr<Statement> convertExpressionStatement(const ASTNode& s);
Ethan Nicholas9ead3df2021-01-06 12:10:48 -0500221 std::unique_ptr<Expression> convertField(std::unique_ptr<Expression> base,
222 StringFragment field);
Ethan Nicholas2ed0d942021-01-20 07:51:23 -0500223 std::unique_ptr<Statement> convertFor(int offset,
224 std::unique_ptr<Statement> initializer,
225 std::unique_ptr<Expression> test,
226 std::unique_ptr<Expression> next,
227 std::unique_ptr<Statement> statement);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400228 std::unique_ptr<Statement> convertFor(const ASTNode& f);
Ethan Nicholas9ead3df2021-01-06 12:10:48 -0500229 std::unique_ptr<Expression> convertIdentifier(int offset, StringFragment identifier);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400230 std::unique_ptr<Expression> convertIdentifier(const ASTNode& identifier);
231 std::unique_ptr<Statement> convertIf(const ASTNode& s);
John Stiles5ede6e32021-01-11 13:00:49 -0500232 std::unique_ptr<Statement> convertIf(int offset, bool isStatic,
233 std::unique_ptr<Expression> test,
234 std::unique_ptr<Statement> ifTrue,
235 std::unique_ptr<Statement> ifFalse);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400236 std::unique_ptr<InterfaceBlock> convertInterfaceBlock(const ASTNode& s);
Ethan Nicholas11d53972016-11-28 11:23:23 -0500237 Modifiers convertModifiers(const Modifiers& m);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400238 std::unique_ptr<Expression> convertPrefixExpression(const ASTNode& expression);
Ethan Nicholas1ff76092021-01-28 10:02:43 -0500239 std::unique_ptr<Statement> convertReturn(int offset, std::unique_ptr<Expression> result);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400240 std::unique_ptr<Statement> convertReturn(const ASTNode& r);
241 std::unique_ptr<Section> convertSection(const ASTNode& e);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400242 std::unique_ptr<Expression> convertCallExpression(const ASTNode& expression);
243 std::unique_ptr<Expression> convertFieldExpression(const ASTNode& expression);
244 std::unique_ptr<Expression> convertIndexExpression(const ASTNode& expression);
John Stiles1b27c3d2020-12-07 12:14:55 -0500245 std::unique_ptr<Expression> convertIndex(std::unique_ptr<Expression> base,
Ethan Nicholas4d2bbbb2021-01-12 11:56:23 -0500246 std::unique_ptr<Expression> index);
Ethan Nicholas9ead3df2021-01-06 12:10:48 -0500247 std::unique_ptr<Expression> convertPostfixExpression(std::unique_ptr<Expression> base,
248 Token::Kind op);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400249 std::unique_ptr<Expression> convertPostfixExpression(const ASTNode& expression);
Ethan Nicholas9ead3df2021-01-06 12:10:48 -0500250 std::unique_ptr<Expression> convertPrefixExpression(Token::Kind op,
251 std::unique_ptr<Expression> base);
Brian Osman6518d772020-09-10 16:50:06 -0400252 std::unique_ptr<Expression> convertScopeExpression(const ASTNode& expression);
John Stilesdc75a972020-11-25 16:24:55 -0500253 std::unique_ptr<StructDefinition> convertStructDefinition(const ASTNode& expression);
Ethan Nicholasaae47c82017-11-10 15:34:03 -0500254 std::unique_ptr<Expression> convertTypeField(int offset, const Type& type,
255 StringFragment field);
Ethan Nicholas9ead3df2021-01-06 12:10:48 -0500256 std::unique_ptr<Expression> convertSwizzle(std::unique_ptr<Expression> base, String fields);
257 std::unique_ptr<Expression> convertTernaryExpression(std::unique_ptr<Expression> test,
258 std::unique_ptr<Expression> ifTrue,
259 std::unique_ptr<Expression> ifFalse);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400260 std::unique_ptr<Expression> convertTernaryExpression(const ASTNode& expression);
261 std::unique_ptr<Statement> convertVarDeclarationStatement(const ASTNode& s);
Ethan Nicholas9ead3df2021-01-06 12:10:48 -0500262 std::unique_ptr<Statement> convertWhile(int offset, std::unique_ptr<Expression> test,
263 std::unique_ptr<Statement> statement);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400264 std::unique_ptr<Statement> convertWhile(const ASTNode& w);
John Stiles7bd70332020-11-30 17:04:09 -0500265 void convertGlobalVarDeclarations(const ASTNode& decl);
Ethan Nicholasfc994162019-06-06 10:04:27 -0400266 void convertEnum(const ASTNode& e);
Ethan Nicholasaae47c82017-11-10 15:34:03 -0500267 std::unique_ptr<Block> applyInvocationIDWorkaround(std::unique_ptr<Block> main);
Robert Phillipsfe8da172018-01-24 14:52:02 +0000268 // returns a statement which converts sk_Position from device to normalized coordinates
269 std::unique_ptr<Statement> getNormalizeSkPositionCode();
ethannicholasb3058bd2016-07-01 08:22:01 -0700270
271 void checkValid(const Expression& expr);
John Stilesb4b627e2020-11-13 15:55:27 -0500272 bool typeContainsPrivateFields(const Type& type);
John Stiles403a3632020-08-20 12:11:48 -0400273 bool setRefKind(Expression& expr, VariableReference::RefKind kind);
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400274 void copyIntrinsicIfNeeded(const FunctionDeclaration& function);
Brian Osman9496fe52020-11-18 14:48:19 -0500275 void findAndDeclareBuiltinVariables();
John Stilese3a91cf2021-01-26 10:13:58 -0500276 bool detectVarDeclarationWithoutScope(const Statement& stmt);
Ethan Nicholase2c05042021-02-03 10:27:22 -0500277 // Coerces returns to correct type and detects invalid break / continue placement
278 void finalizeFunction(FunctionDefinition& f);
ethannicholasb3058bd2016-07-01 08:22:01 -0700279
Brian Osman818fd6d2020-12-30 15:06:22 -0500280 // Runtime effects (and the interpreter, which uses the same CPU runtime) require adherence to
281 // the strict rules from The OpenGL ES Shading Language Version 1.00. (Including Appendix A).
282 bool strictES2Mode() const {
283 return fKind == Program::kRuntimeEffect_Kind || fKind == Program::kGeneric_Kind;
284 }
285
Brian Osman88cda172020-10-09 12:05:16 -0400286 Program::Inputs fInputs;
287 const Program::Settings* fSettings = nullptr;
Brian Osmand7e76592020-11-02 12:26:22 -0500288 const ShaderCapsClass* fCaps = nullptr;
Brian Osman88cda172020-10-09 12:05:16 -0400289 Program::Kind fKind;
290
Ethan Nicholasfc994162019-06-06 10:04:27 -0400291 std::unique_ptr<ASTFile> fFile;
Ethan Nicholas762466e2017-06-29 10:03:38 -0400292 std::unordered_map<String, Program::Settings::Value> fCapsMap;
Brian Osman88cda172020-10-09 12:05:16 -0400293 std::shared_ptr<SymbolTable> fSymbolTable = nullptr;
Ethan Nicholas70728ef2020-05-28 07:09:00 -0400294 // additional statements that need to be inserted before the one that convertStatement is
295 // currently working on
John Stiles8f2a0cf2020-10-13 12:48:21 -0400296 StatementArray fExtraStatements;
John Stiles810c8cf2020-08-26 19:46:27 -0400297 // Symbols which have definitions in the include files.
298 IRIntrinsicMap* fIntrinsics = nullptr;
John Stilesb8e010c2020-08-11 18:05:39 -0400299 std::unordered_set<const FunctionDeclaration*> fReferencedIntrinsics;
Ethan Nicholas7da6dfa2017-06-21 11:25:18 -0400300 int fInvocations;
Brian Osman02bc5222021-01-28 11:00:20 -0500301 std::unordered_set<const Type*> fDefinedStructs;
Brian Osman88cda172020-10-09 12:05:16 -0400302 std::vector<std::unique_ptr<ProgramElement>>* fProgramElements = nullptr;
Brian Osman133724c2020-10-28 14:14:39 -0400303 std::vector<const ProgramElement*>* fSharedElements = nullptr;
Brian Osman88cda172020-10-09 12:05:16 -0400304 const Variable* fRTAdjust = nullptr;
305 const Variable* fRTAdjustInterfaceBlock = nullptr;
Robert Phillipsfe8da172018-01-24 14:52:02 +0000306 int fRTAdjustFieldIndex;
John Stiles881a10c2020-09-19 10:13:24 -0400307 bool fCanInline = true;
Ethan Nicholasc18bb512020-07-28 14:46:53 -0400308 // true if we are currently processing one of the built-in SkSL include files
Ethan Nicholasba9a04f2020-11-06 09:28:04 -0500309 bool fIsBuiltinCode = false;
Ethan Nicholas041fd0a2020-10-07 16:42:04 -0400310 std::unique_ptr<ModifiersPool> fModifiers;
ethannicholasb3058bd2016-07-01 08:22:01 -0700311
312 friend class AutoSymbolTable;
ethannicholas22f939e2016-10-13 13:25:34 -0700313 friend class AutoLoopLevel;
Ethan Nicholasaf197692017-02-27 13:26:45 -0500314 friend class AutoSwitchLevel;
John Stilesd1c4dac2020-08-11 18:50:50 -0400315 friend class AutoDisableInline;
ethannicholasb3058bd2016-07-01 08:22:01 -0700316 friend class Compiler;
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500317 friend class dsl::DSLCore;
Ethan Nicholas1ff76092021-01-28 10:02:43 -0500318 friend class dsl::DSLFunction;
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500319 friend class dsl::DSLVar;
Ethan Nicholas95046142021-01-07 10:57:27 -0500320 friend class dsl::DSLWriter;
ethannicholasb3058bd2016-07-01 08:22:01 -0700321};
322
John Stilesa6841be2020-08-06 14:11:56 -0400323} // namespace SkSL
ethannicholasb3058bd2016-07-01 08:22:01 -0700324
325#endif