/*
Copyright (C) 2002-2003 Victor Luchits

This program 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 2
of the License, or (at your option) any later version.

This program 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 this program; if not, write to the Free Software
Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.

*/
// r_shader.c - based on code by Stephen C. Taylor
// Ported to FTE from qfusion, there are numerous changes since then.


#include "quakedef.h"
#ifndef SERVERONLY
#include "glquake.h"
#include "shader.h"

#include "hash.h"



#include <ctype.h>

extern texid_t missing_texture;
texid_t r_whiteimage, r_blackimage;
qboolean shader_reload_needed;
static qboolean shader_rescan_needed;

sh_config_t sh_config;

//cvars that affect shader generation
cvar_t r_vertexlight = CVARFD("r_vertexlight", "0", CVAR_SHADERSYSTEM, "Hack loaded shaders to remove detail pass and lightmap sampling for faster rendering.");
cvar_t r_forceprogramify = CVARAFD("r_forceprogramify", "0", "dpcompat_makeshitup", CVAR_SHADERSYSTEM, "Reduce the shader to a single texture, and then make stuff up about its mother. The resulting fist fight results in more colour when you shine a light upon its face.\nSet to 2 to ignore 'depthfunc equal' and 'tcmod scale' in order to tolerate bizzare shaders made for a bizzare engine.\nBecause most shaders made for DP are by people who _clearly_ have no idea what the heck they're doing, you'll typically need the '2' setting.");
cvar_t dpcompat_nopremulpics = CVARFD("dpcompat_nopremulpics", "0", CVAR_SHADERSYSTEM, "By default FTE uses premultiplied alpha for hud/2d images, while DP does not (which results in halos with low-res content). Unfortunately DDS files would need to be recompressed, resulting in visible issues.");
extern cvar_t r_glsl_offsetmapping_reliefmapping;
extern cvar_t r_drawflat;
extern cvar_t r_shaderblobs;
extern cvar_t r_tessellation;
extern cvar_t gl_compress;

//backend fills this in to say the max fixed-function pass count (often 1, where its emulated by us, because we're too lazy).
int be_maxpasses;


#define Q_stricmp stricmp
#define Q_strnicmp strnicmp
#define clamp(v,min, max) (v) = (((v)<(min))?(min):(((v)>(max))?(max):(v)));

typedef union {
	float			f;
	unsigned int	i;
} float_int_t;
qbyte FloatToByte( float x )
{
	static float_int_t f2i;

	// shift float to have 8bit fraction at base of number
	f2i.f = x + 32768.0f;

	// then read as integer and kill float bits...
	return (qbyte) min(f2i.i & 0x7FFFFF, 255);
}



cvar_t r_detailtextures;

#define MAX_TOKEN_CHARS sizeof(com_token)

char *COM_ParseExt (char **data_p, qboolean nl, qboolean comma)
{
	int		c;
	int		len;
	char	*data;

	COM_AssertMainThread("COM_ParseExt");

	data = *data_p;
	len = 0;
	com_token[0] = 0;

	if (!data)
	{
		*data_p = NULL;
		return "";
	}

// skip whitespace
skipwhite:
	while ((c = (unsigned char)*data) <= ' ')
	{
		if (c == 0)
		{
			*data_p = NULL;
			return "";
		}
		if (c == '\n' && !nl)
		{
			*data_p = data;
			return com_token;
		}
		data++;
	}

// skip // comments
	if (c == '/' && data[1] == '/')
	{
		while (*data && *data != '\n')
			data++;
		goto skipwhite;
	}
// skip /* comments
	if (c == '/' && data[1] == '*')
	{
		char *start = data;
		data+=2;
		for(;data[0];)
		{
			if (data[0] == '*' && data[1] == '/')
			{
				data+=2;
				break;
			}
			if (*data == '\n' && !nl)
			{
				*data_p = start;
				return com_token;
			}
			data++;
		}
		goto skipwhite;
	}

// handle quoted strings specially
	if (c == '\"')
	{
		data++;
		while (1)
		{
			c = *data++;
			if (c=='\"' || !c)
			{
				com_token[len] = 0;
				*data_p = data;
				return com_token;
			}
			if (len < MAX_TOKEN_CHARS)
			{
				com_token[len] = c;
				len++;
			}
		}
	}

// parse a regular word
	do
	{
		if (len < MAX_TOKEN_CHARS)
		{
			com_token[len] = c;
			len++;
		}
		data++;
		c = *data;
		if (c == ',' && len && comma)
			break;
	} while (c>32);

	if (len == MAX_TOKEN_CHARS)
	{
		Con_DPrintf ("Token exceeded %i chars, discarded.\n", (int)MAX_TOKEN_CHARS);
		len = 0;
	}
	com_token[len] = 0;

	*data_p = data;
	return com_token;
}

static float Com_FloatArgument(const char *shadername, char *arg, size_t arglen, float def)
{
	const char *var;

	//grab an argument instead, otherwise 0
	var = shadername;
	while((var = strchr(var, '#')))
	{
		var++;
		if (!strnicmp(var, arg, arglen))
		{
			if (var[arglen] == '=')
				return strtod(var+arglen+1, NULL);
			if (var[arglen] == '#' || !var[arglen])
				return 1;	//present, but no value
		}
		var++;
	}
	return def;	//not present.
}
#define Shader_FloatArgument(s,k) (Com_FloatArgument(s->name,k,strlen(k),0))





#define HASH_SIZE	128

#define SPF_DEFAULT		0u	/*quake3/fte internal*/
#define SPF_PROGRAMIFY	(1u<<0)	/*automatically replace known glsl, pulling in additional textures+effects from a single primary pass*/
#define SPF_DOOM3		(1u<<1)	/*any commands, args, etc, should be interpretted according to doom3's norms*/

typedef struct shaderparsestate_s
{
	shader_t *s;		//the shader we're parsing
	shaderpass_t *pass;	//the pass we're currently parsing
	char *ptr;			//the src file pointer we're at
	char *sourcename;	//the name of the shader file being read (or '<code>')

	const char *forcedshader;
	unsigned int parseflags;	//SPF_*
	qboolean droppass;

	//for dpwater compat, used to generate a program
	int dpwatertype;
	float reflectmin;
	float reflectmax;
	float reflectfactor;
	float refractfactor;
	vec3_t refractcolour;
	vec3_t reflectcolour;
	float wateralpha;

	char **saveshaderbody;

	//FIXME: rtlights can't respond to these
	int offsetmappingmode;
	float offsetmappingscale;
	float offsetmappingbias;
	float specularexpscale; //*32 ish
	float specularvalscale; //*1 ish
} parsestate_t;

typedef struct shaderkey_s
{
	char			*keyword;
	void			(*func)( parsestate_t *ps, char **ptr );
	char			*prefix;
} shaderkey_t;
typedef struct shadercachefile_s {
	char *data;
	size_t length;
	unsigned int parseflags;
	char forcedshadername[64];
	struct shadercachefile_s *next;
	char name[1];
} shadercachefile_t;
typedef struct shadercache_s {
	shadercachefile_t *source;
	size_t offset;
	struct shadercache_s *hash_next;
	char name[1];
} shadercache_t;

static shadercachefile_t *shaderfiles;	//contents of a .shader file
static shadercache_t **shader_hash;		//locations of known inactive shaders.

unsigned int r_numshaders;	//number of active slots in r_shaders array.
unsigned int r_maxshaders;	//max length of r_shaders array. resized if exceeded.
shader_t	**r_shaders;	//list of active shaders for a id->shader lookup
static hashtable_t shader_active_hash;	//list of active shaders for a name->shader lookup
void *shader_active_hash_mem;

//static char		r_skyboxname[MAX_QPATH];
//static float	r_skyheight;

static char *Shader_Skip(const char *file, const char *shadername, char *ptr);
static qboolean Shader_Parsetok(parsestate_t *ps, shaderkey_t *keys, char *token);
static void Shader_ParseFunc(parsestate_t *ps, const char *functype, char **args, shaderfunc_t *func);
static void Shader_MakeCache(const char *path, unsigned int parseflags);
static qboolean Shader_LocateSource(char *name, char **buf, size_t *bufsize, size_t *offset, shadercachefile_t **sourcefile);
static void Shader_ReadShader(parsestate_t *ps, char *shadersource, shadercachefile_t *sourcefile);
static qboolean Shader_ParseShader(parsestate_t *ps, char *parsename);

//===========================================================================

static qboolean Shader_EvaluateCondition(shader_t *shader, char **ptr)
{
	char *token;
	cvar_t *cv;
	float lhs;
	qboolean conditiontrue = true;
	token = COM_ParseExt(ptr, false, false);
	if (*token == '!')
	{
		conditiontrue = false;
		token++;
		if (!*token)
			token = COM_ParseExt(ptr, false, false);
	}

	if (*token >= '0' && *token <= '9')
		lhs = strtod(token, NULL);
	else
	{
		if (*token == '$')
			token++;

		if (*token == '#')
			lhs = !!Shader_FloatArgument(shader, token+1);
		else if (!Q_stricmp(token, "lpp"))
			lhs = r_lightprepass;
		else if (!Q_stricmp(token, "lightmap"))
			lhs = !r_fullbright.value;
		else if (!Q_stricmp(token, "deluxmap") || !Q_stricmp(token, "deluxe"))
			lhs = r_deluxemapping;
		else if (!Q_stricmp(token, "softwarebanding"))
			lhs = r_softwarebanding;

		//normalmaps are generated if they're not already known.
		else if (!Q_stricmp(token, "normalmap"))
			lhs = r_loadbumpmapping;

		else if (!Q_stricmp(token, "vulkan"))
			lhs = (qrenderer == QR_VULKAN);
		else if (!Q_stricmp(token, "opengl"))
			lhs = (qrenderer == QR_OPENGL);
		else if (!Q_stricmp(token, "d3d8"))
			lhs = (qrenderer == QR_DIRECT3D8);
		else if (!Q_stricmp(token, "d3d9"))
			lhs = (qrenderer == QR_DIRECT3D9);
		else if (!Q_stricmp(token, "d3d11"))
			lhs = (qrenderer == QR_DIRECT3D11);
		else if (!Q_stricmp(token, "gles"))
		{
#ifdef GLQUAKE
			lhs = ((qrenderer == QR_OPENGL) && gl_config.gles);
#else
			lhs = false;
#endif
		}
		else if (!Q_stricmp(token, "nofixed"))
			lhs = sh_config.progs_required;
		else if (!Q_stricmp(token, "glsl"))
			lhs = ((qrenderer == QR_OPENGL) && sh_config.progs_supported);
		else if (!Q_stricmp(token, "hlsl"))
			lhs = ((qrenderer == QR_DIRECT3D9 || qrenderer == QR_DIRECT3D11) && sh_config.progs_supported);	
		else if (!Q_stricmp(token, "haveprogram"))
			lhs = !!shader->prog;
		else if (!Q_stricmp(token, "programs"))
			lhs = sh_config.progs_supported;
		else if (!Q_stricmp(token, "diffuse"))
			lhs = true;
		else if (!Q_stricmp(token, "specular"))
			lhs = false;
		else if (!Q_stricmp(token, "fullbright"))
			lhs = false;
		else if (!Q_stricmp(token, "topoverlay"))
			lhs = false;
		else if (!Q_stricmp(token, "loweroverlay"))
			lhs = false;

		//these are for compat/documentation purposes with qfusion/warsow
		else if (!Q_stricmp(token, "maxTextureSize"))
			lhs = sh_config.texture2d_maxsize;
		else if (!Q_stricmp(token, "maxTextureCubemapSize"))
			lhs = sh_config.texturecube_maxsize;
		else if (!Q_stricmp(token, "maxTextureUnits"))
			lhs = 0;
		else if (!Q_stricmp(token, "textureCubeMap"))
			lhs = sh_config.havecubemaps;
//		else if (!Q_stricmp(token, "GLSL"))
//			lhs = 1;
		else if (!Q_stricmp(token, "deluxeMaps") || !Q_stricmp(token, "deluxe"))
			lhs = r_deluxemapping;
		else if (!Q_stricmp(token, "portalMaps"))
			lhs = false;
		//end qfusion

		else
		{
			cv = Cvar_Get(token, "", 0, "Shader Conditions");
			if (cv)
			{
				cv->flags |= CVAR_SHADERSYSTEM;
				lhs = cv->value;
			}
			else
			{
				Con_Printf("Shader_EvaluateCondition: '%s' is not a cvar\n", token);
				lhs = 0;
			}
		}
	}
	if (*token)
		token = COM_ParseExt(ptr, false, false);
	if (*token)
	{
		float rhs;
		char cmp[4];
		memcpy(cmp, token, 4);
		token = COM_ParseExt(ptr, false, false);
		rhs = atof(token);
		if (!strcmp(cmp, "!="))
			conditiontrue = lhs != rhs;
		else if (!strcmp(cmp, "=="))
			conditiontrue = lhs == rhs;
		else if (!strcmp(cmp, "<"))
			conditiontrue = lhs < rhs;
		else if (!strcmp(cmp, "<="))
			conditiontrue = lhs <= rhs;
		else if (!strcmp(cmp, ">"))
			conditiontrue = lhs > rhs;
		else if (!strcmp(cmp, ">="))
			conditiontrue = lhs >= rhs;
		else
			conditiontrue = false;
	}
	else
	{
		conditiontrue = conditiontrue == !!lhs;
	}
	if (*token)
		token = COM_ParseExt(ptr, false, false);
	if (!strcmp(token, "&&"))
		return Shader_EvaluateCondition(shader, ptr) && conditiontrue;
	if (!strcmp(token, "||"))
		return Shader_EvaluateCondition(shader, ptr) || conditiontrue;

	return conditiontrue;
}

static char *Shader_ParseExactString(char **ptr)
{
	char *token;

	if (!ptr || !(*ptr))
		return "";
	if (!**ptr || **ptr == '}')
		return "";

	token = COM_ParseExt(ptr, false, false);
	return token;
}

static char *Shader_ParseString(char **ptr)
{
	char *token;

	if (!ptr || !(*ptr))
		return "";
	while(**ptr == ' ' || **ptr == '\t')
		*ptr+=1;
	if (!**ptr || **ptr == '}')
		return "";

	token = COM_ParseExt(ptr, false, true);
	Q_strlwr ( token );

	return token;
}

static char *Shader_ParseSensString(char **ptr)
{
	char *token;

	if (!ptr || !(*ptr))
		return "";
	if (!**ptr || **ptr == '}')
		return "";

	token = COM_ParseExt(ptr, false, true);

	return token;
}

static float Shader_ParseFloat(shader_t *shader, char **ptr, float defaultval)
{
	char *token;
	if (!ptr || !(*ptr))
		return defaultval;
	if (!**ptr || **ptr == '}')
		return defaultval;

	token = COM_ParseExt(ptr, false, true);
	if (*token == '$')
	{
		if (token[1] == '#')
		{
			return Shader_FloatArgument(shader, token+2);
		}
		else
		{
			cvar_t *var;
			var = Cvar_FindVar(token+1);
			if (var)
			{
				if (*var->string)
					return var->value;
				else
					return defaultval;
			}
		}
	}
	if (!*token)
		return defaultval;
	return atof(token);
}

static void Shader_ParseVector(shader_t *shader, char **ptr, vec3_t v)
{
	char *scratch;
	char *token;
	qboolean bracket;
	qboolean fromcvar = false;

	token = Shader_ParseString(ptr);
	if (*token == '$')
	{
		cvar_t *var;
		var = Cvar_FindVar(token+1);
		if (!var)
		{
			v[0] = 1;
			v[1] = 1;
			v[2] = 1;
			return;
		}
		var->flags |= CVAR_SHADERSYSTEM;
		ptr = &scratch;
		scratch = var->string;

		token = Shader_ParseString( ptr);
		fromcvar = true;
	}
	if (!Q_stricmp (token, "("))
	{
		bracket = true;
		token = Shader_ParseString(ptr);
	}
	else if (token[0] == '(')
	{
		bracket = true;
		token = &token[1];
	}
	else
		bracket = false;

	if (!strncmp(token, "0x", 2))
	{	//0xRRGGBB
		unsigned int hex = strtoul(token, NULL, 0);
		v[0] = ((hex>>16)&255)/255.;
		v[1] = ((hex>> 8)&255)/255.;
		v[2] = ((hex>> 0)&255)/255.;
		return;
	}

	v[0] = atof ( token );
	
	token = Shader_ParseString ( ptr );
	if ( !token[0] ) {
		v[1] = fromcvar?v[0]:0;
	} else if (bracket &&  token[strlen(token)-1] == ')' ) {
		bracket = false;
		token[strlen(token)-1] = 0;
		v[1] = atof ( token );
	} else {
		v[1] = atof ( token );
	}

	token = Shader_ParseString ( ptr );
	if ( !token[0] ) {
		v[2] = fromcvar?v[1]:0;
	} else if (bracket && token[strlen(token)-1] == ')' ) {
		token[strlen(token)-1] = 0;
		v[2] = atof ( token );
	} else {
		v[2] = atof ( token );
		if ( bracket ) {
			Shader_ParseString ( ptr );
		}
	}

	/*
	if (v[0] > 5 || v[1] > 5 || v[2] > 5)
	{
		VectorScale(v, 1.0f/255, v);
	}
	*/
}

qboolean Shader_ParseSkySides (char *shadername, char *texturename, texid_t *images)
{	//FIXME: use Image_LoadCubemapTextureData to load the faces
	//if possible directly use a 7th/cubemap texture instead
	//this requires fixing the sky code to not do the random transforms thing though.
	qboolean allokay = true;
	int i, ss, sp;
	char path[MAX_QPATH];

	static char	*skyname_suffix[][6] = {
		{"rt", "bk", "lf", "ft", "up", "dn"},
//		{"px", "py", "nx", "ny", "pz", "nz"},
//		{"posx", "posy", "negx", "negy", "posz", "negz"},
//		{"_px", "_py", "_nx", "_ny", "_pz", "_nz"},
//		{"_posx", "_posy", "_negx", "_negy", "_posz", "_negz"},
		{"_rt", "_bk", "_lf", "_ft", "_up", "_dn"}
	};

	static char *skyname_pattern[] = {
		"%s_%s",
		"%s%s",
		"env/%s%s",
		"gfx/env/%s%s"
	};

	if (*texturename == '$')
	{
		cvar_t *v;
		v = Cvar_FindVar(texturename+1);
		if (v)
			texturename = v->string;
	}
	if (!*texturename)
		texturename = "-";

	for ( i = 0; i < 6; i++ )
	{
		if ( texturename[0] == '-' )
		{
			images[i] = r_nulltex;
			allokay = true;
		}
		else
		{
			for (sp = 0; sp < sizeof(skyname_pattern)/sizeof(skyname_pattern[0]); sp++)
			{
				for (ss = 0; ss < sizeof(skyname_suffix)/sizeof(skyname_suffix[0]); ss++)
				{
					Q_snprintfz ( path, sizeof(path), skyname_pattern[sp], texturename, skyname_suffix[ss][i] );
					images[i] = R_LoadHiResTexture ( path, NULL, IF_NOALPHA|IF_CLAMP|IF_LOADNOW);
					if (images[i]->width)
						break;
				}
				if (images[i]->width)
					break;
			}
			if (!images[i]->width)
			{
				Con_Printf("Sky \"%s\" missing texture: %s\n", shadername, path);
				images[i] = missing_texture;
				allokay = false;
			}
		}
	}
	return allokay;
}

static void Shader_ParseFunc (parsestate_t *ps, const char *functype, char **ptr, shaderfunc_t *func)
{
	shader_t *shader = ps->s;
	char *token;

	token = Shader_ParseString (ptr);
	if (!Q_stricmp (token, "sin"))
		func->type = SHADER_FUNC_SIN;
	else if (!Q_stricmp (token, "triangle"))
		func->type = SHADER_FUNC_TRIANGLE;
	else if (!Q_stricmp (token, "square"))
		func->type = SHADER_FUNC_SQUARE;
	else if (!Q_stricmp (token, "sawtooth"))
		func->type = SHADER_FUNC_SAWTOOTH;
	else if (!Q_stricmp (token, "inversesawtooth"))
		func->type = SHADER_FUNC_INVERSESAWTOOTH;
	else if (!Q_stricmp (token, "noise"))
		func->type = SHADER_FUNC_NOISE;
	else
	{
		if (!Q_stricmp (token, "distanceramp"))	//QFusion
			;
		else
			Con_Printf("%s: %s: unknown %s \"%s\"\n", ps->sourcename, shader->name, functype, token);
		func->type = SHADER_FUNC_CONSTANT;	//not supported...
		Shader_ParseFloat (shader, ptr, 0);
		Shader_ParseFloat (shader, ptr, 0);
		Shader_ParseFloat (shader, ptr, 0);
		Shader_ParseFloat (shader, ptr, 0);
		Vector4Set(func->args, 255, 255, 255, 255);
		return;
	}

	func->args[0] = Shader_ParseFloat (shader, ptr, 0);
	func->args[1] = Shader_ParseFloat (shader, ptr, 0);
	func->args[2] = Shader_ParseFloat (shader, ptr, 0);
	func->args[3] = Shader_ParseFloat (shader, ptr, 0);
}

//===========================================================================

static int Shader_SetImageFlags(parsestate_t *parsestate, shaderpass_t *pass, char **name, int flags)
{
	//fixme: pass flags should be handled elsewhere.
	for(;name;)
	{
		if (!Q_strnicmp(*name, "$rt:", 4))
		{
			*name += 4;
			flags |= IF_NOMIPMAP|IF_CLAMP|IF_RENDERTARGET;
			if (!(flags & (IF_NEAREST|IF_LINEAR)))
				flags |= IF_LINEAR;
		}
		else if (!Q_strnicmp(*name, "$clamp:", 7))
		{
			*name += 7;
			flags |= IF_CLAMP;
		}
		else if (!Q_strnicmp(*name, "$premul:", 8))
		{	//have the engine premultiply textures for you, instead of needing to do it in an editor.
			*name += 8;
			flags |= IF_PREMULTIPLYALPHA;
		}
		else if (!Q_strnicmp(*name, "$2d:", 4))
		{
			*name+=4;
			flags = (flags&~IF_TEXTYPEMASK) | IF_TEXTYPE_2D;
		}
		else if (!Q_strnicmp(*name, "$2darray:", 9))
		{
			*name+=9;
			flags = (flags&~IF_TEXTYPEMASK) | IF_TEXTYPE_2D_ARRAY;
		}
		else if (!Q_strnicmp(*name, "$3d:", 4))
		{
			*name+=4;
			flags = (flags&~IF_TEXTYPEMASK) | IF_TEXTYPE_3D;
		}
		else if (!Q_strnicmp(*name, "$cube:", 6))
		{
			*name+=6;
			flags = (flags&~IF_TEXTYPEMASK) | IF_TEXTYPE_CUBE;
		}
		else if (!Q_strnicmp(*name, "$cubearray:", 11))
		{
			*name+=11;
			flags = (flags&~IF_TEXTYPEMASK) | IF_TEXTYPE_CUBE_ARRAY;
		}
		else if (!Q_strnicmp(*name, "$srgb:", 6))
		{
			*name+=6;
			flags &= ~IF_NOSRGB;
			flags |= IF_SRGB;
		}
		else if (!Q_strnicmp(*name, "$nosrgb:", 8))
		{
			*name+=8;
			flags &= ~IF_SRGB;
			flags |= IF_NOSRGB;
		}
		else if (!Q_strnicmp(*name, "$nearest:", 9))
		{
			*name+=9;
			flags &= ~IF_LINEAR;
			flags |= IF_NEAREST;
		}
		else if (!Q_strnicmp(*name, "$linear:", 8))
		{
			*name+=8;
			flags &= ~IF_NEAREST;
			flags |= IF_LINEAR;
		}
		else
			break;
	}

//	if (shader->flags & SHADER_SKY)
//		flags |= IF_SKY;
	if (parsestate->s->flags & SHADER_NOMIPMAPS)
		flags |= IF_NOMIPMAP;
	if (parsestate->s->flags & SHADER_NOPICMIP)
		flags |= IF_NOPICMIP;
	flags |= IF_MIPCAP;

	if (pass)
	{
		if (flags & IF_CLAMP)
			pass->flags |= SHADER_PASS_CLAMP;
		if (flags & IF_LINEAR)
		{
			pass->flags &= ~SHADER_PASS_NEAREST;
			pass->flags |= SHADER_PASS_LINEAR;
		}
		else if (flags & IF_NEAREST)
		{
			pass->flags &= ~SHADER_PASS_LINEAR;
			pass->flags |= SHADER_PASS_NEAREST;
		}
	}

	return flags;
}

texid_t R_LoadColourmapImage(void)
{
	//FIXME: cache the result, because this is abusive
	unsigned int w = 256, h = VID_GRADES;
	unsigned int x;
	unsigned int data[256*(VID_GRADES)];
	qbyte *colourmappal = (qbyte *)FS_LoadMallocFile ("gfx/colormap.lmp", NULL);
#if defined(Q2CLIENT) && defined(IMAGEFMT_PCX)
	if (!colourmappal)
	{
		size_t sz;
		qbyte *pcx = FS_LoadMallocFile("pics/colormap.pcx", &sz);
		if (pcx)
		{
			colourmappal = Z_Malloc(256*VID_GRADES);
			ReadPCXData(pcx, sz, 256, VID_GRADES, colourmappal);
			BZ_Free(pcx);
		}
	}
#endif
	if (colourmappal)
	{
		for (x = 0; x < sizeof(data)/sizeof(data[0]); x++)
			data[x] = d_8to24rgbtable[colourmappal[x]];
	}
	else
	{	//erk
		//fixme: generate a proper colourmap
		for (x = 0; x < sizeof(data)/sizeof(data[0]); x++)
		{
			int r, g, b;
			float l = 1.0-((x/256)/(float)VID_GRADES);
			r = d_8to24rgbtable[x & 0xff];
			g = (r>>16)&0xff;
			b = (r>>8)&0xff;
			r = (r>>0)&0xff;
			data[x] = d_8to24rgbtable[GetPaletteIndex(r*l,g*l,b*l)];
		}
	}
	BZ_Free(colourmappal);
	return R_LoadTexture("$colourmap", w, h, TF_RGBA32, data, IF_NOMIPMAP|IF_NOPICMIP|IF_NEAREST|IF_NOGAMMA|IF_CLAMP);
}

static texid_t Shader_FindImage (parsestate_t *parsestate, char *name, int flags )
{
	extern texid_t missing_texture_normal;
	if (parsestate->parseflags & SPF_DOOM3)
	{
		if (!Q_stricmp (name, "_default"))
			return r_whiteimage; /*fixme*/
		if (!Q_stricmp (name, "_white"))
			return r_whiteimage;
		if (!Q_stricmp (name, "_black"))
		{
			int wibuf[16] = {0};
			return R_LoadTexture("$blackimage", 4, 4, TF_RGBA32, wibuf, IF_NOMIPMAP|IF_NOPICMIP|IF_NEAREST|IF_NOGAMMA);
		}
	}
	else
	{
		if (!Q_stricmp (name, "$whiteimage"))
			return r_whiteimage;
		if (!Q_stricmp (name, "$blackimage"))
		{
			int wibuf[16] = {0};
			return R_LoadTexture("$blackimage", 4, 4, TF_RGBA32, wibuf, IF_NOMIPMAP|IF_NOPICMIP|IF_NEAREST|IF_NOGAMMA);
		}
		if (!Q_stricmp (name, "$identitynormal"))
			return missing_texture_normal;
		if (!Q_stricmp (name, "$colourmap"))
			return R_LoadColourmapImage();
	}
	if (flags & IF_RENDERTARGET)
		return R2D_RT_Configure(name, 0, 0, TF_INVALID, flags);
	return R_LoadHiResTexture(name, NULL, flags);
}


/****************** shader keyword functions ************************/

static void Shader_Cull (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	char *token;

	shader->flags &= ~(SHADER_CULL_FRONT|SHADER_CULL_BACK);

	token = Shader_ParseString ( ptr );
	if ( !Q_stricmp (token, "disable") || !Q_stricmp (token, "none") || !Q_stricmp (token, "twosided") ) {
	} else if ( !Q_stricmp (token, "front") ) {
		shader->flags |= SHADER_CULL_FRONT;
	} else if ( !Q_stricmp (token, "back") || !Q_stricmp (token, "backside") || !Q_stricmp (token, "backsided") ) {
		shader->flags |= SHADER_CULL_BACK;
	} else {
		shader->flags |= SHADER_CULL_FRONT;
	}
}

static void Shader_NoMipMaps (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shader->flags |= (SHADER_NOMIPMAPS|SHADER_NOPICMIP);
}

static void Shader_Affine (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shader->flags |= SBITS_AFFINE;
}


static void Shader_NoPicMip (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shader->flags |= SHADER_NOPICMIP;
}

static void Shader_DeformVertexes (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	char *token;
	deformv_t *deformv;

	if ( shader->numdeforms >= SHADER_DEFORM_MAX )
		return;

	deformv = &shader->deforms[shader->numdeforms];

	shader->flags |= SHADER_NOMARKS;	//just in case...

	token = Shader_ParseString ( ptr );
	if ( !Q_stricmp (token, "wave") )
	{
		deformv->type = DEFORMV_WAVE;
		deformv->args[0] = Shader_ParseFloat (shader, ptr, 0);
		if (deformv->args[0])
			deformv->args[0] = 1.0f / deformv->args[0];
		Shader_ParseFunc (ps, "deformvertexes wave", ptr, &deformv->func );
	}
	else if ( !Q_stricmp (token, "normal") )
	{
		deformv->type = DEFORMV_NORMAL;
		deformv->args[0] = Shader_ParseFloat (shader, ptr, 0);
		deformv->args[1] = Shader_ParseFloat (shader, ptr, 0);
	}
	else if ( !Q_stricmp (token, "bulge") )
	{
		deformv->type = DEFORMV_BULGE;
		Shader_ParseVector (shader, ptr, deformv->args);
	}
	else if ( !Q_stricmp (token, "move") )
	{
		deformv->type = DEFORMV_MOVE;
		Shader_ParseVector (shader, ptr, deformv->args );
		Shader_ParseFunc (ps, "deformvertexes move", ptr, &deformv->func );
	}
	else if ( !Q_stricmp (token, "autosprite") )
	{
		deformv->type = DEFORMV_AUTOSPRITE;
	}
	else if ( !Q_stricmp (token, "autosprite2") )
	{
		deformv->type = DEFORMV_AUTOSPRITE2;
	}
	else if ( !Q_stricmp (token, "projectionShadow") )
		deformv->type = DEFORMV_PROJECTION_SHADOW;
	else if ( !Q_strnicmp (token, "text", 4) )
	{
		deformv->type = DEFORMV_TEXT;
		deformv->args[0] = atoi(token+4);
	}
	else
		return;

	shader->numdeforms++;
}

static void Shader_ClutterParms(parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	struct shader_clutter_s *clut;
	char *modelname;

	modelname = Shader_ParseString(ptr);
	clut = Z_Malloc(sizeof(*clut) + strlen(modelname));
	strcpy(clut->modelname, modelname);
	clut->spacing	= Shader_ParseFloat(shader, ptr, 1000);
	clut->scalemin	= Shader_ParseFloat(shader, ptr, 1);
	clut->scalemax	= Shader_ParseFloat(shader, ptr, 1);
	clut->zofs		= Shader_ParseFloat(shader, ptr, 0);
	clut->anglemin	= Shader_ParseFloat(shader, ptr, 0) * M_PI * 2 / 360.;
	clut->anglemax	= Shader_ParseFloat(shader, ptr, 360) * M_PI * 2 / 360.;

	clut->next = shader->clutter;
	shader->clutter = clut;
}

static void Shader_SkyParms(parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	skydome_t *skydome;
//	float skyheight;
	char *boxname;

	if (shader->skydome)
	{
		Z_Free(shader->skydome);
	}

	skydome = (skydome_t *)Z_Malloc(sizeof(skydome_t));
	shader->skydome = skydome;

	boxname = Shader_ParseString(ptr);
	Shader_ParseSkySides(shader->name, boxname, skydome->farbox_textures);

	/*skyheight =*/ Shader_ParseFloat(shader, ptr, 512);

	boxname = Shader_ParseString(ptr);
//	Shader_ParseSkySides(shader->name, boxname, skydome->nearbox_textures);

	shader->flags |= SHADER_SKY;
	shader->sort = SHADER_SORT_SKY;
}

static void Shader_FogParms (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	float div;
	vec3_t color, fcolor;

//	if ( !r_ignorehwgamma->value )
//		div = 1.0f / pow(2, max(0, floor(r_overbrightbits->value)));
//	else
		div = 1.0f;

	Shader_ParseVector (shader, ptr, color );
	VectorScale ( color, div, color );
	ColorNormalize ( color, fcolor );

	shader->fog_color[0] = FloatToByte ( fcolor[0] );
	shader->fog_color[1] = FloatToByte ( fcolor[1] );
	shader->fog_color[2] = FloatToByte ( fcolor[2] );
	shader->fog_color[3] = 255;
	shader->fog_dist = Shader_ParseFloat (shader, ptr, 128);

	if ( shader->fog_dist <= 0.0f ) {
		shader->fog_dist = 128.0f;
	}
	shader->fog_dist = 1.0f / shader->fog_dist;

	shader->flags |= SHADER_NODLIGHT|SHADER_NOSHADOWS;
}

static void Shader_SurfaceParm (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	char *token;

	token = Shader_ParseString ( ptr );
	if ( !Q_stricmp( token, "nodraw" ) )
		shader->flags |= SHADER_NODRAW;
	else if ( !Q_stricmp( token, "nodraw2" ) )
		shader->flags |= SHADER_NODRAW;	//an alternative so that q3map2 won't see+strip it.
	else if ( !Q_stricmp( token, "nodlight" ) )
		shader->flags |= SHADER_NODLIGHT;
	else if ( !Q_stricmp( token, "noshadows" ) )
		shader->flags |= SHADER_NOSHADOWS;

	else if ( !Q_stricmp( token, "sky" ) )
		shader->flags |= SHADER_SKY;

	else if ( !Q_stricmp( token, "noimpact" ) )
		shader->flags |= SHADER_NOMARKS;	//wrong, but whatever.
	else if ( !Q_stricmp( token, "nomarks" ) )
		shader->flags |= SHADER_NOMARKS;

	//forceshader type things inherit certain textures from the original material
	//however, that original material might not need those textures and thus won't have them loaded, which breaks replacement.
	//these provide a way to override that.
	else if (!Q_stricmp( token, "hasdiffuse"))
		shader->flags |= SHADER_HASDIFFUSE;
	else if (!Q_stricmp( token, "hasnormalmap"))
		shader->flags |= SHADER_HASNORMALMAP;
	else if (!Q_stricmp( token, "hasgloss"))
		shader->flags |= SHADER_HASGLOSS;
	else if (!Q_stricmp( token, "hasfullbright"))
		shader->flags |= SHADER_HASFULLBRIGHT;
	else if (!Q_stricmp( token, "haspaletted"))
		shader->flags |= SHADER_HASPALETTED;
	else if (!Q_stricmp(token, "hastop") || !Q_stricmp(token, "hasbottom") || !Q_stricmp(token, "hastopbottom"))
		shader->flags |= SHADER_HASTOPBOTTOM;
	else
		Con_DLPrintf(2, "Shader %s, Unknown surface parm \"%s\"\n", ps->s->name, token);	//note that there are game-specific names used to override mod surfaceflags+contents
}

static void Shader_Sort (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	char *token;

	token = Shader_ParseString ( ptr );
	if (r_forceprogramify.ival >= 2)
	{
		Con_DPrintf("Shader %s, ignoring 'sort %s'\n", ps->s->name, token);
		return;	//dp ignores 'sort' entirely.
	}
	else if ( !Q_stricmp( token, "portal" ) )
		shader->sort = SHADER_SORT_PORTAL;
	else if( !Q_stricmp( token, "sky" ) )
		shader->sort = SHADER_SORT_SKY;
	else if( !Q_stricmp( token, "opaque" ) )
		shader->sort = SHADER_SORT_OPAQUE;
	else if( !Q_stricmp( token, "decal" ) ||  !Q_stricmp( token, "litdecal" ) )
		shader->sort = SHADER_SORT_DECAL;
	else if( !Q_stricmp( token, "seethrough" ) )
		shader->sort = SHADER_SORT_SEETHROUGH;
	else if( !Q_stricmp( token, "unlitdecal" ) )
		shader->sort = SHADER_SORT_UNLITDECAL;
	else if( !Q_stricmp( token, "banner" ) )
		shader->sort = SHADER_SORT_BANNER;
	else if( !Q_stricmp( token, "additive" ) )
		shader->sort = SHADER_SORT_ADDITIVE;
	else if( !Q_stricmp( token, "underwater" ) )
		shader->sort = SHADER_SORT_UNDERWATER;
	else if( !Q_stricmp( token, "nearest" ) )
		shader->sort = SHADER_SORT_NEAREST;
	else if( !Q_stricmp( token, "blend" ) )
		shader->sort = SHADER_SORT_BLEND;
	else if ( !Q_stricmp( token, "deferredlight" ) )
		shader->sort = SHADER_SORT_DEFERREDLIGHT;
	else if ( !Q_stricmp( token, "ripple" ) )
		shader->sort = SHADER_SORT_RIPPLE;
	else
	{
		shader->sort = atoi ( token );
		clamp ( shader->sort, SHADER_SORT_NONE, SHADER_SORT_NEAREST );
	}
}

static void Shader_Deferredlight (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shader->sort = SHADER_SORT_DEFERREDLIGHT;
}

static void Shader_Portal (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shader->sort = SHADER_SORT_PORTAL;
}

static void Shader_PolygonOffset (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	float m = Shader_ParseFloat(shader, ptr, 1);

	shader->polyoffset.unit = -25 * m;
	shader->polyoffset.factor = -0.05;
	shader->flags |= SHADER_POLYGONOFFSET;	//some backends might be lazy and only allow simple values.
}

static void Shader_EntityMergable (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shader->flags |= SHADER_ENTITY_MERGABLE;
}

