blob: f681b71663225e7bbd8da3e5977911e47ce029a5 [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
7#ifndef COMPILER_OUTPUTHLSL_H_
8#define COMPILER_OUTPUTHLSL_H_
9
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>
13
shannon.woods%transgaming.com@gtempaccount.comf26ddae2013-04-13 03:29:13 +000014#include <GLES3/gl3.h>
daniel@transgaming.com652468c2012-12-20 21:11:57 +000015#include <GLES2/gl2.h>
alokp@chromium.org4e89d232010-05-14 19:37:21 +000016
Geoff Lang17732822013-08-29 13:46:49 -040017#include "compiler/translator/intermediate.h"
Jamie Madill6b9cb252013-10-17 10:45:47 -040018#include "compiler/translator/ParseContext.h"
Jamie Madill834e8b72014-04-11 13:33:58 -040019#include "common/shadervars.h"
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000020
21namespace sh
22{
daniel@transgaming.comf8f8f362012-04-28 00:35:00 +000023class UnfoldShortCircuit;
daniel@transgaming.comb5875982010-04-15 20:44:53 +000024
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000025class OutputHLSL : public TIntermTraverser
26{
27 public:
shannon.woods%transgaming.com@gtempaccount.com18b4c4b2013-04-13 03:31:40 +000028 OutputHLSL(TParseContext &context, const ShBuiltInResources& resources, ShShaderOutput outputType);
daniel@transgaming.comb5875982010-04-15 20:44:53 +000029 ~OutputHLSL();
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000030
daniel@transgaming.com950f9932010-04-13 03:26:14 +000031 void output();
32
daniel@transgaming.comb5875982010-04-15 20:44:53 +000033 TInfoSinkBase &getBodyStream();
Jamie Madill834e8b72014-04-11 13:33:58 -040034 const std::vector<gl::Uniform> &getUniforms();
35 const std::vector<gl::InterfaceBlock> &getInterfaceBlocks() const;
36 const std::vector<gl::Attribute> &getOutputVariables() const;
37 const std::vector<gl::Attribute> &getAttributes() const;
38 const std::vector<gl::Varying> &getVaryings() const;
daniel@transgaming.comb5875982010-04-15 20:44:53 +000039
daniel@transgaming.coma2a95e72010-05-20 19:17:55 +000040 TString typeString(const TType &type);
shannon.woods@transgaming.comfb256be2013-01-25 21:49:25 +000041 TString textureString(const TType &type);
Nicolas Capenscb127d32013-07-15 17:26:18 -040042 TString samplerString(const TType &type);
shannon.woods%transgaming.com@gtempaccount.com1886fd42013-04-13 03:41:45 +000043 TString interpolationString(TQualifier qualifier);
Jamie Madill98493dd2013-07-08 14:39:03 -040044 TString structureString(const TStructure &structure, bool useHLSLRowMajorPacking, bool useStd140Packing);
45 TString structureTypeName(const TStructure &structure, bool useHLSLRowMajorPacking, bool useStd140Packing);
daniel@transgaming.comb5875982010-04-15 20:44:53 +000046 static TString qualifierString(TQualifier qualifier);
daniel@transgaming.comb5875982010-04-15 20:44:53 +000047 static TString arrayString(const TType &type);
48 static TString initializer(const TType &type);
daniel@transgaming.comc72c6412011-09-20 16:09:17 +000049 static TString decorate(const TString &string); // Prepends an underscore to avoid naming clashes
apatrick@chromium.org65756022012-01-17 21:45:38 +000050 static TString decorateUniform(const TString &string, const TType &type);
Jamie Madill98493dd2013-07-08 14:39:03 -040051 static TString decorateField(const TString &string, const TStructure &structure);
daniel@transgaming.comb5875982010-04-15 20:44:53 +000052
daniel@transgaming.com950f9932010-04-13 03:26:14 +000053 protected:
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000054 void header();
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000055
daniel@transgaming.com950f9932010-04-13 03:26:14 +000056 // Visit AST nodes and output their code to the body stream
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000057 void visitSymbol(TIntermSymbol*);
58 void visitConstantUnion(TIntermConstantUnion*);
59 bool visitBinary(Visit visit, TIntermBinary*);
60 bool visitUnary(Visit visit, TIntermUnary*);
61 bool visitSelection(Visit visit, TIntermSelection*);
62 bool visitAggregate(Visit visit, TIntermAggregate*);
63 bool visitLoop(Visit visit, TIntermLoop*);
64 bool visitBranch(Visit visit, TIntermBranch*);
65
daniel@transgaming.com44fffee2012-04-28 00:34:20 +000066 void traverseStatements(TIntermNode *node);
daniel@transgaming.comb5875982010-04-15 20:44:53 +000067 bool isSingleStatement(TIntermNode *node);
daniel@transgaming.com4a35ef22010-04-08 03:51:06 +000068 bool handleExcessiveLoop(TIntermLoop *node);
daniel@transgaming.com67de6d62010-04-29 03:35:30 +000069 void outputTriplet(Visit visit, const TString &preString, const TString &inString, const TString &postString);
apatrick@chromium.org0f4cefe2011-01-26 19:30:57 +000070 void outputLineDirective(int line);
daniel@transgaming.com005c7392010-04-15 20:45:27 +000071 TString argumentString(const TIntermSymbol *symbol);
daniel@transgaming.com0b6b8342010-04-26 15:33:45 +000072 int vectorSize(const TType &type) const;
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000073
daniel@transgaming.com67de6d62010-04-29 03:35:30 +000074 void addConstructor(const TType &type, const TString &name, const TIntermSequence *parameters);
daniel@transgaming.coma54da4e2010-05-07 13:03:28 +000075 const ConstantUnion *writeConstantUnion(const TType &type, const ConstantUnion *constUnion);
daniel@transgaming.com63691862010-04-29 03:32:42 +000076
Jamie Madillbfa91f42014-06-05 15:45:18 -040077 TString structNameString(const TStructure &structure);
daniel@transgaming.coma2a95e72010-05-20 19:17:55 +000078
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
daniel@transgaming.com8803b852012-12-20 21:11:47 +000089 typedef std::map<TString, TIntermSymbol*> ReferencedSymbols;
90 ReferencedSymbols mReferencedUniforms;
shannonwoods@chromium.org4a643ae2013-05-30 00:12:27 +000091 ReferencedSymbols mReferencedInterfaceBlocks;
daniel@transgaming.com8803b852012-12-20 21:11:47 +000092 ReferencedSymbols mReferencedAttributes;
93 ReferencedSymbols mReferencedVaryings;
Jamie Madill46131a32013-06-20 11:55:50 -040094 ReferencedSymbols mReferencedOutputVariables;
daniel@transgaming.com86f7c9d2010-04-20 18:52:06 +000095
Nicolas Capense0ba27a2013-06-24 16:10:52 -040096 struct TextureFunction
97 {
Nicolas Capens75fb4752013-07-10 15:14:47 -040098 enum Method
Nicolas Capense0ba27a2013-06-24 16:10:52 -040099 {
Nicolas Capens75fb4752013-07-10 15:14:47 -0400100 IMPLICIT, // Mipmap LOD determined implicitly (standard lookup)
Nicolas Capense0ba27a2013-06-24 16:10:52 -0400101 BIAS,
102 LOD,
Nicolas Capens75fb4752013-07-10 15:14:47 -0400103 LOD0,
Nicolas Capens84cfa122014-04-14 13:48:45 -0400104 LOD0BIAS,
Nicolas Capensfc014542014-02-18 14:47:13 -0500105 SIZE, // textureSize()
Nicolas Capensd11d5492014-02-19 17:06:10 -0500106 FETCH,
107 GRAD
Nicolas Capense0ba27a2013-06-24 16:10:52 -0400108 };
109
110 TBasicType sampler;
111 int coords;
112 bool proj;
Nicolas Capensb1f45b72013-12-19 17:37:19 -0500113 bool offset;
Nicolas Capens75fb4752013-07-10 15:14:47 -0400114 Method method;
Nicolas Capense0ba27a2013-06-24 16:10:52 -0400115
116 TString name() const;
117
118 bool operator<(const TextureFunction &rhs) const;
119 };
120
121 typedef std::set<TextureFunction> TextureFunctionSet;
122
daniel@transgaming.comd91cfe72010-04-13 03:26:17 +0000123 // Parameters determining what goes in the header output
Nicolas Capense0ba27a2013-06-24 16:10:52 -0400124 TextureFunctionSet mUsesTexture;
shannon.woods%transgaming.com@gtempaccount.comaa8b5cf2013-04-13 03:31:55 +0000125 bool mUsesFragColor;
126 bool mUsesFragData;
daniel@transgaming.comd7c98102010-05-14 17:30:48 +0000127 bool mUsesDepthRange;
daniel@transgaming.com4af7acc2010-05-14 17:30:53 +0000128 bool mUsesFragCoord;
129 bool mUsesPointCoord;
130 bool mUsesFrontFacing;
131 bool mUsesPointSize;
Jamie Madill2aeb26a2013-07-08 14:02:55 -0400132 bool mUsesFragDepth;
daniel@transgaming.comd7c98102010-05-14 17:30:48 +0000133 bool mUsesXor;
134 bool mUsesMod1;
daniel@transgaming.com4229f592011-11-24 22:34:04 +0000135 bool mUsesMod2v;
136 bool mUsesMod2f;
137 bool mUsesMod3v;
138 bool mUsesMod3f;
139 bool mUsesMod4v;
140 bool mUsesMod4f;
daniel@transgaming.com0bbb0312010-04-26 15:33:39 +0000141 bool mUsesFaceforward1;
142 bool mUsesFaceforward2;
143 bool mUsesFaceforward3;
144 bool mUsesFaceforward4;
daniel@transgaming.com35342dc2012-02-28 02:01:22 +0000145 bool mUsesAtan2_1;
146 bool mUsesAtan2_2;
147 bool mUsesAtan2_3;
148 bool mUsesAtan2_4;
Jamie Madill3c9eeb92013-11-04 11:09:26 -0500149 bool mUsesDiscardRewriting;
Nicolas Capens655fe362014-04-11 13:12:34 -0400150 bool mUsesNestedBreak;
daniel@transgaming.com005c7392010-04-15 20:45:27 +0000151
shannon.woods%transgaming.com@gtempaccount.comaa8b5cf2013-04-13 03:31:55 +0000152 int mNumRenderTargets;
153
daniel@transgaming.coma2a95e72010-05-20 19:17:55 +0000154 typedef std::set<TString> Constructors;
155 Constructors mConstructors;
daniel@transgaming.com63691862010-04-29 03:32:42 +0000156
daniel@transgaming.coma2a95e72010-05-20 19:17:55 +0000157 typedef std::set<TString> StructNames;
158 StructNames mStructNames;
daniel@transgaming.com63691862010-04-29 03:32:42 +0000159
daniel@transgaming.coma2a95e72010-05-20 19:17:55 +0000160 typedef std::list<TString> StructDeclarations;
161 StructDeclarations mStructDeclarations;
daniel@transgaming.com63691862010-04-29 03:32:42 +0000162
daniel@transgaming.comb6ef8f12010-11-15 16:41:14 +0000163 int mUniqueIndex; // For creating unique names
daniel@transgaming.com89431aa2012-05-31 01:20:29 +0000164
165 bool mContainsLoopDiscontinuity;
166 bool mOutputLod0Function;
daniel@transgaming.come11100c2012-05-31 01:20:32 +0000167 bool mInsideDiscontinuousLoop;
Nicolas Capens655fe362014-04-11 13:12:34 -0400168 int mNestedLoopDepth;
daniel@transgaming.come9b3f602012-07-11 20:37:31 +0000169
170 TIntermSymbol *mExcessiveLoopIndex;
daniel@transgaming.com652468c2012-12-20 21:11:57 +0000171
172 int mUniformRegister;
shannonwoods@chromium.org4a643ae2013-05-30 00:12:27 +0000173 int mInterfaceBlockRegister;
daniel@transgaming.com652468c2012-12-20 21:11:57 +0000174 int mSamplerRegister;
Jamie Madill574d9dd2013-06-20 11:55:56 -0400175 int mPaddingCounter;
daniel@transgaming.com652468c2012-12-20 21:11:57 +0000176
177 TString registerString(TIntermSymbol *operand);
178 int samplerRegister(TIntermSymbol *sampler);
179 int uniformRegister(TIntermSymbol *uniform);
Jamie Madill834e8b72014-04-11 13:33:58 -0400180 void declareInterfaceBlockField(const TType &type, const TString &name, std::vector<gl::InterfaceBlockField>& output);
181 gl::Uniform declareUniformToList(const TType &type, const TString &name, int registerIndex, std::vector<gl::Uniform>& output);
daniel@transgaming.comf4d9fef2012-12-20 21:12:13 +0000182 void declareUniform(const TType &type, const TString &name, int index);
Jamie Madill834e8b72014-04-11 13:33:58 -0400183 void declareVaryingToList(const TType &type, TQualifier baseTypeQualifier, const TString &name, std::vector<gl::Varying>& fieldsOut);
shannonwoods@chromium.org4430b0d2013-05-30 00:12:34 +0000184
Jamie Madillc2141fb2013-08-30 13:21:08 -0400185 // Returns the uniform's register index
186 int declareUniformAndAssignRegister(const TType &type, const TString &name);
187
Jamie Madill98493dd2013-07-08 14:39:03 -0400188 TString interfaceBlockFieldString(const TInterfaceBlock &interfaceBlock, const TField &field);
shannonwoods@chromium.orge429ab72013-05-30 00:12:52 +0000189 TString decoratePrivate(const TString &privateText);
Jamie Madill98493dd2013-07-08 14:39:03 -0400190 TString interfaceBlockStructNameString(const TInterfaceBlock &interfaceBlockType);
191 TString interfaceBlockInstanceString(const TInterfaceBlock& interfaceBlock, unsigned int arrayIndex);
192 TString interfaceBlockFieldTypeString(const TField &field, TLayoutBlockStorage blockStorage);
193 TString interfaceBlockFieldString(const TInterfaceBlock &interfaceBlock, TLayoutBlockStorage blockStorage);
194 TString interfaceBlockStructString(const TInterfaceBlock &interfaceBlock);
195 TString interfaceBlockString(const TInterfaceBlock &interfaceBlock, unsigned int registerIndex, unsigned int arrayIndex);
Jamie Madill574d9dd2013-06-20 11:55:56 -0400196 TString std140PrePaddingString(const TType &type, int *elementIndex);
Jamie Madille4075c92013-06-21 09:15:32 -0400197 TString std140PostPaddingString(const TType &type, bool useHLSLRowMajorPacking);
Jamie Madill98493dd2013-07-08 14:39:03 -0400198 TString structInitializerString(int indent, const TStructure &structure, const TString &rhsStructName);
shannon.woods%transgaming.com@gtempaccount.com6f273e32013-04-13 03:41:15 +0000199
daniel@transgaming.comf4d9fef2012-12-20 21:12:13 +0000200 static GLenum glVariableType(const TType &type);
shannon.woods@transgaming.comfe3c0ef2013-02-28 23:17:22 +0000201 static GLenum glVariablePrecision(const TType &type);
shannon.woods%transgaming.com@gtempaccount.com6f273e32013-04-13 03:41:15 +0000202 static bool isVaryingIn(TQualifier qualifier);
203 static bool isVaryingOut(TQualifier qualifier);
204 static bool isVarying(TQualifier qualifier);
daniel@transgaming.comf4d9fef2012-12-20 21:12:13 +0000205
Jamie Madill834e8b72014-04-11 13:33:58 -0400206 std::vector<gl::Uniform> mActiveUniforms;
207 std::vector<gl::InterfaceBlock> mActiveInterfaceBlocks;
208 std::vector<gl::Attribute> mActiveOutputVariables;
209 std::vector<gl::Attribute> mActiveAttributes;
210 std::vector<gl::Varying> mActiveVaryings;
Jamie Madille4075c92013-06-21 09:15:32 -0400211 std::map<TString, int> mStd140StructElementIndexes;
Jamie Madill570e04d2013-06-21 09:15:33 -0400212 std::map<TIntermTyped*, TString> mFlaggedStructMappedNames;
213 std::map<TIntermTyped*, TString> mFlaggedStructOriginalNames;
214
215 void makeFlaggedStructMaps(const std::vector<TIntermTyped *> &flaggedStructs);
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000216};
217}
218
219#endif // COMPILER_OUTPUTHLSL_H_