1 // Copyright 2016 The SwiftShader Authors. All Rights Reserved.
2 //
3 // Licensed under the Apache License, Version 2.0 (the "License");
4 // you may not use this file except in compliance with the License.
5 // You may obtain a copy of the License at
6 //
7 //    http://www.apache.org/licenses/LICENSE-2.0
8 //
9 // Unless required by applicable law or agreed to in writing, software
10 // distributed under the License is distributed on an "AS IS" BASIS,
11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 // See the License for the specific language governing permissions and
13 // limitations under the License.
14 
15 #ifndef COMPILER_OUTPUTASM_H_
16 #define COMPILER_OUTPUTASM_H_
17 
18 #include "intermediate.h"
19 #include "ParseHelper.h"
20 #include "Shader/PixelShader.hpp"
21 #include "Shader/VertexShader.hpp"
22 
23 #include <list>
24 #include <set>
25 #include <map>
26 
27 namespace es2
28 {
29 	class Shader;
30 }
31 
32 typedef unsigned int GLenum;
33 
34 namespace glsl
35 {
36 	struct BlockMemberInfo
37 	{
BlockMemberInfoBlockMemberInfo38 		BlockMemberInfo() : offset(-1), arrayStride(-1), matrixStride(-1), isRowMajorMatrix(false) {}
39 
BlockMemberInfoBlockMemberInfo40 		BlockMemberInfo(int offset, int arrayStride, int matrixStride, bool isRowMajorMatrix)
41 			: offset(offset),
42 			arrayStride(arrayStride),
43 			matrixStride(matrixStride),
44 			isRowMajorMatrix(isRowMajorMatrix)
45 		{}
46 
getDefaultBlockInfoBlockMemberInfo47 		static BlockMemberInfo getDefaultBlockInfo()
48 		{
49 			return BlockMemberInfo(-1, -1, -1, false);
50 		}
51 
52 		int offset;
53 		int arrayStride;
54 		int matrixStride;
55 		bool isRowMajorMatrix;
56 	};
57 
58 	struct Uniform
59 	{
60 		Uniform(GLenum type, GLenum precision, const std::string &name, int arraySize, int registerIndex, int blockId, const BlockMemberInfo& blockMemberInfo);
61 
62 		GLenum type;
63 		GLenum precision;
64 		std::string name;
65 		int arraySize;
66 
67 		int registerIndex;
68 
69 		int blockId;
70 		BlockMemberInfo blockInfo;
71 	};
72 
73 	typedef std::vector<Uniform> ActiveUniforms;
74 
75 	struct UniformBlock
76 	{
77 		UniformBlock(const std::string& name, unsigned int dataSize, unsigned int arraySize,
78 		             TLayoutBlockStorage layout, bool isRowMajorLayout, int registerIndex, int blockId);
79 
80 		std::string name;
81 		unsigned int dataSize;
82 		unsigned int arraySize;
83 		TLayoutBlockStorage layout;
84 		bool isRowMajorLayout;
85 		std::vector<int> fields;
86 
87 		int registerIndex;
88 
89 		int blockId;
90 	};
91 
92 	class BlockLayoutEncoder
93 	{
94 	public:
95 		BlockLayoutEncoder(bool rowMajor);
~BlockLayoutEncoder()96 		virtual ~BlockLayoutEncoder() {}
97 
98 		BlockMemberInfo encodeType(const TType &type);
99 
getBlockSize()100 		size_t getBlockSize() const { return mCurrentOffset * BytesPerComponent; }
101 
102 		virtual void enterAggregateType() = 0;
103 		virtual void exitAggregateType() = 0;
104 
105 		static const size_t BytesPerComponent = 4u;
106 		static const unsigned int ComponentsPerRegister = 4u;
107 
108 		static size_t getBlockRegister(const BlockMemberInfo &info);
109 		static size_t getBlockRegisterElement(const BlockMemberInfo &info);
110 
111 	protected:
112 		size_t mCurrentOffset;
113 		bool isRowMajor;
114 
115 		void nextRegister();
116 
117 		virtual void getBlockLayoutInfo(const TType &type, unsigned int arraySize, bool isRowMajorMatrix, int *arrayStrideOut, int *matrixStrideOut) = 0;
118 		virtual void advanceOffset(const TType &type, unsigned int arraySize, bool isRowMajorMatrix, int arrayStride, int matrixStride) = 0;
119 	};
120 
121 	// Block layout according to the std140 block layout
122 	// See "Standard Uniform Block Layout" in Section 2.11.6 of the OpenGL ES 3.0 specification
123 	class Std140BlockEncoder : public BlockLayoutEncoder
124 	{
125 	public:
126 		Std140BlockEncoder(bool rowMajor);
127 
128 		void enterAggregateType() override;
129 		void exitAggregateType() override;
130 
131 	protected:
132 		void getBlockLayoutInfo(const TType &type, unsigned int arraySize, bool isRowMajorMatrix, int *arrayStrideOut, int *matrixStrideOut) override;
133 		void advanceOffset(const TType &type, unsigned int arraySize, bool isRowMajorMatrix, int arrayStride, int matrixStride) override;
134 	};
135 
136 	typedef std::vector<UniformBlock> ActiveUniformBlocks;
137 
138 	struct Attribute
139 	{
140 		Attribute();
141 		Attribute(GLenum type, const std::string &name, int arraySize, int location, int registerIndex);
142 
143 		GLenum type;
144 		std::string name;
145 		int arraySize;
146 		int location;
147 
148 		int registerIndex;
149 	};
150 
151 	typedef std::vector<Attribute> ActiveAttributes;
152 
153 	struct Varying
154 	{
155 		Varying(GLenum type, const std::string &name, int arraySize, int reg = -1, int col = -1)
typeVarying156 			: type(type), name(name), arraySize(arraySize), reg(reg), col(col)
157 		{
158 		}
159 
isArrayVarying160 		bool isArray() const
161 		{
162 			return arraySize >= 1;
163 		}
164 
sizeVarying165 		int size() const   // Unify with es2::Uniform?
166 		{
167 			return arraySize > 0 ? arraySize : 1;
168 		}
169 
170 		GLenum type;
171 		std::string name;
172 		int arraySize;
173 
174 		int reg;    // First varying register, assigned during link
175 		int col;    // First register element, assigned during link
176 	};
177 
178 	typedef std::list<Varying> VaryingList;
179 
180 	class Shader
181 	{
182 		friend class OutputASM;
183 	public:
~Shader()184 		virtual ~Shader() {};
185 		virtual sw::Shader *getShader() const = 0;
186 		virtual sw::PixelShader *getPixelShader() const;
187 		virtual sw::VertexShader *getVertexShader() const;
188 
189 	protected:
190 		VaryingList varyings;
191 		ActiveUniforms activeUniforms;
192 		ActiveAttributes activeAttributes;
193 		ActiveUniformBlocks activeUniformBlocks;
194 	};
195 
196 	struct Function
197 	{
FunctionFunction198 		Function(int label, const char *name, TIntermSequence *arg, TIntermTyped *ret) : label(label), name(name), arg(arg), ret(ret)
199 		{
200 		}
201 
FunctionFunction202 		Function(int label, const TString &name, TIntermSequence *arg, TIntermTyped *ret) : label(label), name(name), arg(arg), ret(ret)
203 		{
204 		}
205 
206 		int label;
207 		TString name;
208 		TIntermSequence *arg;
209 		TIntermTyped *ret;
210 	};
211 
212 	typedef sw::Shader::Instruction Instruction;
213 
214 	class Temporary;
215 
216 	class OutputASM : public TIntermTraverser
217 	{
218 	public:
219 		explicit OutputASM(TParseContext &context, Shader *shaderObject);
220 		~OutputASM();
221 
222 		void output();
223 
224 		void freeTemporary(Temporary *temporary);
225 
226 	private:
227 		enum Scope
228 		{
229 			GLOBAL,
230 			FUNCTION
231 		};
232 
233 		struct TextureFunction
234 		{
235 			TextureFunction(const TString& name);
236 
237 			enum Method
238 			{
239 				IMPLICIT,   // Mipmap LOD determined implicitly (standard lookup)
240 				LOD,
241 				SIZE,   // textureSize()
242 				FETCH,
243 				GRAD
244 			};
245 
246 			Method method;
247 			bool proj;
248 			bool offset;
249 		};
250 
251 		void emitShader(Scope scope);
252 
253 		// Visit AST nodes and output their code to the body stream
254 		virtual void visitSymbol(TIntermSymbol*);
255 		virtual bool visitBinary(Visit visit, TIntermBinary*);
256 		virtual bool visitUnary(Visit visit, TIntermUnary*);
257 		virtual bool visitSelection(Visit visit, TIntermSelection*);
258 		virtual bool visitAggregate(Visit visit, TIntermAggregate*);
259 		virtual bool visitLoop(Visit visit, TIntermLoop*);
260 		virtual bool visitBranch(Visit visit, TIntermBranch*);
261 		virtual bool visitSwitch(Visit, TIntermSwitch*);
262 
263 		sw::Shader::Opcode getOpcode(sw::Shader::Opcode op, TIntermTyped *in) const;
264 		Instruction *emit(sw::Shader::Opcode op, TIntermTyped *dst = 0, TIntermNode *src0 = 0, TIntermNode *src1 = 0, TIntermNode *src2 = 0, TIntermNode *src3 = 0, TIntermNode *src4 = 0);
265 		Instruction *emit(sw::Shader::Opcode op, TIntermTyped *dst, int dstIndex, TIntermNode *src0 = 0, int index0 = 0, TIntermNode *src1 = 0, int index1 = 0,
266 		                  TIntermNode *src2 = 0, int index2 = 0, TIntermNode *src3 = 0, int index3 = 0, TIntermNode *src4 = 0, int index4 = 0);
267 		Instruction *emitCast(TIntermTyped *dst, TIntermTyped *src);
268 		Instruction *emitCast(TIntermTyped *dst, int dstIndex, TIntermTyped *src, int srcIndex);
269 		void emitBinary(sw::Shader::Opcode op, TIntermTyped *dst = 0, TIntermNode *src0 = 0, TIntermNode *src1 = 0, TIntermNode *src2 = 0);
270 		void emitAssign(sw::Shader::Opcode op, TIntermTyped *result, TIntermTyped *lhs, TIntermTyped *src0, TIntermTyped *src1 = 0);
271 		void emitCmp(sw::Shader::Control cmpOp, TIntermTyped *dst, TIntermNode *left, TIntermNode *right, int index = 0);
272 		void emitDeterminant(TIntermTyped *result, TIntermTyped *arg, int size, int col = -1, int row = -1, int outCol = 0, int outRow = 0);
273 		void argument(sw::Shader::SourceParameter &parameter, TIntermNode *argument, int index = 0);
274 		void copy(TIntermTyped *dst, TIntermNode *src, int offset = 0);
275 		void assignLvalue(TIntermTyped *dst, TIntermTyped *src);
276 		int lvalue(sw::Shader::DestinationParameter &dst, Temporary &address, TIntermTyped *node);
277 		sw::Shader::ParameterType registerType(TIntermTyped *operand);
278 		unsigned int registerIndex(TIntermTyped *operand);
279 		int writeMask(TIntermTyped *destination, int index = 0);
280 		int readSwizzle(TIntermTyped *argument, int size);
281 		bool trivial(TIntermTyped *expression, int budget);   // Fast to compute and no side effects
282 		int cost(TIntermNode *expression, int budget);
283 		const Function *findFunction(const TString &name);
284 
285 		int temporaryRegister(TIntermTyped *temporary);
286 		int varyingRegister(TIntermTyped *varying);
287 		void declareVarying(TIntermTyped *varying, int reg);
288 		int uniformRegister(TIntermTyped *uniform);
289 		int attributeRegister(TIntermTyped *attribute);
290 		int fragmentOutputRegister(TIntermTyped *fragmentOutput);
291 		int samplerRegister(TIntermTyped *sampler);
292 		int samplerRegister(TIntermSymbol *sampler);
293 		bool isSamplerRegister(TIntermTyped *operand);
294 
295 		typedef std::vector<TIntermTyped*> VariableArray;
296 
297 		int lookup(VariableArray &list, TIntermTyped *variable);
298 		int lookup(VariableArray &list, TInterfaceBlock *block);
299 		int blockMemberLookup(const TType &type, const TString &name, int registerIndex);
300 		int allocate(VariableArray &list, TIntermTyped *variable);
301 		void free(VariableArray &list, TIntermTyped *variable);
302 
303 		void declareUniform(const TType &type, const TString &name, int registerIndex, int blockId = -1, BlockLayoutEncoder* encoder = nullptr);
304 		GLenum glVariableType(const TType &type);
305 		GLenum glVariablePrecision(const TType &type);
306 
307 		static int dim(TIntermNode *v);
308 		static int dim2(TIntermNode *m);
309 		static unsigned int loopCount(TIntermLoop *node);
310 
311 		Shader *const shaderObject;
312 		sw::Shader *shader;
313 		sw::PixelShader *pixelShader;
314 		sw::VertexShader *vertexShader;
315 
316 		VariableArray temporaries;
317 		VariableArray uniforms;
318 		VariableArray varyings;
319 		VariableArray attributes;
320 		VariableArray samplers;
321 		VariableArray fragmentOutputs;
322 
323 		struct TypedMemberInfo : public BlockMemberInfo
324 		{
TypedMemberInfoTypedMemberInfo325 			TypedMemberInfo() {}
TypedMemberInfoTypedMemberInfo326 			TypedMemberInfo(const BlockMemberInfo& b, const TType& t) : BlockMemberInfo(b), type(t) {}
327 			TType type;
328 		};
329 		struct ArgumentInfo
330 		{
ArgumentInfoArgumentInfo331 			ArgumentInfo(const BlockMemberInfo& b, const TType& t, int clampedIndex, int bufferIndex) :
332 			    typedMemberInfo(b, t), clampedIndex(clampedIndex), bufferIndex(bufferIndex) {}
333 			TypedMemberInfo typedMemberInfo;
334 			int clampedIndex;
335 			int bufferIndex;
336 		};
337 		int getBlockId(TIntermTyped *argument);
338 		ArgumentInfo getArgumentInfo(TIntermTyped *argument, int index);
339 
340 		typedef std::map<int, TypedMemberInfo> BlockDefinitionIndexMap;
341 		std::vector<BlockDefinitionIndexMap> blockDefinitions;
342 
343 		Scope emitScope;
344 		Scope currentScope;
345 
346 		int currentFunction;
347 		std::vector<Function> functionArray;
348 
349 		TQualifier outputQualifier;
350 
351 		TParseContext &mContext;
352 	};
353 
354 	class LoopUnrollable : public TIntermTraverser
355 	{
356 	public:
357 		bool traverse(TIntermNode *node);
358 
359 	private:
360 		bool visitBranch(Visit visit, TIntermBranch *node);
361 		bool visitLoop(Visit visit, TIntermLoop *loop);
362 		bool visitAggregate(Visit visit, TIntermAggregate *node);
363 
364 		int loopDepth;
365 		bool loopUnrollable;
366 	};
367 }
368 
369 #endif   // COMPILER_OUTPUTASM_H_
370