#if defined(GLQUAKE) || defined(D3DQUAKE)
static qboolean Shader_ParseProgramCvar(char *script, cvar_t **cvarrefs, char **cvarnames, int *cvartypes, int cvartype)
{
	char body[MAX_QPATH];
	char *out;
	char *namestart;
	while (*script == ' ' || *script == '\t')
		script++;
	namestart = script;
	while ((*script >= 'A' && *script <= 'Z') || (*script >= 'a' && *script <= 'z') || (*script >= '0' && *script <= '9') || *script == '_')
		script++;

	cvartypes[0] = cvartype;
	cvarnames[0] = Z_Malloc(script - namestart + 1);
	memcpy(cvarnames[0], namestart, script - namestart);
	cvarnames[0][script - namestart] = 0;

	while (*script == ' ' || *script == '\t')
		script++;
	if (*script == '=')
	{
		script++;
		while (*script == ' ' || *script == '\t')
			script++;

		out = body;
		while (out < com_token+countof(body)-1 && *script != '\n' && !(script[0] == '/' && script[1] == '/')) 
			*out++ = *script++;
		*out++ = 0;
		cvarrefs[0] = Cvar_Get(cvarnames[0], body, 0, "GLSL Variables");
	}
	else
		cvarrefs[0] = Cvar_Get(cvarnames[0], "", 0, "GLSL Variables");
	return true;
}
#endif

const struct sh_defaultsamplers_s sh_defaultsamplers[] =
{
	{"s_shadowmap",		1u<<0},
	{"s_projectionmap",	1u<<1},
	{"s_diffuse",		1u<<2},
	{"s_normalmap",		1u<<3},
	{"s_specular",		1u<<4},
	{"s_upper",			1u<<5},
	{"s_lower",			1u<<6},
	{"s_fullbright",	1u<<7},
	{"s_paletted",		1u<<8},
	{"s_reflectcube",	1u<<9},
	{"s_reflectmask",	1u<<10},
	{"s_displacement",	1u<<11},
	{"s_occlusion",		1u<<12},
	{"s_lightmap",		1u<<13},
	{"s_deluxemap",		1u<<14},
#if MAXRLIGHTMAPS > 1
	{"s_lightmap1",		1u<<15},
	{"s_lightmap2",		1u<<16},
	{"s_lightmap3",		1u<<17},
	{"s_deluxemap1",	1u<<18},
	{"s_deluxemap2",	1u<<19},
	{"s_deluxemap3",	1u<<20},
#else
	{"s_lightmap1",		0},
	{"s_lightmap2",		0},
	{"s_lightmap3",		0},
	{"s_deluxemap1",	0},
	{"s_deluxemap2",	0},
	{"s_deluxemap3",	0},
#endif
	{NULL}
};

static struct
{
	char *name;
	unsigned int bitmask;
} permutations[] =
{
	{"BUMP", PERMUTATION_BUMPMAP},
	{"FULLBRIGHT", PERMUTATION_FULLBRIGHT},
	{"UPPERLOWER", PERMUTATION_UPPERLOWER},
	{"REFLECTCUBEMASK", PERMUTATION_REFLECTCUBEMASK},
	{"SKELETAL", PERMUTATION_SKELETAL},
	{"FOG", PERMUTATION_FOG},
	{"FRAMEBLEND", PERMUTATION_FRAMEBLEND},
	{"LIGHTSTYLED", PERMUTATION_LIGHTSTYLES}
};
#define MAXMODIFIERS 64

void VARGS Q_strlcatfz (char *dest, size_t *offset, size_t size, const char *fmt, ...) LIKEPRINTF(4);
void VARGS Q_strlcatfz (char *dest, size_t *offset, size_t size, const char *fmt, ...)
{
	va_list		argptr;

	dest += *offset;
	size -= *offset;

	va_start (argptr, fmt);
	Q_vsnprintfz(dest, size, fmt, argptr);
	va_end (argptr);
	*offset += strlen(dest);
}
struct programpermu_s *Shader_LoadPermutation(program_t *prog, unsigned int p)
{
	const char *permutationdefines[3];
	struct programpermu_s *pp;
	size_t n, pn = 0;
	char defines[8192];
	size_t offset;
	qboolean fail = false;

	extern cvar_t gl_specular, gl_specular_power;

	if (~prog->supportedpermutations & p)
		return NULL;	//o.O
	pp = Z_Malloc(sizeof(*pp));
	pp->permutation = p;
	*defines = 0;
	offset = 0;
	if (p & PERMUTATION_SKELETAL)
		Q_strlcatfz(defines, &offset, sizeof(defines), "#define MAX_GPU_BONES %i\n", sh_config.max_gpu_bones);
	if (gl_specular.value)
		Q_strlcatfz(defines, &offset, sizeof(defines), "#define SPECULAR\n#define SPECULAR_BASE_MUL %f\n#define SPECULAR_BASE_POW %f\n", 1.0*gl_specular.value, max(1,gl_specular_power.value));
#ifdef RTLIGHTS
	if (r_fakeshadows)
		Q_strlcatfz(defines, &offset, sizeof(defines), "#define FAKESHADOWS\n%s",
#ifdef GLQUAKE
				gl_config.arb_shadow?"#define USE_ARB_SHADOW\n":
#endif
				"");
#endif

	for (n = 0; n < countof(permutations); n++)
	{
		if (p & permutations[n].bitmask)
			Q_strlcatfz(defines, &offset, sizeof(defines), "#define %s\n", permutations[n].name);
	}
	if (p & PERMUTATION_UPPERLOWER)
		Q_strlcatfz(defines, &offset, sizeof(defines), "#define UPPER\n#define LOWER\n");
	if (p & PERMUTATION_BUMPMAP)
	{
		if (r_glsl_offsetmapping.ival)
		{
			Q_strlcatfz(defines, &offset, sizeof(defines), "#define OFFSETMAPPING\n");
			if (r_glsl_offsetmapping_reliefmapping.ival && (p & PERMUTATION_BUMPMAP))
				Q_strlcatfz(defines, &offset, sizeof(defines), "#define RELIEFMAPPING\n");
		}

		if (r_deluxemapping)	//fixme: should be per-model really
			Q_strlcatfz(defines, &offset, sizeof(defines), "#define DELUXE\n");
	}
	permutationdefines[pn++] = defines;
	permutationdefines[pn++] = prog->preshade;
	permutationdefines[pn++] = NULL;

	if (!sh_config.pCreateProgram(prog, pp, prog->shaderver, permutationdefines, prog->shadertext, prog->tess?prog->shadertext:NULL, prog->tess?prog->shadertext:NULL, prog->geom?prog->shadertext:NULL, prog->shadertext, prog->warned, NULL))
		prog->warned = fail = true;

	//extra loop to validate the programs actually linked properly.
	//delaying it like this gives certain threaded drivers a chance to compile them all while we're messing around with other junk
	if (!fail && sh_config.pValidateProgram && !sh_config.pValidateProgram(prog, pp, prog->warned, NULL))
		prog->warned = fail = true;

	if (!fail && sh_config.pProgAutoFields)
	{
		cvar_t *cvarrefs[64];
		char *cvarnames[64+1];
		int cvartypes[64];

		unsigned char *cvardata = prog->cvardata;
		size_t size = prog->cvardatasize, i;
		for (i = 0; i < countof(cvartypes) && size; i++)
		{
			memcpy(&cvartypes[i], cvardata, sizeof(int));
			cvarnames[i] = cvardata+sizeof(int);
			size -= sizeof(int)+strlen(cvarnames[i])+1;
			cvardata += sizeof(int)+strlen(cvarnames[i])+1;
			cvarrefs[i] = Cvar_FindVar(cvarnames[i]);
		}
		cvarnames[i] = NULL; //no more
		sh_config.pProgAutoFields(prog, pp, cvarrefs, cvarnames, cvartypes);
	}
	if (fail)
	{
		Z_Free(pp);
		return NULL;
	}
	return pp;
}

qboolean Shader_PermutationEnabled(unsigned int bit)
{
	if (bit == PERMUTATION_REFLECTCUBEMASK)
		return gl_load24bit.ival;
	if (bit == PERMUTATION_BUMPMAP)
		return r_loadbumpmapping;
	return true;
}
qboolean Com_PermuOrFloatArgument(const char *shadername, char *arg, size_t arglen, float def)
{
	extern cvar_t gl_specular;
	size_t p;
	//load-time-only permutations...
	if (arglen == 8 && !strncmp("SPECULAR", arg, arglen) && gl_specular.value)
		return true;
#ifdef RTLIGHTS
	if (arglen == 11 && !strncmp("FAKESHADOWS", arg, arglen) && r_fakeshadows)
		return true;
#endif
	if ((arglen==5||arglen==6) && !strncmp("DELUXE", arg, arglen) && r_deluxemapping && Shader_PermutationEnabled(PERMUTATION_BUMPMAP))
		return true;
	if (arglen == 13 && !strncmp("OFFSETMAPPING", arg, arglen) && r_glsl_offsetmapping.ival)
		return true;
	if (arglen == 13 && !strncmp("RELIEFMAPPING", arg, arglen) && r_glsl_offsetmapping.ival && r_glsl_offsetmapping_reliefmapping.ival)
		return true;

	//real permutations
	if (arglen == 5 && (!strncmp("UPPER", arg, arglen)||!strncmp("LOWER", arg, arglen)) && Shader_PermutationEnabled(PERMUTATION_BIT_UPPERLOWER))
		return true;
	for (p = 0; p < countof(permutations); p++)
	{
		if (arglen == strlen(permutations[p].name) && !strncmp(permutations[p].name, arg, arglen))
		{
			if (Shader_PermutationEnabled(permutations[p].bitmask))
				return true;
			break;
		}
	}

	return Com_FloatArgument(shadername, arg, arglen, def) != 0;
}

static qboolean Shader_LoadPermutations(char *name, program_t *prog, char *script, int qrtype, int ver, char *blobfilename)
{
#if defined(GLQUAKE) || defined(D3DQUAKE)
//	const char *permutationdefines[countof(permutations) + MAXMODIFIERS + 1];
	unsigned int nopermutation = PERMUTATIONS-1;
//	int nummodifiers = 0;
	int p;
	char *end;

	cvar_t *cvarrefs[64];
	char *cvarnames[64];
	int cvartypes[64];
	size_t cvarcount = 0, i;
	extern cvar_t gl_specular, gl_specular_power;
	qboolean cantess = false;	//not forced.
	char prescript[8192];
	size_t offset = 0;
#endif

#ifdef VKQUAKE
	if (qrenderer == QR_VULKAN && (qrtype == QR_VULKAN || qrtype == QR_OPENGL))
	{	//vulkan can potentially load glsl, f it has the extensions enabled.
		if (qrtype == QR_VULKAN && VK_LoadBlob(prog, script, name))
			return true;
	}
	else
#endif
	if (qrenderer != qrtype)
	{
		return false;
	}

#if defined(GLQUAKE) || defined(D3DQUAKE)
	ver = 0;

	if (!sh_config.pCreateProgram && !sh_config.pLoadBlob)
		return false;

	if (prog->name)
		return false;	//o.O

	*prescript = 0;
	offset = 0;
	memset(prog->permu, 0, sizeof(prog->permu));
	prog->name = Z_StrDup(name);
	prog->geom = false;
	prog->tess = false;
	prog->nofixedcompat = true;
	prog->numsamplers = 0;
	prog->defaulttextures = 0;
	for(;;)
	{
		while (*script == ' ' || *script == '\r' || *script == '\n' || *script == '\t')
			script++;
		if (!strncmp(script, "!!fixed", 7))
		{
			prog->nofixedcompat = false;
			script += 7;
		}
		else if (!strncmp(script, "!!geom", 6))
		{
			prog->geom = true;
			script += 6;
		}
		else if (!strncmp(script, "!!tess", 6))
		{
			prog->tess = true;
			script += 6;
		}
		else if (!strncmp(script, "!!samps", 7))
		{
			com_tokentype_t tt;
			qboolean ignore = false;
			script += 7;
			for(;;)
			{
				size_t len;
				int i;
				char *type, *idx, *next;
				char *token = com_token;

				next = COM_ParseTokenOut(script, "", com_token, sizeof(com_token), &tt);
				if (tt == TTP_LINEENDING || tt == TTP_EOF)
					break;
				script = next;

				if (*token == '=' || *token == '!')
				{
					len = strlen(token);
					if (*token == (Com_PermuOrFloatArgument(name, token+1, len-1, 0)?'!':'='))
						ignore = true;
					continue;
				}
				else if (ignore)
					continue;
#if 1//def HAVE_LEGACY
				else if (!strncmp(token, "deluxmap", 8))
				{	//FIXME: remove this some time.
					Con_DPrintf("Outdated texture name \"%s\" in program \"%s\"\n", token, name);
					token = va("deluxemap%s",token+8);
				}
#endif
				type = strchr(token, ':');
				idx = strchr(token, '=');
				if (type || idx)
				{	//name:type=idx
					if (type)
						*type++ = 0;
					else
						type = "2D";
					if (idx)
					{
						*idx++ = 0;
						i = atoi(idx);
					}
					else
						i = prog->numsamplers;
					if (prog->numsamplers < i+1)
						prog->numsamplers = i+1;

					/*for (j = 0; sh_defaultsamplers[j].name; j++)
					{
						if (!strcmp(token, sh_defaultsamplers[j].name+2))
						{
							Con_Printf("%s: %s is an internal texture name\n", name, token);
							break;
						}
					}*/

					//I really want to use layout(binding = %i) here, but its specific to the glsl version (which we don't really know yet)
					Q_strlcatfz(prescript, &offset, sizeof(prescript), "#define s_%s s_t%u\nuniform %s%s s_%s;\n", token, i, strncmp(type, "sampler", 7)?"sampler":"", type, token);
				}
				else
				{
					len = strlen(token);
					for (i = 0; sh_defaultsamplers[i].name; i++)
					{
						if (!strcmp(token, sh_defaultsamplers[i].name+2))
						{
							prog->defaulttextures |= sh_defaultsamplers[i].defaulttexbits;
							break;
						}
					}
					if (!sh_defaultsamplers[i].name)
					{	//this path is deprecated.
						i = atoi(token);
						if (i)
						{
							if (qrenderer == QR_OPENGL)
							{
								while (prog->numsamplers < i)
									Q_strlcatfz(prescript, &offset, sizeof(prescript), "uniform sampler2D s_t%u;\n", prog->numsamplers++);
							}
							else if (prog->numsamplers < i)
								prog->numsamplers = i;
						}
						else
							Con_Printf("Unknown texture name \"%s\" in program \"%s\"\n", token, name);
					}
				}
			}
		}
		else if (!strncmp(script, "!!cvardf", 8) || !strncmp(script, "!!cvard3", 8) || !strncmp(script, "!!cvard4", 8) || !strncmp(script, "!!cvard_srgb", 11))
		{
			qboolean srgb = false;
			float div = 1;
			char type = script[7];
			script+=8;
			if (type == '_')
			{
				if (*script == 's')
				{
					srgb = true;
					script+=1;
				}

				if (!strncmp(script, "rgba", 4))
				{
					type = '4';
					script+=4;
				}
				else if (!strncmp(script, "rgb", 3))
				{
					type = '3';
					script+=3;
				}
				else if (!strncmp(script, "rg", 2))
				{
					type = '2';
					script+=2;
				}
				else if (!strncmp(script, "r", 1) || !strncmp(script, "f", 1))
				{
					type = 'f';
					script+=1;
				}

				if (!strncmp(script, "_b", 2))
				{
					div = 255;
					script+=2;
				}
				else if (!strncmp(script, "_", 1))
					div = strtod(script, &script);
			}
			while (*script == ' ' || *script == '\t')
				script++;
			end = script;
			while ((*end >= 'A' && *end <= 'Z') || (*end >= 'a' && *end <= 'z') || (*end >= '0' && *end <= '9') || *end == '_')
				end++;
			if (end - script < 64)
			{
				cvar_t *var;
				char namebuf[64];
				char valuebuf[64];
				memcpy(namebuf, script, end - script);
				namebuf[end - script] = 0;
				while (*end == ' ' || *end == '\t')
					end++;
				if (*end == '=')
				{
					script = ++end;
					while (*end && *end != '\n' && *end != '\r' && end < script+sizeof(namebuf)-1)
						end++;
					memcpy(valuebuf, script, end - script);
					valuebuf[end - script] = 0;
				}
				else
					strcpy(valuebuf, "0");
				var = Cvar_Get(namebuf, valuebuf, CVAR_SHADERSYSTEM, "GLSL Variables");
				if (var)
				{
					if (srgb)
					{
						if (type == '4')
							Q_strlcatfz(prescript, &offset, sizeof(prescript), "#define %s %s(%g,%g,%g,%g)\n", namebuf, ((qrenderer == QR_OPENGL)?"vec4":"float4"), SRGBf(var->vec4[0]/div), SRGBf(var->vec4[1]/div), SRGBf(var->vec4[2]/div), var->vec4[3]/div);
						else if (type == '3')
							Q_strlcatfz(prescript, &offset, sizeof(prescript), "#define %s %s(%g,%g,%g)\n", namebuf, ((qrenderer == QR_OPENGL)?"vec3":"float3"), SRGBf(var->vec4[0]/div), SRGBf(var->vec4[1]/div), SRGBf(var->vec4[2]/div));
						else
							Q_strlcatfz(prescript, &offset, sizeof(prescript), "#define %s %g\n", namebuf, SRGBf(var->value/div));
					}
					else
					{
						if (type == '4')
							Q_strlcatfz(prescript, &offset, sizeof(prescript), "#define %s %s(%g,%g,%g,%g)\n", namebuf, ((qrenderer == QR_OPENGL)?"vec4":"float4"), var->vec4[0]/div, var->vec4[1]/div, var->vec4[2]/div, var->vec4[3]/div);
						else if (type == '3')
							Q_strlcatfz(prescript, &offset, sizeof(prescript), "#define %s %s(%g,%g,%g)\n", namebuf, ((qrenderer == QR_OPENGL)?"vec3":"float3"), var->vec4[0]/div, var->vec4[1]/div, var->vec4[2]/div);
						else
							Q_strlcatfz(prescript, &offset, sizeof(prescript), "#define %s %g\n", namebuf, var->value/div);
					}
				}
			}
			script = end;
		}
		else if (!strncmp(script, "!!cvarf", 7))
		{
			if (cvarcount != sizeof(cvarnames)/sizeof(cvarnames[0]))
				cvarcount += Shader_ParseProgramCvar(script+7, &cvarrefs[cvarcount], &cvarnames[cvarcount], &cvartypes[cvarcount], SP_CVARF);
		}
		else if (!strncmp(script, "!!cvari", 7))
		{
			if (cvarcount != sizeof(cvarnames)/sizeof(cvarnames[0]))
				cvarcount += Shader_ParseProgramCvar(script+7, &cvarrefs[cvarcount], &cvarnames[cvarcount], &cvartypes[cvarcount], SP_CVARI);
		}
		else if (!strncmp(script, "!!cvarv", 7))
		{
			if (cvarcount != sizeof(cvarnames)/sizeof(cvarnames[0]))
				cvarcount += Shader_ParseProgramCvar(script+7, &cvarrefs[cvarcount], &cvarnames[cvarcount], &cvartypes[cvarcount], SP_CVAR3F);
		}
		else if (!strncmp(script, "!!arg", 5))
		{	//compat with our vulkan glsl, generate (specialisation) constants from #args
			char namebuf[MAX_QPATH];
			char valuebuf[MAX_QPATH];
			char *out;
			char *namestart;
			char *atype;
			script+=5;
			if (*script == 'b')
			{
				atype = "bool";
				strcpy(valuebuf, "false");
			}
			else if (*script == 'f')
			{
				atype = "float";
				strcpy(valuebuf, "0.0");
			}
			else if (*script == 'd')
			{
				atype = "double";
				strcpy(valuebuf, "0.0");
			}
			else if (*script == 'i')
			{
				atype = "int";
				strcpy(valuebuf, "0");
			}
			else if (*script == 'u')
			{
				atype = "uint";
				strcpy(valuebuf, "0");
			}
			else
			{
				atype = "float";	//I guess
				strcpy(valuebuf, "0.0");
			}
			while (*script >= 'a' && *script <= 'z')
				script++;
			while (*script == ' ' || *script == '\t')
				script++;
			namestart = script;
			while ((*script >= 'A' && *script <= 'Z') || (*script >= 'a' && *script <= 'z') || (*script >= '0' && *script <= '9') || *script == '_')
				script++;

			if (script-namestart < countof(namebuf))
			{
				float def = 0;
				memcpy(namebuf, namestart, script - namestart);
				namebuf[script - namestart] = 0;

				while (*script == ' ' || *script == '\t')
					script++;
				if (*script == '=')
				{
					script++;
					while (*script == ' ' || *script == '\t')
						script++;

					out = valuebuf;
					while (out < com_token+countof(valuebuf)-1 && *script != '\n' && !(script[0] == '/' && script[1] == '/'))
						*out++ = *script++;
					*out++ = 0;
					if (!strcmp(valuebuf, "true"))
						def = 1;
					else
						def = atof(valuebuf);
				}
				Com_FloatArgument(name, valuebuf, sizeof(valuebuf), def);
				Q_strlcatfz(prescript, &offset, sizeof(prescript), "const %s arg_%s = %s(%s);\n", atype, namebuf, atype, valuebuf);
			}
		}
		else if (!strncmp(script, "!!permu", 7))
		{
			script += 7;
			while (*script == ' ' || *script == '\t')
				script++;
			end = script;
			while ((*end >= 'A' && *end <= 'Z') || (*end >= 'a' && *end <= 'z') || (*end >= '0' && *end <= '9') || *end == '_')
				end++;
			for (p = 0; p < countof(permutations); p++)
			{
				if (!strncmp(permutations[p].name, script, end - script) && permutations[p].name[end-script] == '\0')
				{
					if (Shader_PermutationEnabled(permutations[p].bitmask))
						nopermutation &= ~permutations[p].bitmask;
					break;
				}
			}
			if (p == countof(permutations))
			{
				//we 'recognise' ones that are force-defined, despite not being actual permutations.
				if (end - script == 4 && !strncmp("TESS", script, 4))
					cantess = true;
				else if (strncmp("SPECULAR", script, end - script))
				if (strncmp("DELUXE", script, end - script))
				if (strncmp("DELUX", script, end - script))
				if (strncmp("OFFSETMAPPING", script, end - script))
				if (strncmp("RELIEFMAPPING", script, end - script))
				if (strncmp("FAKESHADOWS", script, end - script))
					Con_DPrintf("Unknown pemutation in glsl program %s\n", name);
			}
			script = end;
		}
		else if (!strncmp(script, "!!ver", 5))
		{
			int minver, maxver;
			script += 5;
			while (*script == ' ' || *script == '\t')
				script++;
			end = script;
			while ((*end >= 'A' && *end <= 'Z') || (*end >= 'a' && *end <= 'z') || (*end >= '0' && *end <= '9') || *end == '_')
				end++;
			minver = strtol(script, &script, 0);
			while (*script == ' ' || *script == '\t')
				script++;
			maxver = strtol(script, NULL, 0);
			if (!maxver)
				maxver = minver;

			ver = maxver;
			if (ver > sh_config.maxver)
				ver = sh_config.maxver;
			if (ver < minver)
				ver = minver;	//some kind of error.

			script = end;
		}
		else if (!strncmp(script, "!!", 2))
		{
			Con_DPrintf("Unknown directinve in glsl program %s\n", name);
			script += 2;
			while (*script == ' ' || *script == '\t')
				script++;
		}
		else if (!strncmp(script, "//", 2))
		{
			script += 2;
			while (*script == ' ' || *script == '\t')
				script++;
		}
		else
			break;
		while (*script && *script != '\n')
			script++;
	};
	prog->shadertext = Z_StrDup(script);

	if (qrenderer == qrtype && ver < 150)
		prog->tess = cantess = false;	//GL_ARB_tessellation_shader requires glsl 150(gl3.2) (or glessl 3.1). nvidia complains about layouts if you try anyway

	if (!r_fog_permutation.ival)
		nopermutation |= PERMUTATION_BIT_FOG;
	if (!sh_config.max_gpu_bones)
		nopermutation |= PERMUTATION_SKELETAL;

	//multiple lightmaps is kinda hacky. if any are set, all must be.
#define ALTLIGHTMAPSAMP 14
	if (prog->defaulttextures & ((1u<<(ALTLIGHTMAPSAMP+0)) | (1u<<(ALTLIGHTMAPSAMP+1)) | (1u<<(ALTLIGHTMAPSAMP+2))))
		prog->defaulttextures |=((1u<<(ALTLIGHTMAPSAMP+0)) | (1u<<(ALTLIGHTMAPSAMP+1)) | (1u<<(ALTLIGHTMAPSAMP+2)));
#define ALTDELUXMAPSAMP 17
	if (prog->defaulttextures & ((1u<<(ALTDELUXMAPSAMP+0)) | (1u<<(ALTDELUXMAPSAMP+1)) | (1u<<(ALTDELUXMAPSAMP+2))))
		prog->defaulttextures |=((1u<<(ALTDELUXMAPSAMP+0)) | (1u<<(ALTDELUXMAPSAMP+1)) | (1u<<(ALTDELUXMAPSAMP+2)));

	for (end = strchr(name, '#'); end && *end; )
	{
		char *start = end+1;
		end = strchr(start, '#');
		if (!end)
			end = start + strlen(start);
		if (end-start == 7 && !Q_strncasecmp(start, "usemods", 7))
			prog->nofixedcompat = false;

		if (end-start == 4 && !Q_strncasecmp(start, "tess", 4))
			prog->tess |= cantess;

		Q_strlcatfz(prescript, &offset, sizeof(prescript), "#define ");
		while (offset < sizeof(prescript) && start < end)
		{
			if (*start == '=')
			{
				start++;
				prescript[offset++] = ' ';
				break;
			}
			prescript[offset++] = toupper(*start++);
		}
		while (offset < sizeof(prescript) && start < end)
			prescript[offset++] = toupper(*start++);
		Q_strlcatfz(prescript, &offset, sizeof(prescript), "\n");
	}

	prog->preshade = Z_StrDup(prescript);
	prog->supportedpermutations = (~nopermutation) & (PERMUTATIONS-1);
	prog->shaderver = ver;

	if (cvarcount)
	{
		*prescript = 0;
		offset = 0;
		for (i = 0; i < cvarcount && offset < sizeof(prescript); i++)
		{
			if (cvarrefs[i])
			{
				memcpy(prescript+offset, &cvartypes[i], sizeof(int));
				offset+=4;
				Q_strlcatfz(prescript, &offset, sizeof(prescript), "%s", cvarnames[i]);
				offset++;
			}
		}
		prog->cvardata = Z_Malloc(offset);
		prog->cvardatasize = offset;
		memcpy(prog->cvardata, prescript, prog->cvardatasize);
	}

	while(cvarcount)
		Z_Free((char*)cvarnames[--cvarcount]);

	//ensure that permutation 0 works correctly as a fallback.
	//FIXME: add debug mode to compile all.
	prog->permu[0] = Shader_LoadPermutation(prog, 0);
	return !!prog->permu[0];
#else
	return false;
#endif
}

typedef struct sgeneric_s
{
	program_t prog;
	struct sgeneric_s *next;
	char *name;
	qboolean failed;
} sgeneric_t;
static sgeneric_t *sgenerics;
struct sbuiltin_s
{
	int qrtype;
	int apiver;
	char name[MAX_QPATH];
	char *body;
} sbuiltins[] =
{
#include "r_bishaders.h"
	{QR_NONE}
};
void Shader_UnloadProg(program_t *prog)
{
	if (sh_config.pDeleteProg)
		sh_config.pDeleteProg(prog);

	Z_Free(prog->name);
	Z_Free(prog->preshade);
	Z_Free(prog->shadertext);
	Z_Free(prog->cvardata);

	Z_Free(prog);
}
static void Shader_FlushGenerics(void)
{
	sgeneric_t *g;
	while (sgenerics)
	{
		g = sgenerics;
		sgenerics = g->next;

		if (g->prog.refs == 1)
		{
			g->prog.refs--;
			Shader_UnloadProg(&g->prog);
		}
		else
			Con_Printf("generic shader still used\n"); 
	}
}
static void Shader_LoadGeneric(sgeneric_t *g, int qrtype)
{
	unsigned int i;
	void *file;
	char basicname[MAX_QPATH];
	char blobname[MAX_QPATH];
	char *h;

	g->failed = true;

	TRACE(("Loading program %s...\n", g->name));

	basicname[1] = 0;
	Q_strncpyz(basicname, g->name, sizeof(basicname));
	h = strchr(basicname+1, '#');
	if (h)
		*h = '\0';

	if (strchr(basicname, '/') || strchr(basicname, '.'))
	{	//explicit path
		FS_LoadFile(basicname, &file);
		*blobname = 0;
	}
	else if (ruleset_allow_shaders.ival)
	{	//renderer-specific files
		if (sh_config.progpath)
		{
			Q_snprintfz(blobname, sizeof(blobname), sh_config.progpath, basicname);
			FS_LoadFile(blobname, &file);
		}
		else
			file = NULL;
		if (sh_config.blobpath && r_shaderblobs.ival)
			Q_snprintfz(blobname, sizeof(blobname), sh_config.blobpath, basicname);
		else
			*blobname = 0;
	}
	else
	{
		file = NULL;
		*blobname = 0;
	}

	if (sh_config.pDeleteProg)
	{
		sh_config.pDeleteProg(&g->prog);
	}
	Z_Free(g->prog.name);
	g->prog.name = NULL;
	Z_Free(g->prog.preshade);
	g->prog.preshade = NULL;
	Z_Free(g->prog.shadertext);
	g->prog.shadertext = NULL;
	Z_Free(g->prog.cvardata);
	g->prog.cvardata = NULL;

	if (file)
	{
		TRACE(("Loading from disk (%s)\n", g->name));
//		Con_DPrintf("Loaded %s from disk\n", sh_config.progpath?va(sh_config.progpath, basicname):basicname);
		g->failed = !Shader_LoadPermutations(g->name, &g->prog, file, qrtype, 0, blobname);
		FS_FreeFile(file);
		return;
	}
	else
	{
		int ver;
		for (i = 0; *sbuiltins[i].name; i++)
		{
			if (sbuiltins[i].qrtype == qrtype && !strcmp(sbuiltins[i].name, basicname))
			{
				ver = sbuiltins[i].apiver;

				if (ver < sh_config.minver || ver > sh_config.maxver)
					if (!(qrenderer==QR_OPENGL&&ver==110))
						continue;

				TRACE(("Loading Embedded %s\n", g->name));
				g->failed = !Shader_LoadPermutations(g->name, &g->prog, sbuiltins[i].body, qrtype, ver, blobname);

				if (g->failed)
					continue;

				return;
			}
		}
		TRACE(("Program unloadable %s\n", g->name));
	}
}

program_t *Shader_FindGeneric(char *name, int qrtype)
{
	sgeneric_t *g;

	for (g = sgenerics; g; g = g->next)
	{
		if (!strcmp(name, g->name))
		{
			if (g->failed)
				return NULL;
			g->prog.refs++;
			return &g->prog;
		}
	}

	//don't even try if we know it won't work.
	if (!sh_config.progs_supported)
		return NULL;

	g = BZ_Malloc(sizeof(*g) + strlen(name)+1);
	memset(g, 0, sizeof(*g));
	g->name = (char*)(g+1);
	strcpy(g->name, name);
	g->next = sgenerics;
	sgenerics = g;

	g->prog.refs = 1;

	Shader_LoadGeneric(g, qrtype);
	if (g->failed)
		return NULL;
	g->prog.refs++;
	return &g->prog;
}
static void Shader_ReloadGenerics(void)
{
	sgeneric_t *g;
	for (g = sgenerics; g; g = g->next)
	{
		//this happens if some cvar changed that affects the glsl itself. supposedly.
		Shader_LoadGeneric(g, qrenderer);
	}

	//this shader can take a while to load due to its number of permutations.
	//because this all happens on the main thread, try to avoid random stalls by pre-loading it.
	if (sh_config.progs_supported)
	{
		program_t *p = Shader_FindGeneric("defaultskin", qrenderer);
		if (p)	//generics get held on to in order to avoid so much churn. so we can just release the reference we just created and it'll be held until shutdown anyway.
			p->refs--;
	}
}
void Shader_WriteOutGenerics_f(void)
{
	int i;
	char *name;
	for (i = 0; *sbuiltins[i].name; i++)
	{
		name = NULL;
		if (sbuiltins[i].qrtype == QR_OPENGL)
		{
			if (sbuiltins[i].apiver == 100)
				name = va("gles/eg_%s.glsl", sbuiltins[i].name);
			else
				name = va("glsl/eg_%s.glsl", sbuiltins[i].name);
		}
		else if (sbuiltins[i].qrtype == QR_DIRECT3D9)
			name = va("hlsl/eg_%s.hlsl", sbuiltins[i].name);
		else if (sbuiltins[i].qrtype == QR_DIRECT3D11)
			name = va("hlsl11/eg_%s.hlsl", sbuiltins[i].name);

		if (name)
		{
			vfsfile_t *f = FS_OpenVFS(name, "rb", FS_GAMEONLY);
			if (f)
			{
				int len = VFS_GETLEN(f);
				char *buf = Hunk_TempAlloc(len);
				VFS_READ(f, buf, len);
				if (len != strlen(sbuiltins[i].body) || memcmp(buf, sbuiltins[i].body, len))
					Con_Printf("Not writing %s - modified version in the way\n", name);
				else
					Con_Printf("%s is unmodified\n", name);
				VFS_CLOSE(f);
			}
			else
			{
				Con_Printf("Writing %s\n", name);
				FS_WriteFile(name, sbuiltins[i].body, strlen(sbuiltins[i].body), FS_GAMEONLY);
			}
		}
	}
}

struct shader_field_names_s shader_attr_names[] =
{
	/*vertex attributes*/
	{"v_position1",				VATTR_VERTEX1},
	{"v_position2",				VATTR_VERTEX2},
	{"v_colour",				VATTR_COLOUR},
	{"v_texcoord",				VATTR_TEXCOORD},
	{"v_lmcoord",				VATTR_LMCOORD},
	{"v_normal",				VATTR_NORMALS},
	{"v_svector",				VATTR_SNORMALS},
	{"v_tvector",				VATTR_TNORMALS},
	{"v_bone",					VATTR_BONENUMS},
	{"v_weight",				VATTR_BONEWEIGHTS},
#if MAXRLIGHTMAPS > 1
	{"v_lmcoord1",				VATTR_LMCOORD},
	{"v_lmcoord2",				VATTR_LMCOORD2},
	{"v_lmcoord3",				VATTR_LMCOORD3},
	{"v_lmcoord4",				VATTR_LMCOORD4},
	{"v_colour1",				VATTR_COLOUR},
	{"v_colour2",				VATTR_COLOUR2},
	{"v_colour3",				VATTR_COLOUR3},
	{"v_colour4",				VATTR_COLOUR4},
#endif
	{NULL}
};

struct shader_field_names_s shader_unif_names[] =
{
	/**///tagged names are available to vulkan

	/*matricies*/
/**/{"m_model",					SP_M_MODEL},	//the model matrix
	{"m_view",					SP_M_VIEW},		//the view matrix
	{"m_modelview",				SP_M_MODELVIEW},//the combined modelview matrix
	{"m_projection",			SP_M_PROJECTION},//projection matrix
/**/{"m_modelviewprojection",	SP_M_MODELVIEWPROJECTION},//fancy mvp matrix. probably has degraded precision.
	{"m_bones_packed",			SP_M_ENTBONES_PACKED},	//bone matrix array. should normally be read via sys/skeletal.h
	{"m_bones_mat3x4",			SP_M_ENTBONES_MAT3X4},	//bone matrix array. should normally be read via sys/skeletal.h
	{"m_invviewprojection",		SP_M_INVVIEWPROJECTION},//inverted vp matrix
	{"m_invmodelviewprojection",SP_M_INVMODELVIEWPROJECTION},//inverted mvp matrix.
/**///m_modelinv

	/*viewer properties*/
	{"v_eyepos",				SP_V_EYEPOS},	//eye pos in worldspace
/**/{"w_fog",					SP_W_FOG},		//read by sys/fog.h
	{"w_user",					SP_W_USER},		//set via VF_USERDATA

	/*ent properties*/
	{"e_vblend",				SP_E_VBLEND},	//v_position1+v_position2 scalers
/**/{"e_lmscale",				SP_E_LMSCALE},	/*overbright shifting*/
	{"e_vlscale",				SP_E_VLSCALE},	/*no lightmaps, no overbrights*/
	{"e_origin",				SP_E_ORIGIN},		//the ents origin in worldspace
/**/{"e_time",					SP_E_TIME},			//r_refdef.time - entity->time
/**/{"e_eyepos",				SP_E_EYEPOS},		//eye pos in modelspace
	{"e_colour",				SP_E_COLOURS},		//colormod/alpha, even if colormod isn't set
/**/{"e_colourident",			SP_E_COLOURSIDENT},	//colormod,alpha or 1,1,1,alpha if colormod isn't set
/**/{"e_glowmod",				SP_E_GLOWMOD},		//fullbright scalers (for hdr mostly)
/**/{"e_uppercolour",			SP_E_TOPCOLOURS},	//q1 player colours
/**/{"e_lowercolour",			SP_E_BOTTOMCOLOURS},//q1 player colours
/**/{"e_light_dir",				SP_E_L_DIR},		//lightgrid light dir. dotproducts should be clamped to 0-1.
/**/{"e_light_mul",				SP_E_L_MUL},		//lightgrid light scaler.
/**/{"e_light_ambient",			SP_E_L_AMBIENT},	//lightgrid light value for the unlit side.

	{"s_colour",				SP_S_COLOUR},	//the rgbgen/alphagen stuff. obviously doesn't work with per-vertex ones.

	/*rtlight properties, use with caution*/
	{"l_lightscreen",			SP_LIGHTSCREEN},	//screenspace position of the current rtlight
/**/{"l_lightradius",			SP_LIGHTRADIUS},	//radius of the current rtlight
/**/{"l_lightcolour",			SP_LIGHTCOLOUR},	//rgb values of the current rtlight
/**/{"l_lightposition",			SP_LIGHTPOSITION},	//light position in modelspace
	{"l_lightdirection",		SP_LIGHTDIRECTION},	//light direction in modelspace (ortho lights only, instead of position)
/**/{"l_lightcolourscale",		SP_LIGHTCOLOURSCALE},//ambient/diffuse/specular scalers
/**/{"l_cubematrix",			SP_LIGHTCUBEMATRIX},//matrix used to control the rtlight's cubemap projection
/**/{"l_shadowmapproj",			SP_LIGHTSHADOWMAPPROJ},	//compacted projection matrix for shadowmaps
/**/{"l_shadowmapscale",		SP_LIGHTSHADOWMAPSCALE},//should probably use a samplerRect instead...

