blob: a602c38c93f6605a524cb897f283337984336394 [file] [log] [blame]
alokp@chromium.org07620a52010-09-23 17:53:56 +00001//
shannon.woods%transgaming.com@gtempaccount.com0bbed382013-04-13 03:38:07 +00002// Copyright (c) 2002-2013 The ANGLE Project Authors. All rights reserved.
alokp@chromium.org07620a52010-09-23 17:53:56 +00003// Use of this source code is governed by a BSD-style license that can be
4// found in the LICENSE file.
5//
6
zmo@google.com32e97312011-08-24 01:03:11 +00007#include "compiler/BuiltInFunctionEmulator.h"
zmo@google.comb1762df2011-07-30 02:04:23 +00008#include "compiler/DetectRecursion.h"
zmo@google.com0c6bb7a2011-08-17 19:39:58 +00009#include "compiler/ForLoopUnroll.h"
alokp@chromium.org07620a52010-09-23 17:53:56 +000010#include "compiler/Initialize.h"
alokp@chromium.org8b851c62012-06-15 16:25:11 +000011#include "compiler/InitializeParseContext.h"
zmo@google.comb9f64aa2012-01-20 00:35:15 +000012#include "compiler/MapLongVariableNames.h"
alokp@chromium.org07620a52010-09-23 17:53:56 +000013#include "compiler/ParseHelper.h"
maxvujovic@gmail.com430f5e02012-06-08 17:47:59 +000014#include "compiler/RenameFunction.h"
alokp@chromium.org07620a52010-09-23 17:53:56 +000015#include "compiler/ShHandle.h"
alokp@chromium.orgb59a7782010-11-24 18:38:33 +000016#include "compiler/ValidateLimitations.h"
gman@chromium.org8d804792012-10-17 21:33:48 +000017#include "compiler/VariablePacker.h"
maxvujovic@gmail.com66ebd012012-05-30 22:18:11 +000018#include "compiler/depgraph/DependencyGraph.h"
19#include "compiler/depgraph/DependencyGraphOutput.h"
20#include "compiler/timing/RestrictFragmentShaderTiming.h"
21#include "compiler/timing/RestrictVertexShaderTiming.h"
shannon.woods@transgaming.comda1ed362013-01-25 21:54:57 +000022#include "third_party/compiler/ArrayBoundsClamper.h"
alokp@chromium.org07620a52010-09-23 17:53:56 +000023
maxvujovic@gmail.com430f5e02012-06-08 17:47:59 +000024bool isWebGLBasedSpec(ShShaderSpec spec)
25{
26 return spec == SH_WEBGL_SPEC || spec == SH_CSS_SHADERS_SPEC;
27}
28
alokp@chromium.orgbafcbaa2010-11-23 19:07:43 +000029namespace {
30bool InitializeSymbolTable(
31 const TBuiltInStrings& builtInStrings,
32 ShShaderType type, ShShaderSpec spec, const ShBuiltInResources& resources,
33 TInfoSink& infoSink, TSymbolTable& symbolTable)
alokp@chromium.org07620a52010-09-23 17:53:56 +000034{
35 TIntermediate intermediate(infoSink);
36 TExtensionBehavior extBehavior;
zmo@google.com09c323a2011-08-12 18:22:25 +000037 InitExtensionBehavior(resources, extBehavior);
zmo@google.comdc4b4f82011-06-17 00:42:53 +000038 // The builtins deliberately don't specify precisions for the function
39 // arguments and return types. For that reason we don't try to check them.
40 TParseContext parseContext(symbolTable, extBehavior, intermediate, type, spec, 0, false, NULL, infoSink);
shannon.woods%transgaming.com@gtempaccount.comcbb6b6a2013-04-13 03:27:47 +000041 parseContext.fragmentPrecisionHigh = resources.FragmentPrecisionHigh == 1;
alokp@chromium.org07620a52010-09-23 17:53:56 +000042
43 GlobalParseContext = &parseContext;
44
alokp@chromium.org07620a52010-09-23 17:53:56 +000045 assert(symbolTable.isEmpty());
46 //
47 // Parse the built-ins. This should only happen once per
48 // language symbol table.
49 //
50 // Push the symbol table to give it an initial scope. This
51 // push should not have a corresponding pop, so that built-ins
52 // are preserved, and the test for an empty table fails.
53 //
54 symbolTable.push();
alokp@chromium.org07620a52010-09-23 17:53:56 +000055
56 for (TBuiltInStrings::const_iterator i = builtInStrings.begin(); i != builtInStrings.end(); ++i)
57 {
alokp@chromium.org570bfc72010-09-24 17:19:25 +000058 const char* builtInShaders = i->c_str();
59 int builtInLengths = static_cast<int>(i->size());
60 if (builtInLengths <= 0)
61 continue;
alokp@chromium.org07620a52010-09-23 17:53:56 +000062
alokp@chromium.org044a5cf2010-11-12 15:42:16 +000063 if (PaParseStrings(1, &builtInShaders, &builtInLengths, &parseContext) != 0)
alokp@chromium.org07620a52010-09-23 17:53:56 +000064 {
65 infoSink.info.message(EPrefixInternalError, "Unable to parse built-ins");
66 return false;
67 }
68 }
69
alokp@chromium.org4888ceb2010-10-01 21:13:12 +000070 IdentifyBuiltIns(type, spec, resources, symbolTable);
alokp@chromium.org07620a52010-09-23 17:53:56 +000071
alokp@chromium.org07620a52010-09-23 17:53:56 +000072 return true;
73}
74
alokp@chromium.orgbafcbaa2010-11-23 19:07:43 +000075class TScopedPoolAllocator {
76public:
77 TScopedPoolAllocator(TPoolAllocator* allocator, bool pushPop)
78 : mAllocator(allocator), mPushPopAllocator(pushPop) {
79 if (mPushPopAllocator) mAllocator->push();
80 SetGlobalPoolAllocator(mAllocator);
81 }
82 ~TScopedPoolAllocator() {
83 SetGlobalPoolAllocator(NULL);
84 if (mPushPopAllocator) mAllocator->pop();
85 }
86
87private:
88 TPoolAllocator* mAllocator;
89 bool mPushPopAllocator;
90};
91} // namespace
92
93TShHandleBase::TShHandleBase() {
94 allocator.push();
95 SetGlobalPoolAllocator(&allocator);
96}
97
98TShHandleBase::~TShHandleBase() {
99 SetGlobalPoolAllocator(NULL);
100 allocator.popAll();
101}
102
alokp@chromium.org4888ceb2010-10-01 21:13:12 +0000103TCompiler::TCompiler(ShShaderType type, ShShaderSpec spec)
104 : shaderType(type),
zmo@google.comf420c422011-09-12 18:27:59 +0000105 shaderSpec(spec),
shannon.woods%transgaming.com@gtempaccount.comcbb6b6a2013-04-13 03:27:47 +0000106 fragmentPrecisionHigh(false),
shannon.woods@transgaming.com1d432bb2013-01-25 21:57:28 +0000107 clampingStrategy(SH_CLAMP_WITH_CLAMP_INTRINSIC),
zmo@google.com9996b8e2012-01-19 01:43:55 +0000108 builtInFunctionEmulator(type)
alokp@chromium.org4888ceb2010-10-01 21:13:12 +0000109{
zmo@google.comb9f64aa2012-01-20 00:35:15 +0000110 longNameMap = LongNameMap::GetInstance();
alokp@chromium.org4888ceb2010-10-01 21:13:12 +0000111}
112
113TCompiler::~TCompiler()
114{
zmo@google.comb9f64aa2012-01-20 00:35:15 +0000115 ASSERT(longNameMap);
116 longNameMap->Release();
alokp@chromium.org4888ceb2010-10-01 21:13:12 +0000117}
118
119bool TCompiler::Init(const ShBuiltInResources& resources)
alokp@chromium.org07620a52010-09-23 17:53:56 +0000120{
shannon.woods%transgaming.com@gtempaccount.com0bbed382013-04-13 03:38:07 +0000121 shaderVersion = 100;
gman@chromium.org8d804792012-10-17 21:33:48 +0000122 maxUniformVectors = (shaderType == SH_VERTEX_SHADER) ?
123 resources.MaxVertexUniformVectors :
124 resources.MaxFragmentUniformVectors;
alokp@chromium.orgbafcbaa2010-11-23 19:07:43 +0000125 TScopedPoolAllocator scopedAlloc(&allocator, false);
126
alokp@chromium.org07620a52010-09-23 17:53:56 +0000127 // Generate built-in symbol table.
128 if (!InitBuiltInSymbolTable(resources))
129 return false;
alokp@chromium.org07620a52010-09-23 17:53:56 +0000130 InitExtensionBehavior(resources, extensionBehavior);
shannon.woods%transgaming.com@gtempaccount.comcbb6b6a2013-04-13 03:27:47 +0000131 fragmentPrecisionHigh = resources.FragmentPrecisionHigh == 1;
alokp@chromium.orgbafcbaa2010-11-23 19:07:43 +0000132
shannon.woods@transgaming.com1d432bb2013-01-25 21:57:28 +0000133 arrayBoundsClamper.SetClampingStrategy(resources.ArrayIndexClampingStrategy);
134 clampingStrategy = resources.ArrayIndexClampingStrategy;
135
daniel@transgaming.comc23f4612012-11-28 19:42:57 +0000136 hashFunction = resources.HashFunction;
137
alokp@chromium.org07620a52010-09-23 17:53:56 +0000138 return true;
139}
140
141bool TCompiler::compile(const char* const shaderStrings[],
shannon.woods@transgaming.comd64b3da2013-02-28 23:19:26 +0000142 size_t numStrings,
alokp@chromium.org07620a52010-09-23 17:53:56 +0000143 int compileOptions)
144{
alokp@chromium.orgbafcbaa2010-11-23 19:07:43 +0000145 TScopedPoolAllocator scopedAlloc(&allocator, true);
alokp@chromium.org07620a52010-09-23 17:53:56 +0000146 clearResults();
147
148 if (numStrings == 0)
149 return true;
150
alokp@chromium.orgb59a7782010-11-24 18:38:33 +0000151 // If compiling for WebGL, validate loop and indexing as well.
maxvujovic@gmail.com430f5e02012-06-08 17:47:59 +0000152 if (isWebGLBasedSpec(shaderSpec))
alokp@chromium.orgb59a7782010-11-24 18:38:33 +0000153 compileOptions |= SH_VALIDATE_LOOP_INDEXING;
alokp@chromium.org1f299542010-11-12 15:50:23 +0000154
apatrick@chromium.org0f4cefe2011-01-26 19:30:57 +0000155 // First string is path of source file if flag is set. The actual source follows.
156 const char* sourcePath = NULL;
shannon.woods@transgaming.comd64b3da2013-02-28 23:19:26 +0000157 size_t firstSource = 0;
apatrick@chromium.org0f4cefe2011-01-26 19:30:57 +0000158 if (compileOptions & SH_SOURCE_PATH)
159 {
160 sourcePath = shaderStrings[0];
161 ++firstSource;
162 }
163
alokp@chromium.org07620a52010-09-23 17:53:56 +0000164 TIntermediate intermediate(infoSink);
165 TParseContext parseContext(symbolTable, extensionBehavior, intermediate,
zmo@google.comdc4b4f82011-06-17 00:42:53 +0000166 shaderType, shaderSpec, compileOptions, true,
apatrick@chromium.org0f4cefe2011-01-26 19:30:57 +0000167 sourcePath, infoSink);
shannon.woods%transgaming.com@gtempaccount.comcbb6b6a2013-04-13 03:27:47 +0000168 parseContext.fragmentPrecisionHigh = fragmentPrecisionHigh;
alokp@chromium.org07620a52010-09-23 17:53:56 +0000169 GlobalParseContext = &parseContext;
alokp@chromium.org07620a52010-09-23 17:53:56 +0000170
171 // We preserve symbols at the built-in level from compile-to-compile.
172 // Start pushing the user-defined symbols at global level.
173 symbolTable.push();
174 if (!symbolTable.atGlobalLevel())
175 infoSink.info.message(EPrefixInternalError, "Wrong symbol table level");
176
177 // Parse shader.
178 bool success =
apatrick@chromium.org0f4cefe2011-01-26 19:30:57 +0000179 (PaParseStrings(numStrings - firstSource, &shaderStrings[firstSource], NULL, &parseContext) == 0) &&
alokp@chromium.org07620a52010-09-23 17:53:56 +0000180 (parseContext.treeRoot != NULL);
shannon.woods%transgaming.com@gtempaccount.com0bbed382013-04-13 03:38:07 +0000181
182 shaderVersion = parseContext.shaderVersion();
183
alokp@chromium.org07620a52010-09-23 17:53:56 +0000184 if (success) {
alokp@chromium.orgb59a7782010-11-24 18:38:33 +0000185 TIntermNode* root = parseContext.treeRoot;
186 success = intermediate.postProcess(root);
187
zmo@google.comb1762df2011-07-30 02:04:23 +0000188 if (success)
189 success = detectRecursion(root);
190
alokp@chromium.orgb59a7782010-11-24 18:38:33 +0000191 if (success && (compileOptions & SH_VALIDATE_LOOP_INDEXING))
192 success = validateLimitations(root);
alokp@chromium.org07620a52010-09-23 17:53:56 +0000193
maxvujovic@gmail.com66ebd012012-05-30 22:18:11 +0000194 if (success && (compileOptions & SH_TIMING_RESTRICTIONS))
maxvujovic@gmail.com77222c92012-06-04 21:06:05 +0000195 success = enforceTimingRestrictions(root, (compileOptions & SH_DEPENDENCY_GRAPH) != 0);
maxvujovic@gmail.com66ebd012012-05-30 22:18:11 +0000196
maxvujovic@gmail.com430f5e02012-06-08 17:47:59 +0000197 if (success && shaderSpec == SH_CSS_SHADERS_SPEC)
198 rewriteCSSShader(root);
199
zmo@google.com0c6bb7a2011-08-17 19:39:58 +0000200 // Unroll for-loop markup needs to happen after validateLimitations pass.
201 if (success && (compileOptions & SH_UNROLL_FOR_LOOP_WITH_INTEGER_INDEX))
202 ForLoopUnroll::MarkForLoopsWithIntegerIndicesForUnrolling(root);
203
zmo@google.com32e97312011-08-24 01:03:11 +0000204 // Built-in function emulation needs to happen after validateLimitations pass.
205 if (success && (compileOptions & SH_EMULATE_BUILT_IN_FUNCTIONS))
206 builtInFunctionEmulator.MarkBuiltInFunctionsForEmulation(root);
207
daniel@transgaming.com4167cc92013-01-11 04:11:53 +0000208 // Clamping uniform array bounds needs to happen after validateLimitations pass.
209 if (success && (compileOptions & SH_CLAMP_INDIRECT_ARRAY_BOUNDS))
210 arrayBoundsClamper.MarkIndirectArrayBoundsForClamping(root);
211
zmo@google.comfd747b82011-04-23 01:30:07 +0000212 // Call mapLongVariableNames() before collectAttribsUniforms() so in
213 // collectAttribsUniforms() we already have the mapped symbol names and
214 // we could composite mapped and original variable names.
daniel@transgaming.com0aa3b5a2012-11-28 19:43:24 +0000215 // Also, if we hash all the names, then no need to do this for long names.
216 if (success && (compileOptions & SH_MAP_LONG_VARIABLE_NAMES) && hashFunction == NULL)
zmo@google.comfd747b82011-04-23 01:30:07 +0000217 mapLongVariableNames(root);
218
gman@chromium.org8d804792012-10-17 21:33:48 +0000219 if (success && (compileOptions & SH_ATTRIBUTES_UNIFORMS)) {
zmo@google.comfd747b82011-04-23 01:30:07 +0000220 collectAttribsUniforms(root);
gman@chromium.org8d804792012-10-17 21:33:48 +0000221 if (compileOptions & SH_ENFORCE_PACKING_RESTRICTIONS) {
222 success = enforcePackingRestrictions();
223 if (!success) {
224 infoSink.info.message(EPrefixError, "too many uniforms");
225 }
226 }
227 }
zmo@google.comfd747b82011-04-23 01:30:07 +0000228
alokp@chromium.org4888ceb2010-10-01 21:13:12 +0000229 if (success && (compileOptions & SH_INTERMEDIATE_TREE))
alokp@chromium.orgb59a7782010-11-24 18:38:33 +0000230 intermediate.outputTree(root);
alokp@chromium.org07620a52010-09-23 17:53:56 +0000231
alokp@chromium.org4888ceb2010-10-01 21:13:12 +0000232 if (success && (compileOptions & SH_OBJECT_CODE))
alokp@chromium.orgb59a7782010-11-24 18:38:33 +0000233 translate(root);
alokp@chromium.org07620a52010-09-23 17:53:56 +0000234 }
235
236 // Cleanup memory.
237 intermediate.remove(parseContext.treeRoot);
238 // Ensure symbol table is returned to the built-in level,
239 // throwing away all but the built-ins.
240 while (!symbolTable.atBuiltInLevel())
241 symbolTable.pop();
alokp@chromium.org07620a52010-09-23 17:53:56 +0000242
243 return success;
244}
245
alokp@chromium.org4888ceb2010-10-01 21:13:12 +0000246bool TCompiler::InitBuiltInSymbolTable(const ShBuiltInResources& resources)
alokp@chromium.org07620a52010-09-23 17:53:56 +0000247{
248 TBuiltIns builtIns;
249
shannon.woods%transgaming.com@gtempaccount.com18b4c4b2013-04-13 03:31:40 +0000250 compileResources = resources;
alokp@chromium.org4888ceb2010-10-01 21:13:12 +0000251 builtIns.initialize(shaderType, shaderSpec, resources);
252 return InitializeSymbolTable(builtIns.getBuiltInStrings(),
253 shaderType, shaderSpec, resources, infoSink, symbolTable);
alokp@chromium.org07620a52010-09-23 17:53:56 +0000254}
255
256void TCompiler::clearResults()
257{
daniel@transgaming.com4167cc92013-01-11 04:11:53 +0000258 arrayBoundsClamper.Cleanup();
alokp@chromium.org07620a52010-09-23 17:53:56 +0000259 infoSink.info.erase();
260 infoSink.obj.erase();
261 infoSink.debug.erase();
262
263 attribs.clear();
264 uniforms.clear();
zmo@google.coma3b4ab42011-09-16 00:53:26 +0000265
266 builtInFunctionEmulator.Cleanup();
daniel@transgaming.com0aa3b5a2012-11-28 19:43:24 +0000267
268 nameMap.clear();
alokp@chromium.org07620a52010-09-23 17:53:56 +0000269}
270
zmo@google.comb1762df2011-07-30 02:04:23 +0000271bool TCompiler::detectRecursion(TIntermNode* root)
272{
273 DetectRecursion detect;
274 root->traverse(&detect);
275 switch (detect.detectRecursion()) {
276 case DetectRecursion::kErrorNone:
277 return true;
278 case DetectRecursion::kErrorMissingMain:
279 infoSink.info.message(EPrefixError, "Missing main()");
280 return false;
281 case DetectRecursion::kErrorRecursion:
282 infoSink.info.message(EPrefixError, "Function recursion detected");
283 return false;
284 default:
285 UNREACHABLE();
286 return false;
287 }
288}
289
maxvujovic@gmail.com430f5e02012-06-08 17:47:59 +0000290void TCompiler::rewriteCSSShader(TIntermNode* root)
291{
292 RenameFunction renamer("main(", "css_main(");
293 root->traverse(&renamer);
294}
295
alokp@chromium.orgb59a7782010-11-24 18:38:33 +0000296bool TCompiler::validateLimitations(TIntermNode* root) {
297 ValidateLimitations validate(shaderType, infoSink.info);
298 root->traverse(&validate);
299 return validate.numErrors() == 0;
300}
301
maxvujovic@gmail.com77222c92012-06-04 21:06:05 +0000302bool TCompiler::enforceTimingRestrictions(TIntermNode* root, bool outputGraph)
maxvujovic@gmail.com66ebd012012-05-30 22:18:11 +0000303{
304 if (shaderSpec != SH_WEBGL_SPEC) {
305 infoSink.info << "Timing restrictions must be enforced under the WebGL spec.";
306 return false;
307 }
308
309 if (shaderType == SH_FRAGMENT_SHADER) {
310 TDependencyGraph graph(root);
311
312 // Output any errors first.
maxvujovic@gmail.com77222c92012-06-04 21:06:05 +0000313 bool success = enforceFragmentShaderTimingRestrictions(graph);
maxvujovic@gmail.com66ebd012012-05-30 22:18:11 +0000314
315 // Then, output the dependency graph.
316 if (outputGraph) {
317 TDependencyGraphOutput output(infoSink.info);
318 output.outputAllSpanningTrees(graph);
319 }
320
321 return success;
322 }
323 else {
maxvujovic@gmail.com77222c92012-06-04 21:06:05 +0000324 return enforceVertexShaderTimingRestrictions(root);
maxvujovic@gmail.com66ebd012012-05-30 22:18:11 +0000325 }
326}
327
maxvujovic@gmail.com77222c92012-06-04 21:06:05 +0000328bool TCompiler::enforceFragmentShaderTimingRestrictions(const TDependencyGraph& graph)
maxvujovic@gmail.com66ebd012012-05-30 22:18:11 +0000329{
maxvujovic@gmail.com77222c92012-06-04 21:06:05 +0000330 RestrictFragmentShaderTiming restrictor(infoSink.info);
maxvujovic@gmail.com66ebd012012-05-30 22:18:11 +0000331 restrictor.enforceRestrictions(graph);
332 return restrictor.numErrors() == 0;
333}
334
maxvujovic@gmail.com77222c92012-06-04 21:06:05 +0000335bool TCompiler::enforceVertexShaderTimingRestrictions(TIntermNode* root)
maxvujovic@gmail.com66ebd012012-05-30 22:18:11 +0000336{
maxvujovic@gmail.com77222c92012-06-04 21:06:05 +0000337 RestrictVertexShaderTiming restrictor(infoSink.info);
maxvujovic@gmail.com66ebd012012-05-30 22:18:11 +0000338 restrictor.enforceRestrictions(root);
339 return restrictor.numErrors() == 0;
340}
341
alokp@chromium.org07620a52010-09-23 17:53:56 +0000342void TCompiler::collectAttribsUniforms(TIntermNode* root)
343{
daniel@transgaming.com0aa3b5a2012-11-28 19:43:24 +0000344 CollectAttribsUniforms collect(attribs, uniforms, hashFunction);
alokp@chromium.org07620a52010-09-23 17:53:56 +0000345 root->traverse(&collect);
346}
zmo@google.comfd747b82011-04-23 01:30:07 +0000347
gman@chromium.org8d804792012-10-17 21:33:48 +0000348bool TCompiler::enforcePackingRestrictions()
349{
350 VariablePacker packer;
351 return packer.CheckVariablesWithinPackingLimits(maxUniformVectors, uniforms);
352}
353
zmo@google.comfd747b82011-04-23 01:30:07 +0000354void TCompiler::mapLongVariableNames(TIntermNode* root)
355{
zmo@google.comb9f64aa2012-01-20 00:35:15 +0000356 ASSERT(longNameMap);
357 MapLongVariableNames map(longNameMap);
zmo@google.com9996b8e2012-01-19 01:43:55 +0000358 root->traverse(&map);
zmo@google.comfd747b82011-04-23 01:30:07 +0000359}
360
361int TCompiler::getMappedNameMaxLength() const
362{
kbr@chromium.org22152112011-10-26 01:18:28 +0000363 return MAX_SHORTENED_IDENTIFIER_SIZE + 1;
zmo@google.comfd747b82011-04-23 01:30:07 +0000364}
zmo@google.com5601ea02011-06-10 18:23:25 +0000365
366const TExtensionBehavior& TCompiler::getExtensionBehavior() const
367{
368 return extensionBehavior;
369}
zmo@google.com32e97312011-08-24 01:03:11 +0000370
shannon.woods%transgaming.com@gtempaccount.com18b4c4b2013-04-13 03:31:40 +0000371const ShBuiltInResources& TCompiler::getResources() const
372{
373 return compileResources;
374}
375
daniel@transgaming.com4167cc92013-01-11 04:11:53 +0000376const ArrayBoundsClamper& TCompiler::getArrayBoundsClamper() const
377{
378 return arrayBoundsClamper;
379}
380
shannon.woods@transgaming.com1d432bb2013-01-25 21:57:28 +0000381ShArrayIndexClampingStrategy TCompiler::getArrayIndexClampingStrategy() const
382{
383 return clampingStrategy;
384}
385
386const BuiltInFunctionEmulator& TCompiler::getBuiltInFunctionEmulator() const
387{
388 return builtInFunctionEmulator;
389}