doom3-bfg/neo/renderer/RenderProgs_GLSL.cpp
2021-04-29 16:32:15 +02:00

1731 lines
45 KiB
C++

/*
===========================================================================
Doom 3 BFG Edition GPL Source Code
Copyright (C) 1993-2012 id Software LLC, a ZeniMax Media company.
Copyright (C) 2013-2018 Robert Beckebans
Copyright (C) 2016-2017 Dustin Land
This file is part of the Doom 3 BFG Edition GPL Source Code ("Doom 3 BFG Edition Source Code").
Doom 3 BFG Edition Source Code is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
Doom 3 BFG Edition Source Code is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with Doom 3 BFG Edition Source Code. If not, see <http://www.gnu.org/licenses/>.
In addition, the Doom 3 BFG Edition Source Code is also subject to certain additional terms. You should have received a copy of these additional terms immediately following the terms and conditions of the GNU General Public License which accompanied the Doom 3 BFG Edition Source Code. If not, please request a copy in writing from id Software at the address below.
If you have questions concerning this license or the applicable additional terms, you may contact in writing id Software LLC, c/o ZeniMax Media Inc., Suite 120, Rockville, Maryland 20850 USA.
===========================================================================
*/
#pragma hdrstop
#include "precompiled.h"
#include "RenderCommon.h"
#include "RenderProgs_embedded.h"
idCVar r_skipStripDeadCode( "r_skipStripDeadCode", "0", CVAR_BOOL, "Skip stripping dead code" );
struct idCGBlock
{
idStr prefix; // tokens that comes before the name
idStr name; // the name
idStr postfix; // tokens that comes after the name
bool used; // whether or not this block is referenced anywhere
};
/*
================================================
attribInfo_t
================================================
*/
attribInfo_t attribsPC[] =
{
// vertex attributes
{ "float4", "position", "POSITION", "in_Position", PC_ATTRIB_INDEX_VERTEX, AT_VS_IN, VERTEX_MASK_XYZ },
{ "float2", "texcoord", "TEXCOORD0", "in_TexCoord", PC_ATTRIB_INDEX_ST, AT_VS_IN, VERTEX_MASK_ST },
{ "float4", "normal", "NORMAL", "in_Normal", PC_ATTRIB_INDEX_NORMAL, AT_VS_IN, VERTEX_MASK_NORMAL },
{ "float4", "tangent", "TANGENT", "in_Tangent", PC_ATTRIB_INDEX_TANGENT, AT_VS_IN, VERTEX_MASK_TANGENT },
{ "float4", "color", "COLOR0", "in_Color", PC_ATTRIB_INDEX_COLOR, AT_VS_IN, VERTEX_MASK_COLOR },
{ "float4", "color2", "COLOR1", "in_Color2", PC_ATTRIB_INDEX_COLOR2, AT_VS_IN, VERTEX_MASK_COLOR2 },
// pre-defined vertex program output
{ "float4", "position", "POSITION", "gl_Position", 0, AT_VS_OUT | AT_VS_OUT_RESERVED, 0 },
{ "float", "clip0", "CLP0", "gl_ClipDistance[0]", 0, AT_VS_OUT, 0 },
{ "float", "clip1", "CLP1", "gl_ClipDistance[1]", 0, AT_VS_OUT, 0 },
{ "float", "clip2", "CLP2", "gl_ClipDistance[2]", 0, AT_VS_OUT, 0 },
{ "float", "clip3", "CLP3", "gl_ClipDistance[3]", 0, AT_VS_OUT, 0 },
{ "float", "clip4", "CLP4", "gl_ClipDistance[4]", 0, AT_VS_OUT, 0 },
{ "float", "clip5", "CLP5", "gl_ClipDistance[5]", 0, AT_VS_OUT, 0 },
// pre-defined fragment program input
{ "float4", "position", "WPOS", "gl_FragCoord", 0, AT_PS_IN | AT_PS_IN_RESERVED, 0 },
{ "half4", "hposition", "WPOS", "gl_FragCoord", 0, AT_PS_IN | AT_PS_IN_RESERVED, 0 },
{ "float", "facing", "FACE", "gl_FrontFacing", 0, AT_PS_IN | AT_PS_IN_RESERVED, 0 },
// fragment program output
{ "float4", "color", "COLOR", "fo_FragColor", 0, AT_PS_OUT /*| AT_PS_OUT_RESERVED*/, 0 },
{ "half4", "hcolor", "COLOR", "fo_FragColor", 0, AT_PS_OUT /*| AT_PS_OUT_RESERVED*/, 0 },
{ "float4", "color0", "COLOR0", "fo_FragColor", 0, AT_PS_OUT /*| AT_PS_OUT_RESERVED*/, 0 },
{ "float4", "color1", "COLOR1", "fo_FragColor", 1, AT_PS_OUT /*| AT_PS_OUT_RESERVED*/, 0 },
{ "float4", "color2", "COLOR2", "fo_FragColor", 2, AT_PS_OUT /*| AT_PS_OUT_RESERVED*/, 0 },
{ "float4", "color3", "COLOR3", "fo_FragColor", 3, AT_PS_OUT /*| AT_PS_OUT_RESERVED*/, 0 },
{ "float", "depth", "DEPTH", "gl_FragDepth", 4, AT_PS_OUT | AT_PS_OUT_RESERVED, 0 },
// vertex to fragment program pass through
{ "float4", "color", "COLOR", "vofi_Color", 0, AT_VS_OUT, 0 },
{ "float4", "color0", "COLOR0", "vofi_Color", 0, AT_VS_OUT, 0 },
{ "float4", "color1", "COLOR1", "vofi_SecondaryColor", 0, AT_VS_OUT, 0 },
{ "float4", "color", "COLOR", "vofi_Color", 0, AT_PS_IN, 0 },
{ "float4", "color0", "COLOR0", "vofi_Color", 0, AT_PS_IN, 0 },
{ "float4", "color1", "COLOR1", "vofi_SecondaryColor", 0, AT_PS_IN, 0 },
{ "half4", "hcolor", "COLOR", "vofi_Color", 0, AT_PS_IN, 0 },
{ "half4", "hcolor0", "COLOR0", "vofi_Color", 0, AT_PS_IN, 0 },
{ "half4", "hcolor1", "COLOR1", "vofi_SecondaryColor", 0, AT_PS_IN, 0 },
{ "float4", "texcoord0", "TEXCOORD0_centroid", "vofi_TexCoord0", 0, AT_PS_IN, 0 },
{ "float4", "texcoord1", "TEXCOORD1_centroid", "vofi_TexCoord1", 0, AT_PS_IN, 0 },
{ "float4", "texcoord2", "TEXCOORD2_centroid", "vofi_TexCoord2", 0, AT_PS_IN, 0 },
{ "float4", "texcoord3", "TEXCOORD3_centroid", "vofi_TexCoord3", 0, AT_PS_IN, 0 },
{ "float4", "texcoord4", "TEXCOORD4_centroid", "vofi_TexCoord4", 0, AT_PS_IN, 0 },
{ "float4", "texcoord5", "TEXCOORD5_centroid", "vofi_TexCoord5", 0, AT_PS_IN, 0 },
{ "float4", "texcoord6", "TEXCOORD6_centroid", "vofi_TexCoord6", 0, AT_PS_IN, 0 },
{ "float4", "texcoord7", "TEXCOORD7_centroid", "vofi_TexCoord7", 0, AT_PS_IN, 0 },
{ "float4", "texcoord8", "TEXCOORD8_centroid", "vofi_TexCoord8", 0, AT_PS_IN, 0 },
{ "float4", "texcoord9", "TEXCOORD9_centroid", "vofi_TexCoord9", 0, AT_PS_IN, 0 },
{ "float4", "texcoord0", "TEXCOORD0", "vofi_TexCoord0", 0, AT_VS_OUT | AT_PS_IN, 0 },
{ "float4", "texcoord1", "TEXCOORD1", "vofi_TexCoord1", 0, AT_VS_OUT | AT_PS_IN, 0 },
{ "float4", "texcoord2", "TEXCOORD2", "vofi_TexCoord2", 0, AT_VS_OUT | AT_PS_IN, 0 },
{ "float4", "texcoord3", "TEXCOORD3", "vofi_TexCoord3", 0, AT_VS_OUT | AT_PS_IN, 0 },
{ "float4", "texcoord4", "TEXCOORD4", "vofi_TexCoord4", 0, AT_VS_OUT | AT_PS_IN, 0 },
{ "float4", "texcoord5", "TEXCOORD5", "vofi_TexCoord5", 0, AT_VS_OUT | AT_PS_IN, 0 },
{ "float4", "texcoord6", "TEXCOORD6", "vofi_TexCoord6", 0, AT_VS_OUT | AT_PS_IN, 0 },
{ "float4", "texcoord7", "TEXCOORD7", "vofi_TexCoord7", 0, AT_VS_OUT | AT_PS_IN, 0 },
{ "float4", "texcoord8", "TEXCOORD8", "vofi_TexCoord8", 0, AT_VS_OUT | AT_PS_IN, 0 },
{ "float4", "texcoord9", "TEXCOORD9", "vofi_TexCoord9", 0, AT_VS_OUT | AT_PS_IN, 0 },
{ "half4", "htexcoord0", "TEXCOORD0", "vofi_TexCoord0", 0, AT_PS_IN, 0 },
{ "half4", "htexcoord1", "TEXCOORD1", "vofi_TexCoord1", 0, AT_PS_IN, 0 },
{ "half4", "htexcoord2", "TEXCOORD2", "vofi_TexCoord2", 0, AT_PS_IN, 0 },
{ "half4", "htexcoord3", "TEXCOORD3", "vofi_TexCoord3", 0, AT_PS_IN, 0 },
{ "half4", "htexcoord4", "TEXCOORD4", "vofi_TexCoord4", 0, AT_PS_IN, 0 },
{ "half4", "htexcoord5", "TEXCOORD5", "vofi_TexCoord5", 0, AT_PS_IN, 0 },
{ "half4", "htexcoord6", "TEXCOORD6", "vofi_TexCoord6", 0, AT_PS_IN, 0 },
{ "half4", "htexcoord7", "TEXCOORD7", "vofi_TexCoord7", 0, AT_PS_IN, 0 },
{ "half4", "htexcoord8", "TEXCOORD8", "vofi_TexCoord8", 0, AT_PS_IN, 0 },
{ "half4", "htexcoord9", "TEXCOORD9", "vofi_TexCoord9", 0, AT_PS_IN, 0 },
{ "float", "fog", "FOG", "gl_FogFragCoord", 0, AT_VS_OUT, 0 },
{ "float4", "fog", "FOG", "gl_FogFragCoord", 0, AT_PS_IN, 0 },
{ NULL, NULL, NULL, NULL, 0, 0, 0 }
};
const char* types[] =
{
"int",
"float",
"half",
"fixed",
"bool",
"cint",
"cfloat",
"void"
};
static const int numTypes = sizeof( types ) / sizeof( types[0] );
const char* typePosts[] =
{
"1", "2", "3", "4",
"1x1", "1x2", "1x3", "1x4",
"2x1", "2x2", "2x3", "2x4",
"3x1", "3x2", "3x3", "3x4",
"4x1", "4x2", "4x3", "4x4"
};
static const int numTypePosts = sizeof( typePosts ) / sizeof( typePosts[0] );
const char* prefixes[] =
{
"static",
"const",
"uniform",
"struct",
"sampler",
"sampler1D",
"sampler2D",
"sampler3D",
"samplerCUBE",
"sampler1DShadow", // GLSL
"sampler2DShadow", // GLSL
"sampler2DArrayShadow", // GLSL
"sampler3DShadow", // GLSL
"samplerCubeShadow", // GLSL
"sampler2DArray", // GLSL"
"sampler2DMS", // GLSL
};
static const int numPrefixes = sizeof( prefixes ) / sizeof( prefixes[0] );
// For GLSL we need to have the names for the renderparms so we can look up their run time indices within the renderprograms
static const char* GLSLParmNames[RENDERPARM_TOTAL] =
{
"rpScreenCorrectionFactor",
"rpWindowCoord",
"rpDiffuseModifier",
"rpSpecularModifier",
"rpLocalLightOrigin",
"rpLocalViewOrigin",
"rpLightProjectionS",
"rpLightProjectionT",
"rpLightProjectionQ",
"rpLightFalloffS",
"rpBumpMatrixS",
"rpBumpMatrixT",
"rpDiffuseMatrixS",
"rpDiffuseMatrixT",
"rpSpecularMatrixS",
"rpSpecularMatrixT",
"rpVertexColorModulate",
"rpVertexColorAdd",
"rpColor",
"rpViewOrigin",
"rpGlobalEyePos",
"rpMVPmatrixX",
"rpMVPmatrixY",
"rpMVPmatrixZ",
"rpMVPmatrixW",
"rpModelMatrixX",
"rpModelMatrixY",
"rpModelMatrixZ",
"rpModelMatrixW",
"rpProjectionMatrixX",
"rpProjectionMatrixY",
"rpProjectionMatrixZ",
"rpProjectionMatrixW",
"rpModelViewMatrixX",
"rpModelViewMatrixY",
"rpModelViewMatrixZ",
"rpModelViewMatrixW",
"rpTextureMatrixS",
"rpTextureMatrixT",
"rpTexGen0S",
"rpTexGen0T",
"rpTexGen0Q",
"rpTexGen0Enabled",
"rpTexGen1S",
"rpTexGen1T",
"rpTexGen1Q",
"rpTexGen1Enabled",
"rpWobbleSkyX",
"rpWobbleSkyY",
"rpWobbleSkyZ",
"rpOverbright",
"rpEnableSkinning",
"rpAlphaTest",
// RB begin
"rpAmbientColor",
"rpGlobalLightOrigin",
"rpJitterTexScale",
"rpJitterTexOffset",
"rpCascadeDistances",
"rpShadowMatrices",
"rpShadowMatrix0Y",
"rpShadowMatrix0Z",
"rpShadowMatrix0W",
"rpShadowMatrix1X",
"rpShadowMatrix1Y",
"rpShadowMatrix1Z",
"rpShadowMatrix1W",
"rpShadowMatrix2X",
"rpShadowMatrix2Y",
"rpShadowMatrix2Z",
"rpShadowMatrix2W",
"rpShadowMatrix3X",
"rpShadowMatrix3Y",
"rpShadowMatrix3Z",
"rpShadowMatrix3W",
"rpShadowMatrix4X",
"rpShadowMatrix4Y",
"rpShadowMatrix4Z",
"rpShadowMatrix4W",
"rpShadowMatrix5X",
"rpShadowMatrix5Y",
"rpShadowMatrix5Z",
"rpShadowMatrix5W",
"rpUser0",
"rpUser1",
"rpUser2",
"rpUser3",
"rpUser4",
"rpUser5",
"rpUser6",
"rpUser7"
// RB end
};
// RB begin
const char* idRenderProgManager::GLSLMacroNames[MAX_SHADER_MACRO_NAMES] =
{
"USE_GPU_SKINNING",
"LIGHT_POINT",
"LIGHT_PARALLEL",
"BRIGHTPASS",
"HDR_DEBUG",
"USE_SRGB",
"USE_PBR"
};
// RB end
// RB: added embedded Cg shader resources
const char* idRenderProgManager::FindEmbeddedSourceShader( const char* name )
{
const char* embeddedSource = NULL;
for( int i = 0 ; cg_renderprogs[i].name ; i++ )
{
if( !idStr::Icmp( cg_renderprogs[i].name, name ) )
{
embeddedSource = cg_renderprogs[i].shaderText;
break;
}
}
return embeddedSource;
}
class idParser_EmbeddedGLSL : public idParser
{
public:
idParser_EmbeddedGLSL( int flags ) : idParser( flags )
{
}
private:
int Directive_include( idToken* token, bool supressWarning )
{
if( idParser::Directive_include( token, true ) )
{
// RB: try local shaders in base/renderprogs/ first
return true;
}
idLexer* script;
idStr path;
/*
token was already parsed
if( !idParser::ReadSourceToken( &token ) )
{
idParser::Error( "#include without file name" );
return false;
}
*/
if( token->linesCrossed > 0 )
{
idParser::Error( "#include without file name" );
return false;
}
if( token->type == TT_STRING )
{
script = new idLexer;
// try relative to the current file
path = scriptstack->GetFileName();
path.StripFilename();
path += "/";
path += *token;
//if( !script->LoadFile( path, OSPath ) )
const char* embeddedSource = idRenderProgManager::FindEmbeddedSourceShader( path );
if( embeddedSource == NULL )
{
// try absolute path
path = *token;
embeddedSource = idRenderProgManager::FindEmbeddedSourceShader( path );
if( embeddedSource == NULL )
{
// try from the include path
path = includepath + *token;
embeddedSource = idRenderProgManager::FindEmbeddedSourceShader( path );
}
}
if( embeddedSource == NULL || !script->LoadMemory( embeddedSource, strlen( embeddedSource ), path ) )
{
delete script;
script = NULL;
}
}
else if( token->type == TT_PUNCTUATION && *token == "<" )
{
path = idParser::includepath;
while( idParser::ReadSourceToken( token ) )
{
if( token->linesCrossed > 0 )
{
idParser::UnreadSourceToken( token );
break;
}
if( token->type == TT_PUNCTUATION && *token == ">" )
{
break;
}
path += *token;
}
if( *token != ">" )
{
idParser::Warning( "#include missing trailing >" );
}
if( !path.Length() )
{
idParser::Error( "#include without file name between < >" );
return false;
}
if( idParser::flags & LEXFL_NOBASEINCLUDES )
{
return true;
}
script = new idLexer;
const char* embeddedSource = idRenderProgManager::FindEmbeddedSourceShader( includepath + path );
if( embeddedSource == NULL || !script->LoadMemory( embeddedSource, strlen( embeddedSource ), path ) )
{
delete script;
script = NULL;
}
}
else
{
idParser::Error( "#include without file name" );
return false;
}
if( !script )
{
idParser::Error( "file '%s' not found", path.c_str() );
return false;
}
script->SetFlags( idParser::flags );
script->SetPunctuations( idParser::punctuations );
idParser::PushScript( script );
return true;
}
};
// RB end
/*
========================
StripDeadCode
========================
*/
idStr idRenderProgManager::StripDeadCode( const idStr& in, const char* name, const idStrList& compileMacros, bool builtin )
{
if( r_skipStripDeadCode.GetBool() )
{
return in;
}
//idLexer src( LEXFL_NOFATALERRORS );
idParser_EmbeddedGLSL src( LEXFL_NOFATALERRORS );
src.LoadMemory( in.c_str(), in.Length(), name );
idStrStatic<256> sourceName = "filename ";
sourceName += name;
src.AddDefine( sourceName );
src.AddDefine( "PC" );
for( int i = 0; i < compileMacros.Num(); i++ )
{
src.AddDefine( compileMacros[i] );
}
switch( glConfig.driverType )
{
case GLDRV_OPENGL_ES2:
case GLDRV_OPENGL_ES3:
//src.AddDefine( "GLES2" );
break;
}
if( !builtin && glConfig.gpuSkinningAvailable )
{
src.AddDefine( "USE_GPU_SKINNING" );
}
src.AddDefine( "USE_UNIFORM_ARRAYS" );
if( r_useHalfLambertLighting.GetBool() )
{
src.AddDefine( "USE_HALF_LAMBERT" );
}
if( r_useHDR.GetBool() )
{
src.AddDefine( "USE_LINEAR_RGB" );
}
if( r_pbrDebug.GetBool() )
{
src.AddDefine( "DEBUG_PBR" );
}
#if defined( USE_VULKAN )
src.AddDefine( "USE_VULKAN" );
#endif
// SMAA configuration
src.AddDefine( "SMAA_GLSL_3" );
src.AddDefine( "SMAA_RT_METRICS rpScreenCorrectionFactor " );
src.AddDefine( "SMAA_PRESET_HIGH" );
idList< idCGBlock > blocks;
blocks.SetNum( 100 );
idToken token;
while( !src.EndOfFile() )
{
idCGBlock& block = blocks.Alloc();
// read prefix
while( src.ReadToken( &token ) )
{
bool found = false;
for( int i = 0; i < numPrefixes; i++ )
{
if( token == prefixes[i] )
{
found = true;
break;
}
}
if( !found )
{
for( int i = 0; i < numTypes; i++ )
{
if( token == types[i] )
{
found = true;
break;
}
int typeLen = idStr::Length( types[i] );
if( token.Cmpn( types[i], typeLen ) == 0 )
{
for( int j = 0; j < numTypePosts; j++ )
{
if( idStr::Cmp( token.c_str() + typeLen, typePosts[j] ) == 0 )
{
found = true;
break;
}
}
if( found )
{
break;
}
}
}
}
if( found )
{
if( block.prefix.Length() > 0 && token.WhiteSpaceBeforeToken() )
{
block.prefix += ' ';
}
block.prefix += token;
}
else
{
src.UnreadToken( &token );
break;
}
}
if( !src.ReadToken( &token ) )
{
blocks.SetNum( blocks.Num() - 1 );
break;
}
block.name = token;
if( src.PeekTokenString( "=" ) || src.PeekTokenString( ":" ) || src.PeekTokenString( "[" ) )
{
src.ReadToken( &token );
block.postfix = token;
while( src.ReadToken( &token ) )
{
if( token == ";" )
{
block.postfix += ';';
break;
}
else
{
if( token.WhiteSpaceBeforeToken() )
{
block.postfix += ' ';
}
block.postfix += token;
}
}
}
else if( src.PeekTokenString( "(" ) )
{
idStr parms, body;
src.ParseBracedSection( parms, -1, true, '(', ')' );
if( src.CheckTokenString( ";" ) )
{
block.postfix = parms + ";";
}
else
{
src.ParseBracedSection( body, -1, true, '{', '}' );
block.postfix = parms + " " + body;
}
}
else if( src.PeekTokenString( "{" ) )
{
src.ParseBracedSection( block.postfix, -1, true, '{', '}' );
if( src.CheckTokenString( ";" ) )
{
block.postfix += ';';
}
}
else if( src.CheckTokenString( ";" ) )
{
block.postfix = idStr( ';' );
}
else
{
src.Warning( "Could not strip dead code -- unknown token %s\n", token.c_str() );
return in;
}
}
idList<int, TAG_RENDERPROG> stack;
for( int i = 0; i < blocks.Num(); i++ )
{
blocks[i].used = ( ( blocks[i].name == "main" )
|| blocks[i].name.Right( 4 ) == "_ubo"
);
if( blocks[i].name == "include" )
{
blocks[i].used = true;
blocks[i].name = ""; // clear out the include tag
}
if( blocks[i].used )
{
stack.Append( i );
}
}
while( stack.Num() > 0 )
{
int i = stack[stack.Num() - 1];
stack.SetNum( stack.Num() - 1 );
idLexer src( LEXFL_NOFATALERRORS );
src.LoadMemory( blocks[i].postfix.c_str(), blocks[i].postfix.Length(), name );
while( src.ReadToken( &token ) )
{
for( int j = 0; j < blocks.Num(); j++ )
{
if( !blocks[j].used )
{
if( token == blocks[j].name )
{
blocks[j].used = true;
stack.Append( j );
}
}
}
}
}
idStr out;
for( int i = 0; i < blocks.Num(); i++ )
{
if( blocks[i].used )
{
out += blocks[i].prefix;
out += ' ';
out += blocks[i].name;
out += ' ';
out += blocks[i].postfix;
out += '\n';
}
}
return out;
}
struct typeConversion_t
{
const char* typeCG;
const char* typeGLSL;
} typeConversion[] =
{
{ "void", "void" },
{ "fixed", "float" },
{ "float", "float" },
{ "float2", "vec2" },
{ "float3", "vec3" },
{ "float4", "vec4" },
{ "half", "float" },
{ "half2", "vec2" },
{ "half3", "vec3" },
{ "half4", "vec4" },
{ "int", "int" },
{ "int2", "ivec2" },
{ "int3", "ivec3" },
{ "int4", "ivec4" },
{ "bool", "bool" },
{ "bool2", "bvec2" },
{ "bool3", "bvec3" },
{ "bool4", "bvec4" },
{ "float2x2", "mat2x2" },
{ "float2x3", "mat2x3" },
{ "float2x4", "mat2x4" },
{ "float3x2", "mat3x2" },
{ "float3x3", "mat3x3" },
{ "float3x4", "mat3x4" },
{ "float4x2", "mat4x2" },
{ "float4x3", "mat4x3" },
{ "float4x4", "mat4x4" },
{ "sampler1D", "sampler1D" },
{ "sampler2D", "sampler2D" },
{ "sampler3D", "sampler3D" },
{ "samplerCUBE", "samplerCube" },
{ "sampler1DShadow", "sampler1DShadow" },
{ "sampler2DShadow", "sampler2DShadow" },
{ "sampler3DShadow", "sampler3DShadow" },
{ "samplerCubeShadow", "samplerCubeShadow" },
{ "sampler2DMS", "sampler2DMS" },
// RB: HACK convert all text2D stuff to modern texture() usage
{ "texCUBE", "texture" },
{ "tex2Dproj", "textureProj" },
{ "tex2D", "texture" },
{ NULL, NULL }
};
const char* vertexInsert =
{
// SRS - OSX OpenGL only supports up to GLSL 4.1, but current RenderProgs shaders seem to work as-is on OSX OpenGL drivers
#if defined(__APPLE__) && !defined(USE_VULKAN)
"#version 410\n"
#else
"#version 450\n"
#endif
"#pragma shader_stage( vertex )\n"
"#extension GL_ARB_separate_shader_objects : enable\n"
//"#define PC\n"
"\n"
//"float saturate( float v ) { return clamp( v, 0.0, 1.0 ); }\n"
//"vec2 saturate( vec2 v ) { return clamp( v, 0.0, 1.0 ); }\n"
//"vec3 saturate( vec3 v ) { return clamp( v, 0.0, 1.0 ); }\n"
//"vec4 saturate( vec4 v ) { return clamp( v, 0.0, 1.0 ); }\n"
//"vec4 tex2Dlod( sampler2D sampler, vec4 texcoord ) { return textureLod( sampler, texcoord.xy, texcoord.w ); }\n"
//"\n"
};
const char* fragmentInsert =
{
// SRS - OSX OpenGL only supports up to GLSL 4.1, but current RenderProgs shaders seem to work as-is on OSX OpenGL drivers
#if defined(__APPLE__) && !defined(USE_VULKAN)
"#version 410\n"
#else
"#version 450\n"
#endif
"#pragma shader_stage( fragment )\n"
"#extension GL_ARB_separate_shader_objects : enable\n"
//"#define PC\n"
"\n"
"void clip( float v ) { if ( v < 0.0 ) { discard; } }\n"
"void clip( vec2 v ) { if ( any( lessThan( v, vec2( 0.0 ) ) ) ) { discard; } }\n"
"void clip( vec3 v ) { if ( any( lessThan( v, vec3( 0.0 ) ) ) ) { discard; } }\n"
"void clip( vec4 v ) { if ( any( lessThan( v, vec4( 0.0 ) ) ) ) { discard; } }\n"
"\n"
"float saturate( float v ) { return clamp( v, 0.0, 1.0 ); }\n"
"vec2 saturate( vec2 v ) { return clamp( v, 0.0, 1.0 ); }\n"
"vec3 saturate( vec3 v ) { return clamp( v, 0.0, 1.0 ); }\n"
"vec4 saturate( vec4 v ) { return clamp( v, 0.0, 1.0 ); }\n"
"\n"
//"vec4 tex2D( sampler2D sampler, vec2 texcoord ) { return texture( sampler, texcoord.xy ); }\n"
//"vec4 tex2D( sampler2DShadow sampler, vec3 texcoord ) { return vec4( texture( sampler, texcoord.xyz ) ); }\n"
//"\n"
//"vec4 tex2D( sampler2D sampler, vec2 texcoord, vec2 dx, vec2 dy ) { return textureGrad( sampler, texcoord.xy, dx, dy ); }\n"
//"vec4 tex2D( sampler2DShadow sampler, vec3 texcoord, vec2 dx, vec2 dy ) { return vec4( textureGrad( sampler, texcoord.xyz, dx, dy ) ); }\n"
//"\n"
//"vec4 texCUBE( samplerCube sampler, vec3 texcoord ) { return texture( sampler, texcoord.xyz ); }\n"
//"vec4 texCUBE( samplerCubeShadow sampler, vec4 texcoord ) { return vec4( texture( sampler, texcoord.xyzw ) ); }\n"
//"\n"
//"vec4 tex1Dproj( sampler1D sampler, vec2 texcoord ) { return textureProj( sampler, texcoord ); }\n"
//"vec4 tex2Dproj( sampler2D sampler, vec3 texcoord ) { return textureProj( sampler, texcoord ); }\n"
//"vec4 tex3Dproj( sampler3D sampler, vec4 texcoord ) { return textureProj( sampler, texcoord ); }\n"
//"\n"
//"vec4 tex1Dbias( sampler1D sampler, vec4 texcoord ) { return texture( sampler, texcoord.x, texcoord.w ); }\n"
//"vec4 tex2Dbias( sampler2D sampler, vec4 texcoord ) { return texture( sampler, texcoord.xy, texcoord.w ); }\n"
//"vec4 tex3Dbias( sampler3D sampler, vec4 texcoord ) { return texture( sampler, texcoord.xyz, texcoord.w ); }\n"
//"vec4 texCUBEbias( samplerCube sampler, vec4 texcoord ) { return texture( sampler, texcoord.xyz, texcoord.w ); }\n"
//"\n"
//"vec4 tex1Dlod( sampler1D sampler, vec4 texcoord ) { return textureLod( sampler, texcoord.x, texcoord.w ); }\n"
//"vec4 tex2Dlod( sampler2D sampler, vec4 texcoord ) { return textureLod( sampler, texcoord.xy, texcoord.w ); }\n"
//"vec4 tex3Dlod( sampler3D sampler, vec4 texcoord ) { return textureLod( sampler, texcoord.xyz, texcoord.w ); }\n"
//"vec4 texCUBElod( samplerCube sampler, vec4 texcoord ) { return textureLod( sampler, texcoord.xyz, texcoord.w ); }\n"
//"\n"
};
// RB begin
const char* vertexInsert_GLSL_ES_3_00 =
{
"#version 300 es\n"
"#define PC\n"
"precision mediump float;\n"
//"#extension GL_ARB_gpu_shader5 : enable\n"
"\n"
"float saturate( float v ) { return clamp( v, 0.0, 1.0 ); }\n"
"vec2 saturate( vec2 v ) { return clamp( v, 0.0, 1.0 ); }\n"
"vec3 saturate( vec3 v ) { return clamp( v, 0.0, 1.0 ); }\n"
"vec4 saturate( vec4 v ) { return clamp( v, 0.0, 1.0 ); }\n"
//"vec4 tex2Dlod( sampler2D sampler, vec4 texcoord ) { return textureLod( sampler, texcoord.xy, texcoord.w ); }\n"
"\n"
};
const char* fragmentInsert_GLSL_ES_3_00 =
{
"#version 300 es\n"
"#define PC\n"
"precision mediump float;\n"
"precision lowp sampler2D;\n"
"precision lowp sampler2DShadow;\n"
"precision lowp sampler2DArray;\n"
"precision lowp sampler2DArrayShadow;\n"
"precision lowp samplerCube;\n"
"precision lowp samplerCubeShadow;\n"
"precision lowp sampler3D;\n"
"\n"
"void clip( float v ) { if ( v < 0.0 ) { discard; } }\n"
"void clip( vec2 v ) { if ( any( lessThan( v, vec2( 0.0 ) ) ) ) { discard; } }\n"
"void clip( vec3 v ) { if ( any( lessThan( v, vec3( 0.0 ) ) ) ) { discard; } }\n"
"void clip( vec4 v ) { if ( any( lessThan( v, vec4( 0.0 ) ) ) ) { discard; } }\n"
"\n"
"float saturate( float v ) { return clamp( v, 0.0, 1.0 ); }\n"
"vec2 saturate( vec2 v ) { return clamp( v, 0.0, 1.0 ); }\n"
"vec3 saturate( vec3 v ) { return clamp( v, 0.0, 1.0 ); }\n"
"vec4 saturate( vec4 v ) { return clamp( v, 0.0, 1.0 ); }\n"
"\n"
"vec4 tex2D( sampler2D sampler, vec2 texcoord ) { return texture( sampler, texcoord.xy ); }\n"
"vec4 tex2D( sampler2DShadow sampler, vec3 texcoord ) { return vec4( texture( sampler, texcoord.xyz ) ); }\n"
"\n"
"vec4 tex2D( sampler2D sampler, vec2 texcoord, vec2 dx, vec2 dy ) { return textureGrad( sampler, texcoord.xy, dx, dy ); }\n"
"vec4 tex2D( sampler2DShadow sampler, vec3 texcoord, vec2 dx, vec2 dy ) { return vec4( textureGrad( sampler, texcoord.xyz, dx, dy ) ); }\n"
"\n"
"vec4 texCUBE( samplerCube sampler, vec3 texcoord ) { return texture( sampler, texcoord.xyz ); }\n"
"vec4 texCUBE( samplerCubeShadow sampler, vec4 texcoord ) { return vec4( texture( sampler, texcoord.xyzw ) ); }\n"
"\n"
//"vec4 tex1Dproj( sampler1D sampler, vec2 texcoord ) { return textureProj( sampler, texcoord ); }\n"
"vec4 tex2Dproj( sampler2D sampler, vec3 texcoord ) { return textureProj( sampler, texcoord ); }\n"
"vec4 tex3Dproj( sampler3D sampler, vec4 texcoord ) { return textureProj( sampler, texcoord ); }\n"
"\n"
//"vec4 tex1Dbias( sampler1D sampler, vec4 texcoord ) { return texture( sampler, texcoord.x, texcoord.w ); }\n"
"vec4 tex2Dbias( sampler2D sampler, vec4 texcoord ) { return texture( sampler, texcoord.xy, texcoord.w ); }\n"
"vec4 tex3Dbias( sampler3D sampler, vec4 texcoord ) { return texture( sampler, texcoord.xyz, texcoord.w ); }\n"
"vec4 texCUBEbias( samplerCube sampler, vec4 texcoord ) { return texture( sampler, texcoord.xyz, texcoord.w ); }\n"
"\n"
//"vec4 tex1Dlod( sampler1D sampler, vec4 texcoord ) { return textureLod( sampler, texcoord.x, texcoord.w ); }\n"
"vec4 tex2Dlod( sampler2D sampler, vec4 texcoord ) { return textureLod( sampler, texcoord.xy, texcoord.w ); }\n"
"vec4 tex3Dlod( sampler3D sampler, vec4 texcoord ) { return textureLod( sampler, texcoord.xyz, texcoord.w ); }\n"
"vec4 texCUBElod( samplerCube sampler, vec4 texcoord ) { return textureLod( sampler, texcoord.xyz, texcoord.w ); }\n"
"\n"
};
// RB end
struct builtinConversion_t
{
const char* nameCG;
const char* nameGLSL;
} builtinConversion[] =
{
{ "frac", "fract" },
{ "lerp", "mix" },
{ "rsqrt", "inversesqrt" },
{ "ddx", "dFdx" },
{ "ddy", "dFdy" },
{ NULL, NULL }
};
struct inOutVariable_t
{
idStr type;
idStr nameCg;
idStr nameGLSL;
bool declareInOut;
};
/*
========================
ParseInOutStruct
========================
*/
void ParseInOutStruct( idLexer& src, int attribType, int attribIgnoreType, idList< inOutVariable_t >& inOutVars, bool in_Position4 )
{
src.ExpectTokenString( "{" );
while( !src.CheckTokenString( "}" ) )
{
inOutVariable_t var;
idToken token;
src.ReadToken( &token );
var.type = token;
src.ReadToken( &token );
var.nameCg = token;
if( !src.CheckTokenString( ":" ) )
{
src.SkipUntilString( ";" );
continue;
}
src.ReadToken( &token );
var.nameGLSL = token;
src.ExpectTokenString( ";" );
// convert the type
for( int i = 0; typeConversion[i].typeCG != NULL; i++ )
{
if( var.type.Cmp( typeConversion[i].typeCG ) == 0 )
{
var.type = typeConversion[i].typeGLSL;
break;
}
}
// convert the semantic to a GLSL name
for( int i = 0; attribsPC[i].semantic != NULL; i++ )
{
if( ( attribsPC[i].flags & attribType ) != 0 )
{
if( var.nameGLSL.Cmp( attribsPC[i].semantic ) == 0 )
{
var.nameGLSL = attribsPC[i].glsl;
break;
}
}
}
// check if it was defined previously
var.declareInOut = true;
for( int i = 0; i < inOutVars.Num(); i++ )
{
if( var.nameGLSL == inOutVars[i].nameGLSL )
{
var.declareInOut = false;
break;
}
}
// RB: ignore reserved builtin gl_ uniforms
//switch( glConfig.driverType )
{
//case GLDRV_OPENGL32_CORE_PROFILE:
//case GLDRV_OPENGL_ES2:
//case GLDRV_OPENGL_ES3:
//case GLDRV_OPENGL_MESA:
//default:
{
for( int i = 0; attribsPC[i].semantic != NULL; i++ )
{
if( var.nameGLSL.Cmp( attribsPC[i].glsl ) == 0 )
{
if( ( attribsPC[i].flags & attribIgnoreType ) != 0 )
{
var.declareInOut = false;
break;
}
}
}
//break;
}
}
if( glConfig.driverType == GLDRV_VULKAN )
{
// RB: HACK change vec4 in_Position to vec3
if( !in_Position4 && var.nameGLSL == "in_Position" && var.type == "vec4" )
{
var.type = "vec3";
}
}
inOutVars.Append( var );
}
src.ExpectTokenString( ";" );
}
/*
========================
ConvertCG2GLSL
========================
*/
idStr idRenderProgManager::ConvertCG2GLSL( const idStr& in, const char* name, rpStage_t stage, idStr& outLayout, bool vkGLSL, bool hasGPUSkinning, vertexLayoutType_t vertexLayout )
{
idStr program;
program.ReAllocate( in.Length() * 2, false );
idList< inOutVariable_t, TAG_RENDERPROG > varsIn;
idList< inOutVariable_t, TAG_RENDERPROG > varsOut;
idList< idStr > uniformList;
idList< idStr > samplerList;
idLexer src( LEXFL_NOFATALERRORS );
src.LoadMemory( in.c_str(), in.Length(), name );
bool in_Position4 = ( vertexLayout == LAYOUT_DRAW_SHADOW_VERT ) || ( vertexLayout == LAYOUT_DRAW_SHADOW_VERT_SKINNED );
bool inMain = false;
bool justEnteredMain = false;
const char* uniformArrayName = ( stage == SHADER_STAGE_VERTEX ) ? VERTEX_UNIFORM_ARRAY_NAME : FRAGMENT_UNIFORM_ARRAY_NAME;
char newline[128] = { "\n" };
idToken token;
while( src.ReadToken( &token ) )
{
// check for uniforms
// RB: added special case for matrix arrays
while( token == "uniform" && ( src.CheckTokenString( "float4" ) || src.CheckTokenString( "float4x4" ) ) )
{
src.ReadToken( &token );
idStr uniform = token;
// strip ': register()' from uniforms
if( src.CheckTokenString( ":" ) )
{
if( src.CheckTokenString( "register" ) )
{
src.SkipUntilString( ";" );
}
}
if( src.PeekTokenString( "[" ) )
{
while( src.ReadToken( &token ) )
{
uniform += token;
if( token == "]" )
{
break;
}
}
}
uniformList.Append( uniform );
src.ReadToken( &token );
}
// RB: check for sampler uniforms
if( vkGLSL )
{
while( token == "uniform" &&
( src.PeekTokenString( "sampler2D" ) ||
src.PeekTokenString( "samplerCUBE" ) ||
src.PeekTokenString( "sampler3D" ) ||
src.PeekTokenString( "sampler2DArrayShadow" ) ||
src.PeekTokenString( "sampler2DArray" ) ) )
{
idStr sampler;
// sampler2D or whatever
src.ReadToken( &token );
// HACK
if( token == "samplerCUBE" )
{
sampler += "samplerCube";
}
else
{
sampler += token;
}
sampler += " ";
// the actual variable name
src.ReadToken( &token );
sampler += token;
samplerList.Append( sampler );
// strip ': register()' from uniforms
if( src.CheckTokenString( ":" ) )
{
if( src.CheckTokenString( "register" ) )
{
src.SkipUntilString( ";" );
}
}
src.ReadToken( &token );
}
// RB: HACK to add matrices uniform block layer
if( token == "uniform" && src.PeekTokenString( "matrices_ubo" ) )
{
hasGPUSkinning = true;
//src.ReadToken( &token );
//src.SkipRestOfLine();
//program += "\nlayout( binding = 1 ) ";
src.SkipRestOfLine();
continue;
}
}
// convert the in/out structs
if( token == "struct" )
{
if( src.CheckTokenString( "VS_IN" ) )
{
ParseInOutStruct( src, AT_VS_IN, 0, varsIn, in_Position4 );
program += "\n\n";
for( int i = 0; i < varsIn.Num(); i++ )
{
if( varsIn[i].declareInOut )
{
// RB: add layout locations
if( vkGLSL )
{
program += va( "layout( location = %i ) in %s %s;\n", i, varsIn[i].type.c_str(), varsIn[i].nameGLSL.c_str() );
}
else
{
program += "in " + varsIn[i].type + " " + varsIn[i].nameGLSL + ";\n";
}
}
}
continue;
}
else if( src.CheckTokenString( "VS_OUT" ) )
{
// RB
ParseInOutStruct( src, AT_VS_OUT, AT_VS_OUT_RESERVED, varsOut, in_Position4 );
program += "\n";
int numDeclareOut = 0;
for( int i = 0; i < varsOut.Num(); i++ )
{
if( varsOut[i].declareInOut )
{
// RB: add layout locations
if( vkGLSL )
{
program += va( "layout( location = %i ) out %s %s;\n", numDeclareOut, varsOut[i].type.c_str(), varsOut[i].nameGLSL.c_str() );
}
else
{
program += "out " + varsOut[i].type + " " + varsOut[i].nameGLSL + ";\n";
}
numDeclareOut++;
}
}
continue;
}
else if( src.CheckTokenString( "PS_IN" ) )
{
ParseInOutStruct( src, AT_PS_IN, AT_PS_IN_RESERVED, varsIn, in_Position4 );
program += "\n\n";
int numDeclareOut = 0;
for( int i = 0; i < varsIn.Num(); i++ )
{
if( varsIn[i].declareInOut )
{
// RB: add layout locations
if( vkGLSL )
{
program += va( "layout( location = %i ) in %s %s;\n", numDeclareOut, varsIn[i].type.c_str(), varsIn[i].nameGLSL.c_str() );
}
else
{
program += "in " + varsIn[i].type + " " + varsIn[i].nameGLSL + ";\n";
}
numDeclareOut++;
}
}
inOutVariable_t var;
var.type = "vec4";
var.nameCg = "position";
var.nameGLSL = "gl_FragCoord";
varsIn.Append( var );
continue;
}
else if( src.CheckTokenString( "PS_OUT" ) )
{
// RB begin
ParseInOutStruct( src, AT_PS_OUT, AT_PS_OUT_RESERVED, varsOut, in_Position4 );
program += "\n";
for( int i = 0; i < varsOut.Num(); i++ )
{
if( varsOut[i].declareInOut )
{
// RB: add layout locations
if( vkGLSL )
{
program += va( "layout( location = %i ) out %s %s;\n", i, varsOut[i].type.c_str(), varsOut[i].nameGLSL.c_str() );
}
else
{
program += "out " + varsOut[i].type + " " + varsOut[i].nameGLSL + ";\n";
}
}
}
continue;
}
}
// strip 'static'
if( token == "static" )
{
program += ( token.linesCrossed > 0 ) ? newline : ( token.WhiteSpaceBeforeToken() > 0 ? " " : "" );
src.SkipWhiteSpace( true ); // remove white space between 'static' and the next token
continue;
}
// strip ': register()' from uniforms
if( token == ":" )
{
if( src.CheckTokenString( "register" ) )
{
src.SkipUntilString( ";" );
program += ";";
continue;
}
}
// strip in/program parameters from main
if( token == "void" && src.CheckTokenString( "main" ) )
{
src.ExpectTokenString( "(" );
while( src.ReadToken( &token ) )
{
if( token == ")" )
{
break;
}
}
program += "\nvoid main()";
inMain = true;
continue;
}
// strip 'const' from local variables in main()
if( token == "const" && inMain )
{
program += ( token.linesCrossed > 0 ) ? newline : ( token.WhiteSpaceBeforeToken() > 0 ? " " : "" );
src.SkipWhiteSpace( true ); // remove white space between 'const' and the next token
continue;
}
// maintain indentation
if( token == "{" )
{
program += ( token.linesCrossed > 0 ) ? newline : ( token.WhiteSpaceBeforeToken() > 0 ? " " : "" );
program += "{";
int len = Min( idStr::Length( newline ) + 1, ( int )sizeof( newline ) - 1 );
newline[len - 1] = '\t';
newline[len - 0] = '\0';
// RB: add this to every vertex shader
if( inMain && !justEnteredMain && ( stage == SHADER_STAGE_VERTEX ) && vkGLSL && !in_Position4 )
{
program += "\n\tvec4 position4 = vec4( in_Position, 1.0 );\n";
justEnteredMain = true;
}
continue;
}
if( token == "}" )
{
int len = Max( idStr::Length( newline ) - 1, 0 );
newline[len] = '\0';
program += ( token.linesCrossed > 0 ) ? newline : ( token.WhiteSpaceBeforeToken() > 0 ? " " : "" );
program += "}";
continue;
}
// check for a type conversion
bool foundType = false;
for( int i = 0; typeConversion[i].typeCG != NULL; i++ )
{
if( token.Cmp( typeConversion[i].typeCG ) == 0 )
{
program += ( token.linesCrossed > 0 ) ? newline : ( token.WhiteSpaceBeforeToken() > 0 ? " " : "" );
program += typeConversion[i].typeGLSL;
foundType = true;
break;
}
}
if( foundType )
{
continue;
}
// check for uniforms that need to be converted to the array
bool isUniform = false;
for( int i = 0; i < uniformList.Num(); i++ )
{
if( token == uniformList[i] )
{
program += ( token.linesCrossed > 0 ) ? newline : ( token.WhiteSpaceBeforeToken() > 0 ? " " : "" );
// RB: for some unknown reasons has the Nvidia driver problems with regular uniforms when using glUniform4fv
// so we need these uniform arrays
// I added special check rpShadowMatrices so we can still index the uniforms from the shader
if( idStr::Cmp( uniformList[i].c_str(), "rpShadowMatrices" ) == 0 )
{
if( vkGLSL )
{
program += va( "%s[ ", uniformList[i].c_str() );
}
else
{
program += va( "%s[/* %s */ %d + ", uniformArrayName, uniformList[i].c_str(), i );
}
if( src.ExpectTokenString( "[" ) )
{
idStr uniformIndexing;
while( src.ReadToken( &token ) )
{
if( token == "]" )
{
break;
}
uniformIndexing += token;
uniformIndexing += " ";
}
program += uniformIndexing + "]";
}
}
else
{
if( vkGLSL )
{
program += va( "%s", uniformList[i].c_str() );
}
else
{
program += va( "%s[%d /* %s */]", uniformArrayName, i, uniformList[i].c_str() );
}
}
isUniform = true;
break;
}
}
if( isUniform )
{
continue;
}
// check for input/output parameters
if( src.CheckTokenString( "." ) )
{
if( token == "vertex" || token == "fragment" )
{
idToken member;
src.ReadToken( &member );
bool foundInOut = false;
for( int i = 0; i < varsIn.Num(); i++ )
{
if( member.Cmp( varsIn[i].nameCg ) == 0 )
{
program += ( token.linesCrossed > 0 ) ? newline : ( token.WhiteSpaceBeforeToken() > 0 ? " " : "" );
/*
RB: HACK use position4 instead of in_Position directly
because I can't figure out how to do strides with Vulkan
*/
if( !in_Position4 && vkGLSL && ( varsIn[i].nameGLSL == "in_Position" ) )
{
program += "position4";
}
else
{
program += varsIn[i].nameGLSL;
}
foundInOut = true;
break;
}
}
if( !foundInOut )
{
src.Error( "invalid input parameter %s.%s", token.c_str(), member.c_str() );
program += ( token.linesCrossed > 0 ) ? newline : ( token.WhiteSpaceBeforeToken() > 0 ? " " : "" );
program += token;
program += ".";
program += member;
}
continue;
}
if( token == "result" )
{
idToken member;
src.ReadToken( &member );
bool foundInOut = false;
for( int i = 0; i < varsOut.Num(); i++ )
{
if( member.Cmp( varsOut[i].nameCg ) == 0 )
{
program += ( token.linesCrossed > 0 ) ? newline : ( token.WhiteSpaceBeforeToken() > 0 ? " " : "" );
program += varsOut[i].nameGLSL;
foundInOut = true;
break;
}
}
if( !foundInOut )
{
src.Error( "invalid output parameter %s.%s", token.c_str(), member.c_str() );
program += ( token.linesCrossed > 0 ) ? newline : ( token.WhiteSpaceBeforeToken() > 0 ? " " : "" );
program += token;
program += ".";
program += member;
}
continue;
}
program += ( token.linesCrossed > 0 ) ? newline : ( token.WhiteSpaceBeforeToken() > 0 ? " " : "" );
program += token;
program += ".";
continue;
}
// check for a function conversion
bool foundFunction = false;
for( int i = 0; builtinConversion[i].nameCG != NULL; i++ )
{
if( token.Cmp( builtinConversion[i].nameCG ) == 0 )
{
program += ( token.linesCrossed > 0 ) ? newline : ( token.WhiteSpaceBeforeToken() > 0 ? " " : "" );
program += builtinConversion[i].nameGLSL;
foundFunction = true;
break;
}
}
if( foundFunction )
{
continue;
}
program += ( token.linesCrossed > 0 ) ? newline : ( token.WhiteSpaceBeforeToken() > 0 ? " " : "" );
program += token;
}
idStr out;
// RB: tell shader debuggers what shader we look at
idStr filenameHint = "// filename " + idStr( name ) + "\n";
// RB: changed to allow multiple versions of GLSL
if( stage == SHADER_STAGE_VERTEX )
{
switch( glConfig.driverType )
{
case GLDRV_OPENGL_MESA:
{
out.ReAllocate( idStr::Length( vertexInsert_GLSL_ES_3_00 ) + in.Length() * 2, false );
out += filenameHint;
out += vertexInsert_GLSL_ES_3_00;
break;
}
default:
{
out.ReAllocate( idStr::Length( vertexInsert ) + in.Length() * 2, false );
out += filenameHint;
out += vertexInsert;
break;
}
}
}
else
{
switch( glConfig.driverType )
{
case GLDRV_OPENGL_MESA:
{
out.ReAllocate( idStr::Length( fragmentInsert_GLSL_ES_3_00 ) + in.Length() * 2, false );
out += filenameHint;
out += fragmentInsert_GLSL_ES_3_00;
break;
}
default:
{
out.ReAllocate( idStr::Length( fragmentInsert ) + in.Length() * 2, false );
out += filenameHint;
out += fragmentInsert;
break;
}
}
}
// RB end
if( uniformList.Num() > 0 )
{
if( vkGLSL )
{
out += "\n";
if( stage == SHADER_STAGE_VERTEX )
{
out += "layout( binding = 0 ) uniform UBOV {\n";
}
else
{
//out += "layout( binding = 1 ) uniform UBOF {\n";
out += va( "layout( binding = %i ) uniform UBOF {\n", hasGPUSkinning ? 2 : 1 );
}
for( int i = 0; i < uniformList.Num(); i++ )
{
out += "\tvec4 ";
out += uniformList[i];
// TODO fix this ugly hack
if( uniformList[i] == "rpShadowMatrices" )
{
out += "[6*4]";
}
out += ";\n";
}
out += "};\n";
if( hasGPUSkinning && ( stage == SHADER_STAGE_VERTEX ) )
{
out += "\nlayout( binding = 1 ) uniform UBO_MAT {\n";
out += "\t vec4 matrices[ 408 ];\n";
out += "};\n";
}
}
else
{
int extraSize = 0;
for( int i = 0; i < uniformList.Num(); i++ )
{
if( idStr::Cmp( uniformList[i].c_str(), "rpShadowMatrices" ) == 0 )
{
extraSize += ( 6 * 4 );
}
}
out += va( "\nuniform vec4 %s[%d];\n", uniformArrayName, uniformList.Num() + extraSize );
}
}
// RB: add samplers with layout bindings
if( vkGLSL )
{
int bindingOffset = 1;
if( hasGPUSkinning )
{
bindingOffset++;
}
if( uniformList.Num() > 0 )
{
bindingOffset++;
}
for( int i = 0; i < samplerList.Num(); i++ )
{
out += va( "layout( binding = %i ) uniform %s;\n", i + bindingOffset , samplerList[i].c_str() );
}
}
out += program;
outLayout += "uniforms [\n";
for( int i = 0; i < uniformList.Num(); i++ )
{
outLayout += "\t";
outLayout += uniformList[i];
outLayout += "\n";
}
outLayout += "]\n";
outLayout += "bindings [\n";
if( uniformList.Num() > 0 )
{
outLayout += "\tubo\n";
}
if( hasGPUSkinning && ( stage == SHADER_STAGE_VERTEX ) )
{
outLayout += "\tubo\n";
}
for( int i = 0; i < samplerList.Num(); i++ )
{
outLayout += "\tsampler\n";
}
outLayout += "]\n";
return out;
}
/*
================================================================================================
idRenderProgManager::FindGLSLProgram
================================================================================================
*/
int idRenderProgManager::FindGLSLProgram( const char* name, int vIndex, int fIndex )
{
for( int i = 0; i < renderProgs.Num(); ++i )
{
if( ( renderProgs[i].vertexShaderIndex == vIndex ) && ( renderProgs[i].fragmentShaderIndex == fIndex ) )
{
return i;
}
}
renderProg_t program;
program.name = name;
int index = renderProgs.Append( program );
LoadGLSLProgram( index, vIndex, fIndex );
return index;
}
/*
================================================================================================
idRenderProgManager::GetGLSLParmName
================================================================================================
*/
const char* idRenderProgManager::GetGLSLParmName( int rp ) const
{
assert( rp < RENDERPARM_TOTAL );
return GLSLParmNames[ rp ];
}
// RB begin
const char* idRenderProgManager::GetGLSLMacroName( shaderFeature_t sf ) const
{
assert( sf < MAX_SHADER_MACRO_NAMES );
return GLSLMacroNames[ sf ];
}
// RB end
/*
================================================================================================
idRenderProgManager::SetUniformValue
================================================================================================
*/
void idRenderProgManager::SetUniformValue( const renderParm_t rp, const float* value )
{
for( int i = 0; i < 4; i++ )
{
uniforms[rp][i] = value[i];
}
}
/*
================================================================================================
idRenderProgManager::ZeroUniforms
================================================================================================
*/
void idRenderProgManager::ZeroUniforms()
{
memset( uniforms.Ptr(), 0, uniforms.Allocated() );
}