blob: 994442819a8e2351c24b0adb753f5becc6871f6a [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 Nicholas0df1b042017-03-31 13:56:23 -04007
ethannicholasb3058bd2016-07-01 08:22:01 -07008#ifndef SKSL_SPIRVCODEGENERATOR
9#define SKSL_SPIRVCODEGENERATOR
10
ethannicholasb3058bd2016-07-01 08:22:01 -070011#include <stack>
12#include <tuple>
13#include <unordered_map>
14
15#include "SkSLCodeGenerator.h"
ethannicholas8ac838d2016-11-22 08:39:36 -080016#include "SkSLMemoryLayout.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070017#include "ir/SkSLBinaryExpression.h"
18#include "ir/SkSLBoolLiteral.h"
19#include "ir/SkSLConstructor.h"
Ethan Nicholasfd146aa2017-01-13 16:40:35 -050020#include "ir/SkSLDoStatement.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070021#include "ir/SkSLFloatLiteral.h"
22#include "ir/SkSLIfStatement.h"
23#include "ir/SkSLIndexExpression.h"
24#include "ir/SkSLInterfaceBlock.h"
25#include "ir/SkSLIntLiteral.h"
26#include "ir/SkSLFieldAccess.h"
27#include "ir/SkSLForStatement.h"
28#include "ir/SkSLFunctionCall.h"
29#include "ir/SkSLFunctionDeclaration.h"
30#include "ir/SkSLFunctionDefinition.h"
31#include "ir/SkSLPrefixExpression.h"
32#include "ir/SkSLPostfixExpression.h"
33#include "ir/SkSLProgramElement.h"
34#include "ir/SkSLReturnStatement.h"
35#include "ir/SkSLStatement.h"
36#include "ir/SkSLSwizzle.h"
37#include "ir/SkSLTernaryExpression.h"
ethannicholas22f939e2016-10-13 13:25:34 -070038#include "ir/SkSLVarDeclarations.h"
39#include "ir/SkSLVarDeclarationsStatement.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070040#include "ir/SkSLVariableReference.h"
Ethan Nicholasfd146aa2017-01-13 16:40:35 -050041#include "ir/SkSLWhileStatement.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070042#include "spirv.h"
43
44namespace SkSL {
45
46#define kLast_Capability SpvCapabilityMultiViewport
47
48/**
49 * Converts a Program into a SPIR-V binary.
50 */
51class SPIRVCodeGenerator : public CodeGenerator {
52public:
53 class LValue {
54 public:
55 virtual ~LValue() {}
Greg Daniel64773e62016-11-22 09:44:03 -050056
ethannicholasb3058bd2016-07-01 08:22:01 -070057 // returns a pointer to the lvalue, if possible. If the lvalue cannot be directly referenced
58 // by a pointer (e.g. vector swizzles), returns 0.
59 virtual SpvId getPointer() = 0;
60
Ethan Nicholas0df1b042017-03-31 13:56:23 -040061 virtual SpvId load(OutputStream& out) = 0;
ethannicholasb3058bd2016-07-01 08:22:01 -070062
Ethan Nicholas0df1b042017-03-31 13:56:23 -040063 virtual void store(SpvId value, OutputStream& out) = 0;
ethannicholasb3058bd2016-07-01 08:22:01 -070064 };
65
Ethan Nicholas941e7e22016-12-12 15:33:30 -050066 SPIRVCodeGenerator(const Context* context, const Program* program, ErrorReporter* errors,
Ethan Nicholas0df1b042017-03-31 13:56:23 -040067 OutputStream* out)
Ethan Nicholas941e7e22016-12-12 15:33:30 -050068 : INHERITED(program, errors, out)
69 , fContext(*context)
ethannicholas8ac838d2016-11-22 08:39:36 -080070 , fDefaultLayout(MemoryLayout::k140_Standard)
ethannicholasd598f792016-07-25 10:08:54 -070071 , fCapabilities(1 << SpvCapabilityShader)
ethannicholasb3058bd2016-07-01 08:22:01 -070072 , fIdCount(1)
73 , fBoolTrue(0)
74 , fBoolFalse(0)
Ethan Nicholas941e7e22016-12-12 15:33:30 -050075 , fSetupFragPosition(false)
Ethan Nicholas8feeff92017-03-30 14:11:58 -040076 , fCurrentBlock(0)
77 , fSynthetics(nullptr, errors) {
ethannicholasb3058bd2016-07-01 08:22:01 -070078 this->setupIntrinsics();
79 }
80
Ethan Nicholas941e7e22016-12-12 15:33:30 -050081 bool generateCode() override;
ethannicholasb3058bd2016-07-01 08:22:01 -070082
83private:
84 enum IntrinsicKind {
85 kGLSL_STD_450_IntrinsicKind,
86 kSPIRV_IntrinsicKind,
87 kSpecial_IntrinsicKind
88 };
89
90 enum SpecialIntrinsic {
91 kAtan_SpecialIntrinsic,
Greg Daniel64773e62016-11-22 09:44:03 -050092 kSubpassLoad_SpecialIntrinsic,
Ethan Nicholas0187ae62017-05-03 11:03:44 -040093 kTexelFetch_SpecialIntrinsic,
94 kTexture_SpecialIntrinsic,
ethannicholasb3058bd2016-07-01 08:22:01 -070095 };
96
97 void setupIntrinsics();
98
99 SpvId nextId();
100
101 SpvId getType(const Type& type);
102
ethannicholas8ac838d2016-11-22 08:39:36 -0800103 SpvId getType(const Type& type, const MemoryLayout& layout);
104
Ethan Nicholas0187ae62017-05-03 11:03:44 -0400105 SpvId getImageType(const Type& type);
106
ethannicholasd598f792016-07-25 10:08:54 -0700107 SpvId getFunctionType(const FunctionDeclaration& function);
ethannicholasb3058bd2016-07-01 08:22:01 -0700108
ethannicholasd598f792016-07-25 10:08:54 -0700109 SpvId getPointerType(const Type& type, SpvStorageClass_ storageClass);
ethannicholasb3058bd2016-07-01 08:22:01 -0700110
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500111 SpvId getPointerType(const Type& type, const MemoryLayout& layout,
ethannicholas8ac838d2016-11-22 08:39:36 -0800112 SpvStorageClass_ storageClass);
113
Ethan Nicholasa51d7132017-06-09 10:47:31 -0400114 void writePrecisionModifier(const Modifiers& modifiers, SpvId id);
115
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400116 std::vector<SpvId> getAccessChain(const Expression& expr, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700117
118 void writeLayout(const Layout& layout, SpvId target);
119
120 void writeLayout(const Layout& layout, SpvId target, int member);
121
ethannicholas8ac838d2016-11-22 08:39:36 -0800122 void writeStruct(const Type& type, const MemoryLayout& layout, SpvId resultId);
ethannicholasb3058bd2016-07-01 08:22:01 -0700123
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400124 void writeProgramElement(const ProgramElement& pe, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700125
ethannicholasf789b382016-08-03 12:43:36 -0700126 SpvId writeInterfaceBlock(const InterfaceBlock& intf);
ethannicholasb3058bd2016-07-01 08:22:01 -0700127
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400128 SpvId writeFunctionStart(const FunctionDeclaration& f, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700129
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400130 SpvId writeFunctionDeclaration(const FunctionDeclaration& f, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700131
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400132 SpvId writeFunction(const FunctionDefinition& f, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700133
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400134 void writeGlobalVars(Program::Kind kind, const VarDeclarations& v, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700135
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400136 void writeVarDeclarations(const VarDeclarations& decl, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700137
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400138 SpvId writeVariableReference(const VariableReference& ref, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700139
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400140 std::unique_ptr<LValue> getLValue(const Expression& value, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700141
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400142 SpvId writeExpression(const Expression& expr, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700143
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400144 SpvId writeIntrinsicCall(const FunctionCall& c, OutputStream& out);
145
146 SpvId writeFunctionCall(const FunctionCall& c, OutputStream& out);
147
148 SpvId writeSpecialIntrinsic(const FunctionCall& c, SpecialIntrinsic kind, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700149
ethannicholasf789b382016-08-03 12:43:36 -0700150 SpvId writeConstantVector(const Constructor& c);
ethannicholasb3058bd2016-07-01 08:22:01 -0700151
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400152 SpvId writeFloatConstructor(const Constructor& c, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700153
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400154 SpvId writeIntConstructor(const Constructor& c, OutputStream& out);
Ethan Nicholas84645e32017-02-09 13:57:14 -0500155
156 /**
157 * Writes a matrix with the diagonal entries all equal to the provided expression, and all other
158 * entries equal to zero.
159 */
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400160 void writeUniformScaleMatrix(SpvId id, SpvId diagonal, const Type& type, OutputStream& out);
Ethan Nicholas84645e32017-02-09 13:57:14 -0500161
162 /**
163 * Writes a potentially-different-sized copy of a matrix. Entries which do not exist in the
164 * source matrix are filled with zero; entries which do not exist in the destination matrix are
165 * ignored.
166 */
167 void writeMatrixCopy(SpvId id, SpvId src, const Type& srcType, const Type& dstType,
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400168 OutputStream& out);
Ethan Nicholas84645e32017-02-09 13:57:14 -0500169
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400170 SpvId writeMatrixConstructor(const Constructor& c, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700171
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400172 SpvId writeVectorConstructor(const Constructor& c, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700173
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400174 SpvId writeConstructor(const Constructor& c, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700175
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400176 SpvId writeFieldAccess(const FieldAccess& f, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700177
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400178 SpvId writeSwizzle(const Swizzle& swizzle, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700179
Ethan Nicholasef653b82017-02-21 13:50:00 -0500180 /**
181 * Folds the potentially-vector result of a logical operation down to a single bool. If
182 * operandType is a vector type, assumes that the intermediate result in id is a bvec of the
183 * same dimensions, and applys all() to it to fold it down to a single bool value. Otherwise,
184 * returns the original id value.
185 */
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400186 SpvId foldToBool(SpvId id, const Type& operandType, OutputStream& out);
Ethan Nicholasef653b82017-02-21 13:50:00 -0500187
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500188 SpvId writeBinaryOperation(const Type& resultType, const Type& operandType, SpvId lhs,
189 SpvId rhs, SpvOp_ ifFloat, SpvOp_ ifInt, SpvOp_ ifUInt,
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400190 SpvOp_ ifBool, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700191
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500192 SpvId writeBinaryOperation(const BinaryExpression& expr, SpvOp_ ifFloat, SpvOp_ ifInt,
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400193 SpvOp_ ifUInt, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700194
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400195 SpvId writeBinaryExpression(const BinaryExpression& b, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700196
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400197 SpvId writeTernaryExpression(const TernaryExpression& t, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700198
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400199 SpvId writeIndexExpression(const IndexExpression& expr, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700200
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400201 SpvId writeLogicalAnd(const BinaryExpression& b, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700202
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400203 SpvId writeLogicalOr(const BinaryExpression& o, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700204
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400205 SpvId writePrefixExpression(const PrefixExpression& p, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700206
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400207 SpvId writePostfixExpression(const PostfixExpression& p, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700208
ethannicholasf789b382016-08-03 12:43:36 -0700209 SpvId writeBoolLiteral(const BoolLiteral& b);
ethannicholasb3058bd2016-07-01 08:22:01 -0700210
ethannicholasf789b382016-08-03 12:43:36 -0700211 SpvId writeIntLiteral(const IntLiteral& i);
ethannicholasb3058bd2016-07-01 08:22:01 -0700212
ethannicholasf789b382016-08-03 12:43:36 -0700213 SpvId writeFloatLiteral(const FloatLiteral& f);
ethannicholasb3058bd2016-07-01 08:22:01 -0700214
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400215 void writeStatement(const Statement& s, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700216
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400217 void writeBlock(const Block& b, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700218
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400219 void writeIfStatement(const IfStatement& stmt, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700220
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400221 void writeForStatement(const ForStatement& f, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700222
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400223 void writeWhileStatement(const WhileStatement& w, OutputStream& out);
Ethan Nicholasfd146aa2017-01-13 16:40:35 -0500224
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400225 void writeDoStatement(const DoStatement& d, OutputStream& out);
Ethan Nicholasfd146aa2017-01-13 16:40:35 -0500226
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400227 void writeReturnStatement(const ReturnStatement& r, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700228
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400229 void writeCapabilities(OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700230
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400231 void writeInstructions(const Program& program, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700232
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400233 void writeOpCode(SpvOp_ opCode, int length, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700234
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400235 void writeWord(int32_t word, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700236
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400237 void writeString(const char* string, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700238
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400239 void writeLabel(SpvId id, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700240
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400241 void writeInstruction(SpvOp_ opCode, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700242
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400243 void writeInstruction(SpvOp_ opCode, const char* string, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700244
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400245 void writeInstruction(SpvOp_ opCode, int32_t word1, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700246
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400247 void writeInstruction(SpvOp_ opCode, int32_t word1, const char* string, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700248
249 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, const char* string,
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400250 OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700251
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400252 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700253
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500254 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3,
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400255 OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700256
257 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3, int32_t word4,
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400258 OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700259
260 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3, int32_t word4,
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400261 int32_t word5, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700262
263 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3, int32_t word4,
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400264 int32_t word5, int32_t word6, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700265
266 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3, int32_t word4,
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400267 int32_t word5, int32_t word6, int32_t word7, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700268
269 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3, int32_t word4,
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500270 int32_t word5, int32_t word6, int32_t word7, int32_t word8,
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400271 OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700272
ethannicholasd598f792016-07-25 10:08:54 -0700273 const Context& fContext;
ethannicholas8ac838d2016-11-22 08:39:36 -0800274 const MemoryLayout fDefaultLayout;
ethannicholasd598f792016-07-25 10:08:54 -0700275
ethannicholasb3058bd2016-07-01 08:22:01 -0700276 uint64_t fCapabilities;
277 SpvId fIdCount;
278 SpvId fGLSLExtendedInstructions;
279 typedef std::tuple<IntrinsicKind, int32_t, int32_t, int32_t, int32_t> Intrinsic;
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400280 std::unordered_map<String, Intrinsic> fIntrinsicMap;
ethannicholasd598f792016-07-25 10:08:54 -0700281 std::unordered_map<const FunctionDeclaration*, SpvId> fFunctionMap;
282 std::unordered_map<const Variable*, SpvId> fVariableMap;
283 std::unordered_map<const Variable*, int32_t> fInterfaceBlockMap;
Ethan Nicholas0187ae62017-05-03 11:03:44 -0400284 std::unordered_map<String, SpvId> fImageTypeMap;
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400285 std::unordered_map<String, SpvId> fTypeMap;
286 StringStream fCapabilitiesBuffer;
287 StringStream fGlobalInitializersBuffer;
288 StringStream fConstantBuffer;
289 StringStream fExtraGlobalsBuffer;
290 StringStream fExternalFunctionsBuffer;
291 StringStream fVariableBuffer;
292 StringStream fNameBuffer;
293 StringStream fDecorationBuffer;
ethannicholasb3058bd2016-07-01 08:22:01 -0700294
295 SpvId fBoolTrue;
296 SpvId fBoolFalse;
297 std::unordered_map<int64_t, SpvId> fIntConstants;
298 std::unordered_map<uint64_t, SpvId> fUIntConstants;
299 std::unordered_map<float, SpvId> fFloatConstants;
300 std::unordered_map<double, SpvId> fDoubleConstants;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500301 bool fSetupFragPosition;
ethannicholasb3058bd2016-07-01 08:22:01 -0700302 // label of the current block, or 0 if we are not in a block
303 SpvId fCurrentBlock;
304 std::stack<SpvId> fBreakTarget;
305 std::stack<SpvId> fContinueTarget;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500306 SpvId fRTHeightStructId = (SpvId) -1;
307 SpvId fRTHeightFieldIndex = (SpvId) -1;
Ethan Nicholas8feeff92017-03-30 14:11:58 -0400308 // holds variables synthesized during output, for lifetime purposes
309 SymbolTable fSynthetics;
ethannicholasb3058bd2016-07-01 08:22:01 -0700310
311 friend class PointerLValue;
312 friend class SwizzleLValue;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500313
314 typedef CodeGenerator INHERITED;
ethannicholasb3058bd2016-07-01 08:22:01 -0700315};
316
317}
318
319#endif