	{"e_sourcesize",			SP_SOURCESIZE},			//size of VF_SOURCECOLOUR image
	{"e_rendertexturescale",	SP_RENDERTEXTURESCALE},	//scaler for $currentrender, when npot isn't supported.

	{NULL}
};

static char *Shader_ParseBody(char *debugname, char **ptr)
{
	char *body;
	char *start, *end;

	end = *ptr;
	while (*end == ' ' || *end == '\t' || *end == '\r')
		end++;
	if (*end == '\n')
	{
		int count;
		end++;
		while (*end == ' ' || *end == '\t')
			end++;
		if (*end != '{')
		{
			Con_Printf("shader \"%s\" missing program string\n", debugname);
		}
		else
		{
			end++;
			start = end;
			for (count = 1; *end; end++)
			{
				if (*end == '}')
				{
					count--;
					if (!count)
						break;
				}
				else if (*end == '{')
					count++;
			}
			body = BZ_Malloc(end - start + 1);
			memcpy(body, start, end-start);
			body[end-start] = 0;
			*ptr = end+1;/*skip over it all*/

			return body;
		}
	}
	return NULL;
}

static void Shader_SLProgramName (shader_t *shader, shaderpass_t *pass, char **ptr, int qrtype)
{
	/*accepts:
	program
	{
		BLAH
	}
	where BLAH is both vertex+frag with #ifdefs
	or
	program fname
	on one line.
	*/
	char *programbody;
	char *hash;
	program_t *newprog;

	programbody = Shader_ParseBody(shader->name, ptr);
	if (programbody)
	{
		newprog = BZ_Malloc(sizeof(*newprog));
		memset(newprog, 0, sizeof(*newprog));
		newprog->refs = 1;
		if (!Shader_LoadPermutations(shader->name, newprog, programbody, qrtype, 0, NULL))
		{
			BZ_Free(newprog);
			newprog = NULL;
		}

		BZ_Free(programbody);
	}
	else
	{
		hash = strchr(shader->name, '#');
		if (hash)
		{
			//pass the # postfixes from the shader name onto the generic glsl to use
			char newname[512];
			Q_snprintfz(newname, sizeof(newname), "%s%s", Shader_ParseExactString(ptr), hash);
			newprog = Shader_FindGeneric(newname, qrtype);
		}
		else
			newprog = Shader_FindGeneric(Shader_ParseExactString(ptr), qrtype);
	}

	if (pass)
	{
		if (pass->numMergedPasses)
		{
			Shader_ReleaseGeneric(newprog);
			Con_DPrintf("shader %s: program defined after first texture map\n", shader->name);
		}
		else
		{
			Shader_ReleaseGeneric(pass->prog);
			pass->prog = newprog;
		}
	}
	else
	{
		Shader_ReleaseGeneric(shader->prog);
		shader->prog = newprog;
	}
}

static void Shader_GLSLProgramName (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shaderpass_t *pass = ps->pass;
	Shader_SLProgramName(shader,pass,ptr,QR_OPENGL);
}
static void Shader_ProgramName (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shaderpass_t *pass = ps->pass;
	Shader_SLProgramName(shader,pass,ptr,qrenderer);
}
static void Shader_HLSL9ProgramName (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shaderpass_t *pass = ps->pass;
	Shader_SLProgramName(shader,pass,ptr,QR_DIRECT3D9);
}
static void Shader_HLSL11ProgramName (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shaderpass_t *pass = ps->pass;
	Shader_SLProgramName(shader,pass,ptr,QR_DIRECT3D11);
}

static void Shader_ReflectCube(parsestate_t *ps, char **ptr)
{
	char *token = Shader_ParseSensString(ptr);
	unsigned int flags = Shader_SetImageFlags (ps, ps->pass, &token, IF_TEXTYPE_CUBE);
	ps->s->defaulttextures->reflectcube = Shader_FindImage(ps, token, flags);
}
static void Shader_ReflectMask(parsestate_t *ps, char **ptr)
{
	char *token = Shader_ParseSensString(ptr);
	unsigned int flags = Shader_SetImageFlags (ps, ps->pass, &token, 0);
	ps->s->defaulttextures->reflectmask = Shader_FindImage(ps, token, flags);
}

static void Shader_DiffuseMap(parsestate_t *ps, char **ptr)
{
	char *token = Shader_ParseSensString(ptr);
	unsigned int flags = Shader_SetImageFlags (ps, ps->pass, &token, 0);
	ps->s->defaulttextures->base = Shader_FindImage(ps, token, flags);

	Q_strncpyz(ps->s->defaulttextures->mapname, token, sizeof(ps->s->defaulttextures->mapname));
}
static void Shader_SpecularMap(parsestate_t *ps, char **ptr)
{
	char *token = Shader_ParseSensString(ptr);
	unsigned int flags = Shader_SetImageFlags (ps, ps->pass, &token, 0);
	ps->s->defaulttextures->specular = Shader_FindImage(ps, token, flags);
}
static void Shader_NormalMap(parsestate_t *ps, char **ptr)
{
	char *token = Shader_ParseSensString(ptr);
	unsigned int flags = Shader_SetImageFlags (ps, ps->pass, &token, IF_TRYBUMP|IF_NOSRGB);
	ps->s->defaulttextures->bump = Shader_FindImage(ps, token, flags);
}
static void Shader_FullbrightMap(parsestate_t *ps, char **ptr)
{
	char *token = Shader_ParseSensString(ptr);
	unsigned int flags = Shader_SetImageFlags (ps, ps->pass, &token, 0);
	ps->s->defaulttextures->fullbright = Shader_FindImage(ps, token, flags);
}
static void Shader_UpperMap(parsestate_t *ps, char **ptr)
{
	char *token = Shader_ParseSensString(ptr);
	unsigned int flags = Shader_SetImageFlags (ps, ps->pass, &token, 0);
	ps->s->defaulttextures->upperoverlay = Shader_FindImage(ps, token, flags);
}
static void Shader_LowerMap(parsestate_t *ps, char **ptr)
{
	char *token = Shader_ParseSensString(ptr);
	unsigned int flags = Shader_SetImageFlags (ps, ps->pass, &token, 0);
	ps->s->defaulttextures->loweroverlay = Shader_FindImage(ps, token, flags);
}
static void Shader_DisplacementMap(parsestate_t *ps, char **ptr)
{
	char *token = Shader_ParseSensString(ptr);
	unsigned int flags = Shader_SetImageFlags (ps, ps->pass, &token, IF_NOSRGB);
	ps->s->defaulttextures->displacement = Shader_FindImage(ps, token, flags);
}

static void Shaderpass_QF_Material(parsestate_t *ps, char **ptr)
{	//qf_material BASETEXTURE NORMALMAP SPECULARMAP
	unsigned int flags;
	char *progname = "defaultwall";
	char *token;
	char *hash = strchr(ps->s->name, '#');
	if (hash)
	{
		//pass the # postfixes from the shader name onto the generic glsl to use
		char newname[512];
		Q_snprintfz(newname, sizeof(newname), "%s%s", progname, hash);
		ps->pass->prog = Shader_FindGeneric(newname, qrenderer);
	}
	else
		ps->pass->prog = Shader_FindGeneric(progname, qrenderer);

	token = Shader_ParseSensString(ptr);
	if (*token && strcmp(token, "-"))
	{
		flags = Shader_SetImageFlags (ps, ps->pass, &token, 0);
		if (*token)
			ps->s->defaulttextures->base = Shader_FindImage(ps, token, flags);
		else
		{
			token = ps->s->name;
			if (hash)
				*hash = 0;
			ps->s->defaulttextures->base = Shader_FindImage(ps, token, flags);
			if (hash)
				*hash = '#';
		}
	}

	if (*token)
		token = Shader_ParseSensString(ptr);
	if (*token && strcmp(token, "-"))
	{
		flags = Shader_SetImageFlags (ps, NULL, &token, IF_TRYBUMP|IF_NOSRGB);
		ps->s->defaulttextures->bump = Shader_FindImage(ps, token, flags);
	}

	if (*token)
		token = Shader_ParseSensString(ptr);
	if (*token && strcmp(token, "-"))
	{
		flags = Shader_SetImageFlags (ps, NULL, &token, 0);
		ps->s->defaulttextures->specular = Shader_FindImage(ps, token, flags);
	}
}

static qboolean Shaderpass_MapGen (parsestate_t *ps, shaderpass_t *pass, char *tname);

static void Shader_Translucent(parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shader->flags |= SHADER_BLEND;
}

static void Shader_PortalFBOScale(parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shader->portalfboscale = Shader_ParseFloat(shader, ptr, 0);
	shader->portalfboscale = max(shader->portalfboscale, 0);
}

static void Shader_DP_Camera(parsestate_t *ps, char **ptr)
{
	ps->s->sort = SHADER_SORT_PORTAL;
	ps->dpwatertype |= 4;
}
static void Shader_DP_Water(parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	ps->parseflags |= SPF_PROGRAMIFY;

	ps->dpwatertype |= 3;
	ps->reflectmin = Shader_ParseFloat(shader, ptr, 0);
	ps->reflectmax = Shader_ParseFloat(shader, ptr, 0);
	ps->refractfactor = Shader_ParseFloat(shader, ptr, 0);
	ps->reflectfactor = Shader_ParseFloat(shader, ptr, 0);
	Shader_ParseVector(shader, ptr, ps->refractcolour);
	Shader_ParseVector(shader, ptr, ps->reflectcolour);
	ps->wateralpha = Shader_ParseFloat(shader, ptr, 0);
}
static void Shader_DP_Reflect(parsestate_t *ps, char **ptr)
{
	ps->parseflags |= SPF_PROGRAMIFY;

	ps->dpwatertype |= 1;
	ps->reflectmin = 1;
	ps->reflectmax = 1;
	ps->reflectfactor = Shader_ParseFloat(ps->s, ptr, 0);
	Shader_ParseVector(ps->s, ptr, ps->reflectcolour);
}
static void Shader_DP_Refract(parsestate_t *ps, char **ptr)
{
	ps->parseflags |= SPF_PROGRAMIFY;

	ps->dpwatertype |= 2;
	ps->refractfactor = Shader_ParseFloat(ps->s, ptr, 0);
	Shader_ParseVector(ps->s, ptr, ps->refractcolour);
}

static void Shader_DP_OffsetMapping(parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	char *token = Shader_ParseString(ptr);
	if (!strcmp(token, "disable") || !strcmp(token, "none") || !strcmp(token, "off"))
		ps->offsetmappingmode = 0;
	else if (!strcmp(token, "default") || !strcmp(token, "normal"))
		ps->offsetmappingmode = -1;
	else if (!strcmp(token, "linear"))
		ps->offsetmappingmode = 1;
	else if (!strcmp(token, "relief"))
		ps->offsetmappingmode = 2;
	ps->offsetmappingscale = Shader_ParseFloat(shader, ptr, 1);
	token = Shader_ParseString(ptr);
	if (!strcmp(token, "bias"))
		ps->offsetmappingbias = Shader_ParseFloat(shader, ptr, 0.5);
	else if (!strcmp(token, "match"))
		ps->offsetmappingbias = 1.0 - Shader_ParseFloat(shader, ptr, 0.5);
	else if (!strcmp(token, "match8"))
		ps->offsetmappingbias = 1.0 - (1.0/255) * Shader_ParseFloat(shader, ptr, 128);
	else if (!strcmp(token, "match16"))
		ps->offsetmappingbias = 1.0 - (1.0/65535) * Shader_ParseFloat(shader, ptr, 32768);
}
static void Shader_DP_GlossScale(parsestate_t *ps, char **ptr)
{
	ps->specularvalscale = Shader_ParseFloat(ps->s, ptr, 0);
}
static void Shader_DP_GlossExponent(parsestate_t *ps, char **ptr)
{
	ps->specularexpscale = Shader_ParseFloat(ps->s, ptr, 0);
}

static void Shader_FactorBase(parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shader->factors[MATERIAL_FACTOR_BASE][0] = Shader_ParseFloat(shader, ptr, 1);
	shader->factors[MATERIAL_FACTOR_BASE][1] = Shader_ParseFloat(shader, ptr, 1);
	shader->factors[MATERIAL_FACTOR_BASE][2] = Shader_ParseFloat(shader, ptr, 1);
	shader->factors[MATERIAL_FACTOR_BASE][3] = Shader_ParseFloat(shader, ptr, 1);
}
static void Shader_FactorSpec(parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shader->factors[MATERIAL_FACTOR_SPEC][0] = Shader_ParseFloat(shader, ptr, 1);
	shader->factors[MATERIAL_FACTOR_SPEC][1] = Shader_ParseFloat(shader, ptr, 1);
	shader->factors[MATERIAL_FACTOR_SPEC][2] = Shader_ParseFloat(shader, ptr, 1);
	shader->factors[MATERIAL_FACTOR_SPEC][3] = Shader_ParseFloat(shader, ptr, 1);
}
static void Shader_FactorEmit(parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shader->factors[MATERIAL_FACTOR_EMIT][0] = Shader_ParseFloat(shader, ptr, 1);
	shader->factors[MATERIAL_FACTOR_EMIT][1] = Shader_ParseFloat(shader, ptr, 1);
	shader->factors[MATERIAL_FACTOR_EMIT][2] = Shader_ParseFloat(shader, ptr, 1);
	shader->factors[MATERIAL_FACTOR_EMIT][3] = Shader_ParseFloat(shader, ptr, 1);
}

static void Shader_BEMode(parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	char subname[1024];
	int mode;
	char tokencopy[1024];
	char *token;
	char *embed = NULL;
	token = Shader_ParseString(ptr);
	if (!Q_stricmp(token, "rtlight"))
		mode = -1;	//all light types
	else if (!Q_stricmp(token, "rtlight_only"))
		mode = LSHADER_STANDARD;
	else if (!Q_stricmp(token, "rtlight_smap"))
		mode = LSHADER_SMAP;
	else if (!Q_stricmp(token, "rtlight_spot"))
		mode = LSHADER_SPOT;
	else if (!Q_stricmp(token, "rtlight_cube"))
		mode = LSHADER_CUBE;
	else if (!Q_stricmp(token, "rtlight_cube_smap"))
		mode = LSHADER_CUBE|LSHADER_SMAP;
	else if (!Q_stricmp(token, "rtlight_cube_spot"))		//doesn't make sense.
		mode = LSHADER_CUBE|LSHADER_SPOT;
	else if (!Q_stricmp(token, "rtlight_spot_smap"))
		mode = LSHADER_SMAP|LSHADER_SPOT;
	else if (!Q_stricmp(token, "rtlight_cube_spot_smap"))	//doesn't make sense.
		mode = LSHADER_CUBE|LSHADER_SPOT|LSHADER_SMAP;
	else if (!Q_stricmp(token, "crepuscular"))
		mode = bemoverride_crepuscular;
	else if (!Q_stricmp(token, "depthonly"))
		mode = bemoverride_depthonly;
	else if (!Q_stricmp(token, "depthdark"))
		mode = bemoverride_depthdark;
	else if (!Q_stricmp(token, "gbuffer") || !Q_stricmp(token, "prelight"))
		mode = bemoverride_gbuffer;
	else if (!Q_stricmp(token, "fog"))
		mode = bemoverride_fog;
	else
	{
		Con_DPrintf(CON_WARNING "Shader %s specifies unknown bemode %s.\n", shader->name, token);
		return;	//not supported.
	}

	embed = Shader_ParseBody(shader->name, ptr);
	if (embed)
	{
		int l = strlen(embed) + 6;
		char *b = BZ_Malloc(l);
		Q_snprintfz(b, l, "{\n%s\n}\n", embed);
		BZ_Free(embed);
		embed = b;
		//generate a unique name
		Q_snprintfz(tokencopy, sizeof(tokencopy), "%s_mode%i", shader->name, mode);
	}
	else
	{
		token = Shader_ParseString(ptr);
		Q_strncpyz(tokencopy, token, sizeof(tokencopy));	//make sure things don't go squiff.
	}

	if (mode == -1)
	{
		//shorthand for rtlights
		for (mode = 0; mode < LSHADER_MODES; mode++)
		{
			if ((mode & LSHADER_CUBE) && (mode & (LSHADER_SPOT|LSHADER_ORTHO)))
				continue;	//cube projections don't make sense when the light isn't projecting a cube
			if ((mode & LSHADER_ORTHO) && (mode & LSHADER_SPOT))
				continue;	//ortho+spot are mutually exclusive.
			Q_snprintfz(subname, sizeof(subname), "%s%s%s%s%s%s", tokencopy,
																(mode & LSHADER_SMAP)?"#PCF":"",
																(mode & LSHADER_SPOT)?"#SPOT":"",
																(mode & LSHADER_CUBE)?"#CUBE":"",
																(mode & LSHADER_ORTHO)?"#ORTHO":"",
#ifdef GLQUAKE
																(qrenderer == QR_OPENGL && gl_config.arb_shadow && (mode & LSHADER_SMAP))?"#USE_ARB_SHADOW":""
#else
																""
#endif
																);
			shader->bemoverrides[mode] = R_RegisterCustom(subname, shader->usageflags|(embed?SUR_FORCEFALLBACK:0), embed?Shader_DefaultScript:NULL, embed);
		}
	}
	else
	{
		shader->bemoverrides[mode] = R_RegisterCustom(tokencopy, shader->usageflags|(embed?SUR_FORCEFALLBACK:0), embed?Shader_DefaultScript:NULL, embed);
	}
	if (embed)
		BZ_Free(embed);
}

static shaderkey_t shaderkeys[] =
{
#define Q3 NULL
	{"cull",				Shader_Cull,				Q3},
	{"skyparms",			Shader_SkyParms,			Q3},
	{"fogparms",			Shader_FogParms,			Q3},
	{"surfaceparm",			Shader_SurfaceParm,			Q3},
	{"nomipmaps",			Shader_NoMipMaps,			Q3},
	{"nopicmip",			Shader_NoPicMip,			Q3},
	{"polygonoffset",		Shader_PolygonOffset,		Q3},
	{"sort",				Shader_Sort,				Q3},
	{"deformvertexes",		Shader_DeformVertexes,		Q3},
	{"portal",				Shader_Portal,				Q3},
	{"entitymergable",		Shader_EntityMergable,		Q3},

	//fte extensions
	{"clutter",				Shader_ClutterParms,		"fte"},
	{"deferredlight",		Shader_Deferredlight,		"fte"},	//(sort = prelight)
//	{"lpp_light",			Shader_Deferredlight,		"fte"},	//(sort = prelight)
	{"affine",				Shader_Affine,				"fte"},	//some hardware is horribly slow, and can benefit from certain hints.

	{"bemode",				Shader_BEMode,				"fte"},

	{"diffusemap",			Shader_DiffuseMap,			"fte"},
	{"normalmap",			Shader_NormalMap,			"fte"},
	{"specularmap",			Shader_SpecularMap,			"fte"},
	{"fullbrightmap",		Shader_FullbrightMap,		"fte"},
	{"uppermap",			Shader_UpperMap,			"fte"},
	{"lowermap",			Shader_LowerMap,			"fte"},
	{"reflectmask",			Shader_ReflectMask,			"fte"},
	{"displacementmap",		Shader_DisplacementMap,		"fte"},

	{"portalfboscale",		Shader_PortalFBOScale,		"fte"},	//portal/mirror/refraction/reflection FBOs are resized by this scale
	{"basefactor",			Shader_FactorBase,			"fte"},	//material scalers for glsl
	{"specularfactor",		Shader_FactorSpec,			"fte"},	//material scalers for glsl
	{"fullbrightfactor",	Shader_FactorEmit,			"fte"},	//material scalers for glsl

	//TODO: PBR textures...
//	{"albedomap",			Shader_DiffuseMap,			"fte"},	//rgb(a)
//	{"loweruppermap",		Shader_LowerUpperMap,		"fte"}, //r=lower, g=upper (team being more important than personal colours, this allows the texture to gracefully revert to red-only)
	//{"normalmap",			Shader_NormalMap,			"fte"},	//xy-h
//	{"ormmap",				Shader_SpecularMap,			"fte"},	//r=occlusion, g=metalness, b=roughness.
	//{"glowmap",			Shader_FullbrightMap,		"fte"}, //rgb

	/*program stuff at the material level is an outdated practise.*/
	{"program",				Shader_ProgramName,			"fte"},	//usable with any renderer that has a usable shader language...
	{"glslprogram",			Shader_GLSLProgramName,		"fte"},	//for renderers that accept embedded glsl
	{"hlslprogram",			Shader_HLSL9ProgramName,	"fte"},	//for d3d with embedded hlsl
	{"hlsl11program",		Shader_HLSL11ProgramName,	"fte"},	//for d3d with embedded hlsl
//	{"progblendfunc",		Shader_ProgBlendFunc,		"fte"},	//specifies the blend mode (actually just overrides the first subpasses' blendmode.
//	{"progmap",				Shader_ProgMap,				"fte"},	//avoids needing extra subpasses (actually just inserts an extra pass).

	//dp compat
	{"reflectcube",			Shader_ReflectCube,			"dp"},
	{"camera",				Shader_DP_Camera,			"dp"},
	{"water",				Shader_DP_Water,			"dp"},
	{"reflect",				Shader_DP_Reflect,			"dp"},
	{"refract",				Shader_DP_Refract,			"dp"},
	{"offsetmapping",		Shader_DP_OffsetMapping,	"dp"},
	{"shadow",				NULL,						"dp"},
	{"noshadow",			NULL,						"dp"},
	{"polygonoffset",		NULL,						"dp"},
	{"glossintensitymod",	Shader_DP_GlossScale,		"dp"},	//scales r_shadow_glossintensity(=1), aka: gl_specular
	{"glossexponentmod",	Shader_DP_GlossExponent,	"dp"},	//scales r_shadow_glossexponent(=32)

	/*doom3 compat*/
	{"diffusemap",			Shader_DiffuseMap,			"doom3"},	//macro for "{\nstage diffusemap\nmap <map>\n}"
	{"bumpmap",				Shader_NormalMap,			"doom3"},	//macro for "{\nstage bumpmap\nmap <map>\n}"
	{"discrete",			NULL,						"doom3"},
	{"nonsolid",			NULL,						"doom3"},
	{"noimpact",			NULL,						"doom3"},
	{"translucent",			Shader_Translucent,			"doom3"},
	{"noshadows",			NULL,						"doom3"},
	{"nooverlays",			NULL,						"doom3"},
	{"nofragment",			NULL,						"doom3"},

	/*RTCW compat*/
	{"nocompress",			NULL,						"rtcw"},
	{"allowcompress",		NULL,						"rtcw"},
	{"nofog",				NULL,						"rtcw"},
	{"skyfogvars",			NULL,						"rtcw"},
	{"sunshader",			NULL,						"rtcw"},
	{"sun",					NULL,						"q3map2"},	//provides rgb and dir
	{"sunExt",				NULL,						"q3map2"},	//treated as an alias
	{"fogParms",			NULL,						"rtcw"},	//sets a cvar. *shudder*
	{"fogvars",				NULL,						"rtcw"},	//sets a cvar. *shudder*
	{"waterfogvars",		NULL,						"rtcw"},	//sets a cvar. *shudder*
	{"light",				NULL,						"rtcw"},	//for q3map2, not us
	{"lightgridmulamb",		NULL,						"rtcw"},	//urm
	{"lightgridmuldir",		NULL,						"rtcw"},	//not really sure how this is useful to us

	/*qfusion / warsow compat*/
//	{"skyparms2",			NULL,						"qf"},	//skyparms without the underscore.
//	{"skyparmssides",		NULL,						"qf"},	//skyparms with explicitly-named faces
//	{"nocompress",			NULL,						"qf"},	//disables opportunistic compression (doesn't affect compressed source images, apparently)
//	{"nofiltering",			NULL,						"qf"},	//misnomer. there is always 'filtering'. this means to use nearest filtering for min and mag, as well as no mipmaps.
//	{"smallestmipmapsize",	NULL,						"qf"},	//mips with a size less than the specified value are dropped.
//	{"stenciltest",			NULL,						"qf"},	//enables GL_STENCIL_TEST, which is special-case stuff that I see no reason to support
//	{"offsetmappingscale",	NULL,						"qf"},
//	{"glossexponent",		NULL,						"qf"},
//	{"glossintensity",		NULL,						"qf"},
//	{"template",			NULL,						"qf"},	//parses some other shader, with $3 etc arg expansion
	{"skip",				NULL,						"qf"},	//just skips the line. acts like a comment. no idea why they can't just use a comment.
//	{"softparticle",		NULL,						"qf"},	//uses screen depth, if possible. 
//	{"forceworldoutlines",	NULL,						"qf"},	//looks like an ugly hack to me.

	{NULL,				NULL}
};

static struct
{
	char *name;
	char *body;
} shadermacros[] =
{
	{"decal_macro", 	"polygonOffset 1\ndiscrete\nsort decal\nnoShadows"},
//	{"diffusemap", 		"{\nblend diffusemap\nmap %1\n}"},
//	{"bumpmap", 		"{\nblend bumpmap\nmap %1\n}"},
//	{"specularmap", 	"{\nblend specularmap\nmap %1\n}"},
	{NULL}
};

// ===============================================================

static qboolean Shaderpass_MapGen (parsestate_t *ps, shaderpass_t *pass, char *tname)
{
	shader_t *shader = ps->s;
	int tcgen = TC_GEN_BASE;
	if (!Q_stricmp (tname, "$lightmap"))
	{
		tcgen = TC_GEN_LIGHTMAP;
		pass->flags |= SHADER_PASS_LIGHTMAP | SHADER_PASS_NOMIPMAP;
		pass->texgen = T_GEN_LIGHTMAP;
		shader->flags |= SHADER_HASLIGHTMAP;
	}
	else if (!Q_stricmp (tname, "$deluxmap"))
	{
		tcgen = TC_GEN_LIGHTMAP;
		pass->flags |= SHADER_PASS_DELUXMAP | SHADER_PASS_NOMIPMAP;
		pass->texgen = T_GEN_DELUXMAP;
	}
	else if (!Q_stricmp (tname, "$diffuse"))
	{
		pass->texgen = T_GEN_DIFFUSE;
		shader->flags |= SHADER_HASDIFFUSE;
	}
	else if (!Q_stricmp (tname, "$paletted"))
	{
		pass->texgen = T_GEN_PALETTED;
		shader->flags |= SHADER_HASPALETTED;
	}
	else if (!Q_stricmp (tname, "$normalmap"))
	{
		pass->texgen = T_GEN_NORMALMAP;
		shader->flags |= SHADER_HASNORMALMAP;
	}
	else if (!Q_stricmp (tname, "$specular"))
	{
		pass->texgen = T_GEN_SPECULAR;
		shader->flags |= SHADER_HASGLOSS;
	}
	else if (!Q_stricmp (tname, "$fullbright"))
	{
		pass->texgen = T_GEN_FULLBRIGHT;
		shader->flags |= SHADER_HASFULLBRIGHT;
	}
	else if (!Q_stricmp (tname, "$upperoverlay"))
	{
		shader->flags |= SHADER_HASTOPBOTTOM;
		pass->texgen = T_GEN_UPPEROVERLAY;
	}
	else if (!Q_stricmp (tname, "$loweroverlay"))
	{
		shader->flags |= SHADER_HASTOPBOTTOM;
		pass->texgen = T_GEN_LOWEROVERLAY;
	}
	else if (!Q_stricmp (tname, "$reflectcube"))
	{
		shader->flags |= SHADER_HASREFLECTCUBE;
		pass->texgen = T_GEN_REFLECTCUBE;
	}
	else if (!Q_stricmp (tname, "$reflectmask"))
	{
		pass->texgen = T_GEN_REFLECTMASK;
	}
	else if (!Q_stricmp (tname, "$displacement"))
	{
		shader->flags |= SHADER_HASDISPLACEMENT;
		pass->texgen = T_GEN_DISPLACEMENT;
	}
	else if (!Q_stricmp (tname, "$shadowmap"))
	{
		pass->texgen = T_GEN_SHADOWMAP;
		pass->flags |= SHADER_PASS_DEPTHCMP;
	}
	else if (!Q_stricmp (tname, "$lightcubemap"))
	{
		pass->texgen = T_GEN_LIGHTCUBEMAP;
	}
	else if (!Q_stricmp (tname, "$currentrender"))
	{
		pass->texgen = T_GEN_CURRENTRENDER;
		shader->flags |= SHADER_HASCURRENTRENDER;
	}
	else if (!Q_stricmp (tname, "$sourcecolour"))
	{
		pass->texgen = T_GEN_SOURCECOLOUR;
	}
	else if (!Q_stricmp (tname, "$sourcecube"))
	{
		pass->texgen = T_GEN_SOURCECUBE;
	}
	else if (!Q_stricmp (tname, "$sourcedepth"))
	{
		pass->texgen = T_GEN_SOURCEDEPTH;
	}
	else if (!Q_strnicmp (tname, "$gbuffer", 8))
	{
		unsigned idx = strtoul(tname+8, &tname, 10);
		if (*tname || idx >= GBUFFER_COUNT)
			return false;
		pass->texgen = T_GEN_GBUFFER0 + idx;
	}
	else if (!Q_stricmp (tname, "$reflection"))
	{
		shader->flags |= SHADER_HASREFLECT;
		pass->texgen = T_GEN_REFLECTION;
	}
	else if (!Q_stricmp (tname, "$refraction"))
	{
		shader->flags |= SHADER_HASREFRACT;
		pass->texgen = T_GEN_REFRACTION;
	}
	else if (!Q_stricmp (tname, "$refractiondepth"))
	{
		shader->flags |= SHADER_HASREFRACT;
		pass->texgen = T_GEN_REFRACTIONDEPTH;
	}
	else if (!Q_stricmp (tname, "$ripplemap"))
	{
		shader->flags |= SHADER_HASRIPPLEMAP;
		pass->texgen = T_GEN_RIPPLEMAP;
	}
	else if (!Q_stricmp (tname, "$null"))
	{
		pass->flags |= SHADER_PASS_NOMIPMAP|SHADER_PASS_DETAIL;
		pass->texgen = T_GEN_SINGLEMAP;
	}
	else
		return false;

	if (pass->tcgen == TC_GEN_UNSPECIFIED)
		pass->tcgen = tcgen;
	return true;
}

shaderpass_t *Shaderpass_DefineMap(parsestate_t *ps, shaderpass_t *pass)
{
	//'map foo' works a bit differently when there's a program in the same pass.
	//instead of corrupting the previous one, it collects multiple maps so that {prog foo;map t0;map t1; map t2; blendfunc add} can work as expected
	if (pass->prog)
	{
		if (pass->numMergedPasses==0)
			pass->numMergedPasses++;
		else
		{	//FIXME: bounds check!
			if (ps->s->numpasses == SHADER_PASS_MAX || ps->s->numpasses == SHADER_TMU_MAX)
			{
				Con_DPrintf (CON_WARNING "Shader %s has too many texture passes.\n", ps->s->name);
				ps->droppass = true;
			}
//			else if (shader->numpasses == be_maxpasses)
//				ps->droppass = true;
			else
			{
				pass->numMergedPasses++;
				ps->s->numpasses++;
			}
			pass = ps->s->passes+ps->s->numpasses-1;
			memset(pass, 0, sizeof(*pass));
		}
	}
	else
		pass->numMergedPasses = 1;
	return pass;
}

static void Shaderpass_Map (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shaderpass_t *pass = ps->pass;
	int flags;
	char *token;

	pass = Shaderpass_DefineMap(ps, pass);

	pass->anim_frames[0] = r_nulltex;

	token = Shader_ParseSensString (ptr);

	flags = Shader_SetImageFlags (ps, pass, &token, 0);
	if (!Shaderpass_MapGen(ps, pass, token))
	{
		switch((flags & IF_TEXTYPEMASK) >> IF_TEXTYPESHIFT)
		{
		case PTI_2D:
			pass->texgen = T_GEN_SINGLEMAP;
			break;
		case PTI_3D:
			pass->texgen = T_GEN_3DMAP;
			break;
		case PTI_CUBE:
			pass->texgen = T_GEN_CUBEMAP;
			break;
		default:
			pass->texgen = T_GEN_SINGLEMAP;
			break;
		}

		if (pass->tcgen == TC_GEN_UNSPECIFIED)
			pass->tcgen = TC_GEN_BASE;
		if (!*shader->defaulttextures->mapname && *token != '$' && pass->tcgen == TC_GEN_BASE)
			Q_strncpyz(shader->defaulttextures->mapname, token, sizeof(shader->defaulttextures->mapname));
		pass->anim_frames[0] = Shader_FindImage (ps, token, flags);
	}
}

static void Shaderpass_AnimMap_Flag (parsestate_t *ps, char **ptr, unsigned int flags)
{
	shader_t *shader = ps->s;
	shaderpass_t *pass = ps->pass;
	char *token;
	texid_t image;
	qboolean isdiffuse = false;

	flags |= Shader_SetImageFlags (ps, ps->pass, NULL, 0);

	if (pass->tcgen == TC_GEN_UNSPECIFIED)
		pass->tcgen = TC_GEN_BASE;
	pass->flags |= SHADER_PASS_ANIMMAP;
	pass->texgen = T_GEN_ANIMMAP;
	pass->anim_fps = Shader_ParseFloat (shader, ptr, 0);
	pass->anim_numframes = 0;

	for ( ; ; )
	{
		token = Shader_ParseString(ptr);
		if (!token[0])
		{
			break;
		}

		if (!pass->anim_numframes && !*shader->defaulttextures->mapname && *token != '$' && pass->tcgen == TC_GEN_BASE)
		{
			isdiffuse = true;
			shader->defaulttextures_fps = pass->anim_fps;
		}

		if (pass->anim_numframes < SHADER_MAX_ANIMFRAMES)
		{
			image = Shader_FindImage (ps, token, flags);

			if (isdiffuse)
			{
				if (shader->numdefaulttextures < pass->anim_numframes+1)
				{
					int newmax = pass->anim_numframes+1;
					shader->defaulttextures = BZ_Realloc(shader->defaulttextures, sizeof(*shader->defaulttextures) * (newmax));
					memset(shader->defaulttextures+shader->numdefaulttextures, 0, sizeof(*shader->defaulttextures) * (newmax-shader->numdefaulttextures));
					shader->numdefaulttextures = newmax;
				}
				Q_strncpyz(shader->defaulttextures[pass->anim_numframes].mapname, token, sizeof(shader->defaulttextures[pass->anim_numframes].mapname));
			}
			if (!TEXVALID(image))
			{
				pass->anim_frames[pass->anim_numframes++] = missing_texture;
				Con_DPrintf (CON_WARNING "Shader %s has an animmap with no image: %s.\n", shader->name, token );
			}
			else
			{
				pass->anim_frames[pass->anim_numframes++] = image;
			}
		}
	}
}
static void Shaderpass_AnimMap (parsestate_t *ps, char **ptr)
{
	Shaderpass_AnimMap_Flag(ps, ptr, 0);
}
static void Shaderpass_QF_AnimClampMap (parsestate_t *ps, char **ptr)
{
	Shaderpass_AnimMap_Flag(ps, ptr, IF_CLAMP);
}

static void Shaderpass_ClampMap (parsestate_t *ps, char **ptr)
{
	shaderpass_t *pass = ps->pass;
	int flags;
	char *token;

	token = Shader_ParseSensString (ptr);

	flags = Shader_SetImageFlags (ps, pass, &token, IF_CLAMP);
	if (!Shaderpass_MapGen(ps, pass, token))
	{
		if (pass->tcgen == TC_GEN_UNSPECIFIED)
			pass->tcgen = TC_GEN_BASE;
		pass->anim_frames[0] = Shader_FindImage (ps, token, flags);

		switch((flags & IF_TEXTYPEMASK) >> IF_TEXTYPESHIFT)
		{
		default:
		case PTI_2D:
			pass->texgen = T_GEN_SINGLEMAP;
			break;
		case PTI_3D:
			pass->texgen = T_GEN_3DMAP;
			break;
		case PTI_CUBE:
			pass->texgen = T_GEN_CUBEMAP;
			break;
		}

		if (!TEXVALID(pass->anim_frames[0]))
		{
			if ((flags & IF_TEXTYPEMASK)!=IF_TEXTYPE_2D)
				pass->anim_frames[0] = r_nulltex;
			else
				pass->anim_frames[0] = missing_texture;
			Con_DPrintf (CON_WARNING "Shader %s has a stage with no image: %s.\n", ps->s->name, token);
		}
	}
}

static void Shaderpass_VideoMap (parsestate_t *ps, char **ptr)
{
	char		*token = Shader_ParseSensString (ptr);

#ifndef HAVE_MEDIA_DECODER
	(void)token;
#else
	shader_t *shader = ps->s;
	shaderpass_t *pass = ps->pass;

	if (pass->cin)
		Z_Free (pass->cin);

	pass->cin = Media_StartCin(token);
	if (!pass->cin)
	{
		Con_DPrintf (CON_WARNING "(shader %s) Couldn't load video %s\n", shader->name, token);
	}

	if (pass->cin)
	{
		pass->flags |= SHADER_PASS_VIDEOMAP;
		shader->flags |= SHADER_VIDEOMAP;
		pass->texgen = T_GEN_VIDEOMAP;
	}
	else
	{
		pass->texgen = T_GEN_DIFFUSE;
		pass->rgbgen = RGB_GEN_CONST;
		pass->rgbgen_func.type = SHADER_FUNC_CONSTANT;
		pass->rgbgen_func.args[0] = pass->rgbgen_func.args[1] = pass->rgbgen_func.args[2] = 0;
	}
#endif
}

