mirror of
https://github.com/dhewm/dhewm3.git
synced 2025-01-05 17:30:54 +00:00
736ec20d4d
Don't include the lazy precompiled.h everywhere, only what's required for the compilation unit. platform.h needs to be included instead to provide all essential defines and types. All includes use the relative path to the neo or the game specific root. Move all idlib related includes from idlib/Lib.h to precompiled.h. precompiled.h still exists for the MFC stuff in tools/. Add some missing header guards.
2737 lines
71 KiB
C++
2737 lines
71 KiB
C++
/*
|
|
===========================================================================
|
|
|
|
Doom 3 GPL Source Code
|
|
Copyright (C) 1999-2011 id Software LLC, a ZeniMax Media company.
|
|
|
|
This file is part of the Doom 3 GPL Source Code ("Doom 3 Source Code").
|
|
|
|
Doom 3 Source Code is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
Doom 3 Source Code is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with Doom 3 Source Code. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
In addition, the Doom 3 Source Code is also subject to certain additional terms. You should have received a copy of these additional terms immediately following the terms and conditions of the GNU General Public License which accompanied the Doom 3 Source Code. If not, please request a copy in writing from id Software at the address below.
|
|
|
|
If you have questions concerning this license or the applicable additional terms, you may contact in writing id Software LLC, c/o ZeniMax Media Inc., Suite 120, Rockville, Maryland 20850 USA.
|
|
|
|
===========================================================================
|
|
*/
|
|
|
|
#include "sys/platform.h"
|
|
#include "idlib/math/Interpolate.h"
|
|
#include "renderer/Cinematic.h"
|
|
#include "renderer/tr_local.h"
|
|
#include "ui/Window.h"
|
|
#include "ui/UserInterface.h"
|
|
#include "sound/sound.h"
|
|
|
|
#include "renderer/Material.h"
|
|
|
|
/*
|
|
|
|
Any errors during parsing just set MF_DEFAULTED and return, rather than throwing
|
|
a hard error. This will cause the material to fall back to default material,
|
|
but otherwise let things continue.
|
|
|
|
Each material may have a set of calculations that must be evaluated before
|
|
drawing with it.
|
|
|
|
Every expression that a material uses can be evaluated at one time, which
|
|
will allow for perfect common subexpression removal when I get around to
|
|
writing it.
|
|
|
|
Without this, scrolling an entire surface could result in evaluating the
|
|
same texture matrix calculations a half dozen times.
|
|
|
|
Open question: should I allow arbitrary per-vertex color, texCoord, and vertex
|
|
calculations to be specified in the material code?
|
|
|
|
Every stage will definately have a valid image pointer.
|
|
|
|
We might want the ability to change the sort value based on conditionals,
|
|
but it could be a hassle to implement,
|
|
|
|
*/
|
|
|
|
// keep all of these on the stack, when they are static it makes material parsing non-reentrant
|
|
typedef struct mtrParsingData_s {
|
|
bool registerIsTemporary[MAX_EXPRESSION_REGISTERS];
|
|
float shaderRegisters[MAX_EXPRESSION_REGISTERS];
|
|
expOp_t shaderOps[MAX_EXPRESSION_OPS];
|
|
shaderStage_t parseStages[MAX_SHADER_STAGES];
|
|
|
|
bool registersAreConstant;
|
|
bool forceOverlays;
|
|
} mtrParsingData_t;
|
|
|
|
|
|
/*
|
|
=============
|
|
idMaterial::CommonInit
|
|
=============
|
|
*/
|
|
void idMaterial::CommonInit() {
|
|
desc = "<none>";
|
|
renderBump = "";
|
|
contentFlags = CONTENTS_SOLID;
|
|
surfaceFlags = SURFTYPE_NONE;
|
|
materialFlags = 0;
|
|
sort = SS_BAD;
|
|
coverage = MC_BAD;
|
|
cullType = CT_FRONT_SIDED;
|
|
deform = DFRM_NONE;
|
|
numOps = 0;
|
|
ops = NULL;
|
|
numRegisters = 0;
|
|
expressionRegisters = NULL;
|
|
constantRegisters = NULL;
|
|
numStages = 0;
|
|
numAmbientStages = 0;
|
|
stages = NULL;
|
|
editorImage = NULL;
|
|
lightFalloffImage = NULL;
|
|
shouldCreateBackSides = false;
|
|
entityGui = 0;
|
|
fogLight = false;
|
|
blendLight = false;
|
|
ambientLight = false;
|
|
noFog = false;
|
|
hasSubview = false;
|
|
allowOverlays = true;
|
|
unsmoothedTangents = false;
|
|
gui = NULL;
|
|
memset( deformRegisters, 0, sizeof( deformRegisters ) );
|
|
editorAlpha = 1.0;
|
|
spectrum = 0;
|
|
polygonOffset = 0;
|
|
suppressInSubview = false;
|
|
refCount = 0;
|
|
portalSky = false;
|
|
|
|
decalInfo.stayTime = 10000;
|
|
decalInfo.fadeTime = 4000;
|
|
decalInfo.start[0] = 1;
|
|
decalInfo.start[1] = 1;
|
|
decalInfo.start[2] = 1;
|
|
decalInfo.start[3] = 1;
|
|
decalInfo.end[0] = 0;
|
|
decalInfo.end[1] = 0;
|
|
decalInfo.end[2] = 0;
|
|
decalInfo.end[3] = 0;
|
|
}
|
|
|
|
/*
|
|
=============
|
|
idMaterial::idMaterial
|
|
=============
|
|
*/
|
|
idMaterial::idMaterial() {
|
|
CommonInit();
|
|
|
|
// we put this here instead of in CommonInit, because
|
|
// we don't want it cleared when a material is purged
|
|
surfaceArea = 0;
|
|
}
|
|
|
|
/*
|
|
=============
|
|
idMaterial::~idMaterial
|
|
=============
|
|
*/
|
|
idMaterial::~idMaterial() {
|
|
}
|
|
|
|
/*
|
|
===============
|
|
idMaterial::FreeData
|
|
===============
|
|
*/
|
|
void idMaterial::FreeData() {
|
|
int i;
|
|
|
|
if ( stages ) {
|
|
// delete any idCinematic textures
|
|
for ( i = 0; i < numStages; i++ ) {
|
|
if ( stages[i].texture.cinematic != NULL ) {
|
|
delete stages[i].texture.cinematic;
|
|
stages[i].texture.cinematic = NULL;
|
|
}
|
|
if ( stages[i].newStage != NULL ) {
|
|
Mem_Free( stages[i].newStage );
|
|
stages[i].newStage = NULL;
|
|
}
|
|
}
|
|
R_StaticFree( stages );
|
|
stages = NULL;
|
|
}
|
|
if ( expressionRegisters != NULL ) {
|
|
R_StaticFree( expressionRegisters );
|
|
expressionRegisters = NULL;
|
|
}
|
|
if ( constantRegisters != NULL ) {
|
|
R_StaticFree( constantRegisters );
|
|
constantRegisters = NULL;
|
|
}
|
|
if ( ops != NULL ) {
|
|
R_StaticFree( ops );
|
|
ops = NULL;
|
|
}
|
|
}
|
|
|
|
/*
|
|
==============
|
|
idMaterial::GetEditorImage
|
|
==============
|
|
*/
|
|
idImage *idMaterial::GetEditorImage( void ) const {
|
|
if ( editorImage ) {
|
|
return editorImage;
|
|
}
|
|
|
|
// if we don't have an editorImageName, use the first stage image
|
|
if ( !editorImageName.Length()) {
|
|
// _D3XP :: First check for a diffuse image, then use the first
|
|
if ( numStages && stages ) {
|
|
int i;
|
|
for( i = 0; i < numStages; i++ ) {
|
|
if ( stages[i].lighting == SL_DIFFUSE ) {
|
|
editorImage = stages[i].texture.image;
|
|
break;
|
|
}
|
|
}
|
|
if ( !editorImage ) {
|
|
editorImage = stages[0].texture.image;
|
|
}
|
|
} else {
|
|
editorImage = globalImages->defaultImage;
|
|
}
|
|
} else {
|
|
// look for an explicit one
|
|
editorImage = globalImages->ImageFromFile( editorImageName, TF_DEFAULT, true, TR_REPEAT, TD_DEFAULT );
|
|
}
|
|
|
|
if ( !editorImage ) {
|
|
editorImage = globalImages->defaultImage;
|
|
}
|
|
|
|
return editorImage;
|
|
}
|
|
|
|
|
|
// info parms
|
|
typedef struct {
|
|
const char *name;
|
|
int clearSolid, surfaceFlags, contents;
|
|
} infoParm_t;
|
|
|
|
static const infoParm_t infoParms[] = {
|
|
// game relevant attributes
|
|
{"solid", 0, 0, CONTENTS_SOLID }, // may need to override a clearSolid
|
|
{"water", 1, 0, CONTENTS_WATER }, // used for water
|
|
{"playerclip", 0, 0, CONTENTS_PLAYERCLIP }, // solid to players
|
|
{"monsterclip", 0, 0, CONTENTS_MONSTERCLIP }, // solid to monsters
|
|
{"moveableclip",0, 0, CONTENTS_MOVEABLECLIP },// solid to moveable entities
|
|
{"ikclip", 0, 0, CONTENTS_IKCLIP }, // solid to IK
|
|
{"blood", 0, 0, CONTENTS_BLOOD }, // used to detect blood decals
|
|
{"trigger", 0, 0, CONTENTS_TRIGGER }, // used for triggers
|
|
{"aassolid", 0, 0, CONTENTS_AAS_SOLID }, // solid for AAS
|
|
{"aasobstacle", 0, 0, CONTENTS_AAS_OBSTACLE },// used to compile an obstacle into AAS that can be enabled/disabled
|
|
{"flashlight_trigger", 0, 0, CONTENTS_FLASHLIGHT_TRIGGER }, // used for triggers that are activated by the flashlight
|
|
{"nonsolid", 1, 0, 0 }, // clears the solid flag
|
|
{"nullNormal", 0, SURF_NULLNORMAL,0 }, // renderbump will draw as 0x80 0x80 0x80
|
|
|
|
// utility relevant attributes
|
|
{"areaportal", 1, 0, CONTENTS_AREAPORTAL }, // divides areas
|
|
{"qer_nocarve", 1, 0, CONTENTS_NOCSG}, // don't cut brushes in editor
|
|
|
|
{"discrete", 1, SURF_DISCRETE, 0 }, // surfaces should not be automatically merged together or
|
|
// clipped to the world,
|
|
// because they represent discrete objects like gui shaders
|
|
// mirrors, or autosprites
|
|
{"noFragment", 0, SURF_NOFRAGMENT, 0 },
|
|
|
|
{"slick", 0, SURF_SLICK, 0 },
|
|
{"collision", 0, SURF_COLLISION, 0 },
|
|
{"noimpact", 0, SURF_NOIMPACT, 0 }, // don't make impact explosions or marks
|
|
{"nodamage", 0, SURF_NODAMAGE, 0 }, // no falling damage when hitting
|
|
{"ladder", 0, SURF_LADDER, 0 }, // climbable
|
|
{"nosteps", 0, SURF_NOSTEPS, 0 }, // no footsteps
|
|
|
|
// material types for particle, sound, footstep feedback
|
|
{"metal", 0, SURFTYPE_METAL, 0 }, // metal
|
|
{"stone", 0, SURFTYPE_STONE, 0 }, // stone
|
|
{"flesh", 0, SURFTYPE_FLESH, 0 }, // flesh
|
|
{"wood", 0, SURFTYPE_WOOD, 0 }, // wood
|
|
{"cardboard", 0, SURFTYPE_CARDBOARD, 0 }, // cardboard
|
|
{"liquid", 0, SURFTYPE_LIQUID, 0 }, // liquid
|
|
{"glass", 0, SURFTYPE_GLASS, 0 }, // glass
|
|
{"plastic", 0, SURFTYPE_PLASTIC, 0 }, // plastic
|
|
{"ricochet", 0, SURFTYPE_RICOCHET, 0 }, // behaves like metal but causes a ricochet sound
|
|
|
|
// unassigned surface types
|
|
{"surftype10", 0, SURFTYPE_10, 0 },
|
|
{"surftype11", 0, SURFTYPE_11, 0 },
|
|
{"surftype12", 0, SURFTYPE_12, 0 },
|
|
{"surftype13", 0, SURFTYPE_13, 0 },
|
|
{"surftype14", 0, SURFTYPE_14, 0 },
|
|
{"surftype15", 0, SURFTYPE_15, 0 },
|
|
};
|
|
|
|
static const int numInfoParms = sizeof(infoParms) / sizeof (infoParms[0]);
|
|
|
|
|
|
/*
|
|
===============
|
|
idMaterial::CheckSurfaceParm
|
|
|
|
See if the current token matches one of the surface parm bit flags
|
|
===============
|
|
*/
|
|
bool idMaterial::CheckSurfaceParm( idToken *token ) {
|
|
|
|
for ( int i = 0 ; i < numInfoParms ; i++ ) {
|
|
if ( !token->Icmp( infoParms[i].name ) ) {
|
|
if ( infoParms[i].surfaceFlags & SURF_TYPE_MASK ) {
|
|
// ensure we only have one surface type set
|
|
surfaceFlags &= ~SURF_TYPE_MASK;
|
|
}
|
|
surfaceFlags |= infoParms[i].surfaceFlags;
|
|
contentFlags |= infoParms[i].contents;
|
|
if ( infoParms[i].clearSolid ) {
|
|
contentFlags &= ~CONTENTS_SOLID;
|
|
}
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/*
|
|
===============
|
|
idMaterial::MatchToken
|
|
|
|
Sets defaultShader and returns false if the next token doesn't match
|
|
===============
|
|
*/
|
|
bool idMaterial::MatchToken( idLexer &src, const char *match ) {
|
|
if ( !src.ExpectTokenString( match ) ) {
|
|
SetMaterialFlag( MF_DEFAULTED );
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
=================
|
|
idMaterial::ParseSort
|
|
=================
|
|
*/
|
|
void idMaterial::ParseSort( idLexer &src ) {
|
|
idToken token;
|
|
|
|
if ( !src.ReadTokenOnLine( &token ) ) {
|
|
src.Warning( "missing sort parameter" );
|
|
SetMaterialFlag( MF_DEFAULTED );
|
|
return;
|
|
}
|
|
|
|
if ( !token.Icmp( "subview" ) ) {
|
|
sort = SS_SUBVIEW;
|
|
} else if ( !token.Icmp( "opaque" ) ) {
|
|
sort = SS_OPAQUE;
|
|
}else if ( !token.Icmp( "decal" ) ) {
|
|
sort = SS_DECAL;
|
|
} else if ( !token.Icmp( "far" ) ) {
|
|
sort = SS_FAR;
|
|
} else if ( !token.Icmp( "medium" ) ) {
|
|
sort = SS_MEDIUM;
|
|
} else if ( !token.Icmp( "close" ) ) {
|
|
sort = SS_CLOSE;
|
|
} else if ( !token.Icmp( "almostNearest" ) ) {
|
|
sort = SS_ALMOST_NEAREST;
|
|
} else if ( !token.Icmp( "nearest" ) ) {
|
|
sort = SS_NEAREST;
|
|
} else if ( !token.Icmp( "postProcess" ) ) {
|
|
sort = SS_POST_PROCESS;
|
|
} else if ( !token.Icmp( "portalSky" ) ) {
|
|
sort = SS_PORTAL_SKY;
|
|
} else {
|
|
sort = atof( token );
|
|
}
|
|
}
|
|
|
|
/*
|
|
=================
|
|
idMaterial::ParseDecalInfo
|
|
=================
|
|
*/
|
|
void idMaterial::ParseDecalInfo( idLexer &src ) {
|
|
idToken token;
|
|
|
|
decalInfo.stayTime = src.ParseFloat() * 1000;
|
|
decalInfo.fadeTime = src.ParseFloat() * 1000;
|
|
float start[4], end[4];
|
|
src.Parse1DMatrix( 4, start );
|
|
src.Parse1DMatrix( 4, end );
|
|
for ( int i = 0 ; i < 4 ; i++ ) {
|
|
decalInfo.start[i] = start[i];
|
|
decalInfo.end[i] = end[i];
|
|
}
|
|
}
|
|
|
|
/*
|
|
=============
|
|
idMaterial::GetExpressionConstant
|
|
=============
|
|
*/
|
|
int idMaterial::GetExpressionConstant( float f ) {
|
|
int i;
|
|
|
|
for ( i = EXP_REG_NUM_PREDEFINED ; i < numRegisters ; i++ ) {
|
|
if ( !pd->registerIsTemporary[i] && pd->shaderRegisters[i] == f ) {
|
|
return i;
|
|
}
|
|
}
|
|
if ( numRegisters == MAX_EXPRESSION_REGISTERS ) {
|
|
common->Warning( "GetExpressionConstant: material '%s' hit MAX_EXPRESSION_REGISTERS", GetName() );
|
|
SetMaterialFlag( MF_DEFAULTED );
|
|
return 0;
|
|
}
|
|
pd->registerIsTemporary[i] = false;
|
|
pd->shaderRegisters[i] = f;
|
|
numRegisters++;
|
|
|
|
return i;
|
|
}
|
|
|
|
/*
|
|
=============
|
|
idMaterial::GetExpressionTemporary
|
|
=============
|
|
*/
|
|
int idMaterial::GetExpressionTemporary( void ) {
|
|
if ( numRegisters == MAX_EXPRESSION_REGISTERS ) {
|
|
common->Warning( "GetExpressionTemporary: material '%s' hit MAX_EXPRESSION_REGISTERS", GetName() );
|
|
SetMaterialFlag( MF_DEFAULTED );
|
|
return 0;
|
|
}
|
|
pd->registerIsTemporary[numRegisters] = true;
|
|
numRegisters++;
|
|
return numRegisters - 1;
|
|
}
|
|
|
|
/*
|
|
=============
|
|
idMaterial::GetExpressionOp
|
|
=============
|
|
*/
|
|
expOp_t *idMaterial::GetExpressionOp( void ) {
|
|
if ( numOps == MAX_EXPRESSION_OPS ) {
|
|
common->Warning( "GetExpressionOp: material '%s' hit MAX_EXPRESSION_OPS", GetName() );
|
|
SetMaterialFlag( MF_DEFAULTED );
|
|
return &pd->shaderOps[0];
|
|
}
|
|
|
|
return &pd->shaderOps[numOps++];
|
|
}
|
|
|
|
/*
|
|
=================
|
|
idMaterial::EmitOp
|
|
=================
|
|
*/
|
|
int idMaterial::EmitOp( int a, int b, expOpType_t opType ) {
|
|
expOp_t *op;
|
|
|
|
// optimize away identity operations
|
|
if ( opType == OP_TYPE_ADD ) {
|
|
if ( !pd->registerIsTemporary[a] && pd->shaderRegisters[a] == 0 ) {
|
|
return b;
|
|
}
|
|
if ( !pd->registerIsTemporary[b] && pd->shaderRegisters[b] == 0 ) {
|
|
return a;
|
|
}
|
|
if ( !pd->registerIsTemporary[a] && !pd->registerIsTemporary[b] ) {
|
|
return GetExpressionConstant( pd->shaderRegisters[a] + pd->shaderRegisters[b] );
|
|
}
|
|
}
|
|
if ( opType == OP_TYPE_MULTIPLY ) {
|
|
if ( !pd->registerIsTemporary[a] && pd->shaderRegisters[a] == 1 ) {
|
|
return b;
|
|
}
|
|
if ( !pd->registerIsTemporary[a] && pd->shaderRegisters[a] == 0 ) {
|
|
return a;
|
|
}
|
|
if ( !pd->registerIsTemporary[b] && pd->shaderRegisters[b] == 1 ) {
|
|
return a;
|
|
}
|
|
if ( !pd->registerIsTemporary[b] && pd->shaderRegisters[b] == 0 ) {
|
|
return b;
|
|
}
|
|
if ( !pd->registerIsTemporary[a] && !pd->registerIsTemporary[b] ) {
|
|
return GetExpressionConstant( pd->shaderRegisters[a] * pd->shaderRegisters[b] );
|
|
}
|
|
}
|
|
|
|
op = GetExpressionOp();
|
|
op->opType = opType;
|
|
op->a = a;
|
|
op->b = b;
|
|
op->c = GetExpressionTemporary();
|
|
|
|
return op->c;
|
|
}
|
|
|
|
/*
|
|
=================
|
|
idMaterial::ParseEmitOp
|
|
=================
|
|
*/
|
|
int idMaterial::ParseEmitOp( idLexer &src, int a, expOpType_t opType, int priority ) {
|
|
int b;
|
|
|
|
b = ParseExpressionPriority( src, priority );
|
|
return EmitOp( a, b, opType );
|
|
}
|
|
|
|
/*
|
|
=================
|
|
idMaterial::ParseTerm
|
|
|
|
Returns a register index
|
|
=================
|
|
*/
|
|
int idMaterial::ParseTerm( idLexer &src ) {
|
|
idToken token;
|
|
int a, b;
|
|
|
|
src.ReadToken( &token );
|
|
|
|
if ( token == "(" ) {
|
|
a = ParseExpression( src );
|
|
MatchToken( src, ")" );
|
|
return a;
|
|
}
|
|
|
|
if ( !token.Icmp( "time" ) ) {
|
|
pd->registersAreConstant = false;
|
|
return EXP_REG_TIME;
|
|
}
|
|
if ( !token.Icmp( "parm0" ) ) {
|
|
pd->registersAreConstant = false;
|
|
return EXP_REG_PARM0;
|
|
}
|
|
if ( !token.Icmp( "parm1" ) ) {
|
|
pd->registersAreConstant = false;
|
|
return EXP_REG_PARM1;
|
|
}
|
|
if ( !token.Icmp( "parm2" ) ) {
|
|
pd->registersAreConstant = false;
|
|
return EXP_REG_PARM2;
|
|
}
|
|
if ( !token.Icmp( "parm3" ) ) {
|
|
pd->registersAreConstant = false;
|
|
return EXP_REG_PARM3;
|
|
}
|
|
if ( !token.Icmp( "parm4" ) ) {
|
|
pd->registersAreConstant = false;
|
|
return EXP_REG_PARM4;
|
|
}
|
|
if ( !token.Icmp( "parm5" ) ) {
|
|
pd->registersAreConstant = false;
|
|
return EXP_REG_PARM5;
|
|
}
|
|
if ( !token.Icmp( "parm6" ) ) {
|
|
pd->registersAreConstant = false;
|
|
return EXP_REG_PARM6;
|
|
}
|
|
if ( !token.Icmp( "parm7" ) ) {
|
|
pd->registersAreConstant = false;
|
|
return EXP_REG_PARM7;
|
|
}
|
|
if ( !token.Icmp( "parm8" ) ) {
|
|
pd->registersAreConstant = false;
|
|
return EXP_REG_PARM8;
|
|
}
|
|
if ( !token.Icmp( "parm9" ) ) {
|
|
pd->registersAreConstant = false;
|
|
return EXP_REG_PARM9;
|
|
}
|
|
if ( !token.Icmp( "parm10" ) ) {
|
|
pd->registersAreConstant = false;
|
|
return EXP_REG_PARM10;
|
|
}
|
|
if ( !token.Icmp( "parm11" ) ) {
|
|
pd->registersAreConstant = false;
|
|
return EXP_REG_PARM11;
|
|
}
|
|
if ( !token.Icmp( "global0" ) ) {
|
|
pd->registersAreConstant = false;
|
|
return EXP_REG_GLOBAL0;
|
|
}
|
|
if ( !token.Icmp( "global1" ) ) {
|
|
pd->registersAreConstant = false;
|
|
return EXP_REG_GLOBAL1;
|
|
}
|
|
if ( !token.Icmp( "global2" ) ) {
|
|
pd->registersAreConstant = false;
|
|
return EXP_REG_GLOBAL2;
|
|
}
|
|
if ( !token.Icmp( "global3" ) ) {
|
|
pd->registersAreConstant = false;
|
|
return EXP_REG_GLOBAL3;
|
|
}
|
|
if ( !token.Icmp( "global4" ) ) {
|
|
pd->registersAreConstant = false;
|
|
return EXP_REG_GLOBAL4;
|
|
}
|
|
if ( !token.Icmp( "global5" ) ) {
|
|
pd->registersAreConstant = false;
|
|
return EXP_REG_GLOBAL5;
|
|
}
|
|
if ( !token.Icmp( "global6" ) ) {
|
|
pd->registersAreConstant = false;
|
|
return EXP_REG_GLOBAL6;
|
|
}
|
|
if ( !token.Icmp( "global7" ) ) {
|
|
pd->registersAreConstant = false;
|
|
return EXP_REG_GLOBAL7;
|
|
}
|
|
if ( !token.Icmp( "fragmentPrograms" ) ) {
|
|
return GetExpressionConstant( (float) glConfig.ARBFragmentProgramAvailable );
|
|
}
|
|
|
|
if ( !token.Icmp( "sound" ) ) {
|
|
pd->registersAreConstant = false;
|
|
return EmitOp( 0, 0, OP_TYPE_SOUND );
|
|
}
|
|
|
|
// parse negative numbers
|
|
if ( token == "-" ) {
|
|
src.ReadToken( &token );
|
|
if ( token.type == TT_NUMBER || token == "." ) {
|
|
return GetExpressionConstant( -(float) token.GetFloatValue() );
|
|
}
|
|
src.Warning( "Bad negative number '%s'", token.c_str() );
|
|
SetMaterialFlag( MF_DEFAULTED );
|
|
return 0;
|
|
}
|
|
|
|
if ( token.type == TT_NUMBER || token == "." || token == "-" ) {
|
|
return GetExpressionConstant( (float) token.GetFloatValue() );
|
|
}
|
|
|
|
// see if it is a table name
|
|
const idDeclTable *table = static_cast<const idDeclTable *>( declManager->FindType( DECL_TABLE, token.c_str(), false ) );
|
|
if ( !table ) {
|
|
src.Warning( "Bad term '%s'", token.c_str() );
|
|
SetMaterialFlag( MF_DEFAULTED );
|
|
return 0;
|
|
}
|
|
|
|
// parse a table expression
|
|
MatchToken( src, "[" );
|
|
|
|
b = ParseExpression( src );
|
|
|
|
MatchToken( src, "]" );
|
|
|
|
return EmitOp( table->Index(), b, OP_TYPE_TABLE );
|
|
}
|
|
|
|
/*
|
|
=================
|
|
idMaterial::ParseExpressionPriority
|
|
|
|
Returns a register index
|
|
=================
|
|
*/
|
|
#define TOP_PRIORITY 4
|
|
int idMaterial::ParseExpressionPriority( idLexer &src, int priority ) {
|
|
idToken token;
|
|
int a;
|
|
|
|
if ( priority == 0 ) {
|
|
return ParseTerm( src );
|
|
}
|
|
|
|
a = ParseExpressionPriority( src, priority - 1 );
|
|
|
|
if ( TestMaterialFlag( MF_DEFAULTED ) ) { // we have a parse error
|
|
return 0;
|
|
}
|
|
|
|
if ( !src.ReadToken( &token ) ) {
|
|
// we won't get EOF in a real file, but we can
|
|
// when parsing from generated strings
|
|
return a;
|
|
}
|
|
|
|
if ( priority == 1 && token == "*" ) {
|
|
return ParseEmitOp( src, a, OP_TYPE_MULTIPLY, priority );
|
|
}
|
|
if ( priority == 1 && token == "/" ) {
|
|
return ParseEmitOp( src, a, OP_TYPE_DIVIDE, priority );
|
|
}
|
|
if ( priority == 1 && token == "%" ) { // implied truncate both to integer
|
|
return ParseEmitOp( src, a, OP_TYPE_MOD, priority );
|
|
}
|
|
if ( priority == 2 && token == "+" ) {
|
|
return ParseEmitOp( src, a, OP_TYPE_ADD, priority );
|
|
}
|
|
if ( priority == 2 && token == "-" ) {
|
|
return ParseEmitOp( src, a, OP_TYPE_SUBTRACT, priority );
|
|
}
|
|
if ( priority == 3 && token == ">" ) {
|
|
return ParseEmitOp( src, a, OP_TYPE_GT, priority );
|
|
}
|
|
if ( priority == 3 && token == ">=" ) {
|
|
return ParseEmitOp( src, a, OP_TYPE_GE, priority );
|
|
}
|
|
if ( priority == 3 && token == "<" ) {
|
|
return ParseEmitOp( src, a, OP_TYPE_LT, priority );
|
|
}
|
|
if ( priority == 3 && token == "<=" ) {
|
|
return ParseEmitOp( src, a, OP_TYPE_LE, priority );
|
|
}
|
|
if ( priority == 3 && token == "==" ) {
|
|
return ParseEmitOp( src, a, OP_TYPE_EQ, priority );
|
|
}
|
|
if ( priority == 3 && token == "!=" ) {
|
|
return ParseEmitOp( src, a, OP_TYPE_NE, priority );
|
|
}
|
|
if ( priority == 4 && token == "&&" ) {
|
|
return ParseEmitOp( src, a, OP_TYPE_AND, priority );
|
|
}
|
|
if ( priority == 4 && token == "||" ) {
|
|
return ParseEmitOp( src, a, OP_TYPE_OR, priority );
|
|
}
|
|
|
|
// assume that anything else terminates the expression
|
|
// not too robust error checking...
|
|
|
|
src.UnreadToken( &token );
|
|
|
|
return a;
|
|
}
|
|
|
|
/*
|
|
=================
|
|
idMaterial::ParseExpression
|
|
|
|
Returns a register index
|
|
=================
|
|
*/
|
|
int idMaterial::ParseExpression( idLexer &src ) {
|
|
return ParseExpressionPriority( src, TOP_PRIORITY );
|
|
}
|
|
|
|
|
|
/*
|
|
===============
|
|
idMaterial::ClearStage
|
|
===============
|
|
*/
|
|
void idMaterial::ClearStage( shaderStage_t *ss ) {
|
|
ss->drawStateBits = 0;
|
|
ss->conditionRegister = GetExpressionConstant( 1 );
|
|
ss->color.registers[0] =
|
|
ss->color.registers[1] =
|
|
ss->color.registers[2] =
|
|
ss->color.registers[3] = GetExpressionConstant( 1 );
|
|
}
|
|
|
|
/*
|
|
===============
|
|
idMaterial::NameToSrcBlendMode
|
|
===============
|
|
*/
|
|
int idMaterial::NameToSrcBlendMode( const idStr &name ) {
|
|
if ( !name.Icmp( "GL_ONE" ) ) {
|
|
return GLS_SRCBLEND_ONE;
|
|
} else if ( !name.Icmp( "GL_ZERO" ) ) {
|
|
return GLS_SRCBLEND_ZERO;
|
|
} else if ( !name.Icmp( "GL_DST_COLOR" ) ) {
|
|
return GLS_SRCBLEND_DST_COLOR;
|
|
} else if ( !name.Icmp( "GL_ONE_MINUS_DST_COLOR" ) ) {
|
|
return GLS_SRCBLEND_ONE_MINUS_DST_COLOR;
|
|
} else if ( !name.Icmp( "GL_SRC_ALPHA" ) ) {
|
|
return GLS_SRCBLEND_SRC_ALPHA;
|
|
} else if ( !name.Icmp( "GL_ONE_MINUS_SRC_ALPHA" ) ) {
|
|
return GLS_SRCBLEND_ONE_MINUS_SRC_ALPHA;
|
|
} else if ( !name.Icmp( "GL_DST_ALPHA" ) ) {
|
|
return GLS_SRCBLEND_DST_ALPHA;
|
|
} else if ( !name.Icmp( "GL_ONE_MINUS_DST_ALPHA" ) ) {
|
|
return GLS_SRCBLEND_ONE_MINUS_DST_ALPHA;
|
|
} else if ( !name.Icmp( "GL_SRC_ALPHA_SATURATE" ) ) {
|
|
return GLS_SRCBLEND_ALPHA_SATURATE;
|
|
}
|
|
|
|
common->Warning( "unknown blend mode '%s' in material '%s'", name.c_str(), GetName() );
|
|
SetMaterialFlag( MF_DEFAULTED );
|
|
|
|
return GLS_SRCBLEND_ONE;
|
|
}
|
|
|
|
/*
|
|
===============
|
|
idMaterial::NameToDstBlendMode
|
|
===============
|
|
*/
|
|
int idMaterial::NameToDstBlendMode( const idStr &name ) {
|
|
if ( !name.Icmp( "GL_ONE" ) ) {
|
|
return GLS_DSTBLEND_ONE;
|
|
} else if ( !name.Icmp( "GL_ZERO" ) ) {
|
|
return GLS_DSTBLEND_ZERO;
|
|
} else if ( !name.Icmp( "GL_SRC_ALPHA" ) ) {
|
|
return GLS_DSTBLEND_SRC_ALPHA;
|
|
} else if ( !name.Icmp( "GL_ONE_MINUS_SRC_ALPHA" ) ) {
|
|
return GLS_DSTBLEND_ONE_MINUS_SRC_ALPHA;
|
|
} else if ( !name.Icmp( "GL_DST_ALPHA" ) ) {
|
|
return GLS_DSTBLEND_DST_ALPHA;
|
|
} else if ( !name.Icmp( "GL_ONE_MINUS_DST_ALPHA" ) ) {
|
|
return GLS_DSTBLEND_ONE_MINUS_DST_ALPHA;
|
|
} else if ( !name.Icmp( "GL_SRC_COLOR" ) ) {
|
|
return GLS_DSTBLEND_SRC_COLOR;
|
|
} else if ( !name.Icmp( "GL_ONE_MINUS_SRC_COLOR" ) ) {
|
|
return GLS_DSTBLEND_ONE_MINUS_SRC_COLOR;
|
|
}
|
|
|
|
common->Warning( "unknown blend mode '%s' in material '%s'", name.c_str(), GetName() );
|
|
SetMaterialFlag( MF_DEFAULTED );
|
|
|
|
return GLS_DSTBLEND_ONE;
|
|
}
|
|
|
|
/*
|
|
================
|
|
idMaterial::ParseBlend
|
|
================
|
|
*/
|
|
void idMaterial::ParseBlend( idLexer &src, shaderStage_t *stage ) {
|
|
idToken token;
|
|
int srcBlend, dstBlend;
|
|
|
|
if ( !src.ReadToken( &token ) ) {
|
|
return;
|
|
}
|
|
|
|
// blending combinations
|
|
if ( !token.Icmp( "blend" ) ) {
|
|
stage->drawStateBits = GLS_SRCBLEND_SRC_ALPHA | GLS_DSTBLEND_ONE_MINUS_SRC_ALPHA;
|
|
return;
|
|
}
|
|
if ( !token.Icmp( "add" ) ) {
|
|
stage->drawStateBits = GLS_SRCBLEND_ONE | GLS_DSTBLEND_ONE;
|
|
return;
|
|
}
|
|
if ( !token.Icmp( "filter" ) || !token.Icmp( "modulate" ) ) {
|
|
stage->drawStateBits = GLS_SRCBLEND_DST_COLOR | GLS_DSTBLEND_ZERO;
|
|
return;
|
|
}
|
|
if ( !token.Icmp( "none" ) ) {
|
|
// none is used when defining an alpha mask that doesn't draw
|
|
stage->drawStateBits = GLS_SRCBLEND_ZERO | GLS_DSTBLEND_ONE;
|
|
return;
|
|
}
|
|
if ( !token.Icmp( "bumpmap" ) ) {
|
|
stage->lighting = SL_BUMP;
|
|
return;
|
|
}
|
|
if ( !token.Icmp( "diffusemap" ) ) {
|
|
stage->lighting = SL_DIFFUSE;
|
|
return;
|
|
}
|
|
if ( !token.Icmp( "specularmap" ) ) {
|
|
stage->lighting = SL_SPECULAR;
|
|
return;
|
|
}
|
|
|
|
srcBlend = NameToSrcBlendMode( token );
|
|
|
|
MatchToken( src, "," );
|
|
if ( !src.ReadToken( &token ) ) {
|
|
return;
|
|
}
|
|
dstBlend = NameToDstBlendMode( token );
|
|
|
|
stage->drawStateBits = srcBlend | dstBlend;
|
|
}
|
|
|
|
/*
|
|
================
|
|
idMaterial::ParseVertexParm
|
|
|
|
If there is a single value, it will be repeated across all elements
|
|
If there are two values, 3 = 0.0, 4 = 1.0
|
|
if there are three values, 4 = 1.0
|
|
================
|
|
*/
|
|
void idMaterial::ParseVertexParm( idLexer &src, newShaderStage_t *newStage ) {
|
|
idToken token;
|
|
|
|
src.ReadTokenOnLine( &token );
|
|
int parm = token.GetIntValue();
|
|
if ( !token.IsNumeric() || parm < 0 || parm >= MAX_VERTEX_PARMS ) {
|
|
common->Warning( "bad vertexParm number\n" );
|
|
SetMaterialFlag( MF_DEFAULTED );
|
|
return;
|
|
}
|
|
if ( parm >= newStage->numVertexParms ) {
|
|
newStage->numVertexParms = parm+1;
|
|
}
|
|
|
|
newStage->vertexParms[parm][0] = ParseExpression( src );
|
|
|
|
src.ReadTokenOnLine( &token );
|
|
if ( !token[0] || token.Icmp( "," ) ) {
|
|
newStage->vertexParms[parm][1] =
|
|
newStage->vertexParms[parm][2] =
|
|
newStage->vertexParms[parm][3] = newStage->vertexParms[parm][0];
|
|
return;
|
|
}
|
|
|
|
newStage->vertexParms[parm][1] = ParseExpression( src );
|
|
|
|
src.ReadTokenOnLine( &token );
|
|
if ( !token[0] || token.Icmp( "," ) ) {
|
|
newStage->vertexParms[parm][2] = GetExpressionConstant( 0 );
|
|
newStage->vertexParms[parm][3] = GetExpressionConstant( 1 );
|
|
return;
|
|
}
|
|
|
|
newStage->vertexParms[parm][2] = ParseExpression( src );
|
|
|
|
src.ReadTokenOnLine( &token );
|
|
if ( !token[0] || token.Icmp( "," ) ) {
|
|
newStage->vertexParms[parm][3] = GetExpressionConstant( 1 );
|
|
return;
|
|
}
|
|
|
|
newStage->vertexParms[parm][3] = ParseExpression( src );
|
|
}
|
|
|
|
|
|
/*
|
|
================
|
|
idMaterial::ParseFragmentMap
|
|
================
|
|
*/
|
|
void idMaterial::ParseFragmentMap( idLexer &src, newShaderStage_t *newStage ) {
|
|
const char *str;
|
|
textureFilter_t tf;
|
|
textureRepeat_t trp;
|
|
textureDepth_t td;
|
|
cubeFiles_t cubeMap;
|
|
bool allowPicmip;
|
|
idToken token;
|
|
|
|
tf = TF_DEFAULT;
|
|
trp = TR_REPEAT;
|
|
td = TD_DEFAULT;
|
|
allowPicmip = true;
|
|
cubeMap = CF_2D;
|
|
|
|
src.ReadTokenOnLine( &token );
|
|
int unit = token.GetIntValue();
|
|
if ( !token.IsNumeric() || unit < 0 || unit >= MAX_FRAGMENT_IMAGES ) {
|
|
common->Warning( "bad fragmentMap number\n" );
|
|
SetMaterialFlag( MF_DEFAULTED );
|
|
return;
|
|
}
|
|
|
|
// unit 1 is the normal map.. make sure it gets flagged as the proper depth
|
|
if ( unit == 1 ) {
|
|
td = TD_BUMP;
|
|
}
|
|
|
|
if ( unit >= newStage->numFragmentProgramImages ) {
|
|
newStage->numFragmentProgramImages = unit+1;
|
|
}
|
|
|
|
while( 1 ) {
|
|
src.ReadTokenOnLine( &token );
|
|
|
|
if ( !token.Icmp( "cubeMap" ) ) {
|
|
cubeMap = CF_NATIVE;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "cameraCubeMap" ) ) {
|
|
cubeMap = CF_CAMERA;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "nearest" ) ) {
|
|
tf = TF_NEAREST;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "linear" ) ) {
|
|
tf = TF_LINEAR;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "clamp" ) ) {
|
|
trp = TR_CLAMP;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "noclamp" ) ) {
|
|
trp = TR_REPEAT;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "zeroclamp" ) ) {
|
|
trp = TR_CLAMP_TO_ZERO;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "alphazeroclamp" ) ) {
|
|
trp = TR_CLAMP_TO_ZERO_ALPHA;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "forceHighQuality" ) ) {
|
|
td = TD_HIGH_QUALITY;
|
|
continue;
|
|
}
|
|
|
|
if ( !token.Icmp( "uncompressed" ) || !token.Icmp( "highquality" ) ) {
|
|
if ( !globalImages->image_ignoreHighQuality.GetInteger() ) {
|
|
td = TD_HIGH_QUALITY;
|
|
}
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "nopicmip" ) ) {
|
|
allowPicmip = false;
|
|
continue;
|
|
}
|
|
|
|
// assume anything else is the image name
|
|
src.UnreadToken( &token );
|
|
break;
|
|
}
|
|
str = R_ParsePastImageProgram( src );
|
|
|
|
newStage->fragmentProgramImages[unit] =
|
|
globalImages->ImageFromFile( str, tf, allowPicmip, trp, td, cubeMap );
|
|
if ( !newStage->fragmentProgramImages[unit] ) {
|
|
newStage->fragmentProgramImages[unit] = globalImages->defaultImage;
|
|
}
|
|
}
|
|
|
|
/*
|
|
===============
|
|
idMaterial::MultiplyTextureMatrix
|
|
===============
|
|
*/
|
|
void idMaterial::MultiplyTextureMatrix( textureStage_t *ts, int registers[2][3] ) {
|
|
int old[2][3];
|
|
|
|
if ( !ts->hasMatrix ) {
|
|
ts->hasMatrix = true;
|
|
memcpy( ts->matrix, registers, sizeof( ts->matrix ) );
|
|
return;
|
|
}
|
|
|
|
memcpy( old, ts->matrix, sizeof( old ) );
|
|
|
|
// multiply the two maticies
|
|
ts->matrix[0][0] = EmitOp(
|
|
EmitOp( old[0][0], registers[0][0], OP_TYPE_MULTIPLY ),
|
|
EmitOp( old[0][1], registers[1][0], OP_TYPE_MULTIPLY ), OP_TYPE_ADD );
|
|
ts->matrix[0][1] = EmitOp(
|
|
EmitOp( old[0][0], registers[0][1], OP_TYPE_MULTIPLY ),
|
|
EmitOp( old[0][1], registers[1][1], OP_TYPE_MULTIPLY ), OP_TYPE_ADD );
|
|
ts->matrix[0][2] = EmitOp(
|
|
EmitOp(
|
|
EmitOp( old[0][0], registers[0][2], OP_TYPE_MULTIPLY ),
|
|
EmitOp( old[0][1], registers[1][2], OP_TYPE_MULTIPLY ), OP_TYPE_ADD ),
|
|
old[0][2], OP_TYPE_ADD );
|
|
|
|
ts->matrix[1][0] = EmitOp(
|
|
EmitOp( old[1][0], registers[0][0], OP_TYPE_MULTIPLY ),
|
|
EmitOp( old[1][1], registers[1][0], OP_TYPE_MULTIPLY ), OP_TYPE_ADD );
|
|
ts->matrix[1][1] = EmitOp(
|
|
EmitOp( old[1][0], registers[0][1], OP_TYPE_MULTIPLY ),
|
|
EmitOp( old[1][1], registers[1][1], OP_TYPE_MULTIPLY ), OP_TYPE_ADD );
|
|
ts->matrix[1][2] = EmitOp(
|
|
EmitOp(
|
|
EmitOp( old[1][0], registers[0][2], OP_TYPE_MULTIPLY ),
|
|
EmitOp( old[1][1], registers[1][2], OP_TYPE_MULTIPLY ), OP_TYPE_ADD ),
|
|
old[1][2], OP_TYPE_ADD );
|
|
|
|
}
|
|
|
|
/*
|
|
=================
|
|
idMaterial::ParseStage
|
|
|
|
An open brace has been parsed
|
|
|
|
|
|
{
|
|
if <expression>
|
|
map <imageprogram>
|
|
"nearest" "linear" "clamp" "zeroclamp" "uncompressed" "highquality" "nopicmip"
|
|
scroll, scale, rotate
|
|
}
|
|
|
|
=================
|
|
*/
|
|
void idMaterial::ParseStage( idLexer &src, const textureRepeat_t trpDefault ) {
|
|
idToken token;
|
|
const char *str;
|
|
shaderStage_t *ss;
|
|
textureStage_t *ts;
|
|
textureFilter_t tf;
|
|
textureRepeat_t trp;
|
|
textureDepth_t td;
|
|
cubeFiles_t cubeMap;
|
|
bool allowPicmip;
|
|
char imageName[MAX_IMAGE_NAME];
|
|
int a, b;
|
|
int matrix[2][3];
|
|
newShaderStage_t newStage;
|
|
|
|
if ( numStages >= MAX_SHADER_STAGES ) {
|
|
SetMaterialFlag( MF_DEFAULTED );
|
|
common->Warning( "material '%s' exceeded %i stages", GetName(), MAX_SHADER_STAGES );
|
|
}
|
|
|
|
tf = TF_DEFAULT;
|
|
trp = trpDefault;
|
|
td = TD_DEFAULT;
|
|
allowPicmip = true;
|
|
cubeMap = CF_2D;
|
|
|
|
imageName[0] = 0;
|
|
|
|
memset( &newStage, 0, sizeof( newStage ) );
|
|
|
|
ss = &pd->parseStages[numStages];
|
|
ts = &ss->texture;
|
|
|
|
ClearStage( ss );
|
|
|
|
while ( 1 ) {
|
|
if ( TestMaterialFlag( MF_DEFAULTED ) ) { // we have a parse error
|
|
return;
|
|
}
|
|
if ( !src.ExpectAnyToken( &token ) ) {
|
|
SetMaterialFlag( MF_DEFAULTED );
|
|
return;
|
|
}
|
|
|
|
// the close brace for the entire material ends the draw block
|
|
if ( token == "}" ) {
|
|
break;
|
|
}
|
|
|
|
//BSM Nerve: Added for stage naming in the material editor
|
|
if( !token.Icmp( "name") ) {
|
|
src.SkipRestOfLine();
|
|
continue;
|
|
}
|
|
|
|
// image options
|
|
if ( !token.Icmp( "blend" ) ) {
|
|
ParseBlend( src, ss );
|
|
continue;
|
|
}
|
|
|
|
if ( !token.Icmp( "map" ) ) {
|
|
str = R_ParsePastImageProgram( src );
|
|
idStr::Copynz( imageName, str, sizeof( imageName ) );
|
|
continue;
|
|
}
|
|
|
|
if ( !token.Icmp( "remoteRenderMap" ) ) {
|
|
ts->dynamic = DI_REMOTE_RENDER;
|
|
ts->width = src.ParseInt();
|
|
ts->height = src.ParseInt();
|
|
continue;
|
|
}
|
|
|
|
if ( !token.Icmp( "mirrorRenderMap" ) ) {
|
|
ts->dynamic = DI_MIRROR_RENDER;
|
|
ts->width = src.ParseInt();
|
|
ts->height = src.ParseInt();
|
|
ts->texgen = TG_SCREEN;
|
|
continue;
|
|
}
|
|
|
|
if ( !token.Icmp( "xrayRenderMap" ) ) {
|
|
ts->dynamic = DI_XRAY_RENDER;
|
|
ts->width = src.ParseInt();
|
|
ts->height = src.ParseInt();
|
|
ts->texgen = TG_SCREEN;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "screen" ) ) {
|
|
ts->texgen = TG_SCREEN;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "screen2" ) ) {
|
|
ts->texgen = TG_SCREEN2;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "glassWarp" ) ) {
|
|
ts->texgen = TG_GLASSWARP;
|
|
continue;
|
|
}
|
|
|
|
if ( !token.Icmp( "videomap" ) ) {
|
|
// note that videomaps will always be in clamp mode, so texture
|
|
// coordinates had better be in the 0 to 1 range
|
|
if ( !src.ReadToken( &token ) ) {
|
|
common->Warning( "missing parameter for 'videoMap' keyword in material '%s'", GetName() );
|
|
continue;
|
|
}
|
|
bool loop = false;
|
|
if ( !token.Icmp( "loop" ) ) {
|
|
loop = true;
|
|
if ( !src.ReadToken( &token ) ) {
|
|
common->Warning( "missing parameter for 'videoMap' keyword in material '%s'", GetName() );
|
|
continue;
|
|
}
|
|
}
|
|
ts->cinematic = idCinematic::Alloc();
|
|
ts->cinematic->InitFromFile( token.c_str(), loop );
|
|
continue;
|
|
}
|
|
|
|
if ( !token.Icmp( "soundmap" ) ) {
|
|
if ( !src.ReadToken( &token ) ) {
|
|
common->Warning( "missing parameter for 'soundmap' keyword in material '%s'", GetName() );
|
|
continue;
|
|
}
|
|
ts->cinematic = new idSndWindow();
|
|
ts->cinematic->InitFromFile( token.c_str(), true );
|
|
continue;
|
|
}
|
|
|
|
if ( !token.Icmp( "cubeMap" ) ) {
|
|
str = R_ParsePastImageProgram( src );
|
|
idStr::Copynz( imageName, str, sizeof( imageName ) );
|
|
cubeMap = CF_NATIVE;
|
|
continue;
|
|
}
|
|
|
|
if ( !token.Icmp( "cameraCubeMap" ) ) {
|
|
str = R_ParsePastImageProgram( src );
|
|
idStr::Copynz( imageName, str, sizeof( imageName ) );
|
|
cubeMap = CF_CAMERA;
|
|
continue;
|
|
}
|
|
|
|
if ( !token.Icmp( "ignoreAlphaTest" ) ) {
|
|
ss->ignoreAlphaTest = true;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "nearest" ) ) {
|
|
tf = TF_NEAREST;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "linear" ) ) {
|
|
tf = TF_LINEAR;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "clamp" ) ) {
|
|
trp = TR_CLAMP;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "noclamp" ) ) {
|
|
trp = TR_REPEAT;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "zeroclamp" ) ) {
|
|
trp = TR_CLAMP_TO_ZERO;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "alphazeroclamp" ) ) {
|
|
trp = TR_CLAMP_TO_ZERO_ALPHA;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "uncompressed" ) || !token.Icmp( "highquality" ) ) {
|
|
if ( !globalImages->image_ignoreHighQuality.GetInteger() ) {
|
|
td = TD_HIGH_QUALITY;
|
|
}
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "forceHighQuality" ) ) {
|
|
td = TD_HIGH_QUALITY;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "nopicmip" ) ) {
|
|
allowPicmip = false;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "vertexColor" ) ) {
|
|
ss->vertexColor = SVC_MODULATE;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "inverseVertexColor" ) ) {
|
|
ss->vertexColor = SVC_INVERSE_MODULATE;
|
|
continue;
|
|
}
|
|
|
|
// privatePolygonOffset
|
|
else if ( !token.Icmp( "privatePolygonOffset" ) ) {
|
|
if ( !src.ReadTokenOnLine( &token ) ) {
|
|
ss->privatePolygonOffset = 1;
|
|
continue;
|
|
}
|
|
// explict larger (or negative) offset
|
|
src.UnreadToken( &token );
|
|
ss->privatePolygonOffset = src.ParseFloat();
|
|
continue;
|
|
}
|
|
|
|
// texture coordinate generation
|
|
if ( !token.Icmp( "texGen" ) ) {
|
|
src.ExpectAnyToken( &token );
|
|
if ( !token.Icmp( "normal" ) ) {
|
|
ts->texgen = TG_DIFFUSE_CUBE;
|
|
} else if ( !token.Icmp( "reflect" ) ) {
|
|
ts->texgen = TG_REFLECT_CUBE;
|
|
} else if ( !token.Icmp( "skybox" ) ) {
|
|
ts->texgen = TG_SKYBOX_CUBE;
|
|
} else if ( !token.Icmp( "wobbleSky" ) ) {
|
|
ts->texgen = TG_WOBBLESKY_CUBE;
|
|
texGenRegisters[0] = ParseExpression( src );
|
|
texGenRegisters[1] = ParseExpression( src );
|
|
texGenRegisters[2] = ParseExpression( src );
|
|
} else {
|
|
common->Warning( "bad texGen '%s' in material %s", token.c_str(), GetName() );
|
|
SetMaterialFlag( MF_DEFAULTED );
|
|
}
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "scroll" ) || !token.Icmp( "translate" ) ) {
|
|
a = ParseExpression( src );
|
|
MatchToken( src, "," );
|
|
b = ParseExpression( src );
|
|
matrix[0][0] = GetExpressionConstant( 1 );
|
|
matrix[0][1] = GetExpressionConstant( 0 );
|
|
matrix[0][2] = a;
|
|
matrix[1][0] = GetExpressionConstant( 0 );
|
|
matrix[1][1] = GetExpressionConstant( 1 );
|
|
matrix[1][2] = b;
|
|
|
|
MultiplyTextureMatrix( ts, matrix );
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "scale" ) ) {
|
|
a = ParseExpression( src );
|
|
MatchToken( src, "," );
|
|
b = ParseExpression( src );
|
|
// this just scales without a centering
|
|
matrix[0][0] = a;
|
|
matrix[0][1] = GetExpressionConstant( 0 );
|
|
matrix[0][2] = GetExpressionConstant( 0 );
|
|
matrix[1][0] = GetExpressionConstant( 0 );
|
|
matrix[1][1] = b;
|
|
matrix[1][2] = GetExpressionConstant( 0 );
|
|
|
|
MultiplyTextureMatrix( ts, matrix );
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "centerScale" ) ) {
|
|
a = ParseExpression( src );
|
|
MatchToken( src, "," );
|
|
b = ParseExpression( src );
|
|
// this subtracts 0.5, then scales, then adds 0.5
|
|
matrix[0][0] = a;
|
|
matrix[0][1] = GetExpressionConstant( 0 );
|
|
matrix[0][2] = EmitOp( GetExpressionConstant( 0.5 ), EmitOp( GetExpressionConstant( 0.5 ), a, OP_TYPE_MULTIPLY ), OP_TYPE_SUBTRACT );
|
|
matrix[1][0] = GetExpressionConstant( 0 );
|
|
matrix[1][1] = b;
|
|
matrix[1][2] = EmitOp( GetExpressionConstant( 0.5 ), EmitOp( GetExpressionConstant( 0.5 ), b, OP_TYPE_MULTIPLY ), OP_TYPE_SUBTRACT );
|
|
|
|
MultiplyTextureMatrix( ts, matrix );
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "shear" ) ) {
|
|
a = ParseExpression( src );
|
|
MatchToken( src, "," );
|
|
b = ParseExpression( src );
|
|
// this subtracts 0.5, then shears, then adds 0.5
|
|
matrix[0][0] = GetExpressionConstant( 1 );
|
|
matrix[0][1] = a;
|
|
matrix[0][2] = EmitOp( GetExpressionConstant( -0.5 ), a, OP_TYPE_MULTIPLY );
|
|
matrix[1][0] = b;
|
|
matrix[1][1] = GetExpressionConstant( 1 );
|
|
matrix[1][2] = EmitOp( GetExpressionConstant( -0.5 ), b, OP_TYPE_MULTIPLY );
|
|
|
|
MultiplyTextureMatrix( ts, matrix );
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "rotate" ) ) {
|
|
const idDeclTable *table;
|
|
int sinReg, cosReg;
|
|
|
|
// in cycles
|
|
a = ParseExpression( src );
|
|
|
|
table = static_cast<const idDeclTable *>( declManager->FindType( DECL_TABLE, "sinTable", false ) );
|
|
if ( !table ) {
|
|
common->Warning( "no sinTable for rotate defined" );
|
|
SetMaterialFlag( MF_DEFAULTED );
|
|
return;
|
|
}
|
|
sinReg = EmitOp( table->Index(), a, OP_TYPE_TABLE );
|
|
|
|
table = static_cast<const idDeclTable *>( declManager->FindType( DECL_TABLE, "cosTable", false ) );
|
|
if ( !table ) {
|
|
common->Warning( "no cosTable for rotate defined" );
|
|
SetMaterialFlag( MF_DEFAULTED );
|
|
return;
|
|
}
|
|
cosReg = EmitOp( table->Index(), a, OP_TYPE_TABLE );
|
|
|
|
// this subtracts 0.5, then rotates, then adds 0.5
|
|
matrix[0][0] = cosReg;
|
|
matrix[0][1] = EmitOp( GetExpressionConstant( 0 ), sinReg, OP_TYPE_SUBTRACT );
|
|
matrix[0][2] = EmitOp( EmitOp( EmitOp( GetExpressionConstant( -0.5 ), cosReg, OP_TYPE_MULTIPLY ),
|
|
EmitOp( GetExpressionConstant( 0.5 ), sinReg, OP_TYPE_MULTIPLY ), OP_TYPE_ADD ),
|
|
GetExpressionConstant( 0.5 ), OP_TYPE_ADD );
|
|
|
|
matrix[1][0] = sinReg;
|
|
matrix[1][1] = cosReg;
|
|
matrix[1][2] = EmitOp( EmitOp( EmitOp( GetExpressionConstant( -0.5 ), sinReg, OP_TYPE_MULTIPLY ),
|
|
EmitOp( GetExpressionConstant( -0.5 ), cosReg, OP_TYPE_MULTIPLY ), OP_TYPE_ADD ),
|
|
GetExpressionConstant( 0.5 ), OP_TYPE_ADD );
|
|
|
|
MultiplyTextureMatrix( ts, matrix );
|
|
continue;
|
|
}
|
|
|
|
// color mask options
|
|
if ( !token.Icmp( "maskRed" ) ) {
|
|
ss->drawStateBits |= GLS_REDMASK;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "maskGreen" ) ) {
|
|
ss->drawStateBits |= GLS_GREENMASK;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "maskBlue" ) ) {
|
|
ss->drawStateBits |= GLS_BLUEMASK;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "maskAlpha" ) ) {
|
|
ss->drawStateBits |= GLS_ALPHAMASK;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "maskColor" ) ) {
|
|
ss->drawStateBits |= GLS_COLORMASK;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "maskDepth" ) ) {
|
|
ss->drawStateBits |= GLS_DEPTHMASK;
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "alphaTest" ) ) {
|
|
ss->hasAlphaTest = true;
|
|
ss->alphaTestRegister = ParseExpression( src );
|
|
coverage = MC_PERFORATED;
|
|
continue;
|
|
}
|
|
|
|
// shorthand for 2D modulated
|
|
if ( !token.Icmp( "colored" ) ) {
|
|
ss->color.registers[0] = EXP_REG_PARM0;
|
|
ss->color.registers[1] = EXP_REG_PARM1;
|
|
ss->color.registers[2] = EXP_REG_PARM2;
|
|
ss->color.registers[3] = EXP_REG_PARM3;
|
|
pd->registersAreConstant = false;
|
|
continue;
|
|
}
|
|
|
|
if ( !token.Icmp( "color" ) ) {
|
|
ss->color.registers[0] = ParseExpression( src );
|
|
MatchToken( src, "," );
|
|
ss->color.registers[1] = ParseExpression( src );
|
|
MatchToken( src, "," );
|
|
ss->color.registers[2] = ParseExpression( src );
|
|
MatchToken( src, "," );
|
|
ss->color.registers[3] = ParseExpression( src );
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "red" ) ) {
|
|
ss->color.registers[0] = ParseExpression( src );
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "green" ) ) {
|
|
ss->color.registers[1] = ParseExpression( src );
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "blue" ) ) {
|
|
ss->color.registers[2] = ParseExpression( src );
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "alpha" ) ) {
|
|
ss->color.registers[3] = ParseExpression( src );
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "rgb" ) ) {
|
|
ss->color.registers[0] = ss->color.registers[1] =
|
|
ss->color.registers[2] = ParseExpression( src );
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "rgba" ) ) {
|
|
ss->color.registers[0] = ss->color.registers[1] =
|
|
ss->color.registers[2] = ss->color.registers[3] = ParseExpression( src );
|
|
continue;
|
|
}
|
|
|
|
if ( !token.Icmp( "if" ) ) {
|
|
ss->conditionRegister = ParseExpression( src );
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "program" ) ) {
|
|
if ( src.ReadTokenOnLine( &token ) ) {
|
|
newStage.vertexProgram = R_FindARBProgram( GL_VERTEX_PROGRAM_ARB, token.c_str() );
|
|
newStage.fragmentProgram = R_FindARBProgram( GL_FRAGMENT_PROGRAM_ARB, token.c_str() );
|
|
}
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "fragmentProgram" ) ) {
|
|
if ( src.ReadTokenOnLine( &token ) ) {
|
|
newStage.fragmentProgram = R_FindARBProgram( GL_FRAGMENT_PROGRAM_ARB, token.c_str() );
|
|
}
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "vertexProgram" ) ) {
|
|
if ( src.ReadTokenOnLine( &token ) ) {
|
|
newStage.vertexProgram = R_FindARBProgram( GL_VERTEX_PROGRAM_ARB, token.c_str() );
|
|
}
|
|
continue;
|
|
}
|
|
if ( !token.Icmp( "megaTexture" ) ) {
|
|
if ( src.ReadTokenOnLine( &token ) ) {
|
|
newStage.megaTexture = new idMegaTexture;
|
|
if ( !newStage.megaTexture->InitFromMegaFile( token.c_str() ) ) {
|
|
delete newStage.megaTexture;
|
|
SetMaterialFlag( MF_DEFAULTED );
|
|
continue;
|
|
}
|
|
newStage.vertexProgram = R_FindARBProgram( GL_VERTEX_PROGRAM_ARB, "megaTexture.vfp" );
|
|
newStage.fragmentProgram = R_FindARBProgram( GL_FRAGMENT_PROGRAM_ARB, "megaTexture.vfp" );
|
|
continue;
|
|
}
|
|
}
|
|
|
|
|
|
if ( !token.Icmp( "vertexParm" ) ) {
|
|
ParseVertexParm( src, &newStage );
|
|
continue;
|
|
}
|
|
|
|
if ( !token.Icmp( "fragmentMap" ) ) {
|
|
ParseFragmentMap( src, &newStage );
|
|
continue;
|
|
}
|
|
|
|
|
|
common->Warning( "unknown token '%s' in material '%s'", token.c_str(), GetName() );
|
|
SetMaterialFlag( MF_DEFAULTED );
|
|
return;
|
|
}
|
|
|
|
|
|
// if we are using newStage, allocate a copy of it
|
|
if ( newStage.fragmentProgram || newStage.vertexProgram ) {
|
|
ss->newStage = (newShaderStage_t *)Mem_Alloc( sizeof( newStage ) );
|
|
*(ss->newStage) = newStage;
|
|
}
|
|
|
|
// successfully parsed a stage
|
|
numStages++;
|
|
|
|
// select a compressed depth based on what the stage is
|
|
if ( td == TD_DEFAULT ) {
|
|
switch( ss->lighting ) {
|
|
case SL_BUMP:
|
|
td = TD_BUMP;
|
|
break;
|
|
case SL_DIFFUSE:
|
|
td = TD_DIFFUSE;
|
|
break;
|
|
case SL_SPECULAR:
|
|
td = TD_SPECULAR;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
// now load the image with all the parms we parsed
|
|
if ( imageName[0] ) {
|
|
ts->image = globalImages->ImageFromFile( imageName, tf, allowPicmip, trp, td, cubeMap );
|
|
if ( !ts->image ) {
|
|
ts->image = globalImages->defaultImage;
|
|
}
|
|
} else if ( !ts->cinematic && !ts->dynamic && !ss->newStage ) {
|
|
common->Warning( "material '%s' had stage with no image", GetName() );
|
|
ts->image = globalImages->defaultImage;
|
|
}
|
|
}
|
|
|
|
/*
|
|
===============
|
|
idMaterial::ParseDeform
|
|
===============
|
|
*/
|
|
void idMaterial::ParseDeform( idLexer &src ) {
|
|
idToken token;
|
|
|
|
if ( !src.ExpectAnyToken( &token ) ) {
|
|
return;
|
|
}
|
|
|
|
if ( !token.Icmp( "sprite" ) ) {
|
|
deform = DFRM_SPRITE;
|
|
cullType = CT_TWO_SIDED;
|
|
SetMaterialFlag( MF_NOSHADOWS );
|
|
return;
|
|
}
|
|
if ( !token.Icmp( "tube" ) ) {
|
|
deform = DFRM_TUBE;
|
|
cullType = CT_TWO_SIDED;
|
|
SetMaterialFlag( MF_NOSHADOWS );
|
|
return;
|
|
}
|
|
if ( !token.Icmp( "flare" ) ) {
|
|
deform = DFRM_FLARE;
|
|
cullType = CT_TWO_SIDED;
|
|
deformRegisters[0] = ParseExpression( src );
|
|
SetMaterialFlag( MF_NOSHADOWS );
|
|
return;
|
|
}
|
|
if ( !token.Icmp( "expand" ) ) {
|
|
deform = DFRM_EXPAND;
|
|
deformRegisters[0] = ParseExpression( src );
|
|
return;
|
|
}
|
|
if ( !token.Icmp( "move" ) ) {
|
|
deform = DFRM_MOVE;
|
|
deformRegisters[0] = ParseExpression( src );
|
|
return;
|
|
}
|
|
if ( !token.Icmp( "turbulent" ) ) {
|
|
deform = DFRM_TURB;
|
|
|
|
if ( !src.ExpectAnyToken( &token ) ) {
|
|
src.Warning( "deform particle missing particle name" );
|
|
SetMaterialFlag( MF_DEFAULTED );
|
|
return;
|
|
}
|
|
deformDecl = declManager->FindType( DECL_TABLE, token.c_str(), true );
|
|
|
|
deformRegisters[0] = ParseExpression( src );
|
|
deformRegisters[1] = ParseExpression( src );
|
|
deformRegisters[2] = ParseExpression( src );
|
|
return;
|
|
}
|
|
if ( !token.Icmp( "eyeBall" ) ) {
|
|
deform = DFRM_EYEBALL;
|
|
return;
|
|
}
|
|
if ( !token.Icmp( "particle" ) ) {
|
|
deform = DFRM_PARTICLE;
|
|
if ( !src.ExpectAnyToken( &token ) ) {
|
|
src.Warning( "deform particle missing particle name" );
|
|
SetMaterialFlag( MF_DEFAULTED );
|
|
return;
|
|
}
|
|
deformDecl = declManager->FindType( DECL_PARTICLE, token.c_str(), true );
|
|
return;
|
|
}
|
|
if ( !token.Icmp( "particle2" ) ) {
|
|
deform = DFRM_PARTICLE2;
|
|
if ( !src.ExpectAnyToken( &token ) ) {
|
|
src.Warning( "deform particle missing particle name" );
|
|
SetMaterialFlag( MF_DEFAULTED );
|
|
return;
|
|
}
|
|
deformDecl = declManager->FindType( DECL_PARTICLE, token.c_str(), true );
|
|
return;
|
|
}
|
|
src.Warning( "Bad deform type '%s'", token.c_str() );
|
|
SetMaterialFlag( MF_DEFAULTED );
|
|
}
|
|
|
|
|
|
/*
|
|
==============
|
|
idMaterial::AddImplicitStages
|
|
|
|
If a material has diffuse or specular stages without any
|
|
bump stage, add an implicit _flat bumpmap stage.
|
|
|
|
If a material has a bump stage but no diffuse or specular
|
|
stage, add a _white diffuse stage.
|
|
|
|
It is valid to have either a diffuse or specular without the other.
|
|
|
|
It is valid to have a reflection map and a bump map for bumpy reflection
|
|
==============
|
|
*/
|
|
void idMaterial::AddImplicitStages( const textureRepeat_t trpDefault /* = TR_REPEAT */ ) {
|
|
char buffer[1024];
|
|
idLexer newSrc;
|
|
bool hasDiffuse = false;
|
|
bool hasSpecular = false;
|
|
bool hasBump = false;
|
|
bool hasReflection = false;
|
|
|
|
for ( int i = 0 ; i < numStages ; i++ ) {
|
|
if ( pd->parseStages[i].lighting == SL_BUMP ) {
|
|
hasBump = true;
|
|
}
|
|
if ( pd->parseStages[i].lighting == SL_DIFFUSE ) {
|
|
hasDiffuse = true;
|
|
}
|
|
if ( pd->parseStages[i].lighting == SL_SPECULAR ) {
|
|
hasSpecular = true;
|
|
}
|
|
if ( pd->parseStages[i].texture.texgen == TG_REFLECT_CUBE ) {
|
|
hasReflection = true;
|
|
}
|
|
}
|
|
|
|
// if it doesn't have an interaction at all, don't add anything
|
|
if ( !hasBump && !hasDiffuse && !hasSpecular ) {
|
|
return;
|
|
}
|
|
|
|
if ( numStages == MAX_SHADER_STAGES ) {
|
|
return;
|
|
}
|
|
|
|
if ( !hasBump ) {
|
|
idStr::snPrintf( buffer, sizeof( buffer ), "blend bumpmap\nmap _flat\n}\n" );
|
|
newSrc.LoadMemory( buffer, strlen(buffer), "bumpmap" );
|
|
newSrc.SetFlags( LEXFL_NOFATALERRORS | LEXFL_NOSTRINGCONCAT | LEXFL_NOSTRINGESCAPECHARS | LEXFL_ALLOWPATHNAMES );
|
|
ParseStage( newSrc, trpDefault );
|
|
newSrc.FreeSource();
|
|
}
|
|
|
|
if ( !hasDiffuse && !hasSpecular && !hasReflection ) {
|
|
idStr::snPrintf( buffer, sizeof( buffer ), "blend diffusemap\nmap _white\n}\n" );
|
|
newSrc.LoadMemory( buffer, strlen(buffer), "diffusemap" );
|
|
newSrc.SetFlags( LEXFL_NOFATALERRORS | LEXFL_NOSTRINGCONCAT | LEXFL_NOSTRINGESCAPECHARS | LEXFL_ALLOWPATHNAMES );
|
|
ParseStage( newSrc, trpDefault );
|
|
newSrc.FreeSource();
|
|
}
|
|
|
|
}
|
|
|
|
/*
|
|
===============
|
|
idMaterial::SortInteractionStages
|
|
|
|
The renderer expects bump, then diffuse, then specular
|
|
There can be multiple bump maps, followed by additional
|
|
diffuse and specular stages, which allows cross-faded bump mapping.
|
|
|
|
Ambient stages can be interspersed anywhere, but they are
|
|
ignored during interactions, and all the interaction
|
|
stages are ignored during ambient drawing.
|
|
===============
|
|
*/
|
|
void idMaterial::SortInteractionStages() {
|
|
int j;
|
|
|
|
for ( int i = 0 ; i < numStages ; i = j ) {
|
|
// find the next bump map
|
|
for ( j = i + 1 ; j < numStages ; j++ ) {
|
|
if ( pd->parseStages[j].lighting == SL_BUMP ) {
|
|
// if the very first stage wasn't a bumpmap,
|
|
// this bumpmap is part of the first group
|
|
if ( pd->parseStages[i].lighting != SL_BUMP ) {
|
|
continue;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
// bubble sort everything bump / diffuse / specular
|
|
for ( int l = 1 ; l < j-i ; l++ ) {
|
|
for ( int k = i ; k < j-l ; k++ ) {
|
|
if ( pd->parseStages[k].lighting > pd->parseStages[k+1].lighting ) {
|
|
shaderStage_t temp;
|
|
|
|
temp = pd->parseStages[k];
|
|
pd->parseStages[k] = pd->parseStages[k+1];
|
|
pd->parseStages[k+1] = temp;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
=================
|
|
idMaterial::ParseMaterial
|
|
|
|
The current text pointer is at the explicit text definition of the
|
|
Parse it into the global material variable. Later functions will optimize it.
|
|
|
|
If there is any error during parsing, defaultShader will be set.
|
|
=================
|
|
*/
|
|
void idMaterial::ParseMaterial( idLexer &src ) {
|
|
idToken token;
|
|
char buffer[1024];
|
|
const char *str;
|
|
idLexer newSrc;
|
|
int i;
|
|
|
|
numOps = 0;
|
|
numRegisters = EXP_REG_NUM_PREDEFINED; // leave space for the parms to be copied in
|
|
for ( i = 0 ; i < numRegisters ; i++ ) {
|
|
pd->registerIsTemporary[i] = true; // they aren't constants that can be folded
|
|
}
|
|
|
|
numStages = 0;
|
|
|
|
textureRepeat_t trpDefault = TR_REPEAT; // allow a global setting for repeat
|
|
|
|
while ( 1 ) {
|
|
if ( TestMaterialFlag( MF_DEFAULTED ) ) { // we have a parse error
|
|
return;
|
|
}
|
|
if ( !src.ExpectAnyToken( &token ) ) {
|
|
SetMaterialFlag( MF_DEFAULTED );
|
|
return;
|
|
}
|
|
|
|
// end of material definition
|
|
if ( token == "}" ) {
|
|
break;
|
|
}
|
|
else if ( !token.Icmp( "qer_editorimage") ) {
|
|
src.ReadTokenOnLine( &token );
|
|
editorImageName = token.c_str();
|
|
src.SkipRestOfLine();
|
|
continue;
|
|
}
|
|
// description
|
|
else if ( !token.Icmp( "description") ) {
|
|
src.ReadTokenOnLine( &token );
|
|
desc = token.c_str();
|
|
continue;
|
|
}
|
|
// check for the surface / content bit flags
|
|
else if ( CheckSurfaceParm( &token ) ) {
|
|
continue;
|
|
}
|
|
|
|
|
|
// polygonOffset
|
|
else if ( !token.Icmp( "polygonOffset" ) ) {
|
|
SetMaterialFlag( MF_POLYGONOFFSET );
|
|
if ( !src.ReadTokenOnLine( &token ) ) {
|
|
polygonOffset = 1;
|
|
continue;
|
|
}
|
|
// explict larger (or negative) offset
|
|
polygonOffset = token.GetFloatValue();
|
|
continue;
|
|
}
|
|
// noshadow
|
|
else if ( !token.Icmp( "noShadows" ) ) {
|
|
SetMaterialFlag( MF_NOSHADOWS );
|
|
continue;
|
|
}
|
|
else if ( !token.Icmp( "suppressInSubview" ) ) {
|
|
suppressInSubview = true;
|
|
continue;
|
|
}
|
|
else if ( !token.Icmp( "portalSky" ) ) {
|
|
portalSky = true;
|
|
continue;
|
|
}
|
|
// noSelfShadow
|
|
else if ( !token.Icmp( "noSelfShadow" ) ) {
|
|
SetMaterialFlag( MF_NOSELFSHADOW );
|
|
continue;
|
|
}
|
|
// noPortalFog
|
|
else if ( !token.Icmp( "noPortalFog" ) ) {
|
|
SetMaterialFlag( MF_NOPORTALFOG );
|
|
continue;
|
|
}
|
|
// forceShadows allows nodraw surfaces to cast shadows
|
|
else if ( !token.Icmp( "forceShadows" ) ) {
|
|
SetMaterialFlag( MF_FORCESHADOWS );
|
|
continue;
|
|
}
|
|
// overlay / decal suppression
|
|
else if ( !token.Icmp( "noOverlays" ) ) {
|
|
allowOverlays = false;
|
|
continue;
|
|
}
|
|
// moster blood overlay forcing for alpha tested or translucent surfaces
|
|
else if ( !token.Icmp( "forceOverlays" ) ) {
|
|
pd->forceOverlays = true;
|
|
continue;
|
|
}
|
|
// translucent
|
|
else if ( !token.Icmp( "translucent" ) ) {
|
|
coverage = MC_TRANSLUCENT;
|
|
continue;
|
|
}
|
|
// global zero clamp
|
|
else if ( !token.Icmp( "zeroclamp" ) ) {
|
|
trpDefault = TR_CLAMP_TO_ZERO;
|
|
continue;
|
|
}
|
|
// global clamp
|
|
else if ( !token.Icmp( "clamp" ) ) {
|
|
trpDefault = TR_CLAMP;
|
|
continue;
|
|
}
|
|
// global clamp
|
|
else if ( !token.Icmp( "alphazeroclamp" ) ) {
|
|
trpDefault = TR_CLAMP_TO_ZERO;
|
|
continue;
|
|
}
|
|
// forceOpaque is used for skies-behind-windows
|
|
else if ( !token.Icmp( "forceOpaque" ) ) {
|
|
coverage = MC_OPAQUE;
|
|
continue;
|
|
}
|
|
// twoSided
|
|
else if ( !token.Icmp( "twoSided" ) ) {
|
|
cullType = CT_TWO_SIDED;
|
|
// twoSided implies no-shadows, because the shadow
|
|
// volume would be coplanar with the surface, giving depth fighting
|
|
// we could make this no-self-shadows, but it may be more important
|
|
// to receive shadows from no-self-shadow monsters
|
|
SetMaterialFlag( MF_NOSHADOWS );
|
|
}
|
|
// backSided
|
|
else if ( !token.Icmp( "backSided" ) ) {
|
|
cullType = CT_BACK_SIDED;
|
|
// the shadow code doesn't handle this, so just disable shadows.
|
|
// We could fix this in the future if there was a need.
|
|
SetMaterialFlag( MF_NOSHADOWS );
|
|
}
|
|
// foglight
|
|
else if ( !token.Icmp( "fogLight" ) ) {
|
|
fogLight = true;
|
|
continue;
|
|
}
|
|
// blendlight
|
|
else if ( !token.Icmp( "blendLight" ) ) {
|
|
blendLight = true;
|
|
continue;
|
|
}
|
|
// ambientLight
|
|
else if ( !token.Icmp( "ambientLight" ) ) {
|
|
ambientLight = true;
|
|
continue;
|
|
}
|
|
// mirror
|
|
else if ( !token.Icmp( "mirror" ) ) {
|
|
sort = SS_SUBVIEW;
|
|
coverage = MC_OPAQUE;
|
|
continue;
|
|
}
|
|
// noFog
|
|
else if ( !token.Icmp( "noFog" ) ) {
|
|
noFog = true;
|
|
continue;
|
|
}
|
|
// unsmoothedTangents
|
|
else if ( !token.Icmp( "unsmoothedTangents" ) ) {
|
|
unsmoothedTangents = true;
|
|
continue;
|
|
}
|
|
// lightFallofImage <imageprogram>
|
|
// specifies the image to use for the third axis of projected
|
|
// light volumes
|
|
else if ( !token.Icmp( "lightFalloffImage" ) ) {
|
|
str = R_ParsePastImageProgram( src );
|
|
idStr copy;
|
|
|
|
copy = str; // so other things don't step on it
|
|
lightFalloffImage = globalImages->ImageFromFile( copy, TF_DEFAULT, false, TR_CLAMP /* TR_CLAMP_TO_ZERO */, TD_DEFAULT );
|
|
continue;
|
|
}
|
|
// guisurf <guifile> | guisurf entity
|
|
// an entity guisurf must have an idUserInterface
|
|
// specified in the renderEntity
|
|
else if ( !token.Icmp( "guisurf" ) ) {
|
|
src.ReadTokenOnLine( &token );
|
|
if ( !token.Icmp( "entity" ) ) {
|
|
entityGui = 1;
|
|
} else if ( !token.Icmp( "entity2" ) ) {
|
|
entityGui = 2;
|
|
} else if ( !token.Icmp( "entity3" ) ) {
|
|
entityGui = 3;
|
|
} else {
|
|
gui = uiManager->FindGui( token.c_str(), true );
|
|
}
|
|
continue;
|
|
}
|
|
// sort
|
|
else if ( !token.Icmp( "sort" ) ) {
|
|
ParseSort( src );
|
|
continue;
|
|
}
|
|
// spectrum <integer>
|
|
else if ( !token.Icmp( "spectrum" ) ) {
|
|
src.ReadTokenOnLine( &token );
|
|
spectrum = atoi( token.c_str() );
|
|
continue;
|
|
}
|
|
// deform < sprite | tube | flare >
|
|
else if ( !token.Icmp( "deform" ) ) {
|
|
ParseDeform( src );
|
|
continue;
|
|
}
|
|
// decalInfo <staySeconds> <fadeSeconds> ( <start rgb> ) ( <end rgb> )
|
|
else if ( !token.Icmp( "decalInfo" ) ) {
|
|
ParseDecalInfo( src );
|
|
continue;
|
|
}
|
|
// renderbump <args...>
|
|
else if ( !token.Icmp( "renderbump") ) {
|
|
src.ParseRestOfLine( renderBump );
|
|
continue;
|
|
}
|
|
// diffusemap for stage shortcut
|
|
else if ( !token.Icmp( "diffusemap" ) ) {
|
|
str = R_ParsePastImageProgram( src );
|
|
idStr::snPrintf( buffer, sizeof( buffer ), "blend diffusemap\nmap %s\n}\n", str );
|
|
newSrc.LoadMemory( buffer, strlen(buffer), "diffusemap" );
|
|
newSrc.SetFlags( LEXFL_NOFATALERRORS | LEXFL_NOSTRINGCONCAT | LEXFL_NOSTRINGESCAPECHARS | LEXFL_ALLOWPATHNAMES );
|
|
ParseStage( newSrc, trpDefault );
|
|
newSrc.FreeSource();
|
|
continue;
|
|
}
|
|
// specularmap for stage shortcut
|
|
else if ( !token.Icmp( "specularmap" ) ) {
|
|
str = R_ParsePastImageProgram( src );
|
|
idStr::snPrintf( buffer, sizeof( buffer ), "blend specularmap\nmap %s\n}\n", str );
|
|
newSrc.LoadMemory( buffer, strlen(buffer), "specularmap" );
|
|
newSrc.SetFlags( LEXFL_NOFATALERRORS | LEXFL_NOSTRINGCONCAT | LEXFL_NOSTRINGESCAPECHARS | LEXFL_ALLOWPATHNAMES );
|
|
ParseStage( newSrc, trpDefault );
|
|
newSrc.FreeSource();
|
|
continue;
|
|
}
|
|
// normalmap for stage shortcut
|
|
else if ( !token.Icmp( "bumpmap" ) ) {
|
|
str = R_ParsePastImageProgram( src );
|
|
idStr::snPrintf( buffer, sizeof( buffer ), "blend bumpmap\nmap %s\n}\n", str );
|
|
newSrc.LoadMemory( buffer, strlen(buffer), "bumpmap" );
|
|
newSrc.SetFlags( LEXFL_NOFATALERRORS | LEXFL_NOSTRINGCONCAT | LEXFL_NOSTRINGESCAPECHARS | LEXFL_ALLOWPATHNAMES );
|
|
ParseStage( newSrc, trpDefault );
|
|
newSrc.FreeSource();
|
|
continue;
|
|
}
|
|
// DECAL_MACRO for backwards compatibility with the preprocessor macros
|
|
else if ( !token.Icmp( "DECAL_MACRO" ) ) {
|
|
// polygonOffset
|
|
SetMaterialFlag( MF_POLYGONOFFSET );
|
|
polygonOffset = 1;
|
|
|
|
// discrete
|
|
surfaceFlags |= SURF_DISCRETE;
|
|
contentFlags &= ~CONTENTS_SOLID;
|
|
|
|
// sort decal
|
|
sort = SS_DECAL;
|
|
|
|
// noShadows
|
|
SetMaterialFlag( MF_NOSHADOWS );
|
|
continue;
|
|
}
|
|
else if ( token == "{" ) {
|
|
// create the new stage
|
|
ParseStage( src, trpDefault );
|
|
continue;
|
|
}
|
|
else {
|
|
common->Warning( "unknown general material parameter '%s' in '%s'", token.c_str(), GetName() );
|
|
SetMaterialFlag( MF_DEFAULTED );
|
|
return;
|
|
}
|
|
}
|
|
|
|
// add _flat or _white stages if needed
|
|
AddImplicitStages();
|
|
|
|
// order the diffuse / bump / specular stages properly
|
|
SortInteractionStages();
|
|
|
|
// if we need to do anything with normals (lighting or environment mapping)
|
|
// and two sided lighting was asked for, flag
|
|
// shouldCreateBackSides() and change culling back to single sided,
|
|
// so we get proper tangent vectors on both sides
|
|
|
|
// we can't just call ReceivesLighting(), because the stages are still
|
|
// in temporary form
|
|
if ( cullType == CT_TWO_SIDED ) {
|
|
for ( i = 0 ; i < numStages ; i++ ) {
|
|
if ( pd->parseStages[i].lighting != SL_AMBIENT || pd->parseStages[i].texture.texgen != TG_EXPLICIT ) {
|
|
if ( cullType == CT_TWO_SIDED ) {
|
|
cullType = CT_FRONT_SIDED;
|
|
shouldCreateBackSides = true;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// currently a surface can only have one unique texgen for all the stages on old hardware
|
|
texgen_t firstGen = TG_EXPLICIT;
|
|
for ( i = 0; i < numStages; i++ ) {
|
|
if ( pd->parseStages[i].texture.texgen != TG_EXPLICIT ) {
|
|
if ( firstGen == TG_EXPLICIT ) {
|
|
firstGen = pd->parseStages[i].texture.texgen;
|
|
} else if ( firstGen != pd->parseStages[i].texture.texgen ) {
|
|
common->Warning( "material '%s' has multiple stages with a texgen", GetName() );
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
=========================
|
|
idMaterial::SetGui
|
|
=========================
|
|
*/
|
|
void idMaterial::SetGui( const char *_gui ) const {
|
|
gui = uiManager->FindGui( _gui, true, false, true );
|
|
}
|
|
|
|
/*
|
|
=========================
|
|
idMaterial::Parse
|
|
|
|
Parses the current material definition and finds all necessary images.
|
|
=========================
|
|
*/
|
|
bool idMaterial::Parse( const char *text, const int textLength ) {
|
|
idLexer src;
|
|
idToken token;
|
|
mtrParsingData_t parsingData;
|
|
|
|
src.LoadMemory( text, textLength, GetFileName(), GetLineNum() );
|
|
src.SetFlags( DECL_LEXER_FLAGS );
|
|
src.SkipUntilString( "{" );
|
|
|
|
// reset to the unparsed state
|
|
CommonInit();
|
|
|
|
memset( &parsingData, 0, sizeof( parsingData ) );
|
|
|
|
pd = &parsingData; // this is only valid during parse
|
|
|
|
// parse it
|
|
ParseMaterial( src );
|
|
|
|
// if we are doing an fs_copyfiles, also reference the editorImage
|
|
if ( cvarSystem->GetCVarInteger( "fs_copyFiles" ) ) {
|
|
GetEditorImage();
|
|
}
|
|
|
|
//
|
|
// count non-lit stages
|
|
numAmbientStages = 0;
|
|
int i;
|
|
for ( i = 0 ; i < numStages ; i++ ) {
|
|
if ( pd->parseStages[i].lighting == SL_AMBIENT ) {
|
|
numAmbientStages++;
|
|
}
|
|
}
|
|
|
|
// see if there is a subview stage
|
|
if ( sort == SS_SUBVIEW ) {
|
|
hasSubview = true;
|
|
} else {
|
|
hasSubview = false;
|
|
for ( i = 0 ; i < numStages ; i++ ) {
|
|
if ( pd->parseStages[i].texture.dynamic ) {
|
|
hasSubview = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
// automatically determine coverage if not explicitly set
|
|
if ( coverage == MC_BAD ) {
|
|
// automatically set MC_TRANSLUCENT if we don't have any interaction stages and
|
|
// the first stage is blended and not an alpha test mask or a subview
|
|
if ( !numStages ) {
|
|
// non-visible
|
|
coverage = MC_TRANSLUCENT;
|
|
} else if ( numStages != numAmbientStages ) {
|
|
// we have an interaction draw
|
|
coverage = MC_OPAQUE;
|
|
} else if (
|
|
( pd->parseStages[0].drawStateBits & GLS_DSTBLEND_BITS ) != GLS_DSTBLEND_ZERO ||
|
|
( pd->parseStages[0].drawStateBits & GLS_SRCBLEND_BITS ) == GLS_SRCBLEND_DST_COLOR ||
|
|
( pd->parseStages[0].drawStateBits & GLS_SRCBLEND_BITS ) == GLS_SRCBLEND_ONE_MINUS_DST_COLOR ||
|
|
( pd->parseStages[0].drawStateBits & GLS_SRCBLEND_BITS ) == GLS_SRCBLEND_DST_ALPHA ||
|
|
( pd->parseStages[0].drawStateBits & GLS_SRCBLEND_BITS ) == GLS_SRCBLEND_ONE_MINUS_DST_ALPHA
|
|
) {
|
|
// blended with the destination
|
|
coverage = MC_TRANSLUCENT;
|
|
} else {
|
|
coverage = MC_OPAQUE;
|
|
}
|
|
}
|
|
|
|
// translucent automatically implies noshadows
|
|
if ( coverage == MC_TRANSLUCENT ) {
|
|
SetMaterialFlag( MF_NOSHADOWS );
|
|
} else {
|
|
// mark the contents as opaque
|
|
contentFlags |= CONTENTS_OPAQUE;
|
|
}
|
|
|
|
// if we are translucent, draw with an alpha in the editor
|
|
if ( coverage == MC_TRANSLUCENT ) {
|
|
editorAlpha = 0.5;
|
|
} else {
|
|
editorAlpha = 1.0;
|
|
}
|
|
|
|
// the sorts can make reasonable defaults
|
|
if ( sort == SS_BAD ) {
|
|
if ( TestMaterialFlag(MF_POLYGONOFFSET) ) {
|
|
sort = SS_DECAL;
|
|
} else if ( coverage == MC_TRANSLUCENT ) {
|
|
sort = SS_MEDIUM;
|
|
} else {
|
|
sort = SS_OPAQUE;
|
|
}
|
|
}
|
|
|
|
// anything that references _currentRender will automatically get sort = SS_POST_PROCESS
|
|
// and coverage = MC_TRANSLUCENT
|
|
|
|
for ( i = 0 ; i < numStages ; i++ ) {
|
|
shaderStage_t *pStage = &pd->parseStages[i];
|
|
if ( pStage->texture.image == globalImages->currentRenderImage ) {
|
|
if ( sort != SS_PORTAL_SKY ) {
|
|
sort = SS_POST_PROCESS;
|
|
coverage = MC_TRANSLUCENT;
|
|
}
|
|
break;
|
|
}
|
|
if ( pStage->newStage ) {
|
|
for ( int j = 0 ; j < pStage->newStage->numFragmentProgramImages ; j++ ) {
|
|
if ( pStage->newStage->fragmentProgramImages[j] == globalImages->currentRenderImage ) {
|
|
if ( sort != SS_PORTAL_SKY ) {
|
|
sort = SS_POST_PROCESS;
|
|
coverage = MC_TRANSLUCENT;
|
|
}
|
|
i = numStages;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// set the drawStateBits depth flags
|
|
for ( i = 0 ; i < numStages ; i++ ) {
|
|
shaderStage_t *pStage = &pd->parseStages[i];
|
|
if ( sort == SS_POST_PROCESS ) {
|
|
// post-process effects fill the depth buffer as they draw, so only the
|
|
// topmost post-process effect is rendered
|
|
pStage->drawStateBits |= GLS_DEPTHFUNC_LESS;
|
|
} else if ( coverage == MC_TRANSLUCENT || pStage->ignoreAlphaTest ) {
|
|
// translucent surfaces can extend past the exactly marked depth buffer
|
|
pStage->drawStateBits |= GLS_DEPTHFUNC_LESS | GLS_DEPTHMASK;
|
|
} else {
|
|
// opaque and perforated surfaces must exactly match the depth buffer,
|
|
// which gets alpha test correct
|
|
pStage->drawStateBits |= GLS_DEPTHFUNC_EQUAL | GLS_DEPTHMASK;
|
|
}
|
|
}
|
|
|
|
// determine if this surface will accept overlays / decals
|
|
|
|
if ( pd->forceOverlays ) {
|
|
// explicitly flaged in material definition
|
|
allowOverlays = true;
|
|
} else {
|
|
if ( !IsDrawn() ) {
|
|
allowOverlays = false;
|
|
}
|
|
if ( Coverage() != MC_OPAQUE ) {
|
|
allowOverlays = false;
|
|
}
|
|
if ( GetSurfaceFlags() & SURF_NOIMPACT ) {
|
|
allowOverlays = false;
|
|
}
|
|
}
|
|
|
|
// add a tiny offset to the sort orders, so that different materials
|
|
// that have the same sort value will at least sort consistantly, instead
|
|
// of flickering back and forth
|
|
/* this messed up in-game guis
|
|
if ( sort != SS_SUBVIEW ) {
|
|
int hash, l;
|
|
|
|
l = name.Length();
|
|
hash = 0;
|
|
for ( int i = 0 ; i < l ; i++ ) {
|
|
hash ^= name[i];
|
|
}
|
|
sort += hash * 0.01;
|
|
}
|
|
*/
|
|
|
|
if (numStages) {
|
|
stages = (shaderStage_t *)R_StaticAlloc( numStages * sizeof( stages[0] ) );
|
|
memcpy( stages, pd->parseStages, numStages * sizeof( stages[0] ) );
|
|
}
|
|
|
|
if ( numOps ) {
|
|
ops = (expOp_t *)R_StaticAlloc( numOps * sizeof( ops[0] ) );
|
|
memcpy( ops, pd->shaderOps, numOps * sizeof( ops[0] ) );
|
|
}
|
|
|
|
if ( numRegisters ) {
|
|
expressionRegisters = (float *)R_StaticAlloc( numRegisters * sizeof( expressionRegisters[0] ) );
|
|
memcpy( expressionRegisters, pd->shaderRegisters, numRegisters * sizeof( expressionRegisters[0] ) );
|
|
}
|
|
|
|
// see if the registers are completely constant, and don't need to be evaluated
|
|
// per-surface
|
|
CheckForConstantRegisters();
|
|
|
|
pd = NULL; // the pointer will be invalid after exiting this function
|
|
|
|
// finish things up
|
|
if ( TestMaterialFlag( MF_DEFAULTED ) ) {
|
|
MakeDefault();
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
===================
|
|
idMaterial::Print
|
|
===================
|
|
*/
|
|
static const char *opNames[] = {
|
|
"OP_TYPE_ADD",
|
|
"OP_TYPE_SUBTRACT",
|
|
"OP_TYPE_MULTIPLY",
|
|
"OP_TYPE_DIVIDE",
|
|
"OP_TYPE_MOD",
|
|
"OP_TYPE_TABLE",
|
|
"OP_TYPE_GT",
|
|
"OP_TYPE_GE",
|
|
"OP_TYPE_LT",
|
|
"OP_TYPE_LE",
|
|
"OP_TYPE_EQ",
|
|
"OP_TYPE_NE",
|
|
"OP_TYPE_AND",
|
|
"OP_TYPE_OR"
|
|
};
|
|
|
|
void idMaterial::Print() const {
|
|
int i;
|
|
|
|
for ( i = EXP_REG_NUM_PREDEFINED ; i < GetNumRegisters() ; i++ ) {
|
|
common->Printf( "register %i: %f\n", i, expressionRegisters[i] );
|
|
}
|
|
common->Printf( "\n" );
|
|
for ( i = 0 ; i < numOps ; i++ ) {
|
|
const expOp_t *op = &ops[i];
|
|
if ( op->opType == OP_TYPE_TABLE ) {
|
|
common->Printf( "%i = %s[ %i ]\n", op->c, declManager->DeclByIndex( DECL_TABLE, op->a )->GetName(), op->b );
|
|
} else {
|
|
common->Printf( "%i = %i %s %i\n", op->c, op->a, opNames[ op->opType ], op->b );
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
===============
|
|
idMaterial::Save
|
|
===============
|
|
*/
|
|
bool idMaterial::Save( const char *fileName ) {
|
|
return ReplaceSourceFileText();
|
|
}
|
|
|
|
/*
|
|
===============
|
|
idMaterial::AddReference
|
|
===============
|
|
*/
|
|
void idMaterial::AddReference() {
|
|
refCount++;
|
|
|
|
for ( int i = 0; i < numStages; i++ ) {
|
|
shaderStage_t *s = &stages[i];
|
|
|
|
if ( s->texture.image ) {
|
|
s->texture.image->AddReference();
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
===============
|
|
idMaterial::EvaluateRegisters
|
|
|
|
Parameters are taken from the localSpace and the renderView,
|
|
then all expressions are evaluated, leaving the material registers
|
|
set to their apropriate values.
|
|
===============
|
|
*/
|
|
void idMaterial::EvaluateRegisters( float *registers, const float shaderParms[MAX_ENTITY_SHADER_PARMS],
|
|
const viewDef_t *view, idSoundEmitter *soundEmitter ) const {
|
|
int i, b;
|
|
expOp_t *op;
|
|
|
|
// copy the material constants
|
|
for ( i = EXP_REG_NUM_PREDEFINED ; i < numRegisters ; i++ ) {
|
|
registers[i] = expressionRegisters[i];
|
|
}
|
|
|
|
// copy the local and global parameters
|
|
registers[EXP_REG_TIME] = view->floatTime;
|
|
registers[EXP_REG_PARM0] = shaderParms[0];
|
|
registers[EXP_REG_PARM1] = shaderParms[1];
|
|
registers[EXP_REG_PARM2] = shaderParms[2];
|
|
registers[EXP_REG_PARM3] = shaderParms[3];
|
|
registers[EXP_REG_PARM4] = shaderParms[4];
|
|
registers[EXP_REG_PARM5] = shaderParms[5];
|
|
registers[EXP_REG_PARM6] = shaderParms[6];
|
|
registers[EXP_REG_PARM7] = shaderParms[7];
|
|
registers[EXP_REG_PARM8] = shaderParms[8];
|
|
registers[EXP_REG_PARM9] = shaderParms[9];
|
|
registers[EXP_REG_PARM10] = shaderParms[10];
|
|
registers[EXP_REG_PARM11] = shaderParms[11];
|
|
registers[EXP_REG_GLOBAL0] = view->renderView.shaderParms[0];
|
|
registers[EXP_REG_GLOBAL1] = view->renderView.shaderParms[1];
|
|
registers[EXP_REG_GLOBAL2] = view->renderView.shaderParms[2];
|
|
registers[EXP_REG_GLOBAL3] = view->renderView.shaderParms[3];
|
|
registers[EXP_REG_GLOBAL4] = view->renderView.shaderParms[4];
|
|
registers[EXP_REG_GLOBAL5] = view->renderView.shaderParms[5];
|
|
registers[EXP_REG_GLOBAL6] = view->renderView.shaderParms[6];
|
|
registers[EXP_REG_GLOBAL7] = view->renderView.shaderParms[7];
|
|
|
|
op = ops;
|
|
for ( i = 0 ; i < numOps ; i++, op++ ) {
|
|
switch( op->opType ) {
|
|
case OP_TYPE_ADD:
|
|
registers[op->c] = registers[op->a] + registers[op->b];
|
|
break;
|
|
case OP_TYPE_SUBTRACT:
|
|
registers[op->c] = registers[op->a] - registers[op->b];
|
|
break;
|
|
case OP_TYPE_MULTIPLY:
|
|
registers[op->c] = registers[op->a] * registers[op->b];
|
|
break;
|
|
case OP_TYPE_DIVIDE:
|
|
registers[op->c] = registers[op->a] / registers[op->b];
|
|
break;
|
|
case OP_TYPE_MOD:
|
|
b = (int)registers[op->b];
|
|
b = b != 0 ? b : 1;
|
|
registers[op->c] = (int)registers[op->a] % b;
|
|
break;
|
|
case OP_TYPE_TABLE:
|
|
{
|
|
const idDeclTable *table = static_cast<const idDeclTable *>( declManager->DeclByIndex( DECL_TABLE, op->a ) );
|
|
registers[op->c] = table->TableLookup( registers[op->b] );
|
|
}
|
|
break;
|
|
case OP_TYPE_SOUND:
|
|
if ( soundEmitter ) {
|
|
registers[op->c] = soundEmitter->CurrentAmplitude();
|
|
} else {
|
|
registers[op->c] = 0;
|
|
}
|
|
break;
|
|
case OP_TYPE_GT:
|
|
registers[op->c] = registers[ op->a ] > registers[op->b];
|
|
break;
|
|
case OP_TYPE_GE:
|
|
registers[op->c] = registers[ op->a ] >= registers[op->b];
|
|
break;
|
|
case OP_TYPE_LT:
|
|
registers[op->c] = registers[ op->a ] < registers[op->b];
|
|
break;
|
|
case OP_TYPE_LE:
|
|
registers[op->c] = registers[ op->a ] <= registers[op->b];
|
|
break;
|
|
case OP_TYPE_EQ:
|
|
registers[op->c] = registers[ op->a ] == registers[op->b];
|
|
break;
|
|
case OP_TYPE_NE:
|
|
registers[op->c] = registers[ op->a ] != registers[op->b];
|
|
break;
|
|
case OP_TYPE_AND:
|
|
registers[op->c] = registers[ op->a ] && registers[op->b];
|
|
break;
|
|
case OP_TYPE_OR:
|
|
registers[op->c] = registers[ op->a ] || registers[op->b];
|
|
break;
|
|
default:
|
|
common->FatalError( "R_EvaluateExpression: bad opcode" );
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
/*
|
|
=============
|
|
idMaterial::Texgen
|
|
=============
|
|
*/
|
|
texgen_t idMaterial::Texgen() const {
|
|
if ( stages ) {
|
|
for ( int i = 0; i < numStages; i++ ) {
|
|
if ( stages[ i ].texture.texgen != TG_EXPLICIT ) {
|
|
return stages[ i ].texture.texgen;
|
|
}
|
|
}
|
|
}
|
|
|
|
return TG_EXPLICIT;
|
|
}
|
|
|
|
/*
|
|
=============
|
|
idMaterial::GetImageWidth
|
|
=============
|
|
*/
|
|
int idMaterial::GetImageWidth( void ) const {
|
|
assert( GetStage(0) && GetStage(0)->texture.image );
|
|
return GetStage(0)->texture.image->uploadWidth;
|
|
}
|
|
|
|
/*
|
|
=============
|
|
idMaterial::GetImageHeight
|
|
=============
|
|
*/
|
|
int idMaterial::GetImageHeight( void ) const {
|
|
assert( GetStage(0) && GetStage(0)->texture.image );
|
|
return GetStage(0)->texture.image->uploadHeight;
|
|
}
|
|
|
|
/*
|
|
=============
|
|
idMaterial::CinematicLength
|
|
=============
|
|
*/
|
|
int idMaterial::CinematicLength() const {
|
|
if ( !stages || !stages[0].texture.cinematic ) {
|
|
return 0;
|
|
}
|
|
return stages[0].texture.cinematic->AnimationLength();
|
|
}
|
|
|
|
/*
|
|
=============
|
|
idMaterial::UpdateCinematic
|
|
=============
|
|
*/
|
|
void idMaterial::UpdateCinematic( int time ) const {
|
|
if ( !stages || !stages[0].texture.cinematic || !backEnd.viewDef ) {
|
|
return;
|
|
}
|
|
stages[0].texture.cinematic->ImageForTime( tr.primaryRenderView.time );
|
|
}
|
|
|
|
/*
|
|
=============
|
|
idMaterial::CloseCinematic
|
|
=============
|
|
*/
|
|
void idMaterial::CloseCinematic( void ) const {
|
|
for( int i = 0; i < numStages; i++ ) {
|
|
if ( stages[i].texture.cinematic ) {
|
|
stages[i].texture.cinematic->Close();
|
|
delete stages[i].texture.cinematic;
|
|
stages[i].texture.cinematic = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
=============
|
|
idMaterial::ResetCinematicTime
|
|
=============
|
|
*/
|
|
void idMaterial::ResetCinematicTime( int time ) const {
|
|
for( int i = 0; i < numStages; i++ ) {
|
|
if ( stages[i].texture.cinematic ) {
|
|
stages[i].texture.cinematic->ResetTime( time );
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
=============
|
|
idMaterial::ConstantRegisters
|
|
=============
|
|
*/
|
|
const float *idMaterial::ConstantRegisters() const {
|
|
if ( !r_useConstantMaterials.GetBool() ) {
|
|
return NULL;
|
|
}
|
|
return constantRegisters;
|
|
}
|
|
|
|
/*
|
|
==================
|
|
idMaterial::CheckForConstantRegisters
|
|
|
|
As of 5/2/03, about half of the unique materials loaded on typical
|
|
maps are constant, but 2/3 of the surface references are.
|
|
This is probably an optimization of dubious value.
|
|
==================
|
|
*/
|
|
void idMaterial::CheckForConstantRegisters() {
|
|
if ( !pd->registersAreConstant ) {
|
|
return;
|
|
}
|
|
|
|
// evaluate the registers once, and save them
|
|
constantRegisters = (float *)R_ClearedStaticAlloc( GetNumRegisters() * sizeof( float ) );
|
|
|
|
float shaderParms[MAX_ENTITY_SHADER_PARMS];
|
|
memset( shaderParms, 0, sizeof( shaderParms ) );
|
|
viewDef_t viewDef;
|
|
memset( &viewDef, 0, sizeof( viewDef ) );
|
|
|
|
EvaluateRegisters( constantRegisters, shaderParms, &viewDef, 0 );
|
|
}
|
|
|
|
/*
|
|
===================
|
|
idMaterial::ImageName
|
|
===================
|
|
*/
|
|
const char *idMaterial::ImageName( void ) const {
|
|
if ( numStages == 0 ) {
|
|
return "_scratch";
|
|
}
|
|
idImage *image = stages[0].texture.image;
|
|
if ( image ) {
|
|
return image->imgName;
|
|
}
|
|
return "_scratch";
|
|
}
|
|
|
|
/*
|
|
===================
|
|
idMaterial::SetImageClassifications
|
|
|
|
Just for image resource tracking.
|
|
===================
|
|
*/
|
|
void idMaterial::SetImageClassifications( int tag ) const {
|
|
for ( int i = 0 ; i < numStages ; i++ ) {
|
|
idImage *image = stages[i].texture.image;
|
|
if ( image ) {
|
|
image->SetClassification( tag );
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
=================
|
|
idMaterial::Size
|
|
=================
|
|
*/
|
|
size_t idMaterial::Size( void ) const {
|
|
return sizeof( idMaterial );
|
|
}
|
|
|
|
/*
|
|
===================
|
|
idMaterial::SetDefaultText
|
|
===================
|
|
*/
|
|
bool idMaterial::SetDefaultText( void ) {
|
|
// if there exists an image with the same name
|
|
if ( 1 ) { //fileSystem->ReadFile( GetName(), NULL ) != -1 ) {
|
|
char generated[2048];
|
|
idStr::snPrintf( generated, sizeof( generated ),
|
|
"material %s // IMPLICITLY GENERATED\n"
|
|
"{\n"
|
|
"{\n"
|
|
"blend blend\n"
|
|
"colored\n"
|
|
"map \"%s\"\n"
|
|
"clamp\n"
|
|
"}\n"
|
|
"}\n", GetName(), GetName() );
|
|
SetText( generated );
|
|
return true;
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/*
|
|
===================
|
|
idMaterial::DefaultDefinition
|
|
===================
|
|
*/
|
|
const char *idMaterial::DefaultDefinition() const {
|
|
return
|
|
"{\n"
|
|
"\t" "{\n"
|
|
"\t\t" "blend\tblend\n"
|
|
"\t\t" "map\t\t_default\n"
|
|
"\t" "}\n"
|
|
"}";
|
|
}
|
|
|
|
|
|
/*
|
|
===================
|
|
idMaterial::GetBumpStage
|
|
===================
|
|
*/
|
|
const shaderStage_t *idMaterial::GetBumpStage( void ) const {
|
|
for ( int i = 0 ; i < numStages ; i++ ) {
|
|
if ( stages[i].lighting == SL_BUMP ) {
|
|
return &stages[i];
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
===================
|
|
idMaterial::ReloadImages
|
|
===================
|
|
*/
|
|
void idMaterial::ReloadImages( bool force ) const
|
|
{
|
|
for ( int i = 0 ; i < numStages ; i++ ) {
|
|
if ( stages[i].newStage ) {
|
|
for ( int j = 0 ; j < stages[i].newStage->numFragmentProgramImages ; j++ ) {
|
|
if ( stages[i].newStage->fragmentProgramImages[j] ) {
|
|
stages[i].newStage->fragmentProgramImages[j]->Reload( false, force );
|
|
}
|
|
}
|
|
} else if ( stages[i].texture.image ) {
|
|
stages[i].texture.image->Reload( false, force );
|
|
}
|
|
}
|
|
}
|