2012-11-26 18:58:24 +00:00
|
|
|
/*
|
|
|
|
===========================================================================
|
|
|
|
|
|
|
|
Doom 3 BFG Edition GPL Source Code
|
2012-11-28 15:47:07 +00:00
|
|
|
Copyright (C) 1993-2012 id Software LLC, a ZeniMax Media company.
|
2012-11-26 18:58:24 +00:00
|
|
|
|
2012-11-28 15:47:07 +00:00
|
|
|
This file is part of the Doom 3 BFG Edition GPL Source Code ("Doom 3 BFG Edition Source Code").
|
2012-11-26 18:58:24 +00:00
|
|
|
|
|
|
|
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
|
2012-12-22 15:18:19 +00:00
|
|
|
#include "precompiled.h"
|
2012-11-26 18:58:24 +00:00
|
|
|
|
|
|
|
#include "tr_local.h"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
idRenderProgManager renderProgManager;
|
|
|
|
|
|
|
|
/*
|
|
|
|
================================================================================================
|
|
|
|
idRenderProgManager::idRenderProgManager()
|
|
|
|
================================================================================================
|
|
|
|
*/
|
2012-11-28 15:47:07 +00:00
|
|
|
idRenderProgManager::idRenderProgManager()
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
================================================================================================
|
|
|
|
idRenderProgManager::~idRenderProgManager()
|
|
|
|
================================================================================================
|
|
|
|
*/
|
2012-11-28 15:47:07 +00:00
|
|
|
idRenderProgManager::~idRenderProgManager()
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
================================================================================================
|
|
|
|
R_ReloadShaders
|
|
|
|
================================================================================================
|
|
|
|
*/
|
2012-11-28 15:47:07 +00:00
|
|
|
static void R_ReloadShaders( const idCmdArgs& args )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
renderProgManager.KillAllShaders();
|
|
|
|
renderProgManager.LoadAllShaders();
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
================================================================================================
|
|
|
|
idRenderProgManager::Init()
|
|
|
|
================================================================================================
|
|
|
|
*/
|
2012-11-28 15:47:07 +00:00
|
|
|
void idRenderProgManager::Init()
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
common->Printf( "----- Initializing Render Shaders -----\n" );
|
2012-11-28 15:47:07 +00:00
|
|
|
|
|
|
|
|
|
|
|
for( int i = 0; i < MAX_BUILTINS; i++ )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
builtinShaders[i] = -1;
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
struct builtinShaders_t
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
int index;
|
2012-11-28 15:47:07 +00:00
|
|
|
const char* name;
|
|
|
|
} builtins[] =
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
{ BUILTIN_GUI, "gui.vfp" },
|
|
|
|
{ BUILTIN_COLOR, "color.vfp" },
|
|
|
|
{ BUILTIN_SIMPLESHADE, "simpleshade.vfp" },
|
|
|
|
{ BUILTIN_TEXTURED, "texture.vfp" },
|
|
|
|
{ BUILTIN_TEXTURE_VERTEXCOLOR, "texture_color.vfp" },
|
|
|
|
{ BUILTIN_TEXTURE_VERTEXCOLOR_SKINNED, "texture_color_skinned.vfp" },
|
|
|
|
{ BUILTIN_TEXTURE_TEXGEN_VERTEXCOLOR, "texture_color_texgen.vfp" },
|
|
|
|
{ BUILTIN_INTERACTION, "interaction.vfp" },
|
|
|
|
{ BUILTIN_INTERACTION_SKINNED, "interaction_skinned.vfp" },
|
|
|
|
{ BUILTIN_INTERACTION_AMBIENT, "interactionAmbient.vfp" },
|
|
|
|
{ BUILTIN_INTERACTION_AMBIENT_SKINNED, "interactionAmbient_skinned.vfp" },
|
|
|
|
{ BUILTIN_ENVIRONMENT, "environment.vfp" },
|
|
|
|
{ BUILTIN_ENVIRONMENT_SKINNED, "environment_skinned.vfp" },
|
|
|
|
{ BUILTIN_BUMPY_ENVIRONMENT, "bumpyEnvironment.vfp" },
|
|
|
|
{ BUILTIN_BUMPY_ENVIRONMENT_SKINNED, "bumpyEnvironment_skinned.vfp" },
|
2012-11-28 15:47:07 +00:00
|
|
|
|
2012-11-26 18:58:24 +00:00
|
|
|
{ BUILTIN_DEPTH, "depth.vfp" },
|
|
|
|
{ BUILTIN_DEPTH_SKINNED, "depth_skinned.vfp" },
|
|
|
|
{ BUILTIN_SHADOW_DEBUG, "shadowDebug.vfp" },
|
|
|
|
{ BUILTIN_SHADOW_DEBUG_SKINNED, "shadowDebug_skinned.vfp" },
|
2012-11-28 15:47:07 +00:00
|
|
|
|
2012-11-26 18:58:24 +00:00
|
|
|
{ BUILTIN_BLENDLIGHT, "blendlight.vfp" },
|
|
|
|
{ BUILTIN_FOG, "fog.vfp" },
|
|
|
|
{ BUILTIN_FOG_SKINNED, "fog_skinned.vfp" },
|
|
|
|
{ BUILTIN_SKYBOX, "skybox.vfp" },
|
|
|
|
{ BUILTIN_WOBBLESKY, "wobblesky.vfp" },
|
|
|
|
{ BUILTIN_POSTPROCESS, "postprocess.vfp" },
|
|
|
|
{ BUILTIN_STEREO_DEGHOST, "stereoDeGhost.vfp" },
|
|
|
|
{ BUILTIN_STEREO_WARP, "stereoWarp.vfp" },
|
|
|
|
{ BUILTIN_ZCULL_RECONSTRUCT, "zcullReconstruct.vfp" },
|
|
|
|
{ BUILTIN_BINK, "bink.vfp" },
|
|
|
|
{ BUILTIN_BINK_GUI, "bink_gui.vfp" },
|
|
|
|
{ BUILTIN_STEREO_INTERLACE, "stereoInterlace.vfp" },
|
|
|
|
{ BUILTIN_MOTION_BLUR, "motionBlur.vfp" },
|
|
|
|
};
|
|
|
|
int numBuiltins = sizeof( builtins ) / sizeof( builtins[0] );
|
|
|
|
vertexShaders.SetNum( numBuiltins );
|
|
|
|
fragmentShaders.SetNum( numBuiltins );
|
|
|
|
glslPrograms.SetNum( numBuiltins );
|
2012-11-28 15:47:07 +00:00
|
|
|
|
|
|
|
for( int i = 0; i < numBuiltins; i++ )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
vertexShaders[i].name = builtins[i].name;
|
|
|
|
fragmentShaders[i].name = builtins[i].name;
|
|
|
|
builtinShaders[builtins[i].index] = i;
|
|
|
|
LoadVertexShader( i );
|
|
|
|
LoadFragmentShader( i );
|
|
|
|
LoadGLSLProgram( i, i, i );
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
|
2012-11-26 18:58:24 +00:00
|
|
|
// Special case handling for fastZ shaders
|
|
|
|
builtinShaders[BUILTIN_SHADOW] = FindVertexShader( "shadow.vp" );
|
|
|
|
builtinShaders[BUILTIN_SHADOW_SKINNED] = FindVertexShader( "shadow_skinned.vp" );
|
2012-11-28 15:47:07 +00:00
|
|
|
|
2012-11-26 18:58:24 +00:00
|
|
|
FindGLSLProgram( "shadow.vp", builtinShaders[BUILTIN_SHADOW], -1 );
|
|
|
|
FindGLSLProgram( "shadow_skinned.vp", builtinShaders[BUILTIN_SHADOW_SKINNED], -1 );
|
2012-11-28 15:47:07 +00:00
|
|
|
|
2012-11-26 18:58:24 +00:00
|
|
|
glslUniforms.SetNum( RENDERPARM_USER + MAX_GLSL_USER_PARMS, vec4_zero );
|
2012-11-28 15:47:07 +00:00
|
|
|
|
2012-11-26 18:58:24 +00:00
|
|
|
vertexShaders[builtinShaders[BUILTIN_TEXTURE_VERTEXCOLOR_SKINNED]].usesJoints = true;
|
|
|
|
vertexShaders[builtinShaders[BUILTIN_INTERACTION_SKINNED]].usesJoints = true;
|
|
|
|
vertexShaders[builtinShaders[BUILTIN_INTERACTION_AMBIENT_SKINNED]].usesJoints = true;
|
|
|
|
vertexShaders[builtinShaders[BUILTIN_ENVIRONMENT_SKINNED]].usesJoints = true;
|
|
|
|
vertexShaders[builtinShaders[BUILTIN_BUMPY_ENVIRONMENT_SKINNED]].usesJoints = true;
|
|
|
|
vertexShaders[builtinShaders[BUILTIN_DEPTH_SKINNED]].usesJoints = true;
|
|
|
|
vertexShaders[builtinShaders[BUILTIN_SHADOW_SKINNED]].usesJoints = true;
|
|
|
|
vertexShaders[builtinShaders[BUILTIN_SHADOW_DEBUG_SKINNED]].usesJoints = true;
|
|
|
|
vertexShaders[builtinShaders[BUILTIN_FOG_SKINNED]].usesJoints = true;
|
2012-11-28 15:47:07 +00:00
|
|
|
|
2012-11-26 18:58:24 +00:00
|
|
|
cmdSystem->AddCommand( "reloadShaders", R_ReloadShaders, CMD_FL_RENDERER, "reloads shaders" );
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
================================================================================================
|
|
|
|
idRenderProgManager::LoadAllShaders()
|
|
|
|
================================================================================================
|
|
|
|
*/
|
2012-11-28 15:47:07 +00:00
|
|
|
void idRenderProgManager::LoadAllShaders()
|
|
|
|
{
|
|
|
|
for( int i = 0; i < vertexShaders.Num(); i++ )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
LoadVertexShader( i );
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
for( int i = 0; i < fragmentShaders.Num(); i++ )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
LoadFragmentShader( i );
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
|
|
|
|
for( int i = 0; i < glslPrograms.Num(); ++i )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
LoadGLSLProgram( i, glslPrograms[i].vertexShaderIndex, glslPrograms[i].fragmentShaderIndex );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
================================================================================================
|
|
|
|
idRenderProgManager::KillAllShaders()
|
|
|
|
================================================================================================
|
|
|
|
*/
|
2012-11-28 15:47:07 +00:00
|
|
|
void idRenderProgManager::KillAllShaders()
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
Unbind();
|
2012-11-28 15:47:07 +00:00
|
|
|
for( int i = 0; i < vertexShaders.Num(); i++ )
|
|
|
|
{
|
|
|
|
if( vertexShaders[i].progId != INVALID_PROGID )
|
|
|
|
{
|
2012-12-17 16:30:59 +00:00
|
|
|
glDeleteShader( vertexShaders[i].progId );
|
2012-11-26 18:58:24 +00:00
|
|
|
vertexShaders[i].progId = INVALID_PROGID;
|
|
|
|
}
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
for( int i = 0; i < fragmentShaders.Num(); i++ )
|
|
|
|
{
|
|
|
|
if( fragmentShaders[i].progId != INVALID_PROGID )
|
|
|
|
{
|
2012-12-17 16:30:59 +00:00
|
|
|
glDeleteShader( fragmentShaders[i].progId );
|
2012-11-26 18:58:24 +00:00
|
|
|
fragmentShaders[i].progId = INVALID_PROGID;
|
|
|
|
}
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
for( int i = 0; i < glslPrograms.Num(); ++i )
|
|
|
|
{
|
|
|
|
if( glslPrograms[i].progId != INVALID_PROGID )
|
|
|
|
{
|
2012-12-17 16:30:59 +00:00
|
|
|
glDeleteProgram( glslPrograms[i].progId );
|
2012-11-26 18:58:24 +00:00
|
|
|
glslPrograms[i].progId = INVALID_PROGID;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
================================================================================================
|
|
|
|
idRenderProgManager::Shutdown()
|
|
|
|
================================================================================================
|
|
|
|
*/
|
2012-11-28 15:47:07 +00:00
|
|
|
void idRenderProgManager::Shutdown()
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
KillAllShaders();
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
================================================================================================
|
|
|
|
idRenderProgManager::FindVertexShader
|
|
|
|
================================================================================================
|
|
|
|
*/
|
2012-11-28 15:47:07 +00:00
|
|
|
int idRenderProgManager::FindVertexShader( const char* name )
|
|
|
|
{
|
|
|
|
for( int i = 0; i < vertexShaders.Num(); i++ )
|
|
|
|
{
|
|
|
|
if( vertexShaders[i].name.Icmp( name ) == 0 )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
LoadVertexShader( i );
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
vertexShader_t shader;
|
|
|
|
shader.name = name;
|
|
|
|
int index = vertexShaders.Append( shader );
|
|
|
|
LoadVertexShader( index );
|
|
|
|
currentVertexShader = index;
|
2012-11-28 15:47:07 +00:00
|
|
|
|
2012-11-26 18:58:24 +00:00
|
|
|
// FIXME: we should really scan the program source code for using rpEnableSkinning but at this
|
|
|
|
// point we directly load a binary and the program source code is not available on the consoles
|
2012-11-28 15:47:07 +00:00
|
|
|
if( idStr::Icmp( name, "heatHaze.vfp" ) == 0 ||
|
2012-11-26 18:58:24 +00:00
|
|
|
idStr::Icmp( name, "heatHazeWithMask.vfp" ) == 0 ||
|
2012-11-28 15:47:07 +00:00
|
|
|
idStr::Icmp( name, "heatHazeWithMaskAndVertex.vfp" ) == 0 )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
vertexShaders[index].usesJoints = true;
|
|
|
|
vertexShaders[index].optionalSkinning = true;
|
|
|
|
}
|
2012-11-28 15:47:07 +00:00
|
|
|
|
2012-11-26 18:58:24 +00:00
|
|
|
return index;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
================================================================================================
|
|
|
|
idRenderProgManager::FindFragmentShader
|
|
|
|
================================================================================================
|
|
|
|
*/
|
2012-11-28 15:47:07 +00:00
|
|
|
int idRenderProgManager::FindFragmentShader( const char* name )
|
|
|
|
{
|
|
|
|
for( int i = 0; i < fragmentShaders.Num(); i++ )
|
|
|
|
{
|
|
|
|
if( fragmentShaders[i].name.Icmp( name ) == 0 )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
LoadFragmentShader( i );
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fragmentShader_t shader;
|
|
|
|
shader.name = name;
|
|
|
|
int index = fragmentShaders.Append( shader );
|
|
|
|
LoadFragmentShader( index );
|
|
|
|
currentFragmentShader = index;
|
|
|
|
return index;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
================================================================================================
|
|
|
|
idRenderProgManager::LoadVertexShader
|
|
|
|
================================================================================================
|
|
|
|
*/
|
2012-11-28 15:47:07 +00:00
|
|
|
void idRenderProgManager::LoadVertexShader( int index )
|
|
|
|
{
|
|
|
|
if( vertexShaders[index].progId != INVALID_PROGID )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
return; // Already loaded
|
|
|
|
}
|
|
|
|
vertexShaders[index].progId = ( GLuint ) LoadGLSLShader( GL_VERTEX_SHADER, vertexShaders[index].name, vertexShaders[index].uniforms );
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
================================================================================================
|
|
|
|
idRenderProgManager::LoadFragmentShader
|
|
|
|
================================================================================================
|
|
|
|
*/
|
2012-11-28 15:47:07 +00:00
|
|
|
void idRenderProgManager::LoadFragmentShader( int index )
|
|
|
|
{
|
|
|
|
if( fragmentShaders[index].progId != INVALID_PROGID )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
return; // Already loaded
|
|
|
|
}
|
|
|
|
fragmentShaders[index].progId = ( GLuint ) LoadGLSLShader( GL_FRAGMENT_SHADER, fragmentShaders[index].name, fragmentShaders[index].uniforms );
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
================================================================================================
|
|
|
|
idRenderProgManager::BindShader
|
|
|
|
================================================================================================
|
|
|
|
*/
|
2012-11-28 15:47:07 +00:00
|
|
|
void idRenderProgManager::BindShader( int vIndex, int fIndex )
|
|
|
|
{
|
|
|
|
if( currentVertexShader == vIndex && currentFragmentShader == fIndex )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
currentVertexShader = vIndex;
|
|
|
|
currentFragmentShader = fIndex;
|
|
|
|
// vIndex denotes the GLSL program
|
2012-11-28 15:47:07 +00:00
|
|
|
if( vIndex >= 0 && vIndex < glslPrograms.Num() )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
currentRenderProgram = vIndex;
|
|
|
|
RENDERLOG_PRINTF( "Binding GLSL Program %s\n", glslPrograms[vIndex].name.c_str() );
|
2012-12-17 16:30:59 +00:00
|
|
|
glUseProgram( glslPrograms[vIndex].progId );
|
2012-11-26 18:58:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
================================================================================================
|
|
|
|
idRenderProgManager::Unbind
|
|
|
|
================================================================================================
|
|
|
|
*/
|
2012-11-28 15:47:07 +00:00
|
|
|
void idRenderProgManager::Unbind()
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
currentVertexShader = -1;
|
|
|
|
currentFragmentShader = -1;
|
2012-11-28 15:47:07 +00:00
|
|
|
|
2012-12-17 16:30:59 +00:00
|
|
|
glUseProgram( 0 );
|
2012-11-26 18:58:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
================================================================================================
|
|
|
|
idRenderProgManager::SetRenderParms
|
|
|
|
================================================================================================
|
|
|
|
*/
|
2012-11-28 15:47:07 +00:00
|
|
|
void idRenderProgManager::SetRenderParms( renderParm_t rp, const float* value, int num )
|
|
|
|
{
|
|
|
|
for( int i = 0; i < num; i++ )
|
|
|
|
{
|
|
|
|
SetRenderParm( ( renderParm_t )( rp + i ), value + ( i * 4 ) );
|
2012-11-26 18:58:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
================================================================================================
|
|
|
|
idRenderProgManager::SetRenderParm
|
|
|
|
================================================================================================
|
|
|
|
*/
|
2012-11-28 15:47:07 +00:00
|
|
|
void idRenderProgManager::SetRenderParm( renderParm_t rp, const float* value )
|
|
|
|
{
|
2012-11-26 18:58:24 +00:00
|
|
|
SetUniformValue( rp, value );
|
|
|
|
}
|
|
|
|
|