mirror of
https://github.com/blendogames/thirtyflightsofloving.git
synced 2025-01-18 14:31:55 +00:00
add4c7cc46
Added new SCR_DrawPic() variants in cl_screen.c. Added new graphics for text fields and sliders in menus. Improved mouse interaction for menu sliders. Added resettargets developer command to default Lazarus and missionpack DLLs. Added hint_test developer command to missionpack DLL. Fixed freeze developer command in default Lazarus and missionpack DLLs so it can be used more than once. More tweaks to Tactician Gunner prox mine safety checks in misssionpack DLL.
2944 lines
80 KiB
C
2944 lines
80 KiB
C
/*
|
|
===========================================================================
|
|
Copyright (C) 1997-2001 Id Software, Inc.
|
|
|
|
This file is part of Quake 2 source code.
|
|
|
|
Quake 2 source code is free software; you can redistribute it
|
|
and/or modify it under the terms of the GNU General Public License as
|
|
published by the Free Software Foundation; either version 2 of the License,
|
|
or (at your option) any later version.
|
|
|
|
Quake 2 source code is distributed in the hope that it will be
|
|
useful, but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with Quake 2 source code; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
===========================================================================
|
|
*/
|
|
|
|
// r_model.c -- model loading and caching
|
|
|
|
#include "r_local.h"
|
|
|
|
model_t *loadmodel;
|
|
int modfilelen;
|
|
|
|
void *ModChunk_Begin (size_t maxsize);
|
|
void *ModChunk_Alloc (size_t size);
|
|
size_t ModChunk_End (void);
|
|
void ModChunk_Free (void *base);
|
|
size_t Mod_GetAllocSizeSprite (model_t *mod, void *buffer);
|
|
void Mod_LoadSpriteModel (model_t *mod, void *buffer);
|
|
void Mod_Load_Q2_BrushModel (model_t *mod, void *buffer);
|
|
|
|
#ifdef MD2_AS_MD3
|
|
size_t Mod_GetAllocSizeMD2New (model_t *mod, void *buffer);
|
|
void Mod_LoadAliasMD2ModelNew (model_t *mod, void *buffer);
|
|
#else
|
|
size_t Mod_GetAllocSizeMD2Old (model_t *mod, void *buffer);
|
|
void Mod_LoadAliasMD2ModelOld (model_t *mod, void *buffer);
|
|
#endif
|
|
|
|
//Harven++ MD3
|
|
size_t Mod_GetAllocSizeMD3 (model_t *mod, void *buffer);
|
|
void Mod_LoadAliasMD3Model (model_t *mod, void *buffer);
|
|
//Harven-- MD3
|
|
model_t *Mod_LoadModel (model_t *mod, qboolean crash);
|
|
|
|
byte mod_novis[MAX_MAP_LEAFS/8];
|
|
|
|
#define MAX_MOD_KNOWN (MAX_MODELS*2) // Knightmare- was 512
|
|
model_t mod_known[MAX_MOD_KNOWN];
|
|
int mod_numknown;
|
|
|
|
// the inline * models from the current map are kept seperate
|
|
model_t mod_inline[MAX_MOD_KNOWN];
|
|
|
|
int registration_sequence;
|
|
qboolean registration_active; // map registration flag
|
|
|
|
/*
|
|
===============
|
|
Mod_PointInLeaf
|
|
===============
|
|
*/
|
|
mleaf_t *Mod_PointInLeaf (vec3_t p, model_t *model)
|
|
{
|
|
mnode_t *node;
|
|
float d;
|
|
cplane_t *plane;
|
|
|
|
if (!model || !model->nodes)
|
|
VID_Error (ERR_DROP, "Mod_PointInLeaf: bad model");
|
|
|
|
node = model->nodes;
|
|
while (1)
|
|
{
|
|
if (node->contents != -1)
|
|
return (mleaf_t *)node;
|
|
plane = node->plane;
|
|
d = DotProduct (p,plane->normal) - plane->dist;
|
|
if (d > 0)
|
|
node = node->children[0];
|
|
else
|
|
node = node->children[1];
|
|
}
|
|
|
|
return NULL; // never reached
|
|
}
|
|
|
|
|
|
/*
|
|
===================
|
|
Mod_DecompressVis
|
|
===================
|
|
*/
|
|
byte *Mod_DecompressVis (byte *in, model_t *model)
|
|
{
|
|
static byte decompressed[MAX_MAP_LEAFS/8];
|
|
int c;
|
|
byte *out;
|
|
int row;
|
|
|
|
row = (model->vis->numclusters+7)>>3;
|
|
out = decompressed;
|
|
|
|
if (!in)
|
|
{ // no vis info, so make all visible
|
|
while (row)
|
|
{
|
|
*out++ = 0xff;
|
|
row--;
|
|
}
|
|
return decompressed;
|
|
}
|
|
|
|
do
|
|
{
|
|
if (*in)
|
|
{
|
|
*out++ = *in++;
|
|
continue;
|
|
}
|
|
|
|
c = in[1];
|
|
in += 2;
|
|
while (c)
|
|
{
|
|
*out++ = 0;
|
|
c--;
|
|
}
|
|
} while (out - decompressed < row);
|
|
|
|
return decompressed;
|
|
}
|
|
|
|
/*
|
|
==============
|
|
Mod_ClusterPVS
|
|
==============
|
|
*/
|
|
byte *Mod_ClusterPVS (int cluster, model_t *model)
|
|
{
|
|
if (cluster == -1 || !model->vis)
|
|
return mod_novis;
|
|
return Mod_DecompressVis ( (byte *)model->vis + model->vis->bitofs[cluster][DVIS_PVS],
|
|
model);
|
|
}
|
|
|
|
|
|
//===============================================================================
|
|
|
|
/*
|
|
================
|
|
Mod_Modellist_f
|
|
================
|
|
*/
|
|
void Mod_Modellist_f (void)
|
|
{
|
|
int i;
|
|
model_t *mod;
|
|
size_t total;
|
|
|
|
total = 0;
|
|
VID_Printf (PRINT_ALL,"Loaded models:\n");
|
|
for (i=0, mod=mod_known ; i < mod_numknown ; i++, mod++)
|
|
{
|
|
if (!mod->name[0])
|
|
continue;
|
|
VID_Printf (PRINT_ALL, "%8i : %s\n",mod->extradatasize, mod->name);
|
|
total += mod->extradatasize;
|
|
}
|
|
VID_Printf (PRINT_ALL, "Total resident: %i\n", total);
|
|
}
|
|
|
|
/*
|
|
===============
|
|
Mod_Init
|
|
===============
|
|
*/
|
|
void Mod_Init (void)
|
|
{
|
|
memset (mod_novis, 0xff, sizeof(mod_novis));
|
|
|
|
registration_active = false; // map registration flag
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
==================
|
|
Mod_ForName
|
|
|
|
Loads in a model for the given name
|
|
==================
|
|
*/
|
|
model_t *Mod_ForName (char *name, qboolean crash)
|
|
{
|
|
model_t *mod;
|
|
unsigned *buf;
|
|
int i;
|
|
|
|
if (!name[0])
|
|
VID_Error (ERR_DROP, "Mod_ForName: NULL name");
|
|
|
|
//
|
|
// inline models are grabbed only from worldmodel
|
|
//
|
|
if (name[0] == '*')
|
|
{
|
|
i = atoi(name+1);
|
|
if (i < 1 || !r_worldmodel || i >= r_worldmodel->numsubmodels)
|
|
VID_Error (ERR_DROP, "bad inline model number");
|
|
return &mod_inline[i];
|
|
}
|
|
|
|
//
|
|
// search the currently loaded models
|
|
//
|
|
for (i=0, mod=mod_known; i<mod_numknown; i++, mod++)
|
|
{
|
|
if (!mod->name[0])
|
|
continue;
|
|
if (!strcmp (mod->name, name) )
|
|
return mod;
|
|
}
|
|
|
|
//
|
|
// find a free model slot spot
|
|
//
|
|
for (i=0, mod=mod_known; i<mod_numknown; i++, mod++)
|
|
{
|
|
if (!mod->name[0])
|
|
break; // free spot
|
|
}
|
|
if (i == mod_numknown)
|
|
{
|
|
if (mod_numknown == MAX_MOD_KNOWN)
|
|
VID_Error (ERR_DROP, "mod_numknown == MAX_MOD_KNOWN");
|
|
mod_numknown++;
|
|
}
|
|
// strncpy (mod->name, name);
|
|
Q_strncpyz (mod->name, sizeof(mod->name), name);
|
|
|
|
//
|
|
// load the file
|
|
//
|
|
modfilelen = FS_LoadFile (mod->name, &buf);
|
|
if (!buf)
|
|
{
|
|
if (crash)
|
|
VID_Error (ERR_DROP, "Mod_NumForName: %s not found", mod->name);
|
|
memset (mod->name, 0, sizeof(mod->name));
|
|
return NULL;
|
|
}
|
|
|
|
loadmodel = mod;
|
|
|
|
//
|
|
// fill it in
|
|
//
|
|
|
|
|
|
// call the apropriate loader
|
|
|
|
switch (LittleLong(*(unsigned *)buf))
|
|
{
|
|
case IDMD2HEADER:
|
|
#ifdef MD2_AS_MD3
|
|
// loadmodel->extradata = Hunk_Begin (0x800000); // was 0x500000
|
|
loadmodel->extradata = ModChunk_Begin (Mod_GetAllocSizeMD2New (mod, buf));
|
|
Mod_LoadAliasMD2ModelNew (mod, buf);
|
|
#else
|
|
// loadmodel->extradata = Hunk_Begin (0x200000);
|
|
loadmodel->extradata = ModChunk_Begin (Mod_GetAllocSizeMD2Old (mod, buf));
|
|
Mod_LoadAliasMD2ModelOld (mod, buf);
|
|
#endif // MD2_AS_MD3
|
|
loadmodel->extradatasize = ModChunk_End ();
|
|
break;
|
|
//Harven++ MD3
|
|
case IDMD3HEADER:
|
|
// loadmodel->extradata = Hunk_Begin (0x800000);
|
|
loadmodel->extradata = ModChunk_Begin (Mod_GetAllocSizeMD3 (mod, buf));
|
|
Mod_LoadAliasMD3Model (mod, buf);
|
|
loadmodel->extradatasize = ModChunk_End ();
|
|
break;
|
|
//Harven-- MD3
|
|
case IDSP2HEADER:
|
|
// loadmodel->extradata = Hunk_Begin (0x10000);
|
|
loadmodel->extradata = ModChunk_Begin (Mod_GetAllocSizeSprite (mod, buf));
|
|
Mod_LoadSpriteModel (mod, buf);
|
|
loadmodel->extradatasize = ModChunk_End ();
|
|
break;
|
|
|
|
case IDBSPHEADER:
|
|
loadmodel->extradata = Hunk_Begin (0x2000000); // was 0x1000000
|
|
Mod_Load_Q2_BrushModel (mod, buf);
|
|
mod->bspVersion = Q2_BSPVERSION;
|
|
loadmodel->extradatasize = Hunk_End ();
|
|
break;
|
|
|
|
default:
|
|
VID_Error (ERR_DROP,"Mod_NumForName: unknown fileid for %s", mod->name);
|
|
break;
|
|
}
|
|
|
|
// loadmodel->extradatasize = Hunk_End ();
|
|
|
|
FS_FreeFile (buf);
|
|
|
|
return mod;
|
|
}
|
|
|
|
/*
|
|
===============================================================================
|
|
|
|
BRUSHMODEL LOADING
|
|
|
|
===============================================================================
|
|
*/
|
|
|
|
byte *mod_base;
|
|
|
|
|
|
/*
|
|
=================
|
|
Mod_LoadLighting
|
|
=================
|
|
*/
|
|
void Mod_LoadLighting (lump_t *l)
|
|
{
|
|
if (!l->filelen)
|
|
{
|
|
loadmodel->lightdata = NULL;
|
|
return;
|
|
}
|
|
loadmodel->lightdata = Hunk_Alloc ( l->filelen);
|
|
memcpy (loadmodel->lightdata, mod_base + l->fileofs, l->filelen);
|
|
}
|
|
|
|
|
|
/*
|
|
=================
|
|
Mod_LoadVisibility
|
|
=================
|
|
*/
|
|
void Mod_LoadVisibility (lump_t *l)
|
|
{
|
|
int i;
|
|
|
|
if (!l->filelen)
|
|
{
|
|
loadmodel->vis = NULL;
|
|
return;
|
|
}
|
|
loadmodel->vis = Hunk_Alloc ( l->filelen);
|
|
memcpy (loadmodel->vis, mod_base + l->fileofs, l->filelen);
|
|
|
|
loadmodel->vis->numclusters = LittleLong (loadmodel->vis->numclusters);
|
|
for (i=0 ; i<loadmodel->vis->numclusters ; i++)
|
|
{
|
|
loadmodel->vis->bitofs[i][0] = LittleLong (loadmodel->vis->bitofs[i][0]);
|
|
loadmodel->vis->bitofs[i][1] = LittleLong (loadmodel->vis->bitofs[i][1]);
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
=================
|
|
Mod_LoadVertexes
|
|
=================
|
|
*/
|
|
void Mod_LoadVertexes (lump_t *l)
|
|
{
|
|
dvertex_t *in;
|
|
mvertex_t *out;
|
|
int i, count;
|
|
|
|
in = (void *)(mod_base + l->fileofs);
|
|
if (l->filelen % sizeof(*in))
|
|
VID_Error (ERR_DROP, "MOD_LoadBmodel: funny lump size in %s",loadmodel->name);
|
|
count = l->filelen / sizeof(*in);
|
|
out = Hunk_Alloc ( count*sizeof(*out));
|
|
|
|
loadmodel->vertexes = out;
|
|
loadmodel->numvertexes = count;
|
|
|
|
for ( i=0 ; i<count ; i++, in++, out++)
|
|
{
|
|
out->position[0] = LittleFloat (in->point[0]);
|
|
out->position[1] = LittleFloat (in->point[1]);
|
|
out->position[2] = LittleFloat (in->point[2]);
|
|
}
|
|
}
|
|
|
|
/*
|
|
=================
|
|
RadiusFromBounds
|
|
=================
|
|
*/
|
|
float RadiusFromBounds (vec3_t mins, vec3_t maxs)
|
|
{
|
|
int i;
|
|
vec3_t corner;
|
|
|
|
for (i=0 ; i<3 ; i++)
|
|
{
|
|
corner[i] = fabs(mins[i]) > fabs(maxs[i]) ? fabs(mins[i]) : fabs(maxs[i]);
|
|
}
|
|
|
|
return VectorLength (corner);
|
|
}
|
|
|
|
|
|
/*
|
|
=================
|
|
Mod_LoadSubmodels
|
|
=================
|
|
*/
|
|
void Mod_LoadSubmodels (lump_t *l)
|
|
{
|
|
dmodel_t *in;
|
|
mmodel_t *out;
|
|
int i, j, count;
|
|
|
|
in = (void *)(mod_base + l->fileofs);
|
|
if (l->filelen % sizeof(*in))
|
|
VID_Error (ERR_DROP, "MOD_LoadBmodel: funny lump size in %s", loadmodel->name);
|
|
count = l->filelen / sizeof(*in);
|
|
out = Hunk_Alloc ( count*sizeof(*out));
|
|
|
|
// Knightmare- catch submodel overflow
|
|
if (count >= MAX_MOD_KNOWN)
|
|
VID_Error (ERR_DROP, "MOD_LoadBmodel: too many submodels (%i >= %i) in %s", count, MAX_MOD_KNOWN, loadmodel->name);
|
|
|
|
loadmodel->submodels = out;
|
|
loadmodel->numsubmodels = count;
|
|
|
|
for ( i=0 ; i<count ; i++, in++, out++)
|
|
{
|
|
for (j=0 ; j<3 ; j++)
|
|
{ // spread the mins / maxs by a pixel
|
|
out->mins[j] = LittleFloat (in->mins[j]) - 1;
|
|
out->maxs[j] = LittleFloat (in->maxs[j]) + 1;
|
|
out->origin[j] = LittleFloat (in->origin[j]);
|
|
}
|
|
out->radius = RadiusFromBounds (out->mins, out->maxs);
|
|
out->headnode = LittleLong (in->headnode);
|
|
out->firstface = LittleLong (in->firstface);
|
|
out->numfaces = LittleLong (in->numfaces);
|
|
}
|
|
}
|
|
|
|
/*
|
|
=================
|
|
Mod_LoadEdges
|
|
=================
|
|
*/
|
|
void Mod_LoadEdges (lump_t *l)
|
|
{
|
|
dedge_t *in;
|
|
medge_t *out;
|
|
int i, count;
|
|
|
|
in = (void *)(mod_base + l->fileofs);
|
|
if (l->filelen % sizeof(*in))
|
|
VID_Error (ERR_DROP, "MOD_LoadBmodel: funny lump size in %s",loadmodel->name);
|
|
count = l->filelen / sizeof(*in);
|
|
out = Hunk_Alloc ( (count + 1) * sizeof(*out));
|
|
|
|
loadmodel->edges = out;
|
|
loadmodel->numedges = count;
|
|
|
|
for ( i=0 ; i<count ; i++, in++, out++)
|
|
{
|
|
out->v[0] = (unsigned short)LittleShort(in->v[0]);
|
|
out->v[1] = (unsigned short)LittleShort(in->v[1]);
|
|
}
|
|
}
|
|
|
|
//=======================================================
|
|
|
|
// store the names of last textures that failed to load
|
|
#define NUM_FAIL_TEXTURES 256
|
|
char lastFailedTexture[NUM_FAIL_TEXTURES][MAX_OSPATH];
|
|
unsigned int lastFailedTextureHash[NUM_FAIL_TEXTURES];
|
|
static unsigned failedTexListIndex;
|
|
|
|
/*
|
|
===============
|
|
Mod_InitFailedTexList
|
|
===============
|
|
*/
|
|
void Mod_InitFailedTexList (void)
|
|
{
|
|
int i;
|
|
|
|
for (i=0; i<NUM_FAIL_TEXTURES; i++) {
|
|
Com_sprintf(lastFailedTexture[i], sizeof(lastFailedTexture[i]), "\0");
|
|
lastFailedTextureHash[i] = 0;
|
|
}
|
|
|
|
failedTexListIndex = 0;
|
|
}
|
|
|
|
/*
|
|
===============
|
|
Mod_CheckTexFailed
|
|
===============
|
|
*/
|
|
qboolean Mod_CheckTexFailed (char *name)
|
|
{
|
|
int i;
|
|
unsigned int hash;
|
|
|
|
hash = Com_HashFileName(name, 0, false);
|
|
for (i=0; i<NUM_FAIL_TEXTURES; i++)
|
|
{
|
|
if (hash == lastFailedTextureHash[i]) { // compare hash first
|
|
if (lastFailedTexture[i] && strlen(lastFailedTexture[i])
|
|
&& !strcmp(name, lastFailedTexture[i]))
|
|
{ // we already tried to load this image, didn't find it
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/*
|
|
===============
|
|
Mod_AddToFailedTexList
|
|
===============
|
|
*/
|
|
void Mod_AddToFailedTexList (char *name)
|
|
{
|
|
Com_sprintf(lastFailedTexture[failedTexListIndex], sizeof(lastFailedTexture[failedTexListIndex]), "%s", name);
|
|
lastFailedTextureHash[failedTexListIndex] = Com_HashFileName(name, 0, false);
|
|
failedTexListIndex++;
|
|
|
|
// wrap around to start of list
|
|
if (failedTexListIndex >= NUM_FAIL_TEXTURES)
|
|
failedTexListIndex = 0;
|
|
}
|
|
|
|
/*
|
|
===============
|
|
Mod_FindTexture
|
|
A wrapper function that uses a failed list
|
|
to speed map load times
|
|
===============
|
|
*/
|
|
image_t *Mod_FindTexture (char *name, imagetype_t type)
|
|
{
|
|
image_t *image;
|
|
|
|
// don't try again to load a texture that just failed
|
|
if (Mod_CheckTexFailed (name))
|
|
return glMedia.notexture;
|
|
|
|
image = R_FindImage (name, type);
|
|
|
|
if (!image || (image == glMedia.notexture))
|
|
Mod_AddToFailedTexList (name);
|
|
|
|
return image;
|
|
}
|
|
|
|
//=======================================================
|
|
|
|
// store the names and sizes of size reference .wal files
|
|
typedef struct walsize_s
|
|
{
|
|
char name[MAX_OSPATH];
|
|
unsigned int hash;
|
|
int width;
|
|
int height;
|
|
} walsize_t;
|
|
|
|
#define NUM_WALSIZES 1024
|
|
walsize_t walSizeList[NUM_WALSIZES];
|
|
static unsigned walSizeListIndex;
|
|
|
|
/*
|
|
===============
|
|
Mod_InitWalSizeList
|
|
===============
|
|
*/
|
|
void Mod_InitWalSizeList (void)
|
|
{
|
|
int i;
|
|
|
|
for (i=0; i<NUM_WALSIZES; i++) {
|
|
Com_sprintf(walSizeList[i].name, sizeof(walSizeList[i].name), "\0");
|
|
walSizeList[i].hash = 0;
|
|
walSizeList[i].width = 0;
|
|
walSizeList[i].height = 0;
|
|
}
|
|
walSizeListIndex = 0;
|
|
}
|
|
|
|
/*
|
|
===============
|
|
Mod_CheckWalSizeList
|
|
===============
|
|
*/
|
|
qboolean Mod_CheckWalSizeList (const char *name, int *width, int *height)
|
|
{
|
|
int i;
|
|
unsigned int hash;
|
|
|
|
hash = Com_HashFileName(name, 0, false);
|
|
for (i=0; i<NUM_WALSIZES; i++)
|
|
{
|
|
if (hash == walSizeList[i].hash) { // compare hash first
|
|
if (walSizeList[i].name && strlen(walSizeList[i].name)
|
|
&& !strcmp(name, walSizeList[i].name))
|
|
{ // return size of texture
|
|
if (width)
|
|
*width = walSizeList[i].width;
|
|
if (height)
|
|
*height = walSizeList[i].height;
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/*
|
|
===============
|
|
Mod_AddToWalSizeList
|
|
===============
|
|
*/
|
|
void Mod_AddToWalSizeList (const char *name, int width, int height)
|
|
{
|
|
Com_sprintf(walSizeList[walSizeListIndex].name, sizeof(walSizeList[walSizeListIndex].name), "%s", name);
|
|
walSizeList[walSizeListIndex].hash = Com_HashFileName(name, 0, false);
|
|
walSizeList[walSizeListIndex].width = width;
|
|
walSizeList[walSizeListIndex].height = height;
|
|
walSizeListIndex++;
|
|
|
|
// wrap around to start of list
|
|
if (walSizeListIndex >= NUM_WALSIZES)
|
|
walSizeListIndex = 0;
|
|
}
|
|
/*
|
|
=================
|
|
Mod_GetWalSize
|
|
Adapted from Q2E
|
|
=================
|
|
*/
|
|
static void Mod_GetWalSize (const char *name, int *width, int *height)
|
|
{
|
|
char path[MAX_QPATH];
|
|
miptex_t *mt;
|
|
|
|
Com_sprintf (path, sizeof(path), "textures/%s.wal", name);
|
|
|
|
if (Mod_CheckWalSizeList(name, width, height)) // check if already in list
|
|
return;
|
|
|
|
FS_LoadFile (path, (void **)&mt); // load .wal file
|
|
if (!mt)
|
|
{ // set null value to tell us to get actual size of texture
|
|
*width = *height = -1;
|
|
Mod_AddToWalSizeList(name, *width, *height); // add to list
|
|
return;
|
|
}
|
|
*width = LittleLong (mt->width); // grab size from wal
|
|
*height = LittleLong (mt->height);
|
|
FS_FreeFile ((void *)mt); // free the wal
|
|
|
|
Mod_AddToWalSizeList(name, *width, *height); // add to list
|
|
}
|
|
|
|
|
|
/*
|
|
=================
|
|
Mod_LoadTexinfo
|
|
=================
|
|
*/
|
|
void Mod_LoadTexinfo (lump_t *l)
|
|
{
|
|
texinfo_t *in;
|
|
mtexinfo_t *out, *step;
|
|
int i, j, count;
|
|
char name[MAX_QPATH];
|
|
int next;
|
|
|
|
in = (void *)(mod_base + l->fileofs);
|
|
if (l->filelen % sizeof(*in))
|
|
VID_Error (ERR_DROP, "MOD_LoadBmodel: funny lump size in %s",loadmodel->name);
|
|
count = l->filelen / sizeof(*in);
|
|
out = Hunk_Alloc ( count*sizeof(*out));
|
|
|
|
loadmodel->texinfo = out;
|
|
loadmodel->numtexinfo = count;
|
|
|
|
for ( i=0 ; i<count ; i++, in++, out++)
|
|
{
|
|
for (j=0 ; j<8 ; j++)
|
|
out->vecs[0][j] = LittleFloat (in->vecs[0][j]);
|
|
|
|
out->flags = LittleLong (in->flags);
|
|
next = LittleLong (in->nexttexinfo);
|
|
if (next > 0)
|
|
out->next = loadmodel->texinfo + next;
|
|
else
|
|
out->next = NULL;
|
|
|
|
Com_sprintf (name, sizeof(name), "textures/%s.wal", in->texture);
|
|
out->image = Mod_FindTexture (name, it_wall); // was R_FindImage
|
|
|
|
if (!out->image)
|
|
{
|
|
VID_Printf (PRINT_ALL, "Couldn't load %s\n", name);
|
|
out->image = glMedia.notexture;
|
|
}
|
|
|
|
// Added glow
|
|
Com_sprintf (name, sizeof(name), "textures/%s_glow.wal", in->texture);
|
|
out->glow = Mod_FindTexture (name, it_skin); // was R_FindImage
|
|
if (!out->glow)
|
|
out->glow = glMedia.notexture;
|
|
|
|
// Q2E HACK: find .wal dimensions for texture coord generation
|
|
// NOTE: Once Q3 map support is added, be be sure to disable this
|
|
// for Q3 format maps, because they will be natively textured with
|
|
// hi-res textures.
|
|
Mod_GetWalSize (in->texture, &out->texWidth, &out->texHeight);
|
|
|
|
// If no .wal texture was found, use width & height of actual texture
|
|
if (out->texWidth == -1 || out->texHeight == -1)
|
|
{
|
|
out->texWidth = out->image->width;
|
|
out->texHeight = out->image->height;
|
|
}
|
|
}
|
|
|
|
// count animation frames
|
|
for (i=0 ; i<count ; i++)
|
|
{
|
|
out = &loadmodel->texinfo[i];
|
|
out->numframes = 1;
|
|
for (step = out->next ; step && step != out ; step=step->next)
|
|
out->numframes++;
|
|
}
|
|
}
|
|
|
|
/*
|
|
================
|
|
CalcSurfaceExtents
|
|
|
|
Fills in s->texturemins[] and s->extents[]
|
|
================
|
|
*/
|
|
void CalcSurfaceExtents (msurface_t *s)
|
|
{
|
|
float mins[2], maxs[2], val;
|
|
int i,j, e;
|
|
mvertex_t *v;
|
|
mtexinfo_t *tex;
|
|
int bmins[2], bmaxs[2];
|
|
|
|
mins[0] = mins[1] = 999999;
|
|
maxs[0] = maxs[1] = -99999;
|
|
|
|
tex = s->texinfo;
|
|
|
|
for (i=0 ; i<s->numedges ; i++)
|
|
{
|
|
e = loadmodel->surfedges[s->firstedge+i];
|
|
if (e >= 0)
|
|
v = &loadmodel->vertexes[loadmodel->edges[e].v[0]];
|
|
else
|
|
v = &loadmodel->vertexes[loadmodel->edges[-e].v[1]];
|
|
|
|
for (j=0 ; j<2 ; j++)
|
|
{
|
|
val = v->position[0] * tex->vecs[j][0] +
|
|
v->position[1] * tex->vecs[j][1] +
|
|
v->position[2] * tex->vecs[j][2] +
|
|
tex->vecs[j][3];
|
|
if (val < mins[j])
|
|
mins[j] = val;
|
|
if (val > maxs[j])
|
|
maxs[j] = val;
|
|
}
|
|
}
|
|
|
|
for (i=0 ; i<2 ; i++)
|
|
{
|
|
bmins[i] = floor(mins[i]/16);
|
|
bmaxs[i] = ceil(maxs[i]/16);
|
|
|
|
s->texturemins[i] = bmins[i] * 16;
|
|
s->extents[i] = (bmaxs[i] - bmins[i]) * 16;
|
|
|
|
// if ( !(tex->flags & TEX_SPECIAL) && s->extents[i] > 512 /* 256 */ )
|
|
// VID_Error (ERR_DROP, "Bad surface extents");
|
|
}
|
|
}
|
|
|
|
|
|
void R_BuildPolygonFromSurface (msurface_t *fa);
|
|
void R_CreateSurfaceLightmap (msurface_t *surf);
|
|
void R_EndBuildingLightmaps (void);
|
|
void R_BeginBuildingLightmaps (model_t *m);
|
|
|
|
/*
|
|
=================
|
|
Mod_LoadFaces
|
|
=================
|
|
*/
|
|
void Mod_LoadFaces (lump_t *l)
|
|
{
|
|
dface_t *in;
|
|
msurface_t *out;
|
|
int i, count, surfnum;
|
|
int planenum, side;
|
|
int ti;
|
|
|
|
in = (void *)(mod_base + l->fileofs);
|
|
if (l->filelen % sizeof(*in))
|
|
VID_Error (ERR_DROP, "MOD_LoadBmodel: funny lump size in %s",loadmodel->name);
|
|
count = l->filelen / sizeof(*in);
|
|
out = Hunk_Alloc ( count*sizeof(*out));
|
|
|
|
loadmodel->surfaces = out;
|
|
loadmodel->numsurfaces = count;
|
|
|
|
currentmodel = loadmodel;
|
|
|
|
R_BeginBuildingLightmaps (loadmodel);
|
|
|
|
for ( surfnum=0 ; surfnum<count ; surfnum++, in++, out++)
|
|
{
|
|
out->firstedge = LittleLong(in->firstedge);
|
|
out->numedges = LittleShort(in->numedges);
|
|
out->flags = 0;
|
|
out->polys = NULL;
|
|
|
|
planenum = LittleShort(in->planenum);
|
|
side = LittleShort(in->side);
|
|
if (side)
|
|
out->flags |= SURF_PLANEBACK;
|
|
|
|
out->plane = loadmodel->planes + planenum;
|
|
|
|
ti = LittleShort (in->texinfo);
|
|
if (ti < 0 || ti >= loadmodel->numtexinfo)
|
|
VID_Error (ERR_DROP, "MOD_LoadBmodel: bad texinfo number");
|
|
out->texinfo = loadmodel->texinfo + ti;
|
|
|
|
CalcSurfaceExtents (out);
|
|
|
|
// lighting info
|
|
|
|
for (i=0 ; i<MAXLIGHTMAPS ; i++)
|
|
out->styles[i] = in->styles[i];
|
|
i = LittleLong(in->lightofs);
|
|
if (i == -1)
|
|
out->samples = NULL;
|
|
else
|
|
out->samples = loadmodel->lightdata + i;
|
|
|
|
// set the drawing flags
|
|
|
|
if (out->texinfo->flags & SURF_WARP)
|
|
{
|
|
out->flags |= SURF_DRAWTURB;
|
|
#ifdef WARP_LIGHTMAPS
|
|
// Knightmare- create lightmaps if surface has light data and has properly subdivided size
|
|
if ( (loadmodel->bspFeatures & BSPF_WARPLIGHTMAPS) && (out->samples != NULL)
|
|
&& ((out->extents[0]>>4)+1 <= LM_BLOCK_WIDTH)
|
|
&& ((out->extents[1]>>4)+1 <= LM_BLOCK_HEIGHT) )
|
|
{
|
|
out->isLightmapped = true;
|
|
R_CreateSurfaceLightmap (out);
|
|
}
|
|
else // Knightmare- only do this for unlit warp faces!
|
|
#endif // WARP_LIGHTMAPS
|
|
{
|
|
for (i=0; i<2; i++)
|
|
{
|
|
out->extents[i] = (WORLD_SIZE*2); // was 16384
|
|
out->texturemins[i] = -WORLD_SIZE; // was -8192
|
|
}
|
|
out->samples = NULL;
|
|
out->isLightmapped = false;
|
|
}
|
|
|
|
R_SubdivideSurface (out); // cut up polygon for warps
|
|
}
|
|
// Knightmare- Psychospaz's envmapping
|
|
// windows get glass (envmap) effect... -psychospaz
|
|
// but warp surfaces and solid alphas don't
|
|
else if (out->texinfo->flags & (SURF_TRANS33|SURF_TRANS66))
|
|
//&& !((out->texinfo->flags & SURF_TRANS33) && (out->texinfo->flags & SURF_TRANS66)) )
|
|
{
|
|
if (!(out->texinfo->flags & SURF_NOLIGHTENV))
|
|
out->flags |= SURF_ENVMAP;
|
|
}
|
|
|
|
// create lightmaps and polygons
|
|
//if ( !(out->texinfo->flags & (SURF_SKY|SURF_TRANS33|SURF_TRANS66|SURF_WARP) ) )
|
|
if ( !(out->texinfo->flags & (SURF_SKY|SURF_WARP) ) )
|
|
{
|
|
out->isLightmapped = true;
|
|
R_CreateSurfaceLightmap (out);
|
|
}
|
|
|
|
if (! (out->texinfo->flags & SURF_WARP) )
|
|
R_BuildPolygonFromSurface (out);
|
|
}
|
|
|
|
R_EndBuildingLightmaps ();
|
|
}
|
|
|
|
|
|
/*
|
|
=================
|
|
Mod_SetParent
|
|
=================
|
|
*/
|
|
void Mod_SetParent (mnode_t *node, mnode_t *parent)
|
|
{
|
|
node->parent = parent;
|
|
if (node->contents != -1)
|
|
return;
|
|
Mod_SetParent (node->children[0], node);
|
|
Mod_SetParent (node->children[1], node);
|
|
}
|
|
|
|
/*
|
|
=================
|
|
Mod_LoadNodes
|
|
=================
|
|
*/
|
|
void Mod_LoadNodes (lump_t *l)
|
|
{
|
|
int i, j, count, p;
|
|
dnode_t *in;
|
|
mnode_t *out;
|
|
|
|
in = (void *)(mod_base + l->fileofs);
|
|
if (l->filelen % sizeof(*in))
|
|
VID_Error (ERR_DROP, "MOD_LoadBmodel: funny lump size in %s",loadmodel->name);
|
|
count = l->filelen / sizeof(*in);
|
|
out = Hunk_Alloc ( count*sizeof(*out));
|
|
|
|
loadmodel->nodes = out;
|
|
loadmodel->numnodes = count;
|
|
|
|
for ( i=0 ; i<count ; i++, in++, out++)
|
|
{
|
|
for (j=0 ; j<3 ; j++)
|
|
{
|
|
out->minmaxs[j] = LittleShort (in->mins[j]);
|
|
out->minmaxs[3+j] = LittleShort (in->maxs[j]);
|
|
}
|
|
|
|
p = LittleLong(in->planenum);
|
|
out->plane = loadmodel->planes + p;
|
|
|
|
out->firstsurface = LittleShort (in->firstface);
|
|
out->numsurfaces = LittleShort (in->numfaces);
|
|
out->contents = -1; // differentiate from leafs
|
|
|
|
for (j=0 ; j<2 ; j++)
|
|
{
|
|
p = LittleLong (in->children[j]);
|
|
if (p >= 0)
|
|
out->children[j] = loadmodel->nodes + p;
|
|
else
|
|
out->children[j] = (mnode_t *)(loadmodel->leafs + (-1 - p));
|
|
}
|
|
}
|
|
|
|
Mod_SetParent (loadmodel->nodes, NULL); // sets nodes and leafs
|
|
}
|
|
|
|
/*
|
|
=================
|
|
Mod_LoadLeafs
|
|
=================
|
|
*/
|
|
void Mod_LoadLeafs (lump_t *l)
|
|
{
|
|
dleaf_t *in;
|
|
mleaf_t *out;
|
|
int i, j, count, p;
|
|
glpoly_t *poly;
|
|
|
|
in = (void *)(mod_base + l->fileofs);
|
|
if (l->filelen % sizeof(*in))
|
|
VID_Error (ERR_DROP, "MOD_LoadBmodel: funny lump size in %s",loadmodel->name);
|
|
count = l->filelen / sizeof(*in);
|
|
out = Hunk_Alloc ( count*sizeof(*out));
|
|
|
|
loadmodel->leafs = out;
|
|
loadmodel->numleafs = count;
|
|
|
|
for ( i=0 ; i<count ; i++, in++, out++)
|
|
{
|
|
for (j=0 ; j<3 ; j++)
|
|
{
|
|
out->minmaxs[j] = LittleShort (in->mins[j]);
|
|
out->minmaxs[3+j] = LittleShort (in->maxs[j]);
|
|
}
|
|
|
|
p = LittleLong(in->contents);
|
|
out->contents = p;
|
|
|
|
out->cluster = LittleShort(in->cluster);
|
|
out->area = LittleShort(in->area);
|
|
|
|
out->firstmarksurface = loadmodel->marksurfaces +
|
|
(unsigned short)LittleShort(in->firstleafface); // Knightmare- make sure this doesn't turn negative!
|
|
out->nummarksurfaces = LittleShort(in->numleaffaces);
|
|
|
|
// underwater flag for caustics
|
|
//if (out->contents & (CONTENTS_WATER|CONTENTS_SLIME) )
|
|
if (out->contents & (MASK_WATER) )
|
|
{ unsigned int flag;
|
|
if (out->contents & CONTENTS_LAVA)
|
|
flag = SURF_UNDERLAVA;
|
|
else if (out->contents & CONTENTS_SLIME)
|
|
flag = SURF_UNDERSLIME;
|
|
else
|
|
flag = SURF_UNDERWATER;
|
|
for (j=0 ; j<out->nummarksurfaces ; j++)
|
|
{
|
|
out->firstmarksurface[j]->flags |= flag;
|
|
for (poly = out->firstmarksurface[j]->polys; poly; poly=poly->next)
|
|
poly->flags |= flag;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
=================
|
|
Mod_LoadMarksurfaces
|
|
=================
|
|
*/
|
|
void Mod_LoadMarksurfaces (lump_t *l)
|
|
{
|
|
int i, j, count;
|
|
unsigned short *in; // Knightmare- changed to unsigned short
|
|
msurface_t **out;
|
|
|
|
in = (void *)(mod_base + l->fileofs);
|
|
if (l->filelen % sizeof(*in))
|
|
VID_Error (ERR_DROP, "MOD_LoadBmodel: funny lump size in %s",loadmodel->name);
|
|
count = l->filelen / sizeof(*in);
|
|
out = Hunk_Alloc ( count*sizeof(*out));
|
|
|
|
loadmodel->marksurfaces = out;
|
|
loadmodel->nummarksurfaces = count;
|
|
|
|
for ( i=0 ; i<count ; i++)
|
|
{
|
|
j = (unsigned short)LittleShort(in[i]); // Knightmare- make this unsigned!
|
|
if (j < 0 || j >= loadmodel->numsurfaces)
|
|
VID_Error (ERR_DROP, "Mod_ParseMarksurfaces: bad surface number");
|
|
out[i] = loadmodel->surfaces + j;
|
|
}
|
|
}
|
|
|
|
/*
|
|
=================
|
|
Mod_LoadSurfedges
|
|
=================
|
|
*/
|
|
void Mod_LoadSurfedges (lump_t *l)
|
|
{
|
|
int i, count;
|
|
int *in, *out;
|
|
|
|
in = (void *)(mod_base + l->fileofs);
|
|
if (l->filelen % sizeof(*in))
|
|
VID_Error (ERR_DROP, "MOD_LoadBmodel: funny lump size in %s",loadmodel->name);
|
|
count = l->filelen / sizeof(*in);
|
|
if (count < 1 || count >= MAX_MAP_SURFEDGES)
|
|
VID_Error (ERR_DROP, "MOD_LoadBmodel: bad surfedges count in %s: %i",
|
|
loadmodel->name, count);
|
|
|
|
out = Hunk_Alloc ( count*sizeof(*out));
|
|
|
|
loadmodel->surfedges = out;
|
|
loadmodel->numsurfedges = count;
|
|
|
|
for ( i=0 ; i<count ; i++)
|
|
out[i] = LittleLong (in[i]);
|
|
}
|
|
|
|
|
|
/*
|
|
=================
|
|
Mod_LoadPlanes
|
|
=================
|
|
*/
|
|
void Mod_LoadPlanes (lump_t *l)
|
|
{
|
|
int i, j;
|
|
cplane_t *out;
|
|
dplane_t *in;
|
|
int count;
|
|
int bits;
|
|
|
|
in = (void *)(mod_base + l->fileofs);
|
|
if (l->filelen % sizeof(*in))
|
|
VID_Error (ERR_DROP, "MOD_LoadBmodel: funny lump size in %s",loadmodel->name);
|
|
count = l->filelen / sizeof(*in);
|
|
|
|
//original bug- Quake2Max
|
|
// out = Hunk_Alloc ( count*2*sizeof(*out));
|
|
out = Hunk_Alloc ( count*sizeof(*out));
|
|
|
|
loadmodel->planes = out;
|
|
loadmodel->numplanes = count;
|
|
|
|
for ( i=0 ; i<count ; i++, in++, out++)
|
|
{
|
|
bits = 0;
|
|
for (j=0 ; j<3 ; j++)
|
|
{
|
|
out->normal[j] = LittleFloat (in->normal[j]);
|
|
if (out->normal[j] < 0)
|
|
bits |= 1<<j;
|
|
}
|
|
|
|
out->dist = LittleFloat (in->dist);
|
|
out->type = LittleLong (in->type);
|
|
out->signbits = bits;
|
|
}
|
|
}
|
|
|
|
/*
|
|
=================
|
|
Mod_Load_Q2_BrushModel
|
|
=================
|
|
*/
|
|
void Mod_Load_Q2_BrushModel (model_t *mod, void *buffer)
|
|
{
|
|
int i;
|
|
dheader_t *header;
|
|
mmodel_t *bm;
|
|
|
|
loadmodel->type = mod_brush;
|
|
if (loadmodel != mod_known)
|
|
VID_Error (ERR_DROP, "Loaded a brush model after the world");
|
|
|
|
header = (dheader_t *)buffer;
|
|
|
|
i = LittleLong (header->version);
|
|
if (i != Q2_BSPVERSION) {
|
|
VID_Error (ERR_DROP, "Mod_Load_Q2_BrushModel: %s has wrong version number (%i should be %i)", mod->name, i, Q2_BSPVERSION);
|
|
}
|
|
|
|
loadmodel->bspFeatures = 0; // set BSP features flags
|
|
|
|
// swap all the lumps
|
|
mod_base = (byte *)header;
|
|
|
|
for (i=0 ; i<sizeof(dheader_t)/4 ; i++)
|
|
((int *)header)[i] = LittleLong ( ((int *)header)[i]);
|
|
|
|
// load into heap
|
|
Mod_LoadVertexes (&header->lumps[LUMP_VERTEXES]);
|
|
Mod_LoadEdges (&header->lumps[LUMP_EDGES]);
|
|
Mod_LoadSurfedges (&header->lumps[LUMP_SURFEDGES]);
|
|
Mod_LoadLighting (&header->lumps[LUMP_LIGHTING]);
|
|
Mod_LoadPlanes (&header->lumps[LUMP_PLANES]);
|
|
Mod_LoadTexinfo (&header->lumps[LUMP_TEXINFO]);
|
|
Mod_LoadFaces (&header->lumps[LUMP_FACES]);
|
|
Mod_LoadMarksurfaces (&header->lumps[LUMP_LEAFFACES]);
|
|
Mod_LoadVisibility (&header->lumps[LUMP_VISIBILITY]);
|
|
Mod_LoadLeafs (&header->lumps[LUMP_LEAFS]);
|
|
Mod_LoadNodes (&header->lumps[LUMP_NODES]);
|
|
Mod_LoadSubmodels (&header->lumps[LUMP_MODELS]);
|
|
mod->numframes = 2; // regular and alternate animation
|
|
|
|
//
|
|
// set up the submodels
|
|
//
|
|
for (i=0 ; i<mod->numsubmodels ; i++)
|
|
{
|
|
model_t *starmod;
|
|
|
|
bm = &mod->submodels[i];
|
|
starmod = &mod_inline[i];
|
|
|
|
*starmod = *loadmodel;
|
|
|
|
starmod->firstmodelsurface = bm->firstface;
|
|
starmod->nummodelsurfaces = bm->numfaces;
|
|
starmod->firstnode = bm->headnode;
|
|
if (starmod->firstnode >= loadmodel->numnodes)
|
|
VID_Error (ERR_DROP, "Inline model %i has bad firstnode", i);
|
|
|
|
VectorCopy (bm->maxs, starmod->maxs);
|
|
VectorCopy (bm->mins, starmod->mins);
|
|
starmod->radius = bm->radius;
|
|
|
|
if (i == 0)
|
|
*loadmodel = *starmod;
|
|
|
|
starmod->numleafs = bm->visleafs;
|
|
}
|
|
}
|
|
|
|
/*
|
|
===============================================================================
|
|
|
|
MODEL MEMORY ALLOCATION
|
|
|
|
===============================================================================
|
|
*/
|
|
|
|
// It turns out that the Hunk_Begin(), Hunk_Alloc(), and Hunk_Free()
|
|
// functions used for BSP, alias model, and sprite loading are a wrapper around
|
|
// VirtualAlloc()/VirtualFree() on Win32 and they should only be used
|
|
// rarely and for large blocks of memory. After about 185-190 VirtualAlloc()
|
|
// reserve calls are made on Win7 x64, the subsequent call fails. Bad Carmack, bad!
|
|
|
|
// These ModChunk_ functions are a replacement that wrap around malloc()/free()
|
|
// and return pointers into sections aligned on cache lines.
|
|
// The only caveat is that the allocation size passed to ModChunk_Begin() is
|
|
// immediately allocated, not reserved. Calling it with a maximum memory size
|
|
// for each model type would be hugely wasteful. So a size equal or greater to
|
|
// the total amount requested via ModChunk_Alloc() calls in the model loading
|
|
// functions must be calculated first.
|
|
|
|
int modChunkCount;
|
|
|
|
byte *modChunkMemBase;
|
|
size_t modChunkMaxSize;
|
|
size_t modChunkCurSize;
|
|
|
|
|
|
void *ModChunk_Begin (size_t maxsize)
|
|
{
|
|
// alocate a chunk of memory, should be exact size needed!
|
|
modChunkCurSize = 0;
|
|
modChunkMaxSize = maxsize;
|
|
|
|
modChunkMemBase = malloc (maxsize);
|
|
|
|
if (!modChunkMemBase)
|
|
Sys_Error ("ModChunk_Begin: malloc of size %i failed, %i chunks already allocated", maxsize, modChunkCount);
|
|
|
|
memset (modChunkMemBase, 0, maxsize);
|
|
|
|
return (void *)modChunkMemBase;
|
|
}
|
|
|
|
void *ModChunk_Alloc (size_t size)
|
|
{
|
|
// round to cacheline
|
|
size = (size+31)&~31;
|
|
|
|
modChunkCurSize += size;
|
|
if (modChunkCurSize > modChunkMaxSize)
|
|
Sys_Error ("ModChunk_Alloc: overflow");
|
|
|
|
return (void *)(modChunkMemBase + modChunkCurSize - size);
|
|
}
|
|
|
|
size_t ModChunk_End (void)
|
|
{
|
|
// free the remaining unused virtual memory
|
|
|
|
modChunkCount++;
|
|
// VID_Printf (PRINT_ALL, "modChunkCount: %i\n", modChunkCount);
|
|
return modChunkCurSize;
|
|
}
|
|
|
|
void ModChunk_Free (void *base)
|
|
{
|
|
if ( base )
|
|
free (base);
|
|
|
|
modChunkCount--;
|
|
}
|
|
|
|
/*
|
|
==============================================================================
|
|
|
|
ALIAS MODELS
|
|
|
|
==============================================================================
|
|
*/
|
|
|
|
/*
|
|
=================
|
|
Mod_ParseBlendMode
|
|
md3 skin protoshaders
|
|
=================
|
|
*/
|
|
GLenum Mod_ParseBlendMode (char *name)
|
|
{
|
|
if (!name) return -1;
|
|
|
|
if (!Q_strcasecmp(name, "gl_zero"))
|
|
return GL_ZERO;
|
|
if (!Q_strcasecmp(name, "gl_one"))
|
|
return GL_ONE;
|
|
if (!Q_strcasecmp(name, "gl_src_color"))
|
|
return GL_SRC_COLOR;
|
|
if (!Q_strcasecmp(name, "gl_one_minus_src_color"))
|
|
return GL_ONE_MINUS_SRC_COLOR;
|
|
if (!Q_strcasecmp(name, "gl_src_alpha"))
|
|
return GL_SRC_ALPHA;
|
|
if (!Q_strcasecmp(name, "gl_one_minus_src_alpha"))
|
|
return GL_ONE_MINUS_SRC_ALPHA;
|
|
if (!Q_strcasecmp(name, "gl_dst_alpha"))
|
|
return GL_DST_ALPHA;
|
|
if (!Q_strcasecmp(name, "gl_one_minus_dst_alpha"))
|
|
return GL_ONE_MINUS_DST_ALPHA;
|
|
if (!Q_strcasecmp(name, "gl_dst_color"))
|
|
return GL_DST_COLOR;
|
|
if (!Q_strcasecmp(name, "gl_one_minus_dst_color"))
|
|
return GL_ONE_MINUS_DST_COLOR;
|
|
if (!Q_strcasecmp(name, "gl_src_alpha_saturate"))
|
|
return GL_SRC_ALPHA_SATURATE;
|
|
if (!Q_strcasecmp(name, "gl_constant_color"))
|
|
return GL_CONSTANT_COLOR;
|
|
if (!Q_strcasecmp(name, "gl_one_minus_constant_color"))
|
|
return GL_ONE_MINUS_CONSTANT_COLOR;
|
|
if (!Q_strcasecmp(name, "gl_constant_alpha"))
|
|
return GL_CONSTANT_ALPHA;
|
|
if (!Q_strcasecmp(name, "gl_one_minus_constant_alpha"))
|
|
return GL_ONE_MINUS_CONSTANT_ALPHA;
|
|
else
|
|
return -1;
|
|
}
|
|
|
|
/*
|
|
=================
|
|
Mod_GetNextParm
|
|
md3 skin protoshaders
|
|
=================
|
|
*/
|
|
qboolean Mod_GetNextParm (char **data, char **output)
|
|
{
|
|
if (!*data) return false;
|
|
|
|
*output = COM_ParseExt (data, false);
|
|
if (!*data) {
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: EOF without closing brace\n");
|
|
return false;
|
|
}
|
|
if (!*output[0])
|
|
return false;
|
|
if (*output[0] == '}') {
|
|
// VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: closing brace without data\n");
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
=================
|
|
Mod_ParseWaveFunc
|
|
md3 skin protoshaders
|
|
=================
|
|
*/
|
|
qboolean Mod_ParseWaveFunc (char **data, waveForm_t *out)
|
|
{
|
|
char *tok=NULL;
|
|
|
|
if (!*data || !out) return false;
|
|
|
|
if (!Mod_GetNextParm(data, &tok))
|
|
return false;
|
|
|
|
if (!Q_strcasecmp(tok, "sin"))
|
|
*out = WAVEFORM_SIN;
|
|
else if (!Q_strcasecmp(tok, "triangle"))
|
|
*out = WAVEFORM_TRIANGLE;
|
|
else if (!Q_strcasecmp(tok, "square"))
|
|
*out = WAVEFORM_SQUARE;
|
|
else if (!Q_strcasecmp(tok, "sawtooth"))
|
|
*out = WAVEFORM_SAWTOOTH;
|
|
else if (!Q_strcasecmp(tok, "inversesawtooth"))
|
|
*out = WAVEFORM_INVERSESAWTOOTH;
|
|
else if (!Q_strcasecmp(tok, "noise"))
|
|
*out = WAVEFORM_NOISE;
|
|
else {
|
|
*out = -1;
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
=================
|
|
Mod_ParseFloat
|
|
md3 skin protoshaders
|
|
=================
|
|
*/
|
|
qboolean Mod_ParseFloat (char **data, float *outnum, qboolean normalized)
|
|
{
|
|
char *token=NULL;
|
|
|
|
if (!*data || !outnum) return false;
|
|
|
|
if (!Mod_GetNextParm(data, &token))
|
|
return false;
|
|
|
|
if (normalized)
|
|
*outnum = max(min(atof(token), 1.0f), 0.0f);
|
|
else
|
|
*outnum = atof(token);
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
=================
|
|
Mod_ParseModelScript
|
|
skin protoshaders
|
|
=================
|
|
*/
|
|
void Mod_ParseModelScript (maliasskin_t *skin, char **data, char *dataStart, int dataSize, char *meshname, int skinnum, char *scriptname)
|
|
{
|
|
char *token=NULL, *token2=NULL;
|
|
char glowname[MD3_MAX_PATH];
|
|
int i;
|
|
renderparms_t *skinParms = &skin->renderparms;
|
|
|
|
// get the opening curly brace
|
|
token = COM_ParseExt (data, true);
|
|
if (!*data) {
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: unexpected EOF in %s.%i in %s\n", meshname, skinnum, scriptname);
|
|
return;
|
|
}
|
|
if (token[0] != '{') {
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: found %s when expecting { in %s.%i in %s\n", token, meshname, skinnum, scriptname);
|
|
return;
|
|
}
|
|
|
|
// go through all the parms
|
|
while (*data < (dataStart + dataSize))
|
|
{
|
|
token = COM_ParseExt (data, true);
|
|
if (!*data) {
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: EOF in %s.%i in %s without closing brace\n", meshname, skinnum, scriptname);
|
|
break;
|
|
}
|
|
if (token[0] == '}') break; // end of skin
|
|
|
|
if (!Q_strcasecmp(token, "twosided"))
|
|
skinParms->twosided = true;
|
|
else if (!Q_strcasecmp(token, "alphatest")) {
|
|
skinParms->alphatest = true;
|
|
skinParms->noshadow = true; // also noshadow
|
|
}
|
|
else if (!Q_strcasecmp(token, "fullbright"))
|
|
skinParms->fullbright = true;
|
|
else if (!Q_strcasecmp(token, "nodraw")) {
|
|
skinParms->nodraw = true;
|
|
skinParms->noshadow = true; // also noshadow
|
|
}
|
|
else if (!Q_strcasecmp(token, "noshadow"))
|
|
skinParms->noshadow = true;
|
|
else if (!Q_strcasecmp(token, "nodiffuse"))
|
|
skinParms->nodiffuse = true;
|
|
else if (!Q_strcasecmp(token, "envmap"))
|
|
{
|
|
if (!Mod_ParseFloat(data, &skinParms->envmap, true)) {
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: missing parameter for 'envmap' in %s.%i in %s\n", meshname, skinnum, scriptname);
|
|
break;
|
|
}
|
|
}
|
|
else if (!Q_strcasecmp(token, "alpha") || !Q_strcasecmp(token, "trans"))
|
|
{
|
|
if (!Mod_ParseFloat(data, &skinParms->basealpha, true)) {
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: missing parameter for 'alpha' or 'trans' in %s.%i in %s\n", meshname, skinnum, scriptname);
|
|
break;
|
|
}
|
|
}
|
|
else if (!Q_strcasecmp(token, "tcmod"))
|
|
{
|
|
if (!Mod_GetNextParm(data, &token)) {
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: missing parameter for 'tcmod' in %s.%i in %s\n", meshname, skinnum, scriptname);
|
|
break;
|
|
}
|
|
if (!Q_strcasecmp(token, "translate")) {
|
|
if (!Mod_ParseFloat(data, &skinParms->tcmod.translate_x, false)) {
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: missing parameters for 'tcmod translate' in %s.%i in %s\n", meshname, skinnum, scriptname);
|
|
break;
|
|
}
|
|
if (!Mod_ParseFloat(data, &skinParms->tcmod.translate_y, false)) {
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: missing parameter for 'tcmod translate' in %s.%i in %s\n", meshname, skinnum, scriptname);
|
|
skinParms->tcmod.translate_x = 0.0f;
|
|
break;
|
|
}
|
|
}
|
|
else if (!Q_strcasecmp(token, "rotate")) {
|
|
if (!Mod_ParseFloat(data, &skinParms->tcmod.rotate, false)) {
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: missing parameter for 'tcmod rotate' in %s.%i in %s\n", meshname, skinnum, scriptname);
|
|
break;
|
|
}
|
|
}
|
|
else if (!Q_strcasecmp(token, "scale")) {
|
|
if (!Mod_ParseFloat(data, &skinParms->tcmod.scale_x, false)) {
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: missing parameters for 'tcmod scale' in %s.%i in %s\n", meshname, skinnum, scriptname);
|
|
break;
|
|
}
|
|
if (!Mod_ParseFloat(data, &skinParms->tcmod.scale_y, false)) {
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: missing parameter for 'tcmod scale' in %s.%i in %s\n", meshname, skinnum, scriptname);
|
|
skinParms->tcmod.scale_x = 1.0f;
|
|
break;
|
|
}
|
|
}
|
|
else if (!Q_strcasecmp(token, "stretch"))
|
|
{
|
|
if (!Mod_ParseWaveFunc(data, &skinParms->tcmod.stretch.type)) {
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: missing or invalid waveform for 'tcmod stretch' in %s.%i in %s\n", meshname, skinnum, scriptname);
|
|
break;
|
|
}
|
|
for (i=0; i<4; i++)
|
|
if (!Mod_ParseFloat(data, &skinParms->tcmod.stretch.params[i], false)) {
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: missing parameters for 'tcmod stretch' in %s.%i in %s\n", meshname, skinnum, scriptname);
|
|
break;
|
|
}
|
|
}
|
|
else if (!Q_strcasecmp(token, "turb"))
|
|
{ // first parm (base) is unused, so just read twice
|
|
for (i=0; i<4; i++)
|
|
if (!Mod_ParseFloat(data, &skinParms->tcmod.turb.params[i], false)) {
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: missing parameters for 'tcmod turb' in %s.%i in %s\n", meshname, skinnum, scriptname);
|
|
break;
|
|
}
|
|
skinParms->tcmod.turb.type = WAVEFORM_SIN;
|
|
}
|
|
else if (!Q_strcasecmp(token, "scroll"))
|
|
{
|
|
if (!Mod_ParseFloat(data, &skinParms->tcmod.scroll_x, false)) {
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: missing parameters for 'tcmod scroll' in %s.%i in %s\n", meshname, skinnum, scriptname);
|
|
break;
|
|
}
|
|
if (!Mod_ParseFloat(data, &skinParms->tcmod.scroll_y, false)) {
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: missing parameter for 'tcmod scroll' in %s.%i in %s\n", meshname, skinnum, scriptname);
|
|
skinParms->tcmod.scroll_x = 0.0f;
|
|
break;
|
|
}
|
|
}
|
|
else {
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: unknown type '%s' for 'tcmod' in %s.%i in %s\n", token, meshname, skinnum, scriptname);
|
|
break;
|
|
}
|
|
}
|
|
else if (!Q_strcasecmp(token, "blendfunc"))
|
|
{ // parse blend parm
|
|
if (!Mod_GetNextParm(data, &token)) {
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: missing parameter(s) for 'blendfunc' in %s.%i in %s\n", meshname, skinnum, scriptname);
|
|
break;
|
|
}
|
|
|
|
if (!Q_strcasecmp(token, "add")) {
|
|
skinParms->blendfunc_src = GL_ONE;
|
|
skinParms->blendfunc_dst = GL_ONE;
|
|
}
|
|
else if (!Q_strcasecmp(token, "filter")) {
|
|
skinParms->blendfunc_src = GL_DST_COLOR;
|
|
skinParms->blendfunc_dst = GL_ZERO;
|
|
}
|
|
else if (!Q_strcasecmp(token, "blend")) {
|
|
skinParms->blendfunc_src = GL_SRC_ALPHA;
|
|
skinParms->blendfunc_dst = GL_ONE_MINUS_SRC_ALPHA;
|
|
}
|
|
else
|
|
{ // parse 2nd blend parm
|
|
if (!Mod_GetNextParm(data, &token2)) {
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: missing 2nd parameter for 'blendfunc' in %s.%i in %s\n", meshname, skinnum, scriptname);
|
|
break;
|
|
}
|
|
|
|
skinParms->blendfunc_src = Mod_ParseBlendMode (token);
|
|
if (skinParms->blendfunc_src == -1) {
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: invalid src blend func in %s.%i in %s\n", meshname, skinnum, scriptname);
|
|
break;
|
|
}
|
|
|
|
skinParms->blendfunc_dst = Mod_ParseBlendMode (token2);
|
|
if (skinParms->blendfunc_dst == -1) {
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: invalid dst blend func in %s.%i in %s\n", meshname, skinnum, scriptname);
|
|
break;
|
|
}
|
|
}
|
|
skinParms->blend = true;
|
|
}
|
|
else if (!Q_strcasecmp(token, "glow"))
|
|
{
|
|
if (!Mod_GetNextParm(data, &token)) { // glowname
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: missing image name for 'glow' in %s.%i in %s\n", meshname, skinnum, scriptname);
|
|
break;
|
|
}
|
|
memcpy (glowname, token, MD3_MAX_PATH);
|
|
if (!Mod_GetNextParm(data, &token)) { // type
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: missing 'identity' or 'wave' for 'glow' in %s.%i in %s\n", meshname, skinnum, scriptname);
|
|
break;
|
|
}
|
|
if (!Q_strcasecmp(token, "wave")) {
|
|
if (!Mod_ParseWaveFunc(data, &skinParms->glow.type)) {
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: missing or invalid waveform for 'glow <glowskin> wave' in %s.%i in %s\n", meshname, skinnum, scriptname);
|
|
break;
|
|
}
|
|
for (i=0; i<4; i++)
|
|
if (!Mod_ParseFloat(data, &skinParms->glow.params[i], false)) {
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: missing parameters for 'glow <glowskin> wave' in %s.%i in %s\n", meshname, skinnum, scriptname);
|
|
break;
|
|
}
|
|
}
|
|
else if (!Q_strcasecmp(token, "identity"))
|
|
skinParms->glow.type = -1;
|
|
else { // only wave or identity
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: unknown type '%s' for 'glow' in %s.%i in %s\n", token, meshname, skinnum, scriptname);
|
|
break;
|
|
}
|
|
skin->glowimage = R_FindImage (glowname, it_skin);
|
|
if (skin->glowimage)
|
|
memcpy (skin->glowname, glowname, MD3_MAX_PATH);
|
|
}
|
|
else { // invalid parameter
|
|
VID_Printf (PRINT_ALL, S_COLOR_YELLOW"Mod_ParseModelScript: unknown parameter '%s' in %s.%i in %s\n", token, meshname, skinnum, scriptname);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
=================
|
|
Mod_SetTCModParmsDefaults
|
|
md3 skin protoshaders
|
|
=================
|
|
*/
|
|
void Mod_SetTCModParmsDefaults (tcmodParms_t *tcmod)
|
|
{
|
|
if (!tcmod) return;
|
|
|
|
tcmod->translate_x = 0.0f;
|
|
tcmod->translate_y = 0.0f;
|
|
tcmod->rotate = 0.0f;
|
|
tcmod->scale_x = 1.0f;
|
|
tcmod->scale_y = 1.0f;
|
|
tcmod->stretch.type = -1;
|
|
tcmod->turb.type = -1;
|
|
tcmod->scroll_x = 0.0f;
|
|
tcmod->scroll_y = 0.0f;
|
|
}
|
|
|
|
/*
|
|
=================
|
|
Mod_SetRenderParmsDefaults
|
|
md3 skin protoshaders
|
|
=================
|
|
*/
|
|
void Mod_SetRenderParmsDefaults (renderparms_t *parms)
|
|
{
|
|
parms->twosided = false;
|
|
parms->alphatest = false;
|
|
parms->fullbright = false;
|
|
parms->nodraw = false;
|
|
parms->noshadow = false;
|
|
parms->nodiffuse = false;
|
|
parms->envmap = 0.0f;
|
|
parms->basealpha = 1.0f;
|
|
parms->blend = false;
|
|
parms->blendfunc_src = -1;
|
|
parms->blendfunc_dst = -1;
|
|
parms->glow.type = -1;
|
|
parms->tcmod.translate_x = 0.0f;
|
|
parms->tcmod.translate_y = 0.0f;
|
|
parms->tcmod.rotate = 0.0f;
|
|
parms->tcmod.scale_x = 1.0f;
|
|
parms->tcmod.scale_y = 1.0f;
|
|
parms->tcmod.stretch.type = -1;
|
|
parms->tcmod.turb.type = -1;
|
|
parms->tcmod.scroll_x = 0.0f;
|
|
parms->tcmod.scroll_y = 0.0f;
|
|
}
|
|
|
|
/*
|
|
=================
|
|
Mod_LoadModelScript
|
|
md3 skin protoshaders
|
|
=================
|
|
*/
|
|
void Mod_LoadModelScript (model_t *mod, maliasmodel_t *aliasmod)
|
|
{
|
|
char scriptname[MAX_QPATH];
|
|
char *buf, *parse_data, *token=NULL, *token2=NULL;
|
|
int buf_size, i, j; //, k;
|
|
qboolean skinname_found;
|
|
renderparms_t *skinParms;
|
|
|
|
// set defaults
|
|
for (i=0; i < aliasmod->num_meshes; i++)
|
|
for (j=0; j < aliasmod->meshes[i].num_skins; j++) {
|
|
Mod_SetRenderParmsDefaults (&aliasmod->meshes[i].skins[j].renderparms);
|
|
aliasmod->meshes[i].skins[j].glowimage = NULL;
|
|
}
|
|
|
|
COM_StripExtension (mod->name, scriptname, sizeof(scriptname));
|
|
// strncat (scriptname, ".script");
|
|
Q_strncatz (scriptname, sizeof(scriptname), ".script");
|
|
buf_size = FS_LoadFile (scriptname, &buf);
|
|
|
|
if (buf_size < 1)
|
|
return;
|
|
//else
|
|
// VID_Printf (PRINT_ALL, "Mod_LoadModelScript: loaded model script for %s with %i meshes.\n", mod->name, aliasmod->num_meshes);
|
|
|
|
for (i=0; i<aliasmod->num_meshes; i++)
|
|
{
|
|
for (j=0; j<aliasmod->meshes[i].num_skins; j++)
|
|
{
|
|
skinParms = &aliasmod->meshes[i].skins[j].renderparms;
|
|
|
|
// search the script file for that meshname
|
|
skinname_found = false;
|
|
parse_data = buf; // copy data postion
|
|
while (parse_data < (buf + buf_size))
|
|
{
|
|
token = COM_Parse (&parse_data);
|
|
if (!parse_data) break;
|
|
if (!token) break;
|
|
|
|
if ( !Q_strncasecmp(token, aliasmod->meshes[i].name, strlen(aliasmod->meshes[i].name))
|
|
&& (atoi(COM_FileExtension(token)) == j)) {
|
|
skinname_found = true;
|
|
break;
|
|
}
|
|
}
|
|
// then if that skinname is found, load the parms for that skin
|
|
if (skinname_found)
|
|
Mod_ParseModelScript (&aliasmod->meshes[i].skins[j], &parse_data, buf, buf_size, aliasmod->meshes[i].name, j, scriptname);
|
|
}
|
|
}
|
|
FS_FreeFile(buf);
|
|
|
|
// check if model has any alpha surfaces for sorting
|
|
for (i=0; i < aliasmod->num_meshes; i++)
|
|
for (j=0; j < aliasmod->meshes[i].num_skins; j++)
|
|
if (aliasmod->meshes[i].skins[j].renderparms.blend || aliasmod->meshes[i].skins[j].renderparms.basealpha < 1.0f)
|
|
mod->hasAlpha = true;
|
|
|
|
//VID_Printf (PRINT_ALL, "Mod_LoadModelScript: loaded model script for %s with %i bytes per skin.\n", mod->name, sizeof(renderparms_t));
|
|
}
|
|
|
|
#ifndef MD2_AS_MD3
|
|
|
|
#ifdef PROJECTION_SHADOWS // projection shadows from BeefQuake R6
|
|
/*
|
|
=================
|
|
Mod_FindMD2TriangleWithEdge
|
|
=================
|
|
*/
|
|
signed int Mod_FindMD2TriangleWithEdge (short p1, short p2, dmd2triangle_t *ignore, dmd2_t *hdr)
|
|
{
|
|
dmd2triangle_t *tris = (dmd2triangle_t *)((unsigned char*)hdr + hdr->ofs_tris);
|
|
int i, match, count;
|
|
|
|
count = 0;
|
|
match = -1;
|
|
|
|
for (i=0; i<hdr->num_tris; i++, tris++)
|
|
{
|
|
if ( (tris->index_xyz[0] == p2 && tris->index_xyz[1] == p1)
|
|
|| (tris->index_xyz[1] == p2 && tris->index_xyz[2] == p1)
|
|
|| (tris->index_xyz[2] == p2 && tris->index_xyz[0] == p1) ) {
|
|
if (tris != ignore)
|
|
match = i;
|
|
count++;
|
|
}
|
|
else if ( (tris->index_xyz[0] == p1 && tris->index_xyz[1] == p2)
|
|
|| (tris->index_xyz[1] == p1 && tris->index_xyz[2] == p2)
|
|
|| (tris->index_xyz[2] == p1 && tris->index_xyz[0] == p2) )
|
|
count++;
|
|
}
|
|
|
|
if (count > 2)
|
|
match = -1;
|
|
return match;
|
|
}
|
|
|
|
|
|
/*
|
|
=================
|
|
Mod_BuildMD2TriangleNeighbors
|
|
=================
|
|
*/
|
|
void Mod_BuildMD2TriangleNeighbors (model_t *mod)
|
|
{
|
|
dmd2_t *hdr = (dmd2_t *)mod->extradata;
|
|
dmd2triangle_t *tris = (dmd2triangle_t *)((unsigned char*)hdr + hdr->ofs_tris);
|
|
int i, *n;
|
|
|
|
for (i=0, n=mod->edge_tri; i<hdr->num_tris; i++, n+=3, tris++)
|
|
{
|
|
n[0] = Mod_FindMD2TriangleWithEdge(tris->index_xyz[0], tris->index_xyz[1], tris, hdr);
|
|
n[1] = Mod_FindMD2TriangleWithEdge(tris->index_xyz[1], tris->index_xyz[2], tris, hdr);
|
|
n[2] = Mod_FindMD2TriangleWithEdge(tris->index_xyz[2], tris->index_xyz[0], tris, hdr);
|
|
}
|
|
}
|
|
#endif // end projection shadows from BeefQuake R6
|
|
|
|
|
|
/*
|
|
=================
|
|
Mod_GetAllocSizeMD2Old
|
|
|
|
Calc exact alloc size for MD2 in memory
|
|
=================
|
|
*/
|
|
size_t Mod_GetAllocSizeMD2Old (model_t *mod, void *buffer)
|
|
{
|
|
dmd2_t *pinmodel;
|
|
size_t allocSize;
|
|
|
|
pinmodel = (dmd2_t *)buffer;
|
|
allocSize = (LittleLong(pinmodel->ofs_end) + 31) & ~31;
|
|
return allocSize;
|
|
}
|
|
|
|
|
|
/*
|
|
=================
|
|
Mod_LoadAliasMD2ModelOld
|
|
=================
|
|
*/
|
|
void Mod_LoadAliasMD2ModelOld (model_t *mod, void *buffer)
|
|
{
|
|
int i, j;
|
|
dmd2_t *pinmodel, *pheader;
|
|
dmd2coord_t *pinst, *poutst;
|
|
dmd2triangle_t *pintri, *pouttri;
|
|
dmd2frame_t *pinframe, *poutframe;
|
|
int *pincmd, *poutcmd;
|
|
int version;
|
|
|
|
pinmodel = (dmd2_t *)buffer;
|
|
|
|
version = LittleLong (pinmodel->version);
|
|
if (version != ALIAS_VERSION)
|
|
VID_Error (ERR_DROP, "%s has wrong version number (%i should be %i)",
|
|
mod->name, version, ALIAS_VERSION);
|
|
|
|
// pheader = Hunk_Alloc (LittleLong(pinmodel->ofs_end));
|
|
pheader = ModChunk_Alloc (LittleLong(pinmodel->ofs_end));
|
|
|
|
// byte swap the header fields and sanity check
|
|
for (i=0; i<sizeof(dmd2_t)/4; i++)
|
|
((int *)pheader)[i] = LittleLong (((int *)buffer)[i]);
|
|
|
|
if (pheader->skinheight > MAX_LBM_HEIGHT)
|
|
VID_Error (ERR_DROP, "model %s has a skin taller than %d", mod->name,
|
|
MAX_LBM_HEIGHT);
|
|
|
|
if (pheader->num_xyz <= 0)
|
|
VID_Error (ERR_DROP, "model %s has no vertices", mod->name);
|
|
|
|
if (pheader->num_xyz > MAX_VERTS)
|
|
VID_Error (ERR_DROP, "model %s has too many vertices", mod->name);
|
|
|
|
if (pheader->num_st <= 0)
|
|
VID_Error (ERR_DROP, "model %s has no st vertices", mod->name);
|
|
|
|
if (pheader->num_tris <= 0)
|
|
VID_Error (ERR_DROP, "model %s has no triangles", mod->name);
|
|
|
|
if (pheader->num_frames <= 0)
|
|
VID_Error (ERR_DROP, "model %s has no frames", mod->name);
|
|
|
|
//
|
|
// load base s and t vertices (not used in gl version)
|
|
//
|
|
pinst = (dmd2coord_t *) ((byte *)pinmodel + pheader->ofs_st);
|
|
poutst = (dmd2coord_t *) ((byte *)pheader + pheader->ofs_st);
|
|
|
|
for (i=0; i<pheader->num_st; i++)
|
|
{
|
|
poutst[i].s = LittleShort (pinst[i].s);
|
|
poutst[i].t = LittleShort (pinst[i].t);
|
|
}
|
|
|
|
//
|
|
// load triangle lists
|
|
//
|
|
pintri = (dmd2triangle_t *) ((byte *)pinmodel + pheader->ofs_tris);
|
|
pouttri = (dmd2triangle_t *) ((byte *)pheader + pheader->ofs_tris);
|
|
|
|
for (i=0; i<pheader->num_tris; i++)
|
|
{
|
|
for (j=0; j<3; j++)
|
|
{
|
|
pouttri[i].index_xyz[j] = LittleShort (pintri[i].index_xyz[j]);
|
|
pouttri[i].index_st[j] = LittleShort (pintri[i].index_st[j]);
|
|
}
|
|
}
|
|
|
|
//
|
|
// load the frames
|
|
//
|
|
for (i=0; i<pheader->num_frames; i++)
|
|
{
|
|
pinframe = (dmd2frame_t *) ((byte *)pinmodel
|
|
+ pheader->ofs_frames + i * pheader->framesize);
|
|
poutframe = (dmd2frame_t *) ((byte *)pheader
|
|
+ pheader->ofs_frames + i * pheader->framesize);
|
|
|
|
memcpy (poutframe->name, pinframe->name, sizeof(poutframe->name));
|
|
for (j=0 ; j<3 ; j++)
|
|
{
|
|
poutframe->scale[j] = LittleFloat (pinframe->scale[j]);
|
|
poutframe->translate[j] = LittleFloat (pinframe->translate[j]);
|
|
}
|
|
// verts are all 8 bit, so no swapping needed
|
|
memcpy (poutframe->verts, pinframe->verts,
|
|
pheader->num_xyz*sizeof(dmd2vertex_t));
|
|
}
|
|
|
|
//
|
|
// load the glcmds
|
|
//
|
|
pincmd = (int *) ((byte *)pinmodel + pheader->ofs_glcmds);
|
|
poutcmd = (int *) ((byte *)pheader + pheader->ofs_glcmds);
|
|
for (i=0; i<pheader->num_glcmds; i++)
|
|
poutcmd[i] = LittleLong (pincmd[i]);
|
|
|
|
// register all skins
|
|
memcpy ((char *)pheader + pheader->ofs_skins, (char *)pinmodel + pheader->ofs_skins,
|
|
pheader->num_skins*MAX_SKINNAME);
|
|
for (i=0; i<pheader->num_skins; i++)
|
|
{
|
|
mod->skins[0][i] = R_FindImage ((char *)pheader + pheader->ofs_skins + i*MAX_SKINNAME, it_skin);
|
|
}
|
|
|
|
mod->mins[0] = -32;
|
|
mod->mins[1] = -32;
|
|
mod->mins[2] = -32;
|
|
mod->maxs[0] = 32;
|
|
mod->maxs[1] = 32;
|
|
mod->maxs[2] = 32;
|
|
|
|
#ifdef PROJECTION_SHADOWS // projection shadows from BeefQuake R6
|
|
mod->edge_tri = malloc(sizeof(int) * pheader->num_tris * 3);
|
|
Mod_BuildMD2TriangleNeighbors (mod);
|
|
#endif // end projection shadows from BeefQuake R6
|
|
|
|
mod->hasAlpha = false;
|
|
// Mod_LoadMD2ModelScript (mod, pheader); // md2 skin scripting
|
|
|
|
mod->type = mod_md2;
|
|
}
|
|
#endif // MD2_AS_MD3
|
|
|
|
|
|
//Harven++ MD3
|
|
|
|
//
|
|
// Some Vic code here not fully used
|
|
//
|
|
|
|
/*
|
|
===============
|
|
Mod_FindTriangleWithEdge
|
|
===============
|
|
*/
|
|
int Mod_FindTriangleWithEdge (maliasmesh_t *mesh, index_t p1, index_t p2, int ignore)
|
|
{
|
|
int i, match, count;
|
|
index_t *indexes;
|
|
|
|
count = 0;
|
|
match = -1;
|
|
|
|
for (i=0, indexes=mesh->indexes; i<mesh->num_tris; i++, indexes+=3)
|
|
{
|
|
if ( (indexes[0] == p2 && indexes[1] == p1)
|
|
|| (indexes[1] == p2 && indexes[2] == p1)
|
|
|| (indexes[2] == p2 && indexes[0] == p1) ) {
|
|
if (i != ignore)
|
|
match = i;
|
|
count++;
|
|
}
|
|
else if ( (indexes[0] == p1 && indexes[1] == p2)
|
|
|| (indexes[1] == p1 && indexes[2] == p2)
|
|
|| (indexes[2] == p1 && indexes[0] == p2) )
|
|
count++;
|
|
}
|
|
|
|
// detect edges shared by three triangles and make them seams
|
|
if (count > 2)
|
|
match = -1;
|
|
|
|
return match;
|
|
}
|
|
|
|
/*
|
|
===============
|
|
Mod_BuildTriangleNeighbors
|
|
===============
|
|
*/
|
|
void Mod_BuildTriangleNeighbors (maliasmesh_t *mesh)
|
|
{
|
|
int i, *n;
|
|
index_t *index;
|
|
|
|
for (i=0, n=mesh->trneighbors, index=mesh->indexes; i<mesh->num_tris; i++, n+=3, index+=3)
|
|
{
|
|
n[0] = Mod_FindTriangleWithEdge (mesh, index[0], index[1], i);
|
|
n[1] = Mod_FindTriangleWithEdge (mesh, index[1], index[2], i);
|
|
n[2] = Mod_FindTriangleWithEdge (mesh, index[2], index[0], i);
|
|
}
|
|
}
|
|
|
|
|
|
#ifdef MD2_AS_MD3
|
|
/*
|
|
=================
|
|
NormalToLatLong
|
|
=================
|
|
*/
|
|
void NormalToLatLong (const vec3_t normal, byte bytes[2])
|
|
{
|
|
int lat, lng;
|
|
|
|
if (normal[0] == 0 && normal[1] == 0){
|
|
if (normal[2] > 0){
|
|
// Lattitude = 0, Longitude = 0
|
|
bytes[0] = 0;
|
|
bytes[1] = 0;
|
|
}
|
|
else {
|
|
// Lattitude = 0, Longitude = 128
|
|
bytes[0] = 128;
|
|
bytes[1] = 0;
|
|
}
|
|
}
|
|
else {
|
|
lat = RAD2DEG(atan2(normal[1], normal[0])) * (255.0 / 360.0);
|
|
lng = RAD2DEG(acos(normal[2])) * (255.0 / 360.0);
|
|
|
|
bytes[0] = lng & 0xff;
|
|
bytes[1] = lat & 0xff;
|
|
}
|
|
}
|
|
|
|
|
|
int md2IndRemap[MD2_MAX_TRIANGLES*3];
|
|
unsigned int md2TempIndex[MD2_MAX_TRIANGLES*3];
|
|
unsigned int md2TempStIndex[MD2_MAX_TRIANGLES*3];
|
|
/*
|
|
=================
|
|
Mod_GetAllocSizeMD2New
|
|
Calc exact alloc size for MD2 in memory
|
|
=================
|
|
*/
|
|
size_t Mod_GetAllocSizeMD2New (model_t *mod, void *buffer)
|
|
{
|
|
int i, j, numFrames, numMeshes, numTris, numIndex, numVerts, numSkins;
|
|
dmd2_t *pinmodel;
|
|
dmd2triangle_t *pintri;
|
|
// int md2IndRemap[MAX_TRIANGLES*3];
|
|
// unsigned int md2TempIndex[MAX_TRIANGLES*3], md2TempStIndex[MAX_TRIANGLES*3];
|
|
size_t headerSize, meshSize, indexSize, coordSize, frameSize, vertSize, trNeighborsSize, skinSize;
|
|
size_t allocSize;
|
|
|
|
pinmodel = (dmd2_t *)buffer;
|
|
|
|
numFrames = LittleLong(pinmodel->num_frames);
|
|
numMeshes = 1;
|
|
numTris = LittleLong(pinmodel->num_tris);
|
|
|
|
// count unique verts
|
|
// numVerts = LittleLong(pinmodel->num_xyz);
|
|
numVerts = 0;
|
|
numIndex = numTris * 3;
|
|
pintri = (dmd2triangle_t *)((byte *)pinmodel + LittleLong(pinmodel->ofs_tris));
|
|
|
|
for (i=0; i < numTris; i++)
|
|
{
|
|
md2TempIndex[i*3+0] = (unsigned)LittleShort(pintri[i].index_xyz[0]);
|
|
md2TempIndex[i*3+1] = (unsigned)LittleShort(pintri[i].index_xyz[1]);
|
|
md2TempIndex[i*3+2] = (unsigned)LittleShort(pintri[i].index_xyz[2]);
|
|
|
|
md2TempStIndex[i*3+0] = (unsigned)LittleShort(pintri[i].index_st[0]);
|
|
md2TempStIndex[i*3+1] = (unsigned)LittleShort(pintri[i].index_st[1]);
|
|
md2TempStIndex[i*3+2] = (unsigned)LittleShort(pintri[i].index_st[2]);
|
|
}
|
|
memset(md2IndRemap, -1, MD2_MAX_TRIANGLES * 3 * sizeof(int));
|
|
for (i=0; i < numIndex; i++)
|
|
{
|
|
if (md2IndRemap[i] != -1) continue;
|
|
for (j=0; j < numIndex; j++)
|
|
{
|
|
if (j == i) continue;
|
|
if ((md2TempIndex[j] == md2TempIndex[i]) && (md2TempStIndex[j] == md2TempStIndex[i]))
|
|
md2IndRemap[j] = i;
|
|
}
|
|
}
|
|
for (i=0; i < numIndex; i++)
|
|
{
|
|
if (md2IndRemap[i] == -1) numVerts++;
|
|
}
|
|
|
|
numSkins = max(LittleLong(pinmodel->num_skins), 1); // hack because player models have no skin refs
|
|
|
|
// calc sizes rounded to cacheline
|
|
headerSize = (sizeof(maliasmodel_t) + 31) & ~31;
|
|
meshSize = ((sizeof(maliasmesh_t) * numMeshes) + 31) & ~31;
|
|
indexSize = ((sizeof(index_t) * numTris * 3) + 31) & ~31;
|
|
coordSize = ((sizeof(maliascoord_t) * numVerts) + 31) & ~31;
|
|
frameSize = ((sizeof(maliasframe_t) * numFrames) + 31) & ~31;
|
|
vertSize = ((numFrames * numVerts * sizeof(maliasvertex_t)) + 31) & ~31;
|
|
trNeighborsSize = ((sizeof(int) * numTris * 3) + 31) & ~31;
|
|
skinSize = ((sizeof(maliasskin_t) * numSkins) + 31) & ~31;
|
|
|
|
allocSize = headerSize + meshSize + indexSize + coordSize + frameSize + vertSize + trNeighborsSize + skinSize;
|
|
|
|
return allocSize;
|
|
}
|
|
|
|
|
|
/*
|
|
=================
|
|
Mod_LoadAliasMD2ModelNew
|
|
Based on md2 loading code in Q2E 0.40
|
|
|
|
Mod_GetAllocSizeMD2() must be called first to init
|
|
index and remap arrays.
|
|
=================
|
|
*/
|
|
void Mod_LoadAliasMD2ModelNew (model_t *mod, void *buffer)
|
|
{
|
|
int i, j;
|
|
int version;
|
|
dmd2_t *pinmodel;
|
|
dmd2frame_t *pinframe;
|
|
dmd2triangle_t *pintri;
|
|
dmd2coord_t *pincoord;
|
|
maliasmodel_t *poutmodel;
|
|
maliasframe_t *poutframe;
|
|
maliasmesh_t *poutmesh;
|
|
maliasskin_t *poutskin;
|
|
maliasvertex_t *poutvert;
|
|
maliascoord_t *poutcoord;
|
|
index_t *poutindex;
|
|
char name[MD3_MAX_PATH];
|
|
// int md2IndRemap[MD2_MAX_TRIANGLES*3];
|
|
// unsigned md2TempIndex[MD2_MAX_TRIANGLES*3], md2TempStIndex[MD2_MAX_TRIANGLES*3];
|
|
int numIndices, numVertices;
|
|
double skinWidth, skinHeight;
|
|
vec3_t normal;
|
|
|
|
pinmodel = (dmd2_t *)buffer;
|
|
|
|
// poutmodel = Hunk_Alloc (sizeof(maliasmodel_t));
|
|
poutmodel = ModChunk_Alloc (sizeof(maliasmodel_t));
|
|
|
|
// byte swap the header fields and sanity check
|
|
version = LittleLong (pinmodel->version);
|
|
if (version != MD2_ALIAS_VERSION) {
|
|
VID_Error (ERR_DROP, "%s has wrong version number (%i should be %i)",
|
|
mod->name, version, MD2_ALIAS_VERSION);
|
|
}
|
|
|
|
poutmodel->num_frames = LittleLong ( pinmodel->num_frames );
|
|
if (poutmodel->num_frames > MD2_MAX_FRAMES || poutmodel->num_frames <= 0)
|
|
VID_Error (ERR_DROP, "model %s has invalid number of frames (%i)", mod->name, poutmodel->num_frames);
|
|
|
|
poutmodel->num_tags = 0;
|
|
poutmodel->num_meshes = 1;
|
|
|
|
skinWidth = 1.0 / (double)LittleLong(pinmodel->skinwidth);
|
|
skinHeight = 1.0 / (double)LittleLong(pinmodel->skinheight);
|
|
|
|
//
|
|
// load mesh info
|
|
//
|
|
// poutmesh = poutmodel->meshes = Hunk_Alloc(sizeof(maliasmesh_t));
|
|
poutmesh = poutmodel->meshes = ModChunk_Alloc(sizeof(maliasmesh_t));
|
|
|
|
Com_sprintf(poutmesh->name, sizeof(poutmesh->name), "md2mesh"); // mesh name in script must match this
|
|
|
|
poutmesh->num_tris = LittleLong(pinmodel->num_tris);
|
|
if (poutmesh->num_tris > MD2_MAX_TRIANGLES || poutmesh->num_tris <= 0)
|
|
VID_Error (ERR_DROP, "model %s has invalid number of triangles (%i)", mod->name, poutmesh->num_tris);
|
|
|
|
poutmesh->num_verts = LittleLong(pinmodel->num_xyz);
|
|
if (poutmesh->num_verts > MD2_MAX_VERTS || poutmesh->num_verts <= 0)
|
|
VID_Error (ERR_DROP, "model %s has invalid number of vertices (%i)", mod->name, poutmesh->num_verts);
|
|
|
|
poutmesh->num_skins = LittleLong(pinmodel->num_skins);
|
|
if (poutmesh->num_skins > MD2_MAX_SKINS || poutmesh->num_skins < 0)
|
|
Com_Error(ERR_DROP, "model %s has invalid number of skins (%i)", mod->name, poutmesh->num_skins);
|
|
|
|
pintri = (dmd2triangle_t *)((byte *)pinmodel + LittleLong(pinmodel->ofs_tris));
|
|
|
|
// DO NOT DELETE!!! Needed if Mod_GetAllocSizeMD2() is removed!
|
|
/* for (i=0; i < poutmesh->num_tris; i++)
|
|
{
|
|
md2TempIndex[i*3+0] = (unsigned)LittleShort(pintri[i].index_xyz[0]);
|
|
md2TempIndex[i*3+1] = (unsigned)LittleShort(pintri[i].index_xyz[1]);
|
|
md2TempIndex[i*3+2] = (unsigned)LittleShort(pintri[i].index_xyz[2]);
|
|
|
|
md2TempStIndex[i*3+0] = (unsigned)LittleShort(pintri[i].index_st[0]);
|
|
md2TempStIndex[i*3+1] = (unsigned)LittleShort(pintri[i].index_st[1]);
|
|
md2TempStIndex[i*3+2] = (unsigned)LittleShort(pintri[i].index_st[2]);
|
|
}*/
|
|
|
|
//
|
|
// build list of unique vertices
|
|
//
|
|
numIndices = poutmesh->num_tris * 3;
|
|
numVertices = 0;
|
|
|
|
// poutindex = poutmesh->indexes = Hunk_Alloc( sizeof(index_t) * poutmesh->num_tris * 3 );
|
|
poutindex = poutmesh->indexes = ModChunk_Alloc(sizeof(index_t) * poutmesh->num_tris * 3 );
|
|
|
|
// DO NOT DELETE!!! Needed if Mod_GetAllocSizeMD2() is removed!
|
|
/* memset(md2IndRemap, -1, MD2_MAX_TRIANGLES * 3 * sizeof(int));
|
|
|
|
for (i=0; i < numIndices; i++)
|
|
{
|
|
if (md2IndRemap[i] != -1)
|
|
continue;
|
|
for (j=0; j < numIndices; j++)
|
|
{
|
|
if (j == i)
|
|
continue;
|
|
if ((md2TempIndex[j] == md2TempIndex[i]) && (md2TempStIndex[j] == md2TempStIndex[i]))
|
|
md2IndRemap[j] = i;
|
|
}
|
|
}*/
|
|
|
|
//
|
|
// count unique vertices
|
|
//
|
|
for (i=0; i < numIndices; i++)
|
|
{
|
|
if (md2IndRemap[i] != -1)
|
|
continue;
|
|
poutindex[i] = numVertices++;
|
|
md2IndRemap[i] = i;
|
|
}
|
|
poutmesh->num_verts = numVertices;
|
|
|
|
//
|
|
// remap remaining indices
|
|
//
|
|
for (i=0; i < numIndices; i++) {
|
|
if (md2IndRemap[i] != i)
|
|
poutindex[i] = poutindex[md2IndRemap[i]];
|
|
}
|
|
|
|
//
|
|
// load base S and T vertices
|
|
//
|
|
pincoord = (dmd2coord_t *)((byte *)pinmodel + LittleLong(pinmodel->ofs_st));
|
|
// poutcoord = poutmesh->stcoords = Hunk_Alloc ( sizeof(maliascoord_t) * poutmesh->num_verts );
|
|
poutcoord = poutmesh->stcoords = ModChunk_Alloc (sizeof(maliascoord_t) * poutmesh->num_verts);
|
|
|
|
for (i=0; i < numIndices; i++) {
|
|
poutcoord[poutindex[i]].st[0] = (float)(((double)LittleShort(pincoord[md2TempStIndex[md2IndRemap[i]]].s) + 0.5) * skinWidth);
|
|
poutcoord[poutindex[i]].st[1] = (float)(((double)LittleShort(pincoord[md2TempStIndex[md2IndRemap[i]]].t) + 0.5) * skinHeight);
|
|
}
|
|
|
|
//
|
|
// load the frames
|
|
//
|
|
// poutframe = poutmodel->frames = Hunk_Alloc ( sizeof(maliasframe_t) * poutmodel->num_frames );
|
|
// poutvert = poutmesh->vertexes = Hunk_Alloc ( poutmodel->num_frames * poutmesh->num_verts * sizeof(maliasvertex_t) );
|
|
poutframe = poutmodel->frames = ModChunk_Alloc (sizeof(maliasframe_t) * poutmodel->num_frames);
|
|
poutvert = poutmesh->vertexes = ModChunk_Alloc (poutmodel->num_frames * poutmesh->num_verts * sizeof(maliasvertex_t));
|
|
|
|
mod->radius = 0;
|
|
ClearBounds(mod->mins, mod->maxs);
|
|
|
|
for (i=0; i < poutmodel->num_frames; i++, pinframe++, poutframe++, poutvert += numVertices)
|
|
{
|
|
pinframe = (dmd2frame_t *)((byte *)pinmodel + LittleLong(pinmodel->ofs_frames) + i*LittleLong(pinmodel->framesize));
|
|
|
|
poutframe->scale[0] = LittleFloat(pinframe->scale[0]);
|
|
poutframe->scale[1] = LittleFloat(pinframe->scale[1]);
|
|
poutframe->scale[2] = LittleFloat(pinframe->scale[2]);
|
|
|
|
poutframe->translate[0] = LittleFloat(pinframe->translate[0]);
|
|
poutframe->translate[1] = LittleFloat(pinframe->translate[1]);
|
|
poutframe->translate[2] = LittleFloat(pinframe->translate[2]);
|
|
|
|
VectorCopy (poutframe->translate, poutframe->mins);
|
|
VectorMA (poutframe->translate, 255, poutframe->scale, poutframe->maxs);
|
|
|
|
poutframe->radius = RadiusFromBounds(poutframe->mins, poutframe->maxs);
|
|
|
|
mod->radius = max ( mod->radius, poutframe->radius );
|
|
AddPointToBounds(poutframe->mins, mod->mins, mod->maxs);
|
|
AddPointToBounds(poutframe->maxs, mod->mins, mod->maxs);
|
|
|
|
//
|
|
// load the vertexes and normals
|
|
//
|
|
for (j=0; j < numIndices; j++)
|
|
{
|
|
poutvert[poutindex[j]].xyz[0] = (short)pinframe->verts[md2TempIndex[md2IndRemap[j]]].v[0];
|
|
poutvert[poutindex[j]].xyz[1] = (short)pinframe->verts[md2TempIndex[md2IndRemap[j]]].v[1];
|
|
poutvert[poutindex[j]].xyz[2] = (short)pinframe->verts[md2TempIndex[md2IndRemap[j]]].v[2];
|
|
|
|
poutvert[poutindex[j]].lightnormalindex = pinframe->verts[md2TempIndex[md2IndRemap[j]]].lightnormalindex;
|
|
|
|
normal[0] = r_avertexnormals[poutvert[poutindex[j]].lightnormalindex][0];
|
|
normal[1] = r_avertexnormals[poutvert[poutindex[j]].lightnormalindex][1];
|
|
normal[2] = r_avertexnormals[poutvert[poutindex[j]].lightnormalindex][2];
|
|
|
|
NormalToLatLong(normal, poutvert[poutindex[j]].normal);
|
|
}
|
|
}
|
|
|
|
//
|
|
// build triangle neighbors
|
|
//
|
|
// poutmesh->trneighbors = Hunk_Alloc ( sizeof(int) * poutmesh->num_tris * 3 );
|
|
poutmesh->trneighbors = ModChunk_Alloc (sizeof(int) * poutmesh->num_tris * 3);
|
|
Mod_BuildTriangleNeighbors (poutmesh);
|
|
|
|
//
|
|
// register all skins
|
|
//
|
|
if (poutmesh->num_skins <= 0) // hack for player models with no skin refs
|
|
{
|
|
// poutskin = poutmesh->skins = Hunk_Alloc ( sizeof(maliasskin_t) * 1 );
|
|
poutskin = poutmesh->skins = ModChunk_Alloc (sizeof(maliasskin_t) * 1);
|
|
poutmesh->num_skins = 1;
|
|
Com_sprintf(name, sizeof(name), "players/male/grunt.pcx");
|
|
memcpy(poutskin->name, name, MD3_MAX_PATH);
|
|
Com_sprintf(poutskin->glowname, sizeof(poutskin->glowname), "\0"); // set null glowskin
|
|
mod->skins[0][0] = R_FindImage (name, it_skin);
|
|
}
|
|
else
|
|
{
|
|
// poutskin = poutmesh->skins = Hunk_Alloc ( sizeof(maliasskin_t) * poutmesh->num_skins );
|
|
poutskin = poutmesh->skins = ModChunk_Alloc (sizeof(maliasskin_t) * poutmesh->num_skins);
|
|
for (i=0; i < poutmesh->num_skins; i++, poutskin++)
|
|
{
|
|
memcpy(name, ((char *)pinmodel + LittleLong(pinmodel->ofs_skins) + i*MD2_MAX_SKINNAME), MD3_MAX_PATH);
|
|
memcpy(poutskin->name, name, MD3_MAX_PATH);
|
|
Com_sprintf(poutskin->glowname, sizeof(poutskin->glowname), "\0"); // set null glowskin
|
|
mod->skins[0][i] = R_FindImage (name, it_skin);
|
|
}
|
|
}
|
|
|
|
mod->hasAlpha = false;
|
|
Mod_LoadModelScript (mod, poutmodel); // md3 skin scripting
|
|
|
|
//if (mod->hasAlpha)
|
|
// VID_Printf (PRINT_ALL, "Mod_LoadAliasMD2Model: model %s has trans mesh\n", mod->name);
|
|
|
|
mod->type = mod_alias;
|
|
}
|
|
#endif // MD2_AS_MD3
|
|
|
|
|
|
//
|
|
// And here some Vic and some my (Harven)
|
|
//
|
|
|
|
/*
|
|
=================
|
|
Mod_GetAllocSizeMD3
|
|
Calc exact alloc size for MD3 in memory
|
|
=================
|
|
*/
|
|
size_t Mod_GetAllocSizeMD3 (model_t *mod, void *buffer)
|
|
{
|
|
int i, numFrames, numTags, numMeshes, numSkins, numTris, numVerts;
|
|
dmd3_t *pinmodel;
|
|
dmd3mesh_t *pinmesh;
|
|
size_t headerSize, frameSize, tagSize, meshSize;
|
|
size_t skinSize=0, indexSize=0, coordSize=0, vertSize=0, trNeighborsSize=0;
|
|
size_t allocSize;
|
|
|
|
pinmodel = (dmd3_t *)buffer;
|
|
numFrames = LittleLong(pinmodel->num_frames);
|
|
numTags = LittleLong(pinmodel->num_tags);
|
|
numMeshes = LittleLong(pinmodel->num_meshes);
|
|
|
|
// calc sizes rounded to cacheline
|
|
headerSize = (sizeof(maliasmodel_t) + 31) & ~31;
|
|
frameSize = ((sizeof(maliasframe_t) * numFrames) + 31) & ~31;
|
|
tagSize = ((sizeof(maliastag_t) * numFrames * numTags) + 31) & ~31;
|
|
meshSize = ((sizeof(maliasmesh_t) * numMeshes) + 31) & ~31;
|
|
|
|
pinmesh = (dmd3mesh_t *)((byte *)pinmodel + LittleLong (pinmodel->ofs_meshes));
|
|
for (i=0; i < numMeshes; i++)
|
|
{
|
|
numSkins = LittleLong(pinmesh->num_skins);
|
|
numTris = LittleLong(pinmesh->num_tris);
|
|
numVerts = LittleLong(pinmesh->num_verts);
|
|
|
|
skinSize += ((sizeof(maliasskin_t) * numSkins) + 31) & ~31;
|
|
indexSize += ((sizeof(index_t) * numTris * 3) + 31) & ~31;
|
|
coordSize += ((sizeof(maliascoord_t) * numVerts) + 31) & ~31;
|
|
vertSize += ((numFrames * numVerts * sizeof(maliasvertex_t)) + 31) & ~31;
|
|
trNeighborsSize += ((sizeof(int) * numTris * 3) + 31) & ~31;
|
|
|
|
pinmesh = (dmd3mesh_t *)((byte *)pinmesh + LittleLong (pinmesh->meshsize));
|
|
}
|
|
allocSize = headerSize + frameSize + tagSize + meshSize + skinSize + indexSize + coordSize + vertSize + trNeighborsSize;
|
|
|
|
return allocSize;
|
|
}
|
|
|
|
|
|
/*
|
|
=================
|
|
Mod_LoadAliasMD3Model
|
|
=================
|
|
*/
|
|
void Mod_LoadAliasMD3Model (model_t *mod, void *buffer)
|
|
{
|
|
int version, maxdotindex, i, j, l, k;
|
|
dmd3_t *pinmodel;
|
|
dmd3frame_t *pinframe;
|
|
dmd3tag_t *pintag;
|
|
dmd3mesh_t *pinmesh;
|
|
dmd3skin_t *pinskin;
|
|
dmd3coord_t *pincoord;
|
|
dmd3vertex_t *pinvert;
|
|
index_t *pinindex, *poutindex;
|
|
maliasvertex_t *poutvert;
|
|
maliascoord_t *poutcoord;
|
|
maliasskin_t *poutskin;
|
|
maliasmesh_t *poutmesh;
|
|
maliastag_t *pouttag;
|
|
maliasframe_t *poutframe;
|
|
maliasmodel_t *poutmodel;
|
|
char name[MD3_MAX_PATH];
|
|
float lat, lng, maxdot;
|
|
vec3_t normal;
|
|
|
|
pinmodel = ( dmd3_t * )buffer;
|
|
version = LittleLong( pinmodel->version );
|
|
|
|
if ( version != MD3_ALIAS_VERSION )
|
|
{
|
|
VID_Error (ERR_DROP, "%s has wrong version number (%i should be %i)",
|
|
mod->name, version, MD3_ALIAS_VERSION);
|
|
}
|
|
|
|
// poutmodel = Hunk_Alloc (sizeof(maliasmodel_t));
|
|
poutmodel = ModChunk_Alloc (sizeof(maliasmodel_t));
|
|
|
|
// byte swap the header fields and sanity check
|
|
poutmodel->num_frames = LittleLong ( pinmodel->num_frames );
|
|
poutmodel->num_tags = LittleLong ( pinmodel->num_tags );
|
|
poutmodel->num_meshes = LittleLong ( pinmodel->num_meshes );
|
|
|
|
if ( poutmodel->num_frames <= 0 )
|
|
VID_Error ( ERR_DROP, "model %s has no frames", mod->name );
|
|
else if ( poutmodel->num_frames > MD3_MAX_FRAMES )
|
|
VID_Error ( ERR_DROP, "model %s has too many frames", mod->name );
|
|
|
|
if ( poutmodel->num_tags > MD3_MAX_TAGS )
|
|
VID_Error ( ERR_DROP, "model %s has too many tags", mod->name );
|
|
else if ( poutmodel->num_tags < 0 )
|
|
VID_Error ( ERR_DROP, "model %s has invalid number of tags", mod->name );
|
|
|
|
if ( poutmodel->num_meshes <= 0 )
|
|
VID_Error ( ERR_DROP, "model %s has no meshes", mod->name );
|
|
else if ( poutmodel->num_meshes > MD3_MAX_MESHES )
|
|
VID_Error ( ERR_DROP, "model %s has too many meshes", mod->name );
|
|
|
|
//
|
|
// load the frames
|
|
//
|
|
pinframe = (dmd3frame_t *)((byte *)pinmodel + LittleLong (pinmodel->ofs_frames));
|
|
// poutframe = poutmodel->frames = Hunk_Alloc ( sizeof(maliasframe_t) * poutmodel->num_frames);
|
|
poutframe = poutmodel->frames = ModChunk_Alloc ( sizeof(maliasframe_t) * poutmodel->num_frames);
|
|
|
|
mod->radius = 0;
|
|
ClearBounds ( mod->mins, mod->maxs );
|
|
|
|
for ( i = 0; i < poutmodel->num_frames; i++, pinframe++, poutframe++ )
|
|
{
|
|
for ( j = 0; j < 3; j++ )
|
|
{
|
|
poutframe->mins[j] = LittleFloat ( pinframe->mins[j] );
|
|
poutframe->maxs[j] = LittleFloat ( pinframe->maxs[j] );
|
|
poutframe->scale[j] = MD3_XYZ_SCALE;
|
|
poutframe->translate[j] = LittleFloat ( pinframe->translate[j] );
|
|
}
|
|
|
|
poutframe->radius = LittleFloat ( pinframe->radius );
|
|
|
|
mod->radius = max ( mod->radius, poutframe->radius );
|
|
AddPointToBounds ( poutframe->mins, mod->mins, mod->maxs );
|
|
AddPointToBounds ( poutframe->maxs, mod->mins, mod->maxs );
|
|
}
|
|
|
|
//
|
|
// load the tags
|
|
//
|
|
pintag = (dmd3tag_t *)((byte *)pinmodel + LittleLong (pinmodel->ofs_tags));
|
|
// pouttag = poutmodel->tags = Hunk_Alloc( sizeof(maliastag_t) * poutmodel->num_frames * poutmodel->num_tags);
|
|
pouttag = poutmodel->tags = ModChunk_Alloc(sizeof(maliastag_t) * poutmodel->num_frames * poutmodel->num_tags);
|
|
|
|
for ( i = 0; i < poutmodel->num_frames; i++ )
|
|
{
|
|
for ( l = 0; l < poutmodel->num_tags; l++, pintag++, pouttag++ )
|
|
{
|
|
memcpy ( pouttag->name, pintag->name, MD3_MAX_PATH );
|
|
for ( j = 0; j < 3; j++ ) {
|
|
pouttag->orient.origin[j] = LittleFloat ( pintag->orient.origin[j] );
|
|
pouttag->orient.axis[0][j] = LittleFloat ( pintag->orient.axis[0][j] );
|
|
pouttag->orient.axis[1][j] = LittleFloat ( pintag->orient.axis[1][j] );
|
|
pouttag->orient.axis[2][j] = LittleFloat ( pintag->orient.axis[2][j] );
|
|
}
|
|
/* VID_Printf (PRINT_ALL, "Tag origin: (%f %f %f) axis X: (%f %f %f) Y: (%f %f %f) Z: (%f %f %f)\n",
|
|
pouttag->orient.origin[0], pouttag->orient.origin[1], pouttag->orient.origin[2],
|
|
pouttag->orient.axis[0][0], pouttag->orient.axis[0][1], pouttag->orient.axis[0][2],
|
|
pouttag->orient.axis[1][0], pouttag->orient.axis[1][1], pouttag->orient.axis[1][2],
|
|
pouttag->orient.axis[2][0], pouttag->orient.axis[2][1], pouttag->orient.axis[2][2]);
|
|
*/
|
|
}
|
|
}
|
|
|
|
//
|
|
// load the meshes
|
|
//
|
|
pinmesh = (dmd3mesh_t *)((byte *)pinmodel + LittleLong (pinmodel->ofs_meshes));
|
|
// poutmesh = poutmodel->meshes = Hunk_Alloc ( sizeof(maliasmesh_t)*poutmodel->num_meshes);
|
|
poutmesh = poutmodel->meshes = ModChunk_Alloc (sizeof(maliasmesh_t)*poutmodel->num_meshes);
|
|
|
|
for ( i = 0; i < poutmodel->num_meshes; i++, poutmesh++)
|
|
{
|
|
memcpy (poutmesh->name, pinmesh->name, MD3_MAX_PATH);
|
|
|
|
if ( strncmp ((const char *)pinmesh->id, "IDP3", 4))
|
|
{
|
|
VID_Error ( ERR_DROP, "mesh %s in model %s has wrong id (%i should be %i)",
|
|
poutmesh->name, mod->name, LittleLong((int)pinmesh->id), IDMD3HEADER );
|
|
}
|
|
|
|
poutmesh->num_tris = LittleLong ( pinmesh->num_tris );
|
|
poutmesh->num_skins = LittleLong ( pinmesh->num_skins );
|
|
poutmesh->num_verts = LittleLong ( pinmesh->num_verts );
|
|
|
|
if ( poutmesh->num_skins <= 0 )
|
|
VID_Error ( ERR_DROP, "mesh %i in model %s has no skins", i, mod->name );
|
|
else if ( poutmesh->num_skins > MD3_MAX_SHADERS )
|
|
VID_Error ( ERR_DROP, "mesh %i in model %s has too many skins", i, mod->name );
|
|
|
|
if ( poutmesh->num_tris <= 0 )
|
|
VID_Error ( ERR_DROP, "mesh %i in model %s has no triangles", i, mod->name );
|
|
else if ( poutmesh->num_tris > MD3_MAX_TRIANGLES )
|
|
VID_Error ( ERR_DROP, "mesh %i in model %s has too many triangles", i, mod->name );
|
|
|
|
if ( poutmesh->num_verts <= 0 )
|
|
VID_Error ( ERR_DROP, "mesh %i in model %s has no vertices", i, mod->name );
|
|
else if ( poutmesh->num_verts > MD3_MAX_VERTS )
|
|
VID_Error ( ERR_DROP, "mesh %i in model %s has too many vertices", i, mod->name );
|
|
|
|
//
|
|
// register all skins
|
|
//
|
|
pinskin = (dmd3skin_t *)((byte *)pinmesh + LittleLong (pinmesh->ofs_skins));
|
|
// poutskin = poutmesh->skins = Hunk_Alloc( sizeof(maliasskin_t) * poutmesh->num_skins );
|
|
poutskin = poutmesh->skins = ModChunk_Alloc(sizeof(maliasskin_t) * poutmesh->num_skins);
|
|
|
|
for ( j = 0; j < poutmesh->num_skins; j++, pinskin++, poutskin++ )
|
|
{
|
|
memcpy (name, pinskin->name, MD3_MAX_PATH);
|
|
if (name[1] == 'o')
|
|
name[0] = 'm';
|
|
if (name[1] == 'l')
|
|
name[0] = 'p';
|
|
memcpy (poutskin->name, name, MD3_MAX_PATH);
|
|
Com_sprintf(poutskin->glowname, sizeof(poutskin->glowname), "\0"); // set null glowskin
|
|
mod->skins[i][j] = R_FindImage (name, it_skin);
|
|
}
|
|
|
|
//
|
|
// load the indexes
|
|
//
|
|
pinindex = (index_t *)((byte *)pinmesh + LittleLong (pinmesh->ofs_tris));
|
|
// poutindex = poutmesh->indexes = Hunk_Alloc( sizeof(index_t) * poutmesh->num_tris * 3 );
|
|
poutindex = poutmesh->indexes = ModChunk_Alloc(sizeof(index_t) * poutmesh->num_tris * 3);
|
|
|
|
for ( j = 0; j < poutmesh->num_tris; j++, pinindex += 3, poutindex += 3 )
|
|
{
|
|
poutindex[0] = (index_t)LittleLong ( pinindex[0] );
|
|
poutindex[1] = (index_t)LittleLong ( pinindex[1] );
|
|
poutindex[2] = (index_t)LittleLong ( pinindex[2] );
|
|
}
|
|
|
|
//
|
|
// load the texture coordinates
|
|
//
|
|
pincoord = (dmd3coord_t *)((byte *)pinmesh + LittleLong (pinmesh->ofs_tcs));
|
|
// poutcoord = poutmesh->stcoords = Hunk_Alloc( sizeof(maliascoord_t) * poutmesh->num_verts);
|
|
poutcoord = poutmesh->stcoords = ModChunk_Alloc(sizeof(maliascoord_t) * poutmesh->num_verts);
|
|
|
|
for ( j = 0; j < poutmesh->num_verts; j++, pincoord++, poutcoord++ )
|
|
{
|
|
poutcoord->st[0] = LittleFloat ( pincoord->st[0] );
|
|
poutcoord->st[1] = LittleFloat ( pincoord->st[1] );
|
|
}
|
|
|
|
//
|
|
// load the vertexes and normals
|
|
//
|
|
pinvert = (dmd3vertex_t *)((byte *)pinmesh + LittleLong (pinmesh->ofs_verts));
|
|
// poutvert = poutmesh->vertexes = Hunk_Alloc(poutmodel->num_frames * poutmesh->num_verts * sizeof(maliasvertex_t));
|
|
poutvert = poutmesh->vertexes = ModChunk_Alloc(poutmodel->num_frames * poutmesh->num_verts * sizeof(maliasvertex_t));
|
|
|
|
for ( l = 0; l < poutmodel->num_frames; l++ )
|
|
{
|
|
for ( j = 0; j < poutmesh->num_verts; j++, pinvert++, poutvert++ )
|
|
{
|
|
poutvert->xyz[0] = LittleShort(pinvert->point[0]);
|
|
poutvert->xyz[1] = LittleShort(pinvert->point[1]);
|
|
poutvert->xyz[2] = LittleShort(pinvert->point[2]);
|
|
|
|
poutvert->normal[0] = (LittleShort(pinvert->norm) >> 0) & 0xff;
|
|
poutvert->normal[1] = (LittleShort(pinvert->norm) >> 8) & 0xff;
|
|
|
|
lat = (pinvert->norm >> 8) & 0xff;
|
|
lng = (pinvert->norm & 0xff);
|
|
|
|
lat *= M_PI/128;
|
|
lng *= M_PI/128;
|
|
|
|
normal[0] = cos(lat) * sin(lng);
|
|
normal[1] = sin(lat) * sin(lng);
|
|
normal[2] = cos(lng);
|
|
|
|
// use ye olde quantized normals for shading
|
|
maxdot = -999999.0;
|
|
maxdotindex = -1;
|
|
|
|
for (k=0; k<NUMVERTEXNORMALS; k++)
|
|
{
|
|
float dot = DotProduct (normal, r_avertexnormals[k]);
|
|
if (dot > maxdot) {
|
|
maxdot = dot;
|
|
maxdotindex = k;
|
|
}
|
|
}
|
|
poutvert->lightnormalindex = maxdotindex;
|
|
}
|
|
}
|
|
pinmesh = (dmd3mesh_t *)((byte *)pinmesh + LittleLong (pinmesh->meshsize));
|
|
|
|
//
|
|
// build triangle neighbors
|
|
//
|
|
// poutmesh->trneighbors = Hunk_Alloc (sizeof(int) * poutmesh->num_tris * 3);
|
|
poutmesh->trneighbors = ModChunk_Alloc (sizeof(int) * poutmesh->num_tris * 3);
|
|
Mod_BuildTriangleNeighbors (poutmesh);
|
|
}
|
|
|
|
mod->hasAlpha = false;
|
|
Mod_LoadModelScript (mod, poutmodel); // md3 skin scripting
|
|
|
|
//if (mod->hasAlpha)
|
|
// VID_Printf (PRINT_ALL, "Mod_LoadAliasMD3Model: model %s has trans mesh\n", mod->name);
|
|
|
|
mod->type = mod_alias;
|
|
}
|
|
//Harven-- MD3
|
|
|
|
|
|
/*
|
|
==============================================================================
|
|
|
|
SPRITE MODELS
|
|
|
|
==============================================================================
|
|
*/
|
|
|
|
/*
|
|
=================
|
|
Mod_GetAllocSizeSprite
|
|
Calc exact alloc size for sprite in memory
|
|
=================
|
|
*/
|
|
size_t Mod_GetAllocSizeSprite (model_t *mod, void *buffer)
|
|
{
|
|
size_t allocSize;
|
|
|
|
// calc memory allocation size
|
|
allocSize = ((modfilelen) + 31) & ~31;
|
|
|
|
return allocSize;
|
|
}
|
|
|
|
|
|
/*
|
|
=================
|
|
Mod_LoadSpriteModel
|
|
=================
|
|
*/
|
|
void Mod_LoadSpriteModel (model_t *mod, void *buffer)
|
|
{
|
|
dspr2_t *sprin, *sprout;
|
|
int i;
|
|
|
|
sprin = (dspr2_t *)buffer;
|
|
// sprout = Hunk_Alloc (modfilelen);
|
|
sprout = ModChunk_Alloc (modfilelen);
|
|
|
|
sprout->ident = LittleLong (sprin->ident);
|
|
sprout->version = LittleLong (sprin->version);
|
|
sprout->numframes = LittleLong (sprin->numframes);
|
|
|
|
if (sprout->version != SP2_VERSION)
|
|
VID_Error (ERR_DROP, "%s has wrong version number (%i should be %i)",
|
|
mod->name, sprout->version, SP2_VERSION);
|
|
|
|
if (sprout->numframes > MD2_MAX_SKINS)
|
|
VID_Error (ERR_DROP, "%s has too many frames (%i > %i)",
|
|
mod->name, sprout->numframes, MD2_MAX_SKINS);
|
|
|
|
// byte swap everything
|
|
for (i=0; i<sprout->numframes; i++)
|
|
{
|
|
sprout->frames[i].width = LittleLong (sprin->frames[i].width);
|
|
sprout->frames[i].height = LittleLong (sprin->frames[i].height);
|
|
sprout->frames[i].origin_x = LittleLong (sprin->frames[i].origin_x);
|
|
sprout->frames[i].origin_y = LittleLong (sprin->frames[i].origin_y);
|
|
memcpy (sprout->frames[i].name, sprin->frames[i].name, MD2_MAX_SKINNAME);
|
|
mod->skins[0][i] = R_FindImage (sprout->frames[i].name,
|
|
it_sprite);
|
|
}
|
|
|
|
mod->type = mod_sprite;
|
|
}
|
|
|
|
//=============================================================================
|
|
|
|
/*
|
|
@@@@@@@@@@@@@@@@@@@@@
|
|
R_BeginRegistration
|
|
|
|
Specifies the model that will be used as the world
|
|
@@@@@@@@@@@@@@@@@@@@@
|
|
*/
|
|
void R_BeginRegistration (char *model)
|
|
{
|
|
char fullname[MAX_QPATH];
|
|
cvar_t *flushmap;
|
|
|
|
registration_sequence++;
|
|
r_oldviewcluster = -1; // force markleafs
|
|
|
|
Mod_InitFailedTexList (); // clear failed texture list
|
|
Mod_InitWalSizeList (); // clear wal size list
|
|
|
|
Com_sprintf (fullname, sizeof(fullname), "maps/%s.bsp", model);
|
|
|
|
// explicitly free the old map if different
|
|
// this guarantees that mod_known[0] is the world map
|
|
flushmap = Cvar_Get ("flushmap", "0", 0);
|
|
Cvar_SetDescription ("flushmap", "Forces reload of same map when set to 1.");
|
|
if (strcmp(mod_known[0].name, fullname) || flushmap->integer) {
|
|
Mod_Free (&mod_known[0]);
|
|
// clear this on map change (case of different server and autodownloading)
|
|
R_InitFailedImgList ();
|
|
}
|
|
r_worldmodel = Mod_ForName(fullname, true);
|
|
|
|
r_viewcluster = -1;
|
|
|
|
registration_active = true; // map registration flag
|
|
}
|
|
|
|
|
|
/*
|
|
@@@@@@@@@@@@@@@@@@@@@
|
|
R_RegisterModel
|
|
|
|
@@@@@@@@@@@@@@@@@@@@@
|
|
*/
|
|
struct model_s *R_RegisterModel (char *name)
|
|
{
|
|
model_t *mod;
|
|
int i, nameLen;
|
|
dspr2_t *sprout;
|
|
#ifndef MD2_AS_MD3 // Knightmare- no longer used!
|
|
dmd2_t *pheader;
|
|
#endif // MD2_AS_MD3
|
|
// Harven++ MD3
|
|
maliasmodel_t *pheader3;
|
|
// Harven-- MD3
|
|
|
|
// Knightmare- MD3 autoreplace code
|
|
nameLen = (int)strlen(name);
|
|
if ( (nameLen >= 5) && !strcmp(name+nameLen-4, ".md2") ) // look if we have a .md2 file
|
|
{
|
|
char s[128];
|
|
// strncpy(s,name);
|
|
Q_strncpyz (s, sizeof(s), name);
|
|
s[nameLen-1]='3';
|
|
mod = R_RegisterModel (s);
|
|
if (mod)
|
|
return mod;
|
|
}
|
|
// end Knightmare
|
|
|
|
mod = Mod_ForName (name, false);
|
|
if (mod)
|
|
{
|
|
mod->registration_sequence = registration_sequence;
|
|
|
|
// register any images used by the models
|
|
if (mod->type == mod_sprite)
|
|
{
|
|
sprout = (dspr2_t *)mod->extradata;
|
|
for (i=0 ; i<sprout->numframes ; i++)
|
|
mod->skins[0][i] = R_FindImage (sprout->frames[i].name, it_sprite);
|
|
}
|
|
#ifndef MD2_AS_MD3 // Knightmare- no longer used!
|
|
else if (mod->type == mod_md2)
|
|
{
|
|
pheader = (dmd2_t *)mod->extradata;
|
|
for (i=0 ; i<pheader->num_skins ; i++)
|
|
mod->skins[0][i] = R_FindImage ((char *)pheader + pheader->ofs_skins + i*MAX_SKINNAME, it_skin);
|
|
mod->numframes = pheader->num_frames;
|
|
}
|
|
#endif // MD2_AS_MD3
|
|
// Harven++ MD3
|
|
else if (mod->type == mod_alias)
|
|
{
|
|
int k;
|
|
|
|
pheader3 = (maliasmodel_t *)mod->extradata;
|
|
|
|
for (i=0; i<pheader3->num_meshes; i++)
|
|
{
|
|
for (k=0; k<pheader3->meshes[i].num_skins; k++) {
|
|
mod->skins[i][k] = R_FindImage(pheader3->meshes[i].skins[k].name, it_skin);
|
|
if (strlen(pheader3->meshes[i].skins[k].glowname))
|
|
pheader3->meshes[i].skins[k].glowimage = R_FindImage(pheader3->meshes[i].skins[k].glowname, it_skin);
|
|
}
|
|
}
|
|
mod->numframes = pheader3->num_frames;
|
|
}
|
|
// Harven-- MD3
|
|
else if (mod->type == mod_brush)
|
|
{
|
|
for (i=0 ; i<mod->numtexinfo ; i++) {
|
|
mod->texinfo[i].image->registration_sequence = registration_sequence;
|
|
mod->texinfo[i].glow->registration_sequence = registration_sequence;
|
|
}
|
|
}
|
|
}
|
|
return mod;
|
|
}
|
|
|
|
|
|
/*
|
|
@@@@@@@@@@@@@@@@@@@@@
|
|
R_EndRegistration
|
|
|
|
@@@@@@@@@@@@@@@@@@@@@
|
|
*/
|
|
void R_EndRegistration (void)
|
|
{
|
|
int i;
|
|
model_t *mod;
|
|
|
|
for (i=0, mod=mod_known ; i<mod_numknown ; i++, mod++)
|
|
{
|
|
if (!mod->name[0])
|
|
continue;
|
|
if (mod->registration_sequence != registration_sequence)
|
|
{ // don't need this model
|
|
Mod_Free (mod);
|
|
}
|
|
}
|
|
|
|
R_FreeUnusedImages ();
|
|
|
|
registration_active = false; // map registration flag
|
|
}
|
|
|
|
|
|
//=============================================================================
|
|
|
|
|
|
/*
|
|
================
|
|
Mod_Free
|
|
================
|
|
*/
|
|
void Mod_Free (model_t *mod)
|
|
{
|
|
// Knightmare- only brush models use the hunk system now
|
|
// Hunk_Free (mod->extradata);
|
|
switch (mod->type)
|
|
{
|
|
case mod_brush:
|
|
Hunk_Free (mod->extradata);
|
|
break;
|
|
case mod_alias:
|
|
case mod_sprite:
|
|
default:
|
|
ModChunk_Free (mod->extradata);
|
|
break;
|
|
}
|
|
|
|
#ifdef PROJECTION_SHADOWS // projection shadows from BeefQuake R6
|
|
if (mod->edge_tri)
|
|
free (mod->edge_tri);
|
|
#endif // end projection shadows from BeefQuake R6
|
|
|
|
memset (mod, 0, sizeof(*mod));
|
|
}
|
|
|
|
/*
|
|
================
|
|
Mod_FreeAll
|
|
================
|
|
*/
|
|
void Mod_FreeAll (void)
|
|
{
|
|
int i;
|
|
|
|
for (i=0 ; i<mod_numknown ; i++)
|
|
{
|
|
if (mod_known[i].extradatasize)
|
|
Mod_Free (&mod_known[i]);
|
|
}
|
|
}
|