2014-12-04 07:16:20 +03:00
|
|
|
/*
|
2018-01-01 22:16:06 +03:00
|
|
|
* Copyright 2011-2018 Branimir Karadzic. All rights reserved.
|
2016-01-01 11:11:04 +03:00
|
|
|
* License: https://github.com/bkaradzic/bgfx#license-bsd-2-clause
|
2014-12-04 07:16:20 +03:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "shaderc.h"
|
|
|
|
#include "glsl_optimizer.h"
|
|
|
|
|
2016-10-01 04:16:04 +03:00
|
|
|
namespace bgfx { namespace glsl
|
2016-01-31 03:15:25 +03:00
|
|
|
{
|
2017-11-24 21:01:13 +03:00
|
|
|
static bool compile(const Options& _options, uint32_t _version, const std::string& _code, bx::WriterI* _writer)
|
2014-12-04 07:16:20 +03:00
|
|
|
{
|
2017-11-24 21:01:13 +03:00
|
|
|
char ch = _options.shaderType;
|
2016-01-31 08:13:41 +03:00
|
|
|
const glslopt_shader_type type = ch == 'f'
|
|
|
|
? kGlslOptShaderFragment
|
|
|
|
: (ch == 'c' ? kGlslOptShaderCompute : kGlslOptShaderVertex);
|
2014-12-04 07:16:20 +03:00
|
|
|
|
2016-01-31 08:13:41 +03:00
|
|
|
glslopt_target target = kGlslTargetOpenGL;
|
2016-10-01 04:16:04 +03:00
|
|
|
switch (_version)
|
2014-12-04 07:16:20 +03:00
|
|
|
{
|
2016-01-31 08:13:41 +03:00
|
|
|
case BX_MAKEFOURCC('M', 'T', 'L', 0):
|
|
|
|
target = kGlslTargetMetal;
|
|
|
|
break;
|
2014-12-04 07:16:20 +03:00
|
|
|
|
2016-01-31 08:13:41 +03:00
|
|
|
case 2:
|
|
|
|
target = kGlslTargetOpenGLES20;
|
|
|
|
break;
|
2014-12-04 07:16:20 +03:00
|
|
|
|
2016-01-31 08:13:41 +03:00
|
|
|
case 3:
|
|
|
|
target = kGlslTargetOpenGLES30;
|
|
|
|
break;
|
2014-12-04 07:16:20 +03:00
|
|
|
|
2016-01-31 08:13:41 +03:00
|
|
|
default:
|
|
|
|
target = kGlslTargetOpenGL;
|
|
|
|
break;
|
|
|
|
}
|
2014-12-04 07:16:20 +03:00
|
|
|
|
2016-01-31 08:13:41 +03:00
|
|
|
glslopt_ctx* ctx = glslopt_initialize(target);
|
2014-12-04 07:16:20 +03:00
|
|
|
|
2016-01-31 08:13:41 +03:00
|
|
|
glslopt_shader* shader = glslopt_optimize(ctx, type, _code.c_str(), 0);
|
2014-12-04 07:16:20 +03:00
|
|
|
|
2016-01-31 08:13:41 +03:00
|
|
|
if (!glslopt_get_status(shader) )
|
2014-12-04 07:16:20 +03:00
|
|
|
{
|
2016-01-31 08:13:41 +03:00
|
|
|
const char* log = glslopt_get_log(shader);
|
2016-10-01 22:38:41 +03:00
|
|
|
int32_t source = 0;
|
|
|
|
int32_t line = 0;
|
|
|
|
int32_t column = 0;
|
|
|
|
int32_t start = 0;
|
|
|
|
int32_t end = INT32_MAX;
|
2016-01-31 08:13:41 +03:00
|
|
|
|
2016-10-01 22:38:41 +03:00
|
|
|
bool found = false
|
|
|
|
|| 3 == sscanf(log, "%u:%u(%u):", &source, &line, &column)
|
2017-04-04 03:53:14 +03:00
|
|
|
|| 2 == sscanf(log, "(%u,%u):", &line, &column)
|
2016-10-01 22:38:41 +03:00
|
|
|
;
|
|
|
|
|
|
|
|
if (found
|
2016-01-31 08:13:41 +03:00
|
|
|
&& 0 != line)
|
2014-12-04 07:16:20 +03:00
|
|
|
{
|
2016-01-31 08:13:41 +03:00
|
|
|
start = bx::uint32_imax(1, line-10);
|
2016-10-01 22:38:41 +03:00
|
|
|
end = start + 20;
|
2016-01-31 08:13:41 +03:00
|
|
|
}
|
2014-12-04 07:16:20 +03:00
|
|
|
|
2016-10-01 22:38:41 +03:00
|
|
|
printCode(_code.c_str(), line, start, end, column);
|
2016-01-31 08:13:41 +03:00
|
|
|
fprintf(stderr, "Error: %s\n", log);
|
2018-01-13 21:08:03 +03:00
|
|
|
glslopt_shader_delete(shader);
|
2016-01-31 08:13:41 +03:00
|
|
|
glslopt_cleanup(ctx);
|
|
|
|
return false;
|
|
|
|
}
|
2014-12-04 07:16:20 +03:00
|
|
|
|
2016-01-31 08:13:41 +03:00
|
|
|
const char* optimizedShader = glslopt_get_output(shader);
|
2014-12-04 07:16:20 +03:00
|
|
|
|
2016-01-31 08:13:41 +03:00
|
|
|
// Trim all directives.
|
|
|
|
while ('#' == *optimizedShader)
|
|
|
|
{
|
2018-10-21 09:33:31 +03:00
|
|
|
optimizedShader = bx::strFindNl(optimizedShader).getPtr();
|
2016-01-31 08:13:41 +03:00
|
|
|
}
|
2014-12-04 07:16:20 +03:00
|
|
|
|
2016-01-31 08:13:41 +03:00
|
|
|
{
|
|
|
|
char* code = const_cast<char*>(optimizedShader);
|
|
|
|
strReplace(code, "gl_FragDepthEXT", "gl_FragDepth");
|
|
|
|
|
2017-03-24 03:33:29 +03:00
|
|
|
strReplace(code, "texture2DLodARB", "texture2DLod");
|
2016-01-31 08:13:41 +03:00
|
|
|
strReplace(code, "texture2DLodEXT", "texture2DLod");
|
2017-03-24 03:33:29 +03:00
|
|
|
strReplace(code, "texture2DGradARB", "texture2DGrad");
|
2016-01-31 08:13:41 +03:00
|
|
|
strReplace(code, "texture2DGradEXT", "texture2DGrad");
|
2017-03-24 03:33:29 +03:00
|
|
|
|
|
|
|
strReplace(code, "textureCubeLodARB", "textureCubeLod");
|
|
|
|
strReplace(code, "textureCubeLodEXT", "textureCubeLod");
|
|
|
|
strReplace(code, "textureCubeGradARB", "textureCubeGrad");
|
2016-01-31 08:13:41 +03:00
|
|
|
strReplace(code, "textureCubeGradEXT", "textureCubeGrad");
|
|
|
|
|
2017-03-24 03:33:29 +03:00
|
|
|
strReplace(code, "texture2DProjLodARB", "texture2DProjLod");
|
|
|
|
strReplace(code, "texture2DProjLodEXT", "texture2DProjLod");
|
|
|
|
strReplace(code, "texture2DProjGradARB", "texture2DProjGrad");
|
|
|
|
strReplace(code, "texture2DProjGradEXT", "texture2DProjGrad");
|
|
|
|
|
|
|
|
strReplace(code, "shadow2DARB", "shadow2D");
|
2016-01-31 08:13:41 +03:00
|
|
|
strReplace(code, "shadow2DEXT", "shadow2D");
|
2017-03-24 03:33:29 +03:00
|
|
|
strReplace(code, "shadow2DProjARB", "shadow2DProj");
|
2016-01-31 08:13:41 +03:00
|
|
|
strReplace(code, "shadow2DProjEXT", "shadow2DProj");
|
|
|
|
}
|
2014-12-04 07:16:20 +03:00
|
|
|
|
2016-01-31 08:13:41 +03:00
|
|
|
UniformArray uniforms;
|
2014-12-04 07:16:20 +03:00
|
|
|
|
2016-11-01 06:33:14 +03:00
|
|
|
if (target != kGlslTargetMetal)
|
2016-01-31 08:13:41 +03:00
|
|
|
{
|
2018-10-26 08:16:10 +03:00
|
|
|
bx::StringView parse(optimizedShader);
|
2014-12-04 07:16:20 +03:00
|
|
|
|
2018-10-26 08:16:10 +03:00
|
|
|
while (!parse.isEmpty() )
|
2016-01-31 08:13:41 +03:00
|
|
|
{
|
2018-10-26 08:16:10 +03:00
|
|
|
parse = bx::strLTrimSpace(parse);
|
|
|
|
bx::StringView eol = bx::strFind(parse, ';');
|
2018-10-21 09:33:31 +03:00
|
|
|
if (!eol.isEmpty() )
|
2014-12-04 07:16:20 +03:00
|
|
|
{
|
2018-10-26 08:16:10 +03:00
|
|
|
bx::StringView qualifier = nextWord(parse);
|
2016-01-31 08:13:41 +03:00
|
|
|
|
2017-04-23 00:47:02 +03:00
|
|
|
if (0 == bx::strCmp(qualifier, "attribute", 9)
|
|
|
|
|| 0 == bx::strCmp(qualifier, "varying", 7)
|
|
|
|
|| 0 == bx::strCmp(qualifier, "in", 2)
|
|
|
|
|| 0 == bx::strCmp(qualifier, "out", 3)
|
2017-04-04 03:53:14 +03:00
|
|
|
)
|
2016-01-31 08:13:41 +03:00
|
|
|
{
|
|
|
|
// skip attributes and varyings.
|
2018-10-26 08:16:10 +03:00
|
|
|
parse.set(eol.getPtr() + 1, parse.getTerm() );
|
2016-01-31 08:13:41 +03:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-09-18 02:54:37 +03:00
|
|
|
if (0 == bx::strCmp(parse, "tmpvar", 6) )
|
|
|
|
{
|
|
|
|
// skip temporaries
|
2018-10-26 08:16:10 +03:00
|
|
|
parse.set(eol.getPtr() + 1, parse.getTerm() );
|
2017-09-18 02:54:37 +03:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-04-23 00:47:02 +03:00
|
|
|
if (0 != bx::strCmp(qualifier, "uniform", 7) )
|
2016-01-31 08:13:41 +03:00
|
|
|
{
|
|
|
|
// end if there is no uniform keyword.
|
2018-10-26 08:16:10 +03:00
|
|
|
parse.clear();
|
2016-01-31 08:13:41 +03:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-10-26 08:16:10 +03:00
|
|
|
bx::StringView precision;
|
|
|
|
bx::StringView typen = nextWord(parse);
|
2016-01-31 08:13:41 +03:00
|
|
|
|
2017-04-23 00:47:02 +03:00
|
|
|
if (0 == bx::strCmp(typen, "lowp", 4)
|
|
|
|
|| 0 == bx::strCmp(typen, "mediump", 7)
|
|
|
|
|| 0 == bx::strCmp(typen, "highp", 5) )
|
2016-01-31 08:13:41 +03:00
|
|
|
{
|
|
|
|
precision = typen;
|
2018-10-26 08:16:10 +03:00
|
|
|
typen = nextWord(parse);
|
2016-01-31 08:13:41 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
BX_UNUSED(precision);
|
|
|
|
|
|
|
|
char uniformType[256];
|
|
|
|
|
2017-04-23 00:47:02 +03:00
|
|
|
if (0 == bx::strCmp(typen, "sampler", 7) )
|
2016-01-31 08:13:41 +03:00
|
|
|
{
|
2017-04-17 07:56:17 +03:00
|
|
|
bx::strCopy(uniformType, BX_COUNTOF(uniformType), "int");
|
2016-01-31 08:13:41 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-10-26 08:16:10 +03:00
|
|
|
bx::strCopy(uniformType, BX_COUNTOF(uniformType), typen);
|
2016-01-31 08:13:41 +03:00
|
|
|
}
|
|
|
|
|
2018-10-26 08:16:10 +03:00
|
|
|
bx::StringView name = nextWord(parse);
|
2016-01-31 08:13:41 +03:00
|
|
|
|
|
|
|
uint8_t num = 1;
|
2018-10-26 08:16:10 +03:00
|
|
|
bx::StringView array = bx::strSubstr(parse, 0, 1);
|
|
|
|
if (0 == bx::strCmp(array, "[", 1) )
|
2016-11-01 06:33:14 +03:00
|
|
|
{
|
2018-10-26 08:16:10 +03:00
|
|
|
parse = bx::strLTrimSpace(bx::StringView(parse.getPtr() + 1, parse.getTerm() ) );
|
2016-11-01 06:33:14 +03:00
|
|
|
|
2018-10-22 06:33:29 +03:00
|
|
|
uint32_t tmp;
|
2018-10-26 08:16:10 +03:00
|
|
|
bx::fromString(&tmp, parse);
|
2018-10-22 06:33:29 +03:00
|
|
|
num = uint8_t(tmp);
|
2016-11-01 06:33:14 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
Uniform un;
|
|
|
|
un.type = nameToUniformTypeEnum(uniformType);
|
|
|
|
|
|
|
|
if (UniformType::Count != un.type)
|
|
|
|
{
|
2018-10-26 08:16:10 +03:00
|
|
|
un.name.assign(name.getPtr(), name.getTerm());
|
|
|
|
|
|
|
|
BX_TRACE("name: %s (type %d, num %d)", un.name.c_str(), un.type, num);
|
2016-11-01 06:33:14 +03:00
|
|
|
|
|
|
|
un.num = num;
|
|
|
|
un.regIndex = 0;
|
|
|
|
un.regCount = num;
|
|
|
|
uniforms.push_back(un);
|
|
|
|
}
|
|
|
|
|
2018-10-26 08:16:10 +03:00
|
|
|
parse = bx::strLTrimSpace(bx::strFindNl(bx::StringView(eol.getPtr(), parse.getTerm() ) ) );
|
2016-11-01 06:33:14 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-10-21 09:33:31 +03:00
|
|
|
const bx::StringView optShader(optimizedShader);
|
|
|
|
bx::StringView parse = bx::strFind(optimizedShader, "struct xlatMtlShaderUniform {");
|
|
|
|
bx::StringView end = parse;
|
|
|
|
if (!parse.isEmpty() )
|
2016-11-01 06:33:14 +03:00
|
|
|
{
|
2018-10-21 09:33:31 +03:00
|
|
|
parse.set(parse.getPtr() + bx::strLen("struct xlatMtlShaderUniform {"), optShader.getTerm() );
|
|
|
|
end = bx::strFind(parse, "};");
|
2016-11-01 06:33:14 +03:00
|
|
|
}
|
|
|
|
|
2018-10-21 09:33:31 +03:00
|
|
|
while ( parse.getPtr() < end.getPtr()
|
|
|
|
&& !parse.isEmpty() )
|
2016-11-01 06:33:14 +03:00
|
|
|
{
|
2018-10-21 09:33:31 +03:00
|
|
|
parse.set(bx::strLTrimSpace(parse).getPtr(), optShader.getTerm() );
|
|
|
|
const bx::StringView eol = bx::strFind(parse, ';');
|
|
|
|
if (!eol.isEmpty() )
|
2016-11-01 06:33:14 +03:00
|
|
|
{
|
2018-10-21 09:33:31 +03:00
|
|
|
const char* typen = parse.getPtr();
|
2016-11-01 06:33:14 +03:00
|
|
|
|
|
|
|
char uniformType[256];
|
2018-10-26 08:16:10 +03:00
|
|
|
parse = bx::strWord(parse).getPtr();
|
2018-10-21 09:33:31 +03:00
|
|
|
bx::strCopy(uniformType, int32_t(parse.getPtr()-typen+1), typen);
|
|
|
|
const char* name = bx::strLTrimSpace(parse).getPtr();
|
|
|
|
parse.set(name, optShader.getTerm() );
|
2016-11-01 06:33:14 +03:00
|
|
|
|
|
|
|
char uniformName[256];
|
|
|
|
uint8_t num = 1;
|
2018-10-21 09:33:31 +03:00
|
|
|
bx::StringView array = bx::strFind(bx::StringView(name, int32_t(eol.getPtr()-parse.getPtr() ) ), "[");
|
|
|
|
if (!array.isEmpty() )
|
2016-01-31 08:13:41 +03:00
|
|
|
{
|
2018-10-21 09:33:31 +03:00
|
|
|
bx::strCopy(uniformName, int32_t(array.getPtr()-name+1), name);
|
2016-01-31 08:13:41 +03:00
|
|
|
|
|
|
|
char arraySize[32];
|
2018-10-21 09:33:31 +03:00
|
|
|
bx::StringView arrayEnd = bx::strFind(bx::StringView(array.getPtr(), int32_t(eol.getPtr()-array.getPtr() ) ), "]");
|
|
|
|
bx::strCopy(arraySize, int32_t(arrayEnd.getPtr()-array.getPtr() ), array.getPtr()+1);
|
2018-10-22 06:33:29 +03:00
|
|
|
uint32_t tmp;
|
|
|
|
bx::fromString(&tmp, arraySize);
|
|
|
|
num = uint8_t(tmp);
|
2016-01-31 08:13:41 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-10-21 09:33:31 +03:00
|
|
|
bx::strCopy(uniformName, int32_t(eol.getPtr()-name+1), name);
|
2016-01-31 08:13:41 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
Uniform un;
|
|
|
|
un.type = nameToUniformTypeEnum(uniformType);
|
|
|
|
|
|
|
|
if (UniformType::Count != un.type)
|
|
|
|
{
|
|
|
|
BX_TRACE("name: %s (type %d, num %d)", uniformName, un.type, num);
|
|
|
|
|
|
|
|
un.name = uniformName;
|
|
|
|
un.num = num;
|
|
|
|
un.regIndex = 0;
|
|
|
|
un.regCount = num;
|
|
|
|
uniforms.push_back(un);
|
|
|
|
}
|
2014-12-04 07:16:20 +03:00
|
|
|
|
2018-10-21 09:33:31 +03:00
|
|
|
parse = eol.getPtr() + 1;
|
2014-12-04 07:16:20 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-01-31 08:13:41 +03:00
|
|
|
uint16_t count = (uint16_t)uniforms.size();
|
|
|
|
bx::write(_writer, count);
|
2014-12-04 07:16:20 +03:00
|
|
|
|
2016-01-31 08:13:41 +03:00
|
|
|
for (UniformArray::const_iterator it = uniforms.begin(); it != uniforms.end(); ++it)
|
|
|
|
{
|
|
|
|
const Uniform& un = *it;
|
|
|
|
uint8_t nameSize = (uint8_t)un.name.size();
|
|
|
|
bx::write(_writer, nameSize);
|
|
|
|
bx::write(_writer, un.name.c_str(), nameSize);
|
2017-03-12 01:44:00 +03:00
|
|
|
uint8_t uniformType = uint8_t(un.type);
|
2016-01-31 08:13:41 +03:00
|
|
|
bx::write(_writer, uniformType);
|
|
|
|
bx::write(_writer, un.num);
|
|
|
|
bx::write(_writer, un.regIndex);
|
|
|
|
bx::write(_writer, un.regCount);
|
|
|
|
|
|
|
|
BX_TRACE("%s, %s, %d, %d, %d"
|
|
|
|
, un.name.c_str()
|
|
|
|
, getUniformTypeName(un.type)
|
|
|
|
, un.num
|
|
|
|
, un.regIndex
|
|
|
|
, un.regCount
|
|
|
|
);
|
|
|
|
}
|
2014-12-04 07:16:20 +03:00
|
|
|
|
2017-04-17 07:56:17 +03:00
|
|
|
uint32_t shaderSize = (uint32_t)bx::strLen(optimizedShader);
|
2016-01-31 08:13:41 +03:00
|
|
|
bx::write(_writer, shaderSize);
|
|
|
|
bx::write(_writer, optimizedShader, shaderSize);
|
|
|
|
uint8_t nul = 0;
|
|
|
|
bx::write(_writer, nul);
|
2014-12-04 07:16:20 +03:00
|
|
|
|
2017-11-24 21:01:13 +03:00
|
|
|
if (_options.disasm )
|
2016-04-30 21:44:38 +03:00
|
|
|
{
|
2017-11-24 21:01:13 +03:00
|
|
|
std::string disasmfp = _options.outputFilePath + ".disasm";
|
2016-04-30 21:44:38 +03:00
|
|
|
writeFile(disasmfp.c_str(), optimizedShader, shaderSize);
|
|
|
|
}
|
|
|
|
|
2018-01-13 21:08:03 +03:00
|
|
|
glslopt_shader_delete(shader);
|
2016-01-31 08:13:41 +03:00
|
|
|
glslopt_cleanup(ctx);
|
2014-12-04 07:16:20 +03:00
|
|
|
|
2016-01-31 08:13:41 +03:00
|
|
|
return true;
|
|
|
|
}
|
2016-01-31 03:15:25 +03:00
|
|
|
|
2016-10-01 04:16:04 +03:00
|
|
|
} // namespace glsl
|
|
|
|
|
2017-11-24 21:01:13 +03:00
|
|
|
bool compileGLSLShader(const Options& _options, uint32_t _version, const std::string& _code, bx::WriterI* _writer)
|
2016-10-01 04:16:04 +03:00
|
|
|
{
|
2017-11-24 21:01:13 +03:00
|
|
|
return glsl::compile(_options, _version, _code, _writer);
|
2016-10-01 04:16:04 +03:00
|
|
|
}
|
|
|
|
|
2016-01-31 03:15:25 +03:00
|
|
|
} // namespace bgfx
|