blob: 84c582e53e932aab3b9208acf9bb6131c63592e6 [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 Nicholasd8df21a2016-11-17 16:13:37 -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() {}
54
55 // 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 Nicholasd8df21a2016-11-17 16:13:37 -050059 virtual SpvId load(SkWStream& out) = 0;
ethannicholasb3058bd2016-07-01 08:22:01 -070060
Ethan Nicholasd8df21a2016-11-17 16:13:37 -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 Nicholasd8df21a2016-11-17 16:13:37 -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,
87 kTextureProj_SpecialIntrinsic
88 };
89
90 void setupIntrinsics();
91
92 SpvId nextId();
93
94 SpvId getType(const Type& type);
95
ethannicholasd598f792016-07-25 10:08:54 -070096 SpvId getFunctionType(const FunctionDeclaration& function);
ethannicholasb3058bd2016-07-01 08:22:01 -070097
ethannicholasd598f792016-07-25 10:08:54 -070098 SpvId getPointerType(const Type& type, SpvStorageClass_ storageClass);
ethannicholasb3058bd2016-07-01 08:22:01 -070099
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500100 std::vector<SpvId> getAccessChain(const Expression& expr, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700101
102 void writeLayout(const Layout& layout, SpvId target);
103
104 void writeLayout(const Layout& layout, SpvId target, int member);
105
egdaniel988283c2016-11-16 07:29:51 -0800106 void writeStruct(const Type& type, SpvId resultId);
ethannicholasb3058bd2016-07-01 08:22:01 -0700107
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500108 void writeProgramElement(const ProgramElement& pe, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700109
ethannicholasf789b382016-08-03 12:43:36 -0700110 SpvId writeInterfaceBlock(const InterfaceBlock& intf);
ethannicholasb3058bd2016-07-01 08:22:01 -0700111
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500112 SpvId writeFunctionStart(const FunctionDeclaration& f, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700113
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500114 SpvId writeFunctionDeclaration(const FunctionDeclaration& f, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700115
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500116 SpvId writeFunction(const FunctionDefinition& f, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700117
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500118 void writeGlobalVars(Program::Kind kind, const VarDeclarations& v, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700119
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500120 void writeVarDeclarations(const VarDeclarations& decl, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700121
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500122 SpvId writeVariableReference(const VariableReference& ref, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700123
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500124 std::unique_ptr<LValue> getLValue(const Expression& value, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700125
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500126 SpvId writeExpression(const Expression& expr, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700127
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500128 SpvId writeIntrinsicCall(const FunctionCall& c, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700129
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500130 SpvId writeFunctionCall(const FunctionCall& c, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700131
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500132 SpvId writeSpecialIntrinsic(const FunctionCall& c, SpecialIntrinsic kind, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700133
ethannicholasf789b382016-08-03 12:43:36 -0700134 SpvId writeConstantVector(const Constructor& c);
ethannicholasb3058bd2016-07-01 08:22:01 -0700135
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500136 SpvId writeFloatConstructor(const Constructor& c, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700137
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500138 SpvId writeIntConstructor(const Constructor& c, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700139
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500140 SpvId writeMatrixConstructor(const Constructor& c, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700141
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500142 SpvId writeVectorConstructor(const Constructor& c, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700143
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500144 SpvId writeConstructor(const Constructor& c, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700145
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500146 SpvId writeFieldAccess(const FieldAccess& f, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700147
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500148 SpvId writeSwizzle(const Swizzle& swizzle, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700149
150 SpvId writeBinaryOperation(const Type& resultType, const Type& operandType, SpvId lhs,
151 SpvId rhs, SpvOp_ ifFloat, SpvOp_ ifInt, SpvOp_ ifUInt,
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500152 SpvOp_ ifBool, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700153
ethannicholasf789b382016-08-03 12:43:36 -0700154 SpvId writeBinaryOperation(const BinaryExpression& expr, SpvOp_ ifFloat, SpvOp_ ifInt,
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500155 SpvOp_ ifUInt, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700156
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500157 SpvId writeBinaryExpression(const BinaryExpression& b, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700158
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500159 SpvId writeTernaryExpression(const TernaryExpression& t, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700160
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500161 SpvId writeIndexExpression(const IndexExpression& expr, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700162
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500163 SpvId writeLogicalAnd(const BinaryExpression& b, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700164
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500165 SpvId writeLogicalOr(const BinaryExpression& o, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700166
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500167 SpvId writePrefixExpression(const PrefixExpression& p, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700168
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500169 SpvId writePostfixExpression(const PostfixExpression& p, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700170
ethannicholasf789b382016-08-03 12:43:36 -0700171 SpvId writeBoolLiteral(const BoolLiteral& b);
ethannicholasb3058bd2016-07-01 08:22:01 -0700172
ethannicholasf789b382016-08-03 12:43:36 -0700173 SpvId writeIntLiteral(const IntLiteral& i);
ethannicholasb3058bd2016-07-01 08:22:01 -0700174
ethannicholasf789b382016-08-03 12:43:36 -0700175 SpvId writeFloatLiteral(const FloatLiteral& f);
ethannicholasb3058bd2016-07-01 08:22:01 -0700176
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500177 void writeStatement(const Statement& s, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700178
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500179 void writeBlock(const Block& b, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700180
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500181 void writeIfStatement(const IfStatement& stmt, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700182
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500183 void writeForStatement(const ForStatement& f, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700184
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500185 void writeReturnStatement(const ReturnStatement& r, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700186
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500187 void writeCapabilities(SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700188
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500189 void writeInstructions(const Program& program, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700190
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500191 void writeOpCode(SpvOp_ opCode, int length, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700192
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500193 void writeWord(int32_t word, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700194
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500195 void writeString(const char* string, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700196
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500197 void writeLabel(SpvId id, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700198
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500199 void writeInstruction(SpvOp_ opCode, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700200
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500201 void writeInstruction(SpvOp_ opCode, const char* string, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700202
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500203 void writeInstruction(SpvOp_ opCode, int32_t word1, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700204
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500205 void writeInstruction(SpvOp_ opCode, int32_t word1, const char* string, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700206
207 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, const char* string,
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500208 SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700209
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500210 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700211
212 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3,
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500213 SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700214
215 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3, int32_t word4,
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500216 SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700217
218 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3, int32_t word4,
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500219 int32_t word5, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700220
221 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3, int32_t word4,
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500222 int32_t word5, int32_t word6, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700223
224 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3, int32_t word4,
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500225 int32_t word5, int32_t word6, int32_t word7, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700226
227 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3, int32_t word4,
228 int32_t word5, int32_t word6, int32_t word7, int32_t word8,
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500229 SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700230
ethannicholasd598f792016-07-25 10:08:54 -0700231 const Context& fContext;
232
ethannicholasb3058bd2016-07-01 08:22:01 -0700233 uint64_t fCapabilities;
234 SpvId fIdCount;
235 SpvId fGLSLExtendedInstructions;
236 typedef std::tuple<IntrinsicKind, int32_t, int32_t, int32_t, int32_t> Intrinsic;
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500237 std::unordered_map<SkString, Intrinsic> fIntrinsicMap;
ethannicholasd598f792016-07-25 10:08:54 -0700238 std::unordered_map<const FunctionDeclaration*, SpvId> fFunctionMap;
239 std::unordered_map<const Variable*, SpvId> fVariableMap;
240 std::unordered_map<const Variable*, int32_t> fInterfaceBlockMap;
Ethan Nicholasd8df21a2016-11-17 16:13:37 -0500241 std::unordered_map<SkString, SpvId> fTypeMap;
242 SkDynamicMemoryWStream fCapabilitiesBuffer;
243 SkDynamicMemoryWStream fGlobalInitializersBuffer;
244 SkDynamicMemoryWStream fConstantBuffer;
245 SkDynamicMemoryWStream fExternalFunctionsBuffer;
246 SkDynamicMemoryWStream fVariableBuffer;
247 SkDynamicMemoryWStream fNameBuffer;
248 SkDynamicMemoryWStream fDecorationBuffer;
ethannicholasb3058bd2016-07-01 08:22:01 -0700249
250 SpvId fBoolTrue;
251 SpvId fBoolFalse;
252 std::unordered_map<int64_t, SpvId> fIntConstants;
253 std::unordered_map<uint64_t, SpvId> fUIntConstants;
254 std::unordered_map<float, SpvId> fFloatConstants;
255 std::unordered_map<double, SpvId> fDoubleConstants;
256 // label of the current block, or 0 if we are not in a block
257 SpvId fCurrentBlock;
258 std::stack<SpvId> fBreakTarget;
259 std::stack<SpvId> fContinueTarget;
260
261 friend class PointerLValue;
262 friend class SwizzleLValue;
263};
264
265}
266
267#endif