blob: 8bd426f21b70abbd9edbda818af56356db44694d [file] [log] [blame]
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +00001//
shannon.woods%transgaming.com@gtempaccount.com0bbed382013-04-13 03:38:07 +00002// Copyright (c) 2002-2013 The ANGLE Project Authors. All rights reserved.
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +00003// Use of this source code is governed by a BSD-style license that can be
4// found in the LICENSE file.
5//
6
7//
alokp@chromium.org774d7062010-07-21 18:55:45 +00008// Implement the top-level of interface to the compiler,
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +00009// as defined in ShaderLang.h
10//
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000011
alokp@chromium.orgea0e1af2010-03-22 19:33:14 +000012#include "GLSLANG/ShaderLang.h"
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000013
Jamie Madilld4a3a312014-06-25 16:04:56 -040014#include "compiler/translator/Compiler.h"
Geoff Lang17732822013-08-29 13:46:49 -040015#include "compiler/translator/InitializeDll.h"
Jamie Madill5508f392014-02-20 13:31:36 -050016#include "compiler/translator/length_limits.h"
Daniel Bratell73941de2015-02-25 14:34:49 +010017#ifdef ANGLE_ENABLE_HLSL
Geoff Lang17732822013-08-29 13:46:49 -040018#include "compiler/translator/TranslatorHLSL.h"
Daniel Bratell73941de2015-02-25 14:34:49 +010019#endif // ANGLE_ENABLE_HLSL
Geoff Lang17732822013-08-29 13:46:49 -040020#include "compiler/translator/VariablePacker.h"
Jamie Madilla718c1e2014-07-02 15:31:22 -040021#include "angle_gl.h"
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000022
Jamie Madille294bb82014-07-17 14:16:26 -040023namespace
24{
25
26enum ShaderVariableType
27{
28 SHADERVAR_UNIFORM,
29 SHADERVAR_VARYING,
30 SHADERVAR_ATTRIBUTE,
31 SHADERVAR_OUTPUTVARIABLE,
32 SHADERVAR_INTERFACEBLOCK
33};
34
35bool isInitialized = false;
Jamie Madillb4d192c2014-02-26 09:54:10 -050036
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000037//
38// This is the platform independent interface between an OGL driver
alokp@chromium.org774d7062010-07-21 18:55:45 +000039// and the shading language compiler.
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +000040//
41
Jamie Madille294bb82014-07-17 14:16:26 -040042template <typename VarT>
43const std::vector<VarT> *GetVariableList(const TCompiler *compiler, ShaderVariableType variableType);
44
45template <>
46const std::vector<sh::Uniform> *GetVariableList(const TCompiler *compiler, ShaderVariableType)
47{
48 return &compiler->getUniforms();
49}
50
51template <>
52const std::vector<sh::Varying> *GetVariableList(const TCompiler *compiler, ShaderVariableType)
53{
54 return &compiler->getVaryings();
55}
56
57template <>
58const std::vector<sh::Attribute> *GetVariableList(const TCompiler *compiler, ShaderVariableType variableType)
59{
60 return (variableType == SHADERVAR_ATTRIBUTE ?
61 &compiler->getAttributes() :
62 &compiler->getOutputVariables());
63}
64
65template <>
66const std::vector<sh::InterfaceBlock> *GetVariableList(const TCompiler *compiler, ShaderVariableType)
67{
68 return &compiler->getInterfaceBlocks();
69}
70
71template <typename VarT>
72const std::vector<VarT> *GetShaderVariables(const ShHandle handle, ShaderVariableType variableType)
73{
74 if (!handle)
75 {
76 return NULL;
77 }
78
79 TShHandleBase* base = static_cast<TShHandleBase*>(handle);
80 TCompiler* compiler = base->getAsCompiler();
81 if (!compiler)
82 {
83 return NULL;
84 }
85
86 return GetVariableList<VarT>(compiler, variableType);
87}
88
Zhenyao Mo4de44cb2014-10-29 18:03:46 -070089TCompiler *GetCompilerFromHandle(ShHandle handle)
90{
91 if (!handle)
92 return NULL;
93 TShHandleBase *base = static_cast<TShHandleBase *>(handle);
94 return base->getAsCompiler();
Jamie Madille294bb82014-07-17 14:16:26 -040095}
96
Daniel Bratell73941de2015-02-25 14:34:49 +010097#ifdef ANGLE_ENABLE_HLSL
Zhenyao Mo4de44cb2014-10-29 18:03:46 -070098TranslatorHLSL *GetTranslatorHLSLFromHandle(ShHandle handle)
99{
100 if (!handle)
101 return NULL;
102 TShHandleBase *base = static_cast<TShHandleBase *>(handle);
103 return base->getAsTranslatorHLSL();
104}
Daniel Bratell73941de2015-02-25 14:34:49 +0100105#endif // ANGLE_ENABLE_HLSL
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700106
107} // namespace anonymous
108
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000109//
Alok Priyadarshib11713f2013-08-01 16:02:39 -0700110// Driver must call this first, once, before doing any other compiler operations.
111// Subsequent calls to this function are no-op.
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000112//
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700113bool ShInitialize()
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000114{
Jamie Madill477bc782014-02-26 09:54:17 -0500115 if (!isInitialized)
116 {
117 isInitialized = InitProcess();
118 }
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700119 return isInitialized;
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000120}
121
122//
alokp@chromium.org94a86ad2010-08-25 20:02:11 +0000123// Cleanup symbol tables
124//
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700125bool ShFinalize()
alokp@chromium.org94a86ad2010-08-25 20:02:11 +0000126{
Geoff Langf20f0202014-04-28 11:02:07 -0400127 if (isInitialized)
128 {
129 DetachProcess();
130 isInitialized = false;
131 }
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700132 return true;
alokp@chromium.org94a86ad2010-08-25 20:02:11 +0000133}
134
135//
136// Initialize built-in resources with minimum expected values.
137//
alokp@chromium.org4888ceb2010-10-01 21:13:12 +0000138void ShInitBuiltInResources(ShBuiltInResources* resources)
alokp@chromium.org94a86ad2010-08-25 20:02:11 +0000139{
Kimmo Kinnunen7c1cfd62014-10-15 14:59:57 +0300140 // Make comparable.
141 memset(resources, 0, sizeof(*resources));
142
alokp@chromium.org94a86ad2010-08-25 20:02:11 +0000143 // Constants.
144 resources->MaxVertexAttribs = 8;
145 resources->MaxVertexUniformVectors = 128;
146 resources->MaxVaryingVectors = 8;
147 resources->MaxVertexTextureImageUnits = 0;
148 resources->MaxCombinedTextureImageUnits = 8;
149 resources->MaxTextureImageUnits = 8;
150 resources->MaxFragmentUniformVectors = 16;
151 resources->MaxDrawBuffers = 1;
152
153 // Extensions.
154 resources->OES_standard_derivatives = 0;
zmo@google.com09c323a2011-08-12 18:22:25 +0000155 resources->OES_EGL_image_external = 0;
kbr@chromium.org205fef32011-11-22 20:50:02 +0000156 resources->ARB_texture_rectangle = 0;
Kimmo Kinnunenb18609b2015-07-16 14:13:11 +0300157 resources->EXT_blend_func_extended = 0;
shannon.woods@transgaming.com550cd092013-02-28 23:19:54 +0000158 resources->EXT_draw_buffers = 0;
Jamie Madill2aeb26a2013-07-08 14:02:55 -0400159 resources->EXT_frag_depth = 0;
Nicolas Capens46485082014-04-15 13:12:50 -0400160 resources->EXT_shader_texture_lod = 0;
Olli Etuaho853dc1a2014-11-06 17:25:48 +0200161 resources->WEBGL_debug_shader_precision = 0;
Erik Dahlströmea7a2122014-11-17 16:15:57 +0100162 resources->EXT_shader_framebuffer_fetch = 0;
163 resources->NV_shader_framebuffer_fetch = 0;
164 resources->ARM_shader_framebuffer_fetch = 0;
daniel@transgaming.comc23f4612012-11-28 19:42:57 +0000165
Olli Etuahoe61209a2014-09-26 12:01:17 +0300166 resources->NV_draw_buffers = 0;
167
shannon.woods%transgaming.com@gtempaccount.comcbb6b6a2013-04-13 03:27:47 +0000168 // Disable highp precision in fragment shader by default.
169 resources->FragmentPrecisionHigh = 0;
170
shannonwoods@chromium.org74b86cf2013-05-30 00:02:58 +0000171 // GLSL ES 3.0 constants.
172 resources->MaxVertexOutputVectors = 16;
173 resources->MaxFragmentInputVectors = 15;
174 resources->MinProgramTexelOffset = -8;
175 resources->MaxProgramTexelOffset = 7;
176
Kimmo Kinnunenb18609b2015-07-16 14:13:11 +0300177 // Extensions constants.
178 resources->MaxDualSourceDrawBuffers = 0;
179
daniel@transgaming.comc23f4612012-11-28 19:42:57 +0000180 // Disable name hashing by default.
181 resources->HashFunction = NULL;
shannon.woods@transgaming.com1d432bb2013-01-25 21:57:28 +0000182
183 resources->ArrayIndexClampingStrategy = SH_CLAMP_WITH_CLAMP_INTRINSIC;
Nicolas Capens7d649a02014-02-07 11:24:32 -0500184
185 resources->MaxExpressionComplexity = 256;
186 resources->MaxCallStackDepth = 256;
alokp@chromium.org94a86ad2010-08-25 20:02:11 +0000187}
188
189//
alokp@chromium.org774d7062010-07-21 18:55:45 +0000190// Driver calls these to create and destroy compiler objects.
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000191//
Jamie Madill183bde52014-07-02 15:31:19 -0400192ShHandle ShConstructCompiler(sh::GLenum type, ShShaderSpec spec,
zmo@google.com5601ea02011-06-10 18:23:25 +0000193 ShShaderOutput output,
alokp@chromium.org4888ceb2010-10-01 21:13:12 +0000194 const ShBuiltInResources* resources)
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000195{
zmo@google.com5601ea02011-06-10 18:23:25 +0000196 TShHandleBase* base = static_cast<TShHandleBase*>(ConstructCompiler(type, spec, output));
alokp@chromium.orge4249f02010-07-26 18:13:52 +0000197 TCompiler* compiler = base->getAsCompiler();
198 if (compiler == 0)
199 return 0;
200
201 // Generate built-in symbol table.
alokp@chromium.org07620a52010-09-23 17:53:56 +0000202 if (!compiler->Init(*resources)) {
alokp@chromium.orge4249f02010-07-26 18:13:52 +0000203 ShDestruct(base);
204 return 0;
205 }
206
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000207 return reinterpret_cast<void*>(base);
208}
209
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000210void ShDestruct(ShHandle handle)
211{
212 if (handle == 0)
213 return;
214
215 TShHandleBase* base = static_cast<TShHandleBase*>(handle);
216
217 if (base->getAsCompiler())
218 DeleteCompiler(base->getAsCompiler());
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000219}
220
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700221const std::string &ShGetBuiltInResourcesString(const ShHandle handle)
Shannon Woods2d76e5f2014-05-16 17:46:41 -0400222{
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700223 TCompiler *compiler = GetCompilerFromHandle(handle);
224 ASSERT(compiler);
225 return compiler->getBuiltInResourcesString();
Shannon Woods2d76e5f2014-05-16 17:46:41 -0400226}
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700227
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000228//
Shannon Woods2d76e5f2014-05-16 17:46:41 -0400229// Do an actual compile on the given strings. The result is left
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000230// in the given compile object.
231//
232// Return: The return value of ShCompile is really boolean, indicating
233// success or failure.
234//
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700235bool ShCompile(
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000236 const ShHandle handle,
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700237 const char *const shaderStrings[],
shannon.woods@transgaming.comd64b3da2013-02-28 23:19:26 +0000238 size_t numStrings,
alokp@chromium.org7beea402010-09-15 21:18:34 +0000239 int compileOptions)
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000240{
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700241 TCompiler *compiler = GetCompilerFromHandle(handle);
242 ASSERT(compiler);
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000243
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700244 return compiler->compile(shaderStrings, numStrings, compileOptions);
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000245}
246
Dmitry Skiba2539fff2015-06-16 17:56:09 -0700247void ShClearResults(const ShHandle handle)
248{
249 TCompiler *compiler = GetCompilerFromHandle(handle);
250 ASSERT(compiler);
251 compiler->clearResults();
252}
253
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700254int ShGetShaderVersion(const ShHandle handle)
alokp@chromium.org7beea402010-09-15 21:18:34 +0000255{
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700256 TCompiler* compiler = GetCompilerFromHandle(handle);
257 ASSERT(compiler);
258 return compiler->getShaderVersion();
259}
alokp@chromium.org7beea402010-09-15 21:18:34 +0000260
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700261ShShaderOutput ShGetShaderOutputType(const ShHandle handle)
262{
263 TCompiler* compiler = GetCompilerFromHandle(handle);
264 ASSERT(compiler);
265 return compiler->getOutputType();
alokp@chromium.org7beea402010-09-15 21:18:34 +0000266}
267
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000268//
alokp@chromium.org774d7062010-07-21 18:55:45 +0000269// Return any compiler log of messages for the application.
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000270//
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700271const std::string &ShGetInfoLog(const ShHandle handle)
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000272{
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700273 TCompiler *compiler = GetCompilerFromHandle(handle);
274 ASSERT(compiler);
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000275
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700276 TInfoSink &infoSink = compiler->getInfoSink();
277 return infoSink.info.str();
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000278}
279
280//
alokp@chromium.org774d7062010-07-21 18:55:45 +0000281// Return any object code.
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000282//
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700283const std::string &ShGetObjectCode(const ShHandle handle)
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000284{
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700285 TCompiler *compiler = GetCompilerFromHandle(handle);
286 ASSERT(compiler);
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000287
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700288 TInfoSink &infoSink = compiler->getInfoSink();
289 return infoSink.obj.str();
alokp@chromium.org7beea402010-09-15 21:18:34 +0000290}
daniel@transgaming.com4f39fd92010-03-08 20:26:45 +0000291
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700292const std::map<std::string, std::string> *ShGetNameHashingMap(
293 const ShHandle handle)
daniel@transgaming.comc23f4612012-11-28 19:42:57 +0000294{
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700295 TCompiler *compiler = GetCompilerFromHandle(handle);
296 ASSERT(compiler);
297 return &(compiler->getNameMap());
daniel@transgaming.comc23f4612012-11-28 19:42:57 +0000298}
daniel@transgaming.com043da132012-12-20 21:12:22 +0000299
Jamie Madille294bb82014-07-17 14:16:26 -0400300const std::vector<sh::Uniform> *ShGetUniforms(const ShHandle handle)
daniel@transgaming.com043da132012-12-20 21:12:22 +0000301{
Jamie Madille294bb82014-07-17 14:16:26 -0400302 return GetShaderVariables<sh::Uniform>(handle, SHADERVAR_UNIFORM);
303}
daniel@transgaming.com043da132012-12-20 21:12:22 +0000304
Jamie Madille294bb82014-07-17 14:16:26 -0400305const std::vector<sh::Varying> *ShGetVaryings(const ShHandle handle)
306{
307 return GetShaderVariables<sh::Varying>(handle, SHADERVAR_VARYING);
308}
daniel@transgaming.com043da132012-12-20 21:12:22 +0000309
Jamie Madille294bb82014-07-17 14:16:26 -0400310const std::vector<sh::Attribute> *ShGetAttributes(const ShHandle handle)
311{
312 return GetShaderVariables<sh::Attribute>(handle, SHADERVAR_ATTRIBUTE);
313}
314
315const std::vector<sh::Attribute> *ShGetOutputVariables(const ShHandle handle)
316{
317 return GetShaderVariables<sh::Attribute>(handle, SHADERVAR_OUTPUTVARIABLE);
318}
319
320const std::vector<sh::InterfaceBlock> *ShGetInterfaceBlocks(const ShHandle handle)
321{
322 return GetShaderVariables<sh::InterfaceBlock>(handle, SHADERVAR_INTERFACEBLOCK);
shannon.woods@transgaming.com1d432bb2013-01-25 21:57:28 +0000323}
Zhenyao Moa15f3e82013-09-23 14:57:08 -0400324
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700325bool ShCheckVariablesWithinPackingLimits(
326 int maxVectors, ShVariableInfo *varInfoArray, size_t varInfoArraySize)
Zhenyao Moa15f3e82013-09-23 14:57:08 -0400327{
328 if (varInfoArraySize == 0)
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700329 return true;
Zhenyao Moa15f3e82013-09-23 14:57:08 -0400330 ASSERT(varInfoArray);
Jamie Madilla718c1e2014-07-02 15:31:22 -0400331 std::vector<sh::ShaderVariable> variables;
Zhenyao Moa15f3e82013-09-23 14:57:08 -0400332 for (size_t ii = 0; ii < varInfoArraySize; ++ii)
333 {
Jamie Madilla3fe2b42014-07-18 10:33:13 -0400334 sh::ShaderVariable var(varInfoArray[ii].type, varInfoArray[ii].size);
Zhenyao Moa15f3e82013-09-23 14:57:08 -0400335 variables.push_back(var);
336 }
337 VariablePacker packer;
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700338 return packer.CheckVariablesWithinPackingLimits(maxVectors, variables);
Zhenyao Moa15f3e82013-09-23 14:57:08 -0400339}
Jamie Madill4e1fd412014-07-10 17:50:10 -0400340
341bool ShGetInterfaceBlockRegister(const ShHandle handle,
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700342 const std::string &interfaceBlockName,
Jamie Madill4e1fd412014-07-10 17:50:10 -0400343 unsigned int *indexOut)
344{
Daniel Bratell73941de2015-02-25 14:34:49 +0100345#ifdef ANGLE_ENABLE_HLSL
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700346 ASSERT(indexOut);
Jamie Madill4e1fd412014-07-10 17:50:10 -0400347
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700348 TranslatorHLSL *translator = GetTranslatorHLSLFromHandle(handle);
349 ASSERT(translator);
Jamie Madill4e1fd412014-07-10 17:50:10 -0400350
351 if (!translator->hasInterfaceBlock(interfaceBlockName))
352 {
353 return false;
354 }
355
356 *indexOut = translator->getInterfaceBlockRegister(interfaceBlockName);
357 return true;
Daniel Bratell73941de2015-02-25 14:34:49 +0100358#else
359 return false;
360#endif // ANGLE_ENABLE_HLSL
Jamie Madill4e1fd412014-07-10 17:50:10 -0400361}
Jamie Madill9fe25e92014-07-18 10:33:08 -0400362
363bool ShGetUniformRegister(const ShHandle handle,
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700364 const std::string &uniformName,
Jamie Madill9fe25e92014-07-18 10:33:08 -0400365 unsigned int *indexOut)
366{
Daniel Bratell73941de2015-02-25 14:34:49 +0100367#ifdef ANGLE_ENABLE_HLSL
Zhenyao Mo4de44cb2014-10-29 18:03:46 -0700368 ASSERT(indexOut);
369 TranslatorHLSL *translator = GetTranslatorHLSLFromHandle(handle);
370 ASSERT(translator);
Jamie Madill9fe25e92014-07-18 10:33:08 -0400371
372 if (!translator->hasUniform(uniformName))
373 {
374 return false;
375 }
376
377 *indexOut = translator->getUniformRegister(uniformName);
378 return true;
Daniel Bratell73941de2015-02-25 14:34:49 +0100379#else
380 return false;
381#endif // ANGLE_ENABLE_HLSL
Jamie Madill9fe25e92014-07-18 10:33:08 -0400382}