Counter Strike : Global Offensive Source Code
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

4471 lines
152 KiB

//========= Copyright © 1996-2008, Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
// $NoKeywords: $
//=============================================================================//
#pragma warning( disable : 4786 )
#pragma warning( disable : 4748 ) // buffer overrun with optimizations off
// This file has tons of problems with global optimizations. . turn 'em off.
// NOTE: Would be nice to have a test case for this! - not verified in vs2005
#pragma optimize( "g", off )
// use this much memory to build an output file in memory.
#define FILEBUFFER_SIZE ( 4 * 1024 * 1024 )
//#define IGNORE_BONES
#define NVTRISTRIP
// This is the minimum number of regular subd patches a mesh needs to contain before we split it into 2 draw calls ( regular vs extraordinary )
#define MINIMUM_REGULAR_PATCHES 300
#include <stdio.h>
#include <stdlib.h>
#include <float.h>
#include "mathlib/mathlib.h"
#include "cmdlib.h"
#include "studio.h"
#include "studiomdl.h"
#include "HardwareMatrixState.h"
#include "HardwareVertexCache.h"
#include "optimize.h"
#include <malloc.h>
#include <nvtristrip.h>
#include "FileBuffer.h"
#include "tier1/UtlVector.h"
#include "materialsystem/IMaterial.h"
#include "tier1/utllinkedlist.h"
#include "tier1/smartptr.h"
#include "tier2/p4helpers.h"
#define SUBD_STUDIOMDL
#include "optimize_subd.h"
bool g_bDumpGLViewFiles;
extern bool g_IHVTest;
// flush bones between strips rather than deallocating the LRU.
#define USE_FLUSH
extern int FindMaterialByName( const char *pMaterialName );
namespace OptimizedModel
{
//-----------------------------------------------------------------------------
// Defines which debugging output file we will see
//-----------------------------------------------------------------------------
enum
{
WRITEGLVIEW_SHOWMESH = 0x00000001,
WRITEGLVIEW_SHOWSTRIPGROUP = 0x00000002,
WRITEGLVIEW_SHOWSTRIP = 0x00000004,
WRITEGLVIEW_SHOWSUBSTRIP = 0x00000008,
WRITEGLVIEW_SHOWFLEXED = 0x00000010,
WRITEGLVIEW_SHOWSW = 0x00000020,
WRITEGLVIEW_SHOWMESHPROPS = 0x00000040,
WRITEGLVIEW_SHOWVERTNUMBONES = 0x00000080,
WRITEGLVIEW_SHOWSTRIPNUMBONES = 0x00000100,
WRITEGLVIEW_FORCEUINT = 0xffffffff
};
//-----------------------------------------------------------------------------
// This is used to help us figure out where in the file data should go
// and also to display stats
//-----------------------------------------------------------------------------
struct TotalMeshStats_t
{
int m_TotalBodyParts;
int m_TotalModels;
int m_TotalModelLODs;
int m_TotalMeshes;
int m_TotalStrips;
int m_TotalStripGroups;
int m_TotalVerts;
int m_TotalIndices;
int m_TotalTopologyIndices;
int m_TotalBoneStateChanges;
int m_TotalMaterialReplacements;
};
struct Face_t
{
Face_t()
{
touched = false;
for ( int i=0; i<4; i++)
{
neighborID[i] = -1;
vertID[i] = -1;
}
}
int vertID[4]; // Indices into exploded vertex data with splits at vertex component discontinuities
int neighborID[4]; // Neighboring faces across the four edges
int boneID[MAX_NUM_BONES_PER_VERT * 4]; // Worst case...allowing for quads
int numBones;
bool touched;
};
//-----------------------------------------------------------------------------
// Associates software bone indices with hardware bone indices
//-----------------------------------------------------------------------------
struct BoneStateChange_t
{
int hardwareID;
int newBoneID;
};
struct Strip_t
{
Strip_t::Strip_t()
{
numIndices = numTopologyIndices = 0;
}
// these are the verts and indices that are used while building.
// (there may be verts in here that aren't use in the stripGroup.)
CUtlVector<Vertex_t> verts;
int numIndices;
int numTopologyIndices;
unsigned short *pIndices;
unsigned short *pTopologyIndices;
unsigned int flags;
int numBones;
// These are the final, sorted verts as they appear in the strip group.
int stripGroupIndexOffset; // offset into stripGroup's indices
int numStripGroupIndices;
int stripGroupTopologyOffset; // offset into stripGroup's topology data
int numStripGroupTopologyIndices;
int stripGroupVertexOffset; // offset into stripGroup's verts
int numStripGroupVerts;
int numBoneStateChanges;
BoneStateChange_t boneStateChanges[MAX_NUM_BONES_PER_STRIP];
};
//-----------------------------------------------------------------------------
// a list of vertices + faces for a strip group
//-----------------------------------------------------------------------------
typedef CUtlVector<Vertex_t> VertexList_t;
typedef CUtlVector<Face_t> FaceList_t;
typedef CUtlVector<SubD_Face_t> SubD_FaceList_t;
typedef CUtlVector<unsigned short> VertexIndexList_t;
typedef CUtlVector<unsigned short> IndexTopologyList_t;
typedef CUtlVector<Strip_t> StripList_t;
typedef CUtlVector<bool> FaceProcessedList_t;
//-----------------------------------------------------------------------------
// String table
//-----------------------------------------------------------------------------
class CStringTable
{
public:
int StringTableOffset( const char *string )
{
int i;
int size = 0;
for( i = 0; i < m_Strings.Count(); i++ )
{
if( stricmp( m_Strings[i].Base(), string ) == 0 )
{
return size;
}
size += m_Strings[i].Count();
}
return -1;
}
bool StringPresent( const char *string )
{
int i;
for( i = 0; i < m_Strings.Count(); i++ )
{
if( stricmp( m_Strings[i].Base(), string ) == 0 )
{
return true;
}
}
return false;
}
void AddString( const char *newString )
{
if( StringPresent( newString ) )
{
return;
}
CUtlVector<char> &s = m_Strings[m_Strings.AddToTail()];
int size = strlen( newString ) + 1;
s.AddMultipleToTail( size );
strcpy( s.Base(), newString );
}
void Purge()
{
m_Strings.Purge();
}
int CalcSize( void )
{
int size = 0;
int i;
for( i = 0; i < m_Strings.Count(); i++ )
{
size += m_Strings[i].Count();
}
return size;
}
void WriteToMem( char *pDst )
{
int size = 0;
int i;
for( i = 0; i < m_Strings.Count(); i++ )
{
#ifdef _DEBUG
int j = Q_strlen( m_Strings[i].Base() ) + 1;
int k = m_Strings[i].Count();
Assert( j == k );
#endif
memcpy( pDst + size, m_Strings[i].Base(), m_Strings[i].Count() );
size += m_Strings[i].Count();
}
}
private:
typedef CUtlVector<char> CharVector_t;
CUtlVector<CharVector_t> m_Strings;
};
// global string table for the whole vtx file.
static CStringTable s_StringTable;
//-----------------------------------------------------------------------------
// This is all the indices, vertices, and strips that make up this group
// a group can be rendered all in one call to the material system
//-----------------------------------------------------------------------------
struct StripGroup_t
{
VertexIndexList_t indices;
IndexTopologyList_t topologyIndices;
VertexList_t verts;
StripList_t strips;
unsigned int flags;
};
struct Mesh_t
{
CUtlVector<StripGroup_t> stripGroups;
unsigned int flags;
};
struct ModelLOD_t
{
CUtlVector<Mesh_t> meshes;
float switchPoint;
};
struct Model_t
{
CUtlVector<ModelLOD_t> modelLODs;
};
//-----------------------------------------------------------------------------
// Main class that does all the dirty work to stripy + groupify
//-----------------------------------------------------------------------------
class COptimizedModel
{
public:
bool OptimizeFromStudioHdr( studiohdr_t *phdr, s_bodypart_t *pSrcBodyParts, int vertCacheSize,
bool usesFixedFunction, bool bForceSoftwareSkin, bool bHWFlex, int maxBonesPerVert, int maxBonesPerFace,
int maxBonesPerStrip, const char *fileName, const char *glViewFileName );
private:
void CleanupEverything();
// Setup to get the ball rolling
void SetupMeshProcessing( studiohdr_t *phdr, int vertCacheSize,
bool usesFixedFunction, int maxBonesPerVert, int maxBonesPerFace,
int maxBonesPerStrip, const char *fileName );
//
// Methods associated with mesh processing
//
void SourceMeshToFaceList( s_model_t *pSrcModel, s_mesh_t *pSrcMesh, CUtlVector<mstudioiface_t> &meshFaceList );
void CreateLODFaceList( s_model_t *pSrcModel, int nLodID, s_source_t* pLODSource,
mstudiomodel_t *pStudioModel,
mstudiomesh_t *pStudioMesh,
CUtlVector<mstudioiface_t> &meshFaceList,
bool bQuadSubd, bool writeDebug );
// This processes the model + breaks it into strips
void ProcessModel( studiohdr_t *phdr, s_bodypart_t *pSrcBodyParts, TotalMeshStats_t& stats,
bool bForceSoftwareSkin, bool bHWFlex );
// processes a single mesh within the model
void ProcessMesh( Mesh_t *pMesh, studiohdr_t *pStudioHeader, CUtlVector<mstudioiface_t> &srcFaces,
mstudiomodel_t *pStudioModel, mstudiomesh_t *pStudioMesh, bool bForceNoFlex,
bool bForceSoftwareSkin, bool bHWFlex, bool bQuadSubd );
// Processes a single strip group
void ProcessStripGroup( StripGroup_t *pStripGroup, bool bIsHWSkinned, bool bIsFlexed,
mstudiomodel_t *pStudioModel, mstudiomesh_t *pStudioMesh,
CUtlVector<mstudioiface_t> &srcFaces,
FaceProcessedList_t& facesProcessed,
int maxBonesPerVert, int maxBonesPerFace, int maxBonesPerStrip,
bool bForceNoFlex, bool bHWFlex, bool bQuadSubd );
// Constructs vertices appropriate for a strip group based on source face data
bool GenerateStripGroupVerticesFromFace( mstudioiface_t* pFace,
mstudiomesh_t *pStudioMesh, int maxPreferredBones, Vertex_t* pStripGroupVert, bool bQuadSubd );
// Count the number of unique bones in a set of vertices
int CountUniqueBones( int count, Vertex_t *pVertex ) const;
int CountMaxVertBones( int count, Vertex_t *pVertex ) const;
// Counts the unique # of bones in a strip
int CountUniqueBonesInStrip( StripGroup_t *pStripGroup, Strip_t *pStrip );
// Builds SW + HW skinned strips
void BuildSWSkinnedStrips( FaceList_t& faceList, SubD_FaceList_t& subdFaceList, VertexList_t const& vertices, StripGroup_t *pStripGroup );
void BuildHWSkinnedStrips( FaceList_t& faceList, VertexList_t& verts, StripGroup_t *pStripGroup, int maxBonesPerStrip );
// These methods deal with finding another face to batch together
// in a similar matrix state group
int ComputeNewBonesNeeded( Face_t const& face ) const;
bool AllocateHardwareBonesForFace( Face_t *face );
Face_t* GetNextFace( FaceList_t& faces, bool allowNewStrip );
Face_t* GetNextUntouchedWithoutBoneStateChange( FaceList_t& faces );
Face_t* GetNextUntouchedWithLeastBoneStateChanges( FaceList_t& faces );
// Actually does the stripification
void Stripify( VertexIndexList_t const& sourceIndices, IndexTopologyList_t const& sourceTopologyIndices,
bool bIsHWSkinned, int* pNumIndices, int* pNumTopologyIndices,
unsigned short** ppIndices , unsigned short** ppTopologyIndices, bool bQuadSubd );
// Makes sure our vertices are using the correct bones
void SanityCheckVertBones( VertexIndexList_t const& list, VertexList_t const& vertices );
// Sets the flags associated with a particular strip group + mesh
void ComputeStripGroupFlags( StripGroup_t *pStripGroup, bool bIsHWSkinned, bool bIsFlexed );
void ComputeMeshFlags( Mesh_t *pMesh, studiohdr_t *pStudioHeader, mstudiomesh_t *pStudioMesh );
bool MeshIsTeeth( studiohdr_t *pStudioHeader, mstudiomesh_t *pStudioMesh );
// Tries to add neighboring vertices that'll fit into the matrix transform state
void BuildStripsRecursive( VertexIndexList_t& indices, FaceList_t& list, Face_t *face );
// Figures out all bones affecting a particular face
void BuildFaceBoneData( VertexList_t& list, Face_t& face );
// Memory optimize the strip data
void PostProcessStripGroup( mstudiomodel_t *pStudioModel, mstudiomesh_t *pStudioMesh, StripGroup_t *pStripGroup );
void COptimizedModel::ZeroNumBones( void );
//
// Methods associated with writing VTX files
//
// This writes the strip data out to a VTX file
void WriteVTXFile( studiohdr_t *pHdr, char const* pFileName, TotalMeshStats_t const& stats );
// This writes the GL debugging files
void WriteGLViewFiles( studiohdr_t *pHdr, const char *glViewFileName );
// Subdivision surface related processing methods
bool ConditionMeshForSubdivision( FaceList_t& faceList, CUtlVector<int> *indexMapping, CUtlVector<Vector> *posList );
//bool ConditionFace( Face_t &face, FaceList_t& faceList, CUtlVector<int> *indexMapping, CUtlVector<Vector> *posList );
//int ConditionPoint( FaceList_t &faceList, int nV, int *nOtherPatchV, CUtlVector<int>* pNeighborPoints, CUtlVector<int> *indexMapping, CUtlVector<Vector> *posList );
//bool QuadContainsVertex( Face_t &face, int posID );
//int FindLocalIndexForPointInQuad( Face_t *face, int posID );
//Face_t *FindQuadWithPointsABButNotC( FaceList_t &faceList, int vertA, int vertB, int vertC );
//void SetUniquePositionIndices( CUtlVector<Vector> *posList, CUtlVector<int> *indexMapping, FaceList_t& faceList, int nNumVerts, const mstudio_meshvertexdata_t *pVertexData );
// new subdivision surface related processing methods
void BuildSubDFaceList( FaceList_t& faceList, SubD_FaceList_t& subDFaceList, VertexList_t& vtxList, const mstudio_meshvertexdata_t *pVertexData);
void OutputMemoryUsage( void );
bool IsVertexFlexed( mstudiomesh_t *pStudioMesh, int vertID ) const;
void BuildNeighborInfo( FaceList_t& list, int nMaxVertexId );
void ClearTouched( void );
void PrintVert( Vertex_t *v, mstudiomodel_t *pStudioModel, mstudiomesh_t *pStudioMesh );
void SanityCheckAgainstStudioHDR( studiohdr_t *phdr );
void WriteStringTable( int stringTableOffset );
void WriteMaterialReplacements( int materialReplacementsOffset );
void WriteMaterialReplacementLists( int materialReplacementsOffset, int materialReplacementListOffset );
void WriteHeader( int vertCacheSize, int maxBonesPerVert, int maxBonesPerFace, int maxBonesPerStrip, int numBodyParts, long checkSum );
void WriteBodyPart( int bodyPartID, mstudiobodyparts_t *pBodyPart, int modelID );
void WriteModel( int modelID, mstudiomodel_t *pModel, int lodID );
void WriteModelLOD( int lodID, ModelLOD_t *pLOD, int meshID );
void WriteMesh( int meshID, Mesh_t *pMesh, int stripGroupID );
void WriteStripGroup( int stripGroupID, StripGroup_t *pStripGroup, int vertID, int indexID , int topologyID, int stripID );
int WriteVerts( int vertID, StripGroup_t *pStripGroup );
int WriteIndices( int indexID, StripGroup_t *pStripGroup );
int WriteTopology( int topologyID, StripGroup_t *pStripGroup );
void WriteStrip( int stripID, Strip_t *pStrip, int indexID, int curTopology, int vertID, int boneID );
void WriteBoneStateChange( int boneID, BoneStateChange_t *boneStateChange );
void DrawGLViewTriangle( FILE *fp, Vector& pos1, Vector& pos2, Vector& pos3, Vector& color1, Vector& color2, Vector& color3 );
void WriteGLViewFile( studiohdr_t *phdr, const char *pFileName, unsigned int flags, float shrinkFactor );
void ShrinkVerts( float shrinkFactor );
void GLViewDrawBegin( int mode );
void CheckVertBoneWeights( Vertex_t *pVert, mstudiomodel_t *pStudioModel, mstudiomesh_t *pStudioMesh );
void GLViewVert( FILE *fp, Vertex_t vert, int index, Vector& color, mstudiomodel_t *pStudioModel, mstudiomesh_t *pStudioMesh, bool showSubStrips, float shrinkFraction );
void GLViewDrawEnd( void );
void SetMeshPropsColor( unsigned int meshFlags, Vector& color );
void SetFlexedAndSkinColor( unsigned int glViewFlags, unsigned int stripGroupFlags, Vector& color );
void SetColorFromNumVertexBones( int numBones, Vector& color );
Vector& GetOrigVertPosition( mstudiomodel_t *pStudioModel, mstudiomesh_t *pStudioMesh, Vertex_t *pVert );
float GetOrigVertBoneWeightValue( mstudiomodel_t *pStudioModel, mstudiomesh_t *pStudioMesh, Vertex_t *pVert, int boneID );
mstudioboneweight_t &GetOrigVertBoneWeight( mstudiomodel_t *pStudioModel, mstudiomesh_t *pStudioMesh, Vertex_t *pVert );
int GetOrigVertBoneIndex( mstudiomodel_t *pStudioModel, mstudiomesh_t *pStudioMesh, Vertex_t *pVert, int boneID );
void ShowStats( void );
void MapGlobalBonesToHardwareBoneIDsAndSortBones( studiohdr_t *phdr );
void RemoveRedundantBoneStateChanges( void );
void CheckVert( Vertex_t *pVert, int maxBonesPerFace, int maxBonesPerVert );
void CheckAllVerts( int maxBonesPerFace, int maxBonesPerVert );
void SortBonesWithinVertex( bool flexed, Vertex_t *vert, mstudiomodel_t *pStudioModel, mstudiomesh_t *pStudioMesh, int *globalToHardwareBoneIndex, int *hardwareToGlobalBoneIndex, int maxBonesPerFace, int maxBonesPerVert );
int GetTotalVertsForMesh( Mesh_t *pMesh );
int GetTotalIndicesForMesh( Mesh_t *pMesh );
int GetTotalTopologyIndicesForMesh( Mesh_t *pMesh );
int GetTotalStripsForMesh( Mesh_t *pMesh );
int GetTotalStripGroupsForMesh( Mesh_t *pMesh );
int GetTotalBoneStateChangesForMesh( Mesh_t *pMesh );
bool MeshNeedsRemoval( studiohdr_t *pHdr, mstudiomesh_t *pStudioMesh,
LodScriptData_t& scriptLOD );
void PrintBoneStateChanges( studiohdr_t *pHdr, int lod );
void PrintVerts( studiohdr_t *phdr, int lod );
void SanityCheckVertexBoneLODFlags( studiohdr_t *pStudioHdr, FileHeader_t *pVtxHeader );
//
// SOURCE DATA
//
// These are all the built models
CUtlVector<Model_t> m_Models;
// total number of bones in the studio model
int m_NumBones;
// information about the hardware
int m_VertexCacheSize;
int m_MaxBonesPerFace;
int m_MaxBonesPerVert;
int m_MaxBonesPerStrip;
bool m_bUsesFixedFunction;
// stats
int m_NumSkinnedAndFlexedVerts;
CHardwareMatrixState m_HardwareMatrixState;
// a place to stick file output.
CFileBuffer *m_FileBuffer;
// offset for different items in the output file.
int m_BodyPartsOffset;
int m_ModelsOffset;
int m_ModelLODsOffset;
int m_MeshesOffset;
int m_StripGroupsOffset;
int m_StripsOffset;
int m_VertsOffset;
int m_IndicesOffset;
int m_TopologyOffset;
int m_BoneStateChangesOffset;
int m_StringTableOffset;
int m_MaterialReplacementsOffset;
int m_MaterialReplacementsListOffset;
int m_EndOfFileOffset;
};
//-----------------------------------------------------------------------------
// Singleton instance
//-----------------------------------------------------------------------------
static COptimizedModel s_OptimizedModel;
//-----------------------------------------------------------------------------
// Cleanup method
//-----------------------------------------------------------------------------
void COptimizedModel::CleanupEverything()
{
}
void COptimizedModel::OutputMemoryUsage( void )
{
printf( "body parts: %7d bytes\n", ( int )( m_ModelsOffset - m_BodyPartsOffset ) );
printf( "models: %7d bytes\n", ( int )( m_MeshesOffset - m_ModelsOffset ) );
printf( "model LODs: %7d bytes\n", ( int )( m_MeshesOffset - m_ModelLODsOffset ) );
printf( "meshes: %7d bytes\n", ( int )( m_StripGroupsOffset - m_MeshesOffset ) );
printf( "strip groups: %7d bytes\n", ( int )( m_StripsOffset - m_StripGroupsOffset ) );
printf( "strips: %7d bytes\n", ( int )( m_VertsOffset - m_StripsOffset ) );
printf( "verts: %7d bytes\n", ( int )( m_IndicesOffset - m_VertsOffset ) );
printf( "indices: %7d bytes\n", ( int )( m_BoneStateChangesOffset - m_IndicesOffset ) );
printf( "bone changes: %7d bytes\n", ( int )( m_EndOfFileOffset - m_BoneStateChangesOffset ) );
printf( "everything: %7d bytes\n", ( int )( m_EndOfFileOffset ) );
}
void COptimizedModel::SanityCheckAgainstStudioHDR( studiohdr_t *phdr )
{
#if 0 // garymcthack
printf( "SanityCheckAgainstStudioHDR\n" );
FileHeader_t *header = ( FileHeader_t * )m_FileBuffer->GetPointer( 0 );
Assert( header->numBodyParts == phdr->numbodyparts );
for( int bodyPartID = 0; bodyPartID < header->numBodyParts; bodyPartID++ )
{
BodyPartHeader_t *bodyPart = header->pBodyPart( bodyPartID );
mstudiobodyparts_t *pStudioBodyPart = phdr->pBodypart( bodyPartID );
Assert( bodyPart->numModels == pStudioBodyPart->nummodels );
for( int modelID = 0; modelID < bodyPart->numModels; modelID++ )
{
ModelHeader_t *model = bodyPart->pModel( modelID );
mstudiomodel_t *pStudioModel = pStudioBodyPart->pModel( modelID );
Assert( model->numMeshes == pStudioModel->nummeshes );
for( int meshID = 0; meshID < model->numMeshes; meshID++ )
{
MeshHeader_t *mesh = model->pMesh( meshID );
mstudiomesh_t *pStudioMesh = pStudioModel->pMesh( meshID );
for( int stripGroupID = 0; stripGroupID < mesh->numStripGroups; stripGroupID++ )
{
StripGroupHeader_t *stripGroup = mesh->pStripGroup( stripGroupID );
for( int stripID = 0; stripID < stripGroup->numStrips; stripID++ )
{
StripHeader_t *strip = stripGroup->pStrip( stripID );
}
}
}
}
}
#endif
}
//-----------------------------------------------------------------------------
//
// The following methods are all related to creating groups of meshes to strip
//
//-----------------------------------------------------------------------------
//-----------------------------------------------------------------------------
// pick any face that hasn't been used yet to start with.
//-----------------------------------------------------------------------------
static Face_t *GetNextUntouched( FaceList_t& faces )
{
int i;
for( i = 0; i < faces.Count(); i++ )
{
if( !faces[i].touched )
return &faces[i];
}
return 0;
}
//-----------------------------------------------------------------------------
// Returns the number of bones that are not represented in the current hardware state
//-----------------------------------------------------------------------------
inline int COptimizedModel::ComputeNewBonesNeeded( Face_t const& face ) const
{
int numNewBones = 0;
for( int i = 0; i < face.numBones; ++i )
{
if( !m_HardwareMatrixState.IsMatrixAllocated( face.boneID[i] ) )
++numNewBones;
}
return numNewBones;
}
//-----------------------------------------------------------------------------
// returns face index
// Find the next face without a bone state change
// May add new hardware bones if there is space and is necessary.
//-----------------------------------------------------------------------------
Face_t* COptimizedModel::GetNextUntouchedWithoutBoneStateChange( FaceList_t& faces )
{
Face_t *bestFace = 0;
int vertsPerFace = faces[0].vertID[3] == -1 ? 3 : 4;
int bestNumNewBones = ( MAX_NUM_BONES_PER_VERT * vertsPerFace ) + 1;
int i;
for( i = 0; i < faces.Count(); i++ )
{
// We haven't processed this one, so let's try it
if( !faces[i].touched )
{
// How many bones are not represented in the current state?
int numNewBones = ComputeNewBonesNeeded( faces[i] );
// if this face fit and if it's the best so far, save it.
if ( (numNewBones <= m_HardwareMatrixState.FreeMatrixCount()) &&
(numNewBones < bestNumNewBones ) )
{
bestNumNewBones = numNewBones;
bestFace = &faces[i];
// Can't get any better than this!
if (bestNumNewBones == 0)
break;
}
}
}
return bestFace;
}
//---------------------------------------------------------------------------------
// This will returns the face that requires the least number of bone state changes
//---------------------------------------------------------------------------------
Face_t *COptimizedModel::GetNextUntouchedWithLeastBoneStateChanges( FaceList_t& faces )
{
Face_t *bestFace = 0;
int vertsPerFace = faces[0].vertID[3] == -1 ? 3 : 4;
int bestNumNewBones = ( MAX_NUM_BONES_PER_VERT * vertsPerFace ) + 1;
// For this one, just find the face that needs the least number
// of new bones. That way, we'll not have to change too many states
for( int i = 0; i < faces.Count(); i++ )
{
if( !faces[i].touched )
{
int numNewBones = ComputeNewBonesNeeded( faces[i] );
if( numNewBones < bestNumNewBones )
{
bestNumNewBones = numNewBones;
bestFace = &faces[i];
}
}
}
// This only happens if there are no faces untouched
if( !bestFace )
return 0;
#ifdef USE_FLUSH
m_HardwareMatrixState.DeallocateAll();
#else
// Remove bones until we have enough space...
int numToRemove = bestNumNewBones - m_HardwareMatrixState.FreeMatrixCount();
Assert( numToRemove > 0 );
m_HardwareMatrixState.DeallocateLRU(numToRemove);
#endif
return bestFace;
}
//-----------------------------------------------------------------------------
// Allocate bones for a face from the hardware matrix state
//-----------------------------------------------------------------------------
bool COptimizedModel::AllocateHardwareBonesForFace( Face_t *face )
{
for( int i = 0; i < face->numBones; ++i )
{
int bone = face->boneID[i];
if( !m_HardwareMatrixState.IsMatrixAllocated( bone ) )
{
if( !m_HardwareMatrixState.AllocateMatrix( bone ) )
return false;
}
}
return true;
}
//-----------------------------------------------------------------------------
// Get the next face.
// Try to find one that doesn't cause a bone state change by jumping to
// a new location in the model that works with the bones that we have allocated.
// If we can't find one, and allowNewStrip is true, then flush and pick the next
// best face that is close to the current hardware bone state.
//-----------------------------------------------------------------------------
Face_t *COptimizedModel::GetNextFace( FaceList_t& faceList, bool allowNewStrip )
{
// First try to get a face that doesn't involve changing matrix state
Face_t *face;
face = GetNextUntouchedWithoutBoneStateChange( faceList );
// If that didn't work, pick the face that changes the state the least
if( !face && allowNewStrip )
{
face = GetNextUntouchedWithLeastBoneStateChanges( faceList );
}
// Return the face we found
return face;
}
//-----------------------------------------------------------------------------
// Make sure all vertices we've added up to now use bones in the matrix list
//-----------------------------------------------------------------------------
void COptimizedModel::SanityCheckVertBones( VertexIndexList_t const& list, VertexList_t const& vertices )
{
#ifdef _DEBUG
Vertex_t const *pVert;
int i;
for( i = 0; i < list.Count(); i++ )
{
pVert = &vertices[list[i]];
if( !g_staticprop )
{
Assert( pVert->numBones != 0 );
}
int j;
for( j = 0; j < pVert->numBones; j++ )
{
if( pVert->boneID[j] == -1 )
{
continue;
}
if( !m_HardwareMatrixState.IsMatrixAllocated( pVert->boneID[j] ) )
{
Assert( 0 );
}
}
}
#endif
}
//-----------------------------------------------------------------------------
// Make sure all vertices we've added up to now use bones in the matrix list
//-----------------------------------------------------------------------------
void COptimizedModel::Stripify( VertexIndexList_t const& sourceIndices,
IndexTopologyList_t const& sourceTopologyIndices, bool bIsHWSkinned,
int* pNumIndices, int* pNumTopologyIndices, unsigned short** ppIndices,
unsigned short** ppTopologyIndices, bool bQuadSubd )
{
if( sourceIndices.Count() == 0 )
{
*ppIndices = 0;
*pNumIndices = 0;
return;
}
// Null out topology information by default
if ( pNumTopologyIndices )
*pNumTopologyIndices = 0;
if ( ppTopologyIndices )
*ppTopologyIndices = NULL;
// Skip the tristripping phase if we're building in preview mode, rendering quads or not hardware skinning
if ( bQuadSubd || g_bBuildPreview || !bIsHWSkinned || g_bPreserveTriangleOrder )
{
*pNumIndices = sourceIndices.Count();
*ppIndices = new unsigned short[*pNumIndices];
memcpy( *ppIndices, sourceIndices.Base(), (*pNumIndices) * sizeof(unsigned short) );
// Quad lists have additional topology indexing information, copy it in
if ( bQuadSubd )
{
*pNumTopologyIndices = sourceTopologyIndices.Count();
*ppTopologyIndices = new unsigned short[*pNumTopologyIndices];
memcpy( *ppTopologyIndices, sourceTopologyIndices.Base(), (*pNumTopologyIndices) * sizeof(unsigned short) );
}
return;
}
#ifdef NVTRISTRIP
PrimitiveGroup *primGroups;
unsigned short numPrimGroups;
// Be sure to call delete[] on the returned primGroups to avoid leaking memory
GenerateStrips( &sourceIndices[0], sourceIndices.Count(), &primGroups, &numPrimGroups );
Assert( numPrimGroups == 1 );
*pNumIndices = primGroups->numIndices;
*ppIndices = new unsigned short[*pNumIndices];
memcpy( *ppIndices, primGroups->indices, sizeof( unsigned short ) * *pNumIndices );
delete [] primGroups;
#endif
}
//-----------------------------------------------------------------------------
// Eat up face recursively by flood-filling around the model until
// we run out of bones on the hardware.
//-----------------------------------------------------------------------------
void COptimizedModel::BuildStripsRecursive( VertexIndexList_t& indices, FaceList_t& faceList, Face_t *face )
{
Assert( face );
// Don't process the face if it's already been processed
if( face->touched )
return;
// Only suck in faces that need no state change
if ( ComputeNewBonesNeeded( *face ) )
return;
// We've got enough hardware bones. Lets add this face's vertices, and
// then add the vertices of all the neighboring faces.
face->touched = true;
indices.AddToTail( ( unsigned short )face->vertID[0] );
indices.AddToTail( ( unsigned short )face->vertID[1] );
indices.AddToTail( ( unsigned short )face->vertID[2] );
if ( face->vertID[3] != -1 )
indices.AddToTail( ( unsigned short )face->vertID[3] ); // quad case
// Try to add our neighbors
if( face->neighborID[0] != -1 )
{
BuildStripsRecursive( indices, faceList, &faceList[face->neighborID[0]] );
}
if( face->neighborID[1] != -1 )
{
BuildStripsRecursive( indices, faceList, &faceList[face->neighborID[1]] );
}
if( face->neighborID[2] != -1 )
{
BuildStripsRecursive( indices, faceList, &faceList[face->neighborID[2]] );
}
if( face->neighborID[3] != -1 )
{
BuildStripsRecursive( indices, faceList, &faceList[face->neighborID[3]] );
}
}
//-----------------------------------------------------------------------------
// Processes a HW-skinned strip group
//-----------------------------------------------------------------------------
void COptimizedModel::BuildHWSkinnedStrips( FaceList_t& faceList, VertexList_t& vertices, StripGroup_t *pStripGroup, int maxBonesPerStrip )
{
// Set up the hardware matrix state
m_HardwareMatrixState.Init( maxBonesPerStrip );
int numVerts = faceList[0].vertID[3] == -1 ? 3 : 4;
// Empty out the list of faces to be stripified.
VertexIndexList_t facesToStrip;
facesToStrip.EnsureCapacity( faceList.Count() * numVerts );
// Pick any old unused face to start with.
Face_t *pSeedFace = GetNextUntouched( faceList );
while( pSeedFace )
{
// Make sure we've got our transforms allocated
#ifdef _DEBUG
bool ok =
#endif
AllocateHardwareBonesForFace( pSeedFace );
Assert( ok );
// Eat up face recursively by flood-filling around the model until
// we run out of bones on the hardware.
BuildStripsRecursive( facesToStrip, faceList, pSeedFace );
// Try to jump to a new location in the mesh without
// causing a hardware bone state overflow or flush.
pSeedFace = GetNextFace( faceList, false );
if( pSeedFace )
continue;
// Save the results of the generated strip.
int stripIdx = pStripGroup->strips.AddToTail( );
Strip_t& newStrip = pStripGroup->strips[stripIdx];
// Compute the strip flags
newStrip.flags = numVerts == 3 ? STRIP_IS_TRILIST : STRIP_IS_QUADLIST_EXTRA;
// Sanity check the indices of the bones.
SanityCheckVertBones( facesToStrip, vertices );
// There are no more faces to eat up without causing a flush, so
// go ahead and stripify what we have and flush.
// NOTE: This allocates space for stripIndices.pIndices
Stripify( facesToStrip, IndexTopologyList_t(), true, &newStrip.numIndices, 0, &newStrip.pIndices, NULL, numVerts == 4 );
// hack - should just build directly into newStrip.verts instead of using a global.
for( int i = 0; i < vertices.Count(); i++ )
{
newStrip.verts.AddToTail( vertices[i] );
}
// Compute the number of bones in this strip
newStrip.numBoneStateChanges = m_HardwareMatrixState.AllocatedMatrixCount();
Assert( newStrip.numBoneStateChanges <= maxBonesPerStrip );
// Save off the bones used for this strip.
for( int i = 0; i < m_HardwareMatrixState.AllocatedMatrixCount(); i++ )
{
newStrip.boneStateChanges[i].hardwareID = i;
newStrip.boneStateChanges[i].newBoneID = m_HardwareMatrixState.GetNthBoneGlobalID( i );
}
// Empty out the faces to strip so that we can start again with a new strip.
facesToStrip.RemoveAll();
// Get the next best face, allowing for a bone state flushes.
pSeedFace = GetNextFace( faceList, true );
}
}
//-----------------------------------------------------------------------------
// Processes a SW-skinned strip group
//-----------------------------------------------------------------------------
void COptimizedModel::BuildSWSkinnedStrips( FaceList_t& faceList, SubD_FaceList_t& subdFaceList, VertexList_t const& vertices, StripGroup_t *pStripGroup )
{
int nSubDFaces = subdFaceList.Count();
// Save the results of the generated strip.
int stripIdx = pStripGroup->strips.AddToTail( );
Strip_t *pNewStrip = &pStripGroup->strips[stripIdx];
int nFaceCount = faceList.Count();
int nVertsPerFace = faceList[0].vertID[3] == -1 ? 3 : 4;
pNewStrip->flags = nVertsPerFace == 3 ? STRIP_IS_TRILIST : STRIP_IS_QUADLIST_REG;
bool bSubDQuad = ( nVertsPerFace == 4 );
VertexIndexList_t indices;
indices.EnsureCapacity( nVertsPerFace * nFaceCount );
IndexTopologyList_t topologyIndices;
if ( bSubDQuad )
{
topologyIndices.EnsureCapacity( nVertsPerFace == 4 ? nFaceCount * ( MAX_SUBD_POINTS + 4 + 4 ) : nFaceCount * nVertsPerFace );
}
bool bExtraFaces = false;
for ( int f = 0; f < nFaceCount; ++f )
{
Face_t* face = &faceList[f];
face->touched = true;
if ( bSubDQuad )
{
SubD_Face_t* subDFace = &subdFaceList[f];
if ( !bExtraFaces && !FaceIsRegular( subDFace ) )
{
// First regular face
// We need at least MINIMUM_REGULAR_PATCHES(100) faces to make separating regular and extraordinary worthwhile
if ( topologyIndices.Count() > 0 && ( indices.Count() > MINIMUM_REGULAR_PATCHES * 4 ) )
{
// Finish off the previous strip
Stripify( indices, topologyIndices, false, &pNewStrip->numIndices, &pNewStrip->numTopologyIndices, &pNewStrip->pIndices, &pNewStrip->pTopologyIndices, bSubDQuad );
for ( int v = 0; v < vertices.Count(); v++ )
{
pNewStrip->verts.AddToTail( vertices[v] );
}
pNewStrip->numBoneStateChanges = 0;
for ( int b = 0; b < MAX_NUM_BONES_PER_STRIP; b++ )
{
pNewStrip->boneStateChanges[b].hardwareID = -1;
pNewStrip->boneStateChanges[b].newBoneID = -1;
}
// Start a new regular strip
stripIdx = pStripGroup->strips.AddToTail( );
pNewStrip = &pStripGroup->strips[stripIdx];
indices.RemoveAll();
indices.EnsureCapacity( nVertsPerFace * nFaceCount );
topologyIndices.RemoveAll();
topologyIndices.EnsureCapacity( nFaceCount * ( MAX_SUBD_POINTS + 4 + 4 ) );
}
pNewStrip->flags = STRIP_IS_QUADLIST_EXTRA;
bExtraFaces = true;
}
indices.AddToTail( ( unsigned short )face->vertID[0] );
indices.AddToTail( ( unsigned short )face->vertID[1] );
indices.AddToTail( ( unsigned short )face->vertID[2] );
indices.AddToTail( ( unsigned short )face->vertID[3] );
int j=0; // Next, fill in topology indices...
if ( nSubDFaces == nFaceCount )
{
// Subd verts, plus one-ring neighborhood
for ( j=0; j<4; j++ ) topologyIndices.AddToTail( subDFace->vtx1RingSize[j] );
for ( j=0; j<4; j++ ) topologyIndices.AddToTail( subDFace->vtx1RingCenterQuadOffset[j] );
for ( j=0; j<4; j++ ) topologyIndices.AddToTail( subDFace->valences[j] );
for ( j=0; j<4; j++ ) topologyIndices.AddToTail( subDFace->minOneRingIndex[j] );
for ( j=0; j<4; j++ ) topologyIndices.AddToTail( subDFace->bndVtx[j] );
for ( j=0; j<4; j++ ) topologyIndices.AddToTail( subDFace->bndEdge[j] );
for ( j=0; j<4; j++ ) topologyIndices.AddToTail( subDFace->cornerVtx[j] );
for ( j=0; j<4; j++ ) topologyIndices.AddToTail( subDFace->loopGapAngle[j] );
for ( j=0; j<4; j++ ) topologyIndices.AddToTail( subDFace->nbCornerVtx[j] );
for ( j=0; j<8; j++ ) topologyIndices.AddToTail( subDFace->edgeBias[j] );
for ( j=0; j<4; j++ ) topologyIndices.AddToTail( subDFace->vUV0[j] );
for ( j=0; j<4; j++ ) topologyIndices.AddToTail( subDFace->vUV1[j] );
for ( j=0; j<4; j++ ) topologyIndices.AddToTail( subDFace->vUV2[j] );
for ( j=0; j<4; j++ ) topologyIndices.AddToTail( subDFace->vUV3[j] );
short totalOneRingSize = subDFace->vtx1RingSize[0] + subDFace->vtx1RingSize[1] + subDFace->vtx1RingSize[2] + subDFace->vtx1RingSize[3];
for ( j=0; j<totalOneRingSize; ++j )
{
topologyIndices.AddToTail( subDFace->oneRing[j] );
}
}
}
else
{
indices.AddToTail( ( unsigned short )face->vertID[0] );
indices.AddToTail( ( unsigned short )face->vertID[1] );
indices.AddToTail( ( unsigned short )face->vertID[2] );
}
}
// NOTE: This allocates space for the indices
Stripify( indices, topologyIndices, false, &pNewStrip->numIndices, &pNewStrip->numTopologyIndices, &pNewStrip->pIndices, &pNewStrip->pTopologyIndices, bSubDQuad );
// hack - should just build directly into newStrip.verts instead of using a global.
for ( int v = 0; v < vertices.Count(); v++ )
{
pNewStrip->verts.AddToTail( vertices[v] );
}
pNewStrip->numBoneStateChanges = 0;
for ( int b = 0; b < MAX_NUM_BONES_PER_STRIP; b++ )
{
pNewStrip->boneStateChanges[b].hardwareID = -1;
pNewStrip->boneStateChanges[b].newBoneID = -1;
}
}
//-----------------------------------------------------------------------------
// Returns true if a particular vertex is part of a flex
//-----------------------------------------------------------------------------
bool COptimizedModel::IsVertexFlexed( mstudiomesh_t *pStudioMesh, int vertID ) const
{
mstudioflex_t *pflex = pStudioMesh->pFlex( 0 );
int i, j, n;
// Iterate through all the flexes, figure out if the vertex is part of the flex
for ( i = 0; i < pStudioMesh->numflexes; i++ )
{
byte *pvanim = pflex[i].pBaseVertanim();
int nVAnimSizeBytes = pflex[i].VertAnimSizeBytes();
for (j = 0; j < pflex[i].numverts; j++, pvanim += nVAnimSizeBytes )
{
mstudiovertanim_t *pAnim = (mstudiovertanim_t*)( pvanim );
n = pAnim->index;
if ( n == vertID )
return true;
}
}
return false;
}
//-----------------------------------------------------------------------------
// Computes flags for the strip group
//-----------------------------------------------------------------------------
void COptimizedModel::ComputeStripGroupFlags( StripGroup_t *pStripGroup, bool bIsHWSkinned, bool bIsFlexed )
{
pStripGroup->flags = 0;
if( bIsFlexed )
{
pStripGroup->flags |= STRIPGROUP_IS_DELTA_FLEXED; // Going forward, DX9 models are delta flexed
}
if( bIsHWSkinned )
{
pStripGroup->flags |= STRIPGROUP_IS_HWSKINNED;
}
}
//-----------------------------------------------------------------------------
// Try to reduce the number of bones affecting this vert so hardware can deal
//-----------------------------------------------------------------------------
#define MIN_BONE_INFLUENCE 1.0f
static void TryToReduceBoneInfluence( Vertex_t& stripGroupVert,
mstudioboneweight_t const& boneWeights, int maxBones )
{
int i;
while ( stripGroupVert.numBones > maxBones)
{
// Find the minimum bone weight...
float minWeight = 2.0;
int minIndex = -1;
for( i = 0; i < MAX_NUM_BONES_PER_VERT; ++i )
{
if (stripGroupVert.boneID[i] != 255)
{
float weight = boneWeights.weight[ stripGroupVert.boneWeightIndex[i] ];
if (weight < minWeight)
{
minWeight = weight;
minIndex = i;
}
}
}
Assert( minIndex >= 0 );
// Now that we got it, remove that bone influence if it's small enough
if (minWeight >= MIN_BONE_INFLUENCE)
return;
// Blat out that bone!
for( i = minIndex; i < MAX_NUM_BONES_PER_VERT - 1; ++i )
{
stripGroupVert.boneID[i] = stripGroupVert.boneID[i+1];
stripGroupVert.boneWeightIndex[i] = stripGroupVert.boneWeightIndex[i+1];
}
stripGroupVert.boneID[ MAX_NUM_BONES_PER_VERT - 1] = 255;
stripGroupVert.boneWeightIndex[ MAX_NUM_BONES_PER_VERT - 1] = 0;
--stripGroupVert.numBones;
}
}
//-----------------------------------------------------------------------------
// Generate three strip-group ready vertices from source mesh data
// returns true if the face constants a flexed vertex
//-----------------------------------------------------------------------------
bool COptimizedModel::GenerateStripGroupVerticesFromFace( mstudioiface_t* pFace,
mstudiomesh_t *pStudioMesh,
int maxPreferredBones,
Vertex_t* pStripGroupVert,
bool bQuadSubd )
{
int vertIDs[4] = { pFace->a, pFace->b, pFace->c, pFace->d };
const mstudio_meshvertexdata_t *vertData = pStudioMesh->GetVertexData();
Assert( vertData ); // This can only return NULL on X360 for now
bool bFaceIsFlexed = false;
int numVerts = bQuadSubd ? 4 : 3;
for( int faceIndex = 0; faceIndex < numVerts; ++faceIndex )
{
// Get the original source vertex id
int vertex = vertIDs[faceIndex];
// Check the verts of the face to see if they are flexed
bFaceIsFlexed = bFaceIsFlexed || IsVertexFlexed( pStudioMesh, vertex );
// How many bones affect this vertex
mstudioboneweight_t *pBoneWeight = vertData->BoneWeights(vertex);
int bonesAffectingVertex = pBoneWeight->numbones;
if( !g_staticprop && bonesAffectingVertex <= 0 )
{
MdlWarning( "too few bones/vert (%d) : it has no bones!\n", bonesAffectingVertex );
}
else if( bonesAffectingVertex > MAX_NUM_BONES_PER_VERT )
{
MdlError( "too many bones/vert (%d) : MAX_NUM_BONES_PER_VERT needs to be upped\n", bonesAffectingVertex );
}
// Set the fields of the strip group's vert
pStripGroupVert[faceIndex].origMeshVertID = vertex;
pStripGroupVert[faceIndex].numBones = bonesAffectingVertex;
int boneID;
#ifndef IGNORE_BONES
for( boneID = 0; boneID < bonesAffectingVertex; boneID++ )
{
pStripGroupVert[faceIndex].boneID[boneID] = pBoneWeight->bone[boneID];
pStripGroupVert[faceIndex].boneWeightIndex[boneID] = boneID;
}
for( ; boneID < MAX_NUM_BONES_PER_VERT; boneID++ )
{
pStripGroupVert[faceIndex].boneID[boneID] = 255;
pStripGroupVert[faceIndex].boneWeightIndex[boneID] = boneID;
}
#else
// don't let bones have any influence.
for( boneID = 0; boneID < MAX_NUM_BONES_PER_VERT; boneID++ )
{
pStripGroupVert[faceIndex].boneID[boneID] = 255;
pStripGroupVert[faceIndex].boneWeights[boneID] = 0;
}
#endif
// For hardware skinning, we want to try to reduce the number of
// bone influences, which we'll do here if we can.
// At the moment, we can't do this for fixed function because
// we get seam problems at transitions between SW and HW rendered
if (!m_bUsesFixedFunction)
{
if ( (maxPreferredBones > 0) && (bonesAffectingVertex > maxPreferredBones) )
{
TryToReduceBoneInfluence( pStripGroupVert[faceIndex], *pBoneWeight, maxPreferredBones );
}
}
}
return bFaceIsFlexed;
}
//-----------------------------------------------------------------------------
// Computes the unique number of bones in a specified set of vertices
//-----------------------------------------------------------------------------
int COptimizedModel::CountUniqueBones( int count, Vertex_t *pVertex ) const
{
int uniqueBoneCount = 0;
int uniqueBoneList[MAX_NUM_BONES_PER_STRIP];
while ( --count >= 0 )
{
for ( int i = 0; i < pVertex[count].numBones; ++i )
{
int boneID = pVertex[count].boneID[i];
int j = uniqueBoneCount;
while ( --j >= 0 )
{
if ( uniqueBoneList[j] == boneID )
break;
}
// Didn't find a match!
if ( j < 0 )
{
Assert( uniqueBoneCount < MAX_NUM_BONES_PER_STRIP );
uniqueBoneList[uniqueBoneCount++] = boneID;
}
}
}
return uniqueBoneCount;
}
int COptimizedModel::CountMaxVertBones( int count, Vertex_t *pVertex ) const
{
int maxBones = 0;
while ( --count >= 0 )
{
if ( maxBones < pVertex[count].numBones )
{
maxBones = pVertex[count].numBones;
}
}
return maxBones;
}
// Use a hash table to accelerate ProcessStripGroup/PostProcessStripGroup (initialization is slow, so instantiate it once globally and clear it before use)
struct StripVertLookup_t
{
int origMeshVertID;
int vertID;
};
static bool StripVertLookup_CompareFunc( const StripVertLookup_t &a, const StripVertLookup_t&b )
{
return ( a.origMeshVertID == b.origMeshVertID );
}
static unsigned int StripVertLookup_KeyFunc( const StripVertLookup_t &a )
{
return HashInt( a.origMeshVertID );
}
CUtlHash< StripVertLookup_t > g_StripGroupVertexLookup( 65536, 0, 0, StripVertLookup_CompareFunc, StripVertLookup_KeyFunc );
//-----------------------------------------------------------------------------
// Adds a vertex to the list of vertices to be added to the strip group
//-----------------------------------------------------------------------------
static int FindOrCreateVertex( VertexList_t& list, Vertex_t const& vert )
{
StripVertLookup_t stripVertLookup = { vert.origMeshVertID, -1 };
UtlHashHandle_t vertexHandle = g_StripGroupVertexLookup.Find( stripVertLookup );
if ( vertexHandle == g_StripGroupVertexLookup.InvalidHandle() )
{
int result = list.AddToTail( vert );
stripVertLookup.vertID = result;
g_StripGroupVertexLookup.Insert( stripVertLookup );
return result;
}
else
{
StripVertLookup_t &equivalentVertex = g_StripGroupVertexLookup.Element( vertexHandle );
int result = equivalentVertex.vertID;
// Double-check that the verts match
Assert( !memcmp( &list[result], &vert, sizeof( vert )) );
return result;
}
}
//-----------------------------------------------------------------------------
// Computes the bones used by the face
//-----------------------------------------------------------------------------
void COptimizedModel::BuildFaceBoneData( VertexList_t& list, Face_t& face )
{
int j, k, l;
int vertsPerFace = face.vertID[3] == -1 ? 3 : 4;
int maxBonesPerFace = MAX_NUM_BONES_PER_VERT * vertsPerFace ;
// Blat out the bone ID state
face.numBones = 0;
for( j = 0; j < maxBonesPerFace; j++ )
{
face.boneID[j] = -1;
}
// Iterate through the vertices in the face
for( j = 0; j < vertsPerFace; j++ )
{
// Iterate over the bones influencing the vertex
Vertex_t& vert = list[face.vertID[j]];
for( k = 0; k < vert.numBones; ++k )
{
int bone = vert.boneID[k];
Assert( (bone >= 0) && (bone < m_NumBones) );
// Look for matches with previously found bones
for ( l = face.numBones; --l >= 0; )
{
if ( bone == face.boneID[l] )
break;
}
// No match, add it to our list
if ( l < 0 )
face.boneID[face.numBones++] = bone;
}
}
}
//-----------------------------------------------------------------------------
// Computes neighboring faces along each edge of a face
//-----------------------------------------------------------------------------
struct EdgeInfo_t
{
int m_nConnectedVertId;
int m_nEdgeIndex;
int m_nFaceId;
};
static void FindMatchingEdge( FaceList_t& list, int nFaceId, int nEdgeIndex,
const int *pVertIds, CUtlVector< int >& vertexToEdges, CUtlFixedLinkedList< EdgeInfo_t >& edges )
{
Face_t &face = list[nFaceId];
Assert( face.neighborID[nEdgeIndex] == -1 );
int nOrder = ( pVertIds[0] < pVertIds[1] ) ? 0 : 1;
int nVertIndex = pVertIds[nOrder]; // the 'lower' of the edge's two verts (use this to search for a matching edge)
int nConnectedVertId = pVertIds[1-nOrder]; // the 'higher' of the edge's two verts
int hFirstEdge = vertexToEdges[nVertIndex];
for ( int hEdge = hFirstEdge; hEdge != edges.InvalidIndex(); hEdge = edges.Next(hEdge) )
{
EdgeInfo_t &edge = edges[hEdge];
if ( edge.m_nConnectedVertId != nConnectedVertId )
continue;
// Can't attach faces to themselves
if ( edge.m_nFaceId == nFaceId )
continue;
// Found a match! Mark the two faces as sharing an edge
face.neighborID[nEdgeIndex] = edge.m_nFaceId;
list[ edge.m_nFaceId ].neighborID[ edge.m_nEdgeIndex ] = nFaceId;
if ( hEdge == hFirstEdge )
{
vertexToEdges[nVertIndex] = edges.Next( hFirstEdge );
}
// Remove the edge from the list now it has been matched (should have at most 2 faces connected to an edge!)
edges.Free( hEdge );
return;
}
// No match! Insert the disconnected edge into the edge list
int hNewEdge = edges.Alloc( true );
EdgeInfo_t &newEdge = edges[hNewEdge];
newEdge.m_nConnectedVertId = nConnectedVertId;
newEdge.m_nEdgeIndex = nEdgeIndex;
newEdge.m_nFaceId = nFaceId;
edges.LinkBefore( hFirstEdge, hNewEdge );
vertexToEdges[nVertIndex] = hNewEdge;
}
//
// The next several functions are devoted to subdivision surface neighborhood processing
// This algorithm is adapted from the SubD sample in the DirectX10 SDK
//
/*
bool COptimizedModel::QuadContainsVertex( Face_t &quad, int posID )
{
if( posID == quad.posID[0] || posID == quad.posID[1] || posID == quad.posID[2] || posID == quad.posID[3] )
{
return true;
}
return false;
}
int COptimizedModel::FindLocalIndexForPointInQuad( Face_t *pQuad, int posID )
{
for( int i=0; i<4; i++ )
{
if( posID == pQuad->posID[i] )
return i;
}
return -1;
}
Face_t *COptimizedModel::FindQuadWithPointsABButNotC( FaceList_t &faceList, int vertA, int vertB, int vertC )
{
int maxFaces = faceList.Count();
for( int i=0; i < maxFaces; i++ )
{
Face_t &quad = faceList[i];
if( QuadContainsVertex( quad, vertA ) && QuadContainsVertex( quad, vertB ) && !QuadContainsVertex( quad, vertC ) )
{
return &quad;
}
}
Assert( 0 );
return NULL;
}
void ReportTopologyError( int a, int b, int c, CUtlVector<int> *indexMapping, CUtlVector<Vector> *posList )
{
Msg( "Topology error at vertices %d, %d, %d\n", a, b, c );
Vector &vA = posList->Element( a );
Vector &vB = posList->Element( b );
Vector &vC = posList->Element( c );
Msg( "spaceLocator -p %.4f %.4f %.4f;\n", vA.x, vA.y, vA.z );
Msg( "spaceLocator -p %.4f %.4f %.4f;\n", vB.x, vB.y, vB.z );
Msg( "spaceLocator -p %.4f %.4f %.4f;\n", vC.x, vC.y, vC.z );
}
int COptimizedModel::ConditionPoint( FaceList_t &faceList, int nV, int *nOtherPatchV, CUtlVector<int>* pNeighborPoints, CUtlVector<int> *indexMapping, CUtlVector<Vector> *posList )
{
int OriginalNeighborPoints = pNeighborPoints->Count();
// Find the outer face that shares the edge formed by the current vertex in the quad and
// the previous vertex in the quad (previous based on winding order)
Face_t *pCurrentQuad = FindQuadWithPointsABButNotC( faceList, nV, nOtherPatchV[2], nOtherPatchV[0] );
if ( pCurrentQuad == NULL )
{
ReportTopologyError( nV, nOtherPatchV[2], nOtherPatchV[0], indexMapping, posList );
return -1;
}
Face_t *pEndQuad = FindQuadWithPointsABButNotC( faceList, nV, nOtherPatchV[0], nOtherPatchV[2] );
if ( pEndQuad == NULL )
{
ReportTopologyError( nV, nOtherPatchV[0], nOtherPatchV[2], indexMapping, posList );
return -1;
}
int iFarEdgePoint = FindLocalIndexForPointInQuad( pCurrentQuad, nOtherPatchV[2] );
if ( iFarEdgePoint < 0 )
{
Msg( "Topology error at vertex %d\n", nOtherPatchV[2] );
}
int iOffEdgePoint = pCurrentQuad->posID[ (iFarEdgePoint + 1)%4 ];
int iFanPoint = pCurrentQuad->posID[ (iFarEdgePoint + 2)%4 ];
pNeighborPoints->AddToTail( iFanPoint ); // Add the first point
pCurrentQuad = FindQuadWithPointsABButNotC( faceList, nV, iFanPoint, iOffEdgePoint ); // Move to the next quad
if ( pCurrentQuad == NULL )
{
ReportTopologyError( nV, iFanPoint, iOffEdgePoint, indexMapping, posList );
return -1;
}
while( pCurrentQuad != pEndQuad )
{
int iFarEdgePoint = FindLocalIndexForPointInQuad( pCurrentQuad, iFanPoint );
iOffEdgePoint = pCurrentQuad->posID[ (iFarEdgePoint + 1)%4 ];
iFanPoint = pCurrentQuad->posID[ (iFarEdgePoint + 2)%4 ];
iOffEdgePoint = iOffEdgePoint;
iFanPoint = iFanPoint;
pNeighborPoints->AddToTail( iOffEdgePoint ); // Add the off-edge point and fan point
pNeighborPoints->AddToTail( iFanPoint );
pCurrentQuad = FindQuadWithPointsABButNotC( faceList, nV, iFanPoint, iOffEdgePoint );
if ( pCurrentQuad == NULL )
{
ReportTopologyError( nV, iFanPoint, iOffEdgePoint, indexMapping, posList );
return -1;
}
}
int NewNeighborPoints = pNeighborPoints->Count() - OriginalNeighborPoints; // Valence is a function of neighbor points
return (NewNeighborPoints + 5) / 2;
}
//--------------------------------------------------------------------------------------
// Condition a subd patch. Find the 1-ring neighborhood and precompute
// some offsets around it to make runtime conversion to Bezier easier.
//--------------------------------------------------------------------------------------
bool COptimizedModel::ConditionFace( Face_t &face, FaceList_t& faceList, CUtlVector<int> *indexMapping, CUtlVector<Vector> *posList )
{
CUtlVector< int > NeighborPoints;
int vert0 = face.posID[0], vert1 = face.posID[1];
int vert2 = face.posID[2], vert3 = face.posID[3];
int vOther[3];
vOther[0] = vert1; vOther[1] = vert2; vOther[2] = vert3; // vert0
face.valence[0] = ConditionPoint( faceList, vert0, vOther, &NeighborPoints, indexMapping, posList );
if ( face.valence[0] < 0 )
return false;
face.prefix[0] = NeighborPoints.Count() + 4;
vOther[0] = vert2; vOther[1] = vert3; vOther[2] = vert0; // vert1
face.valence[1] = ConditionPoint( faceList, vert1, vOther, &NeighborPoints, indexMapping, posList );
if ( face.valence[1] < 0 )
return false;
face.prefix[1] = NeighborPoints.Count() + 4;
vOther[0] = vert3; vOther[1] = vert0; vOther[2] = vert1; // vert2
face.valence[2] = ConditionPoint( faceList, vert2, vOther, &NeighborPoints, indexMapping, posList );
if ( face.valence[2] < 0 )
return false;
face.prefix[2] = NeighborPoints.Count() + 4;
vOther[0] = vert0; vOther[1] = vert1; vOther[2] = vert2; // vert3
face.valence[3] = ConditionPoint( faceList, vert3, vOther, &NeighborPoints, indexMapping, posList );
if ( face.valence[3] < 0 )
return false;
face.prefix[3] = NeighborPoints.Count() + 4;
// For now, allow no boundary edges (valence less than three)
Assert( (face.valence[0] > 2) && (face.valence[1] > 2) && (face.valence[2] > 2) && (face.valence[3] > 2));
int NumNeighbors = NeighborPoints.Count();
#ifdef _DEBUG
// Msg( " Neighbors: %d\n", NumNeighbors );
// Msg( " Valences: %d %d %d %d\n", face.valence[0], face.valence[1], face.valence[2], face.valence[3] );
// Msg( " Prefixes: %d %d %d %d\n", face.prefix[0], face.prefix[1], face.prefix[2], face.prefix[3] );
#endif
// Move neighbors into the points list
for( int i=0; i < min( NumNeighbors, MAX_SUBD_POINTS-4 ); i++ )
{
face.posID[4+i] = NeighborPoints[i];
}
return true;
}
bool COptimizedModel::ConditionMeshForSubdivision( FaceList_t& faceList, CUtlVector<int> *indexMapping, CUtlVector<Vector> *posList )
{
int nFaceCount = faceList.Count();
for ( int i = 0; i < nFaceCount; ++i )
{
if ( !ConditionFace( faceList[i], faceList, indexMapping, posList ) )
return false;
}
// For each face, remap unique position indices to global indices
for ( int i = 0; i < nFaceCount; ++i )
{
int j=0;
while( faceList[i].posID[j] != -1 )
{
faceList[i].posID[j] = indexMapping->Element( faceList[i].posID[j] );
j++;
}
}
return true;
}
void COptimizedModel::SetUniquePositionIndices( CUtlVector<Vector> *posList, CUtlVector<int> *indexMapping, FaceList_t& faceList, int nNumVerts, const mstudio_meshvertexdata_t *pVertexData )
{
int nFaceCount = faceList.Count();
for ( int f = 0; f < nFaceCount; f++ )
{
Face_t &face = faceList[f];
for ( int v=0; v<4; v++ )
{
// Position from exploded vertexdata
Vector *vPos = pVertexData->Position( face.vertID[v] );
Assert( vPos );
int posIndex = posList->Find( *vPos ); // Try to find
if ( !posList->IsValidIndex( posIndex ) ) // If not found
{
posIndex = posList->AddToTail( *vPos ); // Insert
indexMapping->AddToTail( face.vertID[v] ); // Insert
}
face.posID[v] = posIndex; // Store index into unique position list
}
}
}
*/
void COptimizedModel::BuildSubDFaceList( FaceList_t& faceList, SubD_FaceList_t& subDFaceList, VertexList_t& vtxList, const mstudio_meshvertexdata_t *pVertexData)
{
int nFaceCount = faceList.Count();
subDFaceList.RemoveAll();
subDFaceList.EnsureCapacity( nFaceCount );
for ( int i = 0; i < nFaceCount; ++i )
{
subDFaceList.AddToTail();
for (int j=0; j<4; j++)
{
subDFaceList[i].vtxIDs[j] = faceList[i].vertID[j];
}
}
// build SubD data
COptimizeSubDBuilder subDBuilder( subDFaceList, vtxList, pVertexData, false );
}
void COptimizedModel::BuildNeighborInfo( FaceList_t& faceList, int nMaxVertexId )
{
// NOTE: vertexToEdges[vertId] contains the index of the head of a linked list stored in edges
CUtlFixedLinkedList< EdgeInfo_t > edges;
CUtlVector< int > vertexToEdges;
vertexToEdges.SetCount( nMaxVertexId );
memset( vertexToEdges.Base(), 0, nMaxVertexId * sizeof(int) );
int numVerts = faceList[0].vertID[3] == -1 ? 3 : 4;
int pEdgeVertIds[2];
int nFaceCount = faceList.Count();
edges.EnsureCapacity( 2 * nFaceCount );
for ( int i = 0; i < nFaceCount; ++i )
{
Face_t &face = faceList[i];
// Add the edges for this face into a lookup table indexed by the lower vertID
pEdgeVertIds[0] = face.vertID[0]; pEdgeVertIds[1] = face.vertID[1];
FindMatchingEdge( faceList, i, 0, pEdgeVertIds, vertexToEdges, edges );
pEdgeVertIds[0] = face.vertID[1]; pEdgeVertIds[1] = face.vertID[2];
FindMatchingEdge( faceList, i, 1, pEdgeVertIds, vertexToEdges, edges );
if ( numVerts == 3 )
{
pEdgeVertIds[0] = face.vertID[2]; pEdgeVertIds[1] = face.vertID[0];
FindMatchingEdge( faceList, i, 2, pEdgeVertIds, vertexToEdges, edges );
}
else // must be a quad
{
pEdgeVertIds[0] = face.vertID[2]; pEdgeVertIds[1] = face.vertID[3];
FindMatchingEdge( faceList, i, 2, pEdgeVertIds, vertexToEdges, edges );
pEdgeVertIds[0] = face.vertID[3]; pEdgeVertIds[1] = face.vertID[0];
FindMatchingEdge( faceList, i, 3, pEdgeVertIds, vertexToEdges, edges );
}
}
}
//-----------------------------------------------------------------------------
// Processes a single strip group
//-----------------------------------------------------------------------------
void COptimizedModel::ProcessStripGroup( StripGroup_t *pStripGroup, bool bIsHWSkinned,
bool bIsFlexed, mstudiomodel_t *pStudioModel,
mstudiomesh_t *pStudioMesh,
CUtlVector<mstudioiface_t> &srcFaces,
FaceProcessedList_t& facesProcessed,
int maxBonesPerVert, int maxBonesPerFace,
int maxBonesPerStrip, bool bForceNoFlex, bool bHWFlex, bool bQuadSubd )
{
ComputeStripGroupFlags( pStripGroup, bIsHWSkinned, bIsFlexed );
// All of the faces before stripping for the current stripgroup
FaceList_t stripGroupSourceFaces;
SubD_FaceList_t stripGroupSubDFaces;
VertexList_t stripGroupVertices;
g_StripGroupVertexLookup.RemoveAll();
// FIXME: Flexed/HWSkinned state of faces don't change with each pass.
// We could precompute those flags just once (instead of doing it 4 times)
// Add each face to the stripgroup, if it's appropriate
for( int n=0; n < srcFaces.Count(); ++n )
{
if ( facesProcessed[n] ) // Don't bother processing a face that's already been done
continue;
mstudioiface_t *pFace = &srcFaces[n];
int preferredBones = bIsHWSkinned && (bHWFlex || !bIsFlexed) ? maxBonesPerVert : 0;
// Start a new strip group header
Vertex_t stripGroupVert[4];
bool bFaceIsFlexed = GenerateStripGroupVerticesFromFace( pFace, pStudioMesh, preferredBones, stripGroupVert, bQuadSubd );
if( bForceNoFlex )
{
bFaceIsFlexed = false;
}
// Unless we're a quad-only subd, don't bother to add the faces to our strip-group's face list if we don't need to
if ( !bQuadSubd && ( bFaceIsFlexed != bIsFlexed ) )
continue;
// If we're doing software skinning, then always accept, even if
// this face should have been a HW skinned face.
if ( bIsHWSkinned )
{
// Check how many unique bones were in the face, and how many vertices maximally in a vertex
int numVertexBones = CountMaxVertBones( bQuadSubd ? 4 : 3, stripGroupVert );
int numFaceBones = CountUniqueBones( bQuadSubd ? 4 : 3, stripGroupVert );
// If we have too many, we'll be skinning in software
bool bFaceIsHWSkinned = ( numFaceBones <= maxBonesPerFace ) && ( numVertexBones <= maxBonesPerVert );
// Don't bother to add vertices which aren't going to be used in this pass
if ( !bFaceIsHWSkinned )
continue;
}
// Add a new face to our list of faces
int nFaceIndex = stripGroupSourceFaces.AddToTail( );
Face_t& newFace = stripGroupSourceFaces[nFaceIndex];
newFace.vertID[0] = FindOrCreateVertex( stripGroupVertices, stripGroupVert[0] );
newFace.vertID[1] = FindOrCreateVertex( stripGroupVertices, stripGroupVert[1] );
newFace.vertID[2] = FindOrCreateVertex( stripGroupVertices, stripGroupVert[2] );
newFace.vertID[3] = bQuadSubd ? FindOrCreateVertex( stripGroupVertices, stripGroupVert[3] ) : -1;
BuildFaceBoneData( stripGroupVertices, newFace );
facesProcessed[n] = true;
}
// No mesh? bye bye
if ( stripGroupSourceFaces.Count() == 0 )
return;
// Figure out neighboring faces (this is generally the bottleneck of studiomdl)
BuildNeighborInfo( stripGroupSourceFaces, stripGroupVertices.Count() );
// Condition subd mesh by computing vertex neighbors and valences
if ( bQuadSubd )
{
BuildSubDFaceList( stripGroupSourceFaces, stripGroupSubDFaces, stripGroupVertices, pStudioMesh->GetVertexData() );
}
// Build the actual strips
if( bIsHWSkinned )
{
BuildHWSkinnedStrips( stripGroupSourceFaces, stripGroupVertices, pStripGroup, maxBonesPerStrip );
}
else
{
BuildSWSkinnedStrips( stripGroupSourceFaces, stripGroupSubDFaces, stripGroupVertices, pStripGroup );
}
}
//-----------------------------------------------------------------------------
// How many unique bones are in a strip?
//-----------------------------------------------------------------------------
int COptimizedModel::CountUniqueBonesInStrip( StripGroup_t *pStripGroup, Strip_t *pStrip )
{
int *boneUsageCounts = ( int * )_alloca( m_NumBones * sizeof( int ) );
memset( boneUsageCounts, 0, sizeof( int ) * m_NumBones );
int i;
for( i = 0; i < pStrip->numStripGroupVerts; i++ )
{
Vertex_t *pVert = &pStripGroup->verts[i+pStrip->stripGroupVertexOffset];
if( !g_staticprop )
{
Assert( pVert->numBones != 0 );
}
int j;
for( j = 0; j < pVert->numBones; j++ )
{
int boneID;
boneID = pVert->boneID[j];
Assert( boneID != 255 );
boneUsageCounts[boneID]++;
}
}
int numBones = 0;
for( i = 0; i < m_NumBones; i++ )
{
if( boneUsageCounts[i] )
{
numBones++;
}
}
return numBones;
}
//-----------------------------------------------------------------------------
// A little work to be done after we construct the strip groups
//-----------------------------------------------------------------------------
void COptimizedModel::PostProcessStripGroup( mstudiomodel_t *pStudioModel, mstudiomesh_t *pStudioMesh, StripGroup_t *pStripGroup )
{
// Compile all of the vertices in the current strip into the strip group's vertex list
for( int i = 0; i < pStripGroup->strips.Count(); i++ )
{
// Create sorted strip verts and indices in the stripgroup
Strip_t *pStrip = &pStripGroup->strips[i];
int vertOffset = pStripGroup->verts.Count();
pStrip->stripGroupVertexOffset = vertOffset;
pStrip->stripGroupIndexOffset = pStripGroup->indices.Count();
pStrip->stripGroupTopologyOffset = pStripGroup->topologyIndices.Count();
// Make sure we have enough memory allocated
pStripGroup->indices.EnsureCapacity( pStripGroup->indices.Count() + pStrip->numIndices );
pStripGroup->topologyIndices.EnsureCapacity( pStripGroup->topologyIndices.Count() + pStrip->numTopologyIndices );
// Try to find each of the strip's vertices in the strip group
int maxNumBones = 0;
bool bSubDQuad = pStrip->flags & STRIP_IS_QUADLIST_EXTRA || pStrip->flags & STRIP_IS_QUADLIST_REG;
int nSearch = vertOffset;
if ( bSubDQuad )
{
// Both strips of the subd mesh share the same vertices, so our vertex offset is 0 we start our
// search at 0 when looking for similar verts in the existing vert list
pStrip->stripGroupVertexOffset = 0;
nSearch = 0;
}
// Use a hash table to speed up this process:
g_StripGroupVertexLookup.RemoveAll();
for ( int k = nSearch; k < pStripGroup->verts.Count(); k++ )
{
StripVertLookup_t stripVertLookup = { pStripGroup->verts[k].origMeshVertID, k };
g_StripGroupVertexLookup.Insert( stripVertLookup );
}
for ( int j = 0; j < pStrip->numIndices; j++ )
{
int newIndex = -1;
int index = pStrip->pIndices[j];
Vertex_t *pVert = &pStrip->verts[index];
// Does this vertex exist in the strip group?
StripVertLookup_t stripVertLookup = { pVert->origMeshVertID, -1 };
UtlHashHandle_t vertexHandle = g_StripGroupVertexLookup.Find( stripVertLookup );
if ( vertexHandle != g_StripGroupVertexLookup.InvalidHandle() )
{
newIndex = g_StripGroupVertexLookup.Element( vertexHandle ).vertID;
}
else
{
// Didn't find it? Add the vertex to the list
newIndex = pStripGroup->verts.AddToTail( *pVert );
stripVertLookup.vertID = newIndex;
g_StripGroupVertexLookup.Insert( stripVertLookup );
}
pStripGroup->indices.AddToTail( newIndex );
Assert( pVert->numBones <= MAX_NUM_BONES_PER_VERT );
if ( pVert->numBones > MAX_NUM_BONES_PER_VERT )
{
MdlWarning( "Error: vertex weight exceeds %i bones on one vertex!\n", MAX_NUM_BONES_PER_VERT );
}
float weight = 0;
for( int i = 0; i < pVert->numBones; i++ )
{
weight += GetOrigVertBoneWeightValue( pStudioModel, pStudioMesh, pVert, i );
}
if ( weight < 0.95 || weight > 1.05 )
{
MdlWarning( "Error: vertex is weighted at %f (that's beyond normalized range)\n", weight );
}
/*
#ifdef _DEBUG
// float GetOrigVertBoneWeightValue( mstudiomodel_t *pStudioModel, mstudiomesh_t *pStudioMesh, Vertex_t *pVert, int boneID );
float weights[ MAX_NUM_BONES_PER_VERT ];
Assert( pVert->numBones <= MAX_NUM_BONES_PER_VERT );
for( int i = 0; i < pVert->numBones; i++ )
{
weights[i] = GetOrigVertBoneWeightValue( pStudioModel, pStudioMesh, pVert, i );
}
#endif
*/
// Keep track of the max # of bones in a vert
if ( pVert->numBones > maxNumBones )
maxNumBones = pVert->numBones;
}
for( int j = 0; j < pStrip->numTopologyIndices; j++ )
{
pStripGroup->topologyIndices.AddToTail( pStrip->pTopologyIndices[j] );
}
pStrip->numStripGroupIndices = pStripGroup->indices.Count() - pStrip->stripGroupIndexOffset;
pStrip->numStripGroupTopologyIndices = pStripGroup->topologyIndices.Count() - pStrip->stripGroupTopologyOffset;
pStrip->numStripGroupVerts = pStripGroup->verts.Count() - pStrip->stripGroupVertexOffset;
// The number of bones in a strip is the max number of
// bones in a vertex in this strip for vertex shaders, and it's the
// number of unique bones in the strip for fixed-function
if ( !m_bUsesFixedFunction )
pStrip->numBones = maxNumBones;
else
pStrip->numBones = CountUniqueBonesInStrip( pStripGroup, pStrip );
}
}
//-----------------------------------------------------------------------------
// Returns true if a particular mesh is teeth
//-----------------------------------------------------------------------------
bool COptimizedModel::MeshIsTeeth( studiohdr_t *pStudioHeader, mstudiomesh_t *pStudioMesh )
{
// The mesh is teeth if it's got a skin whose material has a non-zero flags
int i;
for( i = 0; i < pStudioHeader->numskinfamilies; i++ )
{
short *pskinref = pStudioHeader->pSkinref( 0 );
pskinref += ( i * pStudioHeader->numskinref );
mstudiotexture_t *ptexture;
ptexture = pStudioHeader->pTexture( pskinref[ pStudioMesh->material ] );
if( ptexture->flags )
{
return true;
}
}
return false;
}
//-----------------------------------------------------------------------------
// Computes the flags for a mesh
//-----------------------------------------------------------------------------
void COptimizedModel::ComputeMeshFlags( Mesh_t *pMesh, studiohdr_t *pStudioHeader,
mstudiomesh_t *pStudioMesh )
{
// eyeball?
pMesh->flags = 0;
if( pStudioMesh->materialtype != 0 )
{
pMesh->flags |= MESH_IS_EYES;
}
// teeth?
if( MeshIsTeeth( pStudioHeader, pStudioMesh ) )
{
pMesh->flags |= MESH_IS_TEETH;
}
}
//-----------------------------------------------------------------------------
// Creates 4 strip groups for a mesh, combinations of flexed + hwskinned
// A mesh has a single material
//-----------------------------------------------------------------------------
void COptimizedModel::ProcessMesh( Mesh_t *pMesh, studiohdr_t *pStudioHeader, CUtlVector<mstudioiface_t> &srcFaces,
mstudiomodel_t *pStudioModel, mstudiomesh_t *pStudioMesh, bool bForceNoFlex,
bool bForceSoftwareSkin, bool bHWFlex, bool bQuadSubd )
{
// Compute the mesh flags
ComputeMeshFlags( pMesh, pStudioHeader, pStudioMesh );
// We're gonna keep track of which ones we haven't processed
// because we're gonna add all unprocessed faces to the software
// lists if for some reason they don't get added to the hardware lists
FaceProcessedList_t facesProcessed;
facesProcessed.AddMultipleToTail( srcFaces.Count() );
memset( facesProcessed.Base(), 0, facesProcessed.Count() );
// there are up to 4 stripgroups per mesh
// Note that we're gonna do the HW skinned versions first
// because they have the option of deciding not to be hardware after all
int isHWSkinned, isFlexed;
for( isHWSkinned = bForceSoftwareSkin ? 0 : 1; isHWSkinned >= 0; --isHWSkinned )
{
for( isFlexed = 1; isFlexed >= 0; --isFlexed )
{
int realMaxBonesPerFace, realMaxBonesPerVert, realMaxBonesPerStrip;
if( isFlexed && !bHWFlex )
{
realMaxBonesPerFace = 1;
realMaxBonesPerVert = 1;
realMaxBonesPerStrip = 1;
}
else
{
realMaxBonesPerFace = m_MaxBonesPerFace;
realMaxBonesPerVert = m_MaxBonesPerVert;
realMaxBonesPerStrip = m_MaxBonesPerStrip;
}
int newStripGroupIndex = pMesh->stripGroups.AddToTail( );
StripGroup_t& newStripGroup = pMesh->stripGroups[newStripGroupIndex];
ProcessStripGroup( &newStripGroup,
isHWSkinned ? true : false,
isFlexed ? true : false,
pStudioModel, pStudioMesh, srcFaces, facesProcessed,
realMaxBonesPerVert, realMaxBonesPerFace,
realMaxBonesPerStrip, bForceNoFlex, bHWFlex, bQuadSubd );
PostProcessStripGroup( pStudioModel, pStudioMesh, &newStripGroup );
// Clear out the strip group if there wasn't anything in it
if( !newStripGroup.indices.Count() )
pMesh->stripGroups.FastRemove( newStripGroupIndex );
}
}
}
//-----------------------------------------------------------------------------
// some setup required before we really get into it
//-----------------------------------------------------------------------------
void COptimizedModel::SetupMeshProcessing( studiohdr_t *pHdr, int vertexCacheSize,
bool usesFixedFunction, int maxBonesPerVert, int maxBonesPerFace,
int maxBonesPerStrip, const char *fileName )
{
#ifdef NVTRISTRIP
// Tell nvtristrip all of its params
SetCacheSize( vertexCacheSize );
SetStitchStrips( true );
SetMinStripSize( 0 );
SetListsOnly( true );
#endif // NVTRISTRIP
if( !g_quiet )
{
printf( "---------------------\n" );
printf( "Generating optimized mesh \"%s\":\n", fileName );
#ifdef _DEBUG
printf( "\tvertex cache size: %d\n", vertexCacheSize );
printf( "\tmax bones/tri: %d\n", maxBonesPerFace );
printf( "\tmax bones/vert: %d\n", maxBonesPerVert );
printf( "\tmax bones/strip: %d\n", maxBonesPerStrip );
#endif
}
CleanupEverything();
// Total number of bones in the original model
m_NumBones = pHdr->numbones;
// Hardware limitations
m_MaxBonesPerVert = maxBonesPerVert;
m_MaxBonesPerFace = maxBonesPerFace;
m_MaxBonesPerStrip = maxBonesPerStrip;
m_bUsesFixedFunction = usesFixedFunction;
m_VertexCacheSize = vertexCacheSize;
// stats
m_NumSkinnedAndFlexedVerts = 0;
}
//-----------------------------------------------------------------------------
// Process the entire model, return stats...
//-----------------------------------------------------------------------------
int COptimizedModel::GetTotalVertsForMesh( Mesh_t *pMesh )
{
int numVerts = 0;
for( int i = 0; i < pMesh->stripGroups.Count(); i++ )
{
StripGroup_t *pStripGroup = &pMesh->stripGroups[i];
numVerts += pStripGroup->verts.Count();
}
return numVerts;
}
int COptimizedModel::GetTotalIndicesForMesh( Mesh_t *pMesh )
{
int numIndices = 0;
for( int i = 0; i < pMesh->stripGroups.Count(); i++ )
{
StripGroup_t *pStripGroup = &pMesh->stripGroups[i];
numIndices += pStripGroup->indices.Count();
}
return numIndices;
}
int COptimizedModel::GetTotalTopologyIndicesForMesh( Mesh_t *pMesh )
{
int numTopologyIndices = 0;
for( int i = 0; i < pMesh->stripGroups.Count(); i++ )
{
StripGroup_t *pStripGroup = &pMesh->stripGroups[i];
numTopologyIndices += pStripGroup->topologyIndices.Count();
}
return numTopologyIndices;
}
int COptimizedModel::GetTotalStripsForMesh( Mesh_t *pMesh )
{
int numStrips = 0;
for( int i = 0; i < pMesh->stripGroups.Count(); i++ )
{
StripGroup_t *pStripGroup = &pMesh->stripGroups[i];
numStrips += pStripGroup->strips.Count();
}
return numStrips;
}
int COptimizedModel::GetTotalStripGroupsForMesh( Mesh_t *pMesh )
{
return pMesh->stripGroups.Count();
}
int COptimizedModel::GetTotalBoneStateChangesForMesh( Mesh_t *pMesh )
{
int numBoneStateChanges = 0;
for( int i = 0; i < pMesh->stripGroups.Count(); i++ )
{
StripGroup_t *pStripGroup = &pMesh->stripGroups[i];
int j;
for( j = 0; j < pStripGroup->strips.Count(); j++ )
{
Strip_t *pStrip = &pStripGroup->strips[j];
numBoneStateChanges += pStrip->numBoneStateChanges;
}
}
return numBoneStateChanges;
}
/*
static void WriteDebugFile( const char *fileName, const char *outFileName, float red, float grn, float blu )
{
char *tmpName = ( char * )_alloca( strlen( fileName ) + 1 );
strcpy( tmpName, fileName );
s_source_t *pSrc = Load_Source( tmpName, "SMD" );
Assert( pSrc );
int i, j;
FILE *fp;
fp = fopen( outFileName, "w" );
Assert( fp );
for( i = 0; i < pSrc->nummeshes; i++ )
{
s_mesh_t *pMesh = &pSrc->mesh[i];
for( j = 0; j < pMesh->numfaces; j++ )
{
s_face_t *pFace = &pSrc->face[pMesh->faceoffset + j];
Vector &a = pSrc->vertex[pMesh->vertexoffset + pFace->a];
Vector &b = pSrc->vertex[pMesh->vertexoffset + pFace->b];
Vector &c = pSrc->vertex[pMesh->vertexoffset + pFace->c];
fprintf( fp, "3\n" );
fprintf( fp, "%f %f %f %f %f %f\n", ( float )( a[0] ), ( float )( a[1] ), ( float )( a[2] ),
( float )red, ( float )grn, ( float )blu );
fprintf( fp, "%f %f %f %f %f %f\n", ( float )( b[0] ), ( float )( b[1] ), ( float )( b[2] ),
( float )red, ( float )grn, ( float )blu );
fprintf( fp, "%f %f %f %f %f %f\n", ( float )( c[0] ), ( float )( c[1] ), ( float )( c[2] ),
( float )red, ( float )grn, ( float )blu );
}
}
fclose( fp );
}
*/
void COptimizedModel::SourceMeshToFaceList( s_model_t *pSrcModel, s_mesh_t *pSrcMesh, CUtlVector<mstudioiface_t> &meshFaceList )
{
s_face_t *pFaces = pSrcModel->source->face + pSrcMesh->faceoffset;
for ( int i = 0; i < pSrcMesh->numfaces; i++ )
{
const s_face_t &pFace = pFaces[i];
int j = meshFaceList.AddToTail();
mstudioiface_t &newFace = meshFaceList[j];
newFace.a = pFace.a;
newFace.b = pFace.b;
newFace.c = pFace.c;
newFace.d = pFace.d;
}
}
/*
static void WriteSourceMesh( s_model_t *pSrcModel, s_mesh_t *pSrcMesh, int red_, int grn_, int blu_ )
{
FILE *fp;
fp = fopen( "blah.glv", "a+" );
float red, grn, blu;
red = ( float )rand() / ( float )VALVE_RAND_MAX;
grn = ( float )rand() / ( float )VALVE_RAND_MAX;
blu = ( float )rand() / ( float )VALVE_RAND_MAX;
float len = red * red + grn * grn + blu * blu;
len = sqrt( len );
red *= 255.0f / len;
grn *= 255.0f / len;
blu *= 255.0f / len;
s_face_t *pFaces = pSrcModel->source->face + pSrcMesh->faceoffset;
int i;
for( i = 0; i < pSrcMesh->numfaces; i++ )
{
const s_face_t &face = pFaces[i];
Vector a, b, c;
a = pSrcModel->source->vertex[pSrcMesh->vertexoffset + face.a];
b = pSrcModel->source->vertex[pSrcMesh->vertexoffset + face.b];
c = pSrcModel->source->vertex[pSrcMesh->vertexoffset + face.c];
fprintf( fp, "3\n" );
fprintf( fp, "%f %f %f %f %f %f\n", a[0], a[1], a[2], red, grn, blu );
fprintf( fp, "%f %f %f %f %f %f\n", c[0], c[1], c[2], red, grn, blu );
fprintf( fp, "%f %f %f %f %f %f\n", b[0], b[1], b[2], red, grn, blu );
}
fclose( fp );
}
*/
static void RandomColor( Vector& color )
{
color[0] = ( ( float )rand() ) / ( float )VALVE_RAND_MAX;
color[1] = ( ( float )rand() ) / ( float )VALVE_RAND_MAX;
color[2] = ( ( float )rand() ) / ( float )VALVE_RAND_MAX;
VectorNormalize( color );
}
/*
static void GLViewCube( Vector pos, float size, FILE *fp )
{
fprintf( fp, "4\n" );
fprintf( fp, "%f %f %f 255 0 0\n", pos[0] + size, pos[1] + size, pos[2] + size );
fprintf( fp, "%f %f %f 255 0 0\n", pos[0] - size, pos[1] + size, pos[2] + size );
fprintf( fp, "%f %f %f 255 0 0\n", pos[0] + size, pos[1] - size, pos[2] + size );
fprintf( fp, "%f %f %f 255 0 0\n", pos[0] - size, pos[1] - size, pos[2] + size );
fprintf( fp, "4\n" );
fprintf( fp, "%f %f %f 255 0 0\n", pos[0] + size, pos[1] + size, pos[2] - size );
fprintf( fp, "%f %f %f 255 0 0\n", pos[0] - size, pos[1] + size, pos[2] - size );
fprintf( fp, "%f %f %f 255 0 0\n", pos[0] + size, pos[1] - size, pos[2] - size );
fprintf( fp, "%f %f %f 255 0 0\n", pos[0] - size, pos[1] - size, pos[2] - size );
fprintf( fp, "4\n" );
fprintf( fp, "%f %f %f 255 0 0\n", pos[0] + size, pos[1] - size, pos[2] - size );
fprintf( fp, "%f %f %f 255 0 0\n", pos[0] - size, pos[1] - size, pos[2] - size );
fprintf( fp, "%f %f %f 255 0 0\n", pos[0] + size, pos[1] - size, pos[2] + size );
fprintf( fp, "%f %f %f 255 0 0\n", pos[0] - size, pos[1] - size, pos[2] + size );
fprintf( fp, "4\n" );
fprintf( fp, "%f %f %f 255 0 0\n", pos[0] + size, pos[1] + size, pos[2] - size );
fprintf( fp, "%f %f %f 255 0 0\n", pos[0] - size, pos[1] + size, pos[2] - size );
fprintf( fp, "%f %f %f 255 0 0\n", pos[0] + size, pos[1] + size, pos[2] + size );
fprintf( fp, "%f %f %f 255 0 0\n", pos[0] - size, pos[1] + size, pos[2] + size );
fprintf( fp, "4\n" );
fprintf( fp, "%f %f %f 255 0 0\n", pos[0] - size, pos[1] + size, pos[2] - size );
fprintf( fp, "%f %f %f 255 0 0\n", pos[0] - size, pos[1] - size, pos[2] - size );
fprintf( fp, "%f %f %f 255 0 0\n", pos[0] - size, pos[1] + size, pos[2] + size );
fprintf( fp, "%f %f %f 255 0 0\n", pos[0] - size, pos[1] - size, pos[2] + size );
fprintf( fp, "4\n" );
fprintf( fp, "%f %f %f 255 0 0\n", pos[0] + size, pos[1] + size, pos[2] - size );
fprintf( fp, "%f %f %f 255 0 0\n", pos[0] + size, pos[1] - size, pos[2] - size );
fprintf( fp, "%f %f %f 255 0 0\n", pos[0] + size, pos[1] + size, pos[2] + size );
fprintf( fp, "%f %f %f 255 0 0\n", pos[0] + size, pos[1] - size, pos[2] + size );
}
*/
#if 0
static void MStudioBoneWeightToSBoneWeight( s_boneweight_t &sbone, const mstudioboneweight_t &mbone,
const s_source_t *pSrc )
{
int i;
for( i = 0; i < mbone.numbones; i++ )
{
sbone.bone[i] = pSrc->boneimap[mbone.bone[i]];
// sbone.bone[i] = mbone.bone[i];
sbone.weight[i] = mbone.weight[i];
}
sbone.numbones = mbone.numbones;
}
#endif
void COptimizedModel::CreateLODFaceList( s_model_t *pSrcModel, int nLodID, s_source_t* pSrc,
mstudiomodel_t *pStudioModel, mstudiomesh_t *pStudioMesh,
CUtlVector<mstudioiface_t> &meshFaceList,
bool bQuadSubd, bool writeDebug )
{
if ( !pSrc || !pSrcModel )
return;
#ifdef _DEBUG
// const mstudio_modelvertexdata_t *vertData = pStudioModel->GetVertexData();
// Assert( vertData ); // This can only return NULL on X360 for now
// mstudiovertex_t *modelFirstVert = vertData->Vertex( 0 );
// mstudiovertex_t *modelLastVert = vertData->Vertex( pStudioModel->numvertices - 1 );
// mstudiovertex_t *meshFirstVert = vertData->Vertex( 0 );
// mstudiovertex_t *meshLastVert = vertData->Vertex( pStudioMesh->numvertices - 1 );
#endif
/*
if( writeDebug )
{
printf( "MODEL VERTS:\n" );
int i;
for( i = 0; i < pStudioModel->numvertices; i++ )
{
Vector &v = *pStudioModel->pVertex( i );
Vector &n = *pStudioModel->pNormal( i );
Vector2D &t = *pStudioModel->pTexcoord( i );
printf( "model %d: p %f %f %f n: %f %f %f t: %f %f\n",
i, v[0], v[1], v[2], n[0], n[1], n[2], t[0], t[1] );
}
printf( "MESH VERTS:\n" );
for( i = 0; i < pStudioMesh->numvertices; i++ )
{
Vector &v = *pStudioMesh->pVertex( i );
Vector &n = *pStudioMesh->pNormal( i );
Vector2D &t = *pStudioMesh->pTexcoord( i );
printf( "mesh %d: p %f %f %f n: %f %f %f t: %f %f\n",
i, v[0], v[1], v[2], n[0], n[1], n[2], t[0], t[1] );
}
}
*/
// need to find the mesh in the lod model that matches the original model.
int i;
int textureSearchID = MaterialToTexture( pStudioMesh->material );
// this is icky.. . pSrc->nummeshes really refers to the total number of non-empty meshes
// in the model. In pSrc->meshes, there are some empty meshes in the middle if they
// don't exist for the material, so you have to go through all of the meshes to find
// the non-empty ones.
s_mesh_t *pSrcMesh = NULL;
for ( i = 0; i < pStudioModel->nummeshes; i++ )
{
if ( pSrc->texmap[pSrc->meshindex[i]] == textureSearchID )
{
// woo hoo! found it.
pSrcMesh = &pSrc->mesh[pSrc->meshindex[i]];
break;
}
}
if ( !pSrcMesh )
{
//printf( "%s doesn't have material %d\n", lodSMDName, textureSearchID );
// There aren't any faces in this lower lod with this material on it.
return;
}
CUtlVector<int> indexMapping;
indexMapping.AddMultipleToTail( pSrcMesh->numvertices );
for ( i = 0; i < pSrcMesh->numvertices; i++ )
{
// get the mapping between indices in the lod and their real pool location.
indexMapping[i] = pSrcModel->m_pLodData->pMeshVertIndexMaps[nLodID][pSrcMesh->vertexoffset + i];
}
// build the lod's faces so indexes map to remapped vertexes
for ( i = 0; i < pSrcMesh->numfaces; i++ )
{
int index = meshFaceList.AddToTail();
mstudioiface_t& newFace = meshFaceList[index];
const s_face_t &srcFace = pSrc->face[pSrcMesh->faceoffset + i];
newFace.a = indexMapping[srcFace.a];
newFace.b = indexMapping[srcFace.b];
newFace.c = indexMapping[srcFace.c];
newFace.d = bQuadSubd ? indexMapping[srcFace.d] : 0xFFFF;
}
}
bool ComparePath( const char *a, const char *b )
{
if ( strlen( a ) != strlen( b ) )
{
return false;
}
// Case and separator invariant
for ( ; *a; a++, b++ )
{
if ( *a == *b )
{
continue;
}
if ( tolower( *a ) == tolower( *b ) )
{
continue;
}
if ( ( *a == '/' || *a == '\\' ) &&
( *b == '/' || *b == '\\' ) )
{
continue;
}
return false;
}
return true;
}
bool COptimizedModel::MeshNeedsRemoval( studiohdr_t *pHdr, mstudiomesh_t *pStudioMesh,
LodScriptData_t& scriptLOD )
{
mstudiotexture_t *ptexture;
short *pskinref = pHdr->pSkinref( 0 );
ptexture = pHdr->pTexture( pskinref[ pStudioMesh->material ] );
const char *meshName = ptexture->material->GetName();
int i;
for( i = 0; i < scriptLOD.meshRemovals.Count(); i++ )
{
const char *meshRemovalName = scriptLOD.meshRemovals[i].GetSrcName();
if ( ComparePath( meshName, meshRemovalName ) )
{
return true;
}
}
return false;
}
//-----------------------------------------------------------------------------
// Process the entire model, return stats...
//-----------------------------------------------------------------------------
void COptimizedModel::ProcessModel( studiohdr_t *pHdr, s_bodypart_t *pSrcBodyParts,
TotalMeshStats_t& stats, bool bForceSoftwareSkin, bool bHWFlex )
{
memset( &stats, 0, sizeof(stats) );
m_Models.RemoveAll();
int bodyPartID, modelID, meshID, lodID;
for ( bodyPartID = 0; bodyPartID < pHdr->numbodyparts; bodyPartID++, stats.m_TotalBodyParts++ )
{
mstudiobodyparts_t *pBodyPart = pHdr->pBodypart( bodyPartID );
s_bodypart_t *pSrcBodyPart = &pSrcBodyParts[bodyPartID];
for ( modelID = 0; modelID < pBodyPart->nummodels; modelID++, stats.m_TotalModels++ )
{
int i = m_Models.AddToTail();
Model_t& newModel = m_Models[i];
mstudiomodel_t *pStudioModel = pBodyPart->pModel( modelID );
s_model_t *pSrcModel = pSrcBodyPart->pmodel[modelID];
for ( lodID = 0; lodID < g_ScriptLODs.Count(); lodID++, stats.m_TotalModelLODs++ )
{
LodScriptData_t& scriptLOD = g_ScriptLODs[lodID];
s_source_t *pLODSource = pSrcModel->m_LodSources[lodID];
int i = newModel.modelLODs.AddToTail();
Assert( i == lodID );
ModelLOD_t& newLOD = newModel.modelLODs[i];
newLOD.switchPoint = scriptLOD.switchValue;
// In this case, we've been told to remove the model
if ( !pLODSource )
{
if ( pSrcModel && stricmp( pSrcModel->name, "blank" ) != 0)
{
// This is nonsensical
Assert( lodID != 0 );
}
continue;
}
for ( meshID = 0; meshID < pStudioModel->nummeshes; meshID++, stats.m_TotalMeshes++ )
{
#ifdef _DEBUG
// printf( "bodyPart: %d model: %d modellod: %d mesh: %d\n", bodyPartID, modelID, lodID, meshID );
#endif
mstudiomesh_t *pStudioMesh = pStudioModel->pMesh( meshID );
s_mesh_t *pSrcMesh = &pSrcModel->source->mesh[pSrcModel->source->meshindex[meshID]];
int i = newLOD.meshes.AddToTail();
Assert( i == meshID );
Mesh_t& newMesh = newLOD.meshes[i];
if ( MeshNeedsRemoval( pHdr, pStudioMesh, scriptLOD ) )
continue;
#ifdef _DEBUG
// int textureSearchID = material_to_texture( pStudioMesh->material );
// const char *pDebugName = pHdr->pTexture( textureSearchID )->pszName( );
#endif
// Process Quad-only meshes in software
bool bQuadSubd = ( gflags & STUDIOHDR_FLAGS_SUBDIVISION_SURFACE ) != 0;
bForceSoftwareSkin = bQuadSubd || bForceSoftwareSkin;
// Only one of these will actually get used, depending on topology type (tris or quads)
CUtlVector<mstudioiface_t> meshFaceList;
if ( pLODSource )
{
// map the lod data to faces
// uses the original mesh redirected through a mapping table
// this expects built per lod-to-root mapping tables to generate faces
CreateLODFaceList( pSrcModel, lodID, pLODSource, pStudioModel, pStudioMesh, meshFaceList, bQuadSubd, false );
}
else
{
// build the face list from the unmapped source
SourceMeshToFaceList( pSrcModel, pSrcMesh, meshFaceList );
}
ProcessMesh( &newMesh, pHdr, meshFaceList, pStudioModel, pStudioMesh,
!scriptLOD.GetFacialAnimationEnabled(), bForceSoftwareSkin, bHWFlex, bQuadSubd );
stats.m_TotalVerts += GetTotalVertsForMesh( &newMesh );
stats.m_TotalIndices += GetTotalIndicesForMesh( &newMesh );
stats.m_TotalTopologyIndices += GetTotalTopologyIndicesForMesh( &newMesh );
stats.m_TotalStrips += GetTotalStripsForMesh( &newMesh );
stats.m_TotalStripGroups += GetTotalStripGroupsForMesh( &newMesh );
stats.m_TotalBoneStateChanges += GetTotalBoneStateChangesForMesh( &newMesh );
}
}
}
}
}
//-----------------------------------------------------------------------------
// Some processing that happens at the end
//-----------------------------------------------------------------------------
void COptimizedModel::MapGlobalBonesToHardwareBoneIDsAndSortBones( studiohdr_t *phdr )
{
// garymcthack: holy jesus is this function long!
#ifdef IGNORE_BONES
return;
#endif
int *globalToHardwareBoneIndex;
int hardwareToGlobalBoneIndex[MAX_NUM_BONES_PER_STRIP];
globalToHardwareBoneIndex = ( int * )_alloca( m_NumBones * sizeof( int ) );
Assert( globalToHardwareBoneIndex );
FileHeader_t *header = ( FileHeader_t * )m_FileBuffer->GetPointer( 0 );
for( int bodyPartID = 0; bodyPartID < header->numBodyParts; bodyPartID++ )
{
BodyPartHeader_t *bodyPart = header->pBodyPart( bodyPartID );
mstudiobodyparts_t *pStudioBodyPart = phdr->pBodypart( bodyPartID );
for( int lodID = 0; lodID < header->numLODs; lodID++ )
{
// Dear LORD! This needs to be in another function! (garymcthack)
int i;
for( i = 0; i < m_NumBones; i++ )
{
globalToHardwareBoneIndex[i] = -1;
}
for( i = 0; i < MAX_NUM_BONES_PER_STRIP; i++ )
{
hardwareToGlobalBoneIndex[i] = -1;
}
for( int modelID = 0; modelID < bodyPart->numModels; modelID++ )
{
ModelHeader_t *model = bodyPart->pModel( modelID );
mstudiomodel_t *pStudioModel = pStudioBodyPart->pModel( modelID );
ModelLODHeader_t *pLOD = model->pLOD( lodID );
for( int meshID = 0; meshID < pLOD->numMeshes; meshID++ )
{
MeshHeader_t *mesh = pLOD->pMesh( meshID );
mstudiomesh_t *pStudioMesh = pStudioModel->pMesh( meshID );
for( int stripGroupID = 0; stripGroupID < mesh->numStripGroups; stripGroupID++ )
{
// Only do this to HW-skinned meshes
StripGroupHeader_t *pStripGroup = mesh->pStripGroup( stripGroupID );
if( !( pStripGroup->flags & STRIPGROUP_IS_HWSKINNED ) )
{
continue;
}
for( int stripID = 0; stripID < pStripGroup->numStrips; stripID++ )
{
// printf( "UPDATING BONE STATE\n" );
StripHeader_t *pStrip = pStripGroup->pStrip( stripID );
// Generate mapping back and forth between hardware IDs and original bone IDs
int boneStateChangeID;
for( boneStateChangeID = 0; boneStateChangeID < pStrip->numBoneStateChanges; boneStateChangeID++ )
{
BoneStateChangeHeader_t *pBoneStateChange = pStrip->pBoneStateChange( boneStateChangeID );
globalToHardwareBoneIndex[pBoneStateChange->newBoneID] = pBoneStateChange->hardwareID;
if( pBoneStateChange->hardwareID != -1 )
{
hardwareToGlobalBoneIndex[pBoneStateChange->hardwareID] = pBoneStateChange->newBoneID;
}
}
int vertID;
for( vertID = 0; vertID < pStrip->numVerts; vertID++ )
{
Vertex_t *vert = pStripGroup->pVertex( vertID + pStrip->vertOffset );
int boneID;
for( boneID = 0; boneID < header->maxBonesPerVert; boneID++ )
{
int globalBoneID = vert->boneID[boneID];
if( globalBoneID == 255 )
{
// this index isn't used.
vert->boneID[boneID] = 0;
// make sure it's associated weight is zero.
// Assert( vert->boneWeights[boneID] == 0 );
continue;
}
Assert( globalBoneID >= 0 && globalBoneID < m_NumBones );
vert->boneID[boneID] = globalToHardwareBoneIndex[globalBoneID];
Assert( vert->boneID[boneID] >= 0 && vert->boneID[boneID] < header->maxBonesPerStrip );
}
// We only do index palette skinning when we're not doing fixed function
if (m_bUsesFixedFunction)
{
// bool flexed = pStripGroup->flags & STRIPGROUP_IS_FLEXED;
SortBonesWithinVertex( false /* flexed */, vert, pStudioModel,
pStudioMesh, globalToHardwareBoneIndex, hardwareToGlobalBoneIndex,
m_MaxBonesPerFace, m_MaxBonesPerVert );
}
}
}
}
}
}
}
}
}
//-----------------------------------------------------------------------------
//
// The following methods are associated with writing a .vtx file
//
//-----------------------------------------------------------------------------
void COptimizedModel::WriteHeader( int vertCacheSize, int maxBonesPerVert,
int maxBonesPerFace, int maxBonesPerStrip, int numBodyParts, long checkSum )
{
FileHeader_t fileHeader;
fileHeader.version = OPTIMIZED_MODEL_FILE_VERSION;
fileHeader.vertCacheSize = vertCacheSize;
fileHeader.maxBonesPerFace = IsUShort( maxBonesPerFace );
fileHeader.maxBonesPerVert = maxBonesPerVert;
fileHeader.maxBonesPerStrip = IsUShort( maxBonesPerStrip );
fileHeader.numBodyParts = numBodyParts;
fileHeader.bodyPartOffset = sizeof( FileHeader_t );
fileHeader.checkSum = checkSum;
fileHeader.numLODs = g_ScriptLODs.Count();
fileHeader.materialReplacementListOffset = m_MaterialReplacementsListOffset;
m_FileBuffer->WriteAt( 0, &fileHeader, sizeof( FileHeader_t ), "header" );
#ifdef _DEBUG
// FileHeader_t *debug = ( FileHeader_t * )m_FileBuffer->GetPointer( 0 );
// BodyPartHeader_t *pBodyPart = debug->pBodyPart( 0 );
#endif
}
void COptimizedModel::WriteBodyPart( int bodyPartID, mstudiobodyparts_t *pBodyPart, int modelID )
{
BodyPartHeader_t bodyPart;
bodyPart.numModels = pBodyPart->nummodels;
int bodyPartOffset = m_BodyPartsOffset + bodyPartID * sizeof( BodyPartHeader_t );
int modelFileOffset = m_ModelsOffset + modelID * sizeof( ModelHeader_t );
bodyPart.modelOffset = modelFileOffset - bodyPartOffset;
m_FileBuffer->WriteAt( bodyPartOffset, &bodyPart, sizeof( BodyPartHeader_t ), "bodypart" );
#ifdef _DEBUG
// BodyPartHeader_t *debug = ( BodyPartHeader_t * )m_FileBuffer->GetPointer( bodyPartOffset );
// ModelHeader_t *pModel = debug->pModel( 0 );
#endif
}
void COptimizedModel::WriteModel( int modelID, mstudiomodel_t *pModel, int lodID )
{
ModelHeader_t model;
model.numLODs = IsChar( g_ScriptLODs.Count() );
int modelFileOffset = m_ModelsOffset + modelID * sizeof( ModelHeader_t );
int lodFileOffset = m_ModelLODsOffset + lodID * sizeof( ModelLODHeader_t );
model.lodOffset = lodFileOffset - modelFileOffset;
m_FileBuffer->WriteAt( modelFileOffset, &model, sizeof( ModelHeader_t ), "model" );
#ifdef _DEBUG
// ModelHeader_t *debug = ( ModelHeader_t * )m_FileBuffer->GetPointer( modelFileOffset );
// ModelLODHeader_t *pLOD = debug->pLOD( 0 );
#endif
}
void COptimizedModel::WriteModelLOD( int lodID, ModelLOD_t *pLOD, int meshID )
{
ModelLODHeader_t lod;
int lodFileOffset = m_ModelLODsOffset + lodID * sizeof( ModelLODHeader_t );
int meshFileOffset = m_MeshesOffset + meshID * sizeof( MeshHeader_t );
lod.meshOffset = meshFileOffset - lodFileOffset;
lod.numMeshes = pLOD->meshes.Count();
lod.switchPoint = pLOD->switchPoint;
m_FileBuffer->WriteAt( lodFileOffset, &lod, sizeof( ModelLODHeader_t ), "modellod" );
#ifdef _DEBUG
// ModelLODHeader_t *debug = ( ModelLODHeader_t * )m_FileBuffer->GetPointer( lodFileOffset );
// MeshHeader_t *pMesh = debug->pMesh( 0 );
#endif
}
void COptimizedModel::WriteMesh( int meshID, Mesh_t *pMesh, int stripGroupID )
{
MeshHeader_t mesh;
mesh.numStripGroups = IsChar( pMesh->stripGroups.Count() );
int meshFileOffset = m_MeshesOffset + meshID * sizeof( MeshHeader_t );
int stripGroupFileOffset = m_StripGroupsOffset + stripGroupID * sizeof( StripGroupHeader_t );
mesh.stripGroupHeaderOffset = stripGroupFileOffset - meshFileOffset;
mesh.flags = pMesh->flags;
m_FileBuffer->WriteAt( meshFileOffset, &mesh, sizeof( MeshHeader_t ), "mesh" );
#ifdef _DEBUG
// MeshHeader_t *debug = ( MeshHeader_t * )m_FileBuffer->GetPointer( meshFileOffset );
// StripGroupHeader_t *pStripGroup = debug->pStripGroup( 0 );
#endif
}
void COptimizedModel::WriteStripGroup( int stripGroupID, StripGroup_t *pStripGroup,
int vertID, int indexID, int topologyID, int stripID )
{
StripGroupHeader_t stripGroup;
stripGroup.numVerts = pStripGroup->verts.Count();
stripGroup.numIndices = pStripGroup->indices.Count();
stripGroup.numTopologyIndices = pStripGroup->topologyIndices.Count();
stripGroup.numStrips = pStripGroup->strips.Count();
stripGroup.flags = IsByte( pStripGroup->flags );
int stripGroupFileOffset = m_StripGroupsOffset + stripGroupID * sizeof( StripGroupHeader_t );
int vertsFileOffset = m_VertsOffset + vertID * sizeof( Vertex_t );
int indicesFileOffset = m_IndicesOffset + indexID * sizeof( unsigned short );
int topologyFileOffset = m_TopologyOffset + topologyID * sizeof( unsigned short );
int stripsFileOffset = m_StripsOffset + stripID * sizeof( StripHeader_t );
stripGroup.vertOffset = vertsFileOffset - stripGroupFileOffset;
stripGroup.indexOffset = indicesFileOffset - stripGroupFileOffset;
stripGroup.topologyOffset = topologyFileOffset - stripGroupFileOffset;
stripGroup.stripOffset = stripsFileOffset - stripGroupFileOffset;
m_FileBuffer->WriteAt( stripGroupFileOffset, &stripGroup, sizeof( StripGroupHeader_t ), "strip group" );
#ifdef _DEBUG
// StripGroupHeader_t *debug = ( StripGroupHeader_t * )m_FileBuffer->GetPointer( stripGroupFileOffset );
// unsigned short *pIndex = debug->pIndex( 0 );
// Vertex_t *pVert = debug->pVertex( 0 );
// StripHeader_t *pStripHeader = debug->pStrip( 0 );
#endif
}
int COptimizedModel::WriteVerts( int vertID, StripGroup_t *pStripGroup )
{
int vertFileOffset = m_VertsOffset + vertID * sizeof( Vertex_t );
int numVerts = pStripGroup->verts.Count();
m_FileBuffer->WriteAt( vertFileOffset, &pStripGroup->verts[0], sizeof( Vertex_t ) * numVerts, "verts" );
#ifdef _DEBUG
// Vertex_t *debug = ( Vertex_t * )m_FileBuffer->GetPointer( vertFileOffset );
#endif
return numVerts;
}
int COptimizedModel::WriteIndices( int indexID, StripGroup_t *pStripGroup )
{
int indexFileOffset = m_IndicesOffset + indexID * sizeof( unsigned short );
int numIndices = pStripGroup->indices.Count();
m_FileBuffer->WriteAt( indexFileOffset, &pStripGroup->indices[0], sizeof( unsigned short ) * numIndices, "indices" );
#ifdef _DEBUG
// unsigned short *debug = ( unsigned short * )m_FileBuffer->GetPointer( indexFileOffset );
#endif
return numIndices;
}
int COptimizedModel::WriteTopology( int topologyID, StripGroup_t *pStripGroup )
{
// If we didnt' generate topology data, there won't be any topology indices
if ( pStripGroup->topologyIndices.Count() == 0 )
return 0;
int topologyIndexFileOffset = m_TopologyOffset + topologyID * sizeof( unsigned short );
int numTopologyIndices = pStripGroup->topologyIndices.Count();
m_FileBuffer->WriteAt( topologyIndexFileOffset, &pStripGroup->topologyIndices[0], sizeof( unsigned short ) * numTopologyIndices, "topologyIndices" );
#ifdef _DEBUG
// unsigned short *debug = ( unsigned short * )m_FileBuffer->GetPointer( topologyIndexFileOffset );
#endif
return numTopologyIndices;
}
void COptimizedModel::WriteStrip( int stripID, Strip_t *pStrip, int indexID, int curTopology, int vertID, int boneID )
{
StripHeader_t stripHeader;
stripHeader.numIndices = pStrip->numStripGroupIndices;
stripHeader.numTopologyIndices = pStrip->numStripGroupTopologyIndices;
stripHeader.indexOffset = pStrip->stripGroupIndexOffset;
stripHeader.topologyOffset = pStrip->stripGroupTopologyOffset;
stripHeader.numVerts = pStrip->numStripGroupVerts;
stripHeader.vertOffset = pStrip->stripGroupVertexOffset;
stripHeader.numBoneStateChanges = pStrip->numBoneStateChanges;
stripHeader.numBones = IsShort( pStrip->numBones );
stripHeader.flags = IsByte( pStrip->flags );
int boneFileOffset = m_BoneStateChangesOffset + boneID * sizeof( BoneStateChangeHeader_t );
int stripFileOffset = m_StripsOffset + stripID * sizeof( StripHeader_t );
stripHeader.boneStateChangeOffset = IsInt24( boneFileOffset - stripFileOffset );
m_FileBuffer->WriteAt( stripFileOffset, &stripHeader, sizeof( StripHeader_t ), "strip" );
#ifdef _DEBUG
// StripHeader_t *debug = ( StripHeader_t* )m_FileBuffer->GetPointer( stripFileOffset );
// BoneStateChangeHeader_t *pBoneStateChange = debug->pBoneStateChange( 0 );
#endif
}
void COptimizedModel::WriteBoneStateChange( int boneID, BoneStateChange_t *boneStateChange )
{
BoneStateChangeHeader_t boneHeader;
boneHeader.hardwareID = boneStateChange->hardwareID;
boneHeader.newBoneID = boneStateChange->newBoneID;
int boneFileOffset = m_BoneStateChangesOffset + boneID * sizeof( BoneStateChangeHeader_t );
#if 0
printf( "\tboneStateChange: hwid: %d boneID %d\n", ( int )boneHeader.hardwareID,
( int )boneHeader.newBoneID );
#endif
m_FileBuffer->WriteAt( boneFileOffset, &boneHeader, sizeof( BoneStateChangeHeader_t ), "bone" );
#ifdef _DEBUG
// BoneStateChangeHeader_t *debug = ( BoneStateChangeHeader_t * )m_FileBuffer->GetPointer( boneFileOffset );
#endif
}
void COptimizedModel::ZeroNumBones( void )
{
FileHeader_t *header = ( FileHeader_t * )m_FileBuffer->GetPointer( 0 );
int bodyPartID, modelID, lodID, meshID, stripGroupID, vertID, stripID;
for( bodyPartID = 0; bodyPartID < header->numBodyParts; bodyPartID++ )
{
BodyPartHeader_t *pBodyPart = header->pBodyPart( bodyPartID );
for( modelID = 0; modelID < pBodyPart->numModels; modelID++ )
{
ModelHeader_t *pModel = pBodyPart->pModel( modelID );
for( lodID = 0; lodID < pModel->numLODs; lodID++ )
{
ModelLODHeader_t *pLOD = pModel->pLOD( lodID );
for( meshID = 0; meshID < pLOD->numMeshes; meshID++ )
{
MeshHeader_t *pMesh = pLOD->pMesh( meshID );
for( stripGroupID = 0; stripGroupID < pMesh->numStripGroups; stripGroupID++ )
{
StripGroupHeader_t *pStripGroup = pMesh->pStripGroup( stripGroupID );
for( vertID = 0; vertID < pStripGroup->numVerts; vertID++ )
{
Vertex_t *pVert = pStripGroup->pVertex( vertID );
pVert->numBones = 0;
}
for( stripID = 0; stripID < pStripGroup->numStrips; stripID++ )
{
StripHeader_t *pStrip = pStripGroup->pStrip( stripID );
pStrip->numBones = 0;
pStrip->numBoneStateChanges = 0;
}
}
}
}
}
}
}
void COptimizedModel::WriteStringTable( int stringTableOffset )
{
int stringTableSize = s_StringTable.CalcSize();
if( stringTableSize == 0 )
{
return;
}
char *pTmp = new char[stringTableSize];
s_StringTable.WriteToMem( pTmp );
m_FileBuffer->WriteAt( stringTableOffset, pTmp, stringTableSize, "string table" );
// char *pDebug = ( char * )m_FileBuffer->GetPointer( stringTableOffset );
delete [] pTmp;
}
void COptimizedModel::WriteMaterialReplacements( int materialReplacementsOffset )
{
int offset = materialReplacementsOffset;
int i, j;
int numLODs = g_ScriptLODs.Count();
for( i = 0; i < numLODs; i++ )
{
LodScriptData_t &scriptLOD = g_ScriptLODs[i];
for( j = 0; j < scriptLOD.materialReplacements.Count(); j++ )
{
CLodScriptReplacement_t &materialReplacement = scriptLOD.materialReplacements[j];
MaterialReplacementHeader_t tmpHeader;
tmpHeader.materialID = FindMaterialByName( materialReplacement.GetSrcName() );
tmpHeader.replacementMaterialNameOffset = m_StringTableOffset +
s_StringTable.StringTableOffset( materialReplacement.GetDstName() ) - offset;
m_FileBuffer->WriteAt( offset, &tmpHeader, sizeof( tmpHeader ), "material replacements" );
offset += sizeof( MaterialReplacementHeader_t );
}
}
}
void COptimizedModel::WriteMaterialReplacementLists( int materialReplacementsOffset, int materialReplacementListOffset )
{
int replacementOffset = materialReplacementsOffset;
int offset = materialReplacementListOffset;
int i;
int numLODs = g_ScriptLODs.Count();
for( i = 0; i < numLODs; i++ )
{
LodScriptData_t &scriptLOD = g_ScriptLODs[i];
MaterialReplacementListHeader_t tmpHeader;
tmpHeader.numReplacements = IsChar( scriptLOD.materialReplacements.Count() );
tmpHeader.replacementOffset = IsInt24( replacementOffset - offset );
m_FileBuffer->WriteAt( offset, &tmpHeader, sizeof( tmpHeader ), "material replacement headers" );
MaterialReplacementListHeader_t *pDebugList =
( MaterialReplacementListHeader_t * )m_FileBuffer->GetPointer( offset );
if( pDebugList->numReplacements )
{
// MaterialReplacementHeader_t *pDebug = pDebugList->pMaterialReplacement( 0 );
// const char *string = pDebug->pMaterialReplacementName();
}
replacementOffset += tmpHeader.numReplacements * sizeof( MaterialReplacementHeader_t );
offset += sizeof( MaterialReplacementListHeader_t );
}
}
void COptimizedModel::SanityCheckVertexBoneLODFlags( studiohdr_t *pStudioHdr, FileHeader_t *pVtxHeader )
{
int bodyPartID;
for( bodyPartID = 0; bodyPartID < pStudioHdr->numbodyparts; bodyPartID++ )
{
mstudiobodyparts_t *pBodyPart = pStudioHdr->pBodypart( bodyPartID );
BodyPartHeader_t *pVtxBodyPart = pVtxHeader->pBodyPart( bodyPartID );
int modelID;
for( modelID = 0; modelID < pBodyPart->nummodels; modelID++ )
{
mstudiomodel_t *pModel = pBodyPart->pModel( modelID );
ModelHeader_t *pVtxModel = pVtxBodyPart->pModel( modelID );
int lodID;
for( lodID = 0; lodID < pVtxModel->numLODs; lodID++ )
{
ModelLODHeader_t *pVtxLOD = pVtxModel->pLOD( lodID );
int meshID;
Assert( pVtxLOD->numMeshes == pModel->nummeshes );
for( meshID = 0; meshID < pVtxLOD->numMeshes; meshID++ )
{
MeshHeader_t *pVtxMesh = pVtxLOD->pMesh( meshID );
mstudiomesh_t *pMesh = pModel->pMesh( meshID );
int stripGroupID;
for( stripGroupID = 0; stripGroupID < pVtxMesh->numStripGroups; stripGroupID++ )
{
StripGroupHeader_t *pStripGroup = pVtxMesh->pStripGroup( stripGroupID );
int vertID;
for( vertID = 0; vertID < pStripGroup->numVerts; vertID++ )
{
Vertex_t *pVertex = pStripGroup->pVertex( vertID );
Vector pos = GetOrigVertPosition( pModel, pMesh, pVertex );
const mstudioboneweight_t &boneWeight = GetOrigVertBoneWeight( pModel, pMesh, pVertex );
int i;
for( i = 0; i < boneWeight.numbones; i++ )
{
const mstudiobone_t *pBone = pStudioHdr->pBone( boneWeight.bone[i] );
// const char *pBoneName = pBone->pszName();
if (!( pBone->flags & ( BONE_USED_BY_VERTEX_LOD0 << lodID ) ))
{
MdlError("Mismarked Bone flag");
}
}
}
}
}
}
}
}
}
void COptimizedModel::WriteVTXFile( studiohdr_t *pHdr, const char *pFileName,
TotalMeshStats_t const& stats )
{
// calculate file offsets
m_FileBuffer = new CFileBuffer( FILEBUFFER_SIZE );
m_BodyPartsOffset = sizeof( FileHeader_t );
m_ModelsOffset = m_BodyPartsOffset + sizeof( BodyPartHeader_t ) * stats.m_TotalBodyParts;
m_ModelLODsOffset = m_ModelsOffset + sizeof( ModelHeader_t ) * stats.m_TotalModels;
m_MeshesOffset = m_ModelLODsOffset + sizeof( ModelLODHeader_t ) * stats.m_TotalModelLODs;
m_StripGroupsOffset = m_MeshesOffset + sizeof( MeshHeader_t ) * stats.m_TotalMeshes;
m_StripsOffset = m_StripGroupsOffset + sizeof( StripGroupHeader_t ) * stats.m_TotalStripGroups;
m_VertsOffset = m_StripsOffset + sizeof( StripHeader_t ) * stats.m_TotalStrips;
m_IndicesOffset = m_VertsOffset + sizeof( Vertex_t ) * stats.m_TotalVerts;
m_BoneStateChangesOffset = m_IndicesOffset + sizeof( unsigned short ) * stats.m_TotalIndices;
m_StringTableOffset = m_BoneStateChangesOffset + sizeof( BoneStateChangeHeader_t ) * stats.m_TotalBoneStateChanges;
m_MaterialReplacementsOffset = m_StringTableOffset + s_StringTable.CalcSize();
m_MaterialReplacementsListOffset = m_MaterialReplacementsOffset + stats.m_TotalMaterialReplacements * sizeof( MaterialReplacementHeader_t );
m_TopologyOffset = m_MaterialReplacementsListOffset + g_ScriptLODs.Count() * sizeof( MaterialReplacementListHeader_t );
m_EndOfFileOffset = m_TopologyOffset + sizeof( unsigned short ) * stats.m_TotalTopologyIndices;
int curModel = 0;
int curLOD = 0;
int curMesh = 0;
int curStrip = 0;
int curStripGroup = 0;
int curVert = 0;
int curIndex = 0;
int curTopology = 0;
int curBoneStateChange = 0;
int deltaModel = 0;
int deltaLOD = 0;
int deltaMesh = 0;
int deltaStrip = 0;
int deltaStripGroup = 0;
int deltaVert = 0;
int deltaIndex = 0;
int deltaTopology = 0;
int deltaBoneStateChange = 0;
WriteStringTable( m_StringTableOffset );
WriteMaterialReplacements( m_MaterialReplacementsOffset );
WriteMaterialReplacementLists( m_MaterialReplacementsOffset, m_MaterialReplacementsListOffset );
for( int bodyPartID = 0; bodyPartID < pHdr->numbodyparts; bodyPartID++ )
{
mstudiobodyparts_t *pBodyPart = pHdr->pBodypart( bodyPartID );
for( int modelID = 0; modelID < pBodyPart->nummodels; modelID++ )
{
mstudiomodel_t *pStudioModel = pBodyPart->pModel( modelID );
Model_t *pModel = &m_Models[curModel + deltaModel];
for( int lodID = 0; lodID < g_ScriptLODs.Count(); lodID++ )
{
// printf( "lod: %d\n", lodID );
ModelLOD_t *pLOD = &pModel->modelLODs[lodID];
for( int meshID = 0; meshID < pStudioModel->nummeshes; meshID++ )
{
Mesh_t *pMesh = &pLOD->meshes[meshID];
for( int stripGroupID = 0; stripGroupID < pMesh->stripGroups.Count(); stripGroupID++ )
{
StripGroup_t *pStripGroup = &pMesh->stripGroups[stripGroupID];
deltaVert += WriteVerts( curVert + deltaVert, pStripGroup );
deltaIndex += WriteIndices( curIndex + deltaIndex, pStripGroup );
deltaTopology += WriteTopology( curTopology + deltaTopology, pStripGroup );
int nStripCount = pStripGroup->strips.Count();
for( int stripID = 0; stripID < nStripCount; stripID++ )
{
Strip_t *pStrip = &pStripGroup->strips[stripID];
for( int boneStateChangeID = 0; boneStateChangeID < pStrip->numBoneStateChanges; boneStateChangeID++ )
{
WriteBoneStateChange( curBoneStateChange + deltaBoneStateChange, &pStrip->boneStateChanges[boneStateChangeID] );
deltaBoneStateChange++;
}
WriteStrip( curStrip + deltaStrip, pStrip, curIndex, curTopology, curVert, curBoneStateChange );
deltaStrip++;
curBoneStateChange += deltaBoneStateChange;
deltaBoneStateChange = 0;
}
WriteStripGroup( curStripGroup + deltaStripGroup, pStripGroup, curVert, curIndex, curTopology, curStrip );
deltaStripGroup++;
curStrip += deltaStrip;
deltaStrip = 0;
curVert += deltaVert;
deltaVert = 0;
curIndex += deltaIndex;
deltaIndex = 0;
curTopology += deltaTopology;
deltaTopology = 0;
}
WriteMesh( curMesh + deltaMesh, pMesh, curStripGroup );
deltaMesh++;
curStripGroup += deltaStripGroup;
deltaStripGroup = 0;
}
WriteModelLOD( curLOD + deltaLOD, pLOD, curMesh );
deltaLOD++;
curMesh += deltaMesh;
deltaMesh = 0;
}
WriteModel( curModel + deltaModel, pStudioModel, curLOD );
deltaModel++;
curLOD += deltaLOD;
deltaLOD = 0;
}
WriteBodyPart( bodyPartID, pBodyPart, curModel );
curModel += deltaModel;
deltaModel = 0;
}
WriteHeader( m_VertexCacheSize, m_MaxBonesPerVert, m_MaxBonesPerFace,
m_MaxBonesPerStrip, pHdr->numbodyparts, pHdr->checksum );
#ifdef _DEBUG
m_FileBuffer->TestWritten( m_EndOfFileOffset );
#endif
MapGlobalBonesToHardwareBoneIDsAndSortBones( pHdr );
// DebugCompareVerts( phdr );
SanityCheckAgainstStudioHDR( pHdr );
if ( !g_quiet )
{
OutputMemoryUsage();
}
RemoveRedundantBoneStateChanges();
if( g_staticprop )
{
ZeroNumBones();
}
// Show statistics
#ifdef _DEBUG
// ShowStats();
#endif
m_FileBuffer->WriteToFile( pFileName, m_EndOfFileOffset );
FileHeader_t *pVtxHeader = ( FileHeader_t * )m_FileBuffer->GetPointer( 0 );
SanityCheckVertexBoneLODFlags( pHdr, pVtxHeader );
}
static void MergeLikeBoneIndicesWithinVert( mstudioboneweight_t *pBoneWeight )
{
if( pBoneWeight->numbones == 1 )
{
return;
}
int i, j;
int realNumBones = pBoneWeight->numbones;
for( i = 0; i < pBoneWeight->numbones; i++ )
{
for( j = i+1; j < pBoneWeight->numbones; j++ )
{
if( ( pBoneWeight->bone[i] == pBoneWeight->bone[j] ) && ( pBoneWeight->weight[i] != 0.0f ) )
{
pBoneWeight->weight[i] += pBoneWeight->weight[j];
pBoneWeight->weight[j] = 0.0f;
realNumBones--;
}
}
}
// force all of the -1's at the end with a bubble sort
float tmpWeight;
int tmpIndex;
// bubble sort the bones.
for( j = pBoneWeight->numbones; j > 1; j-- )
{
int k;
for( k = 0; k < j - 1; k++ )
{
if( ( pBoneWeight->weight[k] == 0.0f ) && ( pBoneWeight->weight[k+1] != 0.0f ) )
{
// swap
tmpIndex = pBoneWeight->bone[k];
tmpWeight = pBoneWeight->weight[k];
pBoneWeight->bone[k] = pBoneWeight->bone[k+1];
pBoneWeight->weight[k] = pBoneWeight->weight[k+1];
pBoneWeight->bone[k+1] = tmpIndex;
pBoneWeight->weight[k+1] = tmpWeight;
}
}
}
pBoneWeight->numbones = realNumBones;
}
static void MergeLikeBoneIndicesWithinVerts( studiohdr_t *pHdr )
{
int bodyPartID, modelID, vertID;
for( bodyPartID = 0; bodyPartID < pHdr->numbodyparts; bodyPartID++ )
{
mstudiobodyparts_t *pBodyPart = pHdr->pBodypart( bodyPartID );
for( modelID = 0; modelID < pBodyPart->nummodels; modelID++ )
{
mstudiomodel_t *pModel = pBodyPart->pModel( modelID );
for( vertID = 0; vertID < pModel->numvertices; vertID++ )
{
const mstudio_modelvertexdata_t *vertData = pModel->GetVertexData();
Assert( vertData ); // This can only return NULL on X360 for now
mstudioboneweight_t *pBoneWeight = vertData->BoneWeights( vertID );
MergeLikeBoneIndicesWithinVert( pBoneWeight );
}
}
}
}
void COptimizedModel::PrintBoneStateChanges( studiohdr_t *phdr, int lod )
{
FileHeader_t *header = ( FileHeader_t * )m_FileBuffer->GetPointer( 0 );
for( int bodyPartID = 0; bodyPartID < header->numBodyParts; bodyPartID++ )
{
BodyPartHeader_t *bodyPart = header->pBodyPart( bodyPartID );
// mstudiobodyparts_t *pStudioBodyPart = phdr->pBodypart( bodyPartID );
// for( int lodID = 0; lodID < header->numLODs; lodID++ )
int lodID = lod;
{
for( int modelID = 0; modelID < bodyPart->numModels; modelID++ )
{
ModelHeader_t *model = bodyPart->pModel( modelID );
// mstudiomodel_t *pStudioModel = pStudioBodyPart->pModel( modelID );
ModelLODHeader_t *pLOD = model->pLOD( lodID );
for( int meshID = 0; meshID < pLOD->numMeshes; meshID++ )
{
MeshHeader_t *mesh = pLOD->pMesh( meshID );
// mstudiomesh_t *pStudioMesh = pStudioModel->pMesh( meshID );
for( int stripGroupID = 0; stripGroupID < mesh->numStripGroups; stripGroupID++ )
{
StripGroupHeader_t *pStripGroup = mesh->pStripGroup( stripGroupID );
for( int stripID = 0; stripID < pStripGroup->numStrips; stripID++ )
{
StripHeader_t *pStrip = pStripGroup->pStrip( stripID );
for( int boneStateChangeID = 0; boneStateChangeID < pStrip->numBoneStateChanges; boneStateChangeID++ )
{
BoneStateChangeHeader_t *pBoneStateChange = pStrip->pBoneStateChange( boneStateChangeID );
printf( "bone change: hwid: %d boneid: %d (%s)\n",
( int )pBoneStateChange->hardwareID,
( int )pBoneStateChange->newBoneID,
g_bonetable[pBoneStateChange->newBoneID].name);
}
}
}
}
}
}
}
}
void COptimizedModel::PrintVerts( studiohdr_t *phdr, int lod )
{
FileHeader_t *header = ( FileHeader_t * )m_FileBuffer->GetPointer( 0 );
for( int bodyPartID = 0; bodyPartID < header->numBodyParts; bodyPartID++ )
{
BodyPartHeader_t *bodyPart = header->pBodyPart( bodyPartID );
mstudiobodyparts_t *pStudioBodyPart = phdr->pBodypart( bodyPartID );
// for( int lodID = 0; lodID < header->numLODs; lodID++ )
int lodID = lod;
{
for( int modelID = 0; modelID < bodyPart->numModels; modelID++ )
{
ModelHeader_t *model = bodyPart->pModel( modelID );
mstudiomodel_t *pStudioModel = pStudioBodyPart->pModel( modelID );
ModelLODHeader_t *pLOD = model->pLOD( lodID );
for( int meshID = 0; meshID < pLOD->numMeshes; meshID++ )
{
MeshHeader_t *mesh = pLOD->pMesh( meshID );
mstudiomesh_t *pStudioMesh = pStudioModel->pMesh( meshID );
for( int stripGroupID = 0; stripGroupID < mesh->numStripGroups; stripGroupID++ )
{
StripGroupHeader_t *pStripGroup = mesh->pStripGroup( stripGroupID );
for( int vertID = 0; vertID < pStripGroup->numVerts; vertID++ )
{
PrintVert( pStripGroup->pVertex( vertID ), pStudioModel, pStudioMesh );
}
}
}
}
}
}
}
static int CalcNumMaterialReplacements()
{
int i;
int numReplacements = 0;
int numLODs = g_ScriptLODs.Count();
for( i = 0; i < numLODs; i++ )
{
LodScriptData_t &scriptLOD = g_ScriptLODs[i];
numReplacements += scriptLOD.materialReplacements.Count();
}
return numReplacements;
}
//-----------------------------------------------------------------------------
//
// Main entry point
//
//-----------------------------------------------------------------------------
bool COptimizedModel::OptimizeFromStudioHdr( studiohdr_t *pHdr, s_bodypart_t *pSrcBodyParts, int vertCacheSize,
bool usesFixedFunction, bool bForceSoftwareSkin, bool bHWFlex, int maxBonesPerVert, int maxBonesPerFace,
int maxBonesPerStrip, const char *pFileName, const char *glViewFileName )
{
Assert( maxBonesPerVert <= MAX_NUM_BONES_PER_VERT );
Assert( maxBonesPerStrip <= MAX_NUM_BONES_PER_STRIP );
MergeLikeBoneIndicesWithinVerts( pHdr );
// Some initialization
SetupMeshProcessing( pHdr, vertCacheSize, usesFixedFunction, maxBonesPerVert,
maxBonesPerFace, maxBonesPerStrip, pFileName );
// The dude that does it all
TotalMeshStats_t stats;
ProcessModel( pHdr, pSrcBodyParts, stats, bForceSoftwareSkin, bHWFlex );
stats.m_TotalMaterialReplacements = CalcNumMaterialReplacements();
// Write it out to disk
WriteVTXFile( pHdr, pFileName, stats );
// Write out debugging files....
WriteGLViewFiles( pHdr, glViewFileName );
// DebugCrap( pHdr );
// PrintBoneStateChanges( pHdr, 1 );
// PrintVerts( pHdr, 1 );
delete m_FileBuffer;
m_FileBuffer = NULL;
if( m_NumSkinnedAndFlexedVerts != 0 )
{
MdlWarning( "!!!!WARNING!!!!: %d flexed verts had more than one bone influence. . will use SLOW path in engine\n", m_NumSkinnedAndFlexedVerts );
}
CleanupEverything();
return true;
}
static int numGLViewTrangles = 0;
static int numGLViewSWDegenerates = 0;
static int numGLViewHWDegenerates = 0;
enum
{
GLVIEWDRAW_TRILIST,
GLVIEWDRAW_NONE
};
static int s_DrawMode;
static int s_LastThreeIndices[3];
static Vertex_t s_LastThreeVerts[3];
static Vector s_LastThreePositions[3];
static int s_ListID = 0;
static bool s_Shrunk[3];
//-----------------------------------------------------------------------------
//
// The following methods are used in writing out GL View files
//
//-----------------------------------------------------------------------------
void COptimizedModel::PrintVert( Vertex_t *v, mstudiomodel_t *pStudioModel, mstudiomesh_t *pStudioMesh )
{
printf( "vert:\n" );
#if 0
printf( "\tposition: %f %f %f\n",
v->position[0], v->position[1], v->position[2] );
printf( "\tnormal: %f %f %f\n",
v->normal[0], v->normal[1], v->normal[2] );
printf( "\ttexcoord: %f %f\n",
v->texCoord[0], v->texCoord[1] );
#endif
printf( "\torigMeshVertID: %d\n", v->origMeshVertID );
printf( "\tnumBones: %d\n", v->numBones );
int i;
// for( i = 0; i < MAX_NUM_BONES_PER_VERT; i++ )
for( i = 0; i < v->numBones; i++ )
{
float boneWeight = GetOrigVertBoneWeightValue( pStudioModel, pStudioMesh, v, i );
printf( "\tboneID[%d]: %d weight: %f (%s)\n", i, ( int )v->boneID[i], boneWeight,
g_bonetable[v->boneID[i]].name );
}
}
static float RandomFloat( float min, float max )
{
float ret;
ret = ( ( float )rand() ) / ( float )VALVE_RAND_MAX;
ret *= max - min;
ret += min;
return ret;
}
Vector& COptimizedModel::GetOrigVertPosition( mstudiomodel_t *pStudioModel, mstudiomesh_t *pStudioMesh, Vertex_t *pVert )
{
Assert( pStudioMesh->pModel() == pStudioModel );
const mstudio_meshvertexdata_t *vertData = pStudioMesh->GetVertexData();
Assert( vertData ); // This can only return NULL on X360 for now
return *vertData->Position( pVert->origMeshVertID );
}
float COptimizedModel::GetOrigVertBoneWeightValue( mstudiomodel_t *pStudioModel, mstudiomesh_t *pStudioMesh, Vertex_t *pVert, int boneID )
{
Assert( pStudioMesh->pModel() == pStudioModel );
const mstudio_meshvertexdata_t *vertData = pStudioMesh->GetVertexData();
Assert( vertData ); // This can only return NULL on X360 for now
return vertData->BoneWeights( pVert->origMeshVertID )->weight[pVert->boneWeightIndex[boneID]];
}
mstudioboneweight_t &COptimizedModel::GetOrigVertBoneWeight( mstudiomodel_t *pStudioModel, mstudiomesh_t *pStudioMesh, Vertex_t *pVert )
{
Assert( pStudioMesh->pModel() == pStudioModel );
const mstudio_meshvertexdata_t *vertData = pStudioMesh->GetVertexData();
Assert( vertData ); // This can only return NULL on X360 for now
return *vertData->BoneWeights( pVert->origMeshVertID );
}
int COptimizedModel::GetOrigVertBoneIndex( mstudiomodel_t *pStudioModel, mstudiomesh_t *pStudioMesh, Vertex_t *pVert, int boneID )
{
Assert( pStudioMesh->pModel() == pStudioModel );
const mstudio_meshvertexdata_t *vertData = pStudioMesh->GetVertexData();
Assert( vertData ); // This can only return NULL on X360 for now
return vertData->BoneWeights( pVert->origMeshVertID )->bone[pVert->boneWeightIndex[boneID]];
}
void COptimizedModel::SetMeshPropsColor( unsigned int meshFlags, Vector& color )
{
if( meshFlags & MESH_IS_TEETH )
{
color[0] = 1.0f; color[1] = 0.0f; color[2] = 0.0f;
}
else if( meshFlags & MESH_IS_EYES )
{
color[0] = 1.0f; color[1] = 1.0f; color[2] = 0.0f;
}
else
{
color[0] = 0.0f; color[1] = 1.0f; color[2] = 0.0f;
}
}
void COptimizedModel::SetFlexedAndSkinColor( unsigned int glViewFlags, unsigned int stripGroupFlags, Vector& color )
{
if( ( glViewFlags & WRITEGLVIEW_SHOWFLEXED ) && ( glViewFlags & WRITEGLVIEW_SHOWSW ) )
{
if( ( stripGroupFlags & STRIPGROUP_IS_DELTA_FLEXED ) &&
( stripGroupFlags & STRIPGROUP_IS_HWSKINNED ) )
{
// flexed and hw skinned = yellow
color[0] = 1.0f; color[1] = 1.0f; color[2] = 0.0f;
}
else if( !( stripGroupFlags & STRIPGROUP_IS_DELTA_FLEXED ) &&
( stripGroupFlags & STRIPGROUP_IS_HWSKINNED ) )
{
// not flexed and hw skinned = green
color[0] = 0.0f; color[1] = 1.0f; color[2] = 0.0f;
}
else if( !( stripGroupFlags & STRIPGROUP_IS_DELTA_FLEXED ) &&
!( stripGroupFlags & STRIPGROUP_IS_HWSKINNED ) )
{
// not flexed and sw skinned = blue
color[0] = 0.0f; color[1] = 0.0f; color[2] = 1.0f;
}
else if( ( stripGroupFlags & STRIPGROUP_IS_DELTA_FLEXED ) &&
!( stripGroupFlags & STRIPGROUP_IS_HWSKINNED ) )
{
// flexed and sw skinned = red
color[0] = 1.0f; color[1] = 0.0f; color[2] = 0.0f;
}
else
{
Assert( 0 );
}
}
else if( glViewFlags & WRITEGLVIEW_SHOWFLEXED )
{
if( stripGroupFlags & STRIPGROUP_IS_DELTA_FLEXED )
{
color[0] = 1.0f; color[1] = 0.0f; color[2] = 0.0f;
}
else
{
color[0] = 0.0f; color[1] = 1.0f; color[2] = 0.0f;
}
}
else if( glViewFlags & WRITEGLVIEW_SHOWSW )
{
if( stripGroupFlags & STRIPGROUP_IS_HWSKINNED )
{
color[0] = 0.0f; color[1] = 1.0f; color[2] = 0.0f;
}
else
{
color[0] = 1.0f; color[1] = 0.0f; color[2] = 0.0f;
}
}
}
void COptimizedModel::SetColorFromNumVertexBones( int numBones, Vector& color )
{
Vector numBonesColor[5] = {
Vector( 0.0f, 0.0f, 0.0f ), // 0 bones = black
Vector( 0.0f, 1.0f, 0.0f ), // 1 bone = green
Vector( 1.0f, 1.0f, 0.0f ), // 2 bones = yellow
Vector( 0.0f, 0.0f, 1.0f ), // 3 bones = blue
Vector( 1.0f, 0.0f, 0.0f ) // 4 bones = red
};
Assert( numBones >= 0 && numBones <= 4 );
VectorCopy( numBonesColor[numBones], color );
}
void COptimizedModel::DrawGLViewTriangle( FILE *fp, Vector& pos1, Vector& pos2, Vector& pos3,
Vector& color1, Vector& color2, Vector& color3 )
{
numGLViewTrangles++;
fprintf( fp, "3\n" );
fprintf( fp, "%f %f %f %f %f %f\n", pos1[0], pos1[1], pos1[2], color1[0], color1[1], color1[2] );
fprintf( fp, "%f %f %f %f %f %f\n", pos2[0], pos2[1], pos2[2], color2[0], color2[1], color2[2] );
fprintf( fp, "%f %f %f %f %f %f\n", pos3[0], pos3[1], pos3[2], color3[0], color3[1], color3[2] );
}
//-----------------------------------------------------------------------------
// use index to test for degenerates. . isn't used for anything else.
//-----------------------------------------------------------------------------
void COptimizedModel::GLViewVert( FILE *fp, Vertex_t vert, int index,
Vector& color, mstudiomodel_t *pStudioModel, mstudiomesh_t *pStudioMesh,
bool showSubStrips, float shrinkFactor )
{
// CheckVertBoneWeights( &vert, pStudioModel, pStudioMesh );
Assert( s_DrawMode != GLVIEWDRAW_NONE );
int id = s_ListID % 3;
s_LastThreeIndices[id] = index;
s_LastThreeVerts[id] = vert;
s_Shrunk[id] = false;
VectorCopy( GetOrigVertPosition( pStudioModel, pStudioMesh, &s_LastThreeVerts[id] ), s_LastThreePositions[id] );
if( s_DrawMode == GLVIEWDRAW_TRILIST )
{
// trilist
if( id == 2 )
{
ShrinkVerts( shrinkFactor );
DrawGLViewTriangle( fp,
s_LastThreePositions[0],
s_LastThreePositions[1],
s_LastThreePositions[2],
color, color, color );
}
}
else
{
// tristrip
if( s_ListID >= 2 )
{
// spit out the face with both facings. . doesn't matter if we
// get the facing right for glview
if( s_LastThreeIndices[0] == s_LastThreeIndices[1] ||
s_LastThreeIndices[1] == s_LastThreeIndices[2] ||
s_LastThreeIndices[0] == s_LastThreeIndices[2] )
{
// skip degenerate faces
numGLViewHWDegenerates++;
if( showSubStrips )
{
RandomColor( color );
}
}
else if( !( s_ListID & 1 ) )
{
ShrinkVerts( shrinkFactor );
DrawGLViewTriangle( fp,
s_LastThreePositions[(id+0-2+3)%3],
s_LastThreePositions[(id+1-2+3)%3],
s_LastThreePositions[(id+2-2+3)%3],
color, color, color );
}
else
{
ShrinkVerts( shrinkFactor );
DrawGLViewTriangle( fp,
s_LastThreePositions[(id+2-2+3)%3],
s_LastThreePositions[(id+1-2+3)%3],
s_LastThreePositions[(id+0-2+3)%3],
color, color, color );
}
}
}
s_ListID++;
}
void COptimizedModel::GLViewDrawEnd( void )
{
s_DrawMode = GLVIEWDRAW_NONE;
}
/*
void COptimizedModel::DebugCrap( studiohdr_t *phdr )
{
FileHeader_t *header = ( FileHeader_t * )m_FileBuffer->GetPointer( 0 );
for( int bodyPartID = 0; bodyPartID < header->numBodyParts; bodyPartID++ )
{
BodyPartHeader_t *bodyPart = header->pBodyPart( bodyPartID );
mstudiobodyparts_t *pStudioBodyPart = phdr->pBodypart( bodyPartID );
for( int modelID = 0; modelID < bodyPart->numModels; modelID++ )
{
ModelHeader_t *model = bodyPart->pModel( modelID );
mstudiomodel_t *pStudioModel = pStudioBodyPart->pModel( modelID );
for( int lodID = 0; lodID < model->numLODs; lodID++ )
{
char tmp[256];
sprintf( tmp, "crap.lod%d", lodID );
printf( "writing %s\n", tmp );
FILE *fp = fopen( tmp, "w" );
if( !fp )
{
printf( "can't write crap file %s\n", tmp );
return;
}
ModelLODHeader_t *pLOD = model->pLOD( lodID );
for( int meshID = 0; meshID < pLOD->numMeshes; meshID++ )
{
MeshHeader_t *mesh = pLOD->pMesh( meshID );
mstudiomesh_t *pStudioMesh = pStudioModel->pMesh( meshID );
for( int stripGroupID = 0; stripGroupID < mesh->numStripGroups; stripGroupID++ )
{
StripGroupHeader_t *pStripGroup = mesh->pStripGroup( stripGroupID );
for( int stripID = 0; stripID < pStripGroup->numStrips; stripID++ )
{
StripHeader_t *pStrip = pStripGroup->pStrip( stripID );
for( int indexID = 0; indexID < pStrip->numIndices; indexID++ )
{
int id = *pStripGroup->pIndex( indexID + pStrip->indexOffset );
Vertex_t& vert = *pStripGroup->pVertex( id );
Vector& vertPos = GetOrigVertPosition( pStudioModel, pStudioMesh, &vert );
fprintf( fp, "mesh: %04d origvertid: %04d pos: %0.2f %0.2f %0.2f ", meshID, vert.origMeshVertID, vertPos[0], vertPos[1], vertPos[2] );
int i;
for( i = 0; i < vert.numBones; i++ )
{
float boneWeight;
boneWeight = GetOrigVertBoneWeightValue( pStudioModel, pStudioMesh, &vert, i );
int boneID;
boneID = GetOrigVertBoneIndex( pStudioModel, pStudioMesh, &vert, i );
fprintf( fp, "bone: %d %0.1f ", boneID, boneWeight );
}
fprintf( fp, "\n" );
}
}
}
}
fclose( fp );
}
}
}
}
*/
void COptimizedModel::WriteGLViewFile( studiohdr_t *phdr, const char *pFileName, unsigned int flags, float shrinkFactor )
{
Vector color;
RandomColor( color );
FileHeader_t *header = ( FileHeader_t * )m_FileBuffer->GetPointer( 0 );
for( int bodyPartID = 0; bodyPartID < header->numBodyParts; bodyPartID++ )
{
BodyPartHeader_t *bodyPart = header->pBodyPart( bodyPartID );
mstudiobodyparts_t *pStudioBodyPart = phdr->pBodypart( bodyPartID );
for( int modelID = 0; modelID < bodyPart->numModels; modelID++ )
{
ModelHeader_t *model = bodyPart->pModel( modelID );
mstudiomodel_t *pStudioModel = pStudioBodyPart->pModel( modelID );
for( int lodID = 0; lodID < model->numLODs; lodID++ )
{
char tmp[256];
sprintf( tmp, "%s.lod%d", pFileName, lodID );
printf( "writing %s\n", tmp );
CPlainAutoPtr< CP4File > spFile( g_p4factory->AccessFile( tmp ) );
spFile->Edit();
FILE *fp = fopen( tmp, "w" );
if( !fp )
{
printf( "can't write glview file %s\n", tmp );
return;
}
/*
// write out tangent space vectors
int vertID;
for( vertID = 0; vertID < pStudioModel->numvertices; vertID++ )
{
const Vector &pos = *pStudioModel->pVertex( vertID );
const Vector &norm = *pStudioModel->pNormal( vertID );
const Vector4D &sVect = *pStudioModel->pTangentS( vertID );
Vector tmpVect;
tmpVect = pos + norm * .15f;
fprintf( fp, "2\n" );
fprintf( fp, "%f %f %f 0.0 0.0 1.0\n", pos[0], pos[1], pos[2] );
fprintf( fp, "%f %f %f 0.0 0.0 1.0\n", tmpVect[0], tmpVect[1], tmpVect[2] );
Vector tmpSVect( sVect[0], sVect[1], sVect[2] );
tmpVect = pos + tmpSVect * .15f;
fprintf( fp, "2\n" );
fprintf( fp, "%f %f %f 1.0 0.0 0.0\n", pos[0], pos[1], pos[2] );
fprintf( fp, "%f %f %f 1.0 0.0 0.0\n", tmpVect[0], tmpVect[1], tmpVect[2] );
Vector tmpTVect;
CrossProduct( norm, tmpSVect, tmpTVect );
tmpTVect *= sVect[3];
tmpVect = pos + tmpTVect * .15f;
fprintf( fp, "2\n" );
fprintf( fp, "%f %f %f 0.0 1.0 0.0\n", pos[0], pos[1], pos[2] );
fprintf( fp, "%f %f %f 0.0 1.0 0.0\n", tmpVect[0], tmpVect[1], tmpVect[2] );
}
continue;
*/
ModelLODHeader_t *pLOD = model->pLOD( lodID );
for( int meshID = 0; meshID < pLOD->numMeshes; meshID++ )
{
MeshHeader_t *mesh = pLOD->pMesh( meshID );
mstudiomesh_t *pStudioMesh = pStudioModel->pMesh( meshID );
if( flags & WRITEGLVIEW_SHOWMESH )
{
RandomColor( color );
}
if( flags & WRITEGLVIEW_SHOWMESHPROPS )
{
SetMeshPropsColor( mesh->flags, color );
}
for( int stripGroupID = 0; stripGroupID < mesh->numStripGroups; stripGroupID++ )
{
if( flags & WRITEGLVIEW_SHOWSTRIPGROUP )
{
RandomColor( color );
}
StripGroupHeader_t *pStripGroup = mesh->pStripGroup( stripGroupID );
SetFlexedAndSkinColor( flags, pStripGroup->flags, color );
for( int stripID = 0; stripID < pStripGroup->numStrips; stripID++ )
{
StripHeader_t *pStrip = pStripGroup->pStrip( stripID );
if( flags & WRITEGLVIEW_SHOWSTRIP )
{
RandomColor( color );
}
if( flags & WRITEGLVIEW_SHOWSTRIPNUMBONES )
{
switch (pStrip->numBones)
{
case 0:
case 1:
color.Init( 0, 0, 255 );
break;
case 2:
color.Init( 0, 255, 0 );
break;
case 3:
color.Init( 255, 255, 0 );
break;
case 4:
color.Init( 255, 0, 0 );
break;
}
}
GLViewDrawBegin( GLVIEWDRAW_TRILIST );
for( int indexID = 0; indexID < pStrip->numIndices; indexID++ )
{
int id = *pStripGroup->pIndex( indexID + pStrip->indexOffset );
Vertex_t& vert = *pStripGroup->pVertex( id );
if( flags & WRITEGLVIEW_SHOWVERTNUMBONES )
{
switch (vert.numBones)
{
case 0:
case 1:
color.Init( 0, 0, 255 );
break;
case 2:
color.Init( 0, 255, 0 );
break;
case 3:
color.Init( 255, 255, 0 );
break;
case 4:
color.Init( 255, 0, 0 );
break;
}
}
GLViewVert( fp, vert, id, color, pStudioModel, pStudioMesh,
( flags & WRITEGLVIEW_SHOWSUBSTRIP ) ? true : false, shrinkFactor );
}
GLViewDrawEnd();
}
}
}
fclose( fp );
spFile->Add();
}
}
}
}
//-----------------------------------------------------------------------------
// Write out all GL View files
//-----------------------------------------------------------------------------
void COptimizedModel::WriteGLViewFiles( studiohdr_t *pHdr, char const* glViewFileName )
{
if( !g_bDumpGLViewFiles )
return;
char tmpFileName[128];
strcpy( tmpFileName, glViewFileName );
strcat( tmpFileName, ".mesh" );
WriteGLViewFile( pHdr, tmpFileName, WRITEGLVIEW_SHOWMESH, .8f );
strcpy( tmpFileName, glViewFileName );
strcat( tmpFileName, ".stripgroup" );
WriteGLViewFile( pHdr, tmpFileName, WRITEGLVIEW_SHOWSTRIPGROUP, .8f );
strcpy( tmpFileName, glViewFileName );
strcat( tmpFileName, ".strip" );
WriteGLViewFile( pHdr, tmpFileName, WRITEGLVIEW_SHOWSTRIP, .8f );
strcpy( tmpFileName, glViewFileName );
strcat( tmpFileName, ".substrip" );
WriteGLViewFile( pHdr, tmpFileName, WRITEGLVIEW_SHOWSUBSTRIP, .97f );
strcpy( tmpFileName, glViewFileName );
strcat( tmpFileName, ".flexed" );
WriteGLViewFile( pHdr, tmpFileName, WRITEGLVIEW_SHOWFLEXED, .8f );
strcpy( tmpFileName, glViewFileName );
strcat( tmpFileName, ".sw" );
WriteGLViewFile( pHdr, tmpFileName, WRITEGLVIEW_SHOWSW, .8f );
strcpy( tmpFileName, glViewFileName );
strcat( tmpFileName, ".flexedandsw" );
WriteGLViewFile( pHdr, tmpFileName, WRITEGLVIEW_SHOWSW | WRITEGLVIEW_SHOWFLEXED, .8f );
strcpy( tmpFileName, glViewFileName );
strcat( tmpFileName, ".meshprops" );
WriteGLViewFile( pHdr, tmpFileName, WRITEGLVIEW_SHOWMESHPROPS, .8f );
strcpy( tmpFileName, glViewFileName );
strcat( tmpFileName, ".vertnumbones" );
WriteGLViewFile( pHdr, tmpFileName, WRITEGLVIEW_SHOWVERTNUMBONES, 1.0f );
strcpy( tmpFileName, glViewFileName );
strcat( tmpFileName, ".stripnumbones" );
WriteGLViewFile( pHdr, tmpFileName, WRITEGLVIEW_SHOWSTRIPNUMBONES, 1.0f );
}
void COptimizedModel::GLViewDrawBegin( int mode )
{
s_DrawMode = mode;
s_ListID = 0;
}
void COptimizedModel::ShrinkVerts( float shrinkFactor )
{
Vector center;
Vector delta;
VectorCopy( s_LastThreePositions[0], center );
VectorAdd( center, s_LastThreePositions[1], center );
VectorAdd( center, s_LastThreePositions[2], center );
VectorScale( center, 1.0f / 3.0f, center );
int i;
for( i = 0; i < 3; i++ )
{
if( s_Shrunk[i] )
{
continue;
}
VectorSubtract( s_LastThreePositions[i], center, delta );
VectorScale( delta, shrinkFactor, delta );
VectorAdd( center, delta, s_LastThreePositions[i] );
s_Shrunk[i] = true;
}
}
void COptimizedModel::CheckVertBoneWeights( Vertex_t *pVert, mstudiomodel_t *pStudioModel, mstudiomesh_t *pStudioMesh )
{
int i;
float sum = 0;
for( i = 0; i < MAX_NUM_BONES_PER_VERT; i++ )
{
float boneWeight = GetOrigVertBoneWeightValue( pStudioModel, pStudioMesh, pVert, i );
sum += boneWeight;
}
Assert( sum > 0.95f && sum < 1.1f );
}
#define CACHE_INEFFICIENCY 6
void COptimizedModel::ShowStats( void )
{
int totalHWTriangles = 0;
int totalHWDegenerates = 0;
int totalHWIndices = 0;
int totalHWVertexCacheHits = 0;
int totalHWVertexCacheMisses = 0;
int totalSWTriangles = 0;
int totalSWDegenerates = 0;
int totalSWIndices = 0;
int totalSWVertexCacheHits = 0;
int totalSWVertexCacheMisses = 0;
CHardwareVertexCache hardwareVertexCache;
hardwareVertexCache.Init( m_VertexCacheSize - CACHE_INEFFICIENCY );
FileHeader_t *header = ( FileHeader_t * )m_FileBuffer->GetPointer( 0 );
printf( "header: %d body parts\n", header->numBodyParts );
for( int bodyPartID = 0; bodyPartID < header->numBodyParts; bodyPartID++ )
{
BodyPartHeader_t *bodyPart = header->pBodyPart( bodyPartID );
printf( " bodyPart %d: %d models\n", bodyPartID, bodyPart->numModels );
for( int modelID = 0; modelID < bodyPart->numModels; modelID++ )
{
ModelHeader_t *model = bodyPart->pModel( modelID );
printf( " model: %d lods\n", model->numLODs );
for( int lodID = 0; lodID < 1; lodID++ )
{
ModelLODHeader_t *pLOD = model->pLOD( lodID );
printf( " lod: %d meshes\n", pLOD->numMeshes );
for( int meshID = 0; meshID < pLOD->numMeshes; meshID++ )
{
MeshHeader_t *mesh = pLOD->pMesh( meshID );
printf( " mesh %d: %d stripsgroups\n", meshID, mesh->numStripGroups );
for( int stripGroupID = 0; stripGroupID < mesh->numStripGroups; stripGroupID++ )
{
StripGroupHeader_t *pStripGroup = mesh->pStripGroup( stripGroupID );
// JasonM - just report this stuff for triangle lists for now
if ( pStripGroup->flags & STRIP_IS_TRILIST )
{
for( int stripID = 0; stripID < pStripGroup->numStrips; stripID++ )
{
for( int stripID = 0; stripID < pStripGroup->numStrips; stripID++ )
{
int lastThreeIndices[3];
StripHeader_t *pStrip = pStripGroup->pStrip( stripID );
printf( " strip: %d numIndices: %d indexOffset: %d\n", stripID, pStrip->numIndices, pStrip->indexOffset );
hardwareVertexCache.Flush();
Assert( pStrip->numIndices % 3 == 0 );
totalHWTriangles += pStrip->numIndices / 3;
for( int indexID = 0; indexID < pStrip->numIndices; indexID++ )
{
int newVertOffset = indexID % 3;
int index = *pStripGroup->pIndex( indexID + pStrip->indexOffset );
// printf( "%d\n", index );
lastThreeIndices[newVertOffset] = index;
if( newVertOffset == 2 )
{
if( lastThreeIndices[0] == lastThreeIndices[1] ||
lastThreeIndices[1] == lastThreeIndices[2] ||
lastThreeIndices[0] == lastThreeIndices[2] )
{
// printf( "degenerate triangle!!!! %d %d %d\n", lastThreeIndices[0], lastThreeIndices[1], lastThreeIndices[2] );
totalHWDegenerates++;
}
}
totalHWIndices++;
if( !hardwareVertexCache.IsPresent( index ) )
{
totalHWVertexCacheMisses++;
hardwareVertexCache.Insert( index );
}
else
{
totalHWVertexCacheHits++;
}
}
}
}
} // if STRIP_IS_TRILIST
}
}
}
}
}
int totalRealHWTriangles = totalHWTriangles - totalHWDegenerates;
int totalRealSWTriangles = totalSWTriangles - totalSWDegenerates;
printf( "TotalHWTriangles: %d\n", totalHWTriangles );
printf( "TotalHWDegenerates: %d\n", totalHWDegenerates );
printf( "TotalRealHWTriangles: %d\n", totalRealHWTriangles );
printf( "TotalHWIndices: %d\n", totalHWIndices );
printf( "HW real tris/index: %f\n", ( float )totalRealHWTriangles / ( float )totalHWIndices );
printf( "totalHWVertexCacheHits: %d\n", totalHWVertexCacheHits );
printf( "totalHWVertexCacheMisses: %d\n", totalHWVertexCacheMisses );
printf( "HW vertex cache hit/miss ratio: %f\n", ( float )totalHWVertexCacheHits / ( float )totalHWVertexCacheMisses );
printf( "TotalSWTriangles: %d\n", totalSWTriangles );
printf( "TotalSWDegenerates: %d\n", totalSWDegenerates );
printf( "TotalRealSWTriangles: %d\n", totalRealSWTriangles );
printf( "TotalSWIndices: %d\n", totalSWIndices );
printf( "SW real tris/index: %f\n", ( float )totalRealSWTriangles / ( float )totalSWIndices );
printf( "totalSWVertexCacheHits: %d\n", totalSWVertexCacheHits );
printf( "totalSWVertexCacheMisses: %d\n", totalSWVertexCacheMisses );
printf( "SW vertex cache hit/miss ratio: %f\n", ( float )totalSWVertexCacheHits / ( float )totalSWVertexCacheMisses );
}
void COptimizedModel::CheckVert( Vertex_t *pVert, int maxBonesPerFace, int maxBonesPerVert )
{
#ifndef IGNORE_BONES
#ifdef _DEBUG
int offset = ( int )( ( unsigned char * )pVert - ( unsigned char * )m_FileBuffer->GetPointer( 0 ) );
Assert( offset >= m_VertsOffset && offset < m_IndicesOffset );
Assert( ( ( offset - m_VertsOffset ) % sizeof( Vertex_t ) ) == 0 );
#endif
int j;
for( j = 0; j < maxBonesPerVert; j++ )
{
if( pVert->boneID[j] != 255 )
{
Assert( pVert->boneID[j] >= 0 && pVert->boneID[j] < maxBonesPerFace );
}
#if 0
if( pVert->boneWeights[j] != 0 )
{
Assert( pVert->boneID[j] != 255 );
}
#endif
}
// Test to make sure we are sorted.
for( j = 0; j < maxBonesPerVert-1; j++ )
{
#if 1
// if( pVert->boneWeights[j] != 0 && pVert->boneWeights[j+1] != 0 )
{
Assert( pVert->boneID[j] < pVert->boneID[j+1] );
}
#endif
}
#if 0
// Make sure that all the non-zero weights are first.
bool foundZero = false;
for( j = 0; j < maxBonesPerVert; j++ )
{
if( !foundZero )
{
if( pVert->boneWeights[j] == 0.0f )
{
foundZero = true;
}
}
else
{
Assert( pVert->boneWeights[j] == 0.0f );
}
}
#endif
#endif
}
void COptimizedModel::CheckAllVerts( int maxBonesPerFace, int maxBonesPerVert )
{
int i;
for( i = m_VertsOffset; i < m_IndicesOffset; i += sizeof( Vertex_t ) )
{
Vertex_t *vert = ( Vertex_t * )m_FileBuffer->GetPointer( i );
CheckVert( vert, maxBonesPerFace, maxBonesPerVert );
}
}
void COptimizedModel::SortBonesWithinVertex( bool flexed, Vertex_t *vert, mstudiomodel_t *pStudioModel, mstudiomesh_t *pStudioMesh, int *globalToHardwareBoneIndex, int *hardwareToGlobalBoneIndex, int maxBonesPerFace, int maxBonesPerVert )
{
int i;
/*
for( i = 0; i < m_NumBones; i++ )
{
if( globalToHardwareBoneIndex[i] != -1 )
{
if( flexed )
{
printf( "global bone id: %d hardware bone id: %d\n",
i, globalToHardwareBoneIndex[i] );
}
}
}
*/
#if 0
unsigned char tmpWeightIndex;
int tmpIndex;
int j;
// bubble sort the bones.
for( j = m_MaxBonesPerVert; j > 1; j-- )
{
int k;
for( k = 0; k < j - 1; k++ )
{
if( vert->boneID[k] > vert->boneID[k+1] )
{
// swap
tmpIndex = vert->boneID[k];
tmpWeightIndex = vert->boneWeightIndex[k];
vert->boneID[k] = vert->boneID[k+1];
vert->boneWeightIndex[k] = vert->boneWeightIndex[k+1];
vert->boneID[k+1] = tmpIndex;
vert->boneWeightIndex[k+1] = tmpWeightIndex;
}
}
}
#else
int origBoneWeightIndex[MAX_NUM_BONES_PER_VERT];
int zeroWeightIndex = -1;
// find a orig vert bone index that has a zero weight
for( i = 0; i < MAX_NUM_BONES_PER_VERT; i++ )
{
float boneWeight = GetOrigVertBoneWeightValue( pStudioModel, pStudioMesh, vert, i );
if( boneWeight == 0.0f )
{
zeroWeightIndex = i;
break;
}
}
for( i = 0; i < MAX_NUM_BONES_PER_VERT; i++ )
{
origBoneWeightIndex[i] = zeroWeightIndex;
}
for( i = 0; i < vert->numBones; i++ )
{
float boneWeight = GetOrigVertBoneWeightValue( pStudioModel, pStudioMesh, vert, i );
int globalBoneIndex = GetOrigVertBoneIndex( pStudioModel, pStudioMesh, vert, i );
// if( vert->numBones > 1 )
{
if( flexed )
{
printf( "boneWeight: %f\n", boneWeight );
printf( "globalBoneIndex: %d\n", globalBoneIndex );
}
}
if( boneWeight > 0.0f )
{
int hardwareBoneIndex = globalToHardwareBoneIndex[globalBoneIndex];
Assert( globalBoneIndex != -1 );
origBoneWeightIndex[hardwareBoneIndex] = vert->boneWeightIndex[i];
}
else
{
int hardwareBoneIndex = globalToHardwareBoneIndex[globalBoneIndex];
origBoneWeightIndex[hardwareBoneIndex] = zeroWeightIndex;
Assert( zeroWeightIndex != -1 );
Assert( globalBoneIndex == -1 );
}
}
// if( vert->numBones > 1 )
for( i = 0; i < maxBonesPerFace; i++ )
{
vert->boneID[i] = i;
vert->boneWeightIndex[i] = origBoneWeightIndex[i];
float boneWeight = GetOrigVertBoneWeightValue( pStudioModel, pStudioMesh, vert, i );
int globalBoneIndex = GetOrigVertBoneIndex( pStudioModel, pStudioMesh, vert, i );
if( flexed )
{
Assert( boneWeight >= 0.0f && boneWeight <= 1.0f );
printf( "boneWeight: %f ", boneWeight );
printf( "globalBoneIndex: %d ", globalBoneIndex );
printf( "hardwareBoneID: %d\n", i );
}
}
vert->numBones = maxBonesPerFace; // this may be different for software t&l stuff
#endif
}
void COptimizedModel::RemoveRedundantBoneStateChanges( void )
{
FileHeader_t *header = ( FileHeader_t * )m_FileBuffer->GetPointer( 0 );
for( int bodyPartID = 0; bodyPartID < header->numBodyParts; bodyPartID++ )
{
BodyPartHeader_t *bodyPart = header->pBodyPart( bodyPartID );
bool allocated[MAX_NUM_BONES_PER_STRIP];
int hardwareBoneState[MAX_NUM_BONES_PER_STRIP];
bool changed[MAX_NUM_BONES_PER_STRIP];
// start anew with each body part
// printf( "START BODY PARTY - RESETTING BONE MATRIX STATE\n" );
int i;
for( i = 0; i < MAX_NUM_BONES_PER_STRIP; i++ )
{
hardwareBoneState[i] = -1;
allocated[i] = false;
}
for( int modelID = 0; modelID < bodyPart->numModels; modelID++ )
{
ModelHeader_t *model = bodyPart->pModel( modelID );
for( int lodID = 0; lodID < model->numLODs; lodID++ )
{
ModelLODHeader_t *pLOD = model->pLOD( lodID );
for( int meshID = 0; meshID < pLOD->numMeshes; meshID++ )
{
MeshHeader_t *mesh = pLOD->pMesh( meshID );
for( int stripGroupID = 0; stripGroupID < mesh->numStripGroups; stripGroupID++ )
{
StripGroupHeader_t *pStripGroup = mesh->pStripGroup( stripGroupID );
if( !( pStripGroup->flags & STRIPGROUP_IS_HWSKINNED ) )
{
// printf( "skipping! software skinned stripgroup\n" );
continue;
}
for( int stripID = 0; stripID < pStripGroup->numStrips; stripID++ )
{
StripHeader_t *pStrip = pStripGroup->pStrip( stripID );
// int startNumBoneChanges = pStrip->numBoneStateChanges;
/*
printf( "HARDWARE BONE STATE\n" );
for( i = 0; i < MAX_NUM_BONES_PER_STRIP; i++ )
{
if( allocated[i] )
{
printf( "\thw: %d global: %d\n", i, hardwareBoneState[i] );
}
}
printf( "before optimization\n" );
for( i = 0; i < pStrip->numBoneStateChanges; i++ )
{
printf( "\thw: %d global: %d\n",
( int )pStrip->pBoneStateChange( i )->hardwareID,
( int )pStrip->pBoneStateChange( i )->newBoneID );
}
*/
for( i = 0; i < MAX_NUM_BONES_PER_STRIP; i++ )
{
changed[i] = false;
}
for( int boneStateChangeID = 0; boneStateChangeID < pStrip->numBoneStateChanges; boneStateChangeID++ )
{
BoneStateChangeHeader_t *boneStateChange = pStrip->pBoneStateChange( boneStateChangeID );
Assert( boneStateChange->hardwareID >= 0 && boneStateChange->hardwareID < MAX_NUM_BONES_PER_STRIP );
if( allocated[boneStateChange->hardwareID] &&
hardwareBoneState[boneStateChange->hardwareID] == boneStateChange->newBoneID )
{
// already got this one!
}
else
{
changed[boneStateChange->hardwareID] = true;
allocated[boneStateChange->hardwareID] = true;
hardwareBoneState[boneStateChange->hardwareID] = boneStateChange->newBoneID;
}
}
/*
// now "changed" should tell us which ones we care about.
int curOutBoneID = 0;
for( i = 0; i < pStrip->numBoneStateChanges; i++ )
{
// hack . . going to stomp over what is already there with new data.
if( changed[i] )
{
BoneStateChangeHeader_t *boneStateChange = pStrip->pBoneStateChange( curOutBoneID );
boneStateChange->hardwareID = i;
boneStateChange->newBoneID = hardwareBoneState[i];
curOutBoneID++;
}
}
pStrip->numBoneStateChanges = curOutBoneID;
printf( "start bone changes: %d end bone changes: %d\n", startNumBoneChanges, pStrip->numBoneStateChanges );
printf( "after optimization\n" );
for( i = 0; i < pStrip->numBoneStateChanges; i++ )
{
printf( "\thw: %d global: %d\n",
( int )pStrip->pBoneStateChange( i )->hardwareID,
( int )pStrip->pBoneStateChange( i )->newBoneID );
}
*/
}
}
}
}
}
}
}
static void AddMaterialReplacementsToStringTable( void )
{
int i, j;
int numLODs = g_ScriptLODs.Count();
for( i = 0; i < numLODs; i++ )
{
LodScriptData_t &scriptLOD = g_ScriptLODs[i];
for( j = 0; j < scriptLOD.materialReplacements.Count(); j++ )
{
CLodScriptReplacement_t &materialReplacement = scriptLOD.materialReplacements[j];
s_StringTable.AddString( materialReplacement.GetDstName() );
}
}
}
// Check that all replacematerial/removemesh commands map to valid source materials
void ValidateLODReplacements( studiohdr_t *pHdr )
{
bool failed = false;
int lodID;
for( lodID = 0; lodID < g_ScriptLODs.Count(); lodID++ )
{
LodScriptData_t& scriptLOD = g_ScriptLODs[lodID];
int j;
for( j = 0; j < scriptLOD.meshRemovals.Count(); j++ )
{
const char *pName1 = scriptLOD.meshRemovals[j].GetSrcName();
int i;
for( i = 0; i < pHdr->numtextures; i++ )
{
const char *pName2 = pHdr->pTexture( i )->material->GetName();
if( ComparePath( pName1, pName2 ) )
{
goto got_one;
}
}
// no match
MdlWarning( "\"%s\" doesn't match any of the materals in the model\n", pName1 );
failed = true;
got_one:
;
}
}
if( failed )
{
MdlWarning( "possible materials in model:\n" );
int i;
for( i = 0; i < pHdr->numtextures; i++ )
{
MdlWarning( "\t\"%s\"\n", pHdr->pTexture( i )->material->GetName() );
}
MdlError( "Exiting due to errors\n" );
}
}
void WriteOptimizedFiles( studiohdr_t *phdr, s_bodypart_t *pSrcBodyParts )
{
char filename[MAX_PATH];
char tmpFileName[MAX_PATH];
char glViewFilename[MAX_PATH];
ValidateLODReplacements( phdr );
s_StringTable.Purge();
// hack! This should really go in the mdl file since it's common to all LODs.
AddMaterialReplacementsToStringTable();
strcpy( filename, gamedir );
// if( *g_pPlatformName )
// {
// strcat( filename, "platform_" );
// strcat( filename, g_pPlatformName );
// strcat( filename, "/" );
// }
strcat( filename, "models/" );
strcat( filename, g_outname );
Q_StripExtension( filename, filename, sizeof( filename ) );
if ( g_gameinfo.bSupportsDX8 && !g_bFastBuild )
{
strcpy( tmpFileName, filename );
strcat( tmpFileName, ".sw.vtx" );
strcpy( glViewFilename, filename );
strcat( glViewFilename, ".sw.glview" );
bool bForceSoftwareSkinning = phdr->numbones > 0 && !g_staticprop;
s_OptimizedModel.OptimizeFromStudioHdr( phdr, pSrcBodyParts,
512, //vert cache size FIXME: figure out the correct size for L1
false, /* doesn't use fixed function */
bForceSoftwareSkinning, // force software skinning if not static prop
false, // No hardware flex
3, // bones/vert
3*3, // bones/tri
512, // bones/strip
tmpFileName, glViewFilename );
}
if ( g_gameinfo.bSupportsDX8 && !g_bFastBuild )
{
strcpy( tmpFileName, filename );
strcat( tmpFileName, ".dx80.vtx" );
strcpy( glViewFilename, filename );
strcat( glViewFilename, ".dx80.glview" );
s_OptimizedModel.OptimizeFromStudioHdr( phdr, pSrcBodyParts,
24 /* vert cache size (real size, not effective!)*/,
false, /* doesn't use fixed function */
false, // don't force software skinning
false, // No hardware flex
3 /* bones/vert */,
9 /* bones/tri */,
16 /* bones/strip */,
tmpFileName, glViewFilename );
}
if ( true ) // Always process dx90
{
strcpy( tmpFileName, filename );
strcat( tmpFileName, ".dx90.vtx" );
strcpy( glViewFilename, filename );
strcat( glViewFilename, ".dx90.glview" );
s_OptimizedModel.OptimizeFromStudioHdr( phdr, pSrcBodyParts,
24 /* vert cache size (real size, not effective!)*/,
false, /* doesn't use fixed function */
false, // don't force software skinning
true, // Hardware flex on DX9 parts
3 /* bones/vert */,
9 /* bones/tri */,
53 /* bones/strip */,
tmpFileName, glViewFilename );
}
s_StringTable.Purge();
}
}; // namespace OptimizedModel
#pragma optimize( "", on )