static void Shaderpass_RTCW_Map_16bit (parsestate_t *ps, char **ptr)
{
	if (!gl_load24bit.ival)	//urm, not sure if suitable choice of cvar
		Shaderpass_Map(ps, ptr);
}
static void Shaderpass_RTCW_Map_32bit (parsestate_t *ps, char **ptr)
{
	if (gl_load24bit.ival)
		Shaderpass_Map(ps, ptr);
}
static void Shaderpass_RTCW_Map_s3tc (parsestate_t *ps, char **ptr)
{
	if (sh_config.texfmt[PTI_BC3_RGBA] && gl_compress.ival)
		Shaderpass_Map(ps, ptr);
}
static void Shaderpass_RTCW_Map_nos3tc (parsestate_t *ps, char **ptr)
{
	if (!(sh_config.texfmt[PTI_BC3_RGBA] && gl_compress.ival))
		Shaderpass_Map(ps, ptr);
}
static void Shaderpass_RTCW_AnimMap_s3tc (parsestate_t *ps, char **ptr)
{
	if ((sh_config.texfmt[PTI_BC3_RGBA] && gl_compress.ival))
		Shaderpass_AnimMap(ps, ptr);
}
static void Shaderpass_RTCW_AnimMap_nos3tc (parsestate_t *ps, char **ptr)
{
	if (!(sh_config.texfmt[PTI_BC3_RGBA] && gl_compress.ival))
		Shaderpass_AnimMap(ps, ptr);
}

static void Shaderpass_SLProgramName (shader_t *shader, shaderpass_t *pass, char **ptr, int qrtype)
{
	/*accepts:
	program
	{
		BLAH
	}
	where BLAH is both vertex+frag with #ifdefs
	or
	program fname
	on one line.
	*/
	//char *programbody;
	char *hash;

	/*programbody = Shader_ParseBody(shader->name, ptr);
	if (programbody)
	{
		shader->prog = BZ_Malloc(sizeof(*shader->prog));
		memset(shader->prog, 0, sizeof(*shader->prog));
		shader->prog->refs = 1;
		if (!Shader_LoadPermutations(shader->name, shader->prog, programbody, qrtype, 0, NULL))
		{
			BZ_Free(shader->prog);
			shader->prog = NULL;
		}

		BZ_Free(programbody);
		return;
	}*/

	hash = strchr(shader->name, '#');
	if (hash)
	{
		//pass the # postfixes from the shader name onto the generic glsl to use
		char newname[512];
		Q_snprintfz(newname, sizeof(newname), "%s%s", Shader_ParseExactString(ptr), hash);
		pass->prog = Shader_FindGeneric(newname, qrtype);
	}
	else
		pass->prog = Shader_FindGeneric(Shader_ParseExactString(ptr), qrtype);
}
static void Shaderpass_ProgramName (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shaderpass_t *pass = ps->pass;
	Shaderpass_SLProgramName(shader,pass,ptr,qrenderer);
}

static void Shaderpass_RGBGen (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shaderpass_t *pass = ps->pass;
	char		*token;

	token = Shader_ParseString (ptr);
	if (!Q_stricmp (token, "identitylighting"))
		pass->rgbgen = RGB_GEN_IDENTITY_LIGHTING;
	else if (!Q_stricmp (token, "identity"))
		pass->rgbgen = RGB_GEN_IDENTITY;
	else if (!Q_stricmp (token, "wave"))
	{
		pass->rgbgen = RGB_GEN_WAVE;
		Shader_ParseFunc (ps, "rgbGen wave", ptr, &pass->rgbgen_func);
	}
	else if (!Q_stricmp(token, "entity"))
		pass->rgbgen = RGB_GEN_ENTITY;
	else if (!Q_stricmp (token, "oneMinusEntity"))
		pass->rgbgen = RGB_GEN_ONE_MINUS_ENTITY;
	else if (!Q_stricmp (token, "vertex"))
	{
		pass->rgbgen = RGB_GEN_VERTEX_LIGHTING;
		if (pass->alphagen == ALPHA_GEN_UNDEFINED)	//matches Q3, and is a perf gain, even if its inconsistent.
			pass->alphagen = ALPHA_GEN_VERTEX;
	}
	else if (!Q_stricmp (token, "oneMinusVertex"))
		pass->rgbgen = RGB_GEN_ONE_MINUS_VERTEX;
	else if (!Q_stricmp (token, "lightingDiffuse"))
		pass->rgbgen = RGB_GEN_LIGHTING_DIFFUSE;
	else if (!Q_stricmp (token, "entitylighting"))
		pass->rgbgen = RGB_GEN_ENTITY_LIGHTING_DIFFUSE;
	else if (!Q_stricmp (token, "exactvertex"))
		pass->rgbgen = RGB_GEN_VERTEX_EXACT;
	else if (!Q_stricmp (token, "const") || !Q_stricmp (token, "constant"))
	{
		pass->rgbgen = RGB_GEN_CONST;
		pass->rgbgen_func.type = SHADER_FUNC_CONSTANT;

		Shader_ParseVector (shader, ptr, pass->rgbgen_func.args);
	}
	else if (!Q_stricmp (token, "srgb") || !Q_stricmp (token, "srgbconst"))
	{
		pass->rgbgen = RGB_GEN_CONST;
		pass->rgbgen_func.type = SHADER_FUNC_CONSTANT;

		Shader_ParseVector (shader, ptr, pass->rgbgen_func.args);

		pass->rgbgen_func.args[0] = SRGBf(pass->rgbgen_func.args[0]);
		pass->rgbgen_func.args[1] = SRGBf(pass->rgbgen_func.args[1]);
		pass->rgbgen_func.args[2] = SRGBf(pass->rgbgen_func.args[2]);
	}
	else if (!Q_stricmp (token, "topcolor"))
		pass->rgbgen = RGB_GEN_TOPCOLOR;
	else if (!Q_stricmp (token, "bottomcolor"))
		pass->rgbgen = RGB_GEN_BOTTOMCOLOR;
}

static void Shaderpass_AlphaGen (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shaderpass_t *pass = ps->pass;
	char		*token;

	token = Shader_ParseString(ptr);
	if (!Q_stricmp (token, "portal"))
	{
		pass->alphagen = ALPHA_GEN_PORTAL;
		shader->portaldist = Shader_ParseFloat(shader, ptr, 256);
		if (!shader->portaldist)
			shader->portaldist = 256;
		shader->flags |= SHADER_AGEN_PORTAL;
	}
	else if (!Q_stricmp (token, "vertex"))
	{
		pass->alphagen = ALPHA_GEN_VERTEX;
	}
	else if (!Q_stricmp (token, "entity"))
	{
		pass->alphagen = ALPHA_GEN_ENTITY;
	}
	else if (!Q_stricmp (token, "wave"))
	{
		pass->alphagen = ALPHA_GEN_WAVE;

		Shader_ParseFunc (ps, "alphaGen wave", ptr, &pass->alphagen_func);
	}
	else if ( !Q_stricmp (token, "lightingspecular"))
	{
		pass->alphagen = ALPHA_GEN_SPECULAR;
	}
	else if ( !Q_stricmp (token, "const") || !Q_stricmp (token, "constant"))
	{
		pass->alphagen = ALPHA_GEN_CONST;
		pass->alphagen_func.type = SHADER_FUNC_CONSTANT;
		pass->alphagen_func.args[0] = fabs(Shader_ParseFloat(shader, ptr, 0));
	}
}
static void Shaderpass_AlphaShift (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shaderpass_t *pass = ps->pass;
	float speed;
	float min, max;
	pass->alphagen = ALPHA_GEN_WAVE;

	pass->alphagen_func.type = SHADER_FUNC_SIN;


	//arg0 = add
	//arg1 = scale
	//arg2 = timeshift
	//arg3 = timescale

	speed = Shader_ParseFloat(shader, ptr, 0);
	min = Shader_ParseFloat(shader, ptr, 0);
	max = Shader_ParseFloat(shader, ptr, 0);

	pass->alphagen_func.args[0] = min + (max - min)/2;
	pass->alphagen_func.args[1] = (max - min)/2;
	pass->alphagen_func.args[2] = 0;
	pass->alphagen_func.args[3] = 1/speed;
}

static int Shader_BlendFactor(char *name, qboolean dstnotsrc)
{
	int factor;
	if (!strnicmp(name, "gl_", 3))
		name += 3;

	if (!Q_stricmp(name, "zero"))
		factor = SBITS_SRCBLEND_ZERO;
	else if ( !Q_stricmp(name, "one"))
		factor = SBITS_SRCBLEND_ONE;
	else if (!Q_stricmp(name, "dst_color"))
		factor = SBITS_SRCBLEND_DST_COLOR;
	else if (!Q_stricmp(name, "one_minus_src_alpha"))
		factor = SBITS_SRCBLEND_ONE_MINUS_SRC_ALPHA;
	else if (!Q_stricmp(name, "src_alpha"))
		factor = SBITS_SRCBLEND_SRC_ALPHA;
	else if (!Q_stricmp(name, "src_color"))
		factor = SBITS_SRCBLEND_SRC_COLOR_INVALID;
	else if (!Q_stricmp(name, "one_minus_dst_color"))
		factor = SBITS_SRCBLEND_ONE_MINUS_DST_COLOR;
	else if (!Q_stricmp(name, "one_minus_src_color"))
		factor = SBITS_SRCBLEND_ONE_MINUS_SRC_COLOR_INVALID;
	else if (!Q_stricmp(name, "dst_alpha") )
		factor = SBITS_SRCBLEND_DST_ALPHA;
	else if (!Q_stricmp(name, "one_minus_dst_alpha"))
		factor = SBITS_SRCBLEND_ONE_MINUS_DST_ALPHA;
	else
		factor = SBITS_SRCBLEND_NONE;

	if (dstnotsrc)
	{
		//dest factors are shifted
		factor <<= 4;

		/*gl doesn't accept dst_color for destinations*/
		if (factor == SBITS_DSTBLEND_NONE ||
			factor == SBITS_DSTBLEND_DST_COLOR_INVALID ||
			factor == SBITS_DSTBLEND_ONE_MINUS_DST_COLOR_INVALID ||
			factor == SBITS_DSTBLEND_ALPHA_SATURATE_INVALID)
		{
			Con_DPrintf("Invalid shader dst blend \"%s\"\n", name);
			factor = SBITS_DSTBLEND_ONE;
		}
	}
	else
	{
		/*gl doesn't accept src_color for sources*/
		if (factor == SBITS_SRCBLEND_NONE ||
			factor == SBITS_SRCBLEND_SRC_COLOR_INVALID ||
			factor == SBITS_SRCBLEND_ONE_MINUS_SRC_COLOR_INVALID)
		{
			Con_DPrintf("Unrecognised shader src blend \"%s\"\n", name);
			factor = SBITS_SRCBLEND_ONE;
		}
	}

	return factor;
}

static void Shaderpass_BlendFunc (parsestate_t *ps, char **ptr)
{
	shaderpass_t *pass = ps->pass;
	char		*token;

	//reset to defaults
	pass->shaderbits &= ~(SBITS_BLEND_BITS);
	pass->stagetype = ST_AMBIENT;

	token = Shader_ParseString (ptr);
	if ( !Q_stricmp (token, "bumpmap"))				//doom3 is awkward...
		pass->stagetype = ST_BUMPMAP;
	else if ( !Q_stricmp (token, "specularmap"))	//doom3 is awkward...
		pass->stagetype = ST_SPECULARMAP;
	else if ( !Q_stricmp (token, "diffusemap"))		//doom3 is awkward...
		pass->stagetype = ST_DIFFUSEMAP;
	else if ( !Q_stricmp (token, "blend"))
		pass->shaderbits |= SBITS_SRCBLEND_SRC_ALPHA | SBITS_DSTBLEND_ONE_MINUS_SRC_ALPHA;
	else if ( !Q_stricmp (token, "premul"))			//gets rid of feathering.
		pass->shaderbits |= SBITS_SRCBLEND_ONE | SBITS_DSTBLEND_ONE_MINUS_SRC_ALPHA;
	else if (!Q_stricmp (token, "filter"))
		pass->shaderbits |= SBITS_SRCBLEND_DST_COLOR | SBITS_DSTBLEND_ZERO;
	else if (!Q_stricmp (token, "add"))
		pass->shaderbits |= SBITS_SRCBLEND_ONE | SBITS_DSTBLEND_ONE;
	else if (!Q_stricmp (token, "replace"))
		pass->shaderbits |= SBITS_SRCBLEND_NONE | SBITS_DSTBLEND_NONE;
	else
	{
		pass->shaderbits |= Shader_BlendFactor(token, false);

		token = Shader_ParseString (ptr);
		if (*token == ',')
			token = Shader_ParseString (ptr);
		pass->shaderbits |= Shader_BlendFactor(token, true);
	}
}

static void Shaderpass_AlphaFunc (parsestate_t *ps, char **ptr)
{
	shaderpass_t *pass = ps->pass;
	char *token;

	pass->shaderbits &= ~SBITS_ATEST_BITS;

	token = Shader_ParseString (ptr);
	if (!Q_stricmp (token, "gt0"))
	{
		pass->shaderbits |= SBITS_ATEST_GT0;
	}
	else if (!Q_stricmp (token, "lt128"))
	{
		pass->shaderbits |= SBITS_ATEST_LT128;
	}
	else if (!Q_stricmp (token, "ge128"))
	{
		pass->shaderbits |= SBITS_ATEST_GE128;
	}
}

static void Shaderpass_DepthFunc (parsestate_t *ps, char **ptr)
{
	shaderpass_t *pass = ps->pass;
	char *token;

	pass->shaderbits &= ~(SBITS_DEPTHFUNC_BITS);

	token = Shader_ParseString (ptr);
	if (!Q_stricmp (token, "equal"))
		pass->shaderbits |= SBITS_DEPTHFUNC_EQUAL;
	else if (!Q_stricmp (token, "lequal"))
		pass->shaderbits |= SBITS_DEPTHFUNC_CLOSEREQUAL;	//default
	else if (!Q_stricmp (token, "less"))
		pass->shaderbits |= SBITS_DEPTHFUNC_CLOSER;
	else if (!Q_stricmp (token, "greater"))
		pass->shaderbits |= SBITS_DEPTHFUNC_FURTHER;
//	else if (!Q_stricmp (token, "gequal"))
//		pass->shaderbits |= SBITS_DEPTHFUNC_FURTHEREQUAL;
//	else if (!Q_stricmp (token, "nequal"))
//		pass->shaderbits |= SBITS_DEPTHFUNC_NOTEQUAL;
//	else if (!Q_stricmp (token, "never"))
//		pass->shaderbits |= SBITS_DEPTHFUNC_NEVER;
//	else if (!Q_stricmp (token, "always"))
//		pass->shaderbits |= SBITS_DEPTHFUNC_ALWAYS;
	else
		Con_DPrintf("Invalid depth func %s\n", token);
}

static void Shaderpass_DepthWrite (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shaderpass_t *pass = ps->pass;
	shader->flags |= SHADER_DEPTHWRITE;
	pass->shaderbits |= SBITS_MISC_DEPTHWRITE;
}

static void Shaderpass_NoDepthTest (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shaderpass_t *pass = ps->pass;
	shader->flags |= SHADER_DEPTHWRITE;
	pass->shaderbits |= SBITS_MISC_NODEPTHTEST;
}

static void Shaderpass_NoDepth (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shader->flags |= SHADER_DEPTHWRITE;
}

static void Shaderpass_TcMod (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shaderpass_t *pass = ps->pass;
	int i;
	tcmod_t *tcmod;
	char *token;

	if (pass->numtcmods >= SHADER_MAX_TC_MODS)
	{
		return;
	}

	tcmod = &pass->tcmods[pass->numtcmods];

	token = Shader_ParseString (ptr);
	if (!Q_stricmp (token, "rotate"))
	{
		tcmod->args[0] = -Shader_ParseFloat(shader, ptr, 0) / 360.0f;
		if (!tcmod->args[0])
		{
			return;
		}

		tcmod->type = SHADER_TCMOD_ROTATE;
	}
	else if ( !Q_stricmp (token, "scale") )
	{
		tcmod->args[0] = Shader_ParseFloat (shader, ptr, 0);
		tcmod->args[1] = Shader_ParseFloat (shader, ptr, 0);
		tcmod->type = SHADER_TCMOD_SCALE;
	}
	else if ( !Q_stricmp (token, "scroll") )
	{
		tcmod->args[0] = Shader_ParseFloat (shader, ptr, 0);
		tcmod->args[1] = Shader_ParseFloat (shader, ptr, 0);
		tcmod->type = SHADER_TCMOD_SCROLL;
	}
	else if (!Q_stricmp(token, "stretch"))
	{
		shaderfunc_t func;

		Shader_ParseFunc(ps, "tcmod stretch", ptr, &func);

		tcmod->args[0] = func.type;
		for (i = 1; i < 5; ++i)
			tcmod->args[i] = func.args[i-1];
		tcmod->type = SHADER_TCMOD_STRETCH;
	}
	else if (!Q_stricmp (token, "transform"))
	{
		for (i = 0; i < 6; ++i)
			tcmod->args[i] = Shader_ParseFloat (shader, ptr, 0);
		tcmod->type = SHADER_TCMOD_TRANSFORM;
	}
	else if (!Q_stricmp (token, "turb"))
	{
		for (i = 0; i < 4; i++)
			tcmod->args[i] = Shader_ParseFloat (shader, ptr, 0);
		tcmod->type = SHADER_TCMOD_TURB;
	}
	else if (!Q_stricmp (token, "page"))
	{
		for (i = 0; i < 3; i++)
			tcmod->args[i] = Shader_ParseFloat (shader, ptr, 0);
		tcmod->type = SHADER_TCMOD_PAGE;
	}
//	else if (!Q_stricmp (token, "entityTranslate"))	//RTCW
//	else if (!Q_stricmp (token, "swap"))			//RTCW
	else
	{
		Con_DPrintf("Unknown tcmod %s in %s\n", token, shader->name);
		return;
	}

	pass->numtcmods++;
}

static void Shaderpass_Scale (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shaderpass_t *pass = ps->pass;
	//seperate x and y
	char *token;
	tcmod_t *tcmod;

	tcmod = &pass->tcmods[pass->numtcmods];

	tcmod->type = SHADER_TCMOD_SCALE;

	token = Shader_ParseString (ptr);
	if (!strcmp(token, "static"))
	{
		tcmod->args[0] = Shader_ParseFloat (shader, ptr, 0);
	}
	else
	{
		tcmod->args[0] = atof(token);
	}

	while (**ptr == ' ' || **ptr == '\t')
		*ptr+=1;
	if (**ptr == ',')
		*ptr+=1;

	token = Shader_ParseString (ptr);
	if (!strcmp(token, "static"))
	{
		tcmod->args[1] = Shader_ParseFloat (shader, ptr, 0);
	}
	else
	{
		tcmod->args[1] = atof(token);
	}

	pass->numtcmods++;
}

static void Shaderpass_Scroll (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shaderpass_t *pass = ps->pass;
	//seperate x and y
	char *token;
	tcmod_t *tcmod;

	tcmod = &pass->tcmods[pass->numtcmods];

	token = Shader_ParseString ( ptr );
	if (!strcmp(token, "static"))
	{
		tcmod->type = SHADER_TCMOD_SCROLL;
		tcmod->args[0] = Shader_ParseFloat (shader, ptr, 0);
	}
	else
	{
		Con_Printf("Bad shader scale\n");
		return;
	}

	token = Shader_ParseString ( ptr );
	if (!strcmp(token, "static"))
	{
		tcmod->type = SHADER_TCMOD_SCROLL;
		tcmod->args[1] = Shader_ParseFloat (shader, ptr, 0);
	}
	else
	{
		Con_Printf("Bad shader scale\n");
		return;
	}

	pass->numtcmods++;
}


static void Shaderpass_TcGen (parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shaderpass_t *pass = ps->pass;
	char *token;

	token = Shader_ParseString ( ptr );
	if ( !Q_stricmp (token, "base") ) {
		pass->tcgen = TC_GEN_BASE;
	} else if ( !Q_stricmp (token, "lightmap") ) {
		pass->tcgen = TC_GEN_LIGHTMAP;
	} else if ( !Q_stricmp (token, "environment") ) {
		pass->tcgen = TC_GEN_ENVIRONMENT;
	} else if ( !Q_stricmp (token, "fireriseenv") ) {	//from RTCW
		pass->tcgen = TC_GEN_ENVIRONMENT;	//FIXME: not supported
	} else if ( !Q_stricmp (token, "vector") )
	{
		pass->tcgen = TC_GEN_VECTOR;
		Shader_ParseVector (shader, ptr, pass->tcgenvec[0]);
		Shader_ParseVector (shader, ptr, pass->tcgenvec[1]);
	} else if ( !Q_stricmp (token, "normal") ) {
		pass->tcgen = TC_GEN_NORMAL;
	} else if ( !Q_stricmp (token, "svector") ) {
		pass->tcgen = TC_GEN_SVECTOR;
	} else if ( !Q_stricmp (token, "tvector") ) {
		pass->tcgen = TC_GEN_TVECTOR;
	} else if ( !Q_stricmp (token, "skybox") ) {
		pass->tcgen = TC_GEN_SKYBOX;
	}
}
static void Shaderpass_EnvMap (parsestate_t *ps, char **ptr)
{
	shaderpass_t *pass = ps->pass;
	pass->tcgen = TC_GEN_ENVIRONMENT;
}

static void Shaderpass_Detail (parsestate_t *ps, char **ptr)
{
	shaderpass_t *pass = ps->pass;
	pass->flags |= SHADER_PASS_DETAIL;
}

static void Shaderpass_AlphaMask (parsestate_t *ps, char **ptr)
{
	shaderpass_t *pass = ps->pass;
	pass->shaderbits &= ~SBITS_ATEST_BITS;
	pass->shaderbits |= SBITS_ATEST_GE128;
}

static void Shaderpass_NoLightMap (parsestate_t *ps, char **ptr)
{
	shaderpass_t *pass = ps->pass;
	pass->rgbgen = RGB_GEN_IDENTITY;
}

static void Shaderpass_Red(parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shaderpass_t *pass = ps->pass;
	pass->rgbgen = RGB_GEN_CONST;
	pass->rgbgen_func.args[0] = Shader_ParseFloat(shader, ptr, 0);
}
static void Shaderpass_Green(parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shaderpass_t *pass = ps->pass;
	pass->rgbgen = RGB_GEN_CONST;
	pass->rgbgen_func.args[1] = Shader_ParseFloat(shader, ptr, 0);
}
static void Shaderpass_Blue(parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shaderpass_t *pass = ps->pass;
	pass->rgbgen = RGB_GEN_CONST;
	pass->rgbgen_func.args[2] = Shader_ParseFloat(shader, ptr, 0);
}
static void Shaderpass_Alpha(parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shaderpass_t *pass = ps->pass;
	pass->alphagen = ALPHA_GEN_CONST;
	pass->alphagen_func.args[0] = Shader_ParseFloat(shader, ptr, 0);
}
static void Shaderpass_MaskColor(parsestate_t *ps, char **ptr)
{
	shaderpass_t *pass = ps->pass;
	pass->shaderbits |= SBITS_MASK_RED|SBITS_MASK_GREEN|SBITS_MASK_BLUE;
}
static void Shaderpass_MaskRed(parsestate_t *ps, char **ptr)
{
	shaderpass_t *pass = ps->pass;
	pass->shaderbits |= SBITS_MASK_RED;
}
static void Shaderpass_MaskGreen(parsestate_t *ps, char **ptr)
{
	shaderpass_t *pass = ps->pass;
	pass->shaderbits |= SBITS_MASK_GREEN;
}
static void Shaderpass_MaskBlue(parsestate_t *ps, char **ptr)
{
	shaderpass_t *pass = ps->pass;
	pass->shaderbits |= SBITS_MASK_BLUE;
}
static void Shaderpass_MaskAlpha(parsestate_t *ps, char **ptr)
{
	shaderpass_t *pass = ps->pass;
	pass->shaderbits |= SBITS_MASK_ALPHA;
}
static void Shaderpass_AlphaTest(parsestate_t *ps, char **ptr)
{
	shader_t *shader = ps->s;
	shaderpass_t *pass = ps->pass;
	if (Shader_ParseFloat(shader, ptr, 0) == 0.5)
		pass->shaderbits |= SBITS_ATEST_GE128;
	else
		Con_Printf("unsupported alphatest value\n");
}
static void Shaderpass_TexGen(parsestate_t *ps, char **ptr)
{
	shaderpass_t *pass = ps->pass;
	char *token = Shader_ParseString(ptr);
	if (!strcmp(token, "normal"))
		pass->tcgen = TC_GEN_NORMAL;
	else if (!strcmp(token, "skybox"))
		pass->tcgen = TC_GEN_SKYBOX;
	else if (!strcmp(token, "wobblesky"))
	{
		pass->tcgen = TC_GEN_WOBBLESKY;
		token = Shader_ParseString(ptr);
		token = Shader_ParseString(ptr);
		token = Shader_ParseString(ptr);
	}
	else if (!strcmp(token, "reflect"))
		pass->tcgen = TC_GEN_REFLECT;
	else
	{
		Con_Printf("texgen token not understood\n");
	}
}
static void Shaderpass_CubeMap(parsestate_t *ps, char **ptr)
{
	shaderpass_t *pass = ps->pass;
	char *token = Shader_ParseString(ptr);

	if (pass->tcgen == TC_GEN_UNSPECIFIED)
		pass->tcgen = TC_GEN_SKYBOX;
	pass->texgen = T_GEN_CUBEMAP;
	pass->anim_frames[0] = Shader_FindImage(ps, token, IF_TEXTYPE_CUBE);

	if (!TEXVALID(pass->anim_frames[0]))
	{
		pass->texgen = T_GEN_SINGLEMAP;
		pass->anim_frames[0] = missing_texture;
	}
}

static shaderkey_t shaderpasskeys[] =
{
#define Q3 NULL
	{"rgbgen",		Shaderpass_RGBGen,			Q3},
	{"alphagen",	Shaderpass_AlphaGen,		Q3},
	{"blendfunc",	Shaderpass_BlendFunc,		Q3},
	{"depthfunc",	Shaderpass_DepthFunc,		Q3},
	{"depthwrite",	Shaderpass_DepthWrite,		Q3},
	{"alphafunc",	Shaderpass_AlphaFunc,		Q3},
	{"tcmod",		Shaderpass_TcMod,			Q3},
	{"map",			Shaderpass_Map,				Q3},
	{"animmap",		Shaderpass_AnimMap,			Q3},
	{"clampmap",	Shaderpass_ClampMap,		Q3},
	{"videomap",	Shaderpass_VideoMap,		Q3},
	{"tcgen",		Shaderpass_TcGen,			Q3},
	{"texgen",		Shaderpass_TcGen,			Q3},
	{"detail",		Shaderpass_Detail,			Q3},

	{"nodepthtest",	Shaderpass_NoDepthTest,		NULL},
	{"nodepth",		Shaderpass_NoDepth,			NULL},

	{"envmap",		Shaderpass_EnvMap,			"rscript"},//for alienarena
	{"nolightmap",	Shaderpass_NoLightMap,		"rscript"},//for alienarena
	{"scale",		Shaderpass_Scale,			"rscript"},//for alienarena
	{"scroll",		Shaderpass_Scroll,			"rscript"},//for alienarena
	{"alphashift",	Shaderpass_AlphaShift,		"rscript"},//for alienarena
	{"alphamask",	Shaderpass_AlphaMask,		"rscript"},//for alienarena

	{"program",		Shaderpass_ProgramName,		"fte"},
	
	/*doom3 compat*/
	{"blend",		Shaderpass_BlendFunc,		"doom3"},
	{"maskcolor",	Shaderpass_MaskColor,		"doom3"},
	{"maskred",		Shaderpass_MaskRed,			"doom3"},
	{"maskgreen",	Shaderpass_MaskGreen,		"doom3"},
	{"maskblue",	Shaderpass_MaskBlue,		"doom3"},
	{"maskalpha",	Shaderpass_MaskAlpha,		"doom3"},
	{"alphatest",	Shaderpass_AlphaTest,		"doom3"},
	{"texgen",		Shaderpass_TexGen,			"doom3"},
	{"cubemap",		Shaderpass_CubeMap,			"doom3"},	//one of these is wrong
	{"cameracubemap",Shaderpass_CubeMap,		"doom3"},	//one of these is wrong
	{"red",			Shaderpass_Red,				"doom3"},
	{"green",		Shaderpass_Green,			"doom3"},
	{"blue",		Shaderpass_Blue,			"doom3"},
	{"alpha",		Shaderpass_Alpha,			"doom3"},


	//RTCW
	//fancy map lines use the map if that mode is active.
	//FIXME: actually check these to ensure there's no issues with any shaders overriding the pass's previously specified map
	//      (hopefully no shaders would actually do that due to the engine loading both textures, which would be wasteful)
	{"map16",		Shaderpass_RTCW_Map_16bit,		"rtcw"},
	{"map32",		Shaderpass_RTCW_Map_32bit,		"rtcw"},
	{"mapcomp",		Shaderpass_RTCW_Map_s3tc,		"rtcw"},
	{"mapnocomp",	Shaderpass_RTCW_Map_nos3tc,		"rtcw"},
	{"animcompmap",	Shaderpass_RTCW_AnimMap_s3tc,	"rtcw"},
	{"animnocompmap",Shaderpass_RTCW_AnimMap_nos3tc,"rtcw"},

	//qfusion/warsow compat
	{"material",	Shaderpass_QF_Material,		"qf"},
	{"animclampmap",Shaderpass_QF_AnimClampMap,	"qf"},
//	{"cubemap",		Shaderpass_QF_CubeMap,		"qf"},
//	{"shadecubemap",Shaderpass_QF_ShadeCubeMap,	"qf"},
	{"surroundmap",	Shaderpass_CubeMap,			"qf"},
//	{"distortion",	Shaderpass_QF_Distortion,	"qf"},
//	{"celshade",	Shaderpass_QF_Celshade,		"qf"},
//	{"grayscale",	Shaderpass_QF_Greyscale,	"qf"},
//	{"greyscale",	Shaderpass_QF_Greyscale,	"qf"},
//	{"skip",		Shaderpass_QF_Skip,			"qf"},

	{NULL,			NULL}
};

// ===============================================================


void Shader_FreePass (shaderpass_t *pass)
{
#ifdef HAVE_MEDIA_DECODER
	if ( pass->flags & SHADER_PASS_VIDEOMAP )
	{
		Media_ShutdownCin(pass->cin);
		pass->cin = NULL;
	}
#endif

	if (pass->prog)
	{
		Shader_ReleaseGeneric(pass->prog);
		pass->prog = NULL;
	}
}

void Shader_ReleaseGeneric(program_t *prog)
{
	if (prog)
		if (prog->refs-- == 1)
			Shader_UnloadProg(prog);
}
void Shader_Free (shader_t *shader)
{
	int i;
	shaderpass_t *pass;

	if (shader->bucket.data == shader)
		Hash_RemoveData(&shader_active_hash, shader->name, shader);
	shader->bucket.data = NULL;

	Shader_ReleaseGeneric(shader->prog);
	shader->prog = NULL;

	if (shader->skydome)
		Z_Free (shader->skydome);
	shader->skydome = NULL;
	while (shader->clutter)
	{
		void *t = shader->clutter;
		shader->clutter = shader->clutter->next;
		Z_Free(t);
	}

	pass = shader->passes;
	for (i = 0; i < shader->numpasses; i++, pass++)
	{
		Shader_FreePass (pass);
	}
	shader->numpasses = 0;

	if (shader->genargs)
	{
		free(shader->genargs);
		shader->genargs = NULL;
	}
	shader->uses = 0;

	Z_Free(shader->defaulttextures);
	shader->defaulttextures = NULL;
}





int QDECL Shader_InitCallback (const char *name, qofs_t size, time_t mtime, void *param, searchpathfuncs_t *spath)
{
	Shader_MakeCache(name, SPF_DEFAULT);
	return true;
}
int QDECL Shader_InitCallback_Doom3 (const char *name, qofs_t size, time_t mtime, void *param, searchpathfuncs_t *spath)
{
	Shader_MakeCache(name, SPF_DOOM3);
	return true;
}

qboolean Shader_Init (void)
{
	int wibuf[16];

	if (!r_shaders)
	{
		r_numshaders = 0;
		r_maxshaders = 256;
		r_shaders = calloc(r_maxshaders, sizeof(*r_shaders));

		shader_hash = calloc (HASH_SIZE, sizeof(*shader_hash));

		shader_active_hash_mem = malloc(Hash_BytesForBuckets(1024));
		memset(shader_active_hash_mem, 0, Hash_BytesForBuckets(1024));
		Hash_InitTable(&shader_active_hash, 1024, shader_active_hash_mem);

		Shader_FlushGenerics();

		if (!sh_config.progs_supported)
			sh_config.max_gpu_bones = 0;
		else
		{
			extern cvar_t r_max_gpu_bones;
			if (!*r_max_gpu_bones.string)
			{
#ifdef FTE_TARGET_WEB
				sh_config.max_gpu_bones = 0;	//webgl tends to crap out if this is too high, so 32 is a good enough value to play safe. some browsers have really shitty uniform performance too, so lets just default to pure-cpu transforms. in javascript. yes, its that bad.
#else
				sh_config.max_gpu_bones = 64;	//ATI drivers bug out and start to crash if you put this at 128.
#endif
			}
			else
				sh_config.max_gpu_bones = bound(0, r_max_gpu_bones.ival, MAX_BONES);
		}
	}
	
	if (!qrenderer)
		r_whiteimage = r_nulltex;
	else
	{
		memset(wibuf, 0xff, sizeof(wibuf));
		r_whiteimage = R_LoadTexture("$whiteimage", 4, 4, TF_RGBA32, wibuf, IF_NOMIPMAP|IF_NOPICMIP|IF_NEAREST|IF_NOGAMMA);
		memset(wibuf, 0, sizeof(wibuf));
		r_blackimage = R_LoadTexture("$blackimage", 4, 4, TF_RGBA32, wibuf, IF_NOMIPMAP|IF_NOPICMIP|IF_NEAREST|IF_NOGAMMA);
	}

	Shader_NeedReload(true);
	Shader_DoReload();
	return true;
}

void Shader_FlushCache(void)
{
	shadercachefile_t *sf;
	shadercache_t *cache, *cache_next;
	int i;

	for (i = 0; i < HASH_SIZE; i++)
	{
		cache = shader_hash[i];
		shader_hash[i] = NULL;

		for (; cache; cache = cache_next)
		{
			cache_next = cache->hash_next;
			cache->hash_next = NULL;
			Z_Free(cache);
		}
	}

	while(shaderfiles)
	{
		sf = shaderfiles;
		shaderfiles = sf->next;
		if (sf->data)
			FS_FreeFile(sf->data);
		Z_Free(sf);
	}
}

static void Shader_MakeCache(const char *path, unsigned int parseflags)
{
	unsigned int key;
	char *buf, *ptr, *token;
	shadercache_t *cache;
	shadercachefile_t *cachefile, *filelink = NULL;
	qofs_t size;

	for (cachefile = shaderfiles; cachefile; cachefile = cachefile->next)
	{
		if (!Q_stricmp(cachefile->name, path))
			return;	//already loaded. there's no source package or anything.
		filelink = cachefile;
	}

	
	Con_DPrintf ("...loading '%s'\n", path);

	cachefile = Z_Malloc(sizeof(*cachefile) + strlen(path));
	strcpy(cachefile->name, path);
	size = FS_LoadFile(path, (void **)&cachefile->data);
	cachefile->length = size;
	cachefile->parseflags = parseflags;
	if (filelink)
		filelink->next = cachefile;
	else
		shaderfiles = cachefile;

	if (qofs_Error(size))
	{
		Con_Printf("Unable to read %s\n", path);
		cachefile->length = 0;
		return;
	}
	if (size > 1024*1024*64)	//sanity limit
	{
		Con_Printf("Refusing to parse %s due to size\n", path);
		cachefile->length = 0;
		FS_FreeFile(cachefile->data);
		cachefile->data = NULL;
		return;
	}

	ptr = buf = cachefile->data;
	size = cachefile->length;

	//look for meta comments.
	while (1)
	{
		//parse metas
		while (*ptr == ' ' || *ptr == '\t')
			ptr++;
		if (ptr[0] == '\r' && ptr[1] == '\n')
			ptr+=2;	//blank line with dos ending
		else if (ptr[0] == '\r' || ptr[0] == '\n')
			ptr+=1;	//blank line with mac or unix ending
		else if (ptr[0] == '/' && ptr[1] == '/')
		{
			char *e = strchr(ptr, '\n');
			if (e)
				e++;
			else
				e = ptr + strlen(ptr);

			ptr += 2;
			while (*ptr == ' ' || *ptr == '\t')
				ptr++;
			if (!strncmp(ptr, "meta:", 5))
			{
				ptr+=5;

				token = COM_ParseExt (&ptr, false, true);
				if (!strcmp(token, "forceprogramify"))
				{
					cachefile->parseflags |= SPF_PROGRAMIFY;
					token = COM_ParseExt (&ptr, false, true);
					if (*token)
						Q_strncpyz(cachefile->forcedshadername, token, sizeof(cachefile->forcedshadername));
				}
				else
					Con_DPrintf("unknown shader meta term \"%s\" in %s\n", token, path);

				while (*ptr == ' ' || *ptr == '\t')
					ptr++;
				if (*ptr != '\r' && *ptr != '\n')
				{
					while (*ptr && (*ptr != '\r' && *ptr != '\n'))
						ptr++;
					Con_DPrintf("junk after shader meta in %s\n", path);
				}
			}
			ptr = e;
		}
		else
			break;	//the actual shader started.
	}

	//now scan the file looking for each individual shader.
	do
	{
		if ( ptr - buf >= size )
			break;

		token = COM_ParseExt (&ptr, true, true);
		if ( !token[0] || ptr - buf >= size )
			break;

		COM_CleanUpPath(token);

		if (Shader_LocateSource(token, NULL, NULL, NULL, NULL))
		{
			ptr = Shader_Skip (path, token, ptr);
			continue;
		}

		key = Hash_Key ( token, HASH_SIZE );

		cache = ( shadercache_t * )Z_Malloc(sizeof(shadercache_t) + strlen(token));
		strcpy(cache->name, token);
		cache->hash_next = shader_hash[key];
		cache->source = cachefile;
		cache->offset = ptr - cachefile->data;

		shader_hash[key] = cache;

		ptr = Shader_Skip (path, cache->name, ptr);
	} while ( ptr );
}

static qboolean Shader_LocateSource(char *name, char **buf, size_t *bufsize, size_t *offset, shadercachefile_t **sourcefile)
{
	unsigned int key;
	shadercache_t *cache;

	key = Hash_Key ( name, HASH_SIZE );
	cache = shader_hash[key];

	for ( ; cache; cache = cache->hash_next )
	{
		if ( !Q_stricmp (cache->name, name) )
		{
			if (buf)
			{
				*buf = cache->source->data;
				*bufsize = cache->source->length;
				*offset = cache->offset;
				*sourcefile = cache->source;
			}
			return true;
		}
	}
	return false;
}

