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