xref: /aosp_15_r20/external/skia/src/sksl/ir/SkSLVariable.h (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker  * Copyright 2016 Google Inc.
3*c8dee2aaSAndroid Build Coastguard Worker  *
4*c8dee2aaSAndroid Build Coastguard Worker  * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker  * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker  */
7*c8dee2aaSAndroid Build Coastguard Worker 
8*c8dee2aaSAndroid Build Coastguard Worker #ifndef SKSL_VARIABLE
9*c8dee2aaSAndroid Build Coastguard Worker #define SKSL_VARIABLE
10*c8dee2aaSAndroid Build Coastguard Worker 
11*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkTypes.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLPosition.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLIRNode.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLLayout.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLModifierFlags.h"
16*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLStatement.h"
17*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLSymbol.h"
18*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLType.h"
19*c8dee2aaSAndroid Build Coastguard Worker 
20*c8dee2aaSAndroid Build Coastguard Worker #include <cstdint>
21*c8dee2aaSAndroid Build Coastguard Worker #include <memory>
22*c8dee2aaSAndroid Build Coastguard Worker #include <string>
23*c8dee2aaSAndroid Build Coastguard Worker #include <string_view>
24*c8dee2aaSAndroid Build Coastguard Worker #include <utility>
25*c8dee2aaSAndroid Build Coastguard Worker 
26*c8dee2aaSAndroid Build Coastguard Worker namespace SkSL {
27*c8dee2aaSAndroid Build Coastguard Worker 
28*c8dee2aaSAndroid Build Coastguard Worker class Context;
29*c8dee2aaSAndroid Build Coastguard Worker class Expression;
30*c8dee2aaSAndroid Build Coastguard Worker class GlobalVarDeclaration;
31*c8dee2aaSAndroid Build Coastguard Worker class InterfaceBlock;
32*c8dee2aaSAndroid Build Coastguard Worker class Mangler;
33*c8dee2aaSAndroid Build Coastguard Worker class SymbolTable;
34*c8dee2aaSAndroid Build Coastguard Worker class VarDeclaration;
35*c8dee2aaSAndroid Build Coastguard Worker 
36*c8dee2aaSAndroid Build Coastguard Worker enum class VariableStorage : int8_t {
37*c8dee2aaSAndroid Build Coastguard Worker     kGlobal,
38*c8dee2aaSAndroid Build Coastguard Worker     kInterfaceBlock,
39*c8dee2aaSAndroid Build Coastguard Worker     kLocal,
40*c8dee2aaSAndroid Build Coastguard Worker     kParameter,
41*c8dee2aaSAndroid Build Coastguard Worker };
42*c8dee2aaSAndroid Build Coastguard Worker 
43*c8dee2aaSAndroid Build Coastguard Worker /**
44*c8dee2aaSAndroid Build Coastguard Worker  * Represents a variable, whether local, global, or a function parameter. This represents the
45*c8dee2aaSAndroid Build Coastguard Worker  * variable itself (the storage location), which is shared between all VariableReferences which
46*c8dee2aaSAndroid Build Coastguard Worker  * read or write that storage location.
47*c8dee2aaSAndroid Build Coastguard Worker  */
48*c8dee2aaSAndroid Build Coastguard Worker class Variable : public Symbol {
49*c8dee2aaSAndroid Build Coastguard Worker public:
50*c8dee2aaSAndroid Build Coastguard Worker     using Storage = VariableStorage;
51*c8dee2aaSAndroid Build Coastguard Worker 
52*c8dee2aaSAndroid Build Coastguard Worker     inline static constexpr Kind kIRNodeKind = Kind::kVariable;
53*c8dee2aaSAndroid Build Coastguard Worker 
Variable(Position pos,Position modifiersPosition,ModifierFlags modifierFlags,std::string_view name,const Type * type,bool builtin,Storage storage)54*c8dee2aaSAndroid Build Coastguard Worker     Variable(Position pos, Position modifiersPosition, ModifierFlags modifierFlags,
55*c8dee2aaSAndroid Build Coastguard Worker              std::string_view name, const Type* type, bool builtin, Storage storage)
56*c8dee2aaSAndroid Build Coastguard Worker             : INHERITED(pos, kIRNodeKind, name, type)
57*c8dee2aaSAndroid Build Coastguard Worker             , fModifierFlags(modifierFlags)
58*c8dee2aaSAndroid Build Coastguard Worker             , fModifiersPosition(modifiersPosition)
59*c8dee2aaSAndroid Build Coastguard Worker             , fStorage(storage)
60*c8dee2aaSAndroid Build Coastguard Worker             , fBuiltin(builtin) {}
61*c8dee2aaSAndroid Build Coastguard Worker 
62*c8dee2aaSAndroid Build Coastguard Worker     ~Variable() override;
63*c8dee2aaSAndroid Build Coastguard Worker 
64*c8dee2aaSAndroid Build Coastguard Worker     static std::unique_ptr<Variable> Convert(const Context& context, Position pos,
65*c8dee2aaSAndroid Build Coastguard Worker                                              Position modifiersPos, const Layout& layout,
66*c8dee2aaSAndroid Build Coastguard Worker                                              ModifierFlags flags, const Type* type,
67*c8dee2aaSAndroid Build Coastguard Worker                                              Position namePos, std::string_view name,
68*c8dee2aaSAndroid Build Coastguard Worker                                              Storage storage);
69*c8dee2aaSAndroid Build Coastguard Worker 
70*c8dee2aaSAndroid Build Coastguard Worker     static std::unique_ptr<Variable> Make(Position pos, Position modifiersPosition,
71*c8dee2aaSAndroid Build Coastguard Worker                                           const Layout& layout, ModifierFlags flags,
72*c8dee2aaSAndroid Build Coastguard Worker                                           const Type* type, std::string_view name,
73*c8dee2aaSAndroid Build Coastguard Worker                                           std::string mangledName, bool builtin, Storage storage);
74*c8dee2aaSAndroid Build Coastguard Worker 
75*c8dee2aaSAndroid Build Coastguard Worker     /**
76*c8dee2aaSAndroid Build Coastguard Worker      * Creates a local scratch variable and the associated VarDeclaration statement.
77*c8dee2aaSAndroid Build Coastguard Worker      * Useful when doing IR rewrites, e.g. inlining a function call.
78*c8dee2aaSAndroid Build Coastguard Worker      */
79*c8dee2aaSAndroid Build Coastguard Worker     struct ScratchVariable {
80*c8dee2aaSAndroid Build Coastguard Worker         const Variable* fVarSymbol;
81*c8dee2aaSAndroid Build Coastguard Worker         std::unique_ptr<Statement> fVarDecl;
82*c8dee2aaSAndroid Build Coastguard Worker     };
83*c8dee2aaSAndroid Build Coastguard Worker     static ScratchVariable MakeScratchVariable(const Context& context,
84*c8dee2aaSAndroid Build Coastguard Worker                                                Mangler& mangler,
85*c8dee2aaSAndroid Build Coastguard Worker                                                std::string_view baseName,
86*c8dee2aaSAndroid Build Coastguard Worker                                                const Type* type,
87*c8dee2aaSAndroid Build Coastguard Worker                                                SymbolTable* symbolTable,
88*c8dee2aaSAndroid Build Coastguard Worker                                                std::unique_ptr<Expression> initialValue);
modifierFlags()89*c8dee2aaSAndroid Build Coastguard Worker     ModifierFlags modifierFlags() const {
90*c8dee2aaSAndroid Build Coastguard Worker         return fModifierFlags;
91*c8dee2aaSAndroid Build Coastguard Worker     }
92*c8dee2aaSAndroid Build Coastguard Worker 
93*c8dee2aaSAndroid Build Coastguard Worker     virtual const Layout& layout() const;
94*c8dee2aaSAndroid Build Coastguard Worker 
modifiersPosition()95*c8dee2aaSAndroid Build Coastguard Worker     Position modifiersPosition() const {
96*c8dee2aaSAndroid Build Coastguard Worker         return fModifiersPosition;
97*c8dee2aaSAndroid Build Coastguard Worker     }
98*c8dee2aaSAndroid Build Coastguard Worker 
isBuiltin()99*c8dee2aaSAndroid Build Coastguard Worker     bool isBuiltin() const {
100*c8dee2aaSAndroid Build Coastguard Worker         return fBuiltin;
101*c8dee2aaSAndroid Build Coastguard Worker     }
102*c8dee2aaSAndroid Build Coastguard Worker 
storage()103*c8dee2aaSAndroid Build Coastguard Worker     Storage storage() const {
104*c8dee2aaSAndroid Build Coastguard Worker         return fStorage;
105*c8dee2aaSAndroid Build Coastguard Worker     }
106*c8dee2aaSAndroid Build Coastguard Worker 
107*c8dee2aaSAndroid Build Coastguard Worker     const Expression* initialValue() const;
108*c8dee2aaSAndroid Build Coastguard Worker 
109*c8dee2aaSAndroid Build Coastguard Worker     VarDeclaration* varDeclaration() const;
110*c8dee2aaSAndroid Build Coastguard Worker 
111*c8dee2aaSAndroid Build Coastguard Worker     void setVarDeclaration(VarDeclaration* declaration);
112*c8dee2aaSAndroid Build Coastguard Worker 
113*c8dee2aaSAndroid Build Coastguard Worker     GlobalVarDeclaration* globalVarDeclaration() const;
114*c8dee2aaSAndroid Build Coastguard Worker 
115*c8dee2aaSAndroid Build Coastguard Worker     void setGlobalVarDeclaration(GlobalVarDeclaration* global);
116*c8dee2aaSAndroid Build Coastguard Worker 
detachDeadVarDeclaration()117*c8dee2aaSAndroid Build Coastguard Worker     void detachDeadVarDeclaration() {
118*c8dee2aaSAndroid Build Coastguard Worker         // The VarDeclaration is being deleted, so our reference to it has become stale.
119*c8dee2aaSAndroid Build Coastguard Worker         fDeclaringElement = nullptr;
120*c8dee2aaSAndroid Build Coastguard Worker     }
121*c8dee2aaSAndroid Build Coastguard Worker 
122*c8dee2aaSAndroid Build Coastguard Worker     // The interfaceBlock methods are no-op stubs here. They have proper implementations in
123*c8dee2aaSAndroid Build Coastguard Worker     // ExtendedVariable, declared below this class, which dedicates extra space to store the pointer
124*c8dee2aaSAndroid Build Coastguard Worker     // back to the InterfaceBlock.
interfaceBlock()125*c8dee2aaSAndroid Build Coastguard Worker     virtual InterfaceBlock* interfaceBlock() const { return nullptr; }
126*c8dee2aaSAndroid Build Coastguard Worker 
setInterfaceBlock(InterfaceBlock *)127*c8dee2aaSAndroid Build Coastguard Worker     virtual void setInterfaceBlock(InterfaceBlock*) { SkUNREACHABLE; }
128*c8dee2aaSAndroid Build Coastguard Worker 
detachDeadInterfaceBlock()129*c8dee2aaSAndroid Build Coastguard Worker     virtual void detachDeadInterfaceBlock() {}
130*c8dee2aaSAndroid Build Coastguard Worker 
131*c8dee2aaSAndroid Build Coastguard Worker     // Only ExtendedVariables support mangled names.
mangledName()132*c8dee2aaSAndroid Build Coastguard Worker     virtual std::string_view mangledName() const { return this->name(); }
133*c8dee2aaSAndroid Build Coastguard Worker 
description()134*c8dee2aaSAndroid Build Coastguard Worker     std::string description() const override {
135*c8dee2aaSAndroid Build Coastguard Worker         return this->layout().paddedDescription() + this->modifierFlags().paddedDescription() +
136*c8dee2aaSAndroid Build Coastguard Worker                this->type().displayName() + " " + std::string(this->name());
137*c8dee2aaSAndroid Build Coastguard Worker     }
138*c8dee2aaSAndroid Build Coastguard Worker 
139*c8dee2aaSAndroid Build Coastguard Worker private:
140*c8dee2aaSAndroid Build Coastguard Worker     IRNode* fDeclaringElement = nullptr;
141*c8dee2aaSAndroid Build Coastguard Worker     ModifierFlags fModifierFlags;
142*c8dee2aaSAndroid Build Coastguard Worker     Position fModifiersPosition;
143*c8dee2aaSAndroid Build Coastguard Worker     VariableStorage fStorage;
144*c8dee2aaSAndroid Build Coastguard Worker     bool fBuiltin;
145*c8dee2aaSAndroid Build Coastguard Worker 
146*c8dee2aaSAndroid Build Coastguard Worker     using INHERITED = Symbol;
147*c8dee2aaSAndroid Build Coastguard Worker };
148*c8dee2aaSAndroid Build Coastguard Worker 
149*c8dee2aaSAndroid Build Coastguard Worker /**
150*c8dee2aaSAndroid Build Coastguard Worker  * ExtendedVariable is functionally equivalent to a regular Variable, but it also contains extra
151*c8dee2aaSAndroid Build Coastguard Worker  * fields that most variables don't need:
152*c8dee2aaSAndroid Build Coastguard Worker  * - The variable's associated InterfaceBlock
153*c8dee2aaSAndroid Build Coastguard Worker  * - The variable's layout
154*c8dee2aaSAndroid Build Coastguard Worker  * - The variable's mangled name
155*c8dee2aaSAndroid Build Coastguard Worker  *
156*c8dee2aaSAndroid Build Coastguard Worker  * Some of these fields can be null/empty.
157*c8dee2aaSAndroid Build Coastguard Worker  */
158*c8dee2aaSAndroid Build Coastguard Worker class ExtendedVariable final : public Variable {
159*c8dee2aaSAndroid Build Coastguard Worker public:
ExtendedVariable(Position pos,Position modifiersPosition,const Layout & layout,ModifierFlags flags,std::string_view name,const Type * type,bool builtin,Storage storage,std::string mangledName)160*c8dee2aaSAndroid Build Coastguard Worker     ExtendedVariable(Position pos, Position modifiersPosition, const Layout& layout,
161*c8dee2aaSAndroid Build Coastguard Worker                      ModifierFlags flags, std::string_view name, const Type* type, bool builtin,
162*c8dee2aaSAndroid Build Coastguard Worker                      Storage storage, std::string mangledName)
163*c8dee2aaSAndroid Build Coastguard Worker             : INHERITED(pos, modifiersPosition, flags, name, type, builtin, storage)
164*c8dee2aaSAndroid Build Coastguard Worker             , fLayout(layout)
165*c8dee2aaSAndroid Build Coastguard Worker             , fMangledName(std::move(mangledName)) {}
166*c8dee2aaSAndroid Build Coastguard Worker 
167*c8dee2aaSAndroid Build Coastguard Worker     ~ExtendedVariable() override;
168*c8dee2aaSAndroid Build Coastguard Worker 
interfaceBlock()169*c8dee2aaSAndroid Build Coastguard Worker     InterfaceBlock* interfaceBlock() const override {
170*c8dee2aaSAndroid Build Coastguard Worker         return fInterfaceBlockElement;
171*c8dee2aaSAndroid Build Coastguard Worker     }
172*c8dee2aaSAndroid Build Coastguard Worker 
layout()173*c8dee2aaSAndroid Build Coastguard Worker     const Layout& layout() const override {
174*c8dee2aaSAndroid Build Coastguard Worker         return fLayout;
175*c8dee2aaSAndroid Build Coastguard Worker     }
176*c8dee2aaSAndroid Build Coastguard Worker 
setInterfaceBlock(InterfaceBlock * elem)177*c8dee2aaSAndroid Build Coastguard Worker     void setInterfaceBlock(InterfaceBlock* elem) override {
178*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(!fInterfaceBlockElement);
179*c8dee2aaSAndroid Build Coastguard Worker         fInterfaceBlockElement = elem;
180*c8dee2aaSAndroid Build Coastguard Worker     }
181*c8dee2aaSAndroid Build Coastguard Worker 
detachDeadInterfaceBlock()182*c8dee2aaSAndroid Build Coastguard Worker     void detachDeadInterfaceBlock() override {
183*c8dee2aaSAndroid Build Coastguard Worker         // The InterfaceBlock is being deleted, so our reference to it has become stale.
184*c8dee2aaSAndroid Build Coastguard Worker         fInterfaceBlockElement = nullptr;
185*c8dee2aaSAndroid Build Coastguard Worker     }
186*c8dee2aaSAndroid Build Coastguard Worker 
187*c8dee2aaSAndroid Build Coastguard Worker     std::string_view mangledName() const override;
188*c8dee2aaSAndroid Build Coastguard Worker 
189*c8dee2aaSAndroid Build Coastguard Worker private:
190*c8dee2aaSAndroid Build Coastguard Worker     InterfaceBlock* fInterfaceBlockElement = nullptr;
191*c8dee2aaSAndroid Build Coastguard Worker     Layout fLayout;
192*c8dee2aaSAndroid Build Coastguard Worker     std::string fMangledName;
193*c8dee2aaSAndroid Build Coastguard Worker 
194*c8dee2aaSAndroid Build Coastguard Worker     using INHERITED = Variable;
195*c8dee2aaSAndroid Build Coastguard Worker };
196*c8dee2aaSAndroid Build Coastguard Worker 
197*c8dee2aaSAndroid Build Coastguard Worker } // namespace SkSL
198*c8dee2aaSAndroid Build Coastguard Worker 
199*c8dee2aaSAndroid Build Coastguard Worker #endif
200