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 #include "localintermediate.h"
16 #include "SymbolTable.h"
17 
18 //
19 // Two purposes:
20 // 1.  Show an example of how to iterate tree.  Functions can
21 //     also directly call Traverse() on children themselves to
22 //     have finer grained control over the process than shown here.
23 //     See the last function for how to get started.
24 // 2.  Print out a text based description of the tree.
25 //
26 
27 //
28 // Use this class to carry along data from node to node in
29 // the traversal
30 //
31 class TOutputTraverser : public TIntermTraverser {
32 public:
TOutputTraverser(TInfoSinkBase & i)33 	TOutputTraverser(TInfoSinkBase& i) : sink(i) { }
34 	TInfoSinkBase& sink;
35 
36 protected:
37 	void visitSymbol(TIntermSymbol*);
38 	void visitConstantUnion(TIntermConstantUnion*);
39 	bool visitBinary(Visit visit, TIntermBinary*);
40 	bool visitUnary(Visit visit, TIntermUnary*);
41 	bool visitSelection(Visit visit, TIntermSelection*);
42 	bool visitAggregate(Visit visit, TIntermAggregate*);
43 	bool visitLoop(Visit visit, TIntermLoop*);
44 	bool visitBranch(Visit visit, TIntermBranch*);
45 };
46 
getCompleteString() const47 TString TType::getCompleteString() const
48 {
49 	TStringStream stream;
50 
51 	if (qualifier != EvqTemporary && qualifier != EvqGlobal)
52 		stream << getQualifierString() << " " << getPrecisionString() << " ";
53 	if (array)
54 		stream << "array of ";
55 	if (isMatrix())
56 		stream << static_cast<int>(primarySize) << "X" << static_cast<int>(secondarySize) << " matrix of ";
57 	else if(primarySize > 1)
58 		stream << static_cast<int>(primarySize) << "-component vector of ";
59 
60 	stream << getBasicString();
61 	return stream.str();
62 }
63 
64 //
65 // Helper functions for printing, not part of traversing.
66 //
67 
OutputTreeText(TInfoSinkBase & sink,TIntermNode * node,const int depth)68 void OutputTreeText(TInfoSinkBase& sink, TIntermNode* node, const int depth)
69 {
70 	int i;
71 
72 	sink.location(node->getLine());
73 
74 	for (i = 0; i < depth; ++i)
75 		sink << "  ";
76 }
77 
78 //
79 // The rest of the file are the traversal functions.  The last one
80 // is the one that starts the traversal.
81 //
82 // Return true from interior nodes to have the external traversal
83 // continue on to children.  If you process children yourself,
84 // return false.
85 //
86 
visitSymbol(TIntermSymbol * node)87 void TOutputTraverser::visitSymbol(TIntermSymbol* node)
88 {
89 	OutputTreeText(sink, node, mDepth);
90 
91 	sink << "'" << node->getSymbol() << "' ";
92 	sink << "(" << node->getCompleteString() << ")\n";
93 }
94 
visitBinary(Visit visit,TIntermBinary * node)95 bool TOutputTraverser::visitBinary(Visit visit, TIntermBinary* node)
96 {
97 	TInfoSinkBase& out = sink;
98 
99 	OutputTreeText(out, node, mDepth);
100 
101 	switch (node->getOp()) {
102 	case EOpAssign:                   out << "move second child to first child";           break;
103 	case EOpInitialize:               out << "initialize first child with second child";   break;
104 	case EOpAddAssign:                out << "add second child into first child";          break;
105 	case EOpSubAssign:                out << "subtract second child into first child";     break;
106 	case EOpMulAssign:                out << "multiply second child into first child";     break;
107 	case EOpVectorTimesMatrixAssign:  out << "matrix mult second child into first child";  break;
108 	case EOpVectorTimesScalarAssign:  out << "vector scale second child into first child"; break;
109 	case EOpMatrixTimesScalarAssign:  out << "matrix scale second child into first child"; break;
110 	case EOpMatrixTimesMatrixAssign:  out << "matrix mult second child into first child";  break;
111 	case EOpDivAssign:                out << "divide second child into first child";       break;
112 	case EOpIModAssign:               out << "modulo second child into first child";       break;
113 	case EOpBitShiftLeftAssign:       out << "bit-wise shift first child left by second child";  break;
114 	case EOpBitShiftRightAssign:      out << "bit-wise shift first child right by second child"; break;
115 	case EOpBitwiseAndAssign:         out << "bit-wise and second child into first child"; break;
116 	case EOpBitwiseXorAssign:         out << "bit-wise xor second child into first child"; break;
117 	case EOpBitwiseOrAssign:          out << "bit-wise or second child into first child";  break;
118 	case EOpIndexDirect:   out << "direct index";   break;
119 	case EOpIndexIndirect: out << "indirect index"; break;
120 	case EOpIndexDirectStruct:   out << "direct index for structure";   break;
121 	case EOpVectorSwizzle: out << "vector swizzle"; break;
122 
123 	case EOpAdd:    out << "add";                     break;
124 	case EOpSub:    out << "subtract";                break;
125 	case EOpMul:    out << "component-wise multiply"; break;
126 	case EOpDiv:    out << "divide";                  break;
127 	case EOpIMod:   out << "modulo";                  break;
128 	case EOpBitShiftLeft:     out << "bit-wise shift left";           break;
129 	case EOpBitShiftRight:    out << "bit-wise shift right";          break;
130 	case EOpBitwiseAnd:       out << "bit-wise and";                  break;
131 	case EOpBitwiseXor:       out << "bit-wise xor";                  break;
132 	case EOpBitwiseOr:        out << "bit-wise or";                   break;
133 	case EOpEqual:            out << "Compare Equal";                 break;
134 	case EOpNotEqual:         out << "Compare Not Equal";             break;
135 	case EOpLessThan:         out << "Compare Less Than";             break;
136 	case EOpGreaterThan:      out << "Compare Greater Than";          break;
137 	case EOpLessThanEqual:    out << "Compare Less Than or Equal";    break;
138 	case EOpGreaterThanEqual: out << "Compare Greater Than or Equal"; break;
139 
140 	case EOpVectorTimesScalar: out << "vector-scale";          break;
141 	case EOpVectorTimesMatrix: out << "vector-times-matrix";   break;
142 	case EOpMatrixTimesVector: out << "matrix-times-vector";   break;
143 	case EOpMatrixTimesScalar: out << "matrix-scale";          break;
144 	case EOpMatrixTimesMatrix: out << "matrix-multiply";       break;
145 
146 	case EOpLogicalOr:  out << "logical-or";   break;
147 	case EOpLogicalXor: out << "logical-xor"; break;
148 	case EOpLogicalAnd: out << "logical-and"; break;
149 	default: out << "<unknown op>";
150 	}
151 
152 	out << " (" << node->getCompleteString() << ")";
153 
154 	out << "\n";
155 
156 	return true;
157 }
158 
visitUnary(Visit visit,TIntermUnary * node)159 bool TOutputTraverser::visitUnary(Visit visit, TIntermUnary* node)
160 {
161 	TInfoSinkBase& out = sink;
162 
163 	OutputTreeText(out, node, mDepth);
164 
165 	switch (node->getOp()) {
166 	case EOpNegative:       out << "Negate value";         break;
167 	case EOpVectorLogicalNot:
168 	case EOpLogicalNot:     out << "Negate conditional";   break;
169 	case EOpBitwiseNot:     out << "bit-wise not";         break;
170 
171 	case EOpPostIncrement:  out << "Post-Increment";       break;
172 	case EOpPostDecrement:  out << "Post-Decrement";       break;
173 	case EOpPreIncrement:   out << "Pre-Increment";        break;
174 	case EOpPreDecrement:   out << "Pre-Decrement";        break;
175 
176 	case EOpRadians:        out << "radians";              break;
177 	case EOpDegrees:        out << "degrees";              break;
178 	case EOpSin:            out << "sine";                 break;
179 	case EOpCos:            out << "cosine";               break;
180 	case EOpTan:            out << "tangent";              break;
181 	case EOpAsin:           out << "arc sine";             break;
182 	case EOpAcos:           out << "arc cosine";           break;
183 	case EOpAtan:           out << "arc tangent";          break;
184 	case EOpSinh:           out << "hyperbolic sine";        break;
185 	case EOpCosh:           out << "hyperbolic cosine";      break;
186 	case EOpTanh:           out << "hyperbolic tangent";     break;
187 	case EOpAsinh:          out << "arc hyperbolic sine";    break;
188 	case EOpAcosh:          out << "arc hyperbolic cosine";  break;
189 	case EOpAtanh:          out << "arc hyperbolic tangent"; break;
190 
191 	case EOpExp:            out << "exp";                  break;
192 	case EOpLog:            out << "log";                  break;
193 	case EOpExp2:           out << "exp2";                 break;
194 	case EOpLog2:           out << "log2";                 break;
195 	case EOpSqrt:           out << "sqrt";                 break;
196 	case EOpInverseSqrt:    out << "inverse sqrt";         break;
197 
198 	case EOpAbs:            out << "Absolute value";       break;
199 	case EOpSign:           out << "Sign";                 break;
200 	case EOpFloor:          out << "Floor";                break;
201 	case EOpTrunc:          out << "Trunc";                break;
202 	case EOpRound:          out << "Round";                break;
203 	case EOpRoundEven:      out << "RoundEven";            break;
204 	case EOpCeil:           out << "Ceiling";              break;
205 	case EOpFract:          out << "Fraction";             break;
206 	case EOpIsNan:          out << "Is not a number";      break;
207 	case EOpIsInf:          out << "Is infinity";          break;
208 
209 	case EOpFloatBitsToInt: out << "float bits to int";    break;
210 	case EOpFloatBitsToUint: out << "float bits to uint";  break;
211 	case EOpIntBitsToFloat: out << "int bits to float";    break;
212 	case EOpUintBitsToFloat: out << "uint bits to float";  break;
213 
214 	case EOpPackSnorm2x16:  out << "pack Snorm 2x16";      break;
215 	case EOpPackUnorm2x16:  out << "pack Unorm 2x16";      break;
216 	case EOpPackHalf2x16:   out << "pack half 2x16";       break;
217 
218 	case EOpUnpackSnorm2x16: out << "unpack Snorm 2x16";   break;
219 	case EOpUnpackUnorm2x16: out << "unpack Unorm 2x16";   break;
220 	case EOpUnpackHalf2x16:  out << "unpack half 2x16";    break;
221 
222 	case EOpLength:         out << "length";               break;
223 	case EOpNormalize:      out << "normalize";            break;
224 		//	case EOpDPdx:           out << "dPdx";                 break;
225 		//	case EOpDPdy:           out << "dPdy";                 break;
226 		//	case EOpFwidth:         out << "fwidth";               break;
227 
228 	case EOpDeterminant:    out << "determinant";          break;
229 	case EOpTranspose:      out << "transpose";            break;
230 	case EOpInverse:        out << "inverse";              break;
231 
232 	case EOpAny:            out << "any";                  break;
233 	case EOpAll:            out << "all";                  break;
234 
235 	default: out.message(EPrefixError, "Bad unary op");
236 	}
237 
238 	out << " (" << node->getCompleteString() << ")";
239 
240 	out << "\n";
241 
242 	return true;
243 }
244 
visitAggregate(Visit visit,TIntermAggregate * node)245 bool TOutputTraverser::visitAggregate(Visit visit, TIntermAggregate* node)
246 {
247 	TInfoSinkBase& out = sink;
248 
249 	if (node->getOp() == EOpNull) {
250 		out.message(EPrefixError, "node is still EOpNull!");
251 		return true;
252 	}
253 
254 	OutputTreeText(out, node, mDepth);
255 
256 	switch (node->getOp()) {
257 	case EOpSequence:      out << "Sequence\n"; return true;
258 	case EOpComma:         out << "Comma\n"; return true;
259 	case EOpFunction:      out << "Function Definition: " << node->getName(); break;
260 	case EOpFunctionCall:  out << "Function Call: " << node->getName(); break;
261 	case EOpParameters:    out << "Function Parameters: ";              break;
262 
263 	case EOpConstructFloat: out << "Construct float"; break;
264 	case EOpConstructVec2:  out << "Construct vec2";  break;
265 	case EOpConstructVec3:  out << "Construct vec3";  break;
266 	case EOpConstructVec4:  out << "Construct vec4";  break;
267 	case EOpConstructBool:  out << "Construct bool";  break;
268 	case EOpConstructBVec2: out << "Construct bvec2"; break;
269 	case EOpConstructBVec3: out << "Construct bvec3"; break;
270 	case EOpConstructBVec4: out << "Construct bvec4"; break;
271 	case EOpConstructInt:   out << "Construct int";   break;
272 	case EOpConstructIVec2: out << "Construct ivec2"; break;
273 	case EOpConstructIVec3: out << "Construct ivec3"; break;
274 	case EOpConstructIVec4: out << "Construct ivec4"; break;
275 	case EOpConstructUInt:  out << "Construct uint";  break;
276 	case EOpConstructUVec2: out << "Construct uvec2"; break;
277 	case EOpConstructUVec3: out << "Construct uvec3"; break;
278 	case EOpConstructUVec4: out << "Construct uvec4"; break;
279 	case EOpConstructMat2:  out << "Construct mat2";  break;
280 	case EOpConstructMat3:  out << "Construct mat3";  break;
281 	case EOpConstructMat4:  out << "Construct mat4";  break;
282 	case EOpConstructStruct:  out << "Construct structure";  break;
283 
284 	case EOpLessThan:         out << "Compare Less Than";             break;
285 	case EOpGreaterThan:      out << "Compare Greater Than";          break;
286 	case EOpLessThanEqual:    out << "Compare Less Than or Equal";    break;
287 	case EOpGreaterThanEqual: out << "Compare Greater Than or Equal"; break;
288 	case EOpVectorEqual:      out << "Equal";                         break;
289 	case EOpVectorNotEqual:   out << "NotEqual";                      break;
290 
291 	case EOpMod:           out << "mod";         break;
292 	case EOpModf:          out << "modf";        break;
293 	case EOpPow:           out << "pow";         break;
294 
295 	case EOpAtan:          out << "arc tangent"; break;
296 
297 	case EOpMin:           out << "min";         break;
298 	case EOpMax:           out << "max";         break;
299 	case EOpClamp:         out << "clamp";       break;
300 	case EOpMix:           out << "mix";         break;
301 	case EOpStep:          out << "step";        break;
302 	case EOpSmoothStep:    out << "smoothstep";  break;
303 
304 	case EOpDistance:      out << "distance";                break;
305 	case EOpDot:           out << "dot-product";             break;
306 	case EOpCross:         out << "cross-product";           break;
307 	case EOpFaceForward:   out << "face-forward";            break;
308 	case EOpReflect:       out << "reflect";                 break;
309 	case EOpRefract:       out << "refract";                 break;
310 	case EOpMul:           out << "component-wise multiply"; break;
311 	case EOpOuterProduct:  out << "outer product";           break;
312 
313 	default: out.message(EPrefixError, "Bad aggregation op");
314 	}
315 
316 	if (node->getOp() != EOpSequence && node->getOp() != EOpParameters)
317 		out << " (" << node->getCompleteString() << ")";
318 
319 	out << "\n";
320 
321 	return true;
322 }
323 
visitSelection(Visit visit,TIntermSelection * node)324 bool TOutputTraverser::visitSelection(Visit visit, TIntermSelection* node)
325 {
326 	TInfoSinkBase& out = sink;
327 
328 	OutputTreeText(out, node, mDepth);
329 
330 	out << "Test condition and select";
331 	out << " (" << node->getCompleteString() << ")\n";
332 
333 	++mDepth;
334 
335 	OutputTreeText(sink, node, mDepth);
336 	out << "Condition\n";
337 	node->getCondition()->traverse(this);
338 
339 	OutputTreeText(sink, node, mDepth);
340 	if (node->getTrueBlock()) {
341 		out << "true case\n";
342 		node->getTrueBlock()->traverse(this);
343 	} else
344 		out << "true case is null\n";
345 
346 	if (node->getFalseBlock()) {
347 		OutputTreeText(sink, node, mDepth);
348 		out << "false case\n";
349 		node->getFalseBlock()->traverse(this);
350 	}
351 
352 	--mDepth;
353 
354 	return false;
355 }
356 
visitConstantUnion(TIntermConstantUnion * node)357 void TOutputTraverser::visitConstantUnion(TIntermConstantUnion* node)
358 {
359 	TInfoSinkBase& out = sink;
360 
361 	size_t size = node->getType().getObjectSize();
362 
363 	for(size_t i = 0; i < size; i++) {
364 		OutputTreeText(out, node, mDepth);
365 		switch (node->getUnionArrayPointer()[i].getType()) {
366 		case EbtBool:
367 			if (node->getUnionArrayPointer()[i].getBConst())
368 				out << "true";
369 			else
370 				out << "false";
371 
372 			out << " (" << "const bool" << ")";
373 			out << "\n";
374 			break;
375 		case EbtFloat:
376 			out << node->getUnionArrayPointer()[i].getFConst();
377 			out << " (const float)\n";
378 			break;
379 		case EbtInt:
380 			out << node->getUnionArrayPointer()[i].getIConst();
381 			out << " (const int)\n";
382 			break;
383 		case EbtUInt:
384 			out << node->getUnionArrayPointer()[i].getUConst();
385 			out << " (const uint)\n";
386 			break;
387 		default:
388 			out.message(EPrefixInternalError, "Unknown constant", node->getLine());
389 			break;
390 		}
391 	}
392 }
393 
visitLoop(Visit visit,TIntermLoop * node)394 bool TOutputTraverser::visitLoop(Visit visit, TIntermLoop* node)
395 {
396 	TInfoSinkBase& out = sink;
397 
398 	OutputTreeText(out, node, mDepth);
399 
400 	out << "Loop with condition ";
401 	if (node->getType() == ELoopDoWhile)
402 		out << "not ";
403 	out << "tested first\n";
404 
405 	++mDepth;
406 
407 	OutputTreeText(sink, node, mDepth);
408 	if (node->getCondition()) {
409 		out << "Loop Condition\n";
410 		node->getCondition()->traverse(this);
411 	} else
412 		out << "No loop condition\n";
413 
414 	OutputTreeText(sink, node, mDepth);
415 	if (node->getBody()) {
416 		out << "Loop Body\n";
417 		node->getBody()->traverse(this);
418 	} else
419 		out << "No loop body\n";
420 
421 	if (node->getExpression()) {
422 		OutputTreeText(sink, node, mDepth);
423 		out << "Loop Terminal Expression\n";
424 		node->getExpression()->traverse(this);
425 	}
426 
427 	--mDepth;
428 
429 	return false;
430 }
431 
visitBranch(Visit visit,TIntermBranch * node)432 bool TOutputTraverser::visitBranch(Visit visit, TIntermBranch* node)
433 {
434 	TInfoSinkBase& out = sink;
435 
436 	OutputTreeText(out, node, mDepth);
437 
438 	switch (node->getFlowOp()) {
439 	case EOpKill:      out << "Branch: Kill";           break;
440 	case EOpBreak:     out << "Branch: Break";          break;
441 	case EOpContinue:  out << "Branch: Continue";       break;
442 	case EOpReturn:    out << "Branch: Return";         break;
443 	default:           out << "Branch: Unknown Branch"; break;
444 	}
445 
446 	if (node->getExpression()) {
447 		out << " with expression\n";
448 		++mDepth;
449 		node->getExpression()->traverse(this);
450 		--mDepth;
451 	} else
452 		out << "\n";
453 
454 	return false;
455 }
456 
457 //
458 // This function is the one to call externally to start the traversal.
459 // Individual functions can be initialized to 0 to skip processing of that
460 // type of node.  It's children will still be processed.
461 //
outputTree(TIntermNode * root)462 void TIntermediate::outputTree(TIntermNode* root)
463 {
464 	if (root == 0)
465 		return;
466 
467 	TOutputTraverser it(infoSink.info);
468 
469 	root->traverse(&it);
470 }
471