|
|
//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Spews BSP Info
//
//=====================================================================================//
#include "xbspinfo.h"
BEGIN_BYTESWAP_DATADESC( dheader_t ) DEFINE_FIELD( ident, FIELD_INTEGER ), DEFINE_FIELD( version, FIELD_INTEGER ), DEFINE_EMBEDDED_ARRAY( lumps, HEADER_LUMPS ), DEFINE_FIELD( mapRevision, FIELD_INTEGER ), END_BYTESWAP_DATADESC()
BEGIN_BYTESWAP_DATADESC( lump_t ) DEFINE_FIELD( fileofs, FIELD_INTEGER ), DEFINE_FIELD( filelen, FIELD_INTEGER ), DEFINE_FIELD( version, FIELD_INTEGER ), DEFINE_ARRAY( fourCC, FIELD_CHARACTER, 4 ), END_BYTESWAP_DATADESC()
typedef struct { const char *pFriendlyName; const char *pName; int lumpNum; } lumpName_t;
lumpName_t g_lumpNames[] = { {"Entities", "LUMP_ENTITIES", LUMP_ENTITIES}, {"Planes", "LUMP_PLANES", LUMP_PLANES}, {"TexData", "LUMP_TEXDATA", LUMP_TEXDATA}, {"Vertexes", "LUMP_VERTEXES", LUMP_VERTEXES}, {"Visibility", "LUMP_VISIBILITY", LUMP_VISIBILITY}, {"Nodes", "LUMP_NODES", LUMP_NODES}, {"TexInfo", "LUMP_TEXINFO", LUMP_TEXINFO}, {"Faces", "LUMP_FACES", LUMP_FACES}, {"Face IDs", "LUMP_FACEIDS", LUMP_FACEIDS}, {"Lighting", "LUMP_LIGHTING", LUMP_LIGHTING}, {"Occlusion", "LUMP_OCCLUSION", LUMP_OCCLUSION}, {"Leafs", "LUMP_LEAFS", LUMP_LEAFS}, {"Edges", "LUMP_EDGES", LUMP_EDGES}, {"Surf Edges", "LUMP_SURFEDGES", LUMP_SURFEDGES}, {"Models", "LUMP_MODELS", LUMP_MODELS}, {"World Lights", "LUMP_WORLDLIGHTS", LUMP_WORLDLIGHTS}, {"Leaf Faces", "LUMP_LEAFFACES", LUMP_LEAFFACES}, {"Leaf Brushes", "LUMP_LEAFBRUSHES", LUMP_LEAFBRUSHES}, {"Brushes", "LUMP_BRUSHES", LUMP_BRUSHES}, {"Brush Sides", "LUMP_BRUSHSIDES", LUMP_BRUSHSIDES}, {"Areas", "LUMP_AREAS", LUMP_AREAS}, {"Area Portals", "LUMP_AREAPORTALS", LUMP_AREAPORTALS}, {"Disp Info", "LUMP_DISPINFO", LUMP_DISPINFO}, {"Original Faces", "LUMP_ORIGINALFACES", LUMP_ORIGINALFACES}, {"Phys Disp", "LUMP_PHYSDISP", LUMP_PHYSDISP}, {"Phys Collide", "LUMP_PHYSCOLLIDE", LUMP_PHYSCOLLIDE}, {"Vert Normals", "LUMP_VERTNORMALS", LUMP_VERTNORMALS}, {"Vert Normal Indices", "LUMP_VERTNORMALINDICES", LUMP_VERTNORMALINDICES}, {"Disp Lightmap Alphas", "LUMP_DISP_LIGHTMAP_ALPHAS", LUMP_DISP_LIGHTMAP_ALPHAS}, {"Disp Verts", "LUMP_DISP_VERTS", LUMP_DISP_VERTS}, {"Disp Lightmap Sample Positions", "LUMP_DISP_LIGHTMAP_SAMPLE_POSITIONS", LUMP_DISP_LIGHTMAP_SAMPLE_POSITIONS}, {"Game Lump", "LUMP_GAME_LUMP", LUMP_GAME_LUMP}, {"Leaf Water Data", "LUMP_LEAFWATERDATA", LUMP_LEAFWATERDATA}, {"Primitives", "LUMP_PRIMITIVES", LUMP_PRIMITIVES}, {"Prim Verts", "LUMP_PRIMVERTS", LUMP_PRIMVERTS}, {"Prim Indices", "LUMP_PRIMINDICES", LUMP_PRIMINDICES}, {"Pak File", "LUMP_PAKFILE", LUMP_PAKFILE}, {"Clip Portal Verts", "LUMP_CLIPPORTALVERTS", LUMP_CLIPPORTALVERTS}, {"Cube Maps", "LUMP_CUBEMAPS", LUMP_CUBEMAPS}, {"Tex Data String Data", "LUMP_TEXDATA_STRING_DATA", LUMP_TEXDATA_STRING_DATA}, {"Tex Data String Table", "LUMP_TEXDATA_STRING_TABLE", LUMP_TEXDATA_STRING_TABLE}, {"Overlays", "LUMP_OVERLAYS", LUMP_OVERLAYS}, {"Leaf Min Dist To Water", "LUMP_LEAFMINDISTTOWATER", LUMP_LEAFMINDISTTOWATER}, {"Face Macro Texture Info", "LUMP_FACE_MACRO_TEXTURE_INFO", LUMP_FACE_MACRO_TEXTURE_INFO}, {"Disp Tris", "LUMP_DISP_TRIS", LUMP_DISP_TRIS}, {"Phys Collide Surface", "LUMP_PHYSCOLLIDESURFACE", LUMP_PHYSCOLLIDESURFACE}, {"Water Overlays", "LUMP_WATEROVERLAYS", LUMP_WATEROVERLAYS}, {"Leaf Ambient index HDR", "LUMP_LEAF_AMBIENT_INDEX_HDR", LUMP_LEAF_AMBIENT_INDEX_HDR}, {"Leaf Ambient index", "LUMP_LEAF_AMBIENT_INDEX", LUMP_LEAF_AMBIENT_INDEX}, {"Lighting (HDR)", "LUMP_LIGHTING_HDR", LUMP_LIGHTING_HDR}, {"World Lights (HDR)", "LUMP_WORLDLIGHTS_HDR", LUMP_WORLDLIGHTS_HDR}, {"Leaf Ambient Lighting (HDR)", "LUMP_LEAF_AMBIENT_LIGHTING_HDR", LUMP_LEAF_AMBIENT_LIGHTING_HDR}, {"Leaf Ambient Lighting", "LUMP_LEAF_AMBIENT_LIGHTING", LUMP_LEAF_AMBIENT_LIGHTING}, {"*** DEAD ***", "LUMP_XZIPPAKFILE", LUMP_XZIPPAKFILE}, {"Faces (HDR)", "LUMP_FACES_HDR", LUMP_FACES_HDR}, {"Flags", "LUMP_MAP_FLAGS", LUMP_MAP_FLAGS}, {"Fade Overlays", "LUMP_OVERLAY_FADES", LUMP_OVERLAY_FADES}, };
bool g_bQuiet; bool g_bAsPercent; bool g_bAsBytes; bool g_bSortByOffset; bool g_bSortBySize; bool g_bFriendlyNames;
//-----------------------------------------------------------------------------
// Convert lump ID to descriptive Name
//-----------------------------------------------------------------------------
const char *BSP_LumpNumToName( int lumpNum ) { int i;
for ( i=0; i<ARRAYSIZE( g_lumpNames ); ++i ) { if ( g_lumpNames[i].lumpNum == lumpNum ) { if ( g_bFriendlyNames ) { return g_lumpNames[i].pFriendlyName; } else { return g_lumpNames[i].pName; } } }
return "???"; }
//-----------------------------------------------------------------------------
// Extract Lump Pak
//-----------------------------------------------------------------------------
void ExtractZip( const char *pFilename, void *pBSPFile ) { dheader_t *pBSPHeader = (dheader_t *)pBSPFile;
if ( pBSPHeader->lumps[LUMP_PAKFILE].filelen ) { Msg( "Extracting Zip to %s\n", pFilename );
FILE *fp = fopen( pFilename, "wb" ); if ( !fp ) { Warning( "Failed to create %s\n", pFilename ); return; }
fwrite( (unsigned char *)pBSPFile + pBSPHeader->lumps[LUMP_PAKFILE].fileofs, pBSPHeader->lumps[LUMP_PAKFILE].filelen, 1, fp ); fclose( fp ); } else { Msg( "Nothing to do!\n" ); } }
// compare function for qsort below
static dheader_t *g_pSortBSPHeader; static int LumpCompare( const void *pElem1, const void *pElem2 ) { int lump1 = *(byte *)pElem1; int lump2 = *(byte *)pElem2;
int fileOffset1 = g_pSortBSPHeader->lumps[lump1].fileofs; int fileOffset2 = g_pSortBSPHeader->lumps[lump2].fileofs;
int fileSize1 = g_pSortBSPHeader->lumps[lump1].filelen; int fileSize2 = g_pSortBSPHeader->lumps[lump2].filelen;
if ( g_bSortByOffset ) { // invalid or empty lumps will get sorted together
if ( !fileSize1 ) { fileOffset1 = 0; }
if ( !fileSize2 ) { fileOffset2 = 0; }
// compare by offset
if ( fileOffset1 < fileOffset2 ) { return -1; } else if ( fileOffset1 > fileOffset2 ) { return 1; } } else if ( g_bSortBySize ) { if ( fileSize1 < fileSize2 ) { return -1; } else if ( fileSize1 > fileSize2 ) { return 1; } }
return 0; }
//-----------------------------------------------------------------------------
// Spew Info
//-----------------------------------------------------------------------------
void DumpInfo( const char *pFilename, void *pBSPFile, int bspSize ) { const char *pName; dheader_t *pBSPHeader;
pBSPHeader = (dheader_t *)pBSPFile;
Msg( "\n" ); Msg( "%s\n", pFilename );
// sort by offset order
int readOrder[HEADER_LUMPS]; for ( int i=0; i<HEADER_LUMPS; i++ ) { readOrder[i] = i; }
if ( g_bSortByOffset || g_bSortBySize ) { g_pSortBSPHeader = pBSPHeader; qsort( readOrder, HEADER_LUMPS, sizeof( int ), LumpCompare ); }
for ( int i=0; i<HEADER_LUMPS; ++i ) { int lump = readOrder[i]; pName = BSP_LumpNumToName( lump ); if ( !pName ) continue;
if ( g_bSortByOffset ) { Msg( "[Offset: 0x%8.8x] ", pBSPHeader->lumps[lump].fileofs ); }
if ( g_bAsPercent ) { Msg( "%5.2f%s (%2d) %s\n", 100.0f*pBSPHeader->lumps[lump].filelen/( float )bspSize, "%%", lump, pName ); } else if ( g_bAsBytes ) { Msg( "%8d: (%2d) %s\n", pBSPHeader->lumps[lump].filelen, lump, pName ); } else { Msg( "%5.2f MB: (%2d) %s\n", pBSPHeader->lumps[lump].filelen/( 1024.0f*1024.0f ), lump, pName ); } }
Msg( "-------\n" ); if ( g_bAsBytes ) { Msg( "%8d: %s\n", bspSize, "Total Bytes" ); } else { Msg( "%6.2f MB %s\n", bspSize/( 1024.0f*1024.0f ), "Total" ); } }
//-----------------------------------------------------------------------------
// Load the bsp file
//-----------------------------------------------------------------------------
bool LoadBSPFile( const char* pFilename, void **ppBSPBuffer, int *pBSPSize ) { CByteswap byteSwap;
*ppBSPBuffer = NULL; *pBSPSize = 0;
FILE *fp = fopen( pFilename, "rb" ); if ( fp ) { fseek( fp, 0, SEEK_END ); int size = ftell( fp ); fseek( fp, 0, SEEK_SET );
*ppBSPBuffer = malloc( size ); if ( !*ppBSPBuffer ) { Warning( "Failed to alloc %d bytes\n", size ); goto cleanUp; }
*pBSPSize = size; fread( *ppBSPBuffer, size, 1, fp ); fclose( fp ); } else { if ( !g_bQuiet ) { Warning( "Missing %s\n", pFilename ); } goto cleanUp; }
dheader_t *pBSPHeader = (dheader_t *)*ppBSPBuffer;
if ( pBSPHeader->ident != IDBSPHEADER ) { if ( pBSPHeader->ident != BigLong( IDBSPHEADER ) ) { if ( !g_bQuiet ) { Warning( "BSP %s has bad id: got %d, expected %d\n", pFilename, pBSPHeader->ident, IDBSPHEADER ); } goto cleanUp; } else { // bsp is for 360, swap the header
byteSwap.ActivateByteSwapping( true ); byteSwap.SwapFieldsToTargetEndian( pBSPHeader ); } }
if ( pBSPHeader->version < MINBSPVERSION || pBSPHeader->version > BSPVERSION ) { if ( !g_bQuiet ) { Warning( "BSP %s has bad version: got %d, expected %d\n", pFilename, pBSPHeader->version, BSPVERSION ); } goto cleanUp; }
// sucess
return true;
cleanUp: if ( *ppBSPBuffer ) { free( *ppBSPBuffer ); *ppBSPBuffer = NULL; }
return false; }
//-----------------------------------------------------------------------------
// Purpose: Usage
//-----------------------------------------------------------------------------
void Usage( void ) { Msg( "usage: bspinfo <bspfile> [options]\n" ); Msg( "options:\n" ); Msg( "-percent, -p: Show as percentages\n" ); Msg( "-bytes, -b: Show as bytes\n" ); Msg( "-q: Quiet, no header, no errors\n" ); Msg( "-names: Show friendly lump names\n" ); Msg( "-so: Sort by offset\n" ); Msg( "-ss: Sort by size\n" ); Msg( "-extract <zipname>: Extract pak file\n" );
exit( -1 ); }
//-----------------------------------------------------------------------------
// Purpose: default output func
//-----------------------------------------------------------------------------
SpewRetval_t OutputFunc( SpewType_t spewType, char const *pMsg ) { printf( pMsg );
if ( spewType == SPEW_ERROR ) { return SPEW_ABORT; } return ( spewType == SPEW_ASSERT ) ? SPEW_DEBUGGER : SPEW_CONTINUE; }
//-----------------------------------------------------------------------------
// main
//
//-----------------------------------------------------------------------------
int main( int argc, char* argv[] ) { char bspPath[MAX_PATH];
// set the valve library printer
SpewOutputFunc( OutputFunc );
CommandLine()->CreateCmdLine( argc, argv );
Msg( "\nXBSPINFO - Valve Xbox 360 BSP Info ( Build: %s %s )\n", __DATE__, __TIME__ ); Msg( "( C ) Copyright 1996-2006, Valve Corporation, All rights reserved.\n\n" );
if ( argc < 2 || CommandLine()->FindParm( "?" ) || CommandLine()->FindParm( "-h" ) || CommandLine()->FindParm( "-help" ) ) { Usage(); }
if ( argc >= 2 && argv[1][0] != '-' ) { strcpy( bspPath, argv[1] ); } else { Usage(); }
g_bQuiet = CommandLine()->FindParm( "-q" ) != 0; g_bAsPercent = CommandLine()->FindParm( "-p" ) != 0 || CommandLine()->FindParm( "-percent" ) != 0; g_bAsBytes = CommandLine()->FindParm( "-b" ) != 0 || CommandLine()->FindParm( "-bytes" ) != 0; g_bSortByOffset = CommandLine()->FindParm( "-so" ) != 0; g_bSortBySize = CommandLine()->FindParm( "-ss" ) != 0; g_bFriendlyNames = CommandLine()->FindParm( "-names" ) != 0;
void *pBSPBuffer; int bspSize; if ( LoadBSPFile( bspPath, &pBSPBuffer, &bspSize ) ) { const char *pZipName = CommandLine()->ParmValue( "-extract", "" ); if ( pZipName && pZipName[0] ) { ExtractZip( pZipName, pBSPBuffer ); } else { DumpInfo( bspPath, pBSPBuffer, bspSize ); }
free( pBSPBuffer ); }
return ( 0 ); }
|