|
|
//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
// $NoKeywords: $
//
//=============================================================================//
//
// studiomdl.c: generates a studio .mdl file from a .qc script
// models/<scriptname>.mdl.
//
#pragma warning( disable : 4244 )
#pragma warning( disable : 4237 )
#pragma warning( disable : 4305 )
#include <stdio.h>
#include <stdlib.h>
#include <sys/stat.h>
#include <math.h>
#include "cmdlib.h"
#include "scriplib.h"
#include "mathlib/mathlib.h"
#include "studio.h"
#include "studiomdl.h"
//#include "..\..\dlls\activity.h"
bool IsEnd( char const* pLine ) { if (strncmp( "end", pLine, 3 ) != 0) return false; return (pLine[3] == '\0') || (pLine[3] == '\n'); }
int SortAndBalanceBones( int iCount, int iMaxCount, int bones[], float weights[] ) { int i;
// collapse duplicate bone weights
for (i = 0; i < iCount-1; i++) { int j; for (j = i + 1; j < iCount; j++) { if (bones[i] == bones[j]) { weights[i] += weights[j]; weights[j] = 0.0; } } }
// do sleazy bubble sort
int bShouldSort; do { bShouldSort = false; for (i = 0; i < iCount-1; i++) { if (weights[i+1] > weights[i]) { int j = bones[i+1]; bones[i+1] = bones[i]; bones[i] = j; float w = weights[i+1]; weights[i+1] = weights[i]; weights[i] = w; bShouldSort = true; } } } while (bShouldSort);
// throw away all weights less than 1/20th
while (iCount > 1 && weights[iCount-1] < 0.05) { iCount--; }
// clip to the top iMaxCount bones
if (iCount > iMaxCount) { iCount = iMaxCount; }
float t = 0; for (i = 0; i < iCount; i++) { t += weights[i]; }
if (t <= 0.0) { // missing weights?, go ahead and evenly share?
// FIXME: shouldn't this error out?
t = 1.0 / iCount;
for (i = 0; i < iCount; i++) { weights[i] = t; } } else { // scale to sum to 1.0
t = 1.0 / t;
for (i = 0; i < iCount; i++) { weights[i] = weights[i] * t; } }
return iCount; }
void Grab_Vertexlist( s_source_t *psource ) { while (1) { if (fgets( g_szLine, sizeof( g_szLine ), g_fpInput ) != NULL) { int j; int bone; Vector p; int iCount, bones[4]; float weights[4];
g_iLinecount++;
// check for end
if (IsEnd(g_szLine)) return;
int i = sscanf( g_szLine, "%d %d %f %f %f %d %d %f %d %f %d %f %d %f", &j, &bone, &p[0], &p[1], &p[2], &iCount, &bones[0], &weights[0], &bones[1], &weights[1], &bones[2], &weights[2], &bones[3], &weights[3] ); if (i == 5) { if (bone < 0 || bone >= psource->numbones) { MdlWarning( "bogus bone index\n" ); MdlWarning( "%d %s :\n%s", g_iLinecount, g_szFilename, g_szLine ); MdlError( "Exiting due to errors\n" ); }
VectorCopy( p, g_vertex[j] ); g_bone[j].numbones = 1; g_bone[j].bone[0] = bone; g_bone[j].weight[0] = 1.0; } else if (i > 5) { iCount = SortAndBalanceBones( iCount, MAXSTUDIOBONEWEIGHTS, bones, weights );
VectorCopy( p, g_vertex[j] ); g_bone[j].numbones = iCount; for (i = 0; i < iCount; i++) { g_bone[j].bone[i] = bones[i]; g_bone[j].weight[i] = weights[i]; } } else { MdlError("%s: error on line %d: %s", g_szFilename, g_iLinecount, g_szLine ); } } } }
void Grab_Facelist( s_source_t *psource ) { while (1) { if (fgets( g_szLine, sizeof( g_szLine ), g_fpInput ) != NULL) { int j; s_tmpface_t f;
g_iLinecount++;
// check for end
if (IsEnd(g_szLine)) return;
if (sscanf( g_szLine, "%d %d %d %d", &j, &f.a, &f.b, &f.c) == 4) { g_face[j] = f; } else { MdlError("%s: error on line %d: %s", g_szFilename, g_iLinecount, g_szLine ); } } } }
void Grab_Materiallist( s_source_t *psource ) { while (1) { if (fgets( g_szLine, sizeof( g_szLine ), g_fpInput ) != NULL) { // char name[256];
char path[MAX_PATH]; rgb2_t a, d, s; float g; int j;
g_iLinecount++;
// check for end
if (IsEnd(g_szLine)) return;
if (sscanf( g_szLine, "%d %f %f %f %f %f %f %f %f %f %f %f %f %f \"%[^\"]s", &j, &a.r, &a.g, &a.b, &a.a, &d.r, &d.g, &d.b, &d.a, &s.r, &s.g, &s.b, &s.a, &g, path ) == 15) { if (path[0] == '\0') { psource->texmap[j] = -1; } else if (j < ARRAYSIZE(psource->texmap)) { psource->texmap[j] = LookupTexture( path ); } else { MdlError( "Too many materials, max %d\n", ARRAYSIZE(psource->texmap) ); } } } } }
void Grab_Texcoordlist( s_source_t *psource ) { while (1) { if (fgets( g_szLine, sizeof( g_szLine ), g_fpInput ) != NULL) { int j; Vector2D t;
g_iLinecount++;
// check for end
if (IsEnd(g_szLine)) return;
if (sscanf( g_szLine, "%d %f %f", &j, &t[0], &t[1]) == 3) { t[1] = 1.0 - t[1]; g_texcoord[j][0] = t[0]; g_texcoord[j][1] = t[1]; } else { MdlError("%s: error on line %d: %s", g_szFilename, g_iLinecount, g_szLine ); } } } }
void Grab_Normallist( s_source_t *psource ) { while (1) { if (fgets( g_szLine, sizeof( g_szLine ), g_fpInput ) != NULL) { int j; int bone; Vector n;
g_iLinecount++;
// check for end
if (IsEnd(g_szLine)) return;
if (sscanf( g_szLine, "%d %d %f %f %f", &j, &bone, &n[0], &n[1], &n[2]) == 5) { if (bone < 0 || bone >= psource->numbones) { MdlWarning( "bogus bone index\n" ); MdlWarning( "%d %s :\n%s", g_iLinecount, g_szFilename, g_szLine ); MdlError( "Exiting due to errors\n" ); }
VectorCopy( n, g_normal[j] ); } else { MdlError("%s: error on line %d: %s", g_szFilename, g_iLinecount, g_szLine ); } } } }
void Grab_Faceattriblist( s_source_t *psource ) { while (1) { if (fgets( g_szLine, sizeof( g_szLine ), g_fpInput ) != NULL) { int j; int smooth; int material; s_tmpface_t f; unsigned short s;
g_iLinecount++;
// check for end
if (IsEnd(g_szLine)) return;
if (sscanf( g_szLine, "%d %d %d %d %d %d %d %d %d", &j, &material, &smooth, &f.ta, &f.tb, &f.tc, &f.na, &f.nb, &f.nc) == 9) { f.a = g_face[j].a; f.b = g_face[j].b; f.c = g_face[j].c;
f.material = UseTextureAsMaterial( psource->texmap[material] ); if (f.material < 0) { MdlError( "face %d references NULL texture %d\n", j, material ); } if (1) { s = f.b; f.b = f.c; f.c = s; s = f.tb; f.tb = f.tc; f.tc = s; s = f.nb; f.nb = f.nc; f.nc = s; }
g_face[j] = f; } else { MdlError("%s: error on line %d: %s", g_szFilename, g_iLinecount, g_szLine ); } } } }
int closestNormal( int v, int n ) { float maxdot = -1.0; float dot; int r = n;
v_unify_t *cur = v_list[v];
while (cur) { dot = DotProduct( g_normal[cur->n], g_normal[n] ); if (dot > maxdot) { r = cur->n; maxdot = dot; } cur = cur->next; } return r; }
int AddToVlist( int v, int m, int n, int t, int firstref ) { v_unify_t *prev = NULL; v_unify_t *cur = v_list[v];
while (cur) { if (cur->m == m && cur->n == n && cur->t == t) { cur->refcount++; return cur - v_listdata; } prev = cur; cur = cur->next; }
if (numvlist >= MAXSTUDIOVERTS) { MdlError( "Too many unified vertices\n"); }
cur = &v_listdata[numvlist++]; cur->lastref = -1; cur->refcount = 1; cur->firstref = firstref; cur->v = v; cur->m = m; cur->n = n; cur->t = t;
if (prev) { prev->next = cur; } else { v_list[v] = cur; }
return numvlist - 1; }
void DecrementReferenceVlist( int uv, int numverts ) { if (uv < 0 || uv >= MAXSTUDIOVERTS) MdlError( "decrement outside of range\n");
v_listdata[uv].refcount--;
if (v_listdata[uv].refcount == 0) { v_listdata[uv].lastref = numverts; } else if (v_listdata[uv].refcount < 0) { MdlError("<0 ref\n"); } }
void UnifyIndices( s_source_t *psource ) { int i;
static s_tmpface_t tmpface[MAXSTUDIOTRIANGLES]; // mrm processed g_face
static s_face_t uface[MAXSTUDIOTRIANGLES]; // mrm processed unified face
// clear v_list
numvlist = 0; memset( v_list, 0, sizeof( v_list ) ); memset( v_listdata, 0, sizeof( v_listdata ) );
// create an list of all the
for (i = 0; i < g_numfaces; i++) { tmpface[i] = g_face[i];
uface[i].a = AddToVlist( g_face[i].a, g_face[i].material, g_face[i].na, g_face[i].ta, g_numverts ); uface[i].b = AddToVlist( g_face[i].b, g_face[i].material, g_face[i].nb, g_face[i].tb, g_numverts ); uface[i].c = AddToVlist( g_face[i].c, g_face[i].material, g_face[i].nc, g_face[i].tc, g_numverts );
// keep an original copy
g_src_uface[i] = uface[i]; }
// printf("%d : %d %d %d\n", numvlist, g_numverts, g_numnormals, g_numtexcoords );
}
void CalcModelTangentSpaces( s_source_t *pSrc );
//-----------------------------------------------------------------------------
// Builds a list of unique vertices in a source
//-----------------------------------------------------------------------------
static void BuildUniqueVertexList( s_source_t *pSource, const int *pDesiredToVList ) { // allocate memory
pSource->vertex = (s_vertexinfo_t *)kalloc( pSource->numvertices, sizeof( s_vertexinfo_t ) );
// create arrays of unique vertexes, normals, texcoords.
for (int i = 0; i < pSource->numvertices; i++) { int j = pDesiredToVList[i];
s_vertexinfo_t &vertex = pSource->vertex[i]; VectorCopy( g_vertex[ v_listdata[j].v ], vertex.position ); VectorCopy( g_normal[ v_listdata[j].n ], vertex.normal ); Vector2Copy( g_texcoord[ v_listdata[j].t ], vertex.texcoord );
vertex.boneweight.numbones = g_bone[ v_listdata[j].v ].numbones; int k; for( k = 0; k < MAXSTUDIOBONEWEIGHTS; k++ ) { vertex.boneweight.bone[k] = g_bone[ v_listdata[j].v ].bone[k]; vertex.boneweight.weight[k] = g_bone[ v_listdata[j].v ].weight[k]; }
// store a bunch of other info
vertex.material = v_listdata[j].m;
#if 0
pSource->vertexInfo[i].firstref = v_listdata[j].firstref; pSource->vertexInfo[i].lastref = v_listdata[j].lastref; #endif
// printf("%4d : %2d : %6.2f %6.2f %6.2f\n", i, psource->boneweight[i].bone[0], psource->vertex[i][0], psource->vertex[i][1], psource->vertex[i][2] );
}
}
//-----------------------------------------------------------------------------
// sort new vertices by materials, last used
//-----------------------------------------------------------------------------
static int vlistCompare( const void *elem1, const void *elem2 ) { v_unify_t *u1 = &v_listdata[*(int *)elem1]; v_unify_t *u2 = &v_listdata[*(int *)elem2];
// sort by material
if (u1->m < u2->m) return -1; if (u1->m > u2->m) return 1;
// sort by last used
if (u1->lastref < u2->lastref) return -1; if (u1->lastref > u2->lastref) return 1;
return 0; }
static void SortVerticesByMaterial( int *pDesiredToVList, int *pVListToDesired ) { for ( int i = 0; i < numvlist; i++ ) { pDesiredToVList[i] = i; } qsort( pDesiredToVList, numvlist, sizeof( int ), vlistCompare ); for ( int i = 0; i < numvlist; i++ ) { pVListToDesired[ pDesiredToVList[i] ] = i; } }
//-----------------------------------------------------------------------------
// sort new faces by materials, last used
//-----------------------------------------------------------------------------
static int faceCompare( const void *elem1, const void *elem2 ) { int i1 = *(int *)elem1; int i2 = *(int *)elem2;
// sort by material
if (g_face[i1].material < g_face[i2].material) return -1; if (g_face[i1].material > g_face[i2].material) return 1;
// sort by original usage
if (i1 < i2) return -1; if (i1 > i2) return 1;
return 0; }
static void SortFacesByMaterial( int *pDesiredToSrcFace ) { // NOTE: Unlike SortVerticesByMaterial, srcFaceToDesired isn't needed, so we're not computing it
for ( int i = 0; i < g_numfaces; i++ ) { pDesiredToSrcFace[i] = i; } qsort( pDesiredToSrcFace, g_numfaces, sizeof( int ), faceCompare ); }
//-----------------------------------------------------------------------------
// Builds mesh structures in the source
//-----------------------------------------------------------------------------
static void PointMeshesToVertexAndFaceData( s_source_t *pSource, int *pDesiredToSrcFace ) { // First, assign all meshes to be empty
// A mesh is a set of faces + vertices that all use 1 material
for ( int m = 0; m < MAXSTUDIOSKINS; m++ ) { pSource->mesh[m].numvertices = 0; pSource->mesh[m].vertexoffset = pSource->numvertices;
pSource->mesh[m].numfaces = 0; pSource->mesh[m].faceoffset = pSource->numfaces; }
// find first and count of vertices per material
for ( int i = 0; i < pSource->numvertices; i++ ) { int m = pSource->vertex[i].material; pSource->mesh[m].numvertices++; if (pSource->mesh[m].vertexoffset > i) { pSource->mesh[m].vertexoffset = i; } }
// find first and count of faces per material
for ( int i = 0; i < pSource->numfaces; i++ ) { int m = g_face[ pDesiredToSrcFace[i] ].material;
pSource->mesh[m].numfaces++; if (pSource->mesh[m].faceoffset > i) { pSource->mesh[m].faceoffset = i; } }
/*
for (k = 0; k < MAXSTUDIOSKINS; k++) { printf("%d : %d:%d %d:%d\n", k, psource->mesh[k].numvertices, psource->mesh[k].vertexoffset, psource->mesh[k].numfaces, psource->mesh[k].faceoffset ); } */ }
//-----------------------------------------------------------------------------
// Builds the face list in the mesh
//-----------------------------------------------------------------------------
static void BuildFaceList( s_source_t *pSource, int *pVListToDesired, int *pDesiredToSrcFace ) { pSource->face = (s_face_t *)kalloc( pSource->numfaces, sizeof( s_face_t )); for ( int m = 0; m < MAXSTUDIOSKINS; m++) { if ( !pSource->mesh[m].numfaces ) continue;
pSource->meshindex[ pSource->nummeshes++ ] = m;
for ( int i = pSource->mesh[m].faceoffset; i < pSource->mesh[m].numfaces + pSource->mesh[m].faceoffset; i++) { int j = pDesiredToSrcFace[i];
// NOTE: per-face vertex indices a,b,c are mesh relative (hence the subtraction),
// while g_src_uface are model relative
pSource->face[i].a = pVListToDesired[ g_src_uface[j].a ] - pSource->mesh[m].vertexoffset; pSource->face[i].b = pVListToDesired[ g_src_uface[j].b ] - pSource->mesh[m].vertexoffset; pSource->face[i].c = pVListToDesired[ g_src_uface[j].c ] - pSource->mesh[m].vertexoffset; Assert( ((pSource->face[i].a & 0xF0000000) == 0) && ((pSource->face[i].b & 0xF0000000) == 0) && ((pSource->face[i].c & 0xF0000000) == 0) ); // printf("%3d : %4d %4d %4d\n", i, pSource->face[i].a, pSource->face[i].b, pSource->face[i].c );
} } }
//-----------------------------------------------------------------------------
// Remaps the vertex animations based on the new vertex ordering
//-----------------------------------------------------------------------------
static void RemapVertexAnimations( s_source_t *pSource, int *pVListToDesired ) { int nAnimationCount = pSource->m_Animations.Count(); for ( int i = 0; i < nAnimationCount; ++i ) { s_sourceanim_t &anim = pSource->m_Animations[i]; if ( !anim.newStyleVertexAnimations ) continue;
for ( int j = 0; j < MAXSTUDIOANIMFRAMES; ++j ) { int nVAnimCount = anim.numvanims[j]; if ( nVAnimCount == 0 ) continue;
// Copy off the initial vertex data
// Have to do it in 2 loops because it'll overwrite itself if we do it in 1
int *pTemp = (int*)_alloca( nVAnimCount * sizeof(int) ); for ( int k = 0; k < nVAnimCount; ++k ) { pTemp[k] = anim.vanim[j][k].vertex; }
for ( int k = 0; k < nVAnimCount; ++k ) { // NOTE: vertex animations are model relative, not mesh relative
anim.vanim[j][k].vertex = pVListToDesired[ pTemp[k] ]; } } } }
//-----------------------------------------------------------------------------
// Sorts vertices by material type, re-maps data structures that refer to those vertices
// to use the new indices
//-----------------------------------------------------------------------------
void BuildIndividualMeshes( s_source_t *pSource ) { static int v_listsort[MAXSTUDIOVERTS]; // map desired order to vlist entry
static int v_ilistsort[MAXSTUDIOVERTS]; // map vlist entry to desired order
static int facesort[MAXSTUDIOTRIANGLES]; // map desired order to src_face entry
SortVerticesByMaterial( v_listsort, v_ilistsort ); SortFacesByMaterial( facesort );
pSource->numvertices = numvlist; pSource->numfaces = g_numfaces;
BuildUniqueVertexList( pSource, v_listsort ); PointMeshesToVertexAndFaceData( pSource, facesort ); BuildFaceList( pSource, v_ilistsort, facesort ); RemapVertexAnimations( pSource, v_ilistsort ); CalcModelTangentSpaces( pSource ); }
void Grab_MRMFaceupdates( s_source_t *psource ) { while (1) { if (fgets( g_szLine, sizeof( g_szLine ), g_fpInput ) != NULL) { g_iLinecount++;
// check for end
if (IsEnd(g_szLine)) return; } } }
int Load_VRM ( s_source_t *psource ) { char cmd[1024]; int option;
if (!OpenGlobalFile( psource->filename )) { return 0; }
if( !g_quiet ) { printf ("grabbing %s\n", psource->filename); }
g_iLinecount = 0;
while (fgets( g_szLine, sizeof( g_szLine ), g_fpInput ) != NULL) { g_iLinecount++; sscanf( g_szLine, "%1023s %d", cmd, &option ); if (stricmp( cmd, "version" ) == 0) { if (option != 2) { MdlError("bad version\n"); } } else if (stricmp( cmd, "name" ) == 0) { } else if (stricmp( cmd, "vertices" ) == 0) { g_numverts = option; } else if (stricmp( cmd, "faces" ) == 0) { g_numfaces = option; } else if (stricmp( cmd, "materials" ) == 0) { // doesn't matter;
} else if (stricmp( cmd, "texcoords" ) == 0) { g_numtexcoords = option; if (option == 0) MdlError( "model has no texture coordinates\n"); } else if (stricmp( cmd, "normals" ) == 0) { g_numnormals = option; } else if (stricmp( cmd, "tristrips" ) == 0) { // should be 0;
}
else if (stricmp( cmd, "vertexlist" ) == 0) { Grab_Vertexlist( psource ); } else if (stricmp( cmd, "facelist" ) == 0) { Grab_Facelist( psource ); } else if (stricmp( cmd, "materiallist" ) == 0) { Grab_Materiallist( psource ); } else if (stricmp( cmd, "texcoordlist" ) == 0) { Grab_Texcoordlist( psource ); } else if (stricmp( cmd, "normallist" ) == 0) { Grab_Normallist( psource ); } else if (stricmp( cmd, "faceattriblist" ) == 0) { Grab_Faceattriblist( psource ); }
else if (stricmp( cmd, "MRM" ) == 0) { } else if (stricmp( cmd, "MRMvertices" ) == 0) { } else if (stricmp( cmd, "MRMfaces" ) == 0) { } else if (stricmp( cmd, "MRMfaceupdates" ) == 0) { Grab_MRMFaceupdates( psource ); }
else if (stricmp( cmd, "nodes" ) == 0) { psource->numbones = Grab_Nodes( psource->localBone ); } else if (stricmp( cmd, "skeleton" ) == 0) { Grab_Animation( psource, "BindPose" ); } /*
else if (stricmp( cmd, "triangles" ) == 0) { Grab_Triangles( psource ); } */ else { MdlError("unknown VRM command : %s \n", cmd ); } }
UnifyIndices( psource ); BuildIndividualMeshes( psource );
fclose( g_fpInput );
return 1; }
|