blob: db13d5700c9fe13fd853277ada74dc02196708c3 [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>
11#include <set>
daniel@transgaming.com652468c2012-12-20 21:11:57 +000012#include <map>
Jamie Madill32aab012015-01-27 14:12:26 -050013#include <stack>
daniel@transgaming.com652468c2012-12-20 21:11:57 +000014
Jamie Madillf51639a2014-06-25 16:04:57 -040015#include "angle_gl.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 Etuahoe17e3192015-01-02 12:47:59 +020019class BuiltInFunctionEmulatorHLSL;
20
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000021namespace sh
22{
daniel@transgaming.comf8f8f362012-04-28 00:35:00 +000023class UnfoldShortCircuit;
Jamie Madill8daaba12014-06-13 10:04:33 -040024class StructureHLSL;
Jamie Madillf91ce812014-06-13 10:04:34 -040025class UniformHLSL;
26
27typedef std::map<TString, TIntermSymbol*> ReferencedSymbols;
daniel@transgaming.comb5875982010-04-15 20:44:53 +000028
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000029class OutputHLSL : public TIntermTraverser
30{
31 public:
Jamie Madill54ad4f82014-09-03 09:40:46 -040032 OutputHLSL(TParseContext &context, TranslatorHLSL *parentTranslator);
daniel@transgaming.comb5875982010-04-15 20:44:53 +000033 ~OutputHLSL();
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000034
daniel@transgaming.com950f9932010-04-13 03:26:14 +000035 void output();
36
Jamie Madill4e1fd412014-07-10 17:50:10 -040037 const std::map<std::string, unsigned int> &getInterfaceBlockRegisterMap() const;
Jamie Madill9fe25e92014-07-18 10:33:08 -040038 const std::map<std::string, unsigned int> &getUniformRegisterMap() const;
Jamie Madill4e1fd412014-07-10 17:50:10 -040039
daniel@transgaming.comb5875982010-04-15 20:44:53 +000040 static TString initializer(const TType &type);
daniel@transgaming.comb5875982010-04-15 20:44:53 +000041
Jamie Madill32aab012015-01-27 14:12:26 -050042 TInfoSinkBase &getInfoSink() { ASSERT(!mInfoSinkStack.empty()); return *mInfoSinkStack.top(); }
43
daniel@transgaming.com950f9932010-04-13 03:26:14 +000044 protected:
Olli Etuahoe17e3192015-01-02 12:47:59 +020045 void header(const BuiltInFunctionEmulatorHLSL *builtInFunctionEmulator);
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000046
daniel@transgaming.com950f9932010-04-13 03:26:14 +000047 // Visit AST nodes and output their code to the body stream
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000048 void visitSymbol(TIntermSymbol*);
Jamie Madill4cfb1e82014-07-07 12:49:23 -040049 void visitRaw(TIntermRaw*);
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000050 void visitConstantUnion(TIntermConstantUnion*);
51 bool visitBinary(Visit visit, TIntermBinary*);
52 bool visitUnary(Visit visit, TIntermUnary*);
53 bool visitSelection(Visit visit, TIntermSelection*);
54 bool visitAggregate(Visit visit, TIntermAggregate*);
55 bool visitLoop(Visit visit, TIntermLoop*);
56 bool visitBranch(Visit visit, TIntermBranch*);
57
daniel@transgaming.com44fffee2012-04-28 00:34:20 +000058 void traverseStatements(TIntermNode *node);
daniel@transgaming.comb5875982010-04-15 20:44:53 +000059 bool isSingleStatement(TIntermNode *node);
daniel@transgaming.com4a35ef22010-04-08 03:51:06 +000060 bool handleExcessiveLoop(TIntermLoop *node);
daniel@transgaming.com67de6d62010-04-29 03:35:30 +000061 void outputTriplet(Visit visit, const TString &preString, const TString &inString, const TString &postString);
apatrick@chromium.org0f4cefe2011-01-26 19:30:57 +000062 void outputLineDirective(int line);
daniel@transgaming.com005c7392010-04-15 20:45:27 +000063 TString argumentString(const TIntermSymbol *symbol);
daniel@transgaming.com0b6b8342010-04-26 15:33:45 +000064 int vectorSize(const TType &type) const;
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000065
Nicolas Capens1af18dc2014-06-11 11:07:32 -040066 void outputConstructor(Visit visit, const TType &type, const TString &name, const TIntermSequence *parameters);
daniel@transgaming.coma54da4e2010-05-07 13:03:28 +000067 const ConstantUnion *writeConstantUnion(const TType &type, const ConstantUnion *constUnion);
daniel@transgaming.com63691862010-04-29 03:32:42 +000068
Olli Etuaho5c9cd3d2014-12-18 13:04:25 +020069 void writeEmulatedFunctionTriplet(Visit visit, const char *preStr);
Jamie Madill37997142015-01-28 10:06:34 -050070 void makeFlaggedStructMaps(const std::vector<TIntermTyped *> &flaggedStructs);
71
72 // Returns true if it found a 'same symbol' initializer (initializer that references the variable it's initting)
73 bool writeSameSymbolInitializer(TInfoSinkBase &out, TIntermSymbol *symbolNode, TIntermTyped *expression);
74 void writeDeferredGlobalInitializers(TInfoSinkBase &out);
Olli Etuaho5c9cd3d2014-12-18 13:04:25 +020075
Jamie Madill55e79e02015-02-09 15:35:00 -050076 // Returns the function name
77 TString addStructEqualityFunction(const TStructure &structure);
78
daniel@transgaming.com950f9932010-04-13 03:26:14 +000079 TParseContext &mContext;
shannon.woods@transgaming.comb73964e2013-01-25 21:49:14 +000080 const ShShaderOutput mOutputType;
daniel@transgaming.comf8f8f362012-04-28 00:35:00 +000081 UnfoldShortCircuit *mUnfoldShortCircuit;
daniel@transgaming.comf9ef1072010-04-22 13:35:16 +000082 bool mInsideFunction;
daniel@transgaming.com950f9932010-04-13 03:26:14 +000083
84 // Output streams
85 TInfoSinkBase mHeader;
86 TInfoSinkBase mBody;
87 TInfoSinkBase mFooter;
daniel@transgaming.comd91cfe72010-04-13 03:26:17 +000088
Jamie Madill32aab012015-01-27 14:12:26 -050089 // A stack is useful when we want to traverse in the header, or in helper functions, but not always
90 // write to the body. Instead use an InfoSink stack to keep our current state intact.
91 std::stack<TInfoSinkBase *> mInfoSinkStack;
92
daniel@transgaming.com8803b852012-12-20 21:11:47 +000093 ReferencedSymbols mReferencedUniforms;
shannonwoods@chromium.org4a643ae2013-05-30 00:12:27 +000094 ReferencedSymbols mReferencedInterfaceBlocks;
daniel@transgaming.com8803b852012-12-20 21:11:47 +000095 ReferencedSymbols mReferencedAttributes;
96 ReferencedSymbols mReferencedVaryings;
Jamie Madill46131a32013-06-20 11:55:50 -040097 ReferencedSymbols mReferencedOutputVariables;
daniel@transgaming.com86f7c9d2010-04-20 18:52:06 +000098
Jamie Madill8daaba12014-06-13 10:04:33 -040099 StructureHLSL *mStructureHLSL;
Jamie Madillf91ce812014-06-13 10:04:34 -0400100 UniformHLSL *mUniformHLSL;
Jamie Madill8daaba12014-06-13 10:04:33 -0400101
Nicolas Capense0ba27a2013-06-24 16:10:52 -0400102 struct TextureFunction
103 {
Nicolas Capens75fb4752013-07-10 15:14:47 -0400104 enum Method
Nicolas Capense0ba27a2013-06-24 16:10:52 -0400105 {
Nicolas Capens75fb4752013-07-10 15:14:47 -0400106 IMPLICIT, // Mipmap LOD determined implicitly (standard lookup)
Nicolas Capense0ba27a2013-06-24 16:10:52 -0400107 BIAS,
108 LOD,
Nicolas Capens75fb4752013-07-10 15:14:47 -0400109 LOD0,
Nicolas Capens84cfa122014-04-14 13:48:45 -0400110 LOD0BIAS,
Nicolas Capensfc014542014-02-18 14:47:13 -0500111 SIZE, // textureSize()
Nicolas Capensd11d5492014-02-19 17:06:10 -0500112 FETCH,
113 GRAD
Nicolas Capense0ba27a2013-06-24 16:10:52 -0400114 };
115
116 TBasicType sampler;
117 int coords;
118 bool proj;
Nicolas Capensb1f45b72013-12-19 17:37:19 -0500119 bool offset;
Nicolas Capens75fb4752013-07-10 15:14:47 -0400120 Method method;
Nicolas Capense0ba27a2013-06-24 16:10:52 -0400121
122 TString name() const;
123
124 bool operator<(const TextureFunction &rhs) const;
125 };
126
127 typedef std::set<TextureFunction> TextureFunctionSet;
128
daniel@transgaming.comd91cfe72010-04-13 03:26:17 +0000129 // Parameters determining what goes in the header output
Nicolas Capense0ba27a2013-06-24 16:10:52 -0400130 TextureFunctionSet mUsesTexture;
shannon.woods%transgaming.com@gtempaccount.comaa8b5cf2013-04-13 03:31:55 +0000131 bool mUsesFragColor;
132 bool mUsesFragData;
daniel@transgaming.comd7c98102010-05-14 17:30:48 +0000133 bool mUsesDepthRange;
daniel@transgaming.com4af7acc2010-05-14 17:30:53 +0000134 bool mUsesFragCoord;
135 bool mUsesPointCoord;
136 bool mUsesFrontFacing;
137 bool mUsesPointSize;
Gregoire Payen de La Garanderieb3dced22015-01-12 14:54:55 +0000138 bool mUsesInstanceID;
Jamie Madill2aeb26a2013-07-08 14:02:55 -0400139 bool mUsesFragDepth;
daniel@transgaming.comd7c98102010-05-14 17:30:48 +0000140 bool mUsesXor;
Jamie Madill3c9eeb92013-11-04 11:09:26 -0500141 bool mUsesDiscardRewriting;
Nicolas Capens655fe362014-04-11 13:12:34 -0400142 bool mUsesNestedBreak;
daniel@transgaming.com005c7392010-04-15 20:45:27 +0000143
shannon.woods%transgaming.com@gtempaccount.comaa8b5cf2013-04-13 03:31:55 +0000144 int mNumRenderTargets;
145
daniel@transgaming.comb6ef8f12010-11-15 16:41:14 +0000146 int mUniqueIndex; // For creating unique names
daniel@transgaming.com89431aa2012-05-31 01:20:29 +0000147
148 bool mContainsLoopDiscontinuity;
Corentin Wallez80bacde2014-11-10 12:07:37 -0800149 bool mContainsAnyLoop;
daniel@transgaming.com89431aa2012-05-31 01:20:29 +0000150 bool mOutputLod0Function;
daniel@transgaming.come11100c2012-05-31 01:20:32 +0000151 bool mInsideDiscontinuousLoop;
Nicolas Capens655fe362014-04-11 13:12:34 -0400152 int mNestedLoopDepth;
daniel@transgaming.come9b3f602012-07-11 20:37:31 +0000153
154 TIntermSymbol *mExcessiveLoopIndex;
daniel@transgaming.com652468c2012-12-20 21:11:57 +0000155
Jamie Madill98493dd2013-07-08 14:39:03 -0400156 TString structInitializerString(int indent, const TStructure &structure, const TString &rhsStructName);
daniel@transgaming.comf4d9fef2012-12-20 21:12:13 +0000157
Jamie Madill570e04d2013-06-21 09:15:33 -0400158 std::map<TIntermTyped*, TString> mFlaggedStructMappedNames;
159 std::map<TIntermTyped*, TString> mFlaggedStructOriginalNames;
160
Jamie Madill37997142015-01-28 10:06:34 -0500161 // Some initializers use varyings, uniforms or attributes, thus we can't evaluate some variables
162 // at global static scope in HLSL. These variables depend on values which we retrieve from the
163 // shader input structure, which we set in the D3D main function. Instead, we can initialize
164 // these static globals after we initialize our other globals.
165 std::vector<std::pair<TIntermSymbol*, TIntermTyped*>> mDeferredGlobalInitializers;
Jamie Madill55e79e02015-02-09 15:35:00 -0500166
167 // A list of structure equality comparison functions. It's important to preserve the order at
168 // which we add the functions, since nested structures call each other recursively.
169 struct StructEqualityFunction
170 {
171 const TStructure *structure;
172 TString functionName;
173 TString functionDefinition;
174 };
175
176 std::vector<StructEqualityFunction> mStructEqualityFunctions;
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000177};
Jamie Madill8daaba12014-06-13 10:04:33 -0400178
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000179}
180
Geoff Lang0a73dd82014-11-19 16:18:08 -0500181#endif // COMPILER_TRANSLATOR_OUTPUTHLSL_H_