blob: b79194500a5f8213fc8b20c9b84365a72755b2d4 [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 */
7
8#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
Ethan Nicholas9e1138d2016-11-21 10:39:35 -050015#include "SkStream.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070016#include "SkSLCodeGenerator.h"
17#include "ir/SkSLBinaryExpression.h"
18#include "ir/SkSLBoolLiteral.h"
19#include "ir/SkSLConstructor.h"
20#include "ir/SkSLFloatLiteral.h"
21#include "ir/SkSLIfStatement.h"
22#include "ir/SkSLIndexExpression.h"
23#include "ir/SkSLInterfaceBlock.h"
24#include "ir/SkSLIntLiteral.h"
25#include "ir/SkSLFieldAccess.h"
26#include "ir/SkSLForStatement.h"
27#include "ir/SkSLFunctionCall.h"
28#include "ir/SkSLFunctionDeclaration.h"
29#include "ir/SkSLFunctionDefinition.h"
30#include "ir/SkSLPrefixExpression.h"
31#include "ir/SkSLPostfixExpression.h"
32#include "ir/SkSLProgramElement.h"
33#include "ir/SkSLReturnStatement.h"
34#include "ir/SkSLStatement.h"
35#include "ir/SkSLSwizzle.h"
36#include "ir/SkSLTernaryExpression.h"
ethannicholas22f939e2016-10-13 13:25:34 -070037#include "ir/SkSLVarDeclarations.h"
38#include "ir/SkSLVarDeclarationsStatement.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070039#include "ir/SkSLVariableReference.h"
40#include "spirv.h"
41
42namespace SkSL {
43
44#define kLast_Capability SpvCapabilityMultiViewport
45
46/**
47 * Converts a Program into a SPIR-V binary.
48 */
49class SPIRVCodeGenerator : public CodeGenerator {
50public:
51 class LValue {
52 public:
53 virtual ~LValue() {}
Greg Daniel64773e62016-11-22 09:44:03 -050054
ethannicholasb3058bd2016-07-01 08:22:01 -070055 // returns a pointer to the lvalue, if possible. If the lvalue cannot be directly referenced
56 // by a pointer (e.g. vector swizzles), returns 0.
57 virtual SpvId getPointer() = 0;
58
Ethan Nicholas9e1138d2016-11-21 10:39:35 -050059 virtual SpvId load(SkWStream& out) = 0;
ethannicholasb3058bd2016-07-01 08:22:01 -070060
Ethan Nicholas9e1138d2016-11-21 10:39:35 -050061 virtual void store(SpvId value, SkWStream& out) = 0;
ethannicholasb3058bd2016-07-01 08:22:01 -070062 };
63
ethannicholasd598f792016-07-25 10:08:54 -070064 SPIRVCodeGenerator(const Context* context)
65 : fContext(*context)
66 , fCapabilities(1 << SpvCapabilityShader)
ethannicholasb3058bd2016-07-01 08:22:01 -070067 , fIdCount(1)
68 , fBoolTrue(0)
69 , fBoolFalse(0)
70 , fCurrentBlock(0) {
71 this->setupIntrinsics();
72 }
73
Ethan Nicholas9e1138d2016-11-21 10:39:35 -050074 void generateCode(const Program& program, SkWStream& out) override;
ethannicholasb3058bd2016-07-01 08:22:01 -070075
76private:
77 enum IntrinsicKind {
78 kGLSL_STD_450_IntrinsicKind,
79 kSPIRV_IntrinsicKind,
80 kSpecial_IntrinsicKind
81 };
82
83 enum SpecialIntrinsic {
84 kAtan_SpecialIntrinsic,
85 kTexture_SpecialIntrinsic,
86 kTexture2D_SpecialIntrinsic,
Greg Daniel64773e62016-11-22 09:44:03 -050087 kTextureProj_SpecialIntrinsic,
88 kSubpassLoad_SpecialIntrinsic,
ethannicholasb3058bd2016-07-01 08:22:01 -070089 };
90
91 void setupIntrinsics();
92
93 SpvId nextId();
94
95 SpvId getType(const Type& type);
96
ethannicholasd598f792016-07-25 10:08:54 -070097 SpvId getFunctionType(const FunctionDeclaration& function);
ethannicholasb3058bd2016-07-01 08:22:01 -070098
ethannicholasd598f792016-07-25 10:08:54 -070099 SpvId getPointerType(const Type& type, SpvStorageClass_ storageClass);
ethannicholasb3058bd2016-07-01 08:22:01 -0700100
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500101 std::vector<SpvId> getAccessChain(const Expression& expr, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700102
103 void writeLayout(const Layout& layout, SpvId target);
104
105 void writeLayout(const Layout& layout, SpvId target, int member);
106
egdaniel988283c2016-11-16 07:29:51 -0800107 void writeStruct(const Type& type, SpvId resultId);
ethannicholasb3058bd2016-07-01 08:22:01 -0700108
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500109 void writeProgramElement(const ProgramElement& pe, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700110
ethannicholasf789b382016-08-03 12:43:36 -0700111 SpvId writeInterfaceBlock(const InterfaceBlock& intf);
ethannicholasb3058bd2016-07-01 08:22:01 -0700112
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500113 SpvId writeFunctionStart(const FunctionDeclaration& f, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700114
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500115 SpvId writeFunctionDeclaration(const FunctionDeclaration& f, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700116
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500117 SpvId writeFunction(const FunctionDefinition& f, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700118
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500119 void writeGlobalVars(Program::Kind kind, const VarDeclarations& v, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700120
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500121 void writeVarDeclarations(const VarDeclarations& decl, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700122
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500123 SpvId writeVariableReference(const VariableReference& ref, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700124
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500125 std::unique_ptr<LValue> getLValue(const Expression& value, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700126
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500127 SpvId writeExpression(const Expression& expr, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700128
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500129 SpvId writeIntrinsicCall(const FunctionCall& c, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700130
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500131 SpvId writeFunctionCall(const FunctionCall& c, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700132
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500133 SpvId writeSpecialIntrinsic(const FunctionCall& c, SpecialIntrinsic kind, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700134
ethannicholasf789b382016-08-03 12:43:36 -0700135 SpvId writeConstantVector(const Constructor& c);
ethannicholasb3058bd2016-07-01 08:22:01 -0700136
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500137 SpvId writeFloatConstructor(const Constructor& c, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700138
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500139 SpvId writeIntConstructor(const Constructor& c, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700140
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500141 SpvId writeMatrixConstructor(const Constructor& c, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700142
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500143 SpvId writeVectorConstructor(const Constructor& c, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700144
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500145 SpvId writeConstructor(const Constructor& c, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700146
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500147 SpvId writeFieldAccess(const FieldAccess& f, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700148
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500149 SpvId writeSwizzle(const Swizzle& swizzle, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700150
151 SpvId writeBinaryOperation(const Type& resultType, const Type& operandType, SpvId lhs,
152 SpvId rhs, SpvOp_ ifFloat, SpvOp_ ifInt, SpvOp_ ifUInt,
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500153 SpvOp_ ifBool, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700154
ethannicholasf789b382016-08-03 12:43:36 -0700155 SpvId writeBinaryOperation(const BinaryExpression& expr, SpvOp_ ifFloat, SpvOp_ ifInt,
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500156 SpvOp_ ifUInt, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700157
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500158 SpvId writeBinaryExpression(const BinaryExpression& b, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700159
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500160 SpvId writeTernaryExpression(const TernaryExpression& t, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700161
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500162 SpvId writeIndexExpression(const IndexExpression& expr, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700163
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500164 SpvId writeLogicalAnd(const BinaryExpression& b, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700165
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500166 SpvId writeLogicalOr(const BinaryExpression& o, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700167
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500168 SpvId writePrefixExpression(const PrefixExpression& p, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700169
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500170 SpvId writePostfixExpression(const PostfixExpression& p, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700171
ethannicholasf789b382016-08-03 12:43:36 -0700172 SpvId writeBoolLiteral(const BoolLiteral& b);
ethannicholasb3058bd2016-07-01 08:22:01 -0700173
ethannicholasf789b382016-08-03 12:43:36 -0700174 SpvId writeIntLiteral(const IntLiteral& i);
ethannicholasb3058bd2016-07-01 08:22:01 -0700175
ethannicholasf789b382016-08-03 12:43:36 -0700176 SpvId writeFloatLiteral(const FloatLiteral& f);
ethannicholasb3058bd2016-07-01 08:22:01 -0700177
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500178 void writeStatement(const Statement& s, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700179
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500180 void writeBlock(const Block& b, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700181
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500182 void writeIfStatement(const IfStatement& stmt, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700183
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500184 void writeForStatement(const ForStatement& f, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700185
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500186 void writeReturnStatement(const ReturnStatement& r, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700187
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500188 void writeCapabilities(SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700189
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500190 void writeInstructions(const Program& program, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700191
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500192 void writeOpCode(SpvOp_ opCode, int length, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700193
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500194 void writeWord(int32_t word, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700195
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500196 void writeString(const char* string, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700197
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500198 void writeLabel(SpvId id, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700199
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500200 void writeInstruction(SpvOp_ opCode, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700201
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500202 void writeInstruction(SpvOp_ opCode, const char* string, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700203
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500204 void writeInstruction(SpvOp_ opCode, int32_t word1, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700205
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500206 void writeInstruction(SpvOp_ opCode, int32_t word1, const char* string, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700207
208 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, const char* string,
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500209 SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700210
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500211 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700212
213 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3,
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500214 SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700215
216 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3, int32_t word4,
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500217 SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700218
219 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3, int32_t word4,
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500220 int32_t word5, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700221
222 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3, int32_t word4,
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500223 int32_t word5, int32_t word6, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700224
225 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3, int32_t word4,
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500226 int32_t word5, int32_t word6, int32_t word7, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700227
228 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3, int32_t word4,
229 int32_t word5, int32_t word6, int32_t word7, int32_t word8,
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500230 SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700231
ethannicholasd598f792016-07-25 10:08:54 -0700232 const Context& fContext;
233
ethannicholasb3058bd2016-07-01 08:22:01 -0700234 uint64_t fCapabilities;
235 SpvId fIdCount;
236 SpvId fGLSLExtendedInstructions;
237 typedef std::tuple<IntrinsicKind, int32_t, int32_t, int32_t, int32_t> Intrinsic;
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500238 std::unordered_map<SkString, Intrinsic> fIntrinsicMap;
ethannicholasd598f792016-07-25 10:08:54 -0700239 std::unordered_map<const FunctionDeclaration*, SpvId> fFunctionMap;
240 std::unordered_map<const Variable*, SpvId> fVariableMap;
241 std::unordered_map<const Variable*, int32_t> fInterfaceBlockMap;
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500242 std::unordered_map<SkString, SpvId> fTypeMap;
243 SkDynamicMemoryWStream fCapabilitiesBuffer;
244 SkDynamicMemoryWStream fGlobalInitializersBuffer;
245 SkDynamicMemoryWStream fConstantBuffer;
246 SkDynamicMemoryWStream fExternalFunctionsBuffer;
247 SkDynamicMemoryWStream fVariableBuffer;
248 SkDynamicMemoryWStream fNameBuffer;
249 SkDynamicMemoryWStream fDecorationBuffer;
ethannicholasb3058bd2016-07-01 08:22:01 -0700250
251 SpvId fBoolTrue;
252 SpvId fBoolFalse;
253 std::unordered_map<int64_t, SpvId> fIntConstants;
254 std::unordered_map<uint64_t, SpvId> fUIntConstants;
255 std::unordered_map<float, SpvId> fFloatConstants;
256 std::unordered_map<double, SpvId> fDoubleConstants;
257 // label of the current block, or 0 if we are not in a block
258 SpvId fCurrentBlock;
259 std::stack<SpvId> fBreakTarget;
260 std::stack<SpvId> fContinueTarget;
261
262 friend class PointerLValue;
263 friend class SwizzleLValue;
264};
265
266}
267
268#endif