static char *Shader_Skip(const char *file, const char *shadername, char *ptr)
{
	char *tok;
	int brace_count;

    // Opening brace
	tok = COM_ParseExt(&ptr, true, true);

	if (!ptr)
		return NULL;

	if ( tok[0] != '{' )
	{
		tok = COM_ParseExt (&ptr, true, true);
	}

	for (brace_count = 1; brace_count > 0; )
	{
		tok = COM_ParseExt (&ptr, true, true);

		if ( !tok[0] )
		{
			Con_Printf("%s: unexpected EOF parsing %s\n", file, shadername);
			return NULL;
		}

		if (tok[0] == '{')
		{
			brace_count++;
		} else if (tok[0] == '}')
		{
			brace_count--;
		}
	}

	return ptr;
}

void Shader_Reset(shader_t *s)
{
	extern cvar_t r_refractreflect_scale;
	char name[MAX_QPATH];
	int id = s->id;
	int uses = s->uses;
	shader_gen_t *defaultgen = s->generator;
	char *genargs = s->genargs;
	texnums_t *dt = s->defaulttextures;
	int dtcount = s->numdefaulttextures;
	float dtrate = s->defaulttextures_fps;	//FIXME!
	int w = s->width;
	int h = s->height;
	unsigned int uf = s->usageflags;
	Q_strncpyz(name, s->name, sizeof(name));
	s->genargs = NULL;
	s->defaulttextures = NULL;
	Shader_Free(s);
	memset(s, 0, sizeof(*s));
	s->portalfboscale = r_refractreflect_scale.value;	//unless otherwise specified, this cvar specifies the value.

	s->remapto = s;
	s->id = id;
	s->width = w;
	s->height = h;
	s->defaulttextures = dt;
	s->numdefaulttextures = dtcount;
	s->defaulttextures_fps = dtrate;
	s->generator = defaultgen;
	s->genargs = genargs;
	s->usageflags = uf;
	s->uses = uses;
	Q_strncpyz(s->name, name, sizeof(s->name));
	Hash_Add(&shader_active_hash, s->name, s, &s->bucket);
}

static void Shader_Regenerate(parsestate_t *ps, const char *shortname)
{
	Shader_Reset(ps->s);

	if (!strcmp(shortname, "textures/common/clip") || !strcmp(shortname, "textures/common/nodraw") || !strcmp(shortname, "common/nodraw"))
		Shader_DefaultScript(ps, shortname,
			"{\n"
				"surfaceparm nodraw\n"
				"surfaceparm nodlight\n"
			"}\n");
	else
		ps->s->generator(ps, shortname, ps->s->genargs);
}

void Shader_Shutdown (void)
{
	int i;
	shader_t *shader;

	if (!r_shaders)
		return;	/*nothing needs freeing yet*/
	for (i = 0; i < r_numshaders; i++)
	{
		shader = r_shaders[i];
		if (!shader)
			continue;

		Shader_Free(shader);
		Z_Free(r_shaders[i]);
		r_shaders[i] = NULL;
	}

	Shader_FlushCache();
	Shader_FlushGenerics();

	R_SkyShutdown();

	r_maxshaders = 0;
	r_numshaders = 0;

	free(r_shaders);
	r_shaders = NULL;
	free(shader_hash);
	shader_hash = NULL;
	free(shader_active_hash_mem);
	shader_active_hash_mem = NULL;

	shader_reload_needed = false;
}

void Shader_SetBlendmode (shaderpass_t *pass, shaderpass_t *lastpass)
{
	qboolean lightmapoverbright;
	if (pass->texgen == T_GEN_DELUXMAP)
	{
		pass->blendmode = PBM_DOTPRODUCT;
		return;
	}

	if (pass->texgen < T_GEN_DIFFUSE && !TEXVALID(pass->anim_frames[0]) && !(pass->flags & SHADER_PASS_LIGHTMAP))
	{
		pass->blendmode = PBM_MODULATE;
		return;
	}

	if (!(pass->shaderbits & SBITS_BLEND_BITS))
	{
		if (pass->texgen == T_GEN_LIGHTMAP && lastpass)
			pass->blendmode = PBM_OVERBRIGHT;
		else if ((pass->rgbgen == RGB_GEN_IDENTITY) && (pass->alphagen == ALPHA_GEN_IDENTITY))
		{
			pass->blendmode = PBM_REPLACE;
			return;
		}
		else if ((pass->rgbgen == RGB_GEN_IDENTITY_LIGHTING) && (pass->alphagen == ALPHA_GEN_IDENTITY))
		{
			pass->shaderbits &= ~SBITS_BLEND_BITS;
			pass->shaderbits |= SBITS_SRCBLEND_ONE;
			pass->shaderbits |= SBITS_DSTBLEND_ZERO;
			pass->blendmode = PBM_REPLACELIGHT;
		}
		else
		{
			pass->shaderbits &= ~SBITS_BLEND_BITS;
			pass->shaderbits |= SBITS_SRCBLEND_ONE;
			pass->shaderbits |= SBITS_DSTBLEND_ZERO;
			pass->blendmode = PBM_MODULATE;
		}
		return;
	}

	lightmapoverbright = pass->texgen == T_GEN_LIGHTMAP || (lastpass && lastpass->texgen == T_GEN_LIGHTMAP && lastpass->blendmode != PBM_OVERBRIGHT);

	if (((pass->shaderbits&SBITS_BLEND_BITS) == (SBITS_SRCBLEND_ZERO|SBITS_DSTBLEND_SRC_COLOR)) ||
		((pass->shaderbits&SBITS_BLEND_BITS) == (SBITS_SRCBLEND_DST_COLOR|SBITS_DSTBLEND_ZERO)) ||
		((pass->shaderbits&SBITS_BLEND_BITS) == (SBITS_SRCBLEND_DST_COLOR|SBITS_DSTBLEND_ONE_MINUS_DST_ALPHA)))
		pass->blendmode = lightmapoverbright?PBM_OVERBRIGHT:PBM_MODULATE;
	else if ((pass->shaderbits&SBITS_BLEND_BITS) == (SBITS_SRCBLEND_ONE|SBITS_DSTBLEND_ONE))
		pass->blendmode = PBM_ADD;
	else if ((pass->shaderbits&SBITS_BLEND_BITS) == (SBITS_SRCBLEND_SRC_ALPHA|SBITS_DSTBLEND_ONE_MINUS_SRC_ALPHA))
		pass->blendmode = PBM_DECAL;
	else
		pass->blendmode = lightmapoverbright?PBM_OVERBRIGHT:PBM_MODULATE;
}

void Shader_FixupProgPasses(parsestate_t *ps, shaderpass_t *pass)
{
	shader_t *shader = ps->s;
	int i;
	int maxpasses = SHADER_PASS_MAX - (pass-shader->passes);
	struct
	{
		int gen;
		unsigned int flags;
	} defaulttgen[] =
	{
		//light
		{T_GEN_SHADOWMAP,		0},						//0
		{T_GEN_LIGHTCUBEMAP,	0},						//1

		//material
		{T_GEN_DIFFUSE,			SHADER_HASDIFFUSE},		//2
		{T_GEN_NORMALMAP,		SHADER_HASNORMALMAP},	//3
		{T_GEN_SPECULAR,		SHADER_HASGLOSS},		//4
		{T_GEN_UPPEROVERLAY,	SHADER_HASTOPBOTTOM},	//5
		{T_GEN_LOWEROVERLAY,	SHADER_HASTOPBOTTOM},	//6
		{T_GEN_FULLBRIGHT,		SHADER_HASFULLBRIGHT},	//7
		{T_GEN_PALETTED,		SHADER_HASPALETTED},	//8
		{T_GEN_REFLECTCUBE,		SHADER_HASREFLECTCUBE},	//9
		{T_GEN_REFLECTMASK,		0},						//10
		{T_GEN_DISPLACEMENT,	SHADER_HASDISPLACEMENT},//11
		{T_GEN_OCCLUSION,		0},						//12
//			{T_GEN_REFLECTION,		SHADER_HASREFLECT},		//
//			{T_GEN_REFRACTION,		SHADER_HASREFRACT},		//
//			{T_GEN_REFRACTIONDEPTH,	SHADER_HASREFRACTDEPTH},//
//			{T_GEN_RIPPLEMAP,		SHADER_HASRIPPLEMAP},	//

		//batch
		{T_GEN_LIGHTMAP,		SHADER_HASLIGHTMAP},	//13
		{T_GEN_DELUXMAP,		0},						//14
		//more lightmaps								//15,16,17
		//mode deluxemaps								//18,19,20
	};

#ifdef HAVE_MEDIA_DECODER
	cin_t *cin = R_ShaderGetCinematic(shader);
#endif

	//if the glsl doesn't specify all samplers, just trim them.
	pass->numMergedPasses = pass->prog->numsamplers;

#ifdef HAVE_MEDIA_DECODER
	if (cin && R_ShaderGetCinematic(shader) == cin)
		cin = NULL;
#endif

	//if the glsl has specific textures listed, be sure to provide a pass for them.
	for (i = 0; i < sizeof(defaulttgen)/sizeof(defaulttgen[0]); i++)
	{
		if (pass->prog->defaulttextures & (1u<<i))
		{
			if (pass->numMergedPasses >= maxpasses)
			{	//panic...
				ps->droppass = true;
				break;
			}
			pass[pass->numMergedPasses].flags |= SHADER_PASS_NOCOLORARRAY;
			pass[pass->numMergedPasses].flags &= ~SHADER_PASS_DEPTHCMP;
			if (defaulttgen[i].gen == T_GEN_SHADOWMAP)
				pass[pass->numMergedPasses].flags |= SHADER_PASS_DEPTHCMP;
#ifdef HAVE_MEDIA_DECODER
			if (!i && cin)
			{
				pass[pass->numMergedPasses].texgen = T_GEN_VIDEOMAP;
				pass[pass->numMergedPasses].cin = cin;
				cin = NULL;
			}
			else
#endif
			{
				pass[pass->numMergedPasses].texgen = defaulttgen[i].gen;
#ifdef HAVE_MEDIA_DECODER
				pass[pass->numMergedPasses].cin = NULL;
#endif
			}
			pass->numMergedPasses++;
			shader->flags |= defaulttgen[i].flags;
		}
	}

	//must have at least one texture.
	if (!pass->numMergedPasses)
	{
#ifdef HAVE_MEDIA_DECODER
		pass[0].texgen = cin?T_GEN_VIDEOMAP:T_GEN_DIFFUSE;
		pass[0].cin = cin;
#else
		pass[0].texgen = T_GEN_DIFFUSE;
#endif
		pass->numMergedPasses = 1;
	}
#ifdef HAVE_MEDIA_DECODER
	else if (cin)
		Media_ShutdownCin(cin);
#endif

	shader->numpasses = (pass-shader->passes)+pass->numMergedPasses;
}

struct scondinfo_s
{
	int depth;
	int level[8];
#define COND_IGNORE			1
#define COND_IGNOREPARENT	2
#define COND_ALLOWELSE		4
#define COND_TAKEN			8
};
static qboolean Shader_Conditional_Read(shader_t *shader, struct scondinfo_s *cond, char *token, char **ptr)
{
	if (!Q_stricmp(token, "if"))
	{
		if (cond->depth+1 == countof(cond->level))
		{
			Con_Printf("if statements nest too deeply in shader %s\n", shader->name);
			*ptr += strlen(*ptr);
			return true;
		}
		cond->depth++;
		cond->level[cond->depth] = (Shader_EvaluateCondition(shader, ptr)?0:COND_IGNORE);
		cond->level[cond->depth] |= COND_ALLOWELSE;
		if (cond->level[cond->depth-1] & (COND_IGNORE|COND_IGNOREPARENT))
			cond->level[cond->depth] |= COND_IGNOREPARENT;	//if ignoring the parent, ignore this one too, even if valid
		if (!(cond->level[cond->depth] & (COND_IGNORE|COND_IGNOREPARENT)))
			cond->level[cond->depth] |= COND_TAKEN;		//if we're not ignoring the contained commands then flag it so we don't take any elifs/elses
	}
	else if (!Q_stricmp(token, "elif"))
	{
		if (cond->level[cond->depth] & COND_ALLOWELSE)
		{
			if (cond->level[cond->depth] & COND_TAKEN)
			{	//if we took the if/elif then don't take this elif either
				Shader_EvaluateCondition(shader, ptr);
				cond->level[cond->depth] = COND_ALLOWELSE|COND_TAKEN|COND_IGNORE;
			}
			else
			{
				cond->level[cond->depth] = (Shader_EvaluateCondition(shader, ptr)?0:COND_IGNORE);
				cond->level[cond->depth] |= COND_ALLOWELSE;
			}
			if (cond->level[cond->depth-1] & (COND_IGNORE|COND_IGNOREPARENT))
				cond->level[cond->depth] |= COND_IGNOREPARENT;
			if (!(cond->level[cond->depth] & (COND_IGNORE|COND_IGNOREPARENT)))
				cond->level[cond->depth] |= COND_TAKEN;
		}
		else
		{
			Con_Printf("unexpected elif statement in shader %s\n", shader->name);
			*ptr += strlen(*ptr);
		}
	}
	else if (!Q_stricmp(token, "endif"))
	{
		if (!cond->depth)
		{
			Con_Printf("endif without if in shader %s\n", shader->name);
			*ptr += strlen(*ptr);
			return true;
		}
		cond->depth--;
	}
	else if (!Q_stricmp(token, "else"))
	{
		if (cond->level[cond->depth] & COND_ALLOWELSE)
		{
			if (cond->level[cond->depth] & COND_TAKEN)
				cond->level[cond->depth] |= COND_IGNORE;
			else
				cond->level[cond->depth] ^= COND_IGNORE;
			cond->level[cond->depth] &= ~COND_ALLOWELSE;
		}
		else
		{
			Con_Printf("unexpected else statement in shader %s\n", shader->name);
			*ptr += strlen(*ptr);
		}
	}
	else if (cond->level[cond->depth] & (COND_IGNORE|COND_IGNOREPARENT))
	{
		//eat it
		while (ptr)
		{
			token = COM_ParseExt(ptr, false, true);
			if ( !token[0] )
				break;
		}
	}
	else
		return false;
	return true;
}

void Shader_Readpass (parsestate_t *ps)
{
	shader_t *shader = ps->s;
	char *token;
	shaderpass_t *pass;
	static shader_t dummy;
	struct scondinfo_s cond = {0};
	unsigned int oldflags = shader->flags;

	if ( shader->numpasses >= SHADER_PASS_MAX )
	{
		ps->droppass = true;
		shader = &dummy;
		shader->numpasses = 1;
		pass = shader->passes;
	}
	else
	{
		ps->droppass = false;
		pass = &shader->passes[shader->numpasses++];
	}

	// Set defaults
	pass->flags = 0;
	pass->anim_frames[0] = r_nulltex;
	pass->anim_numframes = 0;
	pass->rgbgen = RGB_GEN_UNKNOWN;
	pass->alphagen = ALPHA_GEN_UNDEFINED;
	pass->tcgen = TC_GEN_UNSPECIFIED;
	pass->numtcmods = 0;
	pass->stagetype = ST_AMBIENT;
	pass->numMergedPasses = 0;

	if (shader->flags & SHADER_NOMIPMAPS)
		pass->flags |= SHADER_PASS_NOMIPMAP;

	ps->pass = pass;

	while ( ps->ptr )
	{
		token = COM_ParseExt (&ps->ptr, true, true);

		if ( !token[0] )
		{
			continue;
		}
		else if (!Shader_Conditional_Read(shader, &cond, token, &ps->ptr))
		{
			if ( token[0] == '}' )
				break;
			else if (token[0] == '{')
				Con_Printf("unexpected indentation in %s\n", shader->name);
			else if ( Shader_Parsetok (ps, shaderpasskeys, token) )
				break;
		}
	}

	if (pass->alphagen == ALPHA_GEN_UNDEFINED)
		pass->alphagen = ALPHA_GEN_IDENTITY;

	//if there was no texgen, then its too late now.
	if (!pass->numMergedPasses)
		pass->numMergedPasses = 1;

	if (cond.depth)
	{
		Con_Printf("if statements without endif in shader %s\n", shader->name);
	}

	if (pass->tcgen == TC_GEN_UNSPECIFIED)
		pass->tcgen = TC_GEN_BASE;

	if (!ps->droppass)
	{
		switch(pass->stagetype)
		{
		case ST_DIFFUSEMAP:
			if (pass->texgen == T_GEN_SINGLEMAP)
				shader->defaulttextures->base = pass->anim_frames[0];
			break;
		case ST_AMBIENT:
			break;
		case ST_BUMPMAP:
			if (pass->texgen == T_GEN_SINGLEMAP)
				shader->defaulttextures->bump = pass->anim_frames[0];
			ps->droppass = true;	//fixme: scrolling etc may be important. but we're not doom3.
			break;
		case ST_SPECULARMAP:
			if (pass->texgen == T_GEN_SINGLEMAP)
				shader->defaulttextures->specular = pass->anim_frames[0];
			ps->droppass = true;	//fixme: scrolling etc may be important. but we're not doom3.
			break;
		}
	}

	// check some things

	if (!ps->droppass)
		if ((pass->shaderbits&SBITS_BLEND_BITS) == (SBITS_SRCBLEND_ONE|SBITS_DSTBLEND_ZERO))
		{
			pass->shaderbits |= SBITS_MISC_DEPTHWRITE;
			shader->flags |= SHADER_DEPTHWRITE;
		}

	switch (pass->rgbgen)
	{
		case RGB_GEN_IDENTITY_LIGHTING:
		case RGB_GEN_IDENTITY:
		case RGB_GEN_CONST:
		case RGB_GEN_WAVE:
		case RGB_GEN_ENTITY:
		case RGB_GEN_ONE_MINUS_ENTITY:
		case RGB_GEN_UNKNOWN:	// assume RGB_GEN_IDENTITY or RGB_GEN_IDENTITY_LIGHTING

			switch (pass->alphagen)
			{
				case ALPHA_GEN_IDENTITY:
				case ALPHA_GEN_CONST:
				case ALPHA_GEN_WAVE:
				case ALPHA_GEN_ENTITY:
					pass->flags |= SHADER_PASS_NOCOLORARRAY;
					break;
				default:
					break;
			}

			break;
		default:
			break;
	}

	/*if ((shader->flags & SHADER_SKY) && (shader->flags & SHADER_DEPTHWRITE))
	{
#ifdef warningmsg
#pragma warningmsg("is this valid?")
#endif
		pass->shaderbits &= ~SBITS_MISC_DEPTHWRITE;
	}
	*/

	//if this pass specified a program, make sure it has all the textures that the program requires
	if (!ps->droppass && pass->prog)
		Shader_FixupProgPasses(ps, pass);

	if (ps->droppass)
	{
		while (pass->numMergedPasses > 0)
		{
			Shader_FreePass (pass+--pass->numMergedPasses);
			shader->numpasses--;
		}
		shader->flags = oldflags;
	}
	ps->pass = NULL;
}

//we've read the first token, now make sense of it and any args
static qboolean Shader_Parsetok(parsestate_t *ps, shaderkey_t *keys, char *token)
{
	shaderkey_t *key;
	char *prefix;
	qboolean toolchainprefix = false;

	if (*token == '_')
	{	//forward compat: make sure there's a way to shut stuff up if you're using future extensions in an outdated engine.
		token++;
		toolchainprefix = true;
	}

	//handle known prefixes.
	if		(!Q_strncasecmp(token, "fte", 3))		{prefix = token; token += 3; }
	else if	(!Q_strncasecmp(token, "dp", 2))		{prefix = token; token += 2; }
	else if (!Q_strncasecmp(token, "doom3", 5))		{prefix = token; token += 5; }
	else if (!Q_strncasecmp(token, "rscript", 7))	{prefix = token; token += 7; }
	else if (!Q_strncasecmp(token, "qer_", 4))		{prefix = token; token += 3; toolchainprefix = true; }
	else if (!Q_strncasecmp(token, "q3map_", 6))	{prefix = token; token += 5; toolchainprefix = true; }
	else if (!Q_strncasecmp(token, "vmap_", 6))		{prefix = token; token += 4; toolchainprefix = true; }
	else	prefix = NULL;
	if (prefix && *token == '_')
		token++;

	for (key = keys; key->keyword != NULL; key++)
	{
		if (!Q_stricmp (token, key->keyword))
		{
			if (!prefix || (prefix && key->prefix && !Q_strncasecmp(prefix, key->prefix, strlen(key->prefix))))
			{
				if (key->func)
					key->func ( ps, &ps->ptr );

				return (ps->ptr && *ps->ptr == '}' );
			}
		}
	}

	if (!toolchainprefix)	//we don't really give a damn about prefixes owned by various toolchains - they shouldn't affect us.
	{
		if (prefix)
			Con_DPrintf("Unknown shader directive parsing %s: \"%s\"\n", ps->s->name, prefix);
		else
			Con_DPrintf("Unknown shader directive parsing %s: \"%s\"\n", ps->s->name, token);
	}

	// Next Line
	while (ps->ptr)
	{
		token = COM_ParseExt(&ps->ptr, false, true);
		if ( !token[0] )
		{
			break;
		}
	}

	return false;
}

void Shader_SetPassFlush (shaderpass_t *pass, shaderpass_t *pass2)
{
	if (((pass->flags & SHADER_PASS_DETAIL) && !r_detailtextures.value) ||
		((pass2->flags & SHADER_PASS_DETAIL) && !r_detailtextures.value) ||
		 (pass->flags & SHADER_PASS_VIDEOMAP) || (pass2->flags & SHADER_PASS_VIDEOMAP))
	{
		return;
	}

	//don't merge passes if they're got their own programs.
	if (pass->prog || pass2->prog)
		return;

	/*identity alpha is required for merging*/
	if (pass->alphagen != ALPHA_GEN_IDENTITY || pass2->alphagen != ALPHA_GEN_IDENTITY)
		return;

	/*don't merge passes if the hardware cannot support it*/
	if (pass->numMergedPasses >= be_maxpasses)
		return;

	/*rgbgen must be identity too except if the later pass is identity_ligting, in which case all is well and we can switch the first pass to identity_lighting instead*/
	if (pass2->rgbgen == RGB_GEN_IDENTITY_LIGHTING && (pass2->blendmode == PBM_OVERBRIGHT || pass2->blendmode == PBM_MODULATE) && pass->rgbgen == RGB_GEN_IDENTITY)
	{
		if (pass->blendmode == PBM_REPLACE)
			pass->blendmode = PBM_REPLACELIGHT;
		pass->rgbgen = RGB_GEN_IDENTITY_LIGHTING;
		pass2->rgbgen = RGB_GEN_IDENTITY;
	}
	/*rgbgen must be identity (or the first is identity_lighting)*/
	else if (pass2->rgbgen != RGB_GEN_IDENTITY || (pass->rgbgen != RGB_GEN_IDENTITY && pass->rgbgen != RGB_GEN_IDENTITY_LIGHTING))
		return;

	/*if its alphatest, don't merge with anything other than lightmap (although equal stuff can be merged)*/
	if ((pass->shaderbits & SBITS_ATEST_BITS) && (((pass2->shaderbits & SBITS_DEPTHFUNC_BITS) != SBITS_DEPTHFUNC_EQUAL) || pass2->texgen != T_GEN_LIGHTMAP))
		return;

	if ((pass->shaderbits & SBITS_MASK_BITS) != (pass2->shaderbits & SBITS_MASK_BITS))
		return;

	// check if we can use multiple passes
	if (pass2->blendmode == PBM_DOTPRODUCT)
	{
		pass->numMergedPasses++;
	}
	else if (pass->numMergedPasses < be_maxpasses)
	{
		if (pass->blendmode == PBM_REPLACE || pass->blendmode == PBM_REPLACELIGHT)
		{
			if ((pass2->blendmode == PBM_DECAL && sh_config.tex_env_combine) ||
				(pass2->blendmode == PBM_ADD && sh_config.env_add) ||
				(pass2->blendmode && pass2->blendmode != PBM_ADD) ||	sh_config.nv_tex_env_combine4)
			{
				pass->numMergedPasses++;
			}
		}
		else if (pass->blendmode == PBM_ADD &&
			pass2->blendmode == PBM_ADD && sh_config.env_add)
		{
			pass->numMergedPasses++;
		}
		else if ((pass->blendmode == PBM_MODULATE || pass->blendmode == PBM_OVERBRIGHT) && (pass2->blendmode == PBM_MODULATE || pass2->blendmode == PBM_OVERBRIGHT))
		{
			pass->numMergedPasses++;
		}
		else if (pass->blendmode == PBM_DECAL && pass2->blendmode == PBM_OVERBRIGHT)
			pass->numMergedPasses++;	//HACK: allow modulating overbright passes with decal passes. overbright passes need to blend with something for their lighting to be correct, so this is a tradeoff.
		else
			return;
	}
	else return;

	/*if (pass->texgen == T_GEN_LIGHTMAP && (pass->blendmode == PBM_REPLACE || pass->blendmode == PBM_REPLACELIGHT) && pass2->blendmode == PBM_MODULATE && sh_config.tex_env_combine)
	{
//		if (pass->rgbgen == RGB_GEN_IDENTITY)
//			pass->rgbgen = RGB_GEN_IDENTITY_OVERBRIGHT;	//get the light levels right
//		pass2->blendmode = PBM_OVERBRIGHT;
	}
	if (pass2->texgen == T_GEN_LIGHTMAP && pass2->blendmode == PBM_OVERBRIGHT && sh_config.tex_env_combine)
	{
//		if (pass->rgbgen == RGB_GEN_IDENTITY)
//			pass->rgbgen = RGB_GEN_IDENTITY_OVERBRIGHT;	//get the light levels right
//		pass->blendmode = PBM_REPLACELIGHT;
//		pass2->blendmode = PBM_OVERBRIGHT;
		pass->blendmode = PBM_OVERBRIGHT;
	}
	*/
}

const char *Shader_AlphaMaskProgArgs(shader_t *s)
{
	if (s->numpasses)
	{
		//alpha mask values ALWAYS come from the first pass.
		shaderpass_t *pass = &s->passes[0];
		switch(pass->shaderbits & SBITS_ATEST_BITS)
		{
		default:
			break;
		//cases inverted. the atest is to retain the pixel, glsl is to discard (alpha OP MASK).
		case SBITS_ATEST_GT0:
			return "#MASK=0.0#MASKLT=1";
		case SBITS_ATEST_LT128:
			return "#MASK=0.5";
		case SBITS_ATEST_GE128:
			return "#MASK=0.5#MASKLT=1";	//ignore the eq part.
		}
	}
	return "";
}

void Shader_Programify (parsestate_t *ps)
{
	shader_t *s = ps->s;
	unsigned int reflectrefract = 0;
	const char *prog = NULL;
	const char *mask;
	char args[1024];
	qboolean eightbit = false;
/*	enum
	{
		T_UNKNOWN,
		T_WALL,
		T_MODEL
	} type = 0;*/
	int i;
	shaderpass_t *pass, *lightmap = NULL, *modellighting = NULL, *vertexlighting = NULL;
	for (i = 0; i < s->numpasses; i++)
	{
		pass = &s->passes[i];
		if (pass->rgbgen == RGB_GEN_LIGHTING_DIFFUSE || pass->rgbgen == RGB_GEN_ENTITY_LIGHTING_DIFFUSE)
		{
			if (s->usageflags & SUF_LIGHTMAP)
				lightmap = pass;
			else
				modellighting = pass;
		}
		else if (pass->rgbgen == RGB_GEN_ENTITY)
			modellighting = pass;
		else if (pass->rgbgen == RGB_GEN_VERTEX_LIGHTING || pass->rgbgen == RGB_GEN_VERTEX_EXACT)
		{
			if (s->usageflags & (SUF_LIGHTMAP|SUF_2D))
				vertexlighting = pass;
			else
				modellighting = pass;	//fucking DP morons who don't know what lightmaps are.
		}
		else if (pass->texgen == T_GEN_LIGHTMAP && pass->tcgen == TC_GEN_LIGHTMAP)
		{
			if (s->usageflags & SUF_LIGHTMAP)
				lightmap = pass;
			else
				modellighting = pass;	//fucking DP morons who don't know what lightmaps are.
		}

		/*if (pass->numtcmods || (pass->shaderbits & SBITS_ATEST_BITS))
			return;
		if (pass->texgen == T_GEN_LIGHTMAP && pass->tcgen == TC_GEN_LIGHTMAP)
			;
		else if (pass->texgen != T_GEN_LIGHTMAP && pass->tcgen == TC_GEN_BASE)
			;
		else
			return;*/
	}

	if (ps->forcedshader)
		prog = ps->forcedshader;
	else if (ps->dpwatertype)
	{
		prog = va("altwater%s#USEMODS#FRESNEL_EXP=2.0"
				//variable parts
				"#STRENGTH_REFR=%g#STRENGTH_REFL=%g"
				"#TINT_REFR=%g,%g,%g"
				"#TINT_REFL=%g,%g,%g"
				"#FRESNEL_MIN=%g#FRESNEL_RANGE=%g"
				"#ALPHA=%g",
				//those args
				(ps->dpwatertype&1)?"#REFLECT":"",
				ps->refractfactor*0.01, ps->reflectfactor*0.01,
				ps->refractcolour[0],ps->refractcolour[1],ps->refractcolour[2],
				ps->reflectcolour[0],ps->reflectcolour[1],ps->reflectcolour[2],
				ps->reflectmin, ps->reflectmax-ps->reflectmin,
				ps->wateralpha
			);
		//clear out blending and force regular depth.
		s->passes[0].shaderbits &= ~(SBITS_BLEND_BITS|SBITS_MISC_NODEPTHTEST|SBITS_DEPTHFUNC_BITS);
		s->passes[0].shaderbits |= SBITS_MISC_DEPTHWRITE;

		if (ps->dpwatertype & 1)
			reflectrefract |= SHADER_HASREFLECT;
		if (ps->dpwatertype & 2)
			reflectrefract |= SHADER_HASREFRACT;
		if (ps->dpwatertype & 4)
		{
			reflectrefract |= SHADER_HASREFRACT|SHADER_HASPORTAL;	//doubles up as a 'camera'
			if (s->sort == SHADER_SORT_PORTAL)
				s->sort = SHADER_SORT_OPAQUE;	//don't do it twice.
		}
	}
	else if (modellighting)
	{
		eightbit = r_softwarebanding && (qrenderer == QR_OPENGL) && sh_config.progs_supported;
		if (eightbit)
			prog = "defaultskin#EIGHTBIT";
		else
			prog = "defaultskin";
	}
	else if (lightmap)
	{
		eightbit = r_softwarebanding && (qrenderer == QR_OPENGL || qrenderer == QR_VULKAN) && sh_config.progs_supported;
		if (eightbit)
			prog = "defaultwall#EIGHTBIT";
		else
			prog = "defaultwall";
	}
	else if (vertexlighting)
	{
		if (r_forceprogramify.ival < 0)
			prog = "defaultfill";
		else
		{
			pass = vertexlighting;
			prog = "default2d";
		}
	}
	else
	{
		if (r_forceprogramify.ival < 0)
			prog = "defaultfill";
		else
			return;
	}

	args[0] = 0;
	if (ps->specularvalscale != 1)
		Q_strncatz(args, va("#specmul=%g", ps->specularvalscale), sizeof(args));
	if (ps->specularexpscale != 1)
		Q_strncatz(args, va("#specexp=%g", ps->specularexpscale), sizeof(args));
/*	switch(ps->offsetmappingmode)
	{
	case 0:	//force off.
		Q_strncatz(args, va("#NOOFFSETMAPPING", ps->specularexpscale), sizeof(args));
		break;
	case 1:	//force linear
		Q_strncatz(args, va("#NORELIEFMAPPING", ps->specularexpscale), sizeof(args));
		break;
	case 2:	//force relief
		Q_strncatz(args, va("#RELIEFMAPPING", ps->specularexpscale), sizeof(args));
		break;
	}
	if (ps->offsetmappingscale != 1)
		Q_strncatz(args, va("#OFFSETMAPPING_SCALE=%g", ps->offsetmappingscale), sizeof(args));
	if (ps->offsetmappingbias != 0)
		Q_strncatz(args, va("#OFFSETMAPPING_BIAS=%g", ps->offsetmappingbias), sizeof(args));
*/
	mask = strchr(s->name, '#');
	if (mask)
		Q_strncatz(args, mask, sizeof(args));

	mask = Shader_AlphaMaskProgArgs(s);

	s->prog = Shader_FindGeneric(va("%s%s%s", prog, mask, args), qrenderer);
	if (s->prog)
	{
		s->numpasses = 0;
		if (reflectrefract)
		{
			if (s->passes[0].numtcmods > 0 && s->passes[0].tcmods[0].type == SHADER_TCMOD_SCALE)
			{	//crappy workaround for DP bug.
				s->passes[0].tcmods[0].args[0] *= 4;
				s->passes[0].tcmods[0].args[1] *= 4;
			}
			s->passes[s->numpasses++].texgen = T_GEN_REFRACTION;
			s->passes[s->numpasses++].texgen = T_GEN_REFLECTION;
//			s->passes[s->numpasses++].texgen = T_GEN_RIPPLEMAP;
//			s->passes[s->numpasses++].texgen = T_GEN_REFRACTIONDEPTH;
			s->flags |= reflectrefract;
		}
		else
		{
			if (eightbit)
			{
				s->passes[s->numpasses].anim_frames[0] = R_LoadColourmapImage();
				s->passes[s->numpasses++].texgen = T_GEN_SINGLEMAP;
			}
			else			
				s->passes[s->numpasses++].texgen = T_GEN_DIFFUSE;
			s->flags |= SHADER_HASDIFFUSE;
		}
	}
}

void Shader_Finish (parsestate_t *ps)
{
	shader_t *s = ps->s;
	int i;
	shaderpass_t *pass;
	
	//FIXME: reorder doom3 stages.
	//put diffuse first. give it a lightmap pass also, if we found a diffuse one with no lightmap.
	//then the ambient stages.
	//and forget about the bump/specular stages as we don't support them and already stripped them.

#if 0
	if (s->flags & SHADER_SKY)
	{
		/*skies go all black if fastsky is set*/
		if (r_fastsky.ival)
			s->flags = 0;
		/*or if its purely a skybox and has missing textures*/
//		if (!s->numpasses)
//			for (i = 0; i < 6; i++)
//				if (missing_texture.ref == s->skydome->farbox_textures[i].ref)
//					s->flags = 0;
		if (!(s->flags & SHADER_SKY))
		{
			Shader_Reset(s);

			Shader_DefaultScript(s->name, s,
						"{\n"
							"sort sky\n"
							"{\n"
								"map $whiteimage\n"
								"rgbgen srgb $r_fastskycolour\n"
							"}\n"
							"surfaceparm nodlight\n"
						"}\n"
					);
			return;
		}
	}
#endif

	if (s->prog && !s->numpasses)
	{
		pass = &s->passes[s->numpasses++];
		pass->tcgen = TC_GEN_BASE;
		pass->texgen = T_GEN_DIFFUSE;
		pass->shaderbits |= SBITS_MISC_DEPTHWRITE;
		pass->rgbgen = RGB_GEN_IDENTITY;
		pass->alphagen = ALPHA_GEN_IDENTITY;
		pass->numMergedPasses = 1;
		Shader_SetBlendmode(pass, NULL);
	}

	if (!s->numpasses && s->sort != SHADER_SORT_PORTAL && !(s->flags & (SHADER_NODRAW|SHADER_SKY)) && !s->fog_dist)
	{
		if (r_forceprogramify.ival >= 2)
		{
			Con_DPrintf("Shader %s with no passes, forcing nodraw\n", s->name);
			s->flags |= SHADER_NODRAW;
		}
		else
		{
			pass = &s->passes[s->numpasses++];
			pass = &s->passes[0];
			pass->tcgen = TC_GEN_BASE;
			if (TEXVALID(s->defaulttextures->base))
				pass->texgen = T_GEN_DIFFUSE;
			else
			{
				pass->texgen = T_GEN_SINGLEMAP;
				TEXASSIGN(pass->anim_frames[0], R_LoadHiResTexture(s->name, NULL, IF_NOALPHA));
				if (!TEXVALID(pass->anim_frames[0]))
				{
					Con_Printf("Shader %s failed to load default texture\n", s->name);
					pass->anim_frames[0] = missing_texture;
				}
				Con_Printf("Shader %s with no passes and no surfaceparm nodraw, inserting pass\n", s->name);
			}
			pass->shaderbits |= SBITS_MISC_DEPTHWRITE;
			pass->rgbgen = RGB_GEN_VERTEX_LIGHTING;
			pass->alphagen = ALPHA_GEN_IDENTITY;
			pass->numMergedPasses = 1;
			Shader_SetBlendmode(pass, NULL);
		}
	}

	if (!Q_stricmp (s->name, "flareShader"))
	{
		s->flags |= SHADER_FLARE;
		s->flags |= SHADER_NODRAW;
	}

	if (!s->numpasses && !s->sort)
	{
		s->sort = SHADER_SORT_ADDITIVE;
		return;
	}

	if (!s->sort && s->passes->texgen == T_GEN_CURRENTRENDER)
		s->sort = SHADER_SORT_NEAREST;


	if ((s->polyoffset.unit < 0) && !s->sort)
	{
		s->sort = SHADER_SORT_DECAL;
	}

	if ((r_vertexlight.value || !(s->usageflags & SUF_LIGHTMAP)) && !s->prog)
	{
		// do we have a lightmap pass?
		pass = s->passes;
		for (i = 0; i < s->numpasses; i++, pass++)
		{
			if (pass->flags & SHADER_PASS_LIGHTMAP)
				break;
		}

		if (i == s->numpasses)
		{
			goto done;
		}

		if (!r_vertexlight.value && pass->rgbgen == RGB_GEN_IDENTITY)
		{	//we found the lightmap pass. if we need a vertex-lit shader then just switch over the rgbgen+texture and hope other things work out
			pass->rgbgen = RGB_GEN_VERTEX_LIGHTING;
			pass->flags &= ~SHADER_PASS_LIGHTMAP;
			pass->tcgen = T_GEN_SINGLEMAP;
			goto done;
		}

		// try to find pass with rgbgen set to RGB_GEN_VERTEX
		pass = s->passes;
		for (i = 0; i < s->numpasses; i++, pass++)
		{
			if (pass->rgbgen == RGB_GEN_VERTEX_LIGHTING)
				break;
		}

		if (i < s->numpasses)
		{		// we found it
			pass->flags |= SHADER_CULL_FRONT;
			pass->flags &= ~SHADER_PASS_ANIMMAP;
			pass->shaderbits &= ~SBITS_BLEND_BITS;
			pass->blendmode = 0;
			pass->shaderbits |= SBITS_MISC_DEPTHWRITE;
			pass->alphagen = ALPHA_GEN_IDENTITY;
			pass->numMergedPasses = 1;
			s->flags |= SHADER_DEPTHWRITE;
			s->sort = SHADER_SORT_OPAQUE;
			s->numpasses = 1;
			memcpy(&s->passes[0], pass, sizeof(shaderpass_t));
		}
		else
		{	// we didn't find it - simply remove all lightmap passes
			pass = s->passes;
			for(i = 0; i < s->numpasses; i++, pass++)
			{
				if (pass->flags & SHADER_PASS_LIGHTMAP)
					break;
			}

			if ( i == s->numpasses -1 )
			{
				s->numpasses--;
			}
			else if ( i < s->numpasses - 1 )
			{
				for ( ; i < s->numpasses - 1; i++, pass++ )
				{
					memcpy ( pass, &s->passes[i+1], sizeof(shaderpass_t) );
				}
				s->numpasses--;
			}

			if ( s->passes[0].numtcmods )
			{
				pass = s->passes;
				for ( i = 0; i < s->numpasses; i++, pass++ )
				{
					if ( !pass->numtcmods )
						break;
				}

				memcpy ( &s->passes[0], pass, sizeof(shaderpass_t) );
			}

			s->passes[0].rgbgen = RGB_GEN_VERTEX_LIGHTING;
			s->passes[0].alphagen = ALPHA_GEN_IDENTITY;
			s->passes[0].blendmode = 0;
			s->passes[0].flags &= ~(SHADER_PASS_ANIMMAP|SHADER_PASS_NOCOLORARRAY);
			s->passes[0].shaderbits &= ~SBITS_BLEND_BITS;
			s->passes[0].shaderbits |= SBITS_MISC_DEPTHWRITE;
			s->passes[0].numMergedPasses = 1;
			s->numpasses = 1;
			s->flags |= SHADER_DEPTHWRITE;
		}
	}
done:;

	//if we've no specular map, try and find whatever the q3 syntax said. hopefully it'll be compatible...
 	if (!TEXVALID(s->defaulttextures->specular) && !(s->flags & SHADER_HASGLOSS))
	{
		for (pass = s->passes, i = 0; i < s->numpasses; i++, pass++)
		{
			if (pass->alphagen == ALPHA_GEN_SPECULAR)
				if (pass->texgen == T_GEN_ANIMMAP || pass->texgen == T_GEN_SINGLEMAP)
					s->defaulttextures->specular = pass->anim_frames[0];
		}
	}

	if (!TEXVALID(s->defaulttextures->base) && !(s->flags & SHADER_HASDIFFUSE) && !s->prog)
	{	//if one of the other passes specifies $diffuse, don't try and guess one, because that means that other pass's texture gets used for BOTH passes, which isn't good.
		//also, don't guess one if a program was specified.
		shaderpass_t *best = NULL;
		int bestweight = 9999999;
		int weight;

		for (pass = s->passes, i = 0; i < s->numpasses; i++, pass++)
		{
			weight = 0;
			if (pass->flags & SHADER_PASS_DETAIL)
				weight += 500;	//prefer not to use a detail pass. these are generally useless.
			if (pass->numtcmods || pass->tcgen != TC_GEN_BASE)
				weight += 200;
			if (pass->rgbgen != RGB_GEN_IDENTITY && pass->rgbgen != RGB_GEN_IDENTITY_OVERBRIGHT && pass->rgbgen != RGB_GEN_IDENTITY_LIGHTING)
				weight += 100;

			if (pass->texgen != T_GEN_ANIMMAP && pass->texgen != T_GEN_SINGLEMAP
#ifdef HAVE_MEDIA_DECODER
					&& pass->texgen != T_GEN_VIDEOMAP
#endif
					)
				weight += 1000;

			if ((pass->texgen == T_GEN_ANIMMAP || pass->texgen == T_GEN_SINGLEMAP) && pass->anim_frames[0] && *pass->anim_frames[0]->ident == '$')
				weight += 1500;
			
			if (weight < bestweight)
			{
				bestweight = weight;
				best = pass;
			}
		}

		if (best)
		{
			if (best->texgen == T_GEN_ANIMMAP || best->texgen == T_GEN_SINGLEMAP)
			{
				if (best->anim_frames[0] && *best->anim_frames[0]->ident != '$')
					s->defaulttextures->base = best->anim_frames[0];
			}
#ifdef HAVE_MEDIA_DECODER
			else if (pass->texgen == T_GEN_VIDEOMAP && pass->cin)
				s->defaulttextures->base = Media_UpdateForShader(best->cin);
#endif
		}
	}

	for (i = 0; i < s->numpasses; i += (pass->prog?pass->numMergedPasses:1))
	{
		pass = s->passes+i;
		if (!(pass->shaderbits & (SBITS_BLEND_BITS|SBITS_MASK_BITS)))
		{
			if (pass->texgen == T_GEN_LIGHTMAP && r_forceprogramify.ival==2)
				continue;	//pretend its blended.
			break;
		}
	}

	// all passes have blendfuncs
	if (i == s->numpasses)
	{
		int maskpass;
		qboolean isopaque = false;

		maskpass = -1;
		pass = s->passes;
		for (i = 0; i < s->numpasses; i++, pass++ )
		{
			if (pass->shaderbits & SBITS_ATEST_BITS)
			{
				maskpass = i;
			}
			else if ((pass->shaderbits & SBITS_MASK_BITS) == 0)
			{	//a few shaders use blendfunc one zero so that they're ignored when using r_vertexlight (while later alpha-masked surfs are not).
				if (/*(pass->shaderbits & (SBITS_SRCBLEND_BITS|SBITS_DSTBLEND_BITS)) == 0 ||*/
					(pass->shaderbits & (SBITS_SRCBLEND_BITS|SBITS_DSTBLEND_BITS)) == (SBITS_SRCBLEND_ONE|SBITS_DSTBLEND_ZERO))
					isopaque = true;
			}

			if (pass->rgbgen == RGB_GEN_UNKNOWN)
			{
				if (   (pass->shaderbits & SBITS_SRCBLEND_BITS) == 0
					|| (pass->shaderbits & SBITS_SRCBLEND_BITS) == SBITS_SRCBLEND_ONE
					|| (pass->shaderbits & SBITS_SRCBLEND_BITS) == SBITS_SRCBLEND_SRC_ALPHA)
					pass->rgbgen = RGB_GEN_IDENTITY_LIGHTING;
				else
					pass->rgbgen = RGB_GEN_IDENTITY;
			}

			Shader_SetBlendmode (pass, i?pass-1:NULL);

			if (pass->blendmode == PBM_ADD && !s->defaulttextures->fullbright)
				s->defaulttextures->fullbright = pass->anim_frames[0];
		}

		if (!(s->flags & SHADER_SKY ) && !s->sort)
		{
			if (isopaque)
				s->sort = SHADER_SORT_OPAQUE;
			else if (maskpass == -1)
			{
				if (s->numpasses && s->passes[0].blendmode == PBM_ADD)
					s->sort = SHADER_SORT_ADDITIVE;
				else
					s->sort = SHADER_SORT_BLEND;
			}
			else
				s->sort = SHADER_SORT_SEETHROUGH;
		}
	}
	else
	{
		int	j;
		shaderpass_t *sp;

		sp = s->passes;
		for (j = 0; j < s->numpasses; j++, sp++)
		{
			if (sp->rgbgen == RGB_GEN_UNKNOWN)
			{
				if (sp->flags & SHADER_PASS_LIGHTMAP)
					sp->rgbgen = RGB_GEN_IDENTITY_LIGHTING;
				else
					sp->rgbgen = RGB_GEN_IDENTITY;
			}

			Shader_SetBlendmode (sp, j?sp-1:NULL);
		}

		if (!s->sort)
		{
			if (i < s->numpasses && (s->passes[i].shaderbits & SBITS_ATEST_BITS))
				s->sort = SHADER_SORT_SEETHROUGH;
		}

		if (!( s->flags & SHADER_DEPTHWRITE) &&
			!(s->flags & SHADER_SKY))
		{
			s->passes->shaderbits |= SBITS_MISC_DEPTHWRITE;
			s->flags |= SHADER_DEPTHWRITE;
		}
	}

	if (s->numpasses >= 2)
	{
		int j;

		pass = s->passes;
		for (i = 0; i < s->numpasses;)
		{
			if (i == s->numpasses - 1)
				break;

			pass = s->passes + i;
			for (j = 1; j < s->numpasses-i && j == pass->numMergedPasses && j+1 < be_maxpasses; j++)
				Shader_SetPassFlush (pass, pass + j);

			i += pass->numMergedPasses;
		}
	}

	if (!s->sort)
	{
		s->sort = SHADER_SORT_OPAQUE;
	}

	if ((s->flags & SHADER_SKY) && (s->flags & SHADER_DEPTHWRITE))
	{
		s->flags &= ~SHADER_DEPTHWRITE;
	}

	if (!s->bemoverrides[bemoverride_depthonly])
	{
		const char *mask = Shader_AlphaMaskProgArgs(s);
		if (*mask || (s->prog&&s->prog->tess))
			s->bemoverrides[bemoverride_depthonly] = R_RegisterShader(va("depthonly%s%s", mask, (s->prog&&s->prog->tess)?"#TESS":""), SUF_NONE, 
				"{\n"
					"program depthonly\n"
					"{\n"
						"map $diffuse\n"
						"depthwrite\n"
						"maskcolor\n"
					"}\n"
				"}\n");
	}
	if (!s->bemoverrides[LSHADER_STANDARD] && (s->prog&&s->prog->tess))
	{
		int mode;
		for (mode = 0; mode < LSHADER_MODES; mode++)
		{
			if ((mode & LSHADER_CUBE) && (mode & LSHADER_SPOT))
				continue;
			if (s->bemoverrides[mode])
				continue;
			s->bemoverrides[mode] = R_RegisterShader(va("rtlight%s%s%s%s#TESS", 
																(mode & LSHADER_SMAP)?"#PCF":"",
																(mode & LSHADER_SPOT)?"#SPOT":"",
																(mode & LSHADER_CUBE)?"#CUBE":"",
#ifdef GLQUAKE
																(qrenderer == QR_OPENGL && gl_config.arb_shadow && (mode & (LSHADER_SMAP|LSHADER_SPOT)))?"#USE_ARB_SHADOW":""
#else
																""
#endif
																)
														, s->usageflags, 
				"{\n"
					"program rtlight\n"
					"{\n"
						"map $diffuse\n"
						"blendfunc add\n"
						"nodepth\n"
					"}\n"
				"}\n");
		}
	}

	if (!s->prog && sh_config.progs_supported && (r_forceprogramify.ival || (ps->parseflags & SPF_PROGRAMIFY)))
	{
		if (r_forceprogramify.ival >= 2)
		{
			if (s->passes[0].numtcmods == 1 && s->passes[0].tcmods[0].type == SHADER_TCMOD_SCALE)
				s->passes[0].numtcmods = 0;	//DP sucks and doesn't use normalized texture coords *if* there's a shader specified. so lets ignore any extra scaling that this imposes.
			if (s->passes[0].shaderbits & SBITS_ATEST_BITS)	//mimic DP's limited alphafunc support
				s->passes[0].shaderbits = (s->passes[0].shaderbits & ~SBITS_ATEST_BITS) | SBITS_ATEST_GE128;
			s->passes[0].shaderbits &= ~SBITS_DEPTHFUNC_BITS;	//DP ignores this too.
			if (s->flags & SHADER_CULL_BACK)	//DP has no back-face culling
				s->flags = (s->flags&~SHADER_CULL_BACK)|SHADER_CULL_FRONT;

			//disable rtlight stuff when blended.
			if ((s->passes[0].shaderbits & SBITS_SRCBLEND_BITS)==SBITS_SRCBLEND_ONE && (s->passes[0].shaderbits & SBITS_DSTBLEND_BITS)==SBITS_DSTBLEND_ZERO)
				;	//replace
			else if ((s->passes[0].shaderbits & SBITS_SRCBLEND_BITS)==SBITS_SRCBLEND_ONE && (s->passes[0].shaderbits & SBITS_DSTBLEND_BITS)==SBITS_DSTBLEND_ONE)
				s->flags |= SHADER_NOSHADOWS;	//add-ignore-alpha
			else if ((s->passes[0].shaderbits & SBITS_SRCBLEND_BITS)==SBITS_SRCBLEND_SRC_ALPHA && (s->passes[0].shaderbits & SBITS_DSTBLEND_BITS)==SBITS_DSTBLEND_ONE)
				s->flags |= SHADER_NOSHADOWS;	//add-with-alpha
			else if ((s->passes[0].shaderbits & SBITS_SRCBLEND_BITS)==SBITS_SRCBLEND_SRC_ALPHA && (s->passes[0].shaderbits & SBITS_DSTBLEND_BITS)==SBITS_DSTBLEND_ONE_MINUS_SRC_ALPHA)
				s->flags |= SHADER_NOSHADOWS;	//blended
			else
				s->flags |= SHADER_NOSHADOWS|SHADER_NODLIGHT;	//erk...

			if (s->passes[0].numtcmods>1)
			{	//reverse the order of tcmods (dp uses a texture matrix which it concats in reverse order)
				tcmod_t		tmp[SHADER_MAX_TC_MODS];
				int j;
				memcpy(tmp, s->passes[0].tcmods, sizeof(*tmp)*s->passes[0].numtcmods);
				for (j = 0; j < s->passes[0].numtcmods; j++)
					s->passes[0].tcmods[j] = tmp[s->passes[0].numtcmods-1-j];
			}
		}
		Shader_Programify(ps);
	}

	if (s->prog)
	{
		struct
		{
			int gen;
			unsigned int flags;
		} defaulttgen[] =
		{
			//light
			{T_GEN_SHADOWMAP,		0},						//0
			{T_GEN_LIGHTCUBEMAP,	0},						//1

			//material
			{T_GEN_DIFFUSE,			SHADER_HASDIFFUSE},		//2
			{T_GEN_NORMALMAP,		SHADER_HASNORMALMAP},	//3
			{T_GEN_SPECULAR,		SHADER_HASGLOSS},		//4
			{T_GEN_UPPEROVERLAY,	SHADER_HASTOPBOTTOM},	//5
			{T_GEN_LOWEROVERLAY,	SHADER_HASTOPBOTTOM},	//6
			{T_GEN_FULLBRIGHT,		SHADER_HASFULLBRIGHT},	//7
			{T_GEN_PALETTED,		SHADER_HASPALETTED},	//8
			{T_GEN_REFLECTCUBE,		SHADER_HASREFLECTCUBE},	//9
			{T_GEN_REFLECTMASK,		0},						//10
			{T_GEN_DISPLACEMENT,	SHADER_HASDISPLACEMENT},//11
			{T_GEN_OCCLUSION,		0},						//12
//			{T_GEN_REFLECTION,		SHADER_HASREFLECT},		//
//			{T_GEN_REFRACTION,		SHADER_HASREFRACT},		//
//			{T_GEN_REFRACTIONDEPTH,	SHADER_HASREFRACTDEPTH},//
//			{T_GEN_RIPPLEMAP,		SHADER_HASRIPPLEMAP},	//

			//batch
			{T_GEN_LIGHTMAP,		SHADER_HASLIGHTMAP},	//13
			{T_GEN_DELUXMAP,		0},						//14
			//more lightmaps								//15,16,17
			//mode deluxemaps								//18,19,20
		};

#ifdef HAVE_MEDIA_DECODER
		cin_t *cin = R_ShaderGetCinematic(s);
#endif

		//if the glsl doesn't specify all samplers, just trim them.
		s->numpasses = s->prog->numsamplers;

#ifdef HAVE_MEDIA_DECODER
		if (cin && R_ShaderGetCinematic(s) == cin)
			cin = NULL;
#endif

		//if the glsl has specific textures listed, be sure to provide a pass for them.
		for (i = 0; i < sizeof(defaulttgen)/sizeof(defaulttgen[0]); i++)
		{
			if (s->prog->defaulttextures & (1u<<i))
			{
				if (s->numpasses >= SHADER_PASS_MAX)
					break;	//panic...
				s->passes[s->numpasses].flags &= ~SHADER_PASS_DEPTHCMP;
				if (defaulttgen[i].gen == T_GEN_SHADOWMAP)
					s->passes[s->numpasses].flags |= SHADER_PASS_DEPTHCMP;
#ifdef HAVE_MEDIA_DECODER
				if (!i && cin)
				{
					s->passes[s->numpasses].texgen = T_GEN_VIDEOMAP;
					s->passes[s->numpasses].cin = cin;
					cin = NULL;
				}
				else
#endif
				{
					s->passes[s->numpasses].texgen = defaulttgen[i].gen;
#ifdef HAVE_MEDIA_DECODER
					s->passes[s->numpasses].cin = NULL;
#endif
				}
				s->numpasses++;
				s->flags |= defaulttgen[i].flags;
			}
		}

		//must have at least one texture.
		if (!s->numpasses)
		{
#ifdef HAVE_MEDIA_DECODER
			s->passes[0].texgen = cin?T_GEN_VIDEOMAP:T_GEN_DIFFUSE;
			s->passes[0].cin = cin;
#else
			s->passes[0].texgen = T_GEN_DIFFUSE;
#endif
			s->numpasses = 1;
		}
#ifdef HAVE_MEDIA_DECODER
		else if (cin)
			Media_ShutdownCin(cin);
#endif
		s->passes->numMergedPasses = s->numpasses;
	}
	else if (s->numdeforms)
		s->flags |= SHADER_NEEDSARRAYS;
	else
	{
		for (i = 0; i < s->numpasses; i++)
		{
			pass = &s->passes[i];
			if (pass->prog)
				continue;
			if (pass->numtcmods || (s->passes[i].tcgen != TC_GEN_BASE && s->passes[i].tcgen != TC_GEN_LIGHTMAP) || !(s->passes[i].flags & SHADER_PASS_NOCOLORARRAY))
			{
				s->flags |= SHADER_NEEDSARRAYS;
				break;
			}
			if (!(pass->flags & SHADER_PASS_NOCOLORARRAY))
			{
				if (!(((pass->rgbgen == RGB_GEN_VERTEX_LIGHTING) ||
					(pass->rgbgen == RGB_GEN_VERTEX_EXACT) ||
					(pass->rgbgen == RGB_GEN_ONE_MINUS_VERTEX)) &&
					(pass->alphagen == ALPHA_GEN_VERTEX)))
				{
					s->flags |= SHADER_NEEDSARRAYS;
					break;
				}
			}
		}
	}
}
/*
void Shader_UpdateRegistration (void)
{
	int i, j, l;
	shader_t *shader;
	shaderpass_t *pass;

	shader = r_shaders;
	for (i = 0; i < MAX_SHADERS; i++, shader++)
	{
		if (!shader->registration_sequence)
			continue;
		if (shader->registration_sequence != registration_sequence)
		{
			Shader_Free ( shader );
			shader->registration_sequence = 0;
			continue;
		}

		pass = shader->passes;
		for (j = 0; j < shader->numpasses; j++, pass++)
		{
			if (pass->flags & SHADER_PASS_ANIMMAP)
			{
				for (l = 0; l < pass->anim_numframes; l++)
				{
					if (pass->anim_frames[l])
						pass->anim_frames[l]->registration_sequence = registration_sequence;
				}
			}
			else if ( pass->flags & SHADER_PASS_VIDEOMAP )
			{
				// Shader_RunCinematic will do the job
//				pass->cin->frame = -1;
			}
			else if ( !(pass->flags & SHADER_PASS_LIGHTMAP) )
			{
				if ( pass->anim_frames[0] )
					pass->anim_frames[0]->registration_sequence = registration_sequence;
			}
		}
	}
}
*/

/*
	if (*shader_diffusemapname)
	{
		if (!s->defaulttextures.base)
			s->defaulttextures.base = Shader_FindImage (va("%s.tga", shader_diffusemapname), 0);
		if (!s->defaulttextures.bump)
			s->defaulttextures.bump = Shader_FindImage (va("%s_norm.tga", shader_diffusemapname), 0);
		if (!s->defaulttextures.fullbright)
			s->defaulttextures.fullbright = Shader_FindImage (va("%s_glow.tga", shader_diffusemapname), 0);
		if (!s->defaulttextures.specular)
			s->defaulttextures.specular = Shader_FindImage (va("%s_gloss.tga", shader_diffusemapname), 0);
		if (!s->defaulttextures.upperoverlay)
			s->defaulttextures.upperoverlay = Shader_FindImage (va("%s_shirt.tga", shader_diffusemapname), 0);
		if (!s->defaulttextures.loweroverlay)
			s->defaulttextures.loweroverlay = Shader_FindImage (va("%s_pants.tga", shader_diffusemapname), 0);	//stupid yanks...
	}
*/
void QDECL R_BuildDefaultTexnums(texnums_t *src, shader_t *shader, unsigned int imageflags)
{
	char *h;
	char imagename[MAX_QPATH];
	char mapname[MAX_QPATH];
	char *subpath = NULL;
	texnums_t *tex;
	unsigned int a, aframes;
	strcpy(imagename, shader->name);
	h = strchr(imagename, '#');
	if (h)
		*h = 0;
	if (*imagename == '/' || strchr(imagename, ':'))
	{	//this is not security. this is anti-spam for the verbose security in the filesystem code.
		Con_Printf("Warning: shader has absolute path: %s\n", shader->name);
		*imagename = 0;
	}

	//skins can use an alternative path in certain cases, to work around dodgy models.
	if (shader->generator == Shader_DefaultSkin)
		subpath = shader->genargs;

	tex = shader->defaulttextures;
	aframes = max(1, shader->numdefaulttextures);
	//if any were specified explicitly, replicate that into all.
	//this means animmap can be used, with any explicit textures overriding all.
	if (!shader->numdefaulttextures && src)
	{
		//only do this if there wasn't an animmap thing to break everything.
		if (!TEXVALID(tex->base))
			tex->base			= src->base;
		if (!TEXVALID(tex->bump))
			tex->bump			= src->bump;
		if (!TEXVALID(tex->fullbright))
			tex->fullbright		= src->fullbright;
		if (!TEXVALID(tex->specular))
			tex->specular		= src->specular;
		if (!TEXVALID(tex->loweroverlay))
			tex->loweroverlay	= src->loweroverlay;
		if (!TEXVALID(tex->upperoverlay))
			tex->upperoverlay	= src->upperoverlay;
		if (!TEXVALID(tex->reflectmask))
			tex->reflectmask	= src->reflectmask;
		if (!TEXVALID(tex->reflectcube))
			tex->reflectcube	= src->reflectcube;
		if (!TEXVALID(tex->displacement))
			tex->displacement	= src->displacement;
	}
	for (a = 1; a < aframes; a++)
	{
		if (!TEXVALID(tex[a].base))
			tex[a].base			= tex[0].base;
		if (!TEXVALID(tex[a].bump))
			tex[a].bump			= tex[0].bump;
		if (!TEXVALID(tex[a].fullbright))
			tex[a].fullbright	= tex[0].fullbright;
		if (!TEXVALID(tex[a].specular))
			tex[a].specular		= tex[0].specular;
		if (!TEXVALID(tex[a].loweroverlay))
			tex[a].loweroverlay	= tex[0].loweroverlay;
		if (!TEXVALID(tex[a].upperoverlay))
			tex[a].upperoverlay	= tex[0].upperoverlay;
		if (!TEXVALID(tex[a].reflectmask))
			tex[a].reflectmask	= tex[0].reflectmask;
		if (!TEXVALID(tex[a].reflectcube))
			tex[a].reflectcube	= tex[0].reflectcube;
		if (!TEXVALID(tex[a].displacement))
			tex[a].displacement	= tex[0].displacement;
	}
	for (a = 0; a < aframes; a++, tex++)
	{
		COM_StripExtension(tex->mapname, mapname, sizeof(mapname));

		if (!TEXVALID(tex->base))
		{
			/*dlights/realtime lighting needs some stuff*/
			if (!TEXVALID(tex->base) && *tex->mapname)// && (shader->flags & SHADER_HASDIFFUSE))
				tex->base = R_LoadHiResTexture(tex->mapname, NULL, imageflags);

			if (!TEXVALID(tex->base))
				tex->base = R_LoadHiResTexture(imagename, subpath, (*imagename=='{')?0:IF_NOALPHA|imageflags);
		}

		if ((shader->flags & SHADER_HASPALETTED) && !TEXVALID(tex->paletted))
		{
			if (!TEXVALID(tex->paletted) && *tex->mapname)
				tex->paletted = R_LoadHiResTexture(va("%s", tex->mapname), NULL, 0|IF_NEAREST|IF_PALETTIZE|imageflags);
			if (!TEXVALID(tex->paletted))
				tex->paletted = R_LoadHiResTexture(va("%s", imagename), subpath, ((*imagename=='{')?0:IF_NOALPHA)|IF_NEAREST|IF_PALETTIZE|imageflags);
		}

		imageflags |= IF_LOWPRIORITY;

		COM_StripExtension(imagename, imagename, sizeof(imagename));

		if (!TEXVALID(tex->bump))
		{
			if (r_loadbumpmapping || (shader->flags & SHADER_HASNORMALMAP))
			{
				if (!TEXVALID(tex->bump) && *mapname && (shader->flags & SHADER_HASNORMALMAP))
					tex->bump = R_LoadHiResTexture(va("%s_norm", mapname), NULL, imageflags|IF_TRYBUMP|IF_NOSRGB|imageflags);
				if (!TEXVALID(tex->bump))
					tex->bump = R_LoadHiResTexture(va("%s_norm", imagename), subpath, imageflags|IF_TRYBUMP|IF_NOSRGB|imageflags);
			}
		}

		if (!TEXVALID(tex->loweroverlay))
		{
			if (shader->flags & SHADER_HASTOPBOTTOM)
			{
				if (!TEXVALID(tex->loweroverlay) && *mapname)
					tex->loweroverlay = R_LoadHiResTexture(va("%s_pants", mapname), NULL, imageflags);
				if (!TEXVALID(tex->loweroverlay))
					tex->loweroverlay = R_LoadHiResTexture(va("%s_pants", imagename), subpath, imageflags);	/*how rude*/
			}
		}

		if (!TEXVALID(tex->upperoverlay))
		{
			if (shader->flags & SHADER_HASTOPBOTTOM)
			{
				if (!TEXVALID(tex->upperoverlay) && *mapname)
					tex->upperoverlay = R_LoadHiResTexture(va("%s_shirt", mapname), NULL, imageflags);
				if (!TEXVALID(tex->upperoverlay))
					tex->upperoverlay = R_LoadHiResTexture(va("%s_shirt", imagename), subpath, imageflags);
			}
		}

		if (!TEXVALID(tex->specular))
		{
			extern cvar_t gl_specular;
			if ((shader->flags & SHADER_HASGLOSS) && gl_specular.value && gl_load24bit.value)
			{
				if (!TEXVALID(tex->specular) && *mapname)
					tex->specular = R_LoadHiResTexture(va("%s_gloss", mapname), NULL, imageflags);
				if (!TEXVALID(tex->specular))
					tex->specular = R_LoadHiResTexture(va("%s_gloss", imagename), subpath, imageflags);
			}
		}

		if (!TEXVALID(tex->fullbright))
		{
			extern cvar_t r_fb_bmodels;
			if ((shader->flags & SHADER_HASFULLBRIGHT) && r_fb_bmodels.value && gl_load24bit.value)
			{
				if (!TEXVALID(tex->fullbright) && *mapname)
					tex->fullbright = R_LoadHiResTexture(va("%s_luma:%s_glow", mapname, mapname), NULL, imageflags);
				if (!TEXVALID(tex->fullbright))
					tex->fullbright = R_LoadHiResTexture(va("%s_luma:%s_glow", imagename, imagename), subpath, imageflags);
			}
		}

		//if there's a reflectcube texture specified by the shader, make sure we have a reflectmask to go with it.
		if (tex->reflectcube)
		{
			if (!TEXVALID(tex->reflectmask) && *mapname)
				tex->reflectmask = R_LoadHiResTexture(va("%s_reflect", mapname), NULL, imageflags);
			if (!TEXVALID(tex->reflectmask))
				tex->reflectmask = R_LoadHiResTexture(va("%s_reflect", imagename), subpath, imageflags);
		}
	}
}

#if 0//def Q2BSPS
static qbyte *ReadRGBA8ImageFile(const char *fname, const char *subpath, int *width, int *height)
{
	qboolean hasalpha;
	qofs_t filesize;
	qbyte *img, *filedata = NULL;
	char *patterns[] = 
	{
		"*overrides/%s.tga",
		"*overrides/%s.jpg",
		"textures/%s.tga",
		"textures/%s.jpg",
	};
	char nname[MAX_QPATH];
	size_t p;
	for (p = 0; p < countof(patterns) && !filedata; p++)
	{
		if (*patterns[p] == '*')
			Q_snprintfz(nname, sizeof(nname), patterns[p]+1, COM_SkipPath(fname));
		else
			Q_snprintfz(nname, sizeof(nname), patterns[p], fname);
		filedata = FS_MallocFile(nname, FS_GAME, &filesize);
	}
	img = filedata?Read32BitImageFile(filedata, filesize, width, height, &hasalpha, fname):NULL;
	BZ_Free(filedata);
	return img;
}
#endif

//call this with some fallback textures to directly load some textures
void QDECL R_BuildLegacyTexnums(shader_t *shader, const char *fallbackname, const char *subpath, unsigned int loadflags, unsigned int imageflags, uploadfmt_t basefmt, size_t width, size_t height, qbyte *srcdata, qbyte *palette)
{
	char *h;
	char imagename[MAX_QPATH];
	char mapname[MAX_QPATH];	//as specified by the shader.
	//extern cvar_t gl_miptexLevel;
	texnums_t *tex = shader->defaulttextures;
	int a, aframes;
	/*else if (gl_miptexLevel.ival)
	{
		unsigned int miplevel = 0, i;
		for (i = 0; i < 3 && i < gl_miptexLevel.ival && mipdata[i]; i++)
			miplevel = i;
		for (i = 0; i < 3; i++)
			dontcrashme[i] = (miplevel+i)>3?NULL:mipdata[miplevel+i];
		width >>= miplevel;
		height >>= miplevel;
		mipdata = dontcrashme;
	}
	*/
	strcpy(imagename, shader->name);
	h = strchr(imagename, '#');
	if (h)
		*h = 0;
	if (*imagename == '/' || strchr(imagename, ':'))
	{	//this is not security. this is anti-spam for the verbose security in the filesystem code.
		Con_Printf("Warning: shader has absolute path: %s\n", shader->name);
		*imagename = 0;
	}

	//for water texture replacements
	while((h = strchr(imagename, '*')))
		*h = '#';

	loadflags &= shader->flags;

	//skins can use an alternative path in certain cases, to work around dodgy models.
	if (shader->generator == Shader_DefaultSkin)
		subpath = shader->genargs;

	//optimise away any palette info if we can...
	if (!palette || palette == host_basepal)
	{
		if (basefmt == TF_MIP4_8PAL24)
			basefmt = TF_MIP4_SOLID8;
//		if (basefmt == TF_MIP4_8PAL24_T255)
//			basefmt = TF_MIP4_TRANS8;
	}

	//make sure the noalpha thing is set properly.
	switch(basefmt)
	{
	case TF_MIP4_P8:
	case TF_MIP4_8PAL24:
	case TF_MIP4_SOLID8:
	case TF_SOLID8:
		imageflags |= IF_NOALPHA;
		//fallthrough
	case TF_MIP4_8PAL24_T255:
		if (!srcdata)
			basefmt = TF_SOLID8;
		break;
	default:
		break;
	}
	imageflags |= IF_MIPCAP;

	COM_StripExtension(imagename, imagename, sizeof(imagename));

	aframes = max(1, shader->numdefaulttextures);
	//if any were specified explicitly, replicate that into all.
	//this means animmap can be used, with any explicit textures overriding all.
	for (a = 1; a < aframes; a++)
	{
		if (!TEXVALID(tex[a].base))
			tex[a].base			= tex[0].base;
		if (!TEXVALID(tex[a].bump))
			tex[a].bump			= tex[0].bump;
		if (!TEXVALID(tex[a].fullbright))
			tex[a].fullbright	= tex[0].fullbright;
		if (!TEXVALID(tex[a].specular))
			tex[a].specular		= tex[0].specular;
		if (!TEXVALID(tex[a].loweroverlay))
			tex[a].loweroverlay	= tex[0].loweroverlay;
		if (!TEXVALID(tex[a].upperoverlay))
			tex[a].upperoverlay	= tex[0].upperoverlay;
		if (!TEXVALID(tex[a].reflectmask))
			tex[a].reflectmask	= tex[0].reflectmask;
		if (!TEXVALID(tex[a].reflectcube))
			tex[a].reflectcube	= tex[0].reflectcube;
		if (!TEXVALID(tex[a].displacement))
			tex[a].displacement	= tex[0].displacement;
	}
	for (a = 0; a < aframes; a++, tex++)
	{
		COM_StripExtension(tex->mapname, mapname, sizeof(mapname));

#if 0//def Q2BSPS
		if (gl_load24bit.ival == 2)
		{
			qbyte *base;
			int basewidth, baseheight;
			qbyte *norm;
			int normwidth, normheight;
			qbyte tmp;
			int x, y;
			base = ReadRGBA8ImageFile(imagename, subpath, &basewidth, &baseheight);
			//base contains diffuse RGB, and height
			if (base)
			{
				tex->base = Image_GetTexture(va("%s_diff", imagename), subpath, imageflags|IF_NOREPLACE, base, NULL, basewidth, baseheight, TF_RGBX32);	//queues the texture creation.
				norm = ReadRGBA8ImageFile(va("%s_bump", imagename), subpath, &normwidth, &normheight);
				if (norm)
				{
					if (normwidth != basewidth || normheight != baseheight)
					{
						//sucks...
						tex->bump = Image_GetTexture(va("%s_norm", imagename), subpath, imageflags|IF_NOREPLACE, norm, NULL, normwidth, normheight, TF_RGBX32);	//queues the texture creation.
					}
					else
					{
						//so we have a base texture, and normalmap
						//we already uploaded the diffuse, so now we can just pretend that the base is the specularmap.
						//we just need to swap the two alpha channels.
						for (y = 0; y < baseheight; y++)
						{
							for (x = 0; x < basewidth; x++)
							{
								tmp = base[(x+y*basewidth)*4+3];
								base[(x+y*basewidth)*4+3] = norm[(x+y*basewidth)*4+3];
								norm[(x+y*basewidth)*4+3] = ((x+y)&1)*255;//tmp;
							}
						}
						tex->bump = Image_GetTexture(va("%s_norm", imagename), subpath, imageflags|IF_NOREPLACE, norm, NULL, normwidth, normheight, TF_RGBA32);	//queues the texture creation.
						tex->specular = Image_GetTexture(va("%s_spec", imagename), subpath, imageflags|IF_NOREPLACE, base, NULL, basewidth, baseheight, TF_RGBA32);	//queues the texture creation.
					}
					BZ_Free(norm);
				}
				else
				{
					//generate a height8 image from the alpha channel. we  can do it in place
					for (y = 0; y < baseheight; y++)
					{
						for (x = 0; x < basewidth; x++)
							base[(x+y*basewidth)] = base[(x+y*basewidth)*4+3];
					}
					tex->bump = Image_GetTexture(va("%s_norm", imagename), subpath, imageflags|IF_NOREPLACE, base, NULL, basewidth, baseheight, TF_HEIGHT8);
				}
				BZ_Free(base);
			}
		}
#endif

		/*dlights/realtime lighting needs some stuff*/
		if (loadflags & SHADER_HASDIFFUSE)
		{
			if (!TEXVALID(tex->base) && *mapname)
				tex->base = R_LoadHiResTexture(mapname, NULL, imageflags);
			if (!TEXVALID(tex->base) && fallbackname)
			{
				if (gl_load24bit.ival)
				{
					tex->base = Image_GetTexture(imagename, subpath, imageflags|IF_NOWORKER, NULL, NULL, width, height, basefmt);
					if (!TEXLOADED(tex->base))
					{
						tex->base = Image_GetTexture(fallbackname, subpath, imageflags|IF_NOWORKER, NULL, NULL, width, height, basefmt);
						if (TEXLOADED(tex->base))
							Q_strncpyz(imagename, fallbackname, sizeof(imagename));
					}
				}
				if (!TEXLOADED(tex->base))
					tex->base = Image_GetTexture(imagename, subpath, imageflags, srcdata, palette, width, height, basefmt);
			}
			else if (!TEXVALID(tex->base))
				tex->base = Image_GetTexture(imagename, subpath, imageflags, srcdata, palette, width, height, basefmt);
		}

		if (loadflags & SHADER_HASPALETTED)
		{
			if (!TEXVALID(tex->paletted) && *mapname)
				tex->paletted = R_LoadHiResTexture(va("%s_pal", mapname), NULL, imageflags|IF_NEAREST);
			if (!TEXVALID(tex->paletted))
				tex->paletted = Image_GetTexture(va("%s_pal", imagename), subpath, imageflags|IF_NEAREST|IF_NOSRGB, srcdata, palette, width, height, (basefmt==TF_MIP4_SOLID8)?TF_MIP4_P8:PTI_P8);
		}

		imageflags |= IF_LOWPRIORITY;
		//all the rest need/want an alpha channel in some form.
		imageflags &= ~IF_NOALPHA;
		imageflags |= IF_NOGAMMA;

		if (loadflags & SHADER_HASNORMALMAP||*imagename=='#')
		{
			extern cvar_t r_shadow_bumpscale_basetexture;
			if (!TEXVALID(tex->bump) && *mapname)
				tex->bump = R_LoadHiResTexture(va("%s_norm", mapname), NULL, imageflags|IF_TRYBUMP|IF_NOSRGB);
			if (!TEXVALID(tex->bump) && (r_shadow_bumpscale_basetexture.ival||*imagename=='#'||gl_load24bit.ival))
			{
				qbyte *fallbackheight;
				if ((r_shadow_bumpscale_basetexture.ival||*imagename=='#') && !(basefmt&PTI_FULLMIPCHAIN))
					fallbackheight = srcdata;	//generate normalmap from assumed heights.
				else
					fallbackheight = NULL;		//disabled
				tex->bump = Image_GetTexture(va("%s_norm", imagename), subpath, imageflags|IF_TRYBUMP|IF_NOSRGB|(*imagename=='#'?IF_LINEAR:0), fallbackheight, palette, width, height, TF_HEIGHT8PAL);
			}
		}

		if (loadflags & SHADER_HASTOPBOTTOM)
		{
			if (!TEXVALID(tex->loweroverlay) && *mapname)
				tex->loweroverlay = R_LoadHiResTexture(va("%s_pants", mapname), NULL, imageflags);
			if (!TEXVALID(tex->loweroverlay))
				tex->loweroverlay = Image_GetTexture(va("%s_pants", imagename), subpath, imageflags, NULL, palette, width, height, 0);
		}
		if (loadflags & SHADER_HASTOPBOTTOM)
		{
			if (!TEXVALID(tex->upperoverlay) && *mapname)
				tex->upperoverlay = R_LoadHiResTexture(va("%s_shirt", mapname), NULL, imageflags);
			if (!TEXVALID(tex->upperoverlay))
				tex->upperoverlay = Image_GetTexture(va("%s_shirt", imagename), subpath, imageflags, NULL, palette, width, height, 0);
		}

		if (loadflags & SHADER_HASGLOSS)
		{
			if (!TEXVALID(tex->specular) && *mapname)
				tex->specular = R_LoadHiResTexture(va("%s_gloss", mapname), NULL, imageflags);
			if (!TEXVALID(tex->specular))
				tex->specular = Image_GetTexture(va("%s_gloss", imagename), subpath, imageflags, NULL, palette, width, height, 0);
		}
		
		if (tex->reflectcube)
		{
			if (!TEXVALID(tex->reflectmask) && *mapname)
				tex->reflectmask = R_LoadHiResTexture(va("%s_reflect", mapname), NULL, imageflags);
			if (!TEXVALID(tex->reflectmask))
				tex->reflectmask = Image_GetTexture(va("%s_reflect", imagename), subpath, imageflags, NULL, NULL, width, height, TF_INVALID);
		}

		if (loadflags & SHADER_HASFULLBRIGHT)
		{
			if (!TEXVALID(tex->fullbright) && *mapname)
				tex->fullbright = R_LoadHiResTexture(va("%s_luma", mapname), NULL, imageflags);
			if (!TEXVALID(tex->fullbright))
			{
				int s=-1;
				if (srcdata && !(basefmt&PTI_FULLMIPCHAIN) && (!palette || palette == host_basepal))
				for(s = width*height-1; s>=0; s--)
				{
					if (srcdata[s] >= 256-vid.fullbright)
						break;
				}
				tex->fullbright = Image_GetTexture(va("%s_luma:%s_glow", imagename,imagename), subpath, imageflags, (s>=0)?srcdata:NULL, palette, width, height, TF_TRANS8_FULLBRIGHT);
			}
		}
	}
}

void Shader_DefaultScript(parsestate_t *ps, const char *shortname, const void *args)
{
	const char *f = args;
	if (!args)
		return;
	while (*f == ' ' || *f == '\t' || *f == '\n' || *f == '\r')
		f++;
	if (*f == '{')
	{
		f++;
		Shader_ReadShader(ps, (void*)f, NULL);
	}
}

void Shader_DefaultBSPLM(parsestate_t *ps, const char *shortname, const void *args)
{
	shader_t *s = ps->s;
	char *builtin = NULL;
	if (Shader_ParseShader(ps, "defaultwall"))
		return;

	if (!builtin && r_softwarebanding && (qrenderer == QR_OPENGL || qrenderer == QR_VULKAN) && sh_config.progs_supported)
		builtin = (
				"{\n"
					"{\n"
						"program defaultwall#EIGHTBIT\n"
						"map $colourmap\n"
					"}\n"
				"}\n"
			);

	if (!builtin && r_lightmap.ival)
		builtin = (
				"{\n"
				"fte_program drawflat_wall#LM\n"
					"{\n"
						"map $lightmap\n"
						"tcgen lightmap\n"
					"}\n"
				"}\n"
			);

	if (!builtin && r_drawflat.ival)
		builtin = (
				"{\n"
					"fte_program drawflat_wall\n"
					"{\n"
						"map $lightmap\n"
						"tcgen lightmap\n"
						"rgbgen srgb $r_floorcolour\n"
					"}\n"
				"}\n"
			);


	if (!builtin && r_lightprepass)
	{
		builtin = (
			"{\n"
				"{\n"
					"fte_program lpp_wall\n"
					"map $gbuffer2\n"	//diffuse lighting info
					"map $gbuffer3\n"	//specular lighting info
				"}\n"

				//this is drawn during the initial gbuffer pass to prepare it
				"fte_bemode gbuffer\n"
				"{\n"
					"{\n"
						"fte_program lpp_depthnorm\n"
//						"map $normalmap\n"
						"tcgen base\n"
					"}\n"
				"}\n"
			"}\n"
		);
	}
	//d3d has no position-invariant. this results in all sorts of glitches, so try not to use it.
	if (!builtin && ((sh_config.progs_supported && (qrenderer == QR_OPENGL/*||qrenderer == QR_DIRECT3D9*/)) || sh_config.progs_required))
	{
			builtin = (
					"{\n"
						"fte_program defaultwall\n"
						"{\n"
							"map $diffuse\n"
						"}\n"
					"}\n"
				);
	}
	if (!builtin)
		builtin = (
				"{\n"
/*					"if $deluxmap\n"
						"{\n"
							"map $normalmap\n"
							"tcgen base\n"
							"depthwrite\n"
						"}\n"
						"{\n"
							"map $deluxmap\n"
							"tcgen lightmap\n"
						"}\n"
					"endif\n"
*///				"if !r_fullbright\n"
						"{\n"
							"map $lightmap\n"
//							"if $deluxmap\n"
//								"blendfunc gl_dst_color gl_zero\n"
//							"endif\n"
						"}\n"
//					"endif\n"
					"{\n"
						"map $diffuse\n"
						"tcgen base\n"
//						"if $deluxmap || !r_fullbright\n"
//							"blendfunc gl_dst_color gl_zero\n"
							"blendfunc filter\n"
//						"endif\n"
					"}\n"
					"if gl_fb_bmodels\n"
						"{\n"
							"map $fullbright\n"
							"blendfunc add\n"
							"depthfunc equal\n"
						"}\n"
					"endif\n"
				"}\n"
			);

	Shader_DefaultScript(ps, shortname, builtin);

	if (r_lightprepass)
		s->flags |= SHADER_HASNORMALMAP;
}

void Shader_DefaultCinematic(parsestate_t *ps, const char *shortname, const void *args)
{
	Shader_DefaultScript(ps, shortname,
		va(
			"{\n"
				"program default2d\n"
				"{\n"
					"videomap \"%s\"\n"
					"blendfunc gl_one gl_one_minus_src_alpha\n"
				"}\n"
			"}\n"
		, (const char*)args)
	);
}

/*shortname should begin with 'skybox_'*/
void Shader_DefaultSkybox(parsestate_t *ps, const char *shortname, const void *args)
{
	shader_t *s = ps->s;
	int i;
	Shader_DefaultScript(ps, shortname,
		va(
			"{\n"
				"sort sky\n"
				"surfaceparm nodlight\n"
				"skyparms %s - -\n"
			"}\n"
		, shortname+7)
	);

	for (i = 0; i < 6; i++)
	{
		if (s->skydome->farbox_textures[i] == missing_texture)
		{
			if (s->skydome)
				Z_Free(s->skydome);
			s->skydome = NULL;
			return;
		}
	}
}

char *Shader_DefaultBSPWater(parsestate_t *ps, const char *shortname, char *buffer, size_t buffersize)
{
	shader_t *s = ps->s;
	int wstyle;
	int type;
	float alpha;
	qboolean explicitalpha = false;
	cvar_t *alphavars[] = {	&r_wateralpha, &r_lavaalpha, &r_slimealpha, &r_telealpha};
	cvar_t *stylevars[] = {	&r_waterstyle, &r_lavastyle, &r_slimestyle, &r_telestyle};

	if (!strncmp(shortname, "*portal", 7))
	{
		return	"{\n"
					"portal\n"
				"}\n";
	}
	else if (!strncmp(shortname, "*lava", 5))
		type = 1;
	else if (!strncmp(shortname, "*slime", 6))
		type = 2;
	else if (!strncmp(shortname, "*tele", 5))
		type = 3;
	else
		type = 0;
	alpha = Shader_FloatArgument(s, "ALPHA");
	if (alpha)
		explicitalpha = true;
	else
	{
		if (ruleset_allow_watervis.ival)
			alpha = *alphavars[type]->string?alphavars[type]->value:alphavars[0]->value;
		else
			alpha = 1;
	}

	if (alpha <= 0)
		wstyle = -1;
	else if (r_fastturb.ival)
		wstyle = 0;
	else if (*stylevars[type]->string)
		wstyle = stylevars[type]->ival;
	else if (stylevars[0]->ival > 0)
		wstyle = stylevars[0]->ival;
	else
		wstyle = 1;

	if (wstyle > 1 && !ruleset_allow_watervis.ival)
		wstyle = 1;

	if (wstyle > 1 && !sh_config.progs_supported)
		wstyle = 1;

	//extra silly limitations
	switch(qrenderer)
	{
#ifdef GLQUAKE
	case QR_OPENGL:
		if (wstyle > 2 && !gl_config.ext_framebuffer_objects)
			wstyle = 2;
		break;
#endif
#ifdef VKQUAKE
	case QR_VULKAN:
		if (wstyle > 3)
			wstyle = 3;
		break;
#endif
	default:	//altwater not supported with other renderers
		if (wstyle > 1)
			wstyle = 1;
	}

	switch(wstyle)
	{
	case -1:	//invisible
		return (
			"{\n"
				"surfaceparm nodraw\n"
				"surfaceparm nodlight\n"
				"surfaceparm nomarks\n"
				"surfaceparm hasdiffuse\n"
			"}\n"
		);
	case -2:	//regular with r_wateralpha forced off.
		return (
			"{\n"
				"{\n"
					"program defaultwarp\n"
					"map $diffuse\n"
					"tcmod turb 0.02 0.1 0.5 0.1\n"
				"}\n"
				"surfaceparm nodlight\n"
				"surfaceparm nomarks\n"
				"surfaceparm hasdiffuse\n"
			"}\n"
		);
	case 0:	//fastturb
		return (
			"{\n"
				"{\n"
//					"program defaultfill\n"
					"map $whiteimage\n"
					"rgbgen srgb $r_fastturbcolour\n"
				"}\n"
				"surfaceparm nodlight\n"
				"surfaceparm nomarks\n"
				"surfaceparm hasdiffuse\n"
			"}\n"
		);
	default:
	case 1:	//vanilla style
		Q_snprintfz(buffer, buffersize, 
				"{\n"
					"surfaceparm nodlight\n"
					"surfaceparm nomarks\n"
					"{\n"
						"program defaultwarp%s\n"
						"map $diffuse\n"
						"tcmod turb 0.02 0.1 0.5 0.1\n"
						"if %g < 1\n"
							"alphagen const %g\n"
							"blendfunc gl_src_alpha gl_one_minus_src_alpha\n"
						"endif\n"
					"}\n"
					"surfaceparm hasdiffuse\n"
				"}\n"
				, explicitalpha?"":va("#ALPHA=%g",alpha), alpha, alpha);
		return buffer;
	case 2:	//refraction of the underwater surface, with a fresnel
		return (
			"{\n"
				"surfaceparm nodlight\n"
				"surfaceparm nomarks\n"
				"{\n"
					"program altwater#FRESNEL=4\n"
					"map $refraction\n"
					"map $null\n"//$reflection
					"map $null\n"//$ripplemap
					"map $null\n"//$refractiondepth
				"}\n"
				"surfaceparm hasdiffuse\n"
			"}\n"
		);
	case 3:	//reflections
		return (
			"{\n"
				"surfaceparm nodlight\n"
				"surfaceparm nomarks\n"
				"{\n"
					"program altwater#REFLECT#FRESNEL=4\n"
					"map $refraction\n"
					"map $reflection\n"
					"map $null\n"//$ripplemap
					"map $null\n"//$refractiondepth
				"}\n"
				"surfaceparm hasdiffuse\n"
			"}\n"
		);
	case 4:	//ripples
		return (
			"{\n"
				"surfaceparm nodlight\n"
				"surfaceparm nomarks\n"
				"{\n"
					"program altwater#RIPPLEMAP#FRESNEL=4\n"
					"map $refraction\n"
					"map $null\n"//$reflection
					"map $ripplemap\n"
					"map $null\n"//$refractiondepth
				"}\n"
				"surfaceparm hasdiffuse\n"
			"}\n"
		);
	case 5:	//ripples+reflections
		return (
			"{\n"
				"surfaceparm nodlight\n"
				"surfaceparm nomarks\n"
				"{\n"
					"program altwater#REFLECT#RIPPLEMAP#FRESNEL=4\n"
					"map $refraction\n"
					"map $reflection\n"
					"map $ripplemap\n"
					"map $null\n"//$refractiondepth
				"}\n"
				"surfaceparm hasdiffuse\n"
			"}\n"
		);
	}
}

void Shader_DefaultWaterShader(parsestate_t *ps, const char *shortname, const void *args)
{
	char tmpbuffer[2048];
	Shader_DefaultScript(ps, shortname, Shader_DefaultBSPWater(ps, shortname, tmpbuffer, sizeof(tmpbuffer)));
}
void Shader_DefaultBSPQ2(parsestate_t *ps, const char *shortname, const void *args)
{
	if (!strncmp(shortname, "sky/", 4))
	{
		Shader_DefaultScript(ps, shortname,
				"{\n"
					"sort sky\n"
					"surfaceparm nodlight\n"
					"skyparms - - -\n"
				"}\n"
			);
	}
	else if (Shader_FloatArgument(ps->s, "WARP"))//!strncmp(shortname, "warp/", 5) || !strncmp(shortname, "warp33/", 7) || !strncmp(shortname, "warp66/", 7))
	{
		char tmpbuffer[2048];
		Shader_DefaultScript(ps, shortname, Shader_DefaultBSPWater(ps, shortname, tmpbuffer, sizeof(tmpbuffer)));
	}
	else if (Shader_FloatArgument(ps->s, "FLOW"))
	{
		if (Shader_FloatArgument(ps->s, "ALPHA")) {
			Shader_DefaultScript(ps, shortname,
					"{\n"
						"{\n"
							"map $diffuse\n"
							"alphagen const $#ALPHA\n"
							"tcmod scroll -1 0\n"
							"blendfunc blend\n"
						"}\n"
					"}\n"
				);
		} else {
			Shader_DefaultScript(ps, shortname,
				"{\n"
					"{\n"
						"map $diffuse\n"
						"tcmod scroll -1 0\n"
					"}\n"
					"{\n"
						"map $lightmap\n"
						"if gl_overbright > 1\n"
						"blendfunc gl_dst_color gl_src_color\n"
						"else\n"
						"blendfunc gl_dst_color gl_zero\n"
						"endif\n"
						"depthfunc equal\n"
					"}\n"
				"}\n"
				);
		}
	}
	else if (Shader_FloatArgument(ps->s, "ALPHA"))//   !strncmp(shortname, "trans/", 6))
	{
		Shader_DefaultScript(ps, shortname,
				"{\n"
					"{\n"
						"map $diffuse\n"
						"alphagen const $#ALPHA\n"
						"blendfunc blend\n"
					"}\n"
				"}\n"
			);
	}
	else
		Shader_DefaultBSPLM(ps, shortname, args);
}

void Shader_DefaultBSPQ1(parsestate_t *ps, const char *shortname, const void *args)
{
	char *builtin = NULL;
	char tmpbuffer[2048];

	if (!strcmp(shortname, "mirror_portal"))
	{
		builtin =	"{\n"
						"portal\n"
					"}\n";
	}
	else if (r_mirroralpha.value < 1 && (!strcmp(shortname, "window02_1") || !strncmp(shortname, "mirror", 6)))
	{
		if (r_mirroralpha.value < 0)
		{
			builtin =	"{\n"
							"portal\n"
							"{\n"
								"map $diffuse\n"
								"blendfunc blend\n"
								"alphagen portal 512\n"
								"depthwrite\n"
							"}\n"
						"}\n";
		}
		else
		{
			builtin =	"{\n"
							"portal\n"
							"{\n"
								"map $diffuse\n"
								"blendfunc blend\n"
								"alphagen const $r_mirroralpha\n"
								"depthwrite\n"
							"}\n"
							"surfaceparm nodlight\n"
						"}\n";
		}

	}

	if (!builtin && (*shortname == '*' || *shortname == '!'))
	{
		builtin = Shader_DefaultBSPWater(ps, shortname, tmpbuffer, sizeof(tmpbuffer));
	}
	if (!builtin && !strncmp(shortname, "sky", 3))
	{
		//q1 sky
		/*if (r_fastsky.ival)
		{
			builtin = (
					"{\n"
						"sort sky\n"
						"{\n"
							"map $whiteimage\n"
							"rgbgen srgb $r_fastskycolour\n"
						"}\n"
						"surfaceparm nodlight\n"
					"}\n"
				);
		}*/
		/*else if (*r_skyboxname.string || *cl.skyname)
		{
			qboolean okay;
			Z_Free(s->skydome);
			s->skydome = (skydome_t *)Z_Malloc(sizeof(skydome_t));

			okay = Shader_ParseSkySides(shortname, "", s->skydome->farbox_textures);
			s->flags |= SHADER_SKY|SHADER_NODLIGHT;
			s->sort = SHADER_SORT_SKY;

			if (okay)
				return;
			builtin = NULL;
			//if the r_skybox failed to load or whatever, reset and fall through and just use the regular sky
			Shader_Reset(s);
		}*/
		if (!builtin)
			builtin = (
				"{\n"
					"sort sky\n"
					"program defaultsky\n"
					"skyparms - 512 -\n"
					/*WARNING: these values are not authentic quake, only close aproximations*/
					"{\n"
						"map $diffuse\n"
						"tcmod scale 10 10\n"
						"tcmod scroll 0.04 0.04\n"
						"depthwrite\n"
					"}\n"
					"{\n"
						"map $fullbright\n"
						"blendfunc blend\n"
						"tcmod scale 10 10\n"
						"tcmod scroll 0.02 0.02\n"
					"}\n"
				"}\n"
			);
	}
	if (!builtin && *shortname == '{')
	{
		/*alpha test*/
		if (sh_config.progs_supported)
			builtin = (
				"{\n"
					"fte_program defaultwall#MASK=0.666#MASKLT\n"
				"}\n"
			);
		else
			builtin = (
				"{\n"
			/*		"if $deluxmap\n"
						"{\n"
							"map $normalmap\n"
							"tcgen base\n"
						"}\n"
						"{\n"
							"map $deluxmap\n"
							"tcgen lightmap\n"
						"}\n"
					"endif\n"*/
					"{\n"
						"map $diffuse\n"
						"tcgen base\n"
						"alphafunc ge128\n"
					"}\n"
	//				"if $lightmap\n"
						"{\n"
							"map $lightmap\n"
							"if gl_overbright > 1\n"
							"blendfunc gl_dst_color gl_src_color\n"	//scale it up twice. will probably still get clamped, but what can you do
							"else\n"
							"blendfunc gl_dst_color gl_zero\n"
							"endif\n"
							"depthfunc equal\n"
						"}\n"
	//				"endif\n"
					"{\n"
						"map $fullbright\n"
						"blendfunc add\n"
						"depthfunc equal\n"
					"}\n"
				"}\n"
			);
	}

	/*Hack: note that halflife would normally expect you to use rendermode/renderampt*/
	if (!builtin && (!strncmp(shortname, "glass", 5)/* || !strncmp(shortname, "window", 6)*/))
	{
		/*alpha bended*/
		builtin = (
			"{\n"
				"{\n"
					"map $diffuse\n"
					"tcgen base\n"
					"blendfunc blend\n"
				"}\n"
			"}\n"
		);
	}

	if (builtin)
		Shader_DefaultScript(ps, shortname, builtin);
	else
		Shader_DefaultBSPLM(ps, shortname, args);
}

void Shader_DefaultBSPVertex(parsestate_t *ps, const char *shortname, const void *args)
{
	char *builtin = NULL;

	if (Shader_ParseShader(ps, "defaultvertexlit"))
		return;

	if (!builtin)
	{
		builtin = (
			"{\n"
				"program defaultwall#VERTEXLIT\n"
				"{\n"
					"map $diffuse\n"
					"rgbgen vertex\n"
					"alphagen vertex\n"
				"}\n"
			"}\n"
		);
	}

	Shader_DefaultScript(ps, shortname, builtin);
}
void Shader_DefaultBSPFlare(parsestate_t *ps, const char *shortname, const void *args)
{
	shader_t *s = ps->s;
	shaderpass_t *pass;
	if (Shader_ParseShader(ps, "defaultflare"))
		return;

	pass = &s->passes[0];
	pass->flags = SHADER_PASS_NOCOLORARRAY;
	pass->shaderbits |= SBITS_SRCBLEND_ONE|SBITS_DSTBLEND_ONE;
	pass->anim_frames[0] = R_LoadHiResTexture(shortname, NULL, 0);
	pass->rgbgen = RGB_GEN_VERTEX_LIGHTING;
	pass->alphagen = ALPHA_GEN_IDENTITY;
	pass->numtcmods = 0;
	pass->tcgen = TC_GEN_BASE;
	pass->numMergedPasses = 1;
	Shader_SetBlendmode(pass, NULL);

	if (!TEXVALID(pass->anim_frames[0]))
	{
		Con_DPrintf (CON_WARNING "Shader %s has a stage with no image: %s.\n", s->name, shortname );
		pass->anim_frames[0] = missing_texture;
	}

	s->numpasses = 1;
	s->numdeforms = 0;
	s->flags = SHADER_FLARE;
	s->sort = SHADER_SORT_ADDITIVE;

	s->flags |= SHADER_NODRAW;
}
void Shader_DefaultSkin(parsestate_t *ps, const char *shortname, const void *args)
{
	if (Shader_ParseShader(ps, "defaultskin"))
		return;

	if (r_softwarebanding && qrenderer == QR_OPENGL && sh_config.progs_supported)
	{
		Shader_DefaultScript(ps, shortname,
			"{\n"
				"program defaultskin#EIGHTBIT\n"
				"affine\n"
				"{\n"
					"map $colourmap\n"
				"}\n"
			"}\n"
			);
		return;
	}
	if (r_lightprepass)
	{
		Shader_DefaultScript(ps, shortname,
			"{\n"
				"{\n"
					"fte_program lpp_wall\n"
					"map $gbuffer2\n"	//diffuse lighting info
					"map $gbuffer3\n"	//specular lighting info
				"}\n"

				//this is drawn during the initial gbuffer pass to prepare it
				"fte_bemode gbuffer\n"
				"{\n"
					"{\n"
						"fte_program lpp_depthnorm\n"
						"tcgen base\n"
					"}\n"
				"}\n"
			"}\n"
		);
		return;
	}

	if (r_tessellation.ival && sh_config.progs_supported)
	{
		Shader_DefaultScript(ps, shortname,
			"{\n"
				"program defaultskin#TESS\n"
			"}\n"
			);
		return;
	}

	Shader_DefaultScript(ps, shortname,
		"{\n"
			"if $lpp\n"
				"program lpp_skin\n"
			"else\n"
				"program defaultskin\n"
			"endif\n"
			"if gl_affinemodels\n"
				"affine\n"
			"endif\n"
			"{\n"
				"map $diffuse\n"
				"rgbgen lightingDiffuse\n"
			"}\n"
			"{\n"
				"map $loweroverlay\n"
				"rgbgen bottomcolor\n"
				"blendfunc gl_src_alpha gl_one\n"
			"}\n"
			"{\n"
				"map $upperoverlay\n"
				"rgbgen topcolor\n"
				"blendfunc gl_src_alpha gl_one\n"
			"}\n"
			"{\n"
				"map $fullbright\n"
				"blendfunc add\n"
			"}\n"
		"}\n"
		);
}
void Shader_DefaultSkinShell(parsestate_t *ps, const char *shortname, const void *args)
{
	if (Shader_ParseShader(ps, "defaultskinshell"))
		return;

	Shader_DefaultScript(ps, shortname,
		"{\n"
			"sort seethrough\n"	//before blend, but after other stuff. should fix most issues with shotgun etc effects obscuring it.
//			"deformvertexes normal 1 1\n"
			//draw it with depth but no colours at all
			"{\n"
				"map $whiteimage\n"
				"maskcolor\n"
				"depthwrite\n"
			"}\n"
			//now draw it again, depthfunc = equal should fill only the near-side, avoiding any excess-brightness issues with overlapping triangles
			"{\n"
				"map $whiteimage\n"
				"rgbgen entity\n"
				"alphagen entity\n"
				"blendfunc blend\n"
			"}\n"
		"}\n"
		);
}
void Shader_Default2D(parsestate_t *ps, const char *shortname, const void *genargs)
{
	shader_t *s = ps->s;
	if (Shader_ParseShader(ps, "default2d"))
		return;
	if (sh_config.progs_supported && qrenderer != QR_DIRECT3D9 && !dpcompat_nopremulpics.ival)
	{
		//hexen2 needs premultiplied alpha to avoid looking ugly
		//but that results in problems where things are drawn with alpha not 0, so scale vertex colour by alpha in the fragment program
		Shader_DefaultScript(ps, shortname,
			"{\n"
				"affine\n"
				"nomipmaps\n"
				"program default2d#PREMUL\n"
				"{\n"
				"clampmap $diffuse\n"
				"blendfunc gl_one gl_one_minus_src_alpha\n"
				"}\n"
				"sort additive\n"
			"}\n"
			);
		TEXASSIGN(s->defaulttextures->base, R_LoadHiResTexture(s->name, genargs, IF_PREMULTIPLYALPHA|IF_UIPIC|IF_NOPICMIP|IF_NOMIPMAP|IF_CLAMP));
	}
	else
	{
		Shader_DefaultScript(ps, shortname,
			"{\n"
				"affine\n"
				"nomipmaps\n"
				"{\n"
					"clampmap $diffuse\n"
					"rgbgen vertex\n"
					"alphagen vertex\n"
					"blendfunc gl_src_alpha gl_one_minus_src_alpha\n"
				"}\n"
				"sort additive\n"
			"}\n"
			);
		TEXASSIGN(s->defaulttextures->base, R_LoadHiResTexture(s->name, genargs, IF_UIPIC|IF_NOPICMIP|IF_NOMIPMAP|IF_CLAMP));
	}
}
void Shader_PolygonShader(struct shaderparsestate_s *ps, const char *shortname, const void *args)
{
	shader_t *s = ps->s;
	Shader_DefaultScript(ps, shortname,
		"{\n"
			"if $lpp\n"
				"program lpp_skin\n"
			"else\n"
				"program defaultskin#NONORMALS\n"
			"endif\n"
			"{\n"
				"map $diffuse\n"
				"rgbgen vertex\n"
				"alphagen vertex\n"
			"}\n"
			"{\n"
				"map $fullbright\n"
				"blendfunc add\n"
			"}\n"
		"}\n"
		);

	if (!s->defaulttextures->base && (s->flags & SHADER_HASDIFFUSE))
		R_BuildDefaultTexnums(NULL, s, 0);
}

static qboolean Shader_ReadShaderTerms(parsestate_t *ps, struct scondinfo_s *cond)
{
	char *token;

	if (!ps->ptr)
		return false;

	token = COM_ParseExt (&ps->ptr, true, true);

	if ( !token[0] )
		return true;
	else if (!Shader_Conditional_Read(ps->s, cond, token, &ps->ptr))
	{
		int i;
		for (i = 0; shadermacros[i].name; i++)
		{
			if (!Q_stricmp (token, shadermacros[i].name))
			{
#define SHADER_MACRO_ARGS 8
				int argn = 0;
				char *oldptr;
				char arg[SHADER_MACRO_ARGS][256];
				char tmp[4096], *out, *in;
				//parse args until the end of the line
				while (ps->ptr)
				{
					token = COM_ParseExt(&ps->ptr, false, true);
					if ( !token[0] )
					{
						break;
					}
					if (argn <= SHADER_MACRO_ARGS)
					{
						Q_strncpyz(arg[argn], token, sizeof(arg[argn]));
						argn++;
					}
				}
				for(out = tmp, in = shadermacros[i].body; *in; )
				{
					if (out == tmp+countof(tmp)-1)
						break;
					if (*in == '%' && in[1] == '%')
						in++;	//skip doubled up percents
					else if (*in == '%')
					{	//expand an arg
						char *e;
						int i = strtol(in+1, &e, 0);
						if (e != in+1)
						{
							i--;
							if (i >= 0 && i < countof(arg))
							{
								for (in = arg[i]; *in; )
								{
									if (out == tmp+countof(tmp)-1)
										break;
									*out++ = *in++;
								}
								in = e;
								continue;
							}
						}
					}
					*out++ = *in++;
				}
				*out = 0;

				oldptr = ps->ptr;
				ps->ptr = tmp;
				Shader_ReadShaderTerms(ps, cond);
				ps->ptr = oldptr;
				return true;
			}
		}
		if (token[0] == '}')
			return false;
		else if (token[0] == '{')
			Shader_Readpass(ps);
		else if (Shader_Parsetok(ps, shaderkeys, token))
			return false;
	}
	return true;
}

//loads a shader string into an existing shader object, and finalises it and stuff
static void Shader_ReadShader(parsestate_t *ps, char *shadersource, shadercachefile_t *sourcefile)
{
	struct scondinfo_s cond = {0};
	char **savebody = ps->saveshaderbody;
	shader_t *s = ps->s;

	memset(ps, 0, sizeof(*ps));
	if (sourcefile)
	{
		ps->forcedshader = *sourcefile->forcedshadername?sourcefile->forcedshadername:NULL;
		ps->parseflags = sourcefile->parseflags;
		ps->sourcename = sourcefile->name;
	}
	else
	{
		ps->parseflags = 0;
		ps->sourcename = "<code>";
	}
	ps->specularexpscale = 1;
	ps->specularvalscale = 1;
	ps->ptr = shadersource;
	ps->s = s;

	if (!s->defaulttextures)
	{
		s->defaulttextures = Z_Malloc(sizeof(*s->defaulttextures));
		s->numdefaulttextures = 0;
	}

// set defaults
	s->flags = SHADER_CULL_FRONT;

	while (Shader_ReadShaderTerms(ps, &cond))
	{
	}

	if (cond.depth)
	{
		Con_Printf("if statements without endif in shader %s\n", s->name);
	}

	Shader_Finish (ps);

	//querying the shader body often requires generating the shader, which then gets parsed.
	if (savebody)
	{
		size_t l = ps->ptr?ps->ptr - shadersource:0;
		Z_Free(*savebody);
		*savebody = BZ_Malloc(l+1);
		(*savebody)[l] = 0;
		memcpy(*savebody, shadersource, l);
	}
}

static qboolean Shader_ParseShader(parsestate_t *ps, char *parsename)
{
	size_t offset = 0, length;
	char *buf = NULL;
	shadercachefile_t *sourcefile = NULL;
	char *file;
	const char *token;

	if (!strchr(parsename, ':'))
	{
		//if the named shader is a .shader file then just directly load it.
		token = COM_GetFileExtension(parsename, NULL);
		if (!strcmp(token, ".mat") || !*token)
		{
			char shaderfile[MAX_QPATH];
			if (!*token)
			{
				Q_snprintfz(shaderfile, sizeof(shaderfile), "%s.mat", parsename);
				file = COM_LoadTempMoreFile(shaderfile, &length);
			}
			else
				file = COM_LoadTempMoreFile(parsename, &length);
			if (file)
			{
				Shader_Reset(ps->s);
				token = COM_ParseExt (&file, true, false);	//we need to skip over the leading {.
				if (*token != '{')
					token = COM_ParseExt (&file, true, false);	//try again, in case we found some legacy name.
				if (*token == '{')
				{
					Shader_ReadShader(ps, file, NULL);
					return true;
				}
				else
					Con_Printf("file %s.shader does not appear to contain a shader\n", shaderfile);
			}
		}
	}

	if (Shader_LocateSource(parsename, &buf, &length, &offset, &sourcefile))
	{
		// the shader is in the shader scripts
		if (buf && offset < length )
		{
			file = buf + offset;
			token = COM_ParseExt (&file, true, true);
			if ( !file || token[0] != '{' )
			{
				FS_FreeFile(buf);
				return false;
			}

			Shader_Reset(ps->s);

			Shader_ReadShader(ps, file, sourcefile);

			return true;
		}
	}

	return false;
}
void R_UnloadShader(shader_t *shader)
{
	if (!shader)
		return;
	if (shader->uses <= 0)
	{
		Con_Printf("Shader double free (%s %i)\n", shader->name, shader->usageflags);
		return;
	}
	if (--shader->uses == 0)
		Shader_Free(shader);
}
static shader_t *R_LoadShader (const char *name, unsigned int usageflags, shader_gen_t *defaultgen, const char *genargs)
{
	int i, f = -1;
	char cleanname[MAX_QPATH];
	shader_t *s;

	if (!*name)
		name = "gfx/unspecified";

	COM_AssertMainThread("R_LoadShader");

	Q_strncpyz(cleanname, name, sizeof(cleanname));
	COM_CleanUpPath(cleanname);

	// check the hash first
	s = Hash_Get(&shader_active_hash, cleanname);
	while (s)
	{
		//make sure the same texture can be used as either a lightmap or vertexlit shader
		//if it has an explicit shader overriding it then that still takes precidence. we might just have multiple copies of it.
		//q3 has a separate (internal) shader for every lightmap.
		if (!((s->usageflags ^ usageflags) & SUF_LIGHTMAP))
		{
			if (!s->uses)
				break;
			s->uses++;
			return s;
		}
		s = Hash_GetNext(&shader_active_hash, cleanname, s);
	}

	// not loaded, find a free slot
	for (i = 0; i < r_numshaders; i++)
	{
		if (!r_shaders[i] || !r_shaders[i]->uses)
		{
			if ( f == -1 )	// free shader
			{
				f = i;
				break;
			}
		}
	}

	if (f == -1)
	{
		shader_t **n;
		int nm;	
		if (strlen(cleanname) >= sizeof(s->name))
		{
			Sys_Error( "R_LoadShader: Shader name too long.");
			return NULL;
		}
		f = r_numshaders;
		if (f == r_maxshaders)
		{
			if (!r_maxshaders)
				Sys_Error( "R_LoadShader: shader system not inited.");

			nm = r_maxshaders * 2;
			n = realloc(r_shaders, nm*sizeof(*n));
			if (!n)
			{
				Sys_Error( "R_LoadShader: Shader limit exceeded.");
				return NULL;
			}
			memset(n+r_maxshaders, 0, (nm - r_maxshaders)*sizeof(*n));
			r_shaders = n;
			r_maxshaders = nm;
		}
	}

	{
		parsestate_t ps;
		char shortname[MAX_QPATH];
		char *argsstart;
		s = r_shaders[f];
		if (!s)
		{
			s = r_shaders[f] = Z_Malloc(sizeof(*s));
		}
		ps.saveshaderbody = NULL;
		ps.s = s;
		s->id = f;
		if (r_numshaders < f+1)
			r_numshaders = f+1;

		if (!s->defaulttextures)
			s->defaulttextures = Z_Malloc(sizeof(*s->defaulttextures));
		else
			memset(s->defaulttextures, 0, sizeof(*s->defaulttextures));
		s->numdefaulttextures = 0;
		Q_strncpyz(s->name, cleanname, sizeof(s->name));
		s->usageflags = usageflags;
		s->generator = defaultgen;
		s->width = 0;
		s->height = 0;
		s->uses = 1;
		if (genargs)
			s->genargs = strdup(genargs);
		else
			s->genargs = NULL;

		//now determine the 'short name'. ie: the shader that is loaded off disk (no args, no extension)
		argsstart = strchr(cleanname, '#');
		if (argsstart)
			*argsstart = 0;
		COM_StripExtension (cleanname, shortname, sizeof(shortname));

		if (ruleset_allow_shaders.ival && !(usageflags & SUR_FORCEFALLBACK))
		{
			if (sh_config.shadernamefmt)
			{
				char drivername[MAX_QPATH];
				Q_snprintfz(drivername, sizeof(drivername), sh_config.shadernamefmt, cleanname);
				if (Shader_ParseShader(&ps, drivername))
					return s;
			}
			if (Shader_ParseShader(&ps, cleanname))
				return s;
			if (strcmp(cleanname, shortname))
				if (Shader_ParseShader(&ps, shortname))
					return s;
		}

		// make a default shader

		if (s->generator)
		{
			Shader_Regenerate(&ps, shortname);
			return s;
		}
		else
		{
			Shader_Free(s);
		}
	}
	return NULL;
}

#ifdef _DEBUG
static char *Shader_DecomposePass(char *o, shaderpass_t *p, qboolean simple)
{
	if (!simple)
	{
		switch(p->rgbgen)
		{
		default: sprintf(o, "RGB_GEN_? "); break;
		case RGB_GEN_ENTITY: sprintf(o, "RGB_GEN_ENTITY "); break;
		case RGB_GEN_ONE_MINUS_ENTITY: sprintf(o, "RGB_GEN_ONE_MINUS_ENTITY "); break;
		case RGB_GEN_VERTEX_LIGHTING: sprintf(o, "RGB_GEN_VERTEX_LIGHTING "); break;
		case RGB_GEN_VERTEX_EXACT: sprintf(o, "RGB_GEN_VERTEX_EXACT "); break;
		case RGB_GEN_ONE_MINUS_VERTEX: sprintf(o, "RGB_GEN_ONE_MINUS_VERTEX "); break;
		case RGB_GEN_IDENTITY_LIGHTING: sprintf(o, "RGB_GEN_IDENTITY_LIGHTING "); break;
		case RGB_GEN_IDENTITY_OVERBRIGHT: sprintf(o, "RGB_GEN_IDENTITY_OVERBRIGHT "); break;
		case RGB_GEN_IDENTITY: sprintf(o, "RGB_GEN_IDENTITY "); break;
		case RGB_GEN_CONST: sprintf(o, "RGB_GEN_CONST "); break;
		case RGB_GEN_ENTITY_LIGHTING_DIFFUSE: sprintf(o, "RGB_GEN_ENTITY_LIGHTING_DIFFUSE "); break;
		case RGB_GEN_LIGHTING_DIFFUSE: sprintf(o, "RGB_GEN_LIGHTING_DIFFUSE "); break;
		case RGB_GEN_WAVE: sprintf(o, "RGB_GEN_WAVE "); break;
		case RGB_GEN_TOPCOLOR: sprintf(o, "RGB_GEN_TOPCOLOR "); break;
		case RGB_GEN_BOTTOMCOLOR: sprintf(o, "RGB_GEN_BOTTOMCOLOR "); break;
		case RGB_GEN_UNKNOWN: sprintf(o, "RGB_GEN_UNKNOWN "); break;
		}
		o+=strlen(o);
		sprintf(o, "\n"); o+=strlen(o);

		switch(p->alphagen)
		{
		default: sprintf(o, "ALPHA_GEN_? "); break;
		case ALPHA_GEN_ENTITY: sprintf(o, "ALPHA_GEN_ENTITY "); break;
		case ALPHA_GEN_WAVE: sprintf(o, "ALPHA_GEN_WAVE "); break;
		case ALPHA_GEN_PORTAL: sprintf(o, "ALPHA_GEN_PORTAL "); break;
		case ALPHA_GEN_SPECULAR: sprintf(o, "ALPHA_GEN_SPECULAR "); break;
		case ALPHA_GEN_IDENTITY: sprintf(o, "ALPHA_GEN_IDENTITY "); break;
		case ALPHA_GEN_VERTEX: sprintf(o, "ALPHA_GEN_VERTEX "); break;
		case ALPHA_GEN_CONST: sprintf(o, "ALPHA_GEN_CONST "); break;
		}
		o+=strlen(o);
		sprintf(o, "\n"); o+=strlen(o);
	}

	if (p->prog)
	{
		sprintf(o, "program %s\n", p->prog->name);
		o+=strlen(o);
	}

	if (p->shaderbits & SBITS_MISC_DEPTHWRITE)		{	sprintf(o, "SBITS_MISC_DEPTHWRITE\n"); o+=strlen(o); }
	if (p->shaderbits & SBITS_MISC_NODEPTHTEST)		{	sprintf(o, "SBITS_MISC_NODEPTHTEST\n"); o+=strlen(o); }
	else switch (p->shaderbits & SBITS_DEPTHFUNC_BITS)
	{
	case SBITS_DEPTHFUNC_EQUAL:			sprintf(o, "depthfunc equal\n");	break;
	case SBITS_DEPTHFUNC_CLOSER:		sprintf(o, "depthfunc less\n");		break;
	case SBITS_DEPTHFUNC_CLOSEREQUAL:	sprintf(o, "depthfunc lequal\n");	break;
	case SBITS_DEPTHFUNC_FURTHER:		sprintf(o, "depthfunc greater\n");	break;
	}
	if (p->shaderbits & SBITS_TESSELLATION)			{	sprintf(o, "SBITS_TESSELLATION\n"); o+=strlen(o); }
	if (p->shaderbits & SBITS_AFFINE)				{	sprintf(o, "SBITS_AFFINE\n"); o+=strlen(o); }
	if (p->shaderbits & SBITS_MASK_BITS)			{	sprintf(o, "SBITS_MASK_BITS\n"); o+=strlen(o); }

	if (p->shaderbits & SBITS_BLEND_BITS)
	{
		sprintf(o, "blendfunc"); 
		o+=strlen(o);
		switch(p->shaderbits & SBITS_SRCBLEND_BITS)
		{
		case SBITS_SRCBLEND_NONE:							sprintf(o, " SBITS_SRCBLEND_NONE"); break;
		case SBITS_SRCBLEND_ZERO:							sprintf(o, " SBITS_SRCBLEND_ZERO"); break;
		case SBITS_SRCBLEND_ONE:							sprintf(o, " SBITS_SRCBLEND_ONE"); break;
		case SBITS_SRCBLEND_DST_COLOR:						sprintf(o, " SBITS_SRCBLEND_DST_COLOR"); break;
		case SBITS_SRCBLEND_ONE_MINUS_DST_COLOR:			sprintf(o, " SBITS_SRCBLEND_ONE_MINUS_DST_COLOR"); break;
		case SBITS_SRCBLEND_SRC_ALPHA:						sprintf(o, " SBITS_SRCBLEND_SRC_ALPHA"); break;
		case SBITS_SRCBLEND_ONE_MINUS_SRC_ALPHA:			sprintf(o, " SBITS_SRCBLEND_ONE_MINUS_SRC_ALPHA"); break;
		case SBITS_SRCBLEND_DST_ALPHA:						sprintf(o, " SBITS_SRCBLEND_DST_ALPHA"); break;
		case SBITS_SRCBLEND_ONE_MINUS_DST_ALPHA:			sprintf(o, " SBITS_SRCBLEND_ONE_MINUS_DST_ALPHA"); break;
		case SBITS_SRCBLEND_SRC_COLOR_INVALID:				sprintf(o, " SBITS_SRCBLEND_SRC_COLOR_INVALID"); break;
		case SBITS_SRCBLEND_ONE_MINUS_SRC_COLOR_INVALID:	sprintf(o, " SBITS_SRCBLEND_ONE_MINUS_SRC_COLOR_INVALID"); break;
		case SBITS_SRCBLEND_ALPHA_SATURATE:					sprintf(o, " SBITS_SRCBLEND_ALPHA_SATURATE"); break;
		default:											sprintf(o, " SBITS_SRCBLEND_INVALID"); break;
		}
		o+=strlen(o);
		switch(p->shaderbits & SBITS_DSTBLEND_BITS)
		{
		case SBITS_DSTBLEND_NONE:							sprintf(o, " SBITS_DSTBLEND_NONE"); break;
		case SBITS_DSTBLEND_ZERO:							sprintf(o, " SBITS_DSTBLEND_ZERO"); break;
		case SBITS_DSTBLEND_ONE:							sprintf(o, " SBITS_DSTBLEND_ONE"); break;
		case SBITS_DSTBLEND_DST_COLOR_INVALID:				sprintf(o, " SBITS_DSTBLEND_DST_COLOR_INVALID"); break;
		case SBITS_DSTBLEND_ONE_MINUS_DST_COLOR_INVALID:	sprintf(o, " SBITS_DSTBLEND_ONE_MINUS_DST_COLOR_INVALID"); break;
		case SBITS_DSTBLEND_SRC_ALPHA:						sprintf(o, " SBITS_DSTBLEND_SRC_ALPHA"); break;
		case SBITS_DSTBLEND_ONE_MINUS_SRC_ALPHA:			sprintf(o, " SBITS_DSTBLEND_ONE_MINUS_SRC_ALPHA"); break;
		case SBITS_DSTBLEND_DST_ALPHA:						sprintf(o, " SBITS_DSTBLEND_DST_ALPHA"); break;
		case SBITS_DSTBLEND_ONE_MINUS_DST_ALPHA:			sprintf(o, " SBITS_DSTBLEND_ONE_MINUS_DST_ALPHA"); break;
		case SBITS_DSTBLEND_SRC_COLOR:						sprintf(o, " SBITS_DSTBLEND_SRC_COLOR"); break;
		case SBITS_DSTBLEND_ONE_MINUS_SRC_COLOR:			sprintf(o, " SBITS_DSTBLEND_ONE_MINUS_SRC_COLOR"); break;
		case SBITS_DSTBLEND_ALPHA_SATURATE_INVALID:			sprintf(o, " SBITS_DSTBLEND_ALPHA_SATURATE_INVALID"); break;
		default:											sprintf(o, " SBITS_DSTBLEND_INVALID"); break;
		}
		o+=strlen(o);

		sprintf(o, "\n"); 
		o+=strlen(o);
	}


	switch(p->shaderbits & SBITS_ATEST_BITS)
	{
	case SBITS_ATEST_NONE:	break;
	case SBITS_ATEST_GE128: sprintf(o, "SBITS_ATEST_GE128\n"); break;
	case SBITS_ATEST_LT128: sprintf(o, "SBITS_ATEST_LT128\n"); break;
	case SBITS_ATEST_GT0: sprintf(o, "SBITS_ATEST_GT0\n"); break;
	}
	o+=strlen(o);

	return o;
}
static void Shader_DecomposeSubPassMap(char *o, shader_t *s, char *name, texid_t tex)
{
	if (tex)
	{
		unsigned int flags = tex->flags;
		sprintf(o, "%s \"%s\" %ix%i%s %s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s%s", name, tex->ident, tex->width, tex->height,
			(tex->status == TEX_FAILED)?" FAILED":"",
			Image_FormatName(tex->format),
			(flags&IF_CLAMP)?" clamp":"",
			(flags&IF_NOMIPMAP)?" nomipmap":"",
			(flags&IF_NEAREST)?" nearest":"",
			(flags&IF_LINEAR)?" linear":"",
			(flags&IF_UIPIC)?" uipic":"",
			(flags&IF_SRGB)?" srgb":"",

			(flags&IF_NOPICMIP)?" nopicmip":"",
			(flags&IF_NOALPHA)?" noalpha":"",
			(flags&IF_NOGAMMA)?" noalpha":"",
			(flags&IF_TEXTYPEMASK)?" non-2d":"",
			(flags&IF_MIPCAP)?"":" nomipcap",
			(flags&IF_PREMULTIPLYALPHA)?" premultiply":"",

			(flags&IF_NOSRGB)?" nosrgb":"",

			(flags&IF_PALETTIZE)?" palettize":"",
			(flags&IF_NOPURGE)?" nopurge":"",
			(flags&IF_HIGHPRIORITY)?" highpri":"",
			(flags&IF_LOWPRIORITY)?" lowpri":"",
			(flags&IF_LOADNOW)?" loadnow":"",
			(flags&IF_TRYBUMP)?" trybump":"",
			(flags&IF_RENDERTARGET)?" rendertarget":"",
			(flags&IF_EXACTEXTENSION)?" exactext":"",
			(flags&IF_NOREPLACE)?" noreplace":"",
			(flags&IF_NOWORKER)?" noworker":""
			);
	}
	else
		sprintf(o, "%s (%s)", name, "UNDEFINED");
}
static char *Shader_DecomposeSubPass(char *o, shader_t *s, shaderpass_t *p, qboolean simple)
{
	int i;
	if (!simple)
	{
		switch(p->tcgen)
		{
		default: sprintf(o, "TC_GEN_? "); break;
		case TC_GEN_BASE: sprintf(o, "TC_GEN_BASE "); break;
		case TC_GEN_LIGHTMAP: sprintf(o, "TC_GEN_LIGHTMAP "); break;
		case TC_GEN_ENVIRONMENT: sprintf(o, "TC_GEN_ENVIRONMENT "); break;
		case TC_GEN_DOTPRODUCT: sprintf(o, "TC_GEN_DOTPRODUCT "); break;
		case TC_GEN_VECTOR: sprintf(o, "TC_GEN_VECTOR "); break;
		case TC_GEN_NORMAL: sprintf(o, "TC_GEN_NORMAL "); break;
		case TC_GEN_SVECTOR: sprintf(o, "TC_GEN_SVECTOR "); break;
		case TC_GEN_TVECTOR: sprintf(o, "TC_GEN_TVECTOR "); break;
		case TC_GEN_SKYBOX: sprintf(o, "TC_GEN_SKYBOX "); break;
		case TC_GEN_WOBBLESKY: sprintf(o, "TC_GEN_WOBBLESKY "); break;
		case TC_GEN_REFLECT: sprintf(o, "TC_GEN_REFLECT "); break;
		case TC_GEN_UNSPECIFIED: sprintf(o, "TC_GEN_UNSPECIFIED "); break;
		}
		o+=strlen(o);
		sprintf(o, "\n"); o+=strlen(o);

		for (i = 0; i < p->numtcmods; i++)
		{
			switch(p->tcmods[i].type)
			{
				default: sprintf(o, "TCMOD_GEN_? "); break;
				case SHADER_TCMOD_NONE: sprintf(o, "SHADER_TCMOD_NONE "); break;
				case SHADER_TCMOD_SCALE: sprintf(o, "SHADER_TCMOD_SCALE "); break;
				case SHADER_TCMOD_SCROLL: sprintf(o, "SHADER_TCMOD_SCROLL "); break;
				case SHADER_TCMOD_STRETCH: sprintf(o, "SHADER_TCMOD_STRETCH "); break;
				case SHADER_TCMOD_ROTATE: sprintf(o, "SHADER_TCMOD_ROTATE "); break;
				case SHADER_TCMOD_TRANSFORM: sprintf(o, "SHADER_TCMOD_TRANSFORM "); break;
				case SHADER_TCMOD_TURB: sprintf(o, "SHADER_TCMOD_TURB "); break;
				case SHADER_TCMOD_PAGE: sprintf(o, "SHADER_TCMOD_PAGE "); break;
			}
			o+=strlen(o);
			sprintf(o, "\n"); o+=strlen(o);
		}


		switch(p->blendmode)
		{
		default: sprintf(o, "PBM_? "); break;
		case PBM_MODULATE: sprintf(o, "PBM_MODULATE "); break;
		case PBM_OVERBRIGHT: sprintf(o, "PBM_OVERBRIGHT "); break;
		case PBM_DECAL: sprintf(o, "PBM_DECAL "); break;
		case PBM_ADD: sprintf(o, "PBM_ADD "); break;
		case PBM_DOTPRODUCT: sprintf(o, "PBM_DOTPRODUCT "); break;
		case PBM_REPLACE: sprintf(o, "PBM_REPLACE "); break;
		case PBM_REPLACELIGHT: sprintf(o, "PBM_REPLACELIGHT "); break;
		case PBM_MODULATE_PREV_COLOUR: sprintf(o, "PBM_MODULATE_PREV_COLOUR "); break;
		}
		o+=strlen(o);
	}

	switch(p->texgen)
	{
	default: sprintf(o, "T_GEN_? "); break;
	case T_GEN_SINGLEMAP:		Shader_DecomposeSubPassMap(o, s, "map", p->anim_frames[0]);	break;
	case T_GEN_ANIMMAP:			Shader_DecomposeSubPassMap(o, s, "animmap", p->anim_frames[0]);	break;
#ifdef HAVE_MEDIA_DECODER
	case T_GEN_VIDEOMAP:		Shader_DecomposeSubPassMap(o, s, "videomap", Media_UpdateForShader(p->cin)); break;
#endif
	case T_GEN_CUBEMAP:			Shader_DecomposeSubPassMap(o, s, "map $cubemap", p->anim_frames[0]); break;
	case T_GEN_3DMAP:			Shader_DecomposeSubPassMap(o, s, "map $3dmap", p->anim_frames[0]); break;
	case T_GEN_LIGHTMAP:		sprintf(o, "map $lightmap "); break;
	case T_GEN_DELUXMAP:		sprintf(o, "map $deluxemap "); break;
	case T_GEN_SHADOWMAP:		sprintf(o, "map $shadowmap "); break;
	case T_GEN_LIGHTCUBEMAP: 	sprintf(o, "map $lightcubemap "); break;
	case T_GEN_DIFFUSE:			Shader_DecomposeSubPassMap(o, s, "map $diffuse", s->defaulttextures[0].base); break;
	case T_GEN_NORMALMAP:		Shader_DecomposeSubPassMap(o, s, "map $normalmap", s->defaulttextures[0].bump); break;
	case T_GEN_SPECULAR:		Shader_DecomposeSubPassMap(o, s, "map $specular", s->defaulttextures[0].specular); break;
	case T_GEN_UPPEROVERLAY:	Shader_DecomposeSubPassMap(o, s, "map $upper", s->defaulttextures[0].upperoverlay); break;
	case T_GEN_LOWEROVERLAY:	Shader_DecomposeSubPassMap(o, s, "map $lower", s->defaulttextures[0].loweroverlay); break;
	case T_GEN_FULLBRIGHT:		Shader_DecomposeSubPassMap(o, s, "map $fullbright", s->defaulttextures[0].fullbright); break;
	case T_GEN_PALETTED:		Shader_DecomposeSubPassMap(o, s, "map $paletted", s->defaulttextures[0].paletted); break;
	case T_GEN_REFLECTCUBE:		Shader_DecomposeSubPassMap(o, s, "map $reflectcube", s->defaulttextures[0].reflectcube); break;
	case T_GEN_REFLECTMASK:		Shader_DecomposeSubPassMap(o, s, "map $reflectmask", s->defaulttextures[0].reflectmask); break;
	case T_GEN_DISPLACEMENT:	Shader_DecomposeSubPassMap(o, s, "map $displacement", s->defaulttextures[0].displacement); break;
	case T_GEN_OCCLUSION:		Shader_DecomposeSubPassMap(o, s, "map $occlusion", s->defaulttextures[0].occlusion); break;
	case T_GEN_CURRENTRENDER:	sprintf(o, "map $currentrender "); break;
	case T_GEN_SOURCECOLOUR:	sprintf(o, "map $sourcecolour"); break;
	case T_GEN_SOURCEDEPTH:		sprintf(o, "map $sourcedepth"); break;
	case T_GEN_REFLECTION:		sprintf(o, "map $reflection"); break;
	case T_GEN_REFRACTION:		sprintf(o, "map $refraction"); break;
	case T_GEN_REFRACTIONDEPTH:	sprintf(o, "map $refractiondepth"); break;
	case T_GEN_RIPPLEMAP:		sprintf(o, "map $ripplemap"); break;
	case T_GEN_SOURCECUBE:		sprintf(o, "map $sourcecube"); break;
	case T_GEN_GBUFFERCASE:		sprintf(o, "map $gbuffer%i ",p->texgen-T_GEN_GBUFFER0); break;
	}
	o+=strlen(o);

	sprintf(o, "\n"); o+=strlen(o);
	return o;
}
char *Shader_Decompose(shader_t *s)
{
	static char decomposebuf[32768];
	char *o = decomposebuf;
	shaderpass_t *p;
	unsigned int i, j;

	sprintf(o, "\n<---\n"); o+=strlen(o);

	switch (s->sort)
	{
	default:						sprintf(o, "sort %i\n", s->sort); break;
	case SHADER_SORT_NONE:			sprintf(o, "sort %i (SHADER_SORT_NONE)\n", s->sort); break;
	case SHADER_SORT_RIPPLE:		sprintf(o, "sort %i (SHADER_SORT_RIPPLE)\n", s->sort); break;
	case SHADER_SORT_DEFERREDLIGHT:	sprintf(o, "sort %i (SHADER_SORT_DEFERREDLIGHT)\n", s->sort); break;
	case SHADER_SORT_PORTAL:		sprintf(o, "sort %i (SHADER_SORT_PORTAL)\n", s->sort); break;
	case SHADER_SORT_SKY:			sprintf(o, "sort %i (SHADER_SORT_SKY)\n", s->sort); break;
	case SHADER_SORT_OPAQUE:		sprintf(o, "sort %i (SHADER_SORT_OPAQUE)\n", s->sort); break;
	case SHADER_SORT_DECAL:			sprintf(o, "sort %i (SHADER_SORT_DECAL)\n", s->sort); break;
	case SHADER_SORT_SEETHROUGH:	sprintf(o, "sort %i (SHADER_SORT_SEETHROUGH)\n", s->sort); break;
	case SHADER_SORT_BANNER:		sprintf(o, "sort %i (SHADER_SORT_BANNER)\n", s->sort); break;
	case SHADER_SORT_UNDERWATER:	sprintf(o, "sort %i (SHADER_SORT_UNDERWATER)\n", s->sort); break;
	case SHADER_SORT_BLEND:			sprintf(o, "sort %i (SHADER_SORT_BLEND)\n", s->sort); break;
	case SHADER_SORT_ADDITIVE:		sprintf(o, "sort %i (SHADER_SORT_ADDITIVE)\n", s->sort); break;
	case SHADER_SORT_NEAREST:		sprintf(o, "sort %i (SHADER_SORT_NEAREST)\n", s->sort); break;
	}
	o+=strlen(o);

	if (s->prog)
	{
		sprintf(o, "program %s\n", s->prog->name);
		o+=strlen(o);

		p = s->passes;
		o = Shader_DecomposePass(o, p, true);
		for (j = 0; j < s->numpasses; j++)
			o = Shader_DecomposeSubPass(o, s, p+j, true);
	}
	else
	{
		for (i = 0; i < s->numpasses; i+= p->numMergedPasses)
		{
			p = &s->passes[i];
			sprintf(o, "{\n"); o+=strlen(o);

			o = Shader_DecomposePass(o, p, false);
			for (j = 0; j < p->numMergedPasses; j++)
				o = Shader_DecomposeSubPass(o, s, p+j, !!p->prog);
			sprintf(o, "}\n"); o+=strlen(o);
		}
	}
	sprintf(o, "--->\n"); o+=strlen(o);
	return decomposebuf;
}
#endif

char *Shader_GetShaderBody(shader_t *s, char *fname, size_t fnamesize)
{
	parsestate_t ps;
	char *adr, *parsename=NULL, *argsstart;
	char cleanname[MAX_QPATH];
	char shortname[MAX_QPATH];
	char drivername[MAX_QPATH];
	int oldsort;
	qboolean resort = false;

	if (!s || !s->uses)
		return NULL;
	ps.s = s;

	adr = Z_StrDup("UNKNOWN BODY");
	ps.saveshaderbody = &adr;

	strcpy(cleanname, s->name);
	argsstart = strchr(cleanname, '#');
	if (argsstart)
		*argsstart = 0;
	COM_StripExtension (cleanname, shortname, sizeof(shortname));
	if (ruleset_allow_shaders.ival && !(s->usageflags & SUR_FORCEFALLBACK))
	{
		if (sh_config.shadernamefmt)
		{
			Q_snprintfz(drivername, sizeof(drivername), sh_config.shadernamefmt, cleanname);
			if (!parsename && Shader_ParseShader(&ps, drivername))
				parsename = drivername;
		}
		if (!parsename && Shader_ParseShader(&ps, cleanname))
			parsename = cleanname;
		if (!parsename && Shader_ParseShader(&ps, shortname))
			parsename = shortname;
	}
	if (!parsename && s->generator)
	{
		oldsort = s->sort;
		Shader_Regenerate(&ps, shortname);

		if (s->sort != oldsort)
			resort = true;
	}

	if (resort)
	{
		Mod_ResortShaders();
	}

	if (fnamesize)
	{
		*fname = 0;

		if (parsename)
		{
			unsigned int key;
			shadercache_t *cache;
			key = Hash_Key (parsename, HASH_SIZE);
			cache = shader_hash[key];
			for ( ; cache; cache = cache->hash_next)
			{
				if (!Q_stricmp (cache->name, parsename))
				{
					char *c, *stop;
					int line = 1;
					//okay, this is the shader we're looking for, we know where it came from too, so there's handy.
					//figure out the line index now, by just counting the \ns up to the offset
					for (c = cache->source->data, stop = c+cache->offset; c < stop; c++)
					{
						if (*c == '\n')
							line++;
					}
					Q_snprintfz(fname, fnamesize, "%s:%i", cache->source->name, line);
					break;
				}
			}

			if (!strchr(parsename, ':'))
			{
				//if the named shader is a .shader file then just directly load it.
				const char *token = COM_GetFileExtension(parsename, NULL);
				if (!strcmp(token, ".shader") || !*token)
				{
					char shaderfile[MAX_QPATH];
					if (!*token)
					{
						Q_snprintfz(shaderfile, sizeof(shaderfile), "%s.shader", parsename);
						if (COM_FCheckExists(shaderfile))
							Q_snprintfz(fname, fnamesize, "%s:%i", shaderfile, 1);
					}
					else if (COM_FCheckExists(parsename))
						Q_snprintfz(fname, fnamesize, "%s:%i", parsename, 1);
				}
			}
		}
	}

#ifdef _DEBUG
	{
		char *add, *ret;
		add = Shader_Decompose(s);
		if (*add)
		{
			ret = Z_Malloc(strlen(add) + strlen(adr) + 1);
			strcpy(ret, adr);
			strcpy(ret + strlen(ret), add);
			Z_Free(adr);
			adr = ret;
		}
	}
#endif

	return adr;
}

void Shader_ShowShader_f(void)
{
	char *sourcename = Cmd_Argv(1);
	shader_t *o = R_LoadShader(sourcename, SUF_NONE, NULL, NULL);
	if (!o)
		o = R_LoadShader(sourcename, SUF_LIGHTMAP, NULL, NULL);
	if (!o)
		o = R_LoadShader(sourcename, SUF_2D, NULL, NULL);
	if (o)
	{
		char fname[256];
		char *body = Shader_GetShaderBody(o, fname, sizeof(fname));
		if (body)
		{
			Con_Printf("^h(%s)^h\n%s\n{%s\n", fname, o->name, body);
			Z_Free(body);
		}
		else
		{
			Con_Printf("Shader \"%s\" is not in use\n", o->name);
		}
	}
	else
		Con_Printf("Shader \"%s\" is not loaded\n", sourcename);
}

void Shader_TouchTextures(void)
{
	int i, j, k;
	shader_t *s;
	shaderpass_t *p;
	texnums_t *t;
	for (i = 0; i < r_numshaders; i++)
	{
		s = r_shaders[i];
		if (!s || !s->uses)
			continue;

		for (j = 0; j < s->numpasses; j++)
		{
			p = &s->passes[j];
			for (k = 0; k < countof(p->anim_frames); k++)
				if (p->anim_frames[k])
					p->anim_frames[k]->regsequence = r_regsequence;
		}
		for (j = 0; j < max(1,s->numdefaulttextures); j++)
		{
			t = &s->defaulttextures[j];
			if (t->base)
				t->base->regsequence = r_regsequence;
			if (t->bump)
				t->bump->regsequence = r_regsequence;
			if (t->fullbright)
				t->fullbright->regsequence = r_regsequence;
			if (t->specular)
				t->specular->regsequence = r_regsequence;
			if (t->upperoverlay)
				t->upperoverlay->regsequence = r_regsequence;
			if (t->loweroverlay)
				t->loweroverlay->regsequence = r_regsequence;
		}
	}
}

void Shader_DoReload(void)
{
	shader_t *s;
	unsigned int i;
	char shortname[MAX_QPATH];
	char cleanname[MAX_QPATH], *argsstart;
	int oldsort;
	qboolean resort = false;
	parsestate_t ps;

	//don't spam shader reloads while we're connecting, as that's just wasteful.
	if (cls.state && cls.state < ca_active)
		return;
	if (!r_shaders)
		return;	//err, not ready yet

	if (shader_rescan_needed)
	{
		Shader_FlushCache();

		if (ruleset_allow_shaders.ival)
		{
			COM_EnumerateFiles("materials/*.mtr", Shader_InitCallback_Doom3, NULL);
			COM_EnumerateFiles("shaders/*.shader", Shader_InitCallback, NULL);
			COM_EnumerateFiles("scripts/*.shader", Shader_InitCallback, NULL);
			COM_EnumerateFiles("scripts/*.rscript", Shader_InitCallback, NULL);
		}

		shader_reload_needed = true;
		shader_rescan_needed = false;

		Con_DPrintf("Rescanning shaders\n");
	}
	else
	{
		if (!shader_reload_needed)
			return;
		Con_DPrintf("Reloading shaders\n");
	}
	shader_reload_needed = false;
	R2D_ImageColours(1,1,1,1);
	TRACE(("Reloading generics\n"));
	Shader_ReloadGenerics();

	for (i = 0; i < r_numshaders; i++)
	{
		s = r_shaders[i];
		if (!s || !s->uses)
			continue;
		ps.s = s;
		ps.saveshaderbody = NULL;

		strcpy(cleanname, s->name);
		argsstart = strchr(cleanname, '#');
		if (argsstart)
			*argsstart = 0;
		COM_StripExtension (cleanname, shortname, sizeof(shortname));
		TRACE(("reparsing %s\n", s->name));
		if (ruleset_allow_shaders.ival && !(s->usageflags & SUR_FORCEFALLBACK))
		{
			if (sh_config.shadernamefmt)
			{
				char drivername[MAX_QPATH];
				Q_snprintfz(drivername, sizeof(drivername), sh_config.shadernamefmt, cleanname);
				if (Shader_ParseShader(&ps, drivername))
					continue;
			}
			if (Shader_ParseShader(&ps, cleanname))
				continue;
			if (strcmp(cleanname, shortname))
				if (Shader_ParseShader(&ps, shortname))
					continue;
		}
		if (s->generator)
		{
			oldsort = s->sort;
			Shader_Regenerate(&ps, shortname);
			if (s->sort != oldsort)
				resort = true;
		}
	}

	TRACE(("Resorting shaders\n"));

	if (resort)
	{
		Mod_ResortShaders();
	}
}

void Shader_NeedReload(qboolean rescanfs)
{
	if (rescanfs)
		shader_rescan_needed = true;
	shader_reload_needed = true;
}

cin_t *R_ShaderGetCinematic(shader_t *s)
{
#ifdef HAVE_MEDIA_DECODER
	int j;
	if (!s)
		return NULL;
	for (j = 0; j < s->numpasses; j++)
		if (s->passes[j].cin)
			return s->passes[j].cin;
#endif
	/*no cinematic in this shader!*/
	return NULL;
}

shader_t *R_ShaderFind(const char *name)
{
	int i;
	char shortname[MAX_QPATH];
	shader_t *s;

	if (!r_shaders)
		return NULL;

	COM_StripExtension ( name, shortname, sizeof(shortname));

	COM_CleanUpPath(shortname);

	//try and find it
	for (i = 0; i < r_numshaders; i++)
	{
		s = r_shaders[i];
		if (!s || !s->uses)
			continue;

		if (!Q_stricmp (shortname, s->name) )
			return s;
	}
	return NULL;
}

cin_t *R_ShaderFindCinematic(const char *name)
{
#ifdef HAVE_MEDIA_DECODER
	return R_ShaderGetCinematic(R_ShaderFind(name));
#else
	return NULL;
#endif
}

void Shader_ResetRemaps(void)
{
	shader_t *s;
	int i;
	for (i = 0; i < r_numshaders; i++)
	{
		s = r_shaders[i];
		if (!s)
			continue;
		s->remapto = s;
		s->remaptime = 0;
	}
}

void R_RemapShader(const char *sourcename, const char *destname, float timeoffset)
{
	shader_t *o;
	shader_t *n;
	int i;
	size_t l;

	char cleansrcname[MAX_QPATH];
	Q_strncpyz(cleansrcname, sourcename, sizeof(cleansrcname));
	COM_CleanUpPath(cleansrcname);
	l = strlen(cleansrcname);

	for (i = 0; i < r_numshaders; i++)
	{
		o = r_shaders[i];
		if (o && o->uses)
		{
			if (!strncmp(o->name, cleansrcname, l) && (!o->name[l] || o->name[l]=='#'))
			{
				n = R_LoadShader (va("%s%s", destname, o->name+l), o->usageflags, NULL, NULL);
				if (!n)
				{	//if it isn't actually available on disk then don't care about usageflags, just find ANY that's already loaded.
					// check the hash first
					char cleandstname[MAX_QPATH];
					Q_strncpyz(cleandstname, destname, sizeof(cleandstname));
					COM_CleanUpPath(cleandstname);
					n = Hash_Get(&shader_active_hash, cleandstname);
					if (!n || !n->uses)
						n = o;
				}
				o->remapto = n;
				o->remaptime = timeoffset;	//this just feels wrong.
			}
		}
	}
}

void Shader_RemapShader_f(void)
{
	char *sourcename = Cmd_Argv(1);
	char *destname = Cmd_Argv(2);
	float timeoffset = atof(Cmd_Argv(3));
	
	if (!Cmd_FromGamecode() && strcmp(InfoBuf_ValueForKey(&cl.serverinfo, "*cheats"), "ON"))
	{
		Con_Printf("%s may only be used from gamecode, or when cheats are enabled\n", Cmd_Argv(0));
		return;
	}
	if (!*sourcename)
	{
		Con_Printf("%s originalshader remappedshader starttime\n", Cmd_Argv(0));
		return;
	}
	R_RemapShader(sourcename, destname, timeoffset);
}

//blocks
int R_GetShaderSizes(shader_t *shader, int *width, int *height, qboolean blocktillloaded)
{
	if (!shader)
		return false;
	if (!shader->width && !shader->height)
	{
		int i;
		if (width)
			*width = 0;
		if (height)
			*height = 0;
		if ((shader->flags & SHADER_HASDIFFUSE) && shader->defaulttextures->base)
		{
			if (shader->defaulttextures->base->status == TEX_LOADING)
			{
				if (!blocktillloaded)
					return -1;
				COM_WorkerPartialSync(shader->defaulttextures->base, &shader->defaulttextures->base->status, TEX_LOADING);
			}
			if (shader->defaulttextures->base->status == TEX_LOADED)
			{
				shader->width = shader->defaulttextures->base->width;
				shader->height = shader->defaulttextures->base->height;
			}
		}
		else if ((shader->flags & SHADER_HASPALETTED) && shader->defaulttextures->paletted)
		{
			if (shader->defaulttextures->paletted->status == TEX_LOADING)
			{
				if (!blocktillloaded)
					return -1;
				COM_WorkerPartialSync(shader->defaulttextures->paletted, &shader->defaulttextures->paletted->status, TEX_LOADING);
			}
			if (shader->defaulttextures->paletted->status == TEX_LOADED)
			{
				shader->width = shader->defaulttextures->paletted->width;
				shader->height = shader->defaulttextures->paletted->height;
			}
		}
		else
		{
			for (i = 0; i < shader->numpasses; i++)
			{
				if (shader->passes[i].texgen == T_GEN_SINGLEMAP && shader->passes[i].anim_frames[0] && shader->passes[i].anim_frames[0]->status == TEX_LOADING)
				{
					if (!blocktillloaded)
						return -1;
					COM_WorkerPartialSync(shader->passes[i].anim_frames[0], &shader->passes[i].anim_frames[0]->status, TEX_LOADING);
				}
				if (shader->passes[i].texgen == T_GEN_DIFFUSE && (shader->defaulttextures->base && shader->defaulttextures->base->status == TEX_LOADING))
				{
					if (!blocktillloaded)
						return -1;
					COM_WorkerPartialSync(shader->defaulttextures->base, &shader->defaulttextures->base->status, TEX_LOADING);
				}
				if (shader->passes[i].texgen == T_GEN_PALETTED && (shader->defaulttextures->paletted && shader->defaulttextures->paletted->status == TEX_LOADING))
				{
					if (!blocktillloaded)
						return -1;
					COM_WorkerPartialSync(shader->defaulttextures->paletted, &shader->defaulttextures->paletted->status, TEX_LOADING);
				}
			}

			for (i = 0; i < shader->numpasses; i++)
			{
				if (shader->passes[i].texgen == T_GEN_SINGLEMAP)
				{
					if (shader->passes[i].anim_frames[0] && shader->passes[i].anim_frames[0]->status == TEX_LOADED)
					{
						shader->width = shader->passes[i].anim_frames[0]->width;
						shader->height = shader->passes[i].anim_frames[0]->height;
					}
					break;
				}
				if (shader->passes[i].texgen == T_GEN_DIFFUSE)
				{
					if (shader->defaulttextures->base && shader->defaulttextures->base->status == TEX_LOADED)
					{
						shader->width = shader->defaulttextures->base->width;
						shader->height = shader->defaulttextures->base->height;
					}
					break;
				}
				if (shader->passes[i].texgen == T_GEN_PALETTED)
				{
					if (shader->defaulttextures->paletted && shader->defaulttextures->paletted->status == TEX_LOADED)
					{
						shader->width = shader->defaulttextures->paletted->width;
						shader->height = shader->defaulttextures->paletted->height;
					}
					break;
				}
			}
			if (i == shader->numpasses)
			{	//this shader has no textures from which to source a width and height
				if (!shader->width)
					shader->width = 64;
				if (!shader->height)
					shader->height = 64;
			}
		}
	}
	if (shader->width && shader->height)
	{
		if (width)
			*width = shader->width;
		if (height)
			*height = shader->height;
		return true;	//final size
	}
	else
	{
		//fill with dummy values
		if (width)
			*width = 64;
		if (height)
			*height = 64;
		return false;
	}
}
shader_t *R_RegisterPic (const char *name, const char *subdirs)
{
	shader_t *shader;
	shader = R_LoadShader (name, SUF_2D, Shader_Default2D, subdirs);
	return shader;
}

shader_t *QDECL R_RegisterShader (const char *name, unsigned int usageflags, const char *shaderscript)
{
	return R_LoadShader (name, usageflags, Shader_DefaultScript, shaderscript);
}

shader_t *R_RegisterShader_Lightmap (const char *name)
{
	return R_LoadShader (name, SUF_LIGHTMAP, Shader_DefaultBSPLM, NULL);
}

shader_t *R_RegisterShader_Vertex (const char *name)
{
	return R_LoadShader (name, 0, Shader_DefaultBSPVertex, NULL);
}

shader_t *R_RegisterShader_Flare (const char *name)
{
	return R_LoadShader (name, 0, Shader_DefaultBSPFlare, NULL);
}

shader_t *QDECL R_RegisterSkin (const char *shadername, const char *modname)
{
	char newsname[MAX_QPATH];
	shader_t *shader;
#ifdef _DEBUG
	if (shadername == com_token)
		Con_Printf("R_RegisterSkin was passed com_token. that will bug out.\n");
#endif

	newsname[0] = 0;
	if (modname && !strchr(shadername, '/') && *shadername)
	{
		char *b = COM_SkipPath(modname);
		if (b != modname && b-modname + strlen(shadername)+1 < sizeof(newsname))
		{
			b--;	//no trailing /
			memcpy(newsname, modname, b - modname);
			newsname[b-modname] = 0;
		}
	}
	if (*newsname)
	{
		int l = strlen(newsname);
		Q_strncpyz(newsname+l, ":models", sizeof(newsname)-l);
	}
	else
		Q_strncpyz(newsname, "models", sizeof(newsname));
	shader = R_LoadShader (shadername, 0, Shader_DefaultSkin, newsname);
	return shader;
}
shader_t *R_RegisterCustom (const char *name, unsigned int usageflags, shader_gen_t *defaultgen, const void *args)
{
	return R_LoadShader (name, usageflags, defaultgen, args);
}
#endif //SERVERONLY