blob: 562bf279e92ea6df0056272bcda2b303401b4f3b [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"
ethannicholas8ac838d2016-11-22 08:39:36 -080017#include "SkSLMemoryLayout.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070018#include "ir/SkSLBinaryExpression.h"
19#include "ir/SkSLBoolLiteral.h"
20#include "ir/SkSLConstructor.h"
Ethan Nicholasfd146aa2017-01-13 16:40:35 -050021#include "ir/SkSLDoStatement.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070022#include "ir/SkSLFloatLiteral.h"
23#include "ir/SkSLIfStatement.h"
24#include "ir/SkSLIndexExpression.h"
25#include "ir/SkSLInterfaceBlock.h"
26#include "ir/SkSLIntLiteral.h"
27#include "ir/SkSLFieldAccess.h"
28#include "ir/SkSLForStatement.h"
29#include "ir/SkSLFunctionCall.h"
30#include "ir/SkSLFunctionDeclaration.h"
31#include "ir/SkSLFunctionDefinition.h"
32#include "ir/SkSLPrefixExpression.h"
33#include "ir/SkSLPostfixExpression.h"
34#include "ir/SkSLProgramElement.h"
35#include "ir/SkSLReturnStatement.h"
36#include "ir/SkSLStatement.h"
37#include "ir/SkSLSwizzle.h"
38#include "ir/SkSLTernaryExpression.h"
ethannicholas22f939e2016-10-13 13:25:34 -070039#include "ir/SkSLVarDeclarations.h"
40#include "ir/SkSLVarDeclarationsStatement.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070041#include "ir/SkSLVariableReference.h"
Ethan Nicholasfd146aa2017-01-13 16:40:35 -050042#include "ir/SkSLWhileStatement.h"
ethannicholasb3058bd2016-07-01 08:22:01 -070043#include "spirv.h"
44
45namespace SkSL {
46
47#define kLast_Capability SpvCapabilityMultiViewport
48
49/**
50 * Converts a Program into a SPIR-V binary.
51 */
52class SPIRVCodeGenerator : public CodeGenerator {
53public:
54 class LValue {
55 public:
56 virtual ~LValue() {}
Greg Daniel64773e62016-11-22 09:44:03 -050057
ethannicholasb3058bd2016-07-01 08:22:01 -070058 // returns a pointer to the lvalue, if possible. If the lvalue cannot be directly referenced
59 // by a pointer (e.g. vector swizzles), returns 0.
60 virtual SpvId getPointer() = 0;
61
Ethan Nicholas9e1138d2016-11-21 10:39:35 -050062 virtual SpvId load(SkWStream& out) = 0;
ethannicholasb3058bd2016-07-01 08:22:01 -070063
Ethan Nicholas9e1138d2016-11-21 10:39:35 -050064 virtual void store(SpvId value, SkWStream& out) = 0;
ethannicholasb3058bd2016-07-01 08:22:01 -070065 };
66
Ethan Nicholas941e7e22016-12-12 15:33:30 -050067 SPIRVCodeGenerator(const Context* context, const Program* program, ErrorReporter* errors,
68 SkWStream* out)
69 : INHERITED(program, errors, out)
70 , fContext(*context)
ethannicholas8ac838d2016-11-22 08:39:36 -080071 , fDefaultLayout(MemoryLayout::k140_Standard)
ethannicholasd598f792016-07-25 10:08:54 -070072 , fCapabilities(1 << SpvCapabilityShader)
ethannicholasb3058bd2016-07-01 08:22:01 -070073 , fIdCount(1)
74 , fBoolTrue(0)
75 , fBoolFalse(0)
Ethan Nicholas941e7e22016-12-12 15:33:30 -050076 , fSetupFragPosition(false)
ethannicholasb3058bd2016-07-01 08:22:01 -070077 , fCurrentBlock(0) {
78 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,
92 kTexture_SpecialIntrinsic,
Greg Daniel64773e62016-11-22 09:44:03 -050093 kSubpassLoad_SpecialIntrinsic,
ethannicholasb3058bd2016-07-01 08:22:01 -070094 };
95
96 void setupIntrinsics();
97
98 SpvId nextId();
99
100 SpvId getType(const Type& type);
101
ethannicholas8ac838d2016-11-22 08:39:36 -0800102 SpvId getType(const Type& type, const MemoryLayout& layout);
103
ethannicholasd598f792016-07-25 10:08:54 -0700104 SpvId getFunctionType(const FunctionDeclaration& function);
ethannicholasb3058bd2016-07-01 08:22:01 -0700105
ethannicholasd598f792016-07-25 10:08:54 -0700106 SpvId getPointerType(const Type& type, SpvStorageClass_ storageClass);
ethannicholasb3058bd2016-07-01 08:22:01 -0700107
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500108 SpvId getPointerType(const Type& type, const MemoryLayout& layout,
ethannicholas8ac838d2016-11-22 08:39:36 -0800109 SpvStorageClass_ storageClass);
110
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500111 std::vector<SpvId> getAccessChain(const Expression& expr, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700112
113 void writeLayout(const Layout& layout, SpvId target);
114
115 void writeLayout(const Layout& layout, SpvId target, int member);
116
ethannicholas8ac838d2016-11-22 08:39:36 -0800117 void writeStruct(const Type& type, const MemoryLayout& layout, SpvId resultId);
ethannicholasb3058bd2016-07-01 08:22:01 -0700118
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500119 void writeProgramElement(const ProgramElement& pe, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700120
ethannicholasf789b382016-08-03 12:43:36 -0700121 SpvId writeInterfaceBlock(const InterfaceBlock& intf);
ethannicholasb3058bd2016-07-01 08:22:01 -0700122
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500123 SpvId writeFunctionStart(const FunctionDeclaration& f, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700124
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500125 SpvId writeFunctionDeclaration(const FunctionDeclaration& f, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700126
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500127 SpvId writeFunction(const FunctionDefinition& f, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700128
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500129 void writeGlobalVars(Program::Kind kind, const VarDeclarations& v, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700130
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500131 void writeVarDeclarations(const VarDeclarations& decl, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700132
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500133 SpvId writeVariableReference(const VariableReference& ref, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700134
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500135 std::unique_ptr<LValue> getLValue(const Expression& value, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700136
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500137 SpvId writeExpression(const Expression& expr, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700138
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500139 SpvId writeIntrinsicCall(const FunctionCall& c, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700140
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500141 SpvId writeFunctionCall(const FunctionCall& c, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700142
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500143 SpvId writeSpecialIntrinsic(const FunctionCall& c, SpecialIntrinsic kind, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700144
ethannicholasf789b382016-08-03 12:43:36 -0700145 SpvId writeConstantVector(const Constructor& c);
ethannicholasb3058bd2016-07-01 08:22:01 -0700146
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500147 SpvId writeFloatConstructor(const Constructor& c, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700148
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500149 SpvId writeIntConstructor(const Constructor& c, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700150
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500151 SpvId writeMatrixConstructor(const Constructor& c, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700152
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500153 SpvId writeVectorConstructor(const Constructor& c, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700154
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500155 SpvId writeConstructor(const Constructor& c, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700156
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500157 SpvId writeFieldAccess(const FieldAccess& f, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700158
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500159 SpvId writeSwizzle(const Swizzle& swizzle, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700160
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500161 SpvId writeBinaryOperation(const Type& resultType, const Type& operandType, SpvId lhs,
162 SpvId rhs, SpvOp_ ifFloat, SpvOp_ ifInt, SpvOp_ ifUInt,
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500163 SpvOp_ ifBool, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700164
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500165 SpvId writeBinaryOperation(const BinaryExpression& expr, SpvOp_ ifFloat, SpvOp_ ifInt,
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500166 SpvOp_ ifUInt, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700167
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500168 SpvId writeBinaryExpression(const BinaryExpression& b, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700169
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500170 SpvId writeTernaryExpression(const TernaryExpression& t, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700171
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500172 SpvId writeIndexExpression(const IndexExpression& expr, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700173
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500174 SpvId writeLogicalAnd(const BinaryExpression& b, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700175
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500176 SpvId writeLogicalOr(const BinaryExpression& o, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700177
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500178 SpvId writePrefixExpression(const PrefixExpression& p, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700179
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500180 SpvId writePostfixExpression(const PostfixExpression& p, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700181
ethannicholasf789b382016-08-03 12:43:36 -0700182 SpvId writeBoolLiteral(const BoolLiteral& b);
ethannicholasb3058bd2016-07-01 08:22:01 -0700183
ethannicholasf789b382016-08-03 12:43:36 -0700184 SpvId writeIntLiteral(const IntLiteral& i);
ethannicholasb3058bd2016-07-01 08:22:01 -0700185
ethannicholasf789b382016-08-03 12:43:36 -0700186 SpvId writeFloatLiteral(const FloatLiteral& f);
ethannicholasb3058bd2016-07-01 08:22:01 -0700187
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500188 void writeStatement(const Statement& s, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700189
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500190 void writeBlock(const Block& b, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700191
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500192 void writeIfStatement(const IfStatement& stmt, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700193
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500194 void writeForStatement(const ForStatement& f, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700195
Ethan Nicholasfd146aa2017-01-13 16:40:35 -0500196 void writeWhileStatement(const WhileStatement& w, SkWStream& out);
197
198 void writeDoStatement(const DoStatement& d, SkWStream& out);
199
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500200 void writeReturnStatement(const ReturnStatement& r, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700201
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500202 void writeCapabilities(SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700203
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500204 void writeInstructions(const Program& program, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700205
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500206 void writeOpCode(SpvOp_ opCode, int length, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700207
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500208 void writeWord(int32_t word, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700209
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500210 void writeString(const char* string, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700211
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500212 void writeLabel(SpvId id, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700213
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500214 void writeInstruction(SpvOp_ opCode, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700215
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500216 void writeInstruction(SpvOp_ opCode, const char* string, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700217
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500218 void writeInstruction(SpvOp_ opCode, int32_t word1, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700219
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500220 void writeInstruction(SpvOp_ opCode, int32_t word1, const char* string, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700221
222 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, const char* string,
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500223 SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700224
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500225 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700226
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500227 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3,
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500228 SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700229
230 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3, int32_t word4,
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500231 SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700232
233 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3, int32_t word4,
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500234 int32_t word5, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700235
236 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3, int32_t word4,
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500237 int32_t word5, int32_t word6, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700238
239 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3, int32_t word4,
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500240 int32_t word5, int32_t word6, int32_t word7, SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700241
242 void writeInstruction(SpvOp_ opCode, int32_t word1, int32_t word2, int32_t word3, int32_t word4,
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500243 int32_t word5, int32_t word6, int32_t word7, int32_t word8,
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500244 SkWStream& out);
ethannicholasb3058bd2016-07-01 08:22:01 -0700245
ethannicholasd598f792016-07-25 10:08:54 -0700246 const Context& fContext;
ethannicholas8ac838d2016-11-22 08:39:36 -0800247 const MemoryLayout fDefaultLayout;
ethannicholasd598f792016-07-25 10:08:54 -0700248
ethannicholasb3058bd2016-07-01 08:22:01 -0700249 uint64_t fCapabilities;
250 SpvId fIdCount;
251 SpvId fGLSLExtendedInstructions;
252 typedef std::tuple<IntrinsicKind, int32_t, int32_t, int32_t, int32_t> Intrinsic;
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500253 std::unordered_map<SkString, Intrinsic> fIntrinsicMap;
ethannicholasd598f792016-07-25 10:08:54 -0700254 std::unordered_map<const FunctionDeclaration*, SpvId> fFunctionMap;
255 std::unordered_map<const Variable*, SpvId> fVariableMap;
256 std::unordered_map<const Variable*, int32_t> fInterfaceBlockMap;
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500257 std::unordered_map<SkString, SpvId> fTypeMap;
258 SkDynamicMemoryWStream fCapabilitiesBuffer;
259 SkDynamicMemoryWStream fGlobalInitializersBuffer;
260 SkDynamicMemoryWStream fConstantBuffer;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500261 SkDynamicMemoryWStream fExtraGlobalsBuffer;
Ethan Nicholas9e1138d2016-11-21 10:39:35 -0500262 SkDynamicMemoryWStream fExternalFunctionsBuffer;
263 SkDynamicMemoryWStream fVariableBuffer;
264 SkDynamicMemoryWStream fNameBuffer;
265 SkDynamicMemoryWStream fDecorationBuffer;
ethannicholasb3058bd2016-07-01 08:22:01 -0700266
267 SpvId fBoolTrue;
268 SpvId fBoolFalse;
269 std::unordered_map<int64_t, SpvId> fIntConstants;
270 std::unordered_map<uint64_t, SpvId> fUIntConstants;
271 std::unordered_map<float, SpvId> fFloatConstants;
272 std::unordered_map<double, SpvId> fDoubleConstants;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500273 bool fSetupFragPosition;
ethannicholasb3058bd2016-07-01 08:22:01 -0700274 // label of the current block, or 0 if we are not in a block
275 SpvId fCurrentBlock;
276 std::stack<SpvId> fBreakTarget;
277 std::stack<SpvId> fContinueTarget;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500278 SpvId fRTHeightStructId = (SpvId) -1;
279 SpvId fRTHeightFieldIndex = (SpvId) -1;
ethannicholasb3058bd2016-07-01 08:22:01 -0700280
281 friend class PointerLValue;
282 friend class SwizzleLValue;
Ethan Nicholas941e7e22016-12-12 15:33:30 -0500283
284 typedef CodeGenerator INHERITED;
ethannicholasb3058bd2016-07-01 08:22:01 -0700285};
286
287}
288
289#endif