blob: 127055ad7f1897e8017079c84546bcb5ce780757 [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
John Stilescd806892021-01-06 13:33:31 -050015#include "src/core/SkOpts.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050016#include "src/sksl/SkSLCodeGenerator.h"
17#include "src/sksl/SkSLMemoryLayout.h"
Mike Klein4b432fa2019-06-06 11:44:05 -050018#include "src/sksl/SkSLStringStream.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050019#include "src/sksl/ir/SkSLBinaryExpression.h"
20#include "src/sksl/ir/SkSLBoolLiteral.h"
21#include "src/sksl/ir/SkSLConstructor.h"
22#include "src/sksl/ir/SkSLDoStatement.h"
23#include "src/sksl/ir/SkSLFieldAccess.h"
24#include "src/sksl/ir/SkSLFloatLiteral.h"
25#include "src/sksl/ir/SkSLForStatement.h"
26#include "src/sksl/ir/SkSLFunctionCall.h"
27#include "src/sksl/ir/SkSLFunctionDeclaration.h"
28#include "src/sksl/ir/SkSLFunctionDefinition.h"
29#include "src/sksl/ir/SkSLIfStatement.h"
30#include "src/sksl/ir/SkSLIndexExpression.h"
31#include "src/sksl/ir/SkSLIntLiteral.h"
32#include "src/sksl/ir/SkSLInterfaceBlock.h"
33#include "src/sksl/ir/SkSLPostfixExpression.h"
34#include "src/sksl/ir/SkSLPrefixExpression.h"
35#include "src/sksl/ir/SkSLProgramElement.h"
36#include "src/sksl/ir/SkSLReturnStatement.h"
37#include "src/sksl/ir/SkSLStatement.h"
38#include "src/sksl/ir/SkSLSwitchStatement.h"
39#include "src/sksl/ir/SkSLSwizzle.h"
40#include "src/sksl/ir/SkSLTernaryExpression.h"
41#include "src/sksl/ir/SkSLVarDeclarations.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050042#include "src/sksl/ir/SkSLVariableReference.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050043#include "src/sksl/spirv.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070044
John Stilescd806892021-01-06 13:33:31 -050045namespace SkSL {
46
Ethan Nicholascc5d3e02019-04-19 09:50:56 -040047union ConstantValue {
John Stilesacb091f2021-01-06 11:57:58 -050048 ConstantValue(SKSL_INT i)
Ethan Nicholasa3f22f12020-10-01 12:13:17 -040049 : fInt(i) {
John Stilesacb091f2021-01-06 11:57:58 -050050 static_assert(sizeof(*this) == sizeof(SKSL_INT));
Ethan Nicholasa3f22f12020-10-01 12:13:17 -040051 }
Ethan Nicholascc5d3e02019-04-19 09:50:56 -040052
Ethan Nicholasa3f22f12020-10-01 12:13:17 -040053 ConstantValue(SKSL_FLOAT f) {
54 memset(this, 0, sizeof(*this));
55 fFloat = f;
56 }
Ethan Nicholascc5d3e02019-04-19 09:50:56 -040057
58 bool operator==(const ConstantValue& other) const {
59 return fInt == other.fInt;
60 }
61
John Stilesacb091f2021-01-06 11:57:58 -050062 SKSL_INT fInt;
Ethan Nicholasa3f22f12020-10-01 12:13:17 -040063 SKSL_FLOAT fFloat;
Ethan Nicholascc5d3e02019-04-19 09:50:56 -040064};
65
John Stilesacb091f2021-01-06 11:57:58 -050066using ConstantValuePair = std::pair<ConstantValue, SkSL::Type::NumberKind>;
Ethan Nicholascc5d3e02019-04-19 09:50:56 -040067
John Stilescd806892021-01-06 13:33:31 -050068struct SPIRVVectorConstant {
69 bool operator==(const SPIRVVectorConstant& that) const {
70 return fTypeId == that.fTypeId &&
71 fValueId[0] == that.fValueId[0] &&
72 fValueId[1] == that.fValueId[1] &&
73 fValueId[2] == that.fValueId[2] &&
74 fValueId[3] == that.fValueId[3];
75 }
76 SpvId fTypeId;
77 SpvId fValueId[4];
78};
79
80} // namespace SkSL
81
Ethan Nicholascc5d3e02019-04-19 09:50:56 -040082namespace std {
83
84template <>
John Stilescd806892021-01-06 13:33:31 -050085struct hash<SkSL::ConstantValuePair> {
86 size_t operator()(const SkSL::ConstantValuePair& key) const {
Ethan Nicholascc5d3e02019-04-19 09:50:56 -040087 return key.first.fInt ^ (int) key.second;
88 }
89};
90
John Stilescd806892021-01-06 13:33:31 -050091template <>
92struct hash<SkSL::SPIRVVectorConstant> {
93 size_t operator()(const SkSL::SPIRVVectorConstant& key) const {
94 return SkOpts::hash(&key, sizeof(key));
95 }
96};
97
John Stilesa6841be2020-08-06 14:11:56 -040098} // namespace std
Ethan Nicholascc5d3e02019-04-19 09:50:56 -040099
ethannicholasb3058bd2016-07-01 08:22:01 -0700100namespace SkSL {
101
ethannicholasb3058bd2016-07-01 08:22:01 -0700102/**
103 * Converts a Program into a SPIR-V binary.
104 */
105class SPIRVCodeGenerator : public CodeGenerator {
106public:
107 class LValue {
108 public:
109 virtual ~LValue() {}
Greg Daniel64773e62016-11-22 09:44:03 -0500110
ethannicholasb3058bd2016-07-01 08:22:01 -0700111 // returns a pointer to the lvalue, if possible. If the lvalue cannot be directly referenced
112 // by a pointer (e.g. vector swizzles), returns 0.
113 virtual SpvId getPointer() = 0;
114
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400115 virtual SpvId load(OutputStream& out) = 0;
ethannicholasb3058bd2016-07-01 08:22:01 -0700116
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400117 virtual void store(SpvId value, OutputStream& out) = 0;
ethannicholasb3058bd2016-07-01 08:22:01 -0700118 };
119
Brian Osman8b43dad2020-10-09 13:31:42 -0400120 SPIRVCodeGenerator(const Context* context,
121 const Program* program,
122 ErrorReporter* errors,
123 OutputStream* out)
124 : INHERITED(program, errors, out)
125 , fContext(*context)
126 , fDefaultLayout(MemoryLayout::k140_Standard)
127 , fCapabilities(0)
128 , fIdCount(1)
129 , fBoolTrue(0)
130 , fBoolFalse(0)
131 , fSetupFragPosition(false)
132 , fCurrentBlock(0)
John Stiles7c3515b2020-10-16 18:38:39 -0400133 , fSynthetics(errors, /*builtin=*/true) {
ethannicholasb3058bd2016-07-01 08:22:01 -0700134 this->setupIntrinsics();
135 }
136
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500137 bool generateCode() override;
ethannicholasb3058bd2016-07-01 08:22:01 -0700138
139private:
140 enum IntrinsicKind {
141 kGLSL_STD_450_IntrinsicKind,
142 kSPIRV_IntrinsicKind,
143 kSpecial_IntrinsicKind
144 };
145
146 enum SpecialIntrinsic {
147 kAtan_SpecialIntrinsic,
Ethan Nicholas0fc07f92018-02-27 15:25:47 -0500148 kClamp_SpecialIntrinsic,
Brian Osmand1b593f2020-12-28 13:00:46 -0500149 kMatrixCompMult_SpecialIntrinsic,
Ethan Nicholas0fc07f92018-02-27 15:25:47 -0500150 kMax_SpecialIntrinsic,
151 kMin_SpecialIntrinsic,
152 kMix_SpecialIntrinsic,
Ethan Nicholas70a44b22017-11-30 09:09:16 -0500153 kMod_SpecialIntrinsic,
Chris Daltonb8af5ad2019-02-25 14:54:21 -0700154 kDFdy_SpecialIntrinsic,
Ethan Nicholas12fb9cf2018-08-03 16:16:57 -0400155 kSaturate_SpecialIntrinsic,
Stephen Whiteff5d7a22019-07-26 17:42:06 -0400156 kSampledImage_SpecialIntrinsic,
Brian Osman6ba3be12020-11-13 16:32:52 -0500157 kSmoothStep_SpecialIntrinsic,
158 kStep_SpecialIntrinsic,
Greg Daniel64773e62016-11-22 09:44:03 -0500159 kSubpassLoad_SpecialIntrinsic,
Ethan Nicholas0187ae62017-05-03 11:03:44 -0400160 kTexture_SpecialIntrinsic,
ethannicholasb3058bd2016-07-01 08:22:01 -0700161 };
162
Ethan Nicholas10e93b62019-03-20 10:46:14 -0400163 enum class Precision {
164 kLow,
165 kHigh,
166 };
167
ethannicholasb3058bd2016-07-01 08:22:01 -0700168 void setupIntrinsics();
169
170 SpvId nextId();
171
Ethan Nicholase2c49992020-10-05 11:49:11 -0400172 const Type& getActualType(const Type& type);
Ethan Nicholasf7b88202017-09-18 14:10:39 -0400173
ethannicholasb3058bd2016-07-01 08:22:01 -0700174 SpvId getType(const Type& type);
175
ethannicholas8ac838d2016-11-22 08:39:36 -0800176 SpvId getType(const Type& type, const MemoryLayout& layout);
177
Ethan Nicholas0187ae62017-05-03 11:03:44 -0400178 SpvId getImageType(const Type& type);
179
ethannicholasd598f792016-07-25 10:08:54 -0700180 SpvId getFunctionType(const FunctionDeclaration& function);
ethannicholasb3058bd2016-07-01 08:22:01 -0700181
ethannicholasd598f792016-07-25 10:08:54 -0700182 SpvId getPointerType(const Type& type, SpvStorageClass_ storageClass);
ethannicholasb3058bd2016-07-01 08:22:01 -0700183
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500184 SpvId getPointerType(const Type& type, const MemoryLayout& layout,
ethannicholas8ac838d2016-11-22 08:39:36 -0800185 SpvStorageClass_ storageClass);
186
Ethan Nicholas10e93b62019-03-20 10:46:14 -0400187 void writePrecisionModifier(Precision precision, SpvId id);
188
Ethan Nicholas858fecc2019-03-07 13:19:18 -0500189 void writePrecisionModifier(const Type& type, SpvId id);
Ethan Nicholasa51d7132017-06-09 10:47:31 -0400190
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400191 std::vector<SpvId> getAccessChain(const Expression& expr, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700192
193 void writeLayout(const Layout& layout, SpvId target);
194
195 void writeLayout(const Layout& layout, SpvId target, int member);
196
ethannicholas8ac838d2016-11-22 08:39:36 -0800197 void writeStruct(const Type& type, const MemoryLayout& layout, SpvId resultId);
ethannicholasb3058bd2016-07-01 08:22:01 -0700198
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400199 void writeProgramElement(const ProgramElement& pe, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700200
Stephen White88574972020-06-23 19:09:29 -0400201 SpvId writeInterfaceBlock(const InterfaceBlock& intf, bool appendRTHeight = true);
ethannicholasb3058bd2016-07-01 08:22:01 -0700202
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400203 SpvId writeFunctionStart(const FunctionDeclaration& f, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700204
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400205 SpvId writeFunctionDeclaration(const FunctionDeclaration& f, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700206
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400207 SpvId writeFunction(const FunctionDefinition& f, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700208
Brian Osmanc0213602020-10-06 14:43:32 -0400209 void writeGlobalVar(Program::Kind kind, const VarDeclaration& v, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700210
Brian Osmanc0213602020-10-06 14:43:32 -0400211 void writeVarDeclaration(const VarDeclaration& var, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700212
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400213 SpvId writeVariableReference(const VariableReference& ref, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700214
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400215 std::unique_ptr<LValue> getLValue(const Expression& value, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700216
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400217 SpvId writeExpression(const Expression& expr, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700218
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400219 SpvId writeIntrinsicCall(const FunctionCall& c, OutputStream& out);
220
221 SpvId writeFunctionCall(const FunctionCall& c, OutputStream& out);
222
Ethan Nicholas0fc07f92018-02-27 15:25:47 -0500223
224 void writeGLSLExtendedInstruction(const Type& type, SpvId id, SpvId floatInst,
225 SpvId signedInst, SpvId unsignedInst,
226 const std::vector<SpvId>& args, OutputStream& out);
227
228 /**
229 * Given a list of potentially mixed scalars and vectors, promotes the scalars to match the
230 * size of the vectors and returns the ids of the written expressions. e.g. given (float, vec2),
231 * returns (vec2(float), vec2). It is an error to use mismatched vector sizes, e.g. (float,
232 * vec2, vec3).
233 */
John Stiles8e3b6be2020-10-13 11:14:08 -0400234 std::vector<SpvId> vectorize(const ExpressionArray& args, OutputStream& out);
Ethan Nicholas0fc07f92018-02-27 15:25:47 -0500235
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400236 SpvId writeSpecialIntrinsic(const FunctionCall& c, SpecialIntrinsic kind, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700237
ethannicholasf789b382016-08-03 12:43:36 -0700238 SpvId writeConstantVector(const Constructor& c);
ethannicholasb3058bd2016-07-01 08:22:01 -0700239
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400240 SpvId writeFloatConstructor(const Constructor& c, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700241
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400242 SpvId writeIntConstructor(const Constructor& c, OutputStream& out);
Ethan Nicholas84645e32017-02-09 13:57:14 -0500243
Ethan Nicholas925f52d2017-07-19 10:42:50 -0400244 SpvId writeUIntConstructor(const Constructor& c, OutputStream& out);
245
Ethan Nicholas84645e32017-02-09 13:57:14 -0500246 /**
247 * Writes a matrix with the diagonal entries all equal to the provided expression, and all other
248 * entries equal to zero.
249 */
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400250 void writeUniformScaleMatrix(SpvId id, SpvId diagonal, const Type& type, OutputStream& out);
Ethan Nicholas84645e32017-02-09 13:57:14 -0500251
252 /**
253 * Writes a potentially-different-sized copy of a matrix. Entries which do not exist in the
254 * source matrix are filled with zero; entries which do not exist in the destination matrix are
255 * ignored.
256 */
257 void writeMatrixCopy(SpvId id, SpvId src, const Type& srcType, const Type& dstType,
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400258 OutputStream& out);
Ethan Nicholas84645e32017-02-09 13:57:14 -0500259
Ethan Nicholascc5d3e02019-04-19 09:50:56 -0400260 void addColumnEntry(SpvId columnType, Precision precision, std::vector<SpvId>* currentColumn,
Ethan Nicholas5c46b722019-03-22 14:32:37 -0400261 std::vector<SpvId>* columnIds, int* currentCount, int rows, SpvId entry,
262 OutputStream& out);
263
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400264 SpvId writeMatrixConstructor(const Constructor& c, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700265
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400266 SpvId writeVectorConstructor(const Constructor& c, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700267
Ethan Nicholasbd553222017-07-18 15:54:59 -0400268 SpvId writeArrayConstructor(const Constructor& c, OutputStream& out);
269
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400270 SpvId writeConstructor(const Constructor& c, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700271
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400272 SpvId writeFieldAccess(const FieldAccess& f, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700273
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400274 SpvId writeSwizzle(const Swizzle& swizzle, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700275
Ethan Nicholasef653b82017-02-21 13:50:00 -0500276 /**
277 * Folds the potentially-vector result of a logical operation down to a single bool. If
278 * operandType is a vector type, assumes that the intermediate result in id is a bvec of the
279 * same dimensions, and applys all() to it to fold it down to a single bool value. Otherwise,
280 * returns the original id value.
281 */
Ethan Nicholas48e24052018-03-14 13:51:39 -0400282 SpvId foldToBool(SpvId id, const Type& operandType, SpvOp op, OutputStream& out);
Ethan Nicholasef653b82017-02-21 13:50:00 -0500283
Ethan Nicholas68990be2017-07-13 09:36:52 -0400284 SpvId writeMatrixComparison(const Type& operandType, SpvId lhs, SpvId rhs, SpvOp_ floatOperator,
Ethan Nicholas0df21132018-07-10 09:37:51 -0400285 SpvOp_ intOperator, SpvOp_ vectorMergeOperator,
286 SpvOp_ mergeOperator, OutputStream& out);
287
288 SpvId writeComponentwiseMatrixBinary(const Type& operandType, SpvId lhs, SpvId rhs,
289 SpvOp_ floatOperator, SpvOp_ intOperator,
290 OutputStream& out);
Ethan Nicholas68990be2017-07-13 09:36:52 -0400291
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500292 SpvId writeBinaryOperation(const Type& resultType, const Type& operandType, SpvId lhs,
293 SpvId rhs, SpvOp_ ifFloat, SpvOp_ ifInt, SpvOp_ ifUInt,
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400294 SpvOp_ ifBool, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700295
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500296 SpvId writeBinaryOperation(const BinaryExpression& expr, SpvOp_ ifFloat, SpvOp_ ifInt,
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400297 SpvOp_ ifUInt, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700298
Ethan Nicholas49465b42019-04-17 12:22:21 -0400299 SpvId writeBinaryExpression(const Type& leftType, SpvId lhs, Token::Kind op,
300 const Type& rightType, SpvId rhs, const Type& resultType,
301 OutputStream& out);
302
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400303 SpvId writeBinaryExpression(const BinaryExpression& b, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700304
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400305 SpvId writeTernaryExpression(const TernaryExpression& t, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700306
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400307 SpvId writeIndexExpression(const IndexExpression& expr, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700308
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400309 SpvId writeLogicalAnd(const BinaryExpression& b, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700310
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400311 SpvId writeLogicalOr(const BinaryExpression& o, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700312
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400313 SpvId writePrefixExpression(const PrefixExpression& p, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700314
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400315 SpvId writePostfixExpression(const PostfixExpression& p, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700316
ethannicholasf789b382016-08-03 12:43:36 -0700317 SpvId writeBoolLiteral(const BoolLiteral& b);
ethannicholasb3058bd2016-07-01 08:22:01 -0700318
ethannicholasf789b382016-08-03 12:43:36 -0700319 SpvId writeIntLiteral(const IntLiteral& i);
ethannicholasb3058bd2016-07-01 08:22:01 -0700320
ethannicholasf789b382016-08-03 12:43:36 -0700321 SpvId writeFloatLiteral(const FloatLiteral& f);
ethannicholasb3058bd2016-07-01 08:22:01 -0700322
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400323 void writeStatement(const Statement& s, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700324
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400325 void writeBlock(const Block& b, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700326
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400327 void writeIfStatement(const IfStatement& stmt, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700328
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400329 void writeForStatement(const ForStatement& f, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700330
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400331 void writeDoStatement(const DoStatement& d, OutputStream& out);
Ethan Nicholasfd146aa2017-01-13 16:40:35 -0500332
Ethan Nicholase92b1b12017-11-13 16:13:21 -0500333 void writeSwitchStatement(const SwitchStatement& s, OutputStream& out);
334
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400335 void writeReturnStatement(const ReturnStatement& r, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700336
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400337 void writeCapabilities(OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700338
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400339 void writeInstructions(const Program& program, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700340
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400341 void writeOpCode(SpvOp_ opCode, int length, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700342
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400343 void writeWord(int32_t word, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700344
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700345 void writeString(const char* string, size_t length, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700346
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400347 void writeLabel(SpvId id, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700348
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400349 void writeInstruction(SpvOp_ opCode, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700350
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700351 void writeInstruction(SpvOp_ opCode, StringFragment string, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700352
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400353 void writeInstruction(SpvOp_ opCode, int32_t word1, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700354
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700355 void writeInstruction(SpvOp_ opCode, int32_t word1, StringFragment string, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700356
Ethan Nicholas5b5f0962017-09-11 13:50:14 -0700357 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, StringFragment string,
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400358 OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700359
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400360 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700361
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500362 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3,
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400363 OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700364
365 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3, int32_t word4,
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400366 OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700367
368 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3, int32_t word4,
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400369 int32_t word5, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700370
371 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3, int32_t word4,
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400372 int32_t word5, int32_t word6, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700373
374 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3, int32_t word4,
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400375 int32_t word5, int32_t word6, int32_t word7, OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700376
377 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3, int32_t word4,
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500378 int32_t word5, int32_t word6, int32_t word7, int32_t word8,
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400379 OutputStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700380
Ethan Nicholasbb155e22017-07-24 10:05:09 -0400381 void writeGeometryShaderExecutionMode(SpvId entryPoint, OutputStream& out);
382
ethannicholasd598f792016-07-25 10:08:54 -0700383 const Context& fContext;
ethannicholas8ac838d2016-11-22 08:39:36 -0800384 const MemoryLayout fDefaultLayout;
ethannicholasd598f792016-07-25 10:08:54 -0700385
ethannicholasb3058bd2016-07-01 08:22:01 -0700386 uint64_t fCapabilities;
387 SpvId fIdCount;
388 SpvId fGLSLExtendedInstructions;
389 typedef std::tuple<IntrinsicKind, int32_t, int32_t, int32_t, int32_t> Intrinsic;
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400390 std::unordered_map<String, Intrinsic> fIntrinsicMap;
ethannicholasd598f792016-07-25 10:08:54 -0700391 std::unordered_map<const FunctionDeclaration*, SpvId> fFunctionMap;
392 std::unordered_map<const Variable*, SpvId> fVariableMap;
393 std::unordered_map<const Variable*, int32_t> fInterfaceBlockMap;
Ethan Nicholas0187ae62017-05-03 11:03:44 -0400394 std::unordered_map<String, SpvId> fImageTypeMap;
Ethan Nicholas0df1b042017-03-31 13:56:23 -0400395 std::unordered_map<String, SpvId> fTypeMap;
396 StringStream fCapabilitiesBuffer;
397 StringStream fGlobalInitializersBuffer;
398 StringStream fConstantBuffer;
399 StringStream fExtraGlobalsBuffer;
400 StringStream fExternalFunctionsBuffer;
401 StringStream fVariableBuffer;
402 StringStream fNameBuffer;
403 StringStream fDecorationBuffer;
ethannicholasb3058bd2016-07-01 08:22:01 -0700404
405 SpvId fBoolTrue;
406 SpvId fBoolFalse;
John Stilesacb091f2021-01-06 11:57:58 -0500407 std::unordered_map<ConstantValuePair, SpvId> fNumberConstants;
John Stilescd806892021-01-06 13:33:31 -0500408 std::unordered_map<SPIRVVectorConstant, SpvId> fVectorConstants;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500409 bool fSetupFragPosition;
ethannicholasb3058bd2016-07-01 08:22:01 -0700410 // label of the current block, or 0 if we are not in a block
411 SpvId fCurrentBlock;
412 std::stack<SpvId> fBreakTarget;
413 std::stack<SpvId> fContinueTarget;
Greg Daniele6ab9982018-08-22 13:56:32 +0000414 SpvId fRTHeightStructId = (SpvId) -1;
415 SpvId fRTHeightFieldIndex = (SpvId) -1;
Jim Van Verthf3ec9832020-10-21 16:09:57 -0400416 SpvStorageClass_ fRTHeightStorageClass;
Ethan Nicholas8feeff92017-03-30 14:11:58 -0400417 // holds variables synthesized during output, for lifetime purposes
418 SymbolTable fSynthetics;
Ethan Nicholas5226b772018-05-03 16:20:41 -0400419 int fSkInCount = 1;
ethannicholasb3058bd2016-07-01 08:22:01 -0700420
421 friend class PointerLValue;
422 friend class SwizzleLValue;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500423
John Stiles7571f9e2020-09-02 22:42:33 -0400424 using INHERITED = CodeGenerator;
ethannicholasb3058bd2016-07-01 08:22:01 -0700425};
426
John Stilesa6841be2020-08-06 14:11:56 -0400427} // namespace SkSL
ethannicholasb3058bd2016-07-01 08:22:01 -0700428
429#endif