blob: 46a721f70cc6c5c29266eb3fef69bac8da9cdaa9 [file] [log] [blame]
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +00001//
Nicolas Capensb1f45b72013-12-19 17:37:19 -05002// Copyright (c) 2002-2014 The ANGLE Project Authors. All rights reserved.
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +00003// Use of this source code is governed by a BSD-style license that can be
4// found in the LICENSE file.
5//
6
Geoff Lang0a73dd82014-11-19 16:18:08 -05007#ifndef COMPILER_TRANSLATOR_OUTPUTHLSL_H_
8#define COMPILER_TRANSLATOR_OUTPUTHLSL_H_
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +00009
alokp@chromium.org4e89d232010-05-14 19:37:21 +000010#include <list>
daniel@transgaming.com652468c2012-12-20 21:11:57 +000011#include <map>
Jamie Madill32aab012015-01-27 14:12:26 -050012#include <stack>
daniel@transgaming.com652468c2012-12-20 21:11:57 +000013
Jamie Madillf51639a2014-06-25 16:04:57 -040014#include "angle_gl.h"
Corentin Wallezf4eab3b2015-03-18 12:55:45 -070015#include "compiler/translator/ASTMetadataHLSL.h"
Jamie Madillb1a85f42014-08-19 15:23:24 -040016#include "compiler/translator/IntermNode.h"
Jamie Madill6b9cb252013-10-17 10:45:47 -040017#include "compiler/translator/ParseContext.h"
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000018
Olli Etuaho8efc5ad2015-03-03 17:21:10 +020019class BuiltInFunctionEmulator;
Olli Etuahoe17e3192015-01-02 12:47:59 +020020
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000021namespace sh
22{
Corentin Wallezc2ed9382016-04-15 13:29:25 +000023class StructureHLSL;
Olli Etuaho5858f7e2016-04-08 13:08:46 +030024class TextureFunctionHLSL;
25class UnfoldShortCircuit;
Jamie Madillf91ce812014-06-13 10:04:34 -040026class UniformHLSL;
27
Jamie Madilld7b1ab52016-12-12 14:42:19 -050028typedef std::map<TString, TIntermSymbol *> ReferencedSymbols;
daniel@transgaming.comb5875982010-04-15 20:44:53 +000029
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000030class OutputHLSL : public TIntermTraverser
31{
32 public:
Qiankun Miao7ebb97f2016-09-08 18:01:50 +080033 OutputHLSL(sh::GLenum shaderType,
34 int shaderVersion,
35 const TExtensionBehavior &extensionBehavior,
36 const char *sourcePath,
37 ShShaderOutput outputType,
38 int numRenderTargets,
39 const std::vector<Uniform> &uniforms,
40 ShCompileOptions compileOptions);
Olli Etuahoa3a5cc62015-02-13 13:12:22 +020041
daniel@transgaming.comb5875982010-04-15 20:44:53 +000042 ~OutputHLSL();
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000043
Olli Etuahoa3a5cc62015-02-13 13:12:22 +020044 void output(TIntermNode *treeRoot, TInfoSinkBase &objSink);
daniel@transgaming.com950f9932010-04-13 03:26:14 +000045
Jamie Madill4e1fd412014-07-10 17:50:10 -040046 const std::map<std::string, unsigned int> &getInterfaceBlockRegisterMap() const;
Jamie Madill9fe25e92014-07-18 10:33:08 -040047 const std::map<std::string, unsigned int> &getUniformRegisterMap() const;
Jamie Madill4e1fd412014-07-10 17:50:10 -040048
daniel@transgaming.comb5875982010-04-15 20:44:53 +000049 static TString initializer(const TType &type);
daniel@transgaming.comb5875982010-04-15 20:44:53 +000050
Jamie Madilld7b1ab52016-12-12 14:42:19 -050051 TInfoSinkBase &getInfoSink()
52 {
53 ASSERT(!mInfoSinkStack.empty());
54 return *mInfoSinkStack.top();
55 }
Jamie Madill32aab012015-01-27 14:12:26 -050056
Olli Etuaho18b9deb2015-11-05 12:14:50 +020057 static bool canWriteAsHLSLLiteral(TIntermTyped *expression);
58
daniel@transgaming.com950f9932010-04-13 03:26:14 +000059 protected:
Jamie Madill8c46ab12015-12-07 16:39:19 -050060 void header(TInfoSinkBase &out, const BuiltInFunctionEmulator *builtInFunctionEmulator);
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000061
Olli Etuaho56a2f952016-12-08 12:16:27 +000062 void writeFloat(TInfoSinkBase &out, float f);
63 void writeSingleConstant(TInfoSinkBase &out, const TConstantUnion *const constUnion);
64 const TConstantUnion *writeConstantUnionArray(TInfoSinkBase &out,
65 const TConstantUnion *const constUnion,
66 const size_t size);
67
daniel@transgaming.com950f9932010-04-13 03:26:14 +000068 // Visit AST nodes and output their code to the body stream
Jamie Madilld7b1ab52016-12-12 14:42:19 -050069 void visitSymbol(TIntermSymbol *);
70 void visitRaw(TIntermRaw *);
71 void visitConstantUnion(TIntermConstantUnion *);
Olli Etuahob6fa0432016-09-28 16:28:05 +010072 bool visitSwizzle(Visit visit, TIntermSwizzle *node) override;
Jamie Madilld7b1ab52016-12-12 14:42:19 -050073 bool visitBinary(Visit visit, TIntermBinary *);
74 bool visitUnary(Visit visit, TIntermUnary *);
Olli Etuahod0bad2c2016-09-09 18:01:16 +030075 bool visitTernary(Visit visit, TIntermTernary *);
Olli Etuaho57961272016-09-14 13:57:46 +030076 bool visitIfElse(Visit visit, TIntermIfElse *);
Olli Etuaho3c1dfb52015-02-20 11:34:03 +020077 bool visitSwitch(Visit visit, TIntermSwitch *);
78 bool visitCase(Visit visit, TIntermCase *);
Olli Etuaho16c745a2017-01-16 17:02:27 +000079 bool visitFunctionPrototype(Visit visit, TIntermFunctionPrototype *node) override;
Olli Etuaho336b1472016-10-05 16:37:55 +010080 bool visitFunctionDefinition(Visit visit, TIntermFunctionDefinition *node) override;
Jamie Madilld7b1ab52016-12-12 14:42:19 -050081 bool visitAggregate(Visit visit, TIntermAggregate *);
Olli Etuaho6d40bbd2016-09-30 13:49:38 +010082 bool visitBlock(Visit visit, TIntermBlock *node);
Olli Etuahobf4e1b72016-12-09 11:30:15 +000083 bool visitInvariantDeclaration(Visit visit, TIntermInvariantDeclaration *node);
Olli Etuaho13389b62016-10-16 11:48:18 +010084 bool visitDeclaration(Visit visit, TIntermDeclaration *node);
Jamie Madilld7b1ab52016-12-12 14:42:19 -050085 bool visitLoop(Visit visit, TIntermLoop *);
86 bool visitBranch(Visit visit, TIntermBranch *);
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000087
Jamie Madill8c46ab12015-12-07 16:39:19 -050088 bool handleExcessiveLoop(TInfoSinkBase &out, TIntermLoop *node);
Daniel Bratell29190082015-02-20 16:42:54 +010089
Jamie Madilld7b1ab52016-12-12 14:42:19 -050090 // Emit one of three strings depending on traverse phase. Called with literal strings so using
91 // const char* instead of TString.
Jamie Madill8c46ab12015-12-07 16:39:19 -050092 void outputTriplet(TInfoSinkBase &out,
93 Visit visit,
94 const char *preString,
95 const char *inString,
96 const char *postString);
97 void outputLineDirective(TInfoSinkBase &out, int line);
daniel@transgaming.com005c7392010-04-15 20:45:27 +000098 TString argumentString(const TIntermSymbol *symbol);
daniel@transgaming.com0b6b8342010-04-26 15:33:45 +000099 int vectorSize(const TType &type) const;
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000100
Daniel Bratell29190082015-02-20 16:42:54 +0100101 // Emit constructor. Called with literal names so using const char* instead of TString.
Jamie Madill8c46ab12015-12-07 16:39:19 -0500102 void outputConstructor(TInfoSinkBase &out,
103 Visit visit,
104 const TType &type,
105 const char *name,
106 const TIntermSequence *parameters);
107 const TConstantUnion *writeConstantUnion(TInfoSinkBase &out,
108 const TType &type,
109 const TConstantUnion *constUnion);
daniel@transgaming.com63691862010-04-29 03:32:42 +0000110
Olli Etuaho7fb49552015-03-18 17:27:44 +0200111 void outputEqual(Visit visit, const TType &type, TOperator op, TInfoSinkBase &out);
112
Olli Etuahod68924e2017-01-02 17:34:40 +0000113 void writeEmulatedFunctionTriplet(TInfoSinkBase &out, Visit visit, TOperator op);
Jamie Madill37997142015-01-28 10:06:34 -0500114 void makeFlaggedStructMaps(const std::vector<TIntermTyped *> &flaggedStructs);
115
Jamie Madilld7b1ab52016-12-12 14:42:19 -0500116 // Returns true if it found a 'same symbol' initializer (initializer that references the
117 // variable it's initting)
118 bool writeSameSymbolInitializer(TInfoSinkBase &out,
119 TIntermSymbol *symbolNode,
120 TIntermTyped *expression);
Olli Etuaho18b9deb2015-11-05 12:14:50 +0200121 // Returns true if variable initializer could be written using literal {} notation.
122 bool writeConstantInitialization(TInfoSinkBase &out,
123 TIntermSymbol *symbolNode,
124 TIntermTyped *expression);
125
Olli Etuaho57961272016-09-14 13:57:46 +0300126 void writeIfElse(TInfoSinkBase &out, TIntermIfElse *node);
Olli Etuaho5c9cd3d2014-12-18 13:04:25 +0200127
Jamie Madill55e79e02015-02-09 15:35:00 -0500128 // Returns the function name
129 TString addStructEqualityFunction(const TStructure &structure);
Olli Etuaho7fb49552015-03-18 17:27:44 +0200130 TString addArrayEqualityFunction(const TType &type);
Olli Etuaho12690762015-03-31 12:55:28 +0300131 TString addArrayAssignmentFunction(const TType &type);
Olli Etuaho9638c352015-04-01 14:34:52 +0300132 TString addArrayConstructIntoFunction(const TType &type);
Jamie Madill55e79e02015-02-09 15:35:00 -0500133
Jamie Madill2e295e22015-04-29 10:41:33 -0400134 // Ensures if the type is a struct, the struct is defined
135 void ensureStructDefined(const TType &type);
136
Olli Etuahoa3a5cc62015-02-13 13:12:22 +0200137 sh::GLenum mShaderType;
138 int mShaderVersion;
139 const TExtensionBehavior &mExtensionBehavior;
140 const char *mSourcePath;
shannon.woods@transgaming.comb73964e2013-01-25 21:49:14 +0000141 const ShShaderOutput mOutputType;
Qiankun Miao7ebb97f2016-09-08 18:01:50 +0800142 ShCompileOptions mCompileOptions;
Olli Etuahoa3a5cc62015-02-13 13:12:22 +0200143
daniel@transgaming.comf9ef1072010-04-22 13:35:16 +0000144 bool mInsideFunction;
daniel@transgaming.com950f9932010-04-13 03:26:14 +0000145
146 // Output streams
147 TInfoSinkBase mHeader;
148 TInfoSinkBase mBody;
149 TInfoSinkBase mFooter;
daniel@transgaming.comd91cfe72010-04-13 03:26:17 +0000150
Jamie Madilld7b1ab52016-12-12 14:42:19 -0500151 // A stack is useful when we want to traverse in the header, or in helper functions, but not
152 // always write to the body. Instead use an InfoSink stack to keep our current state intact.
Olli Etuaho7fb49552015-03-18 17:27:44 +0200153 // TODO (jmadill): Just passing an InfoSink in function parameters would be simpler.
Jamie Madill32aab012015-01-27 14:12:26 -0500154 std::stack<TInfoSinkBase *> mInfoSinkStack;
155
daniel@transgaming.com8803b852012-12-20 21:11:47 +0000156 ReferencedSymbols mReferencedUniforms;
shannonwoods@chromium.org4a643ae2013-05-30 00:12:27 +0000157 ReferencedSymbols mReferencedInterfaceBlocks;
daniel@transgaming.com8803b852012-12-20 21:11:47 +0000158 ReferencedSymbols mReferencedAttributes;
159 ReferencedSymbols mReferencedVaryings;
Jamie Madill46131a32013-06-20 11:55:50 -0400160 ReferencedSymbols mReferencedOutputVariables;
daniel@transgaming.com86f7c9d2010-04-20 18:52:06 +0000161
Jamie Madill8daaba12014-06-13 10:04:33 -0400162 StructureHLSL *mStructureHLSL;
Jamie Madillf91ce812014-06-13 10:04:34 -0400163 UniformHLSL *mUniformHLSL;
Olli Etuaho5858f7e2016-04-08 13:08:46 +0300164 TextureFunctionHLSL *mTextureFunctionHLSL;
Nicolas Capense0ba27a2013-06-24 16:10:52 -0400165
daniel@transgaming.comd91cfe72010-04-13 03:26:17 +0000166 // Parameters determining what goes in the header output
shannon.woods%transgaming.com@gtempaccount.comaa8b5cf2013-04-13 03:31:55 +0000167 bool mUsesFragColor;
168 bool mUsesFragData;
daniel@transgaming.comd7c98102010-05-14 17:30:48 +0000169 bool mUsesDepthRange;
daniel@transgaming.com4af7acc2010-05-14 17:30:53 +0000170 bool mUsesFragCoord;
171 bool mUsesPointCoord;
172 bool mUsesFrontFacing;
173 bool mUsesPointSize;
Gregoire Payen de La Garanderieb3dced22015-01-12 14:54:55 +0000174 bool mUsesInstanceID;
Corentin Wallezb076add2016-01-11 16:45:46 -0500175 bool mUsesVertexID;
Jamie Madill2aeb26a2013-07-08 14:02:55 -0400176 bool mUsesFragDepth;
Xinghua Caob1239382016-12-13 15:07:05 +0800177 bool mUsesNumWorkGroups;
178 bool mUsesWorkGroupID;
179 bool mUsesLocalInvocationID;
180 bool mUsesGlobalInvocationID;
181 bool mUsesLocalInvocationIndex;
daniel@transgaming.comd7c98102010-05-14 17:30:48 +0000182 bool mUsesXor;
Jamie Madill3c9eeb92013-11-04 11:09:26 -0500183 bool mUsesDiscardRewriting;
Nicolas Capens655fe362014-04-11 13:12:34 -0400184 bool mUsesNestedBreak;
Arun Patole44efa0b2015-03-04 17:11:05 +0530185 bool mRequiresIEEEStrictCompiling;
186
shannon.woods%transgaming.com@gtempaccount.comaa8b5cf2013-04-13 03:31:55 +0000187 int mNumRenderTargets;
188
Jamie Madilld7b1ab52016-12-12 14:42:19 -0500189 int mUniqueIndex; // For creating unique names
daniel@transgaming.com89431aa2012-05-31 01:20:29 +0000190
Corentin Wallez1239ee92015-03-19 14:38:02 -0700191 CallDAG mCallDag;
192 MetadataList mASTMetadataList;
193 ASTMetadataHLSL *mCurrentFunctionMetadata;
daniel@transgaming.com89431aa2012-05-31 01:20:29 +0000194 bool mOutputLod0Function;
daniel@transgaming.come11100c2012-05-31 01:20:32 +0000195 bool mInsideDiscontinuousLoop;
Nicolas Capens655fe362014-04-11 13:12:34 -0400196 int mNestedLoopDepth;
daniel@transgaming.come9b3f602012-07-11 20:37:31 +0000197
198 TIntermSymbol *mExcessiveLoopIndex;
daniel@transgaming.com652468c2012-12-20 21:11:57 +0000199
Jamie Madilld7b1ab52016-12-12 14:42:19 -0500200 TString structInitializerString(int indent,
201 const TStructure &structure,
202 const TString &rhsStructName);
daniel@transgaming.comf4d9fef2012-12-20 21:12:13 +0000203
Jamie Madilld7b1ab52016-12-12 14:42:19 -0500204 std::map<TIntermTyped *, TString> mFlaggedStructMappedNames;
205 std::map<TIntermTyped *, TString> mFlaggedStructOriginalNames;
Jamie Madill570e04d2013-06-21 09:15:33 -0400206
Olli Etuaho12690762015-03-31 12:55:28 +0300207 struct HelperFunction
Olli Etuahoae37a5c2015-03-20 16:50:15 +0200208 {
209 TString functionName;
210 TString functionDefinition;
211
Olli Etuaho12690762015-03-31 12:55:28 +0300212 virtual ~HelperFunction() {}
Olli Etuahoae37a5c2015-03-20 16:50:15 +0200213 };
214
215 // A list of all equality comparison functions. It's important to preserve the order at
216 // which we add the functions, since nested structures call each other recursively, and
217 // structure equality functions may need to call array equality functions and vice versa.
218 // The ownership of the pointers is maintained by the type-specific arrays.
Jamie Madilld7b1ab52016-12-12 14:42:19 -0500219 std::vector<HelperFunction *> mEqualityFunctions;
Olli Etuahoae37a5c2015-03-20 16:50:15 +0200220
Olli Etuaho12690762015-03-31 12:55:28 +0300221 struct StructEqualityFunction : public HelperFunction
Jamie Madill55e79e02015-02-09 15:35:00 -0500222 {
223 const TStructure *structure;
Jamie Madill55e79e02015-02-09 15:35:00 -0500224 };
Jamie Madilld7b1ab52016-12-12 14:42:19 -0500225 std::vector<StructEqualityFunction *> mStructEqualityFunctions;
Jamie Madill55e79e02015-02-09 15:35:00 -0500226
Olli Etuaho12690762015-03-31 12:55:28 +0300227 struct ArrayHelperFunction : public HelperFunction
Olli Etuaho7fb49552015-03-18 17:27:44 +0200228 {
229 TType type;
Olli Etuaho7fb49552015-03-18 17:27:44 +0200230 };
Jamie Madilld7b1ab52016-12-12 14:42:19 -0500231 std::vector<ArrayHelperFunction *> mArrayEqualityFunctions;
Olli Etuaho12690762015-03-31 12:55:28 +0300232
233 std::vector<ArrayHelperFunction> mArrayAssignmentFunctions;
Olli Etuaho9638c352015-04-01 14:34:52 +0300234
Jamie Madilld7b1ab52016-12-12 14:42:19 -0500235 // The construct-into functions are functions that fill an N-element array passed as an out
236 // parameter with the other N parameters of the function. This is used to work around that
237 // arrays can't be return values in HLSL.
Olli Etuaho9638c352015-04-01 14:34:52 +0300238 std::vector<ArrayHelperFunction> mArrayConstructIntoFunctions;
Olli Etuaho96963162016-03-21 11:54:33 +0200239
240 private:
241 TString samplerNamePrefixFromStruct(TIntermTyped *node);
242 bool ancestorEvaluatesToSamplerInStruct(Visit visit);
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000243};
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000244}
245
Jamie Madilld7b1ab52016-12-12 14:42:19 -0500246#endif // COMPILER_TRANSLATOR_OUTPUTHLSL_H_