1 //
2 // Copyright (C) 2013 LunarG, Inc.
3 //
4 // All rights reserved.
5 //
6 // Redistribution and use in source and binary forms, with or without
7 // modification, are permitted provided that the following conditions
8 // are met:
9 //
10 // Redistributions of source code must retain the above copyright
11 // notice, this list of conditions and the following disclaimer.
12 //
13 // Redistributions in binary form must reproduce the above
14 // copyright notice, this list of conditions and the following
15 // disclaimer in the documentation and/or other materials provided
16 // with the distribution.
17 //
18 // Neither the name of 3Dlabs Inc. Ltd. nor the names of its
19 // contributors may be used to endorse or promote products derived
20 // from this software without specific prior written permission.
21 //
22 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
23 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
24 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
25 // FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
26 // COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
27 // INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
28 // BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
29 // LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
30 // CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
31 // LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
32 // ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
33 // POSSIBILITY OF SUCH DAMAGE.
34 //
35
36 //
37 // Do sub tree walks for
38 // 1) inductive loop bodies to see if the inductive variable is modified
39 // 2) array-index expressions to see if they are "constant-index-expression"
40 //
41 // These are per Appendix A of ES 2.0:
42 //
43 // "Within the body of the loop, the loop index is not statically assigned to nor is it used as the
44 // argument to a function out or inout parameter."
45 //
46 // "The following are constant-index-expressions:
47 // - Constant expressions
48 // - Loop indices as defined in section 4
49 // - Expressions composed of both of the above"
50 //
51 // N.B.: assuming the last rule excludes function calls
52 //
53
54 #include "ParseHelper.h"
55
56 namespace glslang {
57
58 //
59 // The inductive loop-body traverser.
60 //
61 // Just look at things that might modify the loop index.
62 //
63
64 class TInductiveTraverser : public TIntermTraverser {
65 public:
TInductiveTraverser(int id,TSymbolTable & st)66 TInductiveTraverser(int id, TSymbolTable& st)
67 : loopId(id), symbolTable(st), bad(false) { }
68
69 virtual bool visitBinary(TVisit, TIntermBinary* node);
70 virtual bool visitUnary(TVisit, TIntermUnary* node);
71 virtual bool visitAggregate(TVisit, TIntermAggregate* node);
72
73 int loopId; // unique ID of the symbol that's the loop inductive variable
74 TSymbolTable& symbolTable;
75 bool bad;
76 TSourceLoc badLoc;
77
78 protected:
79 TInductiveTraverser(TInductiveTraverser&);
80 TInductiveTraverser& operator=(TInductiveTraverser&);
81 };
82
83 // check binary operations for those modifying the loop index
visitBinary(TVisit,TIntermBinary * node)84 bool TInductiveTraverser::visitBinary(TVisit /* visit */, TIntermBinary* node)
85 {
86 if (node->modifiesState() && node->getLeft()->getAsSymbolNode() &&
87 node->getLeft()->getAsSymbolNode()->getId() == loopId) {
88 bad = true;
89 badLoc = node->getLoc();
90 }
91
92 return true;
93 }
94
95 // check unary operations for those modifying the loop index
visitUnary(TVisit,TIntermUnary * node)96 bool TInductiveTraverser::visitUnary(TVisit /* visit */, TIntermUnary* node)
97 {
98 if (node->modifiesState() && node->getOperand()->getAsSymbolNode() &&
99 node->getOperand()->getAsSymbolNode()->getId() == loopId) {
100 bad = true;
101 badLoc = node->getLoc();
102 }
103
104 return true;
105 }
106
107 // check function calls for arguments modifying the loop index
visitAggregate(TVisit,TIntermAggregate * node)108 bool TInductiveTraverser::visitAggregate(TVisit /* visit */, TIntermAggregate* node)
109 {
110 if (node->getOp() == EOpFunctionCall) {
111 // see if an out or inout argument is the loop index
112 const TIntermSequence& args = node->getSequence();
113 for (int i = 0; i < (int)args.size(); ++i) {
114 if (args[i]->getAsSymbolNode() && args[i]->getAsSymbolNode()->getId() == loopId) {
115 TSymbol* function = symbolTable.find(node->getName());
116 const TType* type = (*function->getAsFunction())[i].type;
117 if (type->getQualifier().storage == EvqOut ||
118 type->getQualifier().storage == EvqInOut) {
119 bad = true;
120 badLoc = node->getLoc();
121 }
122 }
123 }
124 }
125
126 return true;
127 }
128
129 //
130 // External function to call for loop check.
131 //
inductiveLoopBodyCheck(TIntermNode * body,int loopId,TSymbolTable & symbolTable)132 void TParseContext::inductiveLoopBodyCheck(TIntermNode* body, int loopId, TSymbolTable& symbolTable)
133 {
134 TInductiveTraverser it(loopId, symbolTable);
135
136 if (body == nullptr)
137 return;
138
139 body->traverse(&it);
140
141 if (it.bad)
142 error(it.badLoc, "inductive loop index modified", "limitations", "");
143 }
144
145 //
146 // The "constant-index-expression" tranverser.
147 //
148 // Just look at things that can form an index.
149 //
150
151 class TIndexTraverser : public TIntermTraverser {
152 public:
TIndexTraverser(const TIdSetType & ids)153 TIndexTraverser(const TIdSetType& ids) : inductiveLoopIds(ids), bad(false) { }
154 virtual void visitSymbol(TIntermSymbol* symbol);
155 virtual bool visitAggregate(TVisit, TIntermAggregate* node);
156 const TIdSetType& inductiveLoopIds;
157 bool bad;
158 TSourceLoc badLoc;
159
160 protected:
161 TIndexTraverser(TIndexTraverser&);
162 TIndexTraverser& operator=(TIndexTraverser&);
163 };
164
165 // make sure symbols are inductive-loop indexes
visitSymbol(TIntermSymbol * symbol)166 void TIndexTraverser::visitSymbol(TIntermSymbol* symbol)
167 {
168 if (inductiveLoopIds.find(symbol->getId()) == inductiveLoopIds.end()) {
169 bad = true;
170 badLoc = symbol->getLoc();
171 }
172 }
173
174 // check for function calls, assuming they are bad; spec. doesn't really say
visitAggregate(TVisit,TIntermAggregate * node)175 bool TIndexTraverser::visitAggregate(TVisit /* visit */, TIntermAggregate* node)
176 {
177 if (node->getOp() == EOpFunctionCall) {
178 bad = true;
179 badLoc = node->getLoc();
180 }
181
182 return true;
183 }
184
185 //
186 // External function to call for loop check.
187 //
constantIndexExpressionCheck(TIntermNode * index)188 void TParseContext::constantIndexExpressionCheck(TIntermNode* index)
189 {
190 #ifndef GLSLANG_WEB
191 TIndexTraverser it(inductiveLoopIds);
192
193 index->traverse(&it);
194
195 if (it.bad)
196 error(it.badLoc, "Non-constant-index-expression", "limitations", "");
197 #endif
198 }
199
200 } // end namespace glslang
201