1 //
2 // Copyright 2002 The ANGLE Project Authors. All rights reserved.
3 // Use of this source code is governed by a BSD-style license that can be
4 // found in the LICENSE file.
5 //
6 // Symbol table for parsing. The design principles and most of the functionality are documented in
7 // the header file.
8 //
9
10 #if defined(_MSC_VER)
11 # pragma warning(disable : 4718)
12 #endif
13
14 #include "compiler/translator/SymbolTable.h"
15
16 #include "angle_gl.h"
17 #include "compiler/translator/ImmutableString.h"
18 #include "compiler/translator/IntermNode.h"
19 #include "compiler/translator/StaticType.h"
20 #include "compiler/translator/util.h"
21
22 namespace sh
23 {
24 namespace
25 {
CheckShaderType(Shader expected,GLenum actual)26 bool CheckShaderType(Shader expected, GLenum actual)
27 {
28 switch (expected)
29 {
30 case Shader::ALL:
31 return true;
32 case Shader::FRAGMENT:
33 return actual == GL_FRAGMENT_SHADER;
34 case Shader::VERTEX:
35 return actual == GL_VERTEX_SHADER;
36 case Shader::COMPUTE:
37 return actual == GL_COMPUTE_SHADER;
38 case Shader::GEOMETRY:
39 return actual == GL_GEOMETRY_SHADER;
40 case Shader::GEOMETRY_EXT:
41 return actual == GL_GEOMETRY_SHADER_EXT;
42 case Shader::TESS_CONTROL_EXT:
43 return actual == GL_TESS_CONTROL_SHADER_EXT;
44 case Shader::TESS_EVALUATION_EXT:
45 return actual == GL_TESS_EVALUATION_SHADER_EXT;
46 case Shader::NOT_COMPUTE:
47 return actual != GL_COMPUTE_SHADER;
48 default:
49 UNREACHABLE();
50 return false;
51 }
52 }
53
CheckExtension(uint32_t extensionIndex,const ShBuiltInResources & resources)54 bool CheckExtension(uint32_t extensionIndex, const ShBuiltInResources &resources)
55 {
56 const int *resourcePtr = reinterpret_cast<const int *>(&resources);
57 return resourcePtr[extensionIndex] > 0;
58 }
59 } // namespace
60
61 class TSymbolTable::TSymbolTableLevel
62 {
63 public:
64 TSymbolTableLevel() = default;
65
66 bool insert(TSymbol *symbol);
67
68 // Insert a function using its unmangled name as the key.
69 void insertUnmangled(TFunction *function);
70
71 TSymbol *find(const ImmutableString &name) const;
72
73 private:
74 using tLevel = TUnorderedMap<ImmutableString,
75 TSymbol *,
76 ImmutableString::FowlerNollVoHash<sizeof(size_t)>>;
77 using tLevelPair = const tLevel::value_type;
78 using tInsertResult = std::pair<tLevel::iterator, bool>;
79
80 tLevel level;
81 };
82
insert(TSymbol * symbol)83 bool TSymbolTable::TSymbolTableLevel::insert(TSymbol *symbol)
84 {
85 // returning true means symbol was added to the table
86 tInsertResult result = level.insert(tLevelPair(symbol->getMangledName(), symbol));
87 return result.second;
88 }
89
insertUnmangled(TFunction * function)90 void TSymbolTable::TSymbolTableLevel::insertUnmangled(TFunction *function)
91 {
92 level.insert(tLevelPair(function->name(), function));
93 }
94
find(const ImmutableString & name) const95 TSymbol *TSymbolTable::TSymbolTableLevel::find(const ImmutableString &name) const
96 {
97 tLevel::const_iterator it = level.find(name);
98 if (it == level.end())
99 return nullptr;
100 else
101 return (*it).second;
102 }
103
TSymbolTable()104 TSymbolTable::TSymbolTable()
105 : mGlobalInvariant(false),
106 mUniqueIdCounter(0),
107 mShaderType(GL_FRAGMENT_SHADER),
108 mShaderSpec(SH_GLES2_SPEC),
109 mGlInVariableWithArraySize(nullptr)
110 {}
111
112 TSymbolTable::~TSymbolTable() = default;
113
isEmpty() const114 bool TSymbolTable::isEmpty() const
115 {
116 return mTable.empty();
117 }
118
atGlobalLevel() const119 bool TSymbolTable::atGlobalLevel() const
120 {
121 return mTable.size() == 1u;
122 }
123
push()124 void TSymbolTable::push()
125 {
126 mTable.emplace_back(new TSymbolTableLevel);
127 mPrecisionStack.emplace_back(new PrecisionStackLevel);
128 }
129
pop()130 void TSymbolTable::pop()
131 {
132 mTable.pop_back();
133 mPrecisionStack.pop_back();
134 }
135
markFunctionHasPrototypeDeclaration(const ImmutableString & mangledName,bool * hadPrototypeDeclarationOut) const136 const TFunction *TSymbolTable::markFunctionHasPrototypeDeclaration(
137 const ImmutableString &mangledName,
138 bool *hadPrototypeDeclarationOut) const
139 {
140 TFunction *function = findUserDefinedFunction(mangledName);
141 *hadPrototypeDeclarationOut = function->hasPrototypeDeclaration();
142 function->setHasPrototypeDeclaration();
143 return function;
144 }
145
setFunctionParameterNamesFromDefinition(const TFunction * function,bool * wasDefinedOut) const146 const TFunction *TSymbolTable::setFunctionParameterNamesFromDefinition(const TFunction *function,
147 bool *wasDefinedOut) const
148 {
149 TFunction *firstDeclaration = findUserDefinedFunction(function->getMangledName());
150 ASSERT(firstDeclaration);
151 // Note: 'firstDeclaration' could be 'function' if this is the first time we've seen function as
152 // it would have just been put in the symbol table. Otherwise, we're looking up an earlier
153 // occurance.
154 if (function != firstDeclaration)
155 {
156 // The previous declaration should have the same parameters as the function definition
157 // (parameter names may differ).
158 firstDeclaration->shareParameters(*function);
159 }
160
161 *wasDefinedOut = firstDeclaration->isDefined();
162 firstDeclaration->setDefined();
163 return firstDeclaration;
164 }
165
setGlInArraySize(unsigned int inputArraySize)166 bool TSymbolTable::setGlInArraySize(unsigned int inputArraySize)
167 {
168 if (mGlInVariableWithArraySize)
169 {
170 return mGlInVariableWithArraySize->getType().getOutermostArraySize() == inputArraySize;
171 }
172 const TInterfaceBlock *glPerVertex = static_cast<const TInterfaceBlock *>(m_gl_PerVertex);
173 TType *glInType = new TType(glPerVertex, EvqPerVertexIn, TLayoutQualifier::Create());
174 glInType->makeArray(inputArraySize);
175 mGlInVariableWithArraySize =
176 new TVariable(this, ImmutableString("gl_in"), glInType, SymbolType::BuiltIn,
177 TExtension::EXT_geometry_shader);
178 return true;
179 }
180
getGlInVariableWithArraySize() const181 TVariable *TSymbolTable::getGlInVariableWithArraySize() const
182 {
183 return mGlInVariableWithArraySize;
184 }
185
gl_FragData() const186 const TVariable *TSymbolTable::gl_FragData() const
187 {
188 return static_cast<const TVariable *>(m_gl_FragData);
189 }
190
gl_SecondaryFragDataEXT() const191 const TVariable *TSymbolTable::gl_SecondaryFragDataEXT() const
192 {
193 return static_cast<const TVariable *>(m_gl_SecondaryFragDataEXT);
194 }
195
getOrCreateVariableMetadata(const TVariable & variable)196 TSymbolTable::VariableMetadata *TSymbolTable::getOrCreateVariableMetadata(const TVariable &variable)
197 {
198 int id = variable.uniqueId().get();
199 auto iter = mVariableMetadata.find(id);
200 if (iter == mVariableMetadata.end())
201 {
202 iter = mVariableMetadata.insert(std::make_pair(id, VariableMetadata())).first;
203 }
204 return &iter->second;
205 }
206
markStaticWrite(const TVariable & variable)207 void TSymbolTable::markStaticWrite(const TVariable &variable)
208 {
209 auto metadata = getOrCreateVariableMetadata(variable);
210 metadata->staticWrite = true;
211 }
212
markStaticRead(const TVariable & variable)213 void TSymbolTable::markStaticRead(const TVariable &variable)
214 {
215 auto metadata = getOrCreateVariableMetadata(variable);
216 metadata->staticRead = true;
217 }
218
isStaticallyUsed(const TVariable & variable) const219 bool TSymbolTable::isStaticallyUsed(const TVariable &variable) const
220 {
221 ASSERT(!variable.getConstPointer());
222 int id = variable.uniqueId().get();
223 auto iter = mVariableMetadata.find(id);
224 return iter != mVariableMetadata.end() && (iter->second.staticRead || iter->second.staticWrite);
225 }
226
addInvariantVarying(const TVariable & variable)227 void TSymbolTable::addInvariantVarying(const TVariable &variable)
228 {
229 ASSERT(atGlobalLevel());
230 auto metadata = getOrCreateVariableMetadata(variable);
231 metadata->invariant = true;
232 }
233
isVaryingInvariant(const TVariable & variable) const234 bool TSymbolTable::isVaryingInvariant(const TVariable &variable) const
235 {
236 ASSERT(atGlobalLevel());
237 if (mGlobalInvariant && (IsShaderOutput(variable.getType().getQualifier())))
238 {
239 return true;
240 }
241 int id = variable.uniqueId().get();
242 auto iter = mVariableMetadata.find(id);
243 return iter != mVariableMetadata.end() && iter->second.invariant;
244 }
245
setGlobalInvariant(bool invariant)246 void TSymbolTable::setGlobalInvariant(bool invariant)
247 {
248 ASSERT(atGlobalLevel());
249 mGlobalInvariant = invariant;
250 }
251
find(const ImmutableString & name,int shaderVersion) const252 const TSymbol *TSymbolTable::find(const ImmutableString &name, int shaderVersion) const
253 {
254 const TSymbol *userSymbol = findUserDefined(name);
255 if (userSymbol)
256 {
257 return userSymbol;
258 }
259
260 return findBuiltIn(name, shaderVersion);
261 }
262
findUserDefined(const ImmutableString & name) const263 const TSymbol *TSymbolTable::findUserDefined(const ImmutableString &name) const
264 {
265 int userDefinedLevel = static_cast<int>(mTable.size()) - 1;
266 while (userDefinedLevel >= 0)
267 {
268 const TSymbol *symbol = mTable[userDefinedLevel]->find(name);
269 if (symbol)
270 {
271 return symbol;
272 }
273 userDefinedLevel--;
274 }
275
276 return nullptr;
277 }
278
findUserDefinedFunction(const ImmutableString & name) const279 TFunction *TSymbolTable::findUserDefinedFunction(const ImmutableString &name) const
280 {
281 // User-defined functions are always declared at the global level.
282 ASSERT(!mTable.empty());
283 return static_cast<TFunction *>(mTable[0]->find(name));
284 }
285
findGlobal(const ImmutableString & name) const286 const TSymbol *TSymbolTable::findGlobal(const ImmutableString &name) const
287 {
288 ASSERT(!mTable.empty());
289 return mTable[0]->find(name);
290 }
291
findGlobalWithConversion(const std::vector<ImmutableString> & names) const292 const TSymbol *TSymbolTable::findGlobalWithConversion(
293 const std::vector<ImmutableString> &names) const
294 {
295 for (const ImmutableString &name : names)
296 {
297 const TSymbol *target = findGlobal(name);
298 if (target != nullptr)
299 return target;
300 }
301 return nullptr;
302 }
303
findBuiltInWithConversion(const std::vector<ImmutableString> & names,int shaderVersion) const304 const TSymbol *TSymbolTable::findBuiltInWithConversion(const std::vector<ImmutableString> &names,
305 int shaderVersion) const
306 {
307 for (const ImmutableString &name : names)
308 {
309 const TSymbol *target = findBuiltIn(name, shaderVersion);
310 if (target != nullptr)
311 return target;
312 }
313 return nullptr;
314 }
315
declare(TSymbol * symbol)316 bool TSymbolTable::declare(TSymbol *symbol)
317 {
318 ASSERT(!mTable.empty());
319 // The following built-ins may be redeclared by the shader: gl_ClipDistance, gl_CullDistance,
320 // gl_LastFragData, and gl_LastFragColorARM.
321 ASSERT(symbol->symbolType() == SymbolType::UserDefined ||
322 (symbol->symbolType() == SymbolType::BuiltIn && IsRedeclarableBuiltIn(symbol->name())));
323 ASSERT(!symbol->isFunction());
324 return mTable.back()->insert(symbol);
325 }
326
declareInternal(TSymbol * symbol)327 bool TSymbolTable::declareInternal(TSymbol *symbol)
328 {
329 ASSERT(!mTable.empty());
330 ASSERT(symbol->symbolType() == SymbolType::AngleInternal);
331 ASSERT(!symbol->isFunction());
332 return mTable.back()->insert(symbol);
333 }
334
declareUserDefinedFunction(TFunction * function,bool insertUnmangledName)335 void TSymbolTable::declareUserDefinedFunction(TFunction *function, bool insertUnmangledName)
336 {
337 ASSERT(!mTable.empty());
338 if (insertUnmangledName)
339 {
340 // Insert the unmangled name to detect potential future redefinition as a variable.
341 mTable[0]->insertUnmangled(function);
342 }
343 mTable[0]->insert(function);
344 }
345
setDefaultPrecision(TBasicType type,TPrecision prec)346 void TSymbolTable::setDefaultPrecision(TBasicType type, TPrecision prec)
347 {
348 int indexOfLastElement = static_cast<int>(mPrecisionStack.size()) - 1;
349 // Uses map operator [], overwrites the current value
350 (*mPrecisionStack[indexOfLastElement])[type] = prec;
351 }
352
getDefaultPrecision(TBasicType type) const353 TPrecision TSymbolTable::getDefaultPrecision(TBasicType type) const
354 {
355 if (!SupportsPrecision(type))
356 return EbpUndefined;
357
358 // unsigned integers use the same precision as signed
359 TBasicType baseType = (type == EbtUInt) ? EbtInt : type;
360
361 int level = static_cast<int>(mPrecisionStack.size()) - 1;
362 ASSERT(level >= 0); // Just to be safe. Should not happen.
363 // If we dont find anything we return this. Some types don't have predefined default precision.
364 TPrecision prec = EbpUndefined;
365 while (level >= 0)
366 {
367 PrecisionStackLevel::iterator it = mPrecisionStack[level]->find(baseType);
368 if (it != mPrecisionStack[level]->end())
369 {
370 prec = (*it).second;
371 break;
372 }
373 level--;
374 }
375 return prec;
376 }
377
clearCompilationResults()378 void TSymbolTable::clearCompilationResults()
379 {
380 mGlobalInvariant = false;
381 mUniqueIdCounter = kLastBuiltInId + 1;
382 mVariableMetadata.clear();
383 mGlInVariableWithArraySize = nullptr;
384
385 // User-defined scopes should have already been cleared when the compilation finished.
386 ASSERT(mTable.empty());
387 }
388
nextUniqueIdValue()389 int TSymbolTable::nextUniqueIdValue()
390 {
391 ASSERT(mUniqueIdCounter < std::numeric_limits<int>::max());
392 return ++mUniqueIdCounter;
393 }
394
initializeBuiltIns(sh::GLenum type,ShShaderSpec spec,const ShBuiltInResources & resources)395 void TSymbolTable::initializeBuiltIns(sh::GLenum type,
396 ShShaderSpec spec,
397 const ShBuiltInResources &resources)
398 {
399 mShaderType = type;
400 mShaderSpec = spec;
401 mResources = resources;
402
403 // We need just one precision stack level for predefined precisions.
404 mPrecisionStack.emplace_back(new PrecisionStackLevel);
405
406 if (IsDesktopGLSpec(spec))
407 {
408 setDefaultPrecision(EbtInt, EbpUndefined);
409 setDefaultPrecision(EbtFloat, EbpUndefined);
410 }
411 else
412 {
413 switch (type)
414 {
415 case GL_FRAGMENT_SHADER:
416 setDefaultPrecision(EbtInt, EbpMedium);
417 break;
418 case GL_VERTEX_SHADER:
419 case GL_COMPUTE_SHADER:
420 case GL_GEOMETRY_SHADER_EXT:
421 case GL_TESS_CONTROL_SHADER_EXT:
422 case GL_TESS_EVALUATION_SHADER_EXT:
423 setDefaultPrecision(EbtInt, EbpHigh);
424 setDefaultPrecision(EbtFloat, EbpHigh);
425 break;
426 default:
427 UNREACHABLE();
428 }
429 }
430
431 // Set defaults for sampler types that have default precision, even those that are
432 // only available if an extension exists.
433 // New sampler types in ESSL3 don't have default precision. ESSL1 types do.
434 initSamplerDefaultPrecision(EbtSampler2D);
435 initSamplerDefaultPrecision(EbtSamplerCube);
436 // SamplerExternalOES is specified in the extension to have default precision.
437 initSamplerDefaultPrecision(EbtSamplerExternalOES);
438 // SamplerExternal2DY2YEXT is specified in the extension to have default precision.
439 initSamplerDefaultPrecision(EbtSamplerExternal2DY2YEXT);
440 // It isn't specified whether Sampler2DRect has default precision.
441 initSamplerDefaultPrecision(EbtSampler2DRect);
442
443 if (spec < SH_GLES3_SPEC)
444 {
445 // Only set the default precision of shadow samplers in ESLL1. They become core in ESSL3
446 // where they do not have a defalut precision.
447 initSamplerDefaultPrecision(EbtSampler2DShadow);
448 }
449
450 setDefaultPrecision(EbtAtomicCounter, EbpHigh);
451
452 initializeBuiltInVariables(type, spec, resources);
453 mUniqueIdCounter = kLastBuiltInId + 1;
454 }
455
initSamplerDefaultPrecision(TBasicType samplerType)456 void TSymbolTable::initSamplerDefaultPrecision(TBasicType samplerType)
457 {
458 ASSERT(samplerType >= EbtGuardSamplerBegin && samplerType <= EbtGuardSamplerEnd);
459 setDefaultPrecision(samplerType, EbpLow);
460 }
461
VariableMetadata()462 TSymbolTable::VariableMetadata::VariableMetadata()
463 : staticRead(false), staticWrite(false), invariant(false)
464 {}
465
get(ShShaderSpec shaderSpec,int shaderVersion,sh::GLenum shaderType,const ShBuiltInResources & resources,const TSymbolTableBase & symbolTable) const466 const TSymbol *SymbolRule::get(ShShaderSpec shaderSpec,
467 int shaderVersion,
468 sh::GLenum shaderType,
469 const ShBuiltInResources &resources,
470 const TSymbolTableBase &symbolTable) const
471 {
472 if (IsDesktopGLSpec(shaderSpec) != (mIsDesktop == 1))
473 return nullptr;
474
475 if (mVersion == kESSL1Only && shaderVersion != static_cast<int>(kESSL1Only))
476 return nullptr;
477
478 if (mVersion > shaderVersion)
479 return nullptr;
480
481 if (!CheckShaderType(static_cast<Shader>(mShaders), shaderType))
482 return nullptr;
483
484 if (mExtensionIndex != 0 && !CheckExtension(mExtensionIndex, resources))
485 return nullptr;
486
487 return mIsVar > 0 ? symbolTable.*(mSymbolOrVar.var) : mSymbolOrVar.symbol;
488 }
489
FindMangledBuiltIn(ShShaderSpec shaderSpec,int shaderVersion,sh::GLenum shaderType,const ShBuiltInResources & resources,const TSymbolTableBase & symbolTable,const SymbolRule * rules,uint16_t startIndex,uint16_t endIndex)490 const TSymbol *FindMangledBuiltIn(ShShaderSpec shaderSpec,
491 int shaderVersion,
492 sh::GLenum shaderType,
493 const ShBuiltInResources &resources,
494 const TSymbolTableBase &symbolTable,
495 const SymbolRule *rules,
496 uint16_t startIndex,
497 uint16_t endIndex)
498 {
499 for (uint32_t ruleIndex = startIndex; ruleIndex < endIndex; ++ruleIndex)
500 {
501 const TSymbol *symbol =
502 rules[ruleIndex].get(shaderSpec, shaderVersion, shaderType, resources, symbolTable);
503 if (symbol)
504 {
505 return symbol;
506 }
507 }
508
509 return nullptr;
510 }
511
matches(const ImmutableString & name,ShShaderSpec shaderSpec,int shaderVersion,sh::GLenum shaderType,const TExtensionBehavior & extensions) const512 bool UnmangledEntry::matches(const ImmutableString &name,
513 ShShaderSpec shaderSpec,
514 int shaderVersion,
515 sh::GLenum shaderType,
516 const TExtensionBehavior &extensions) const
517 {
518 if (name != mName)
519 return false;
520
521 if (!CheckShaderType(static_cast<Shader>(mShaderType), shaderType))
522 return false;
523
524 if (IsDesktopGLSpec(shaderSpec))
525 {
526 if (mGLSLVersion > shaderVersion)
527 return false;
528
529 if (mGLSLExtension == TExtension::UNDEFINED)
530 return true;
531
532 return IsExtensionEnabled(extensions, mGLSLExtension);
533 }
534 else
535 {
536 if (mESSLVersion == kESSL1Only && shaderVersion != static_cast<int>(kESSL1Only))
537 return false;
538
539 if (mESSLVersion > shaderVersion)
540 return false;
541
542 bool anyExtension = false;
543 bool anyExtensionEnabled = false;
544 for (TExtension ext : mESSLExtensions)
545 {
546 if (ext != TExtension::UNDEFINED)
547 {
548 anyExtension = true;
549 anyExtensionEnabled = anyExtensionEnabled || IsExtensionEnabled(extensions, ext);
550 }
551 }
552
553 if (!anyExtension)
554 return true;
555
556 return anyExtensionEnabled;
557 }
558 }
559 } // namespace sh
560