jkxr/Projects/Android/jni/OpenJK/codemp/rd-gles/tr_shader.cpp
Simon 4597b03873 Initial Commit
Opens in Android Studio but haven't even tried to build it yet (it won't.. I know that much!)
2022-09-18 16:37:21 +01:00

4775 lines
126 KiB
C++

#include "tr_local.h"
// tr_shader.c -- this file deals with the parsing and definition of shaders
#define USE_NEW_SHADER_HASH
/////////////////////////////////////////////////////////////////////////////////////////////////////////////
// Vertex and Pixel Shader definitions. - AReis
/***********************************************************************************************************/
// This vertex shader basically passes through most values and calculates no lighting. The only
// unusual thing it does is add the inputed texel offsets to all four texture units (this allows
// nearest neighbor pixel peeking).
const unsigned char g_strGlowVShaderARB[] =
{
"!!ARBvp1.0\
\
# Input.\n\
ATTRIB iPos = vertex.position;\
ATTRIB iColor = vertex.color;\
ATTRIB iTex0 = vertex.texcoord[0];\
ATTRIB iTex1 = vertex.texcoord[1];\
ATTRIB iTex2 = vertex.texcoord[2];\
ATTRIB iTex3 = vertex.texcoord[3];\
\
# Output.\n\
OUTPUT oPos = result.position;\
OUTPUT oColor = result.color;\
OUTPUT oTex0 = result.texcoord[0];\
OUTPUT oTex1 = result.texcoord[1];\
OUTPUT oTex2 = result.texcoord[2];\
OUTPUT oTex3 = result.texcoord[3];\
\
# Constants.\n\
PARAM ModelViewProj[4]= { state.matrix.mvp };\
PARAM TexelOffset0 = program.env[0];\
PARAM TexelOffset1 = program.env[1];\
PARAM TexelOffset2 = program.env[2];\
PARAM TexelOffset3 = program.env[3];\
\
# Main.\n\
DP4 oPos.x, ModelViewProj[0], iPos;\
DP4 oPos.y, ModelViewProj[1], iPos;\
DP4 oPos.z, ModelViewProj[2], iPos;\
DP4 oPos.w, ModelViewProj[3], iPos;\
MOV oColor, iColor;\
# Notice the optimization of using one texture coord instead of all four.\n\
ADD oTex0, iTex0, TexelOffset0;\
ADD oTex1, iTex0, TexelOffset1;\
ADD oTex2, iTex0, TexelOffset2;\
ADD oTex3, iTex0, TexelOffset3;\
\
END"
};
// This Pixel Shader loads four texture units and adds them all together (with a modifier
// multiplied to each in the process). The final output is r0 = t0 + t1 + t2 + t3.
const unsigned char g_strGlowPShaderARB[] =
{
"!!ARBfp1.0\
\
# Input.\n\
ATTRIB iColor = fragment.color.primary;\
\
# Output.\n\
OUTPUT oColor = result.color;\
\
# Constants.\n\
PARAM Weight = program.env[0];\
TEMP t0;\
TEMP t1;\
TEMP t2;\
TEMP t3;\
TEMP r0;\
\
# Main.\n\
TEX t0, fragment.texcoord[0], texture[0], RECT;\
TEX t1, fragment.texcoord[1], texture[1], RECT;\
TEX t2, fragment.texcoord[2], texture[2], RECT;\
TEX t3, fragment.texcoord[3], texture[3], RECT;\
\
MUL r0, t0, Weight;\
MAD r0, t1, Weight, r0;\
MAD r0, t2, Weight, r0;\
MAD r0, t3, Weight, r0;\
\
MOV oColor, r0;\
\
END"
};
/***********************************************************************************************************/
#ifdef USE_NEW_SHADER_HASH
static char shader_token[MAX_TOKEN_CHARS];
static char shader_parsename[MAX_TOKEN_CHARS];
static int shader_lines;
static char *Shader_ParseExt( const char **data_p, qboolean allowLineBreaks );
void Shader_BeginParseSession( const char *name )
{
shader_lines = 0;
Com_sprintf(shader_parsename, sizeof(shader_parsename), "%s", name);
}
int Shader_GetCurrentParseLine( void )
{
return shader_lines;
}
static void Shader_ParseWarning( char *format, ... )
{
va_list argptr;
static char string[4096];
va_start (argptr, format);
Q_vsnprintf (string, sizeof(string), format, argptr);
va_end (argptr);
ri->Printf( PRINT_ALL, "WARNING: %s, line %d: %s\n", shader_parsename, shader_lines, string);
}
static int Shader_CompressBracedSection( char **data_p, char **name, char **text, int *nameLength, int *textLength ) { // drakkar - optimized sub-parse function
char *in, *out;
int depth, c;
if( !*data_p ) return -1;
*name = NULL;
*text = NULL;
*nameLength = 0;
*textLength = 0;
depth = 0;
in = out = *data_p;
if( !*in ) return 0;
while( (c = *(unsigned char* /*eurofix*/)in++) != '\0' )
{
if( c <= '/' || c >= '{' ) // skip lot of conditions if c is regular char
{
// whitespace or newline
if( c <= ' ' )
{
if( out > *data_p && out[-1] <= ' ' ) {
out--;
*out = ( c == '\n' ? '\n' : *out );
}
else {
*out = ( c == '\n' ? '\n' : ' ' );
}
while( *in && *(unsigned char* /*eurofix*/)in <= ' ' ) {
if( *in++ == '\n' ) {
shader_lines++;
*out = '\n';
}
}
out++;
continue;
}
// skip comments
if( c == '/' ) {
// double slash comments
if( *in == '/' ) {
in++;
while( *in && *in != '\n' ) in++; // ignore until newline
if( out > *data_p && out[-1] <= ' ' ) out--;
if( *in ) in++;
shader_lines++;
*out++ = '\n';
}
// multiline /* */ comments
else if( *in == '*' ) {
in++;
while( *in && ( *in != '*' || in[1] != '/' ) ) { // ignore until comment close
if( *in++ == '\n' ) {
shader_lines++;
}
}
if( *in ) in += 2;
}
// not comment
else {
*out++ = '/';
}
continue;
}
// handle quoted strings
if( c == '"' )
{
*out++ = '"';
while( *in && *in != '"' ) *out++ = *in++;
*out++ = '"';
in++;
continue;
}
// brace matching
if( c == '{' || c == '}' ) {
if( c == '{' && !*name ) {
*name = *data_p;
if( *(*name) <= ' ' ) (*name)++;
*nameLength = out - *name;
if( (*name)[*nameLength-1] <= ' ' ) (*nameLength)--;
*text = out;
}
if( out > *data_p && out[-1] > ' ' && out+1 < in ) *out++ = ' ';
*out++ = c;
if( out+1 < in ) *out++ = ' ';
depth += ( c == '{' ? +1 : -1 );
if( depth <= 0 ) break;
continue;
}
}
// parse a regular word
while( c ) {
*out++ = c;
c = *in;
// end of regular chars ?
if( c <= '/' ) break;
if( c >= '{' ) break;
in++;
}
}
if( depth ) {
Shader_ParseWarning( "Unmatched braces in shader text" );
}
if( !c ) in--;
if( *text && *(*text) <= ' ' ) (*text)++; // remove begining white char
if( out > *data_p && out[-1] <= ' ' ) out--; // remove ending white char
if( *text ) *textLength = out - *text; // compressed text length
c = out - *data_p; // uncompressed chars parsed
*data_p = in;
return c;
}
static char *Shader_ParseExt( const char **data_p, qboolean allowLineBreaks ) // drakkar - new Shader_ParseExt(), optimized version
{
const char *in;
char *out;
shader_token[0] = 0;
if( !*data_p ) return shader_token;
in = *data_p;
out = shader_token;
if( *in && *in <= '/' ) // skip lot of conditions if *in is regular char
{
// ignore while whitespace or newline
while( *in && *in <= ' ' ) {
if( *in++ == '\n') {
shader_lines++;
if( !allowLineBreaks ) {
*data_p = in;
return shader_token;
}
}
}
// skip comments
while( *in == '/' ) {
in++;
if( *in == '/' ) {
in++;
while( *in && *in != '\n' ) in++; // ignore until newline
if( *in ) in++;
}
else if( *in == '*' ) {
in++;
while( *in && ( *in != '*' || in[1] != '/' ) ) in++; // ignore until comment close
if( *in ) in += 2;
}
else {
*out++ = '/';
break;
}
while( *in && *in <= ' ' ) {
if( *in++ == '\n') {
shader_lines++;
if( !allowLineBreaks ) {
*data_p = in;
return shader_token;
}
}
}
}
// handle quoted strings
if( *in == '"' ) {
in++;
while( *in && *in != '"' ) {
if( (out-shader_token) >= MAX_TOKEN_CHARS-2 ) {
Shader_ParseWarning( "Token exceeded %d chars, truncated.", MAX_TOKEN_CHARS-2 );
break;
}
*out++ = *in++;
}
if( *in ) in++;
*out = '\0';
*data_p = in;
return shader_token;
}
}
// parse a regular word
while( *in > ' ' ) {
if( (out-shader_token) >= MAX_TOKEN_CHARS-1 ) {
Shader_ParseWarning( "Token exceeded %d chars, truncated.", MAX_TOKEN_CHARS-2 );
break;
}
*out++ = *in++;
}
*out = '\0';
*data_p = ( *in ? in : NULL ); // next text point or NULL if end of text reached
return shader_token;
}
/*
=================
Shader_SkipRestOfLine
=================
*/
static void Shader_SkipRestOfLine ( const char **data ) {
const char *p;
int c;
p = *data;
while ( (c = *p++) != 0 ) {
if ( c == '\n' ) {
shader_lines++;
break;
}
}
*data = p;
}
#else
#define Shader_BeginParseSession COM_BeginParseSession
#define Shader_GetCurrentParseLine COM_GetCurrentParseLine
#define Shader_Parse COM_Parse
#define Shader_Shader_SkipWhitespace SkipWhitespace
#define Shader_Compress COM_Comress
#define Shader_ParseExt COM_ParseExt
#define Shader_SkipBracedSection SkipBracedSection
#define Shader_SkipRestOfLine SkipRestOfLine
#endif
#ifndef USE_NEW_SHADER_HASH
static char *s_shaderText;
#endif
// the shader is parsed into these global variables, then copied into
// dynamically allocated memory if it is valid.
static shaderStage_t stages[MAX_SHADER_STAGES];
static shader_t shader;
static texModInfo_t texMods[MAX_SHADER_STAGES][TR_MAX_TEXMODS];
static qboolean deferLoad;
#define FILE_HASH_SIZE 1024
static shader_t* hashTable[FILE_HASH_SIZE];
#ifdef USE_NEW_SHADER_HASH
// drakkar - dynamic shaderTextHashTable
#define MAX_SHADERNAME_LENGTH 127
#define MAX_SHADERTEXT_HASH 4096 // from 2048 to 4096
typedef struct shaderText_s { // 8 bytes + strlen(text)+1
struct shaderText_s *next; // linked list hashtable
char *name; // shader name
char text[1]; // shader text
} shaderText_t;
static shaderText_t *shaderTextHashTable[MAX_SHADERTEXT_HASH];
static int fileShaderCount; // total .shader files found
static int shaderCount; // total shaders parsed
// !drakkar
#else
#define MAX_SHADERTEXT_HASH 2048
static char **shaderTextHashTable[MAX_SHADERTEXT_HASH] = { 0 };
#endif
void KillTheShaderHashTable(void)
{
memset(shaderTextHashTable, 0, sizeof(shaderTextHashTable));
}
qboolean ShaderHashTableExists(void)
{
if (shaderTextHashTable[0])
{
return qtrue;
}
return qfalse;
}
const int lightmapsNone[MAXLIGHTMAPS] =
{
LIGHTMAP_NONE,
LIGHTMAP_NONE,
LIGHTMAP_NONE,
LIGHTMAP_NONE
};
const int lightmaps2d[MAXLIGHTMAPS] =
{
LIGHTMAP_2D,
LIGHTMAP_2D,
LIGHTMAP_2D,
LIGHTMAP_2D
};
const int lightmapsVertex[MAXLIGHTMAPS] =
{
LIGHTMAP_BY_VERTEX,
LIGHTMAP_BY_VERTEX,
LIGHTMAP_BY_VERTEX,
LIGHTMAP_BY_VERTEX
};
const int lightmapsFullBright[MAXLIGHTMAPS] =
{
LIGHTMAP_WHITEIMAGE,
LIGHTMAP_WHITEIMAGE,
LIGHTMAP_WHITEIMAGE,
LIGHTMAP_WHITEIMAGE
};
const byte stylesDefault[MAXLIGHTMAPS] =
{
LS_NORMAL,
LS_LSNONE,
LS_LSNONE,
LS_LSNONE
};
/*
Ghoul2 Insert Start
*/
/*
===============
R_CreateExtendedName
Creates a unique shader name taking into account lightstyles
===============
*/
//rwwRMG - added
void R_CreateExtendedName(char *extendedName, int extendedNameSize, const char *name, const int *lightmapIndex, const byte *styles)
{
int i;
// Set the basename
COM_StripExtension( name, extendedName, extendedNameSize );
// Add in lightmaps
if(lightmapIndex && styles)
{
if(lightmapIndex == lightmapsNone)
{
strcat(extendedName, "_nolightmap");
}
else if(lightmapIndex == lightmaps2d)
{
strcat(extendedName, "_2d");
}
else if(lightmapIndex == lightmapsVertex)
{
strcat(extendedName, "_vertex");
}
else if(lightmapIndex == lightmapsFullBright)
{
strcat(extendedName, "_fullbright");
}
else
{
for(i = 0; (i < 4) && (styles[i] != 255); i++)
{
switch(lightmapIndex[i])
{
case LIGHTMAP_NONE:
strcat(extendedName, va("_style(%d,none)", styles[i]));
break;
case LIGHTMAP_2D:
strcat(extendedName, va("_style(%d,2d)", styles[i]));
break;
case LIGHTMAP_BY_VERTEX:
strcat(extendedName, va("_style(%d,vert)", styles[i]));
break;
case LIGHTMAP_WHITEIMAGE:
strcat(extendedName, va("_style(%d,fb)", styles[i]));
break;
default:
strcat(extendedName, va("_style(%d,%d)", styles[i], lightmapIndex[i]));
break;
}
}
}
}
}
/*
Ghoul2 Insert End
*/
static void ClearGlobalShader(void)
{
int i;
memset( &shader, 0, sizeof( shader ) );
memset( &stages, 0, sizeof( stages ) );
for ( i = 0 ; i < MAX_SHADER_STAGES ; i++ ) {
stages[i].bundle[0].texMods = texMods[i];
stages[i].mGLFogColorOverride = GLFOGOVERRIDE_NONE;
}
shader.contentFlags = CONTENTS_SOLID | CONTENTS_OPAQUE;
}
/*
================
return a hash value for the filename
================
*/
static long generateHashValue( const char *fname, const int size ) {
int i;
long hash;
char letter;
hash = 0;
i = 0;
while (fname[i] != '\0') {
letter = tolower((unsigned char)fname[i]);
if (letter =='.') break; // don't include extension
if (letter =='\\') letter = '/'; // damn path names
if (letter == PATH_SEP) letter = '/'; // damn path names
hash+=(long)(letter)*(i+119);
i++;
}
hash = (hash ^ (hash >> 10) ^ (hash >> 20));
hash &= (size-1);
return hash;
}
void R_RemapShader(const char *shaderName, const char *newShaderName, const char *timeOffset) {
char strippedName[MAX_QPATH];
int hash;
shader_t *sh, *sh2;
qhandle_t h;
sh = R_FindShaderByName( shaderName );
if (sh == NULL || sh == tr.defaultShader) {
h = RE_RegisterShaderLightMap(shaderName, lightmapsNone, stylesDefault);
sh = R_GetShaderByHandle(h);
}
if (sh == NULL || sh == tr.defaultShader) {
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: R_RemapShader: shader %s not found\n", shaderName );
return;
}
sh2 = R_FindShaderByName( newShaderName );
if (sh2 == NULL || sh2 == tr.defaultShader) {
h = RE_RegisterShaderLightMap(newShaderName, lightmapsNone, stylesDefault);
sh2 = R_GetShaderByHandle(h);
}
if (sh2 == NULL || sh2 == tr.defaultShader) {
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: R_RemapShader: new shader %s not found\n", newShaderName );
return;
}
// remap all the shaders with the given name
// even tho they might have different lightmaps
COM_StripExtension( shaderName, strippedName, sizeof( strippedName ) );
hash = generateHashValue(strippedName, FILE_HASH_SIZE);
for (sh = hashTable[hash]; sh; sh = sh->next) {
if (Q_stricmp(sh->name, strippedName) == 0) {
if (sh != sh2) {
sh->remappedShader = sh2;
} else {
sh->remappedShader = NULL;
}
}
}
if (timeOffset) {
sh2->timeOffset = atof(timeOffset);
}
}
/*
===============
ParseVector
===============
*/
qboolean ParseVector( const char **text, int count, float *v ) {
char *token;
int i;
// FIXME: spaces are currently required after parens, should change parseext...
token = Shader_ParseExt( text, qfalse );
if ( strcmp( token, "(" ) ) {
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing parenthesis in shader '%s'\n", shader.name );
return qfalse;
}
for ( i = 0 ; i < count ; i++ ) {
token = Shader_ParseExt( text, qfalse );
if ( !token[0] ) {
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing vector element in shader '%s'\n", shader.name );
return qfalse;
}
v[i] = atof( token );
}
token = Shader_ParseExt( text, qfalse );
if ( strcmp( token, ")" ) ) {
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing parenthesis in shader '%s'\n", shader.name );
return qfalse;
}
return qtrue;
}
/*
===============
NameToAFunc
===============
*/
static unsigned NameToAFunc( const char *funcname )
{
if ( !Q_stricmp( funcname, "GT0" ) )
{
return GLS_ATEST_GT_0;
}
else if ( !Q_stricmp( funcname, "LT128" ) )
{
return GLS_ATEST_LT_80;
}
else if ( !Q_stricmp( funcname, "GE128" ) )
{
return GLS_ATEST_GE_80;
}
else if ( !Q_stricmp( funcname, "GE192" ) )
{
return GLS_ATEST_GE_C0;
}
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: invalid alphaFunc name '%s' in shader '%s'\n", funcname, shader.name );
return 0;
}
/*
===============
NameToSrcBlendMode
===============
*/
static int NameToSrcBlendMode( const char *name )
{
if ( !Q_stricmp( name, "GL_ONE" ) )
{
return GLS_SRCBLEND_ONE;
}
else if ( !Q_stricmp( name, "GL_ZERO" ) )
{
return GLS_SRCBLEND_ZERO;
}
else if ( !Q_stricmp( name, "GL_DST_COLOR" ) )
{
return GLS_SRCBLEND_DST_COLOR;
}
else if ( !Q_stricmp( name, "GL_ONE_MINUS_DST_COLOR" ) )
{
return GLS_SRCBLEND_ONE_MINUS_DST_COLOR;
}
else if ( !Q_stricmp( name, "GL_SRC_ALPHA" ) )
{
return GLS_SRCBLEND_SRC_ALPHA;
}
else if ( !Q_stricmp( name, "GL_ONE_MINUS_SRC_ALPHA" ) )
{
return GLS_SRCBLEND_ONE_MINUS_SRC_ALPHA;
}
else if ( !Q_stricmp( name, "GL_DST_ALPHA" ) )
{
return GLS_SRCBLEND_DST_ALPHA;
}
else if ( !Q_stricmp( name, "GL_ONE_MINUS_DST_ALPHA" ) )
{
return GLS_SRCBLEND_ONE_MINUS_DST_ALPHA;
}
else if ( !Q_stricmp( name, "GL_SRC_ALPHA_SATURATE" ) )
{
return GLS_SRCBLEND_ALPHA_SATURATE;
}
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: unknown blend mode '%s' in shader '%s', substituting GL_ONE\n", name, shader.name );
return GLS_SRCBLEND_ONE;
}
/*
===============
NameToDstBlendMode
===============
*/
static int NameToDstBlendMode( const char *name )
{
if ( !Q_stricmp( name, "GL_ONE" ) )
{
return GLS_DSTBLEND_ONE;
}
else if ( !Q_stricmp( name, "GL_ZERO" ) )
{
return GLS_DSTBLEND_ZERO;
}
else if ( !Q_stricmp( name, "GL_SRC_ALPHA" ) )
{
return GLS_DSTBLEND_SRC_ALPHA;
}
else if ( !Q_stricmp( name, "GL_ONE_MINUS_SRC_ALPHA" ) )
{
return GLS_DSTBLEND_ONE_MINUS_SRC_ALPHA;
}
else if ( !Q_stricmp( name, "GL_DST_ALPHA" ) )
{
return GLS_DSTBLEND_DST_ALPHA;
}
else if ( !Q_stricmp( name, "GL_ONE_MINUS_DST_ALPHA" ) )
{
return GLS_DSTBLEND_ONE_MINUS_DST_ALPHA;
}
else if ( !Q_stricmp( name, "GL_SRC_COLOR" ) )
{
return GLS_DSTBLEND_SRC_COLOR;
}
else if ( !Q_stricmp( name, "GL_ONE_MINUS_SRC_COLOR" ) )
{
return GLS_DSTBLEND_ONE_MINUS_SRC_COLOR;
}
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: unknown blend mode '%s' in shader '%s', substituting GL_ONE\n", name, shader.name );
return GLS_DSTBLEND_ONE;
}
/*
===============
NameToGenFunc
===============
*/
static genFunc_t NameToGenFunc( const char *funcname )
{
if ( !Q_stricmp( funcname, "sin" ) )
{
return GF_SIN;
}
else if ( !Q_stricmp( funcname, "square" ) )
{
return GF_SQUARE;
}
else if ( !Q_stricmp( funcname, "triangle" ) )
{
return GF_TRIANGLE;
}
else if ( !Q_stricmp( funcname, "sawtooth" ) )
{
return GF_SAWTOOTH;
}
else if ( !Q_stricmp( funcname, "inversesawtooth" ) )
{
return GF_INVERSE_SAWTOOTH;
}
else if ( !Q_stricmp( funcname, "noise" ) )
{
return GF_NOISE;
}
else if ( !Q_stricmp( funcname, "random" ) )
{
return GF_RAND;
}
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: invalid genfunc name '%s' in shader '%s'\n", funcname, shader.name );
return GF_SIN;
}
/*
===================
ParseWaveForm
===================
*/
static void ParseWaveForm( const char **text, waveForm_t *wave )
{
char *token;
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing waveform parm in shader '%s'\n", shader.name );
return;
}
wave->func = NameToGenFunc( token );
// BASE, AMP, PHASE, FREQ
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing waveform parm in shader '%s'\n", shader.name );
return;
}
wave->base = atof( token );
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing waveform parm in shader '%s'\n", shader.name );
return;
}
wave->amplitude = atof( token );
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing waveform parm in shader '%s'\n", shader.name );
return;
}
wave->phase = atof( token );
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing waveform parm in shader '%s'\n", shader.name );
return;
}
wave->frequency = atof( token );
}
/*
===================
ParseTexMod
===================
*/
static void ParseTexMod( const char *_text, shaderStage_t *stage )
{
const char *token;
const char **text = &_text;
texModInfo_t *tmi;
if ( stage->bundle[0].numTexMods == TR_MAX_TEXMODS ) {
Com_Error( ERR_DROP, "ERROR: too many tcMod stages in shader '%s'\n", shader.name );
return;
}
tmi = &stage->bundle[0].texMods[stage->bundle[0].numTexMods];
stage->bundle[0].numTexMods++;
token = Shader_ParseExt( text, qfalse );
//
// turb
//
if ( !Q_stricmp( token, "turb" ) )
{
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing tcMod turb parms in shader '%s'\n", shader.name );
return;
}
tmi->wave.base = atof( token );
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing tcMod turb in shader '%s'\n", shader.name );
return;
}
tmi->wave.amplitude = atof( token );
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing tcMod turb in shader '%s'\n", shader.name );
return;
}
tmi->wave.phase = atof( token );
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing tcMod turb in shader '%s'\n", shader.name );
return;
}
tmi->wave.frequency = atof( token );
tmi->type = TMOD_TURBULENT;
}
//
// scale
//
else if ( !Q_stricmp( token, "scale" ) )
{
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing scale parms in shader '%s'\n", shader.name );
return;
}
tmi->translate[0] = atof( token ); //scale unioned
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing scale parms in shader '%s'\n", shader.name );
return;
}
tmi->translate[1] = atof( token ); //scale unioned
tmi->type = TMOD_SCALE;
}
//
// scroll
//
else if ( !Q_stricmp( token, "scroll" ) )
{
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing scale scroll parms in shader '%s'\n", shader.name );
return;
}
tmi->translate[0] = atof( token ); //scroll unioned
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing scale scroll parms in shader '%s'\n", shader.name );
return;
}
tmi->translate[1] = atof( token ); //scroll unioned
tmi->type = TMOD_SCROLL;
}
//
// stretch
//
else if ( !Q_stricmp( token, "stretch" ) )
{
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing stretch parms in shader '%s'\n", shader.name );
return;
}
tmi->wave.func = NameToGenFunc( token );
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing stretch parms in shader '%s'\n", shader.name );
return;
}
tmi->wave.base = atof( token );
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing stretch parms in shader '%s'\n", shader.name );
return;
}
tmi->wave.amplitude = atof( token );
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing stretch parms in shader '%s'\n", shader.name );
return;
}
tmi->wave.phase = atof( token );
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing stretch parms in shader '%s'\n", shader.name );
return;
}
tmi->wave.frequency = atof( token );
tmi->type = TMOD_STRETCH;
}
//
// transform
//
else if ( !Q_stricmp( token, "transform" ) )
{
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing transform parms in shader '%s'\n", shader.name );
return;
}
tmi->matrix[0][0] = atof( token );
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing transform parms in shader '%s'\n", shader.name );
return;
}
tmi->matrix[0][1] = atof( token );
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing transform parms in shader '%s'\n", shader.name );
return;
}
tmi->matrix[1][0] = atof( token );
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing transform parms in shader '%s'\n", shader.name );
return;
}
tmi->matrix[1][1] = atof( token );
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing transform parms in shader '%s'\n", shader.name );
return;
}
tmi->translate[0] = atof( token );
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing transform parms in shader '%s'\n", shader.name );
return;
}
tmi->translate[1] = atof( token );
tmi->type = TMOD_TRANSFORM;
}
//
// rotate
//
else if ( !Q_stricmp( token, "rotate" ) )
{
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing tcMod rotate parms in shader '%s'\n", shader.name );
return;
}
tmi->translate[0]= atof( token ); //rotateSpeed unioned
tmi->type = TMOD_ROTATE;
}
//
// entityTranslate
//
else if ( !Q_stricmp( token, "entityTranslate" ) )
{
tmi->type = TMOD_ENTITY_TRANSLATE;
}
else
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: unknown tcMod '%s' in shader '%s'\n", token, shader.name );
}
}
/*
/////===== Part of the VERTIGON system =====/////
===================
ParseSurfaceSprites
===================
*/
// surfaceSprites <type> <width> <height> <density> <fadedist>
//
// NOTE: This parsing function used to be 12 pages long and very complex. The new version of surfacesprites
// utilizes optional parameters parsed in ParseSurfaceSpriteOptional.
static void ParseSurfaceSprites( const char *_text, shaderStage_t *stage )
{
const char *token;
const char **text = &_text;
float width, height, density, fadedist;
int sstype=SURFSPRITE_NONE;
//
// spritetype
//
token = Shader_ParseExt( text, qfalse );
if (token[0]==0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing surfaceSprites params in shader '%s'\n", shader.name );
return;
}
if (!Q_stricmp(token, "vertical"))
{
sstype = SURFSPRITE_VERTICAL;
}
else if (!Q_stricmp(token, "oriented"))
{
sstype = SURFSPRITE_ORIENTED;
}
else if (!Q_stricmp(token, "effect"))
{
sstype = SURFSPRITE_EFFECT;
}
else if (!Q_stricmp(token, "flattened"))
{
sstype = SURFSPRITE_FLATTENED;
}
else
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: invalid type in shader '%s'\n", shader.name );
return;
}
//
// width
//
token = Shader_ParseExt( text, qfalse );
if (token[0]==0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing surfaceSprites params in shader '%s'\n", shader.name );
return;
}
width=atof(token);
if (width <= 0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: invalid width in shader '%s'\n", shader.name );
return;
}
//
// height
//
token = Shader_ParseExt( text, qfalse );
if (token[0]==0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing surfaceSprites params in shader '%s'\n", shader.name );
return;
}
height=atof(token);
if (height <= 0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: invalid height in shader '%s'\n", shader.name );
return;
}
//
// density
//
token = Shader_ParseExt( text, qfalse );
if (token[0]==0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing surfaceSprites params in shader '%s'\n", shader.name );
return;
}
density=atof(token);
if (density <= 0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: invalid density in shader '%s'\n", shader.name );
return;
}
//
// fadedist
//
token = Shader_ParseExt( text, qfalse );
if (token[0]==0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing surfaceSprites params in shader '%s'\n", shader.name );
return;
}
fadedist=atof(token);
if (fadedist < 32)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: invalid fadedist (%f < 32) in shader '%s'\n", fadedist, shader.name );
return;
}
if (!stage->ss)
{
stage->ss = (surfaceSprite_t *)Hunk_Alloc( sizeof( surfaceSprite_t ), h_low );
}
// These are all set by the command lines.
stage->ss->surfaceSpriteType = sstype;
stage->ss->width = width;
stage->ss->height = height;
stage->ss->density = density;
stage->ss->fadeDist = fadedist;
// These are defaults that can be overwritten.
stage->ss->fadeMax = fadedist*1.33;
stage->ss->fadeScale = 0.0;
stage->ss->wind = 0.0;
stage->ss->windIdle = 0.0;
stage->ss->variance[0] = 0.0;
stage->ss->variance[1] = 0.0;
stage->ss->facing = SURFSPRITE_FACING_NORMAL;
// A vertical parameter that needs a default regardless
stage->ss->vertSkew = 0.0f;
// These are effect parameters that need defaults nonetheless.
stage->ss->fxDuration = 1000; // 1 second
stage->ss->fxGrow[0] = 0.0;
stage->ss->fxGrow[1] = 0.0;
stage->ss->fxAlphaStart = 1.0;
stage->ss->fxAlphaEnd = 0.0;
}
/*
/////===== Part of the VERTIGON system =====/////
===========================
ParseSurfaceSpritesOptional
===========================
*/
//
// ssFademax <fademax>
// ssFadescale <fadescale>
// ssVariance <varwidth> <varheight>
// ssHangdown
// ssAnyangle
// ssFaceup
// ssWind <wind>
// ssWindIdle <windidle>
// ssVertSkew <skew>
// ssFXDuration <duration>
// ssFXGrow <growwidth> <growheight>
// ssFXAlphaRange <alphastart> <startend>
// ssFXWeather
//
// Optional parameters that will override the defaults set in the surfacesprites command above.
//
static void ParseSurfaceSpritesOptional( const char *param, const char *_text, shaderStage_t *stage )
{
const char *token;
const char **text = &_text;
float value;
if (!stage->ss)
{
stage->ss = (surfaceSprite_t *)Hunk_Alloc( sizeof( surfaceSprite_t ), h_low );
}
//
// fademax
//
if (!Q_stricmp(param, "ssFademax"))
{
token = Shader_ParseExt( text, qfalse);
if (token[0]==0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing surfacesprite fademax in shader '%s'\n", shader.name );
return;
}
value = atof(token);
if (value <= stage->ss->fadeDist)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: invalid surfacesprite fademax (%.2f <= fadeDist(%.2f)) in shader '%s'\n", value, stage->ss->fadeDist, shader.name );
return;
}
stage->ss->fadeMax=value;
return;
}
//
// fadescale
//
if (!Q_stricmp(param, "ssFadescale"))
{
token = Shader_ParseExt( text, qfalse);
if (token[0]==0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing surfacesprite fadescale in shader '%s'\n", shader.name );
return;
}
value = atof(token);
stage->ss->fadeScale=value;
return;
}
//
// variance
//
if (!Q_stricmp(param, "ssVariance"))
{
token = Shader_ParseExt( text, qfalse);
if (token[0]==0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing surfacesprite variance width in shader '%s'\n", shader.name );
return;
}
value = atof(token);
if (value < 0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: invalid surfacesprite variance width in shader '%s'\n", shader.name );
return;
}
stage->ss->variance[0]=value;
token = Shader_ParseExt( text, qfalse);
if (token[0]==0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing surfacesprite variance height in shader '%s'\n", shader.name );
return;
}
value = atof(token);
if (value < 0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: invalid surfacesprite variance height in shader '%s'\n", shader.name );
return;
}
stage->ss->variance[1]=value;
return;
}
//
// hangdown
//
if (!Q_stricmp(param, "ssHangdown"))
{
if (stage->ss->facing != SURFSPRITE_FACING_NORMAL)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: Hangdown facing overrides previous facing in shader '%s'\n", shader.name );
return;
}
stage->ss->facing=SURFSPRITE_FACING_DOWN;
return;
}
//
// anyangle
//
if (!Q_stricmp(param, "ssAnyangle"))
{
if (stage->ss->facing != SURFSPRITE_FACING_NORMAL)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: Anyangle facing overrides previous facing in shader '%s'\n", shader.name );
return;
}
stage->ss->facing=SURFSPRITE_FACING_ANY;
return;
}
//
// faceup
//
if (!Q_stricmp(param, "ssFaceup"))
{
if (stage->ss->facing != SURFSPRITE_FACING_NORMAL)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: Faceup facing overrides previous facing in shader '%s'\n", shader.name );
return;
}
stage->ss->facing=SURFSPRITE_FACING_UP;
return;
}
//
// wind
//
if (!Q_stricmp(param, "ssWind"))
{
token = Shader_ParseExt( text, qfalse);
if (token[0]==0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing surfacesprite wind in shader '%s'\n", shader.name );
return;
}
value = atof(token);
if (value < 0.0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: invalid surfacesprite wind in shader '%s'\n", shader.name );
return;
}
stage->ss->wind=value;
if (stage->ss->windIdle <= 0)
{ // Also override the windidle, it usually is the same as wind
stage->ss->windIdle = value;
}
return;
}
//
// windidle
//
if (!Q_stricmp(param, "ssWindidle"))
{
token = Shader_ParseExt( text, qfalse);
if (token[0]==0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing surfacesprite windidle in shader '%s'\n", shader.name );
return;
}
value = atof(token);
if (value < 0.0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: invalid surfacesprite windidle in shader '%s'\n", shader.name );
return;
}
stage->ss->windIdle=value;
return;
}
//
// vertskew
//
if (!Q_stricmp(param, "ssVertskew"))
{
token = Shader_ParseExt( text, qfalse);
if (token[0]==0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing surfacesprite vertskew in shader '%s'\n", shader.name );
return;
}
value = atof(token);
if (value < 0.0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: invalid surfacesprite vertskew in shader '%s'\n", shader.name );
return;
}
stage->ss->vertSkew=value;
return;
}
//
// fxduration
//
if (!Q_stricmp(param, "ssFXDuration"))
{
token = Shader_ParseExt( text, qfalse);
if (token[0]==0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing surfacesprite duration in shader '%s'\n", shader.name );
return;
}
value = atof(token);
if (value <= 0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: invalid surfacesprite duration in shader '%s'\n", shader.name );
return;
}
stage->ss->fxDuration=value;
return;
}
//
// fxgrow
//
if (!Q_stricmp(param, "ssFXGrow"))
{
token = Shader_ParseExt( text, qfalse);
if (token[0]==0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing surfacesprite grow width in shader '%s'\n", shader.name );
return;
}
value = atof(token);
if (value < 0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: invalid surfacesprite grow width in shader '%s'\n", shader.name );
return;
}
stage->ss->fxGrow[0]=value;
token = Shader_ParseExt( text, qfalse);
if (token[0]==0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing surfacesprite grow height in shader '%s'\n", shader.name );
return;
}
value = atof(token);
if (value < 0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: invalid surfacesprite grow height in shader '%s'\n", shader.name );
return;
}
stage->ss->fxGrow[1]=value;
return;
}
//
// fxalpharange
//
if (!Q_stricmp(param, "ssFXAlphaRange"))
{
token = Shader_ParseExt( text, qfalse);
if (token[0]==0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing surfacesprite fxalpha start in shader '%s'\n", shader.name );
return;
}
value = atof(token);
if (value < 0 || value > 1.0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: invalid surfacesprite fxalpha start in shader '%s'\n", shader.name );
return;
}
stage->ss->fxAlphaStart=value;
token = Shader_ParseExt( text, qfalse);
if (token[0]==0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing surfacesprite fxalpha end in shader '%s'\n", shader.name );
return;
}
value = atof(token);
if (value < 0 || value > 1.0)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: invalid surfacesprite fxalpha end in shader '%s'\n", shader.name );
return;
}
stage->ss->fxAlphaEnd=value;
return;
}
//
// fxweather
//
if (!Q_stricmp(param, "ssFXWeather"))
{
if (stage->ss->surfaceSpriteType != SURFSPRITE_EFFECT)
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: weather applied to non-effect surfacesprite in shader '%s'\n", shader.name );
return;
}
stage->ss->surfaceSpriteType = SURFSPRITE_WEATHERFX;
return;
}
//
// invalid ss command.
//
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: invalid optional surfacesprite param '%s' in shader '%s'\n", param, shader.name );
return;
}
/*
===================
ParseStage
===================
*/
static qboolean ParseStage( shaderStage_t *stage, const char **text )
{
char *token;
int depthMaskBits = GLS_DEPTHMASK_TRUE, blendSrcBits = 0, blendDstBits = 0, atestBits = 0, depthFuncBits = 0;
qboolean depthMaskExplicit = qfalse;
stage->active = qtrue;
while ( 1 )
{
token = Shader_ParseExt( text, qtrue );
if ( !token[0] )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: no matching '}' found\n" );
return qfalse;
}
if ( token[0] == '}' )
{
break;
}
//
// map <name>
//
else if ( !Q_stricmp( token, "map" ) )
{
token = Shader_ParseExt( text, qfalse );
if ( !token[0] )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing parameter for 'map' keyword in shader '%s'\n", shader.name );
return qfalse;
}
if ( !Q_stricmp( token, "$whiteimage" ) )
{
stage->bundle[0].image = tr.whiteImage;
continue;
}
else if ( !Q_stricmp( token, "$lightmap" ) )
{
stage->bundle[0].isLightmap = qtrue;
if ( shader.lightmapIndex[0] < 0 || shader.lightmapIndex[0] >= tr.numLightmaps )
{
#ifndef FINAL_BUILD
ri->Printf( PRINT_ALL, S_COLOR_RED"Lightmap requested but none available for shader %s\n", shader.name);
#endif
stage->bundle[0].image = tr.whiteImage;
}
else
{
stage->bundle[0].image = tr.lightmaps[shader.lightmapIndex[0]];
}
continue;
}
else
{
stage->bundle[0].image = R_FindImageFile( token, (qboolean)!shader.noMipMaps, (qboolean)!shader.noPicMip, (qboolean)!shader.noTC, GL_REPEAT );
if ( !stage->bundle[0].image )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: R_FindImageFile could not find '%s' in shader '%s'\n", token, shader.name );
return qfalse;
}
}
}
//
// clampmap <name>
//
else if ( !Q_stricmp( token, "clampmap" ) )
{
token = Shader_ParseExt( text, qfalse );
if ( !token[0] )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing parameter for 'clampmap' keyword in shader '%s'\n", shader.name );
return qfalse;
}
stage->bundle[0].image = R_FindImageFile( token, (qboolean)!shader.noMipMaps, (qboolean)!shader.noPicMip, (qboolean)!shader.noTC, GL_CLAMP );
if ( !stage->bundle[0].image )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: R_FindImageFile could not find '%s' in shader '%s'\n", token, shader.name );
return qfalse;
}
}
//
// animMap <frequency> <image1> .... <imageN>
//
else if ( !Q_stricmp( token, "animMap" ) || !Q_stricmp( token, "clampanimMap" ) || !Q_stricmp( token, "oneshotanimMap" ))
{
#define MAX_IMAGE_ANIMATIONS 32
image_t *images[MAX_IMAGE_ANIMATIONS];
bool bClamp = !Q_stricmp( token, "clampanimMap" );
bool oneShot = !Q_stricmp( token, "oneshotanimMap" );
token = Shader_ParseExt( text, qfalse );
if ( !token[0] )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing parameter for '%s' keyword in shader '%s'\n", (bClamp ? "animMap":"clampanimMap"), shader.name );
return qfalse;
}
stage->bundle[0].imageAnimationSpeed = atof( token );
stage->bundle[0].oneShotAnimMap = oneShot;
// parse up to MAX_IMAGE_ANIMATIONS animations
while ( 1 ) {
int num;
token = Shader_ParseExt( text, qfalse );
if ( !token[0] ) {
break;
}
num = stage->bundle[0].numImageAnimations;
if ( num < MAX_IMAGE_ANIMATIONS ) {
images[num] = R_FindImageFile( token, (qboolean)!shader.noMipMaps, (qboolean)!shader.noPicMip, (qboolean)!shader.noTC, bClamp?GL_CLAMP:GL_REPEAT );
if ( !images[num] )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: R_FindImageFile could not find '%s' in shader '%s'\n", token, shader.name );
return qfalse;
}
stage->bundle[0].numImageAnimations++;
}
}
// Copy image ptrs into an array of ptrs
stage->bundle[0].image = (image_t*) Hunk_Alloc( stage->bundle[0].numImageAnimations * sizeof( image_t* ), h_low );
memcpy( stage->bundle[0].image, images, stage->bundle[0].numImageAnimations * sizeof( image_t* ) );
}
else if ( !Q_stricmp( token, "videoMap" ) )
{
token = Shader_ParseExt( text, qfalse );
if ( !token[0] )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing parameter for 'videoMap' keyword in shader '%s'\n", shader.name );
return qfalse;
}
stage->bundle[0].videoMapHandle = ri->CIN_PlayCinematic( token, 0, 0, 256, 256, (CIN_loop | CIN_silent | CIN_shader));
if (stage->bundle[0].videoMapHandle != -1) {
stage->bundle[0].isVideoMap = qtrue;
assert (stage->bundle[0].videoMapHandle<NUM_SCRATCH_IMAGES);
stage->bundle[0].image = tr.scratchImage[stage->bundle[0].videoMapHandle];
}
}
//
// alphafunc <func>
//
else if ( !Q_stricmp( token, "alphaFunc" ) )
{
token = Shader_ParseExt( text, qfalse );
if ( !token[0] )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing parameter for 'alphaFunc' keyword in shader '%s'\n", shader.name );
return qfalse;
}
atestBits = NameToAFunc( token );
}
//
// depthFunc <func>
//
else if ( !Q_stricmp( token, "depthfunc" ) )
{
token = Shader_ParseExt( text, qfalse );
if ( !token[0] )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing parameter for 'depthfunc' keyword in shader '%s'\n", shader.name );
return qfalse;
}
if ( !Q_stricmp( token, "lequal" ) )
{
depthFuncBits = 0;
}
else if ( !Q_stricmp( token, "equal" ) )
{
depthFuncBits = GLS_DEPTHFUNC_EQUAL;
}
else if ( !Q_stricmp( token, "disable" ) )
{
depthFuncBits = GLS_DEPTHTEST_DISABLE;
}
else
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: unknown depthfunc '%s' in shader '%s'\n", token, shader.name );
continue;
}
}
//
// detail
//
else if ( !Q_stricmp( token, "detail" ) )
{
stage->isDetail = qtrue;
}
//
// blendfunc <srcFactor> <dstFactor>
// or blendfunc <add|filter|blend>
//
else if ( !Q_stricmp( token, "blendfunc" ) )
{
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing parm for blendFunc in shader '%s'\n", shader.name );
continue;
}
// check for "simple" blends first
if ( !Q_stricmp( token, "add" ) ) {
blendSrcBits = GLS_SRCBLEND_ONE;
blendDstBits = GLS_DSTBLEND_ONE;
} else if ( !Q_stricmp( token, "filter" ) ) {
blendSrcBits = GLS_SRCBLEND_DST_COLOR;
blendDstBits = GLS_DSTBLEND_ZERO;
} else if ( !Q_stricmp( token, "blend" ) ) {
blendSrcBits = GLS_SRCBLEND_SRC_ALPHA;
blendDstBits = GLS_DSTBLEND_ONE_MINUS_SRC_ALPHA;
} else {
// complex double blends
blendSrcBits = NameToSrcBlendMode( token );
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing parm for blendFunc in shader '%s'\n", shader.name );
continue;
}
blendDstBits = NameToDstBlendMode( token );
}
// clear depth mask for blended surfaces
if ( !depthMaskExplicit )
{
depthMaskBits = 0;
}
}
//
// rgbGen
//
else if ( !Q_stricmp( token, "rgbGen" ) )
{
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing parameters for rgbGen in shader '%s'\n", shader.name );
continue;
}
if ( !Q_stricmp( token, "wave" ) )
{
ParseWaveForm( text, &stage->rgbWave );
stage->rgbGen = CGEN_WAVEFORM;
}
else if ( !Q_stricmp( token, "const" ) )
{
vec3_t color;
ParseVector( text, 3, color );
stage->constantColor[0] = 255 * color[0];
stage->constantColor[1] = 255 * color[1];
stage->constantColor[2] = 255 * color[2];
stage->rgbGen = CGEN_CONST;
}
else if ( !Q_stricmp( token, "identity" ) )
{
stage->rgbGen = CGEN_IDENTITY;
}
else if ( !Q_stricmp( token, "identityLighting" ) )
{
stage->rgbGen = CGEN_IDENTITY_LIGHTING;
}
else if ( !Q_stricmp( token, "entity" ) )
{
stage->rgbGen = CGEN_ENTITY;
}
else if ( !Q_stricmp( token, "oneMinusEntity" ) )
{
stage->rgbGen = CGEN_ONE_MINUS_ENTITY;
}
else if ( !Q_stricmp( token, "vertex" ) )
{
stage->rgbGen = CGEN_VERTEX;
if ( stage->alphaGen == 0 ) {
stage->alphaGen = AGEN_VERTEX;
}
}
else if ( !Q_stricmp( token, "exactVertex" ) )
{
stage->rgbGen = CGEN_EXACT_VERTEX;
}
else if ( !Q_stricmp( token, "lightingDiffuse" ) )
{
stage->rgbGen = CGEN_LIGHTING_DIFFUSE;
}
else if ( !Q_stricmp( token, "lightingDiffuseEntity" ) )
{
if (shader.lightmapIndex[0] != LIGHTMAP_NONE)
{
ri->Printf( PRINT_ALL, S_COLOR_RED "ERROR: rgbGen lightingDiffuseEntity used on a misc_model! in shader '%s'\n", shader.name );
}
stage->rgbGen = CGEN_LIGHTING_DIFFUSE_ENTITY;
}
else if ( !Q_stricmp( token, "oneMinusVertex" ) )
{
stage->rgbGen = CGEN_ONE_MINUS_VERTEX;
}
else
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: unknown rgbGen parameter '%s' in shader '%s'\n", token, shader.name );
continue;
}
}
//
// alphaGen
//
else if ( !Q_stricmp( token, "alphaGen" ) )
{
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing parameters for alphaGen in shader '%s'\n", shader.name );
continue;
}
if ( !Q_stricmp( token, "wave" ) )
{
ParseWaveForm( text, &stage->alphaWave );
stage->alphaGen = AGEN_WAVEFORM;
}
else if ( !Q_stricmp( token, "const" ) )
{
token = Shader_ParseExt( text, qfalse );
stage->constantColor[3] = 255 * atof( token );
stage->alphaGen = AGEN_CONST;
}
else if ( !Q_stricmp( token, "identity" ) )
{
stage->alphaGen = AGEN_IDENTITY;
}
else if ( !Q_stricmp( token, "entity" ) )
{
stage->alphaGen = AGEN_ENTITY;
}
else if ( !Q_stricmp( token, "oneMinusEntity" ) )
{
stage->alphaGen = AGEN_ONE_MINUS_ENTITY;
}
else if ( !Q_stricmp( token, "vertex" ) )
{
stage->alphaGen = AGEN_VERTEX;
}
else if ( !Q_stricmp( token, "lightingSpecular" ) )
{
stage->alphaGen = AGEN_LIGHTING_SPECULAR;
}
else if ( !Q_stricmp( token, "oneMinusVertex" ) )
{
stage->alphaGen = AGEN_ONE_MINUS_VERTEX;
}
else if ( !Q_stricmp( token, "dot" ) )
{
stage->alphaGen = AGEN_DOT;
}
else if ( !Q_stricmp( token, "oneMinusDot" ) )
{
stage->alphaGen = AGEN_ONE_MINUS_DOT;
}
else if ( !Q_stricmp( token, "portal" ) )
{
stage->alphaGen = AGEN_PORTAL;
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
shader.portalRange = 256;
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing range parameter for alphaGen portal in shader '%s', defaulting to 256\n", shader.name );
}
else
{
shader.portalRange = atof( token );
}
}
else
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: unknown alphaGen parameter '%s' in shader '%s'\n", token, shader.name );
continue;
}
}
//
// tcGen <function>
//
else if ( !Q_stricmp(token, "texgen") || !Q_stricmp( token, "tcGen" ) )
{
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing texgen parm in shader '%s'\n", shader.name );
continue;
}
if ( !Q_stricmp( token, "environment" ) )
{
stage->bundle[0].tcGen = TCGEN_ENVIRONMENT_MAPPED;
}
else if ( !Q_stricmp( token, "lightmap" ) )
{
stage->bundle[0].tcGen = TCGEN_LIGHTMAP;
}
else if ( !Q_stricmp( token, "texture" ) || !Q_stricmp( token, "base" ) )
{
stage->bundle[0].tcGen = TCGEN_TEXTURE;
}
else if ( !Q_stricmp( token, "vector" ) )
{
stage->bundle[0].tcGenVectors = ( vec3_t *) Hunk_Alloc( 2 * sizeof( vec3_t ), h_low );
ParseVector( text, 3, stage->bundle[0].tcGenVectors[0] );
ParseVector( text, 3, stage->bundle[0].tcGenVectors[1] );
stage->bundle[0].tcGen = TCGEN_VECTOR;
}
else
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: unknown texgen parm in shader '%s'\n", shader.name );
}
}
//
// tcMod <type> <...>
//
else if ( !Q_stricmp( token, "tcMod" ) )
{
char buffer[1024] = "";
while ( 1 )
{
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
break;
strcat( buffer, token );
strcat( buffer, " " );
}
ParseTexMod( buffer, stage );
continue;
}
//
// depthmask
//
else if ( !Q_stricmp( token, "depthwrite" ) )
{
depthMaskBits = GLS_DEPTHMASK_TRUE;
depthMaskExplicit = qtrue;
continue;
}
// If this stage has glow... GLOWXXX
else if ( Q_stricmp( token, "glow" ) == 0 )
{
stage->glow = true;
continue;
}
//
// surfaceSprites <type> ...
//
else if ( !Q_stricmp( token, "surfaceSprites" ) )
{
char buffer[1024] = "";
while ( 1 )
{
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
break;
strcat( buffer, token );
strcat( buffer, " " );
}
ParseSurfaceSprites( buffer, stage );
continue;
}
//
// ssFademax <fademax>
// ssFadescale <fadescale>
// ssVariance <varwidth> <varheight>
// ssHangdown
// ssAnyangle
// ssFaceup
// ssWind <wind>
// ssWindIdle <windidle>
// ssDuration <duration>
// ssGrow <growwidth> <growheight>
// ssWeather
//
else if (!Q_stricmpn(token, "ss", 2)) // <--- NOTE ONLY COMPARING FIRST TWO LETTERS
{
char buffer[1024] = "";
char param[128];
strcpy(param,token);
while ( 1 )
{
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
break;
strcat( buffer, token );
strcat( buffer, " " );
}
ParseSurfaceSpritesOptional( param, buffer, stage );
continue;
}
else
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: unknown parameter '%s' in shader '%s'\n", token, shader.name );
return qfalse;
}
}
//
// if cgen isn't explicitly specified, use either identity or identitylighting
//
if ( stage->rgbGen == CGEN_BAD ) {
if ( //blendSrcBits == 0 ||
blendSrcBits == GLS_SRCBLEND_ONE ||
blendSrcBits == GLS_SRCBLEND_SRC_ALPHA ) {
stage->rgbGen = CGEN_IDENTITY_LIGHTING;
} else {
stage->rgbGen = CGEN_IDENTITY;
}
}
//
// implicitly assume that a GL_ONE GL_ZERO blend mask disables blending
//
if ( ( blendSrcBits == GLS_SRCBLEND_ONE ) &&
( blendDstBits == GLS_DSTBLEND_ZERO ) )
{
blendDstBits = blendSrcBits = 0;
depthMaskBits = GLS_DEPTHMASK_TRUE;
}
// decide which agens we can skip
if ( stage->alphaGen == AGEN_IDENTITY ) {
if ( stage->rgbGen == CGEN_IDENTITY
|| stage->rgbGen == CGEN_LIGHTING_DIFFUSE ) {
stage->alphaGen = AGEN_SKIP;
}
}
//
// compute state bits
//
stage->stateBits = depthMaskBits |
blendSrcBits | blendDstBits |
atestBits |
depthFuncBits;
return qtrue;
}
/*
===============
ParseDeform
deformVertexes wave <spread> <waveform> <base> <amplitude> <phase> <frequency>
deformVertexes normal <frequency> <amplitude>
deformVertexes move <vector> <waveform> <base> <amplitude> <phase> <frequency>
deformVertexes bulge <bulgeWidth> <bulgeHeight> <bulgeSpeed>
deformVertexes projectionShadow
deformVertexes autoSprite
deformVertexes autoSprite2
deformVertexes text[0-7]
===============
*/
static void ParseDeform( const char **text ) {
char *token;
deformStage_t *ds;
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing deform parm in shader '%s'\n", shader.name );
return;
}
if ( shader.numDeforms == MAX_SHADER_DEFORMS ) {
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: MAX_SHADER_DEFORMS in '%s'\n", shader.name );
return;
}
shader.deforms[ shader.numDeforms ] = (deformStage_t *)Hunk_Alloc( sizeof( deformStage_t ), h_low );
ds = shader.deforms[ shader.numDeforms ];
shader.numDeforms++;
if ( !Q_stricmp( token, "projectionShadow" ) ) {
ds->deformation = DEFORM_PROJECTION_SHADOW;
return;
}
if ( !Q_stricmp( token, "autosprite" ) ) {
ds->deformation = DEFORM_AUTOSPRITE;
return;
}
if ( !Q_stricmp( token, "autosprite2" ) ) {
ds->deformation = DEFORM_AUTOSPRITE2;
return;
}
if ( !Q_stricmpn( token, "text", 4 ) ) {
int n;
n = token[4] - '0';
if ( n < 0 || n > 7 ) {
n = 0;
}
ds->deformation = (deform_t)(DEFORM_TEXT0 + n);
return;
}
if ( !Q_stricmp( token, "bulge" ) ) {
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing deformVertexes bulge parm in shader '%s'\n", shader.name );
return;
}
ds->bulgeWidth = atof( token );
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing deformVertexes bulge parm in shader '%s'\n", shader.name );
return;
}
ds->bulgeHeight = atof( token );
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing deformVertexes bulge parm in shader '%s'\n", shader.name );
return;
}
ds->bulgeSpeed = atof( token );
ds->deformation = DEFORM_BULGE;
return;
}
if ( !Q_stricmp( token, "wave" ) )
{
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing deformVertexes parm in shader '%s'\n", shader.name );
return;
}
if ( atof( token ) != 0 )
{
ds->deformationSpread = 1.0f / atof( token );
}
else
{
ds->deformationSpread = 100.0f;
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: illegal div value of 0 in deformVertexes command for shader '%s'\n", shader.name );
}
ParseWaveForm( text, &ds->deformationWave );
ds->deformation = DEFORM_WAVE;
return;
}
if ( !Q_stricmp( token, "normal" ) )
{
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing deformVertexes parm in shader '%s'\n", shader.name );
return;
}
ds->deformationWave.amplitude = atof( token );
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing deformVertexes parm in shader '%s'\n", shader.name );
return;
}
ds->deformationWave.frequency = atof( token );
ds->deformation = DEFORM_NORMALS;
return;
}
if ( !Q_stricmp( token, "move" ) ) {
int i;
for ( i = 0 ; i < 3 ; i++ ) {
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 ) {
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing deformVertexes parm in shader '%s'\n", shader.name );
return;
}
ds->moveVector[i] = atof( token );
}
ParseWaveForm( text, &ds->deformationWave );
ds->deformation = DEFORM_MOVE;
return;
}
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: unknown deformVertexes subtype '%s' found in shader '%s'\n", token, shader.name );
}
/*
===============
ParseSkyParms
skyParms <outerbox> <cloudheight> <innerbox>
===============
*/
static void ParseSkyParms( const char **text ) {
char *token;
const char *suf[6] = {"rt", "lf", "bk", "ft", "up", "dn"};
char pathname[MAX_QPATH];
int i;
shader.sky = (skyParms_t *)Hunk_Alloc( sizeof( skyParms_t ), h_low );
// outerbox
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 ) {
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: 'skyParms' missing parameter in shader '%s'\n", shader.name );
return;
}
if ( strcmp( token, "-" ) ) {
for (i=0 ; i<6 ; i++) {
Com_sprintf( pathname, sizeof(pathname), "%s_%s", token, suf[i] );
shader.sky->outerbox[i] = R_FindImageFile( ( char * ) pathname, qtrue, qtrue, (qboolean)!shader.noTC, GL_CLAMP );
if ( !shader.sky->outerbox[i] ) {
if (i) {
shader.sky->outerbox[i] = shader.sky->outerbox[i-1];//not found, so let's use the previous image
}else{
shader.sky->outerbox[i] = tr.defaultImage;
}
}
}
}
// cloudheight
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 ) {
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: 'skyParms' missing cloudheight in shader '%s'\n", shader.name );
return;
}
shader.sky->cloudHeight = atof( token );
if ( !shader.sky->cloudHeight ) {
shader.sky->cloudHeight = 512;
}
R_InitSkyTexCoords( shader.sky->cloudHeight );
// innerbox
token = Shader_ParseExt( text, qfalse );
if ( strcmp( token, "-" ) ) {
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: in shader '%s' 'skyParms', innerbox is not supported!", shader.name);
}
}
/*
=================
ParseSort
=================
*/
static void ParseSort( const char **text ) {
char *token;
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 ) {
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing sort parameter in shader '%s'\n", shader.name );
return;
}
if ( !Q_stricmp( token, "portal" ) ) {
shader.sort = SS_PORTAL;
} else if ( !Q_stricmp( token, "sky" ) ) {
shader.sort = SS_ENVIRONMENT;
} else if ( !Q_stricmp( token, "opaque" ) ) {
shader.sort = SS_OPAQUE;
} else if ( !Q_stricmp( token, "decal" ) ) {
shader.sort = SS_DECAL;
} else if ( !Q_stricmp( token, "seeThrough" ) ) {
shader.sort = SS_SEE_THROUGH;
} else if ( !Q_stricmp( token, "banner" ) ) {
shader.sort = SS_BANNER;
} else if ( !Q_stricmp( token, "additive" ) ) {
shader.sort = SS_BLEND1;
} else if ( !Q_stricmp( token, "nearest" ) ) {
shader.sort = SS_NEAREST;
} else if ( !Q_stricmp( token, "underwater" ) ) {
shader.sort = SS_UNDERWATER;
} else if ( !Q_stricmp( token, "inside" ) ) {
shader.sort = SS_INSIDE;
} else if ( !Q_stricmp( token, "mid_inside" ) ) {
shader.sort = SS_MID_INSIDE;
} else if ( !Q_stricmp( token, "middle" ) ) {
shader.sort = SS_MIDDLE;
} else if ( !Q_stricmp( token, "mid_outside" ) ) {
shader.sort = SS_MID_OUTSIDE;
} else if ( !Q_stricmp( token, "outside" ) ) {
shader.sort = SS_OUTSIDE;
}
else {
shader.sort = atof( token );
}
}
/*
=================
ParseMaterial
=================
*/
const char *materialNames[MATERIAL_LAST] =
{
MATERIALS
};
void ParseMaterial( const char **text )
{
char *token;
int i;
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing material in shader '%s'\n", shader.name );
return;
}
for(i = 0; i < MATERIAL_LAST; i++)
{
if ( !Q_stricmp( token, materialNames[i] ) )
{
shader.surfaceFlags |= i;
break;
}
}
}
// this table is also present in q3map
typedef struct infoParm_s {
const char *name;
uint32_t clearSolid, surfaceFlags, contents;
} infoParm_t;
infoParm_t infoParms[] = {
// Game content Flags
{ "nonsolid", ~CONTENTS_SOLID, SURF_NONE, CONTENTS_NONE }, // special hack to clear solid flag
{ "nonopaque", ~CONTENTS_OPAQUE, SURF_NONE, CONTENTS_NONE }, // special hack to clear opaque flag
{ "lava", ~CONTENTS_SOLID, SURF_NONE, CONTENTS_LAVA }, // very damaging
{ "slime", ~CONTENTS_SOLID, SURF_NONE, CONTENTS_SLIME }, // mildly damaging
{ "water", ~CONTENTS_SOLID, SURF_NONE, CONTENTS_WATER }, //
{ "fog", ~CONTENTS_SOLID, SURF_NONE, CONTENTS_FOG}, // carves surfaces entering
{ "shotclip", ~CONTENTS_SOLID, SURF_NONE, CONTENTS_SHOTCLIP }, // block shots, but not people
{ "playerclip", ~(CONTENTS_SOLID|CONTENTS_OPAQUE), SURF_NONE, CONTENTS_PLAYERCLIP }, // block only the player
{ "monsterclip", ~(CONTENTS_SOLID|CONTENTS_OPAQUE), SURF_NONE, CONTENTS_MONSTERCLIP }, //
{ "botclip", ~(CONTENTS_SOLID|CONTENTS_OPAQUE), SURF_NONE, CONTENTS_BOTCLIP }, // for bots
{ "trigger", ~(CONTENTS_SOLID|CONTENTS_OPAQUE), SURF_NONE, CONTENTS_TRIGGER }, //
{ "nodrop", ~(CONTENTS_SOLID|CONTENTS_OPAQUE), SURF_NONE, CONTENTS_NODROP }, // don't drop items or leave bodies (death fog, lava, etc)
{ "terrain", ~(CONTENTS_SOLID|CONTENTS_OPAQUE), SURF_NONE, CONTENTS_TERRAIN }, // use special terrain collsion
{ "ladder", ~(CONTENTS_SOLID|CONTENTS_OPAQUE), SURF_NONE, CONTENTS_LADDER }, // climb up in it like water
{ "abseil", ~(CONTENTS_SOLID|CONTENTS_OPAQUE), SURF_NONE, CONTENTS_ABSEIL }, // can abseil down this brush
{ "outside", ~(CONTENTS_SOLID|CONTENTS_OPAQUE), SURF_NONE, CONTENTS_OUTSIDE }, // volume is considered to be in the outside (i.e. not indoors)
{ "inside", ~(CONTENTS_SOLID|CONTENTS_OPAQUE), SURF_NONE, CONTENTS_INSIDE }, // volume is considered to be inside (i.e. indoors)
{ "detail", CONTENTS_ALL, SURF_NONE, CONTENTS_DETAIL }, // don't include in structural bsp
{ "trans", CONTENTS_ALL, SURF_NONE, CONTENTS_TRANSLUCENT }, // surface has an alpha component
/* Game surface flags */
{ "sky", CONTENTS_ALL, SURF_SKY, CONTENTS_NONE }, // emit light from an environment map
{ "slick", CONTENTS_ALL, SURF_SLICK, CONTENTS_NONE }, //
{ "nodamage", CONTENTS_ALL, SURF_NODAMAGE, CONTENTS_NONE }, //
{ "noimpact", CONTENTS_ALL, SURF_NOIMPACT, CONTENTS_NONE }, // don't make impact explosions or marks
{ "nomarks", CONTENTS_ALL, SURF_NOMARKS, CONTENTS_NONE }, // don't make impact marks, but still explode
{ "nodraw", CONTENTS_ALL, SURF_NODRAW, CONTENTS_NONE }, // don't generate a drawsurface (or a lightmap)
{ "nosteps", CONTENTS_ALL, SURF_NOSTEPS, CONTENTS_NONE }, //
{ "nodlight", CONTENTS_ALL, SURF_NODLIGHT, CONTENTS_NONE }, // don't ever add dynamic lights
{ "metalsteps", CONTENTS_ALL, SURF_METALSTEPS, CONTENTS_NONE }, //
{ "nomiscents", CONTENTS_ALL, SURF_NOMISCENTS, CONTENTS_NONE }, // No misc ents on this surface
{ "forcefield", CONTENTS_ALL, SURF_FORCEFIELD, CONTENTS_NONE }, //
{ "forcesight", CONTENTS_ALL, SURF_FORCESIGHT, CONTENTS_NONE }, // only visible with force sight
};
/*
===============
ParseSurfaceParm
surfaceparm <name>
===============
*/
static void ParseSurfaceParm( const char **text ) {
char *token;
int numInfoParms = sizeof(infoParms) / sizeof(infoParms[0]);
int i;
token = Shader_ParseExt( text, qfalse );
for ( i = 0 ; i < numInfoParms ; i++ ) {
if ( !Q_stricmp( token, infoParms[i].name ) ) {
shader.surfaceFlags |= infoParms[i].surfaceFlags;
shader.contentFlags |= infoParms[i].contents;
shader.contentFlags &= infoParms[i].clearSolid;
break;
}
}
}
/*
=================
ParseShader
The current text pointer is at the explicit text definition of the
shader. Parse it into the global shader variable. Later functions
will optimize it.
=================
*/
static qboolean ParseShader( const char **text )
{
char *token;
int s;
s = 0;
token = Shader_ParseExt( text, qtrue );
if ( token[0] != '{' )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: expecting '{', found '%s' instead in shader '%s'\n", token, shader.name );
return qfalse;
}
while ( 1 )
{
token = Shader_ParseExt( text, qtrue );
if ( !token[0] )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: no concluding '}' in shader %s\n", shader.name );
return qfalse;
}
// end of shader definition
if ( token[0] == '}' )
{
break;
}
// stage definition
else if ( token[0] == '{' )
{
if ( s >= MAX_SHADER_STAGES ) {
ri->Printf( PRINT_WARNING, "WARNING: too many stages in shader %s\n", shader.name );
return qfalse;
}
if ( !ParseStage( &stages[s], text ) )
{
return qfalse;
}
stages[s].active = qtrue;
if ( stages[s].glow )
{
shader.hasGlow = true;
}
s++;
continue;
}
// skip stuff that only the QuakeEdRadient needs
else if ( !Q_stricmpn( token, "qer", 3 ) ) {
Shader_SkipRestOfLine( text );
continue;
}
// material deprecated as of 11 Jan 01
// material undeprecated as of 7 May 01 - q3map_material deprecated
else if ( !Q_stricmp( token, "material" ) || !Q_stricmp( token, "q3map_material" ) )
{
ParseMaterial( text );
}
// sun parms
else if ( !Q_stricmp( token, "sun" ) || !Q_stricmp( token, "q3map_sun" ) || !Q_stricmp( token, "q3map_sunExt" ) )
{
token = Shader_ParseExt( text, qfalse );
tr.sunLight[0] = atof( token );
token = Shader_ParseExt( text, qfalse );
tr.sunLight[1] = atof( token );
token = Shader_ParseExt( text, qfalse );
tr.sunLight[2] = atof( token );
VectorNormalize( tr.sunLight );
token = Shader_ParseExt( text, qfalse );
float a = atof( token );
VectorScale( tr.sunLight, a, tr.sunLight);
token = Shader_ParseExt( text, qfalse );
a = atof( token );
a = a / 180 * M_PI;
token = Shader_ParseExt( text, qfalse );
float b = atof( token );
b = b / 180 * M_PI;
tr.sunDirection[0] = cos( a ) * cos( b );
tr.sunDirection[1] = sin( a ) * cos( b );
tr.sunDirection[2] = sin( b );
SkipRestOfLine( text );
continue;
}
// q3map_surfacelight deprecated as of 16 Jul 01
else if ( !Q_stricmp( token, "surfacelight" ) || !Q_stricmp( token, "q3map_surfacelight" ) )
{
token = Shader_ParseExt( text, qfalse );
tr.sunSurfaceLight = atoi( token );
}
else if ( !Q_stricmp( token, "lightColor" ) )
{
/*
if ( !ParseVector( text, 3, tr.sunAmbient ) )
{
return qfalse;
}
*/
//SP skips this so I'm skipping it here too.
Shader_SkipRestOfLine( text );
continue;
}
else if ( !Q_stricmp( token, "deformvertexes" ) || !Q_stricmp( token, "deform" )) {
ParseDeform( text );
continue;
}
else if ( !Q_stricmp( token, "tesssize" ) ) {
Shader_SkipRestOfLine( text );
continue;
}
else if ( !Q_stricmp( token, "clampTime" ) ) {
token = Shader_ParseExt( text, qfalse );
if (token[0]) {
shader.clampTime = atof(token);
}
}
// skip stuff that only the q3map needs
else if ( !Q_stricmpn( token, "q3map", 5 ) ) {
Shader_SkipRestOfLine( text );
continue;
}
// skip stuff that only q3map or the server needs
else if ( !Q_stricmp( token, "surfaceParm" ) ) {
ParseSurfaceParm( text );
continue;
}
// no mip maps
else if ( !Q_stricmp( token, "nomipmaps" ) )
{
shader.noMipMaps = true;
shader.noPicMip = true;
continue;
}
// no picmip adjustment
else if ( !Q_stricmp( token, "nopicmip" ) )
{
shader.noPicMip = true;
continue;
}
else if ( !Q_stricmp( token, "noglfog" ) )
{
shader.fogPass = FP_NONE;
continue;
}
// polygonOffset
else if ( !Q_stricmp( token, "polygonOffset" ) )
{
shader.polygonOffset = true;
continue;
}
else if ( !Q_stricmp( token, "noTC" ) )
{
shader.noTC = true;
continue;
}
// entityMergable, allowing sprite surfaces from multiple entities
// to be merged into one batch. This is a savings for smoke
// puffs and blood, but can't be used for anything where the
// shader calcs (not the surface function) reference the entity color or scroll
else if ( !Q_stricmp( token, "entityMergable" ) )
{
shader.entityMergable = true;
continue;
}
// fogParms
else if ( !Q_stricmp( token, "fogParms" ) )
{
shader.fogParms = (fogParms_t *)Hunk_Alloc( sizeof( fogParms_t ), h_low );
if ( !ParseVector( text, 3, shader.fogParms->color ) ) {
return qfalse;
}
token = Shader_ParseExt( text, qfalse );
if ( !token[0] )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing parm for 'fogParms' keyword in shader '%s'\n", shader.name );
continue;
}
shader.fogParms->depthForOpaque = atof( token );
// skip any old gradient directions
Shader_SkipRestOfLine( text );
continue;
}
// portal
else if ( !Q_stricmp(token, "portal") )
{
shader.sort = SS_PORTAL;
continue;
}
// skyparms <cloudheight> <outerbox> <innerbox>
else if ( !Q_stricmp( token, "skyparms" ) )
{
ParseSkyParms( text );
continue;
}
// light <value> determines flaring in q3map, not needed here
else if ( !Q_stricmp(token, "light") )
{
token = Shader_ParseExt( text, qfalse );
continue;
}
// cull <face>
else if ( !Q_stricmp( token, "cull") )
{
token = Shader_ParseExt( text, qfalse );
if ( token[0] == 0 )
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: missing cull parms in shader '%s'\n", shader.name );
continue;
}
if ( !Q_stricmp( token, "none" ) || !Q_stricmp( token, "twosided" ) || !Q_stricmp( token, "disable" ) )
{
shader.cullType = CT_TWO_SIDED;
}
else if ( !Q_stricmp( token, "back" ) || !Q_stricmp( token, "backside" ) || !Q_stricmp( token, "backsided" ) )
{
shader.cullType = CT_BACK_SIDED;
}
else
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: invalid cull parm '%s' in shader '%s'\n", token, shader.name );
}
continue;
}
// sort
else if ( !Q_stricmp( token, "sort" ) )
{
ParseSort( text );
continue;
}
else
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: unknown general shader parameter '%s' in '%s'\n", token, shader.name );
return qfalse;
}
}
//
// ignore shaders that don't have any stages, unless it is a sky or fog
//
if ( s == 0 && !shader.sky && !(shader.contentFlags & CONTENTS_FOG ) ) {
return qfalse;
}
shader.explicitlyDefined = true;
return qtrue;
}
/*
========================================================================================
SHADER OPTIMIZATION AND FOGGING
========================================================================================
*/
typedef struct collapse_s {
int blendA;
int blendB;
int multitextureEnv;
int multitextureBlend;
} collapse_t;
static collapse_t collapse[] = {
{ 0, GLS_DSTBLEND_SRC_COLOR | GLS_SRCBLEND_ZERO,
GL_MODULATE, 0 },
{ 0, GLS_DSTBLEND_ZERO | GLS_SRCBLEND_DST_COLOR,
GL_MODULATE, 0 },
{ GLS_DSTBLEND_ZERO | GLS_SRCBLEND_DST_COLOR, GLS_DSTBLEND_ZERO | GLS_SRCBLEND_DST_COLOR,
GL_MODULATE, GLS_DSTBLEND_ZERO | GLS_SRCBLEND_DST_COLOR },
{ GLS_DSTBLEND_SRC_COLOR | GLS_SRCBLEND_ZERO, GLS_DSTBLEND_ZERO | GLS_SRCBLEND_DST_COLOR,
GL_MODULATE, GLS_DSTBLEND_ZERO | GLS_SRCBLEND_DST_COLOR },
{ GLS_DSTBLEND_ZERO | GLS_SRCBLEND_DST_COLOR, GLS_DSTBLEND_SRC_COLOR | GLS_SRCBLEND_ZERO,
GL_MODULATE, GLS_DSTBLEND_ZERO | GLS_SRCBLEND_DST_COLOR },
{ GLS_DSTBLEND_SRC_COLOR | GLS_SRCBLEND_ZERO, GLS_DSTBLEND_SRC_COLOR | GLS_SRCBLEND_ZERO,
GL_MODULATE, GLS_DSTBLEND_ZERO | GLS_SRCBLEND_DST_COLOR },
{ 0, GLS_DSTBLEND_ONE | GLS_SRCBLEND_ONE,
GL_ADD, 0 },
{ GLS_DSTBLEND_ONE | GLS_SRCBLEND_ONE, GLS_DSTBLEND_ONE | GLS_SRCBLEND_ONE,
GL_ADD, GLS_DSTBLEND_ONE | GLS_SRCBLEND_ONE },
#if 0
{ 0, GLS_DSTBLEND_ONE_MINUS_SRC_ALPHA | GLS_SRCBLEND_SRC_ALPHA,
GL_DECAL, 0 },
#endif
{ -1 }
};
/*
================
CollapseMultitexture
Attempt to combine two stages into a single multitexture stage
FIXME: I think modulated add + modulated add collapses incorrectly
=================
*/
static qboolean CollapseMultitexture( void ) {
int abits, bbits;
int i;
textureBundle_t tmpBundle;
if ( !qglActiveTextureARB ) {
return qfalse;
}
// make sure both stages are active
if ( !stages[0].active || !stages[1].active ) {
return qfalse;
}
abits = stages[0].stateBits;
bbits = stages[1].stateBits;
// make sure that both stages have identical state other than blend modes
if ( ( abits & ~( GLS_DSTBLEND_BITS | GLS_SRCBLEND_BITS | GLS_DEPTHMASK_TRUE ) ) !=
( bbits & ~( GLS_DSTBLEND_BITS | GLS_SRCBLEND_BITS | GLS_DEPTHMASK_TRUE ) ) ) {
return qfalse;
}
abits &= ( GLS_DSTBLEND_BITS | GLS_SRCBLEND_BITS );
bbits &= ( GLS_DSTBLEND_BITS | GLS_SRCBLEND_BITS );
// search for a valid multitexture blend function
for ( i = 0; collapse[i].blendA != -1 ; i++ ) {
if ( abits == collapse[i].blendA
&& bbits == collapse[i].blendB ) {
break;
}
}
// nothing found
if ( collapse[i].blendA == -1 ) {
return qfalse;
}
// GL_ADD is a separate extension
if ( collapse[i].multitextureEnv == GL_ADD && !glConfig.textureEnvAddAvailable ) {
return qfalse;
}
// make sure waveforms have identical parameters
if ( ( stages[0].rgbGen != stages[1].rgbGen ) ||
( stages[0].alphaGen != stages[1].alphaGen ) ) {
return qfalse;
}
// an add collapse can only have identity colors
if ( collapse[i].multitextureEnv == GL_ADD && stages[0].rgbGen != CGEN_IDENTITY ) {
return qfalse;
}
if ( stages[0].rgbGen == CGEN_WAVEFORM )
{
if ( memcmp( &stages[0].rgbWave,
&stages[1].rgbWave,
sizeof( stages[0].rgbWave ) ) )
{
return qfalse;
}
}
if ( stages[0].alphaGen == AGEN_WAVEFORM )
{
if ( memcmp( &stages[0].alphaWave,
&stages[1].alphaWave,
sizeof( stages[0].alphaWave ) ) )
{
return qfalse;
}
}
// make sure that lightmaps are in bundle 1 for 3dfx
if ( stages[0].bundle[0].isLightmap )
{
tmpBundle = stages[0].bundle[0];
stages[0].bundle[0] = stages[1].bundle[0];
stages[0].bundle[1] = tmpBundle;
}
else
{
stages[0].bundle[1] = stages[1].bundle[0];
}
// set the new blend state bits
shader.multitextureEnv = collapse[i].multitextureEnv;
stages[0].stateBits &= ~( GLS_DSTBLEND_BITS | GLS_SRCBLEND_BITS );
stages[0].stateBits |= collapse[i].multitextureBlend;
//
// move down subsequent shaders
//
memmove( &stages[1], &stages[2], sizeof( stages[0] ) * ( MAX_SHADER_STAGES - 2 ) );
memset( &stages[MAX_SHADER_STAGES-1], 0, sizeof( stages[0] ) );
return qtrue;
}
/*
=============
FixRenderCommandList
https://zerowing.idsoftware.com/bugzilla/show_bug.cgi?id=493
Arnout: this is a nasty issue. Shaders can be registered after drawsurfaces are generated
but before the frame is rendered. This will, for the duration of one frame, cause drawsurfaces
to be rendered with bad shaders. To fix this, need to go through all render commands and fix
sortedIndex.
==============
*/
extern bool gServerSkinHack;
static void FixRenderCommandList( int newShader ) {
if( !gServerSkinHack ) {
renderCommandList_t *cmdList = &backEndData->commands;
if( cmdList ) {
const void *curCmd = cmdList->cmds;
while ( 1 ) {
curCmd = PADP(curCmd, sizeof(void *));
switch ( *(const int *)curCmd ) {
case RC_SET_COLOR:
{
const setColorCommand_t *sc_cmd = (const setColorCommand_t *)curCmd;
curCmd = (const void *)(sc_cmd + 1);
break;
}
case RC_STRETCH_PIC:
{
const stretchPicCommand_t *sp_cmd = (const stretchPicCommand_t *)curCmd;
curCmd = (const void *)(sp_cmd + 1);
break;
}
case RC_ROTATE_PIC:
{
const rotatePicCommand_t *sp_cmd = (const rotatePicCommand_t *)curCmd;
curCmd = (const void *)(sp_cmd + 1);
break;
}
case RC_ROTATE_PIC2:
{
const rotatePicCommand_t *sp_cmd = (const rotatePicCommand_t *)curCmd;
curCmd = (const void *)(sp_cmd + 1);
break;
}
case RC_DRAW_SURFS:
{
int i;
drawSurf_t *drawSurf;
shader_t *shader;
int fogNum;
int entityNum;
int dlightMap;
int sortedIndex;
const drawSurfsCommand_t *ds_cmd = (const drawSurfsCommand_t *)curCmd;
for( i = 0, drawSurf = ds_cmd->drawSurfs; i < ds_cmd->numDrawSurfs; i++, drawSurf++ ) {
R_DecomposeSort( drawSurf->sort, &entityNum, &shader, &fogNum, &dlightMap );
sortedIndex = (( drawSurf->sort >> QSORT_SHADERNUM_SHIFT ) & (MAX_SHADERS-1));
if( sortedIndex >= newShader ) {
sortedIndex++;
drawSurf->sort = (sortedIndex << QSORT_SHADERNUM_SHIFT) | (entityNum << QSORT_REFENTITYNUM_SHIFT) | ( fogNum << QSORT_FOGNUM_SHIFT ) | (int)dlightMap;
}
}
curCmd = (const void *)(ds_cmd + 1);
break;
}
case RC_DRAW_BUFFER:
case RC_WORLD_EFFECTS:
case RC_AUTO_MAP:
{
const drawBufferCommand_t *db_cmd = (const drawBufferCommand_t *)curCmd;
curCmd = (const void *)(db_cmd + 1);
break;
}
case RC_SWAP_BUFFERS:
{
const swapBuffersCommand_t *sb_cmd = (const swapBuffersCommand_t *)curCmd;
curCmd = (const void *)(sb_cmd + 1);
break;
}
case RC_END_OF_LIST:
default:
return;
}
}
}
}
}
/*
==============
SortNewShader
Positions the most recently created shader in the tr.sortedShaders[]
array so that the shader->sort key is sorted reletive to the other
shaders.
Sets shader->sortedIndex
==============
*/
static void SortNewShader( void ) {
int i;
float sort;
shader_t *newShader;
newShader = tr.shaders[ tr.numShaders - 1 ];
sort = newShader->sort;
for ( i = tr.numShaders - 2 ; i >= 0 ; i-- ) {
if ( tr.sortedShaders[ i ]->sort <= sort ) {
break;
}
tr.sortedShaders[i+1] = tr.sortedShaders[i];
tr.sortedShaders[i+1]->sortedIndex++;
}
// Arnout: fix rendercommandlist
// https://zerowing.idsoftware.com/bugzilla/show_bug.cgi?id=493
FixRenderCommandList( i+1 );
newShader->sortedIndex = i+1;
tr.sortedShaders[i+1] = newShader;
}
/*
====================
GeneratePermanentShader
====================
*/
static shader_t *GeneratePermanentShader( void ) {
shader_t *newShader;
int i, b;
int size;
if ( tr.numShaders == MAX_SHADERS ) {
//ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: GeneratePermanentShader - MAX_SHADERS hit\n");
ri->Printf( PRINT_ALL, "WARNING: GeneratePermanentShader - MAX_SHADERS hit\n");
return tr.defaultShader;
}
newShader = (struct shader_s *)ri->Hunk_Alloc( sizeof( shader_t ), h_low );
*newShader = shader;
if ( shader.sort <= /*SS_OPAQUE*/SS_SEE_THROUGH ) {
newShader->fogPass = FP_EQUAL;
} else if ( shader.contentFlags & CONTENTS_FOG ) {
newShader->fogPass = FP_LE;
}
tr.shaders[ tr.numShaders ] = newShader;
newShader->index = tr.numShaders;
tr.sortedShaders[ tr.numShaders ] = newShader;
newShader->sortedIndex = tr.numShaders;
tr.numShaders++;
size = newShader->numUnfoggedPasses ? newShader->numUnfoggedPasses * sizeof( stages[0] ) : sizeof( stages[0] );
newShader->stages = (shaderStage_t *) Hunk_Alloc( size, h_low );
for ( i = 0 ; i < newShader->numUnfoggedPasses ; i++ ) {
if ( !stages[i].active ) {
break;
}
newShader->stages[i] = stages[i];
for ( b = 0 ; b < NUM_TEXTURE_BUNDLES ; b++ ) {
if (newShader->stages[i].bundle[b].numTexMods)
{
size = newShader->stages[i].bundle[b].numTexMods * sizeof( texModInfo_t );
newShader->stages[i].bundle[b].texMods = (texModInfo_t *)Hunk_Alloc( size, h_low );
memcpy( newShader->stages[i].bundle[b].texMods, stages[i].bundle[b].texMods, size );
}
else
{
newShader->stages[i].bundle[b].texMods = 0; //clear the globabl ptr jic
}
}
}
SortNewShader();
const int hash = generateHashValue(newShader->name, FILE_HASH_SIZE);
newShader->next = hashTable[hash];
hashTable[hash] = newShader;
return newShader;
}
/*
=================
VertexLightingCollapse
If vertex lighting is enabled, only render a single
pass, trying to guess which is the correct one to best approximate
what it is supposed to look like.
OUTPUT: Number of stages after the collapse (in the case of surfacesprites this isn't one).
=================
*/
//rww - no longer used, at least for now. destroys alpha shaders completely.
#if 0
static int VertexLightingCollapse( void ) {
int stage, nextopenstage;
shaderStage_t *bestStage;
int bestImageRank;
int rank;
int finalstagenum=1;
// if we aren't opaque, just use the first pass
if ( shader.sort == SS_OPAQUE ) {
// pick the best texture for the single pass
bestStage = &stages[0];
bestImageRank = -999999;
for ( stage = 0; stage < MAX_SHADER_STAGES; stage++ ) {
shaderStage_t *pStage = &stages[stage];
if ( !pStage->active ) {
break;
}
rank = 0;
if ( pStage->bundle[0].isLightmap ) {
rank -= 100;
}
if ( pStage->bundle[0].tcGen != TCGEN_TEXTURE ) {
rank -= 5;
}
if ( pStage->bundle[0].numTexMods ) {
rank -= 5;
}
if ( pStage->rgbGen != CGEN_IDENTITY && pStage->rgbGen != CGEN_IDENTITY_LIGHTING ) {
rank -= 3;
}
// SurfaceSprites are most certainly NOT desireable as the collapsed surface texture.
if ( pStage->ss && pstage->ss->surfaceSpriteType)
{
rank -= 1000;
}
if ( rank > bestImageRank ) {
bestImageRank = rank;
bestStage = pStage;
}
}
stages[0].bundle[0] = bestStage->bundle[0];
stages[0].stateBits &= ~( GLS_DSTBLEND_BITS | GLS_SRCBLEND_BITS );
stages[0].stateBits |= GLS_DEPTHMASK_TRUE;
if ( shader.lightmapIndex[0] == LIGHTMAP_NONE ) {
stages[0].rgbGen = CGEN_LIGHTING_DIFFUSE;
} else {
stages[0].rgbGen = CGEN_EXACT_VERTEX;
}
stages[0].alphaGen = AGEN_SKIP;
} else {
// don't use a lightmap (tesla coils)
if ( stages[0].bundle[0].isLightmap ) {
stages[0] = stages[1];
}
// if we were in a cross-fade cgen, hack it to normal
if ( stages[0].rgbGen == CGEN_ONE_MINUS_ENTITY || stages[1].rgbGen == CGEN_ONE_MINUS_ENTITY ) {
stages[0].rgbGen = CGEN_IDENTITY_LIGHTING;
}
if ( ( stages[0].rgbGen == CGEN_WAVEFORM && stages[0].rgbWave.func == GF_SAWTOOTH )
&& ( stages[1].rgbGen == CGEN_WAVEFORM && stages[1].rgbWave.func == GF_INVERSE_SAWTOOTH ) ) {
stages[0].rgbGen = CGEN_IDENTITY_LIGHTING;
}
if ( ( stages[0].rgbGen == CGEN_WAVEFORM && stages[0].rgbWave.func == GF_INVERSE_SAWTOOTH )
&& ( stages[1].rgbGen == CGEN_WAVEFORM && stages[1].rgbWave.func == GF_SAWTOOTH ) ) {
stages[0].rgbGen = CGEN_IDENTITY_LIGHTING;
}
}
for ( stage=1, nextopenstage=1; stage < MAX_SHADER_STAGES; stage++ ) {
shaderStage_t *pStage = &stages[stage];
if ( !pStage->active ) {
break;
}
if ( pStage->ss && pstage->ss->surfaceSpriteType)
{
// Copy this stage to the next open stage list (that is, we don't want any inactive stages before this one)
if (nextopenstage != stage)
{
stages[nextopenstage] = *pStage;
stages[nextopenstage].bundle[0] = pStage->bundle[0];
}
nextopenstage++;
finalstagenum++;
continue;
}
memset( pStage, 0, sizeof( *pStage ) );
}
return finalstagenum;
}
#endif
/*
=========================
FinishShader
Returns a freshly allocated shader with all the needed info
from the current global working shader
=========================
*/
static shader_t *FinishShader( void ) {
int stage, lmStage, stageIndex; //rwwRMG - stageIndex for AGEN_BLEND
qboolean hasLightmapStage;
qboolean vertexLightmap;
hasLightmapStage = qfalse;
vertexLightmap = qfalse;
//
// set sky stuff appropriate
//
if ( shader.sky ) {
shader.sort = SS_ENVIRONMENT;
}
//
// set polygon offset
//
if ( shader.polygonOffset && !shader.sort ) {
shader.sort = SS_DECAL;
}
for(lmStage = 0; lmStage < MAX_SHADER_STAGES; lmStage++)
{
shaderStage_t *pStage = &stages[lmStage];
if (pStage->active && pStage->bundle[0].isLightmap)
{
break;
}
}
if (lmStage < MAX_SHADER_STAGES)
{
if (shader.lightmapIndex[0] == LIGHTMAP_BY_VERTEX)
{
if (lmStage == 0) //< MAX_SHADER_STAGES-1)
{//copy the rest down over the lightmap slot
memmove(&stages[lmStage], &stages[lmStage+1], sizeof(shaderStage_t) * (MAX_SHADER_STAGES-lmStage-1));
memset(&stages[MAX_SHADER_STAGES-1], 0, sizeof(shaderStage_t));
//change blending on the moved down stage
stages[lmStage].stateBits = GLS_DEFAULT;
}
//change anything that was moved down (or the *white if LM is first) to use vertex color
stages[lmStage].rgbGen = CGEN_EXACT_VERTEX;
stages[lmStage].alphaGen = AGEN_SKIP;
lmStage = MAX_SHADER_STAGES; //skip the style checking below
}
}
if (lmStage < MAX_SHADER_STAGES)// && !r_fullbright->value)
{
int numStyles;
int i;
for(numStyles=0;numStyles<MAXLIGHTMAPS;numStyles++)
{
if (shader.styles[numStyles] >= LS_UNUSED)
{
break;
}
}
numStyles--;
if (numStyles > 0)
{
for(i=MAX_SHADER_STAGES-1;i>lmStage+numStyles;i--)
{
stages[i] = stages[i-numStyles];
}
for(i=0;i<numStyles;i++)
{
stages[lmStage+i+1] = stages[lmStage];
if (shader.lightmapIndex[i+1] == LIGHTMAP_BY_VERTEX)
{
stages[lmStage+i+1].bundle[0].image = tr.whiteImage;
}
else if (shader.lightmapIndex[i+1] < 0)
{
Com_Error( ERR_DROP, "FinishShader: light style with no light map or vertex color for shader %s", shader.name);
}
else
{
stages[lmStage+i+1].bundle[0].image = tr.lightmaps[shader.lightmapIndex[i+1]];
stages[lmStage+i+1].bundle[0].tcGen = (texCoordGen_t)(TCGEN_LIGHTMAP+i+1);
}
stages[lmStage+i+1].rgbGen = CGEN_LIGHTMAPSTYLE;
stages[lmStage+i+1].stateBits &= ~(GLS_SRCBLEND_BITS | GLS_DSTBLEND_BITS);
stages[lmStage+i+1].stateBits |= GLS_SRCBLEND_ONE | GLS_DSTBLEND_ONE;
}
}
for(i=0;i<=numStyles;i++)
{
stages[lmStage+i].lightmapStyle = shader.styles[i];
}
}
//
// set appropriate stage information
//
stageIndex = 0; //rwwRMG - needed for AGEN_BLEND
for ( stage = 0; stage < MAX_SHADER_STAGES; ) {
shaderStage_t *pStage = &stages[stage];
if ( !pStage->active ) {
break;
}
// check for a missing texture
if ( !pStage->bundle[0].image ) {
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "Shader %s has a stage with no image\n", shader.name );
pStage->active = qfalse;
stage++;
continue;
}
//
// ditch this stage if it's detail and detail textures are disabled
//
if ( pStage->isDetail && !r_detailTextures->integer ) {
int index;
for ( index=stage+1; index<MAX_SHADER_STAGES; index++ ) {
if ( !stages[index].active )
break;
}
if ( index < MAX_SHADER_STAGES )
memmove( pStage, pStage + 1, sizeof( *pStage ) * ( index - stage ) );
else {
if ( stage + 1 < MAX_SHADER_STAGES )
memmove( pStage, pStage + 1, sizeof( *pStage ) * ( index - stage - 1 ) );
Com_Memset( &stages[index - 1], 0, sizeof( *stages ) );
}
continue;
}
pStage->index = stageIndex; //rwwRMG - needed for AGEN_BLEND
//
// default texture coordinate generation
//
if ( pStage->bundle[0].isLightmap ) {
if ( pStage->bundle[0].tcGen == TCGEN_BAD ) {
pStage->bundle[0].tcGen = TCGEN_LIGHTMAP;
}
hasLightmapStage = qtrue;
} else {
if ( pStage->bundle[0].tcGen == TCGEN_BAD ) {
pStage->bundle[0].tcGen = TCGEN_TEXTURE;
}
}
// not a true lightmap but we want to leave existing
// behaviour in place and not print out a warning
//if (pStage->rgbGen == CGEN_VERTEX) {
// vertexLightmap = qtrue;
//}
//
// determine sort order and fog color adjustment
//
if ( ( pStage->stateBits & ( GLS_SRCBLEND_BITS | GLS_DSTBLEND_BITS ) ) &&
( stages[0].stateBits & ( GLS_SRCBLEND_BITS | GLS_DSTBLEND_BITS ) ) ) {
int blendSrcBits = pStage->stateBits & GLS_SRCBLEND_BITS;
int blendDstBits = pStage->stateBits & GLS_DSTBLEND_BITS;
// fog color adjustment only works for blend modes that have a contribution
// that aproaches 0 as the modulate values aproach 0 --
// GL_ONE, GL_ONE
// GL_ZERO, GL_ONE_MINUS_SRC_COLOR
// GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA
// modulate, additive
if ( ( ( blendSrcBits == GLS_SRCBLEND_ONE ) && ( blendDstBits == GLS_DSTBLEND_ONE ) ) ||
( ( blendSrcBits == GLS_SRCBLEND_ZERO ) && ( blendDstBits == GLS_DSTBLEND_ONE_MINUS_SRC_COLOR ) ) ) {
pStage->adjustColorsForFog = ACFF_MODULATE_RGB;
}
// strict blend
else if ( ( blendSrcBits == GLS_SRCBLEND_SRC_ALPHA ) && ( blendDstBits == GLS_DSTBLEND_ONE_MINUS_SRC_ALPHA ) )
{
pStage->adjustColorsForFog = ACFF_MODULATE_ALPHA;
}
// premultiplied alpha
else if ( ( blendSrcBits == GLS_SRCBLEND_ONE ) && ( blendDstBits == GLS_DSTBLEND_ONE_MINUS_SRC_ALPHA ) )
{
pStage->adjustColorsForFog = ACFF_MODULATE_RGBA;
} else {
// we can't adjust this one correctly, so it won't be exactly correct in fog
}
// don't screw with sort order if this is a portal or environment
if ( !shader.sort ) {
// see through item, like a grill or grate
if ( pStage->stateBits & GLS_DEPTHMASK_TRUE )
{
shader.sort = SS_SEE_THROUGH;
}
else
{
/*
if (( blendSrcBits == GLS_SRCBLEND_ONE ) && ( blendDstBits == GLS_DSTBLEND_ONE ))
{
// GL_ONE GL_ONE needs to come a bit later
shader.sort = SS_BLEND2;
}
else if (( blendSrcBits == GLS_SRCBLEND_SRC_ALPHA ) && ( blendDstBits == GLS_DSTBLEND_ONE_MINUS_SRC_ALPHA ))
{ //rww - Pushed SS_BLEND1 up to SS_BLEND2, inserting this so that saber glow will render above water and things.
//Unfortunately it still affects other shaders with the same blend settings, but it seems more or less alright.
shader.sort = SS_BLEND1;
}
else
{
shader.sort = SS_BLEND0;
}
*/
if (( blendSrcBits == GLS_SRCBLEND_ONE ) && ( blendDstBits == GLS_DSTBLEND_ONE ))
{
// GL_ONE GL_ONE needs to come a bit later
shader.sort = SS_BLEND1;
}
else
{
shader.sort = SS_BLEND0;
}
}
}
}
//rww - begin hw fog
if ((pStage->stateBits & (GLS_SRCBLEND_BITS|GLS_DSTBLEND_BITS)) == (GLS_SRCBLEND_ONE|GLS_DSTBLEND_ONE))
{
pStage->mGLFogColorOverride = GLFOGOVERRIDE_BLACK;
}
else if ((pStage->stateBits & (GLS_SRCBLEND_BITS|GLS_DSTBLEND_BITS)) == (GLS_SRCBLEND_SRC_ALPHA|GLS_DSTBLEND_ONE) &&
pStage->alphaGen == AGEN_LIGHTING_SPECULAR && stage)
{
pStage->mGLFogColorOverride = GLFOGOVERRIDE_BLACK;
}
else if ((pStage->stateBits & (GLS_SRCBLEND_BITS|GLS_DSTBLEND_BITS)) == (GLS_SRCBLEND_ZERO|GLS_DSTBLEND_ZERO))
{
pStage->mGLFogColorOverride = GLFOGOVERRIDE_WHITE;
}
else if ((pStage->stateBits & (GLS_SRCBLEND_BITS|GLS_DSTBLEND_BITS)) == (GLS_SRCBLEND_ONE|GLS_DSTBLEND_ZERO))
{
pStage->mGLFogColorOverride = GLFOGOVERRIDE_WHITE;
}
else if ((pStage->stateBits & (GLS_SRCBLEND_BITS|GLS_DSTBLEND_BITS)) == 0 && stage)
{ //
pStage->mGLFogColorOverride = GLFOGOVERRIDE_WHITE;
}
else if ((pStage->stateBits & (GLS_SRCBLEND_BITS|GLS_DSTBLEND_BITS)) == 0 && pStage->bundle[0].isLightmap && stage < MAX_SHADER_STAGES-1 &&
stages[stage+1].bundle[0].isLightmap)
{ // multiple light map blending
pStage->mGLFogColorOverride = GLFOGOVERRIDE_WHITE;
}
else if ((pStage->stateBits & (GLS_SRCBLEND_BITS|GLS_DSTBLEND_BITS)) == (GLS_SRCBLEND_DST_COLOR|GLS_DSTBLEND_ZERO) && pStage->bundle[0].isLightmap)
{ //I don't know, it works. -rww
pStage->mGLFogColorOverride = GLFOGOVERRIDE_WHITE;
}
else if ((pStage->stateBits & (GLS_SRCBLEND_BITS|GLS_DSTBLEND_BITS)) == (GLS_SRCBLEND_DST_COLOR|GLS_DSTBLEND_ZERO))
{ //I don't know, it works. -rww
pStage->mGLFogColorOverride = GLFOGOVERRIDE_BLACK;
}
else if ((pStage->stateBits & (GLS_SRCBLEND_BITS|GLS_DSTBLEND_BITS)) == (GLS_SRCBLEND_ONE|GLS_DSTBLEND_ONE_MINUS_SRC_COLOR))
{ //I don't know, it works. -rww
pStage->mGLFogColorOverride = GLFOGOVERRIDE_BLACK;
}
else
{
pStage->mGLFogColorOverride = GLFOGOVERRIDE_NONE;
}
//rww - end hw fog
stageIndex++; //rwwRMG - needed for AGEN_BLEND
stage++;
}
// there are times when you will need to manually apply a sort to
// opaque alpha tested shaders that have later blend passes
if ( !shader.sort ) {
shader.sort = SS_OPAQUE;
}
//
// if we are in r_vertexLight mode, never use a lightmap texture
//
if ( stage > 1 && (r_vertexLight->integer && !r_uiFullScreen->integer) ) {
//stage = VertexLightingCollapse();
//rww - since this does bad things, I am commenting it out for now. If you want to attempt a fix, feel free.
hasLightmapStage = qfalse;
}
//
// look for multitexture potential
//
if ( stage > 1 && CollapseMultitexture() ) {
stage--;
}
if ( shader.lightmapIndex[0] >= 0 && !hasLightmapStage )
{
if (vertexLightmap)
{
// ri->DPrintf( "WARNING: shader '%s' has VERTEX forced lightmap!\n", shader.name );
}
else
{
ri->Printf( PRINT_DEVELOPER, "WARNING: shader '%s' has lightmap but no lightmap stage!\n", shader.name );
memcpy(shader.lightmapIndex, lightmapsNone, sizeof(shader.lightmapIndex));
memcpy(shader.styles, stylesDefault, sizeof(shader.styles));
}
}
//
// compute number of passes
//
shader.numUnfoggedPasses = stage;
// fogonly shaders don't have any normal passes
if ( stage == 0 && !shader.sky ) {
shader.sort = SS_FOG;
}
for ( stage = 1; stage < shader.numUnfoggedPasses; stage++ )
{
// Make sure stage is non detail and active
if(stages[stage].isDetail || !stages[stage].active)
{
break;
}
// MT lightmaps are always in bundle 1
if(stages[stage].bundle[0].isLightmap)
{
continue;
}
}
return GeneratePermanentShader();
}
//========================================================================================
/*
====================
FindShaderInShaderText
Scans the combined text description of all the shader files for
the given shader name.
return NULL if not found
If found, it will return a valid shader
=====================
*/
static const char *FindShaderInShaderText( const char *shadername ) {
#ifdef USE_NEW_SHADER_HASH
shaderText_t *st;
int hash;
hash = generateHashValue(shadername, MAX_SHADERTEXT_HASH);
for( st = shaderTextHashTable[hash]; st; st = st->next ) {
if ( !Q_stricmp( st->name, shadername ) ) {
return st->text;
}
}
// drakkar - if shader does not exists in the hashtable then it does not exist in s_shaderText
return NULL;
#else
char *token;
const char *p;
int i, hash;
hash = generateHashValue(shadername, MAX_SHADERTEXT_HASH);
for (i = 0; shaderTextHashTable[hash][i]; i++) {
p = shaderTextHashTable[hash][i];
token = Shader_ParseExt(&p, qtrue);
if ( !Q_stricmp( token, shadername ) ) {
return p;
}
}
p = s_shaderText;
if ( !p ) {
return NULL;
}
// look for label
while ( 1 ) {
token = Shader_ParseExt( &p, qtrue );
if ( token[0] == 0 ) {
break;
}
if ( !Q_stricmp( token, shadername ) ) {
return p;
}
else {
// skip the definition
Shader_SkipBracedSection( &p );
}
}
return NULL;
#endif
}
/*
==================
R_FindShaderByName
Will always return a valid shader, but it might be the
default shader if the real one can't be found.
==================
*/
shader_t *R_FindShaderByName( const char *name ) {
char strippedName[MAX_QPATH];
int hash;
shader_t *sh;
if ( (name==NULL) || (name[0] == 0) ) { // bk001205
return tr.defaultShader;
}
COM_StripExtension( name, strippedName, sizeof( strippedName ) );
hash = generateHashValue(strippedName, FILE_HASH_SIZE);
//
// see if the shader is already loaded
//
for (sh=hashTable[hash]; sh; sh=sh->next) {
// NOTE: if there was no shader or image available with the name strippedName
// then a default shader is created with lightmapIndex == LIGHTMAP_NONE, so we
// have to check all default shaders otherwise for every call to R_FindShader
// with that same strippedName a new default shader is created.
if (Q_stricmp(sh->name, strippedName) == 0) {
// match found
return sh;
}
}
return tr.defaultShader;
}
inline qboolean IsShader(shader_t *sh, const char *name, const int *lightmapIndex, const byte *styles)
{
int i;
if (Q_stricmp(sh->name, name))
{
return qfalse;
}
if (!sh->defaultShader)
{
for(i=0;i<MAXLIGHTMAPS;i++)
{
if (sh->lightmapIndex[i] != lightmapIndex[i])
{
return qfalse;
}
if (sh->styles[i] != styles[i])
{
return qfalse;
}
}
}
return qtrue;
}
/*
===============
R_FindShader
Will always return a valid shader, but it might be the
default shader if the real one can't be found.
In the interest of not requiring an explicit shader text entry to
be defined for every single image used in the game, three default
shader behaviors can be auto-created for any image:
If lightmapIndex == LIGHTMAP_NONE, then the image will have
dynamic diffuse lighting applied to it, as appropriate for most
entity skin surfaces.
If lightmapIndex == LIGHTMAP_2D, then the image will be used
for 2D rendering unless an explicit shader is found
If lightmapIndex == LIGHTMAP_BY_VERTEX, then the image will use
the vertex rgba modulate values, as appropriate for misc_model
pre-lit surfaces.
Other lightmapIndex values will have a lightmap stage created
and src*dest blending applied with the texture, as appropriate for
most world construction surfaces.
===============
*/
shader_t *R_FindShader( const char *name, const int *lightmapIndex, const byte *styles, qboolean mipRawImage )
{
char strippedName[MAX_QPATH];
char fileName[MAX_QPATH];
int hash;
const char *shaderText;
image_t *image;
shader_t *sh;
if ( name[0] == 0 ) {
return tr.defaultShader;
}
// use (fullbright) vertex lighting if the bsp file doesn't have
// lightmaps
if ( lightmapIndex[0] >= 0 && lightmapIndex[0] >= tr.numLightmaps )
{
lightmapIndex = lightmapsVertex;
}
else if ( lightmapIndex[0] < LIGHTMAP_2D )
{
// negative lightmap indexes cause stray pointers (think tr.lightmaps[lightmapIndex])
ri->Printf( PRINT_WARNING, "WARNING: shader '%s' has invalid lightmap index of %d\n", name, lightmapIndex[0] );
lightmapIndex = lightmapsVertex;
}
COM_StripExtension( name, strippedName, sizeof( strippedName ) );
hash = generateHashValue(strippedName, FILE_HASH_SIZE);
//
// see if the shader is already loaded
//
for (sh = hashTable[hash]; sh; sh = sh->next) {
// NOTE: if there was no shader or image available with the name strippedName
// then a default shader is created with lightmapIndex == LIGHTMAP_NONE, so we
// have to check all default shaders otherwise for every call to R_FindShader
// with that same strippedName a new default shader is created.
if (IsShader(sh, strippedName, lightmapIndex, styles))
{
return sh;
}
}
// clear the global shader
ClearGlobalShader();
Q_strncpyz(shader.name, strippedName, sizeof(shader.name));
Com_Memcpy(shader.lightmapIndex, lightmapIndex, sizeof(shader.lightmapIndex));
Com_Memcpy(shader.styles, styles, sizeof(shader.styles));
//
// attempt to define shader from an explicit parameter file
//
shaderText = FindShaderInShaderText( strippedName );
if ( shaderText ) {
if ( !ParseShader( &shaderText ) ) {
// had errors, so use default shader
shader.defaultShader = true;
}
sh = FinishShader();
return sh;
}
//
// if not defined in the in-memory shader descriptions,
// look for a single TGA, BMP, or PCX
//
COM_StripExtension( name, fileName, sizeof( fileName ) );
image = R_FindImageFile( fileName, mipRawImage, mipRawImage, qtrue, mipRawImage ? GL_REPEAT : GL_CLAMP );
if ( !image ) {
ri->Printf( PRINT_DEVELOPER, S_COLOR_RED "Couldn't find image for shader %s\n", name );
shader.defaultShader = true;
return FinishShader();
}
//
// create the default shading commands
//
if ( shader.lightmapIndex[0] == LIGHTMAP_NONE ) {
// dynamic colors at vertexes
stages[0].bundle[0].image = image;
stages[0].active = qtrue;
stages[0].rgbGen = CGEN_LIGHTING_DIFFUSE;
stages[0].stateBits = GLS_DEFAULT;
} else if ( shader.lightmapIndex[0] == LIGHTMAP_BY_VERTEX ) {
// explicit colors at vertexes
stages[0].bundle[0].image = image;
stages[0].active = qtrue;
stages[0].rgbGen = CGEN_EXACT_VERTEX;
stages[0].alphaGen = AGEN_SKIP;
stages[0].stateBits = GLS_DEFAULT;
} else if ( shader.lightmapIndex[0] == LIGHTMAP_2D ) {
// GUI elements
stages[0].bundle[0].image = image;
stages[0].active = qtrue;
stages[0].rgbGen = CGEN_VERTEX;
stages[0].alphaGen = AGEN_VERTEX;
stages[0].stateBits = GLS_DEPTHTEST_DISABLE |
GLS_SRCBLEND_SRC_ALPHA |
GLS_DSTBLEND_ONE_MINUS_SRC_ALPHA;
} else if ( shader.lightmapIndex[0] == LIGHTMAP_WHITEIMAGE ) {
// fullbright level
stages[0].bundle[0].image = tr.whiteImage;
stages[0].active = qtrue;
stages[0].rgbGen = CGEN_IDENTITY_LIGHTING;
stages[0].stateBits = GLS_DEFAULT;
stages[1].bundle[0].image = image;
stages[1].active = qtrue;
stages[1].rgbGen = CGEN_IDENTITY;
stages[1].stateBits |= GLS_SRCBLEND_DST_COLOR | GLS_DSTBLEND_ZERO;
} else {
// two pass lightmap
stages[0].bundle[0].image = tr.lightmaps[shader.lightmapIndex[0]];
stages[0].bundle[0].isLightmap = qtrue;
stages[0].active = qtrue;
stages[0].rgbGen = CGEN_IDENTITY; // lightmaps are scaled on creation
// for identitylight
stages[0].stateBits = GLS_DEFAULT;
stages[1].bundle[0].image = image;
stages[1].active = qtrue;
stages[1].rgbGen = CGEN_IDENTITY;
stages[1].stateBits |= GLS_SRCBLEND_DST_COLOR | GLS_DSTBLEND_ZERO;
}
return FinishShader();
}
static void ScanAndLoadShaderFiles( const char *path );
shader_t *R_FindServerShader( const char *name, const int *lightmapIndex, const byte *styles, qboolean mipRawImage )
{
char strippedName[MAX_QPATH];
int hash;
shader_t *sh;
if ( name[0] == 0 ) {
return tr.defaultShader;
}
COM_StripExtension( name, strippedName, sizeof( strippedName ) );
hash = generateHashValue(strippedName, FILE_HASH_SIZE);
//
// see if the shader is already loaded
//
for (sh = hashTable[hash]; sh; sh = sh->next) {
// NOTE: if there was no shader or image available with the name strippedName
// then a default shader is created with lightmapIndex == LIGHTMAP_NONE, so we
// have to check all default shaders otherwise for every call to R_FindShader
// with that same strippedName a new default shader is created.
if (IsShader(sh, strippedName, lightmapIndex, styles))
{
return sh;
}
}
// clear the global shader
ClearGlobalShader();
Q_strncpyz(shader.name, strippedName, sizeof(shader.name));
memcpy(shader.lightmapIndex, lightmapIndex, sizeof(shader.lightmapIndex));
memcpy(shader.styles, styles, sizeof(shader.styles));
shader.defaultShader = true;
return FinishShader();
}
qhandle_t RE_RegisterShaderFromImage(const char *name, int *lightmapIndex, byte *styles, image_t *image, qboolean mipRawImage) {
int i, hash;
shader_t *sh;
hash = generateHashValue(name, FILE_HASH_SIZE);
// probably not necessary since this function
// only gets called from tr_font.c with lightmapIndex == LIGHTMAP_2D
// but better safe than sorry.
// Doesn't actually ever get called in JA at all
if ( lightmapIndex[0] >= tr.numLightmaps ) {
lightmapIndex = (int *)lightmapsFullBright;
}
//
// see if the shader is already loaded
//
for (sh=hashTable[hash]; sh; sh=sh->next) {
// NOTE: if there was no shader or image available with the name strippedName
// then a default shader is created with lightmapIndex == LIGHTMAP_NONE, so we
// have to check all default shaders otherwise for every call to R_FindShader
// with that same strippedName a new default shader is created.
if (IsShader(sh, name, lightmapIndex, styles))
{
return sh->index;
}
}
// clear the global shader
memset( &shader, 0, sizeof( shader ) );
memset( &stages, 0, sizeof( stages ) );
Q_strncpyz(shader.name, name, sizeof(shader.name));
memcpy(shader.lightmapIndex, lightmapIndex, sizeof(shader.lightmapIndex));
memcpy(shader.styles, styles, sizeof(shader.styles));
for ( i = 0 ; i < MAX_SHADER_STAGES ; i++ ) {
stages[i].bundle[0].texMods = texMods[i];
}
//
// create the default shading commands
//
if ( shader.lightmapIndex[0] == LIGHTMAP_NONE ) {
// dynamic colors at vertexes
stages[0].bundle[0].image = image;
stages[0].active = qtrue;
stages[0].rgbGen = CGEN_LIGHTING_DIFFUSE;
stages[0].stateBits = GLS_DEFAULT;
} else if ( shader.lightmapIndex[0] == LIGHTMAP_BY_VERTEX ) {
// explicit colors at vertexes
stages[0].bundle[0].image = image;
stages[0].active = qtrue;
stages[0].rgbGen = CGEN_EXACT_VERTEX;
stages[0].alphaGen = AGEN_SKIP;
stages[0].stateBits = GLS_DEFAULT;
} else if ( shader.lightmapIndex[0] == LIGHTMAP_2D ) {
// GUI elements
stages[0].bundle[0].image = image;
stages[0].active = qtrue;
stages[0].rgbGen = CGEN_VERTEX;
stages[0].alphaGen = AGEN_VERTEX;
stages[0].stateBits = GLS_DEPTHTEST_DISABLE |
GLS_SRCBLEND_SRC_ALPHA |
GLS_DSTBLEND_ONE_MINUS_SRC_ALPHA;
} else if ( shader.lightmapIndex[0] == LIGHTMAP_WHITEIMAGE ) {
// fullbright level
stages[0].bundle[0].image = tr.whiteImage;
stages[0].active = qtrue;
stages[0].rgbGen = CGEN_IDENTITY_LIGHTING;
stages[0].stateBits = GLS_DEFAULT;
stages[1].bundle[0].image = image;
stages[1].active = qtrue;
stages[1].rgbGen = CGEN_IDENTITY;
stages[1].stateBits |= GLS_SRCBLEND_DST_COLOR | GLS_DSTBLEND_ZERO;
} else {
// two pass lightmap
stages[0].bundle[0].image = tr.lightmaps[shader.lightmapIndex[0]];
stages[0].bundle[0].isLightmap = qtrue;
stages[0].active = qtrue;
stages[0].rgbGen = CGEN_IDENTITY; // lightmaps are scaled on creation
// for identitylight
stages[0].stateBits = GLS_DEFAULT;
stages[1].bundle[0].image = image;
stages[1].active = qtrue;
stages[1].rgbGen = CGEN_IDENTITY;
stages[1].stateBits |= GLS_SRCBLEND_DST_COLOR | GLS_DSTBLEND_ZERO;
}
sh = FinishShader();
return sh->index;
}
/*
====================
RE_RegisterShader
This is the exported shader entry point for the rest of the system
It will always return an index that will be valid.
This should really only be used for explicit shaders, because there is no
way to ask for different implicit lighting modes (vertex, lightmap, etc)
====================
*/
qhandle_t RE_RegisterShaderLightMap( const char *name, const int *lightmapIndex, const byte *styles )
{
shader_t *sh;
if ( strlen( name ) >= MAX_QPATH ) {
ri->Printf( PRINT_ALL, "Shader name exceeds MAX_QPATH\n" );
return 0;
}
sh = R_FindShader( name, lightmapIndex, styles, qtrue );
// we want to return 0 if the shader failed to
// load for some reason, but R_FindShader should
// still keep a name allocated for it, so if
// something calls RE_RegisterShader again with
// the same name, we don't try looking for it again
if ( sh->defaultShader ) {
return 0;
}
return sh->index;
}
/*
====================
RE_RegisterShader
This is the exported shader entry point for the rest of the system
It will always return an index that will be valid.
This should really only be used for explicit shaders, because there is no
way to ask for different implicit lighting modes (vertex, lightmap, etc)
====================
*/
qhandle_t RE_RegisterShader( const char *name ) {
shader_t *sh;
if ( strlen( name ) >= MAX_QPATH ) {
ri->Printf( PRINT_ALL, "Shader name exceeds MAX_QPATH\n" );
return 0;
}
sh = R_FindShader( name, lightmaps2d, stylesDefault, qtrue );
// we want to return 0 if the shader failed to
// load for some reason, but R_FindShader should
// still keep a name allocated for it, so if
// something calls RE_RegisterShader again with
// the same name, we don't try looking for it again
if ( sh->defaultShader ) {
return 0;
}
return sh->index;
}
/*
====================
RE_RegisterShaderNoMip
For menu graphics that should never be picmiped
====================
*/
qhandle_t RE_RegisterShaderNoMip( const char *name ) {
shader_t *sh;
if ( strlen( name ) >= MAX_QPATH ) {
ri->Printf( PRINT_ALL, "Shader name exceeds MAX_QPATH\n" );
return 0;
}
sh = R_FindShader( name, lightmaps2d, stylesDefault, qfalse );
// we want to return 0 if the shader failed to
// load for some reason, but R_FindShader should
// still keep a name allocated for it, so if
// something calls RE_RegisterShader again with
// the same name, we don't try looking for it again
if ( sh->defaultShader ) {
return 0;
}
return sh->index;
}
//added for ui -rww
const char *RE_ShaderNameFromIndex(int index)
{
assert(index >= 0 && index < tr.numShaders && tr.shaders[index]);
return tr.shaders[index]->name;
}
/*
====================
R_GetShaderByHandle
When a handle is passed in by another module, this range checks
it and returns a valid (possibly default) shader_t to be used internally.
====================
*/
shader_t *R_GetShaderByHandle( qhandle_t hShader ) {
if ( hShader < 0 ) {
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "R_GetShaderByHandle: out of range hShader '%d'\n", hShader ); // bk: FIXME name
return tr.defaultShader;
}
if ( hShader >= tr.numShaders ) {
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "R_GetShaderByHandle: out of range hShader '%d'\n", hShader );
return tr.defaultShader;
}
return tr.shaders[hShader];
}
/*
===============
R_ShaderList_f
Dump information on all valid shaders to the console
A second parameter will cause it to print in sorted order
===============
*/
void R_ShaderList_f (void) {
int i;
int count;
shader_t *shader;
ri->Printf( PRINT_ALL, "-----------------------\n");
count = 0;
for ( i = 0 ; i < tr.numShaders ; i++ ) {
if ( ri->Cmd_Argc() > 1 ) {
shader = tr.sortedShaders[i];
} else {
shader = tr.shaders[i];
}
ri->Printf( PRINT_ALL, "%i ", shader->numUnfoggedPasses );
if (shader->lightmapIndex[0] >= 0 ) {
ri->Printf( PRINT_ALL, "L ");
} else {
ri->Printf( PRINT_ALL, " ");
}
if ( shader->multitextureEnv == GL_ADD ) {
ri->Printf( PRINT_ALL, "MT(a) " );
} else if ( shader->multitextureEnv == GL_MODULATE ) {
ri->Printf( PRINT_ALL, "MT(m) " );
} else if ( shader->multitextureEnv == GL_DECAL ) {
ri->Printf( PRINT_ALL, "MT(d) " );
} else {
ri->Printf( PRINT_ALL, " " );
}
if ( shader->explicitlyDefined ) {
ri->Printf( PRINT_ALL, "E " );
} else {
ri->Printf( PRINT_ALL, " " );
}
if ( shader->sky )
{
ri->Printf( PRINT_ALL, "sky " );
} else {
ri->Printf( PRINT_ALL, "gen " );
}
if ( shader->defaultShader ) {
ri->Printf( PRINT_ALL, ": %s (DEFAULTED)\n", shader->name);
} else {
ri->Printf( PRINT_ALL, ": %s\n", shader->name);
}
count++;
}
ri->Printf( PRINT_ALL, "%i total shaders\n", count);
ri->Printf( PRINT_ALL, "------------------\n");
}
#ifdef USE_NEW_SHADER_HASH
// drakkar - extract shaders from the buffer and insert into the hashtable
static void LoadShaderFromBuffer( char *buff )
{
char shadername[MAX_SHADERNAME_LENGTH+1];
shaderText_t *st;
char *p, *name, *text;
int nameLength, textLength;
long size, hash;
int q3ShaderBug = 1;
if( !buff ) return;
p = buff;
while( *p )
{
// get next shader name and shader text from buffer
Shader_CompressBracedSection( &p, &name, &text, &nameLength, &textLength );
if( !nameLength || !textLength ) continue;
if( nameLength >= MAX_SHADERNAME_LENGTH ) {
strncpy( shadername, name, MAX_SHADERNAME_LENGTH );
shadername[MAX_SHADERNAME_LENGTH] = '\0';
ri->Printf( PRINT_DEVELOPER, "Warning: Shader name too long '%s'...\n", shadername );
continue;
}
strncpy( shadername, name, nameLength );
shadername[nameLength] = '\0';
name = shadername;
// if shader already exists ignore the new shader text
hash = generateHashValue( name, MAX_SHADERTEXT_HASH );
for( st = shaderTextHashTable[hash]; st; st = st->next ) {
if( !Q_stricmp( name, st->name ) ) {
if( q3ShaderBug ) { // simulating q3 shader bug: override only the first shader of the buffer
st->name[0] = '\0';
st = NULL;
}
break;
}
}
if( st ) continue;
q3ShaderBug = 0;
// create the new shader
size = sizeof(shaderText_t) + (textLength) + (nameLength+1);
st = (shaderText_t *)ri->Hunk_Alloc( size, h_low );
// copy shader name and shader text
memcpy( st->text, text, textLength );
st->text[textLength] = '\0';
st->name = st->text + (textLength+1);
strncpy( st->name, name, nameLength );
st->name[nameLength] = '\0';
// insert the new shader into hashtable
st->next = shaderTextHashTable[hash];
shaderTextHashTable[hash] = st;
shaderCount++;
}
}
static void ScanAndLoadShaderFiles( const char *path ) // drakkar - using LoadShaderFromBuffer()
{
char filename[MAX_QPATH];
char *buffer;
char **shaderFiles;
int i, numShaderFiles;
// scan for shader files
shaderFiles = ri->FS_ListFiles( path, ".shader", &numShaderFiles );
if ( !shaderFiles || !numShaderFiles )
{
Com_Error(ERR_FATAL, "ERROR: no shader files found\n");
return;
}
// load and parse shader files
for( i = numShaderFiles-1; i >= 0; i-- ) // parse shaders in reverse order
{
Com_sprintf( filename, sizeof(filename), "%s/%s", path, shaderFiles[i] );
ri->Printf( PRINT_DEVELOPER, "...loading '%s'\n", filename );
ri->FS_ReadFile( filename, (void**)&buffer );
if( !buffer ) Com_Error( ERR_DROP, "Couldn't load %s", filename );
LoadShaderFromBuffer( buffer ); // extract and index all shaders from the buffer
ri->FS_FreeFile( buffer );
fileShaderCount++;
}
// free up memory
ri->FS_FreeFileList( shaderFiles );
return;
}
// !drakkar
#else
/*
====================
ScanAndLoadShaderFiles
Finds and loads all .shader files, combining them into
a single large text block that can be scanned for shader names
rww - Do not access any ri or render data stuff that doesn't get init'd on
a dedicated server in here. I am calling it for dedicateds now because
we need all this shader BS for looking up surface indicators in skin
files if we want to be like SP.
bto (VV) - Rather than keeping all the buffer pointers around forever and
creating more bugs, do the hash creation with the finalized shadertext.
Previous code only really worked if ri->FS_ReadFile returned contiguous buffers
in ascending order on consecutive calls.
=====================
*/
#define MAX_SHADER_FILES 4096
static void ScanAndLoadShaderFiles( const char *path )
{
char **shaderFiles;
char *buffers[MAX_SHADER_FILES];
char *p;
int numShaderFiles;
int i;
char *oldp, *token, *hashMem, *textEnd;
int shaderTextHashTableSizes[MAX_SHADERTEXT_HASH], hash, size;
long sum = 0, summand;
// scan for shader files
shaderFiles = ri->FS_ListFiles( path, ".shader", &numShaderFiles );
if ( !shaderFiles || !numShaderFiles )
{
Com_Error(ERR_FATAL, "ERROR: no shader files found\n");
return;
}
if ( numShaderFiles > MAX_SHADER_FILES ) {
numShaderFiles = MAX_SHADER_FILES;
}
// load and parse shader files
for ( i = 0; i < numShaderFiles; i++ )
{
char filename[MAX_QPATH];
Com_sprintf( filename, sizeof( filename ), "%s/%s", path, shaderFiles[i] );
ri->Printf( PRINT_DEVELOPER, "...loading '%s'\n", filename );
summand = ri->FS_ReadFile( filename, (void **)&buffers[i] );
if ( !buffers[i] ) {
ri->Com_Error( ERR_DROP, "Couldn't load %s", filename );
}
// Do a simple check on the shader structure in that file to make sure one bad shader file cannot fuck up all other shaders.
p = buffers[i];
while(1)
{
token = Shader_ParseExt((const char **)&p, qtrue);
if(!*token)
break;
oldp = p;
token = Shader_ParseExt((const char **)&p, qtrue);
if(token[0] != '{' && token[1] != '\0')
{
ri->Printf( PRINT_ALL, S_COLOR_YELLOW "WARNING: Bad shader file %s has incorrect syntax.\n", filename);
ri->FS_FreeFile(buffers[i]);
buffers[i] = NULL;
break;
}
Shader_SkipBracedSection((const char **)&oldp);
p = oldp;
}
if (buffers[i])
sum += summand;
}
// build single large buffer
s_shaderText = (char *)ri->Hunk_Alloc( sum + numShaderFiles*2, h_low );
s_shaderText[ 0 ] = '\0';
textEnd = s_shaderText;
// free in reverse order, so the temp files are all dumped
for ( i = numShaderFiles - 1; i >= 0 ; i-- )
{
if ( !buffers[i] )
continue;
strcat( textEnd, buffers[i] );
strcat( textEnd, "\n" );
textEnd += strlen( textEnd );
ri->FS_FreeFile( buffers[i] );
}
Shader_Compress( s_shaderText );
// free up memory
ri->FS_FreeFileList( shaderFiles );
memset(shaderTextHashTableSizes, 0, sizeof(shaderTextHashTableSizes));
size = 0;
p = s_shaderText;
// look for label
while ( 1 ) {
token = Shader_ParseExt( (const char **)&p, qtrue );
if ( token[0] == 0 ) {
break;
}
hash = generateHashValue(token, MAX_SHADERTEXT_HASH);
shaderTextHashTableSizes[hash]++;
size++;
Shader_SkipBracedSection((const char **)&p);
}
size += MAX_SHADERTEXT_HASH;
hashMem = (char *)ri->Hunk_Alloc( size * sizeof(char *), h_low );
for (i = 0; i < MAX_SHADERTEXT_HASH; i++) {
shaderTextHashTable[i] = (char **) hashMem;
hashMem = ((char *) hashMem) + ((shaderTextHashTableSizes[i] + 1) * sizeof(char *));
}
memset(shaderTextHashTableSizes, 0, sizeof(shaderTextHashTableSizes));
p = s_shaderText;
// look for label
while ( 1 ) {
oldp = p;
token = Shader_ParseExt( (const char **)&p, qtrue );
if ( token[0] == 0 ) {
break;
}
hash = generateHashValue(token, MAX_SHADERTEXT_HASH);
shaderTextHashTable[hash][shaderTextHashTableSizes[hash]++] = oldp;
Shader_SkipBracedSection((const char **)&p);
}
return;
}
#endif
/*
====================
R_CreateBlendedShader
This takes 4 shaders (one per corner of a quad) and creates a blended shader the fades the textures over
eg.
if [A][A]
[B][B]
then the shader would be texture A at the top fading to texture B at the bottom
This is highly biased towards terrain shaders ie vertex lit surfaces
====================
*/
//rwwRMG: Added:
static void R_CopyStage(shaderStage_t *orig, shaderStage_t *stage)
{
// Assumption: this stage has not been collapsed
*stage = *orig; //Just copy the whole thing!
}
static void R_CreateBlendedStage(qhandle_t handle, int idx)
{
shader_t *work;
work = R_GetShaderByHandle(handle);
R_CopyStage(work->stages, stages + idx);
stages[idx].rgbGen = CGEN_EXACT_VERTEX;
stages[idx].alphaGen = AGEN_BLEND;
stages[idx].stateBits = GLS_SRCBLEND_SRC_ALPHA | GLS_DSTBLEND_ONE | GLS_DEPTHMASK_TRUE;
if (stages[idx].ss)
{
stages[idx].ss->density *= 0.33f;
}
}
static qhandle_t R_MergeShaders(const char *blendedName, qhandle_t a, qhandle_t b, qhandle_t c, bool surfaceSprites)
{
shader_t *blended;
shader_t *work;
int current, i;
R_IssuePendingRenderCommands();
// Set up default parameters
ClearGlobalShader();
Q_strncpyz(shader.name, blendedName, sizeof(shader.name));
memcpy(shader.lightmapIndex, lightmapsVertex, sizeof(shader.lightmapIndex));
memcpy(shader.styles, stylesDefault, sizeof(shader.styles));
shader.fogPass = FP_EQUAL;
// Get the top left shader and set it up as pass 0 - it should be completely opaque
work = R_GetShaderByHandle(c);
stages[0].active = qtrue;
R_CopyStage(work->stages, stages);
stages[0].rgbGen = CGEN_EXACT_VERTEX;
stages[0].alphaGen = AGEN_BLEND;
stages[0].stateBits = GLS_SRCBLEND_SRC_ALPHA | GLS_DSTBLEND_ZERO | GLS_DEPTHMASK_TRUE;
shader.multitextureEnv = work->multitextureEnv; //jic
// Go through the other verts and add a pass
R_CreateBlendedStage(a, 1);
R_CreateBlendedStage(b, 2);
if ( surfaceSprites )
{
current = 3;
work = R_GetShaderByHandle(a);
for(i=1;(i<work->numUnfoggedPasses && current<MAX_SHADER_STAGES);i++)
{
if (work->stages[i].ss)
{
stages[current] = work->stages[i];
// stages[current].ss->density *= 0.33f;
stages[current].ss->density *= 3;
current++;
}
}
work = R_GetShaderByHandle(b);
for(i=1;(i<work->numUnfoggedPasses && current<MAX_SHADER_STAGES);i++)
{
if (work->stages[i].ss)
{
stages[current] = work->stages[i];
// stages[current].ss->density *= 0.33f;
stages[current].ss->density *= 3;
current++;
}
}
work = R_GetShaderByHandle(c);
for(i=1;(i<work->numUnfoggedPasses && current<MAX_SHADER_STAGES);i++)
{
if (work->stages[i].ss)
{
stages[current] = work->stages[i];
// stages[current].ss->density *= 0.33f;
stages[current].ss->density *= 3;
current++;
}
}
}
blended = FinishShader();
return(blended->index);
}
// Create a 3 pass shader - the last 2 passes are alpha'd out
qhandle_t R_CreateBlendedShader(qhandle_t a, qhandle_t b, qhandle_t c, bool surfaceSprites )
{
qhandle_t blended;
shader_t *work;
char blendedName[MAX_QPATH];
char extendedName[MAX_QPATH + MAX_QPATH];
Com_sprintf(blendedName, MAX_QPATH, "blend(%d,%d,%d)", a, b, c);
if (!surfaceSprites)
{
strcat(blendedName, "noSS");
}
// Find if this shader has already been created
R_CreateExtendedName(extendedName, sizeof(extendedName), blendedName, lightmapsVertex, stylesDefault);
work = hashTable[generateHashValue(extendedName, FILE_HASH_SIZE)];
for ( ; work; work = work->next)
{
if (Q_stricmp(work->name, extendedName) == 0)
{
return work->index;
}
}
// Create new shader if it doesn't already exist
blended = R_MergeShaders(extendedName, a, b, c, surfaceSprites);
return(blended);
}
/*
====================
CreateInternalShaders
====================
*/
static void CreateInternalShaders( void ) {
tr.numShaders = 0;
// init the default shader
memset( &shader, 0, sizeof( shader ) );
memset( &stages, 0, sizeof( stages ) );
Q_strncpyz( shader.name, "<default>", sizeof( shader.name ) );
memcpy(shader.lightmapIndex, lightmapsNone, sizeof(shader.lightmapIndex));
memcpy(shader.styles, stylesDefault, sizeof(shader.styles));
for ( int i = 0 ; i < MAX_SHADER_STAGES ; i++ ) {
stages[i].bundle[0].texMods = texMods[i];
}
stages[0].bundle[0].image = tr.defaultImage;
stages[0].active = qtrue;
stages[0].stateBits = GLS_DEFAULT;
tr.defaultShader = FinishShader();
// shadow shader is just a marker
Q_strncpyz( shader.name, "<stencil shadow>", sizeof( shader.name ) );
shader.sort = SS_BANNER; //SS_STENCIL_SHADOW;
tr.shadowShader = FinishShader();
// distortion shader is just a marker
Q_strncpyz( shader.name, "internal_distortion", sizeof( shader.name ) );
shader.sort = SS_BLEND0;
shader.defaultShader = qfalse;
tr.distortionShader = FinishShader();
shader.defaultShader = qtrue;
#ifndef HAVE_GLES
#define GL_PROGRAM_ERROR_STRING_ARB 0x8874
#define GL_PROGRAM_ERROR_POSITION_ARB 0x864B
// Allocate and Load the global 'Glow' Vertex Program. - AReis
if ( qglGenProgramsARB )
{
qglGenProgramsARB( 1, &tr.glowVShader );
qglBindProgramARB( GL_VERTEX_PROGRAM_ARB, tr.glowVShader );
qglProgramStringARB( GL_VERTEX_PROGRAM_ARB, GL_PROGRAM_FORMAT_ASCII_ARB, strlen( ( char * ) g_strGlowVShaderARB ), g_strGlowVShaderARB );
// const GLubyte *strErr = qglGetString( GL_PROGRAM_ERROR_STRING_ARB );
int iErrPos = 0;
qglGetIntegerv( GL_PROGRAM_ERROR_POSITION_ARB, &iErrPos );
assert( iErrPos == -1 );
}
// NOTE: I make an assumption here. If you have (current) nvidia hardware, you obviously support register combiners instead of fragment
// programs, so use those. The problem with this is that nv30 WILL support fragment shaders, breaking this logic. The good thing is that
// if you always ask for regcoms before fragment shaders, you'll always just use regcoms (problem solved... for now). - AReis
// Load Pixel Shaders (either regcoms or fragprogs).
if ( qglCombinerParameteriNV )
{
// The purpose of this regcom is to blend all the pixels together from the 4 texture units, but with their
// texture coordinates offset by 1 (or more) texels, effectively letting us blend adjoining pixels. The weight is
// used to either strengthen or weaken the pixel intensity. The more it diffuses (the higher the radius of the glow),
// the higher the intensity should be for a noticable effect.
// Regcom result is: ( tex1 * fBlurWeight ) + ( tex2 * fBlurWeight ) + ( tex2 * fBlurWeight ) + ( tex2 * fBlurWeight )
// VV guys, this is the pixel shader you would use instead :-)
/*
// c0 is the blur weight.
ps 1.1
tex t0
tex t1
tex t2
tex t3
mul r0, c0, t0;
madd r0, c0, t1, r0;
madd r0, c0, t2, r0;
madd r0, c0, t3, r0;
*/
tr.glowPShader = qglGenLists( 1 );
qglNewList( tr.glowPShader, GL_COMPILE );
qglCombinerParameteriNV( GL_NUM_GENERAL_COMBINERS_NV, 2 );
// spare0 = fBlend * tex0 + fBlend * tex1.
qglCombinerInputNV( GL_COMBINER0_NV, GL_RGB, GL_VARIABLE_A_NV, GL_TEXTURE0_ARB, GL_UNSIGNED_IDENTITY_NV, GL_RGB );
qglCombinerInputNV( GL_COMBINER0_NV, GL_RGB, GL_VARIABLE_B_NV, GL_CONSTANT_COLOR0_NV, GL_UNSIGNED_IDENTITY_NV, GL_RGB );
qglCombinerInputNV( GL_COMBINER0_NV, GL_RGB, GL_VARIABLE_C_NV, GL_TEXTURE1_ARB, GL_UNSIGNED_IDENTITY_NV, GL_RGB );
qglCombinerInputNV( GL_COMBINER0_NV, GL_RGB, GL_VARIABLE_D_NV, GL_CONSTANT_COLOR0_NV, GL_UNSIGNED_IDENTITY_NV, GL_RGB );
qglCombinerOutputNV( GL_COMBINER0_NV, GL_RGB, GL_DISCARD_NV, GL_DISCARD_NV, GL_SPARE0_NV, GL_NONE, GL_NONE, GL_FALSE, GL_FALSE, GL_FALSE );
// spare1 = fBlend * tex2 + fBlend * tex3.
qglCombinerInputNV( GL_COMBINER1_NV, GL_RGB, GL_VARIABLE_A_NV, GL_TEXTURE2_ARB, GL_UNSIGNED_IDENTITY_NV, GL_RGB );
qglCombinerInputNV( GL_COMBINER1_NV, GL_RGB, GL_VARIABLE_B_NV, GL_CONSTANT_COLOR0_NV, GL_UNSIGNED_IDENTITY_NV, GL_RGB );
qglCombinerInputNV( GL_COMBINER1_NV, GL_RGB, GL_VARIABLE_C_NV, GL_TEXTURE3_ARB, GL_UNSIGNED_IDENTITY_NV, GL_RGB );
qglCombinerInputNV( GL_COMBINER1_NV, GL_RGB, GL_VARIABLE_D_NV, GL_CONSTANT_COLOR0_NV, GL_UNSIGNED_IDENTITY_NV, GL_RGB );
qglCombinerOutputNV( GL_COMBINER1_NV, GL_RGB, GL_DISCARD_NV, GL_DISCARD_NV, GL_SPARE1_NV, GL_NONE, GL_NONE, GL_FALSE, GL_FALSE, GL_FALSE );
// ( A * B ) + ( ( 1 - A ) * C ) + D = ( spare0 * 1 ) + ( ( 1 - spare0 ) * 0 ) + spare1 == spare0 + spare1.
qglFinalCombinerInputNV( GL_VARIABLE_A_NV, GL_SPARE0_NV, GL_UNSIGNED_IDENTITY_NV, GL_RGB );
qglFinalCombinerInputNV( GL_VARIABLE_B_NV, GL_ZERO, GL_UNSIGNED_INVERT_NV, GL_RGB );
qglFinalCombinerInputNV( GL_VARIABLE_C_NV, GL_ZERO, GL_UNSIGNED_IDENTITY_NV, GL_RGB );
qglFinalCombinerInputNV( GL_VARIABLE_D_NV, GL_SPARE1_NV, GL_UNSIGNED_IDENTITY_NV, GL_RGB );
qglEndList();
}
else if ( qglGenProgramsARB )
{
qglGenProgramsARB( 1, &tr.glowPShader );
qglBindProgramARB( GL_FRAGMENT_PROGRAM_ARB, tr.glowPShader );
qglProgramStringARB( GL_FRAGMENT_PROGRAM_ARB, GL_PROGRAM_FORMAT_ASCII_ARB, strlen( ( char * ) g_strGlowPShaderARB ), g_strGlowPShaderARB );
// const GLubyte *strErr = qglGetString( GL_PROGRAM_ERROR_STRING_ARB );
int iErrPos = 0;
qglGetIntegerv( GL_PROGRAM_ERROR_POSITION_ARB, &iErrPos );
assert( iErrPos == -1 );
}
#endif //HAVE_GLES
}
static void CreateExternalShaders( void ) {
tr.projectionShadowShader = R_FindShader( "projectionShadow", lightmapsNone, stylesDefault, qtrue );
tr.projectionShadowShader->sort = SS_STENCIL_SHADOW;
tr.sunShader = R_FindShader( "sun", lightmapsNone, stylesDefault, qtrue );
}
/*
==================
R_InitShaders
==================
*/
void R_InitShaders(qboolean server)
{
//ri->Printf( PRINT_ALL, "Initializing Shaders\n" );
#if defined(USE_NEW_SHADER_HASH) && !(DEDICATED)
int time, mem;
// drakkar - profiling shader parse session
if (!server)
{
Shader_BeginParseSession( "R_InitShaders" );
time = ri->Milliseconds()*ri->Cvar_VariableValue( "timescale" );
mem = Hunk_MemoryRemaining();
fileShaderCount = 0;
shaderCount = 0;
}
// !drakkar
#endif
memset(hashTable, 0, sizeof(hashTable));
#ifdef USE_NEW_SHADER_HASH
memset(shaderTextHashTable, 0, sizeof(shaderTextHashTable)); // drakkar - clear shader hashtable
#endif
deferLoad = qfalse;
if (!server)
{
CreateInternalShaders();
ScanAndLoadShaderFiles("shaders");
CreateExternalShaders();
}
#if defined(USE_NEW_SHADER_HASH) && !(DEDICATED)
// drakkar - print profiling info
if (!server)
{
time = ri->Milliseconds()*ri->Cvar_VariableValue( "timescale" ) - time;
mem = mem - Hunk_MemoryRemaining();
ri->Printf( PRINT_ALL, "-------------------------\n" );
ri->Printf( PRINT_ALL, "%d shader files read \n", fileShaderCount );
ri->Printf( PRINT_ALL, "%d shaders found\n", shaderCount );
ri->Printf( PRINT_ALL, "%d code lines\n", Shader_GetCurrentParseLine() );
ri->Printf( PRINT_ALL, "%.2f MB shader data\n", mem/1024.0f/1024.0f );
ri->Printf( PRINT_ALL, "%.3f seconds\n", time/1000.0f );
ri->Printf( PRINT_ALL, "-------------------------\n" );
Shader_BeginParseSession( "" );
}
// !drakkar
#endif
}