doom3-bfg/neo/tools/compilers/aas/AASBuild.cpp

1173 lines
25 KiB
C++
Raw Normal View History

/*
===========================================================================
Doom 3 GPL Source Code
Copyright (C) 1999-2011 id Software LLC, a ZeniMax Media company.
This file is part of the Doom 3 GPL Source Code (?Doom 3 Source Code?).
Doom 3 Source Code is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
Doom 3 Source Code is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with Doom 3 Source Code. If not, see <http://www.gnu.org/licenses/>.
In addition, the Doom 3 Source Code is also subject to certain additional terms. You should have received a copy of these additional terms immediately following the terms and conditions of the GNU General Public License which accompanied the Doom 3 Source Code. If not, please request a copy in writing from id Software at the address below.
If you have questions concerning this license or the applicable additional terms, you may contact in writing id Software LLC, c/o ZeniMax Media Inc., Suite 120, Rockville, Maryland 20850 USA.
===========================================================================
*/
#include "precompiled.h"
#pragma hdrstop
#include "AASBuild_local.h"
#define BFL_PATCH 0x1000
//===============================================================
//
// idAASBuild
//
//===============================================================
/*
============
idAASBuild::idAASBuild
============
*/
2014-08-02 12:48:04 +00:00
idAASBuild::idAASBuild()
{
file = NULL;
procNodes = NULL;
numProcNodes = 0;
numGravitationalSubdivisions = 0;
numMergedLeafNodes = 0;
numLedgeSubdivisions = 0;
ledgeMap = NULL;
}
/*
============
idAASBuild::~idAASBuild
============
*/
2014-08-02 12:48:04 +00:00
idAASBuild::~idAASBuild()
{
Shutdown();
}
/*
================
idAASBuild::Shutdown
================
*/
2014-08-02 12:48:04 +00:00
void idAASBuild::Shutdown()
{
aasSettings = NULL;
if( file )
{
delete file;
file = NULL;
}
DeleteProcBSP();
numGravitationalSubdivisions = 0;
numMergedLeafNodes = 0;
numLedgeSubdivisions = 0;
ledgeList.Clear();
if( ledgeMap )
{
delete ledgeMap;
ledgeMap = NULL;
}
}
/*
================
idAASBuild::ParseProcNodes
================
*/
void idAASBuild::ParseProcNodes( idLexer* src )
{
int i;
src->ExpectTokenString( "{" );
idAASBuild::numProcNodes = src->ParseInt();
if( idAASBuild::numProcNodes < 0 )
{
src->Error( "idAASBuild::ParseProcNodes: bad numProcNodes" );
}
idAASBuild::procNodes = ( aasProcNode_t* )Mem_ClearedAlloc( idAASBuild::numProcNodes * sizeof( aasProcNode_t ), TAG_TOOLS );
for( i = 0; i < idAASBuild::numProcNodes; i++ )
{
aasProcNode_t* node;
node = &( idAASBuild::procNodes[i] );
src->Parse1DMatrix( 4, node->plane.ToFloatPtr() );
node->children[0] = src->ParseInt();
node->children[1] = src->ParseInt();
}
src->ExpectTokenString( "}" );
}
/*
================
idAASBuild::LoadProcBSP
================
*/
bool idAASBuild::LoadProcBSP( const char* name, ID_TIME_T minFileTime )
{
idStr fileName;
idToken token;
idLexer* src;
// load it
fileName = name;
fileName.SetFileExtension( PROC_FILE_EXT );
src = new idLexer( fileName, LEXFL_NOSTRINGCONCAT | LEXFL_NODOLLARPRECOMPILE );
if( !src->IsLoaded() )
{
common->Warning( "idAASBuild::LoadProcBSP: couldn't load %s", fileName.c_str() );
delete src;
return false;
}
// if the file is too old
if( src->GetFileTime() < minFileTime )
{
delete src;
return false;
}
if( !src->ReadToken( &token ) || token.Icmp( PROC_FILE_ID ) )
{
common->Warning( "idAASBuild::LoadProcBSP: bad id '%s' instead of '%s'", token.c_str(), PROC_FILE_ID );
delete src;
return false;
}
// parse the file
while( 1 )
{
if( !src->ReadToken( &token ) )
{
break;
}
if( token == "model" )
{
src->SkipBracedSection();
continue;
}
if( token == "shadowModel" )
{
src->SkipBracedSection();
continue;
}
if( token == "interAreaPortals" )
{
src->SkipBracedSection();
continue;
}
if( token == "nodes" )
{
idAASBuild::ParseProcNodes( src );
break;
}
src->Error( "idAASBuild::LoadProcBSP: bad token \"%s\"", token.c_str() );
}
delete src;
return true;
}
/*
============
idAASBuild::DeleteProcBSP
============
*/
2014-08-02 12:48:04 +00:00
void idAASBuild::DeleteProcBSP()
{
if( procNodes )
{
Mem_Free( procNodes );
procNodes = NULL;
}
numProcNodes = 0;
}
/*
============
idAASBuild::ChoppedAwayByProcBSP
============
*/
bool idAASBuild::ChoppedAwayByProcBSP( int nodeNum, idFixedWinding* w, const idVec3& normal, const idVec3& origin, const float radius )
{
int res;
idFixedWinding back;
aasProcNode_t* node;
float dist;
do
{
node = idAASBuild::procNodes + nodeNum;
dist = node->plane.Normal() * origin + node->plane[3];
if( dist > radius )
{
res = SIDE_FRONT;
}
else if( dist < -radius )
{
res = SIDE_BACK;
}
else
{
res = w->Split( &back, node->plane, ON_EPSILON );
}
if( res == SIDE_FRONT )
{
nodeNum = node->children[0];
}
else if( res == SIDE_BACK )
{
nodeNum = node->children[1];
}
else if( res == SIDE_ON )
{
// continue with the side the winding faces
if( node->plane.Normal() * normal > 0.0f )
{
nodeNum = node->children[0];
}
else
{
nodeNum = node->children[1];
}
}
else
{
// if either node is not solid
if( node->children[0] < 0 || node->children[1] < 0 )
{
return false;
}
// only recurse if the node is not solid
if( node->children[1] > 0 )
{
if( !idAASBuild::ChoppedAwayByProcBSP( node->children[1], &back, normal, origin, radius ) )
{
return false;
}
}
nodeNum = node->children[0];
}
}
while( nodeNum > 0 );
if( nodeNum < 0 )
{
return false;
}
return true;
}
/*
============
idAASBuild::ClipBrushSidesWithProcBSP
============
*/
void idAASBuild::ClipBrushSidesWithProcBSP( idBrushList& brushList )
{
int i, clippedSides;
idBrush* brush;
idFixedWinding neww;
idBounds bounds;
float radius;
idVec3 origin;
// if the .proc file has no BSP tree
if( idAASBuild::procNodes == NULL )
{
return;
}
clippedSides = 0;
for( brush = brushList.Head(); brush; brush = brush->Next() )
{
for( i = 0; i < brush->GetNumSides(); i++ )
{
if( !brush->GetSide( i )->GetWinding() )
{
continue;
}
// make a local copy of the winding
neww = *brush->GetSide( i )->GetWinding();
neww.GetBounds( bounds );
origin = ( bounds[1] - bounds[0] ) * 0.5f;
radius = origin.Length() + ON_EPSILON;
origin = bounds[0] + origin;
if( ChoppedAwayByProcBSP( 0, &neww, brush->GetSide( i )->GetPlane().Normal(), origin, radius ) )
{
brush->GetSide( i )->SetFlag( SFL_USED_SPLITTER );
clippedSides++;
}
}
}
common->Printf( "%6d brush sides clipped\n", clippedSides );
}
/*
============
idAASBuild::ContentsForAAS
============
*/
int idAASBuild::ContentsForAAS( int contents )
{
int c;
if( contents & ( CONTENTS_SOLID | CONTENTS_AAS_SOLID | CONTENTS_MONSTERCLIP ) )
{
return AREACONTENTS_SOLID;
}
c = 0;
if( contents & CONTENTS_WATER )
{
c |= AREACONTENTS_WATER;
}
if( contents & CONTENTS_AREAPORTAL )
{
c |= AREACONTENTS_CLUSTERPORTAL;
}
if( contents & CONTENTS_AAS_OBSTACLE )
{
c |= AREACONTENTS_OBSTACLE;
}
return c;
}
/*
============
idAASBuild::AddBrushForMapBrush
============
*/
idBrushList idAASBuild::AddBrushesForMapBrush( const idMapBrush* mapBrush, const idVec3& origin, const idMat3& axis, int entityNum, int primitiveNum, idBrushList brushList )
{
int contents, i;
idMapBrushSide* mapSide;
const idMaterial* mat;
idList<idBrushSide*> sideList;
idBrush* brush;
idPlane plane;
contents = 0;
for( i = 0; i < mapBrush->GetNumSides(); i++ )
{
mapSide = mapBrush->GetSide( i );
mat = declManager->FindMaterial( mapSide->GetMaterial() );
contents |= mat->GetContentFlags();
plane = mapSide->GetPlane();
plane.FixDegeneracies( DEGENERATE_DIST_EPSILON );
sideList.Append( new idBrushSide( plane, -1 ) );
}
contents = ContentsForAAS( contents );
if( !contents )
{
for( i = 0; i < sideList.Num(); i++ )
{
delete sideList[i];
}
return brushList;
}
brush = new idBrush();
brush->SetContents( contents );
if( !brush->FromSides( sideList ) )
{
common->Warning( "brush primitive %d on entity %d is degenerate", primitiveNum, entityNum );
delete brush;
return brushList;
}
brush->SetEntityNum( entityNum );
brush->SetPrimitiveNum( primitiveNum );
brush->Transform( origin, axis );
brushList.AddToTail( brush );
return brushList;
}
/*
============
idAASBuild::AddBrushesForPatch
============
*/
idBrushList idAASBuild::AddBrushesForMapPatch( const idMapPatch* mapPatch, const idVec3& origin, const idMat3& axis, int entityNum, int primitiveNum, idBrushList brushList )
{
int i, j, contents, validBrushes;
float dot;
int v1, v2, v3, v4;
idFixedWinding w;
idPlane plane;
idVec3 d1, d2;
idBrush* brush;
idSurface_Patch mesh;
const idMaterial* mat;
mat = declManager->FindMaterial( mapPatch->GetMaterial() );
contents = ContentsForAAS( mat->GetContentFlags() );
if( !contents )
{
return brushList;
}
mesh = idSurface_Patch( *mapPatch );
// if the patch has an explicit number of subdivisions use it to avoid cracks
if( mapPatch->GetExplicitlySubdivided() )
{
mesh.SubdivideExplicit( mapPatch->GetHorzSubdivisions(), mapPatch->GetVertSubdivisions(), false, true );
}
else
{
mesh.Subdivide( DEFAULT_CURVE_MAX_ERROR_CD, DEFAULT_CURVE_MAX_ERROR_CD, DEFAULT_CURVE_MAX_LENGTH_CD, false );
}
validBrushes = 0;
for( i = 0; i < mesh.GetWidth() - 1; i++ )
{
for( j = 0; j < mesh.GetHeight() - 1; j++ )
{
v1 = j * mesh.GetWidth() + i;
v2 = v1 + 1;
v3 = v1 + mesh.GetWidth() + 1;
v4 = v1 + mesh.GetWidth();
d1 = mesh[v2].xyz - mesh[v1].xyz;
d2 = mesh[v3].xyz - mesh[v1].xyz;
plane.SetNormal( d1.Cross( d2 ) );
if( plane.Normalize() != 0.0f )
{
plane.FitThroughPoint( mesh[v1].xyz );
dot = plane.Distance( mesh[v4].xyz );
// if we can turn it into a quad
if( idMath::Fabs( dot ) < 0.1f )
{
w.Clear();
w += mesh[v1].xyz;
w += mesh[v2].xyz;
w += mesh[v3].xyz;
w += mesh[v4].xyz;
brush = new idBrush();
brush->SetContents( contents );
if( brush->FromWinding( w, plane ) )
{
brush->SetEntityNum( entityNum );
brush->SetPrimitiveNum( primitiveNum );
brush->SetFlag( BFL_PATCH );
brush->Transform( origin, axis );
brushList.AddToTail( brush );
validBrushes++;
}
else
{
delete brush;
}
continue;
}
else
{
// create one of the triangles
w.Clear();
w += mesh[v1].xyz;
w += mesh[v2].xyz;
w += mesh[v3].xyz;
brush = new idBrush();
brush->SetContents( contents );
if( brush->FromWinding( w, plane ) )
{
brush->SetEntityNum( entityNum );
brush->SetPrimitiveNum( primitiveNum );
brush->SetFlag( BFL_PATCH );
brush->Transform( origin, axis );
brushList.AddToTail( brush );
validBrushes++;
}
else
{
delete brush;
}
}
}
// create the other triangle
d1 = mesh[v3].xyz - mesh[v1].xyz;
d2 = mesh[v4].xyz - mesh[v1].xyz;
plane.SetNormal( d1.Cross( d2 ) );
if( plane.Normalize() != 0.0f )
{
plane.FitThroughPoint( mesh[v1].xyz );
w.Clear();
w += mesh[v1].xyz;
w += mesh[v3].xyz;
w += mesh[v4].xyz;
brush = new idBrush();
brush->SetContents( contents );
if( brush->FromWinding( w, plane ) )
{
brush->SetEntityNum( entityNum );
brush->SetPrimitiveNum( primitiveNum );
brush->SetFlag( BFL_PATCH );
brush->Transform( origin, axis );
brushList.AddToTail( brush );
validBrushes++;
}
else
{
delete brush;
}
}
}
}
if( !validBrushes )
{
common->Warning( "patch primitive %d on entity %d is completely degenerate", primitiveNum, entityNum );
}
return brushList;
}
/*
============
idAASBuild::AddBrushesForMapEntity
============
*/
idBrushList idAASBuild::AddBrushesForMapEntity( const idMapEntity* mapEnt, int entityNum, idBrushList brushList )
{
int i;
idVec3 origin;
idMat3 axis;
if( mapEnt->GetNumPrimitives() < 1 )
{
return brushList;
}
mapEnt->epairs.GetVector( "origin", "0 0 0", origin );
if( !mapEnt->epairs.GetMatrix( "rotation", "1 0 0 0 1 0 0 0 1", axis ) )
{
float angle = mapEnt->epairs.GetFloat( "angle" );
if( angle != 0.0f )
{
axis = idAngles( 0.0f, angle, 0.0f ).ToMat3();
}
else
{
axis.Identity();
}
}
for( i = 0; i < mapEnt->GetNumPrimitives(); i++ )
{
idMapPrimitive* mapPrim;
mapPrim = mapEnt->GetPrimitive( i );
if( mapPrim->GetType() == idMapPrimitive::TYPE_BRUSH )
{
brushList = AddBrushesForMapBrush( static_cast<idMapBrush*>( mapPrim ), origin, axis, entityNum, i, brushList );
continue;
}
if( mapPrim->GetType() == idMapPrimitive::TYPE_PATCH )
{
if( aasSettings->usePatches )
{
brushList = AddBrushesForMapPatch( static_cast<idMapPatch*>( mapPrim ), origin, axis, entityNum, i, brushList );
}
continue;
}
}
return brushList;
}
/*
============
idAASBuild::AddBrushesForMapFile
============
*/
idBrushList idAASBuild::AddBrushesForMapFile( const idMapFile* mapFile, idBrushList brushList )
{
int i;
common->Printf( "[Brush Load]\n" );
brushList = AddBrushesForMapEntity( mapFile->GetEntity( 0 ), 0, brushList );
for( i = 1; i < mapFile->GetNumEntities(); i++ )
{
const char* classname = mapFile->GetEntity( i )->epairs.GetString( "classname" );
if( idStr::Icmp( classname, "func_aas_obstacle" ) == 0 )
{
brushList = AddBrushesForMapEntity( mapFile->GetEntity( i ), i, brushList );
}
}
common->Printf( "%6d brushes\n", brushList.Num() );
return brushList;
}
/*
============
idAASBuild::CheckForEntities
============
*/
bool idAASBuild::CheckForEntities( const idMapFile* mapFile, idStrList& entityClassNames ) const
{
int i;
idStr classname;
for( i = 0; i < mapFile->GetNumEntities(); i++ )
{
if( !mapFile->GetEntity( i )->epairs.GetString( "classname", "", classname ) )
{
continue;
}
if( aasSettings->ValidEntity( classname ) )
{
entityClassNames.AddUnique( classname );
}
}
return ( entityClassNames.Num() != 0 );
}
/*
============
MergeAllowed
============
*/
bool MergeAllowed( idBrush* b1, idBrush* b2 )
{
return ( b1->GetContents() == b2->GetContents() && !( ( b1->GetFlags() | b2->GetFlags() ) & BFL_PATCH ) );
}
/*
============
ExpandedChopAllowed
============
*/
bool ExpandedChopAllowed( idBrush* b1, idBrush* b2 )
{
return ( b1->GetContents() == b2->GetContents() );
}
/*
============
ExpandedMergeAllowed
============
*/
bool ExpandedMergeAllowed( idBrush* b1, idBrush* b2 )
{
return ( b1->GetContents() == b2->GetContents() );
}
/*
============
idAASBuild::ChangeMultipleBoundingBoxContents
============
*/
void idAASBuild::ChangeMultipleBoundingBoxContents_r( idBrushBSPNode* node, int mask )
{
while( node )
{
if( !( node->GetContents() & mask ) )
{
node->SetContents( node->GetContents() & ~AREACONTENTS_SOLID );
}
ChangeMultipleBoundingBoxContents_r( node->GetChild( 0 ), mask );
node = node->GetChild( 1 );
}
}
/*
============
idAASBuild::Build
============
*/
bool idAASBuild::Build( const idStr& fileName, const idAASSettings* settings )
{
int i, bit, mask, startTime;
idMapFile* mapFile;
idBrushList brushList;
idList<idBrushList*> expandedBrushes;
idBrush* b;
idBrushBSP bsp;
idStr name;
idAASReach reach;
idAASCluster cluster;
idStrList entityClassNames;
startTime = Sys_Milliseconds();
Shutdown();
aasSettings = settings;
name = fileName;
name.SetFileExtension( "map" );
mapFile = new idMapFile;
if( !mapFile->Parse( name ) )
{
delete mapFile;
common->Error( "Couldn't load map file: '%s'", name.c_str() );
return false;
}
// check if this map has any entities that use this AAS file
if( !CheckForEntities( mapFile, entityClassNames ) )
{
delete mapFile;
common->Printf( "no entities in map that use %s\n", settings->fileExtension.c_str() );
return true;
}
// load map file brushes
brushList = AddBrushesForMapFile( mapFile, brushList );
// if empty map
if( brushList.Num() == 0 )
{
delete mapFile;
common->Error( "%s is empty", name.c_str() );
return false;
}
// merge as many brushes as possible before expansion
brushList.Merge( MergeAllowed );
// if there is a .proc file newer than the .map file
if( LoadProcBSP( fileName, mapFile->GetFileTime() ) )
{
ClipBrushSidesWithProcBSP( brushList );
DeleteProcBSP();
}
// make copies of the brush list
expandedBrushes.Append( &brushList );
for( i = 1; i < aasSettings->numBoundingBoxes; i++ )
{
expandedBrushes.Append( brushList.Copy() );
}
// expand brushes for the axial bounding boxes
mask = AREACONTENTS_SOLID;
for( i = 0; i < expandedBrushes.Num(); i++ )
{
for( b = expandedBrushes[i]->Head(); b; b = b->Next() )
{
b->ExpandForAxialBox( aasSettings->boundingBoxes[i] );
bit = 1 << ( i + AREACONTENTS_BBOX_BIT );
mask |= bit;
b->SetContents( b->GetContents() | bit );
}
}
// move all brushes back into the original list
for( i = 1; i < aasSettings->numBoundingBoxes; i++ )
{
brushList.AddToTail( *expandedBrushes[i] );
delete expandedBrushes[i];
}
if( aasSettings->writeBrushMap )
{
bsp.WriteBrushMap( fileName, "_" + aasSettings->fileExtension, AREACONTENTS_SOLID );
}
// build BSP tree from brushes
bsp.Build( brushList, AREACONTENTS_SOLID, ExpandedChopAllowed, ExpandedMergeAllowed );
// only solid nodes with all bits set for all bounding boxes need to stay solid
ChangeMultipleBoundingBoxContents_r( bsp.GetRootNode(), mask );
// portalize the bsp tree
bsp.Portalize();
// remove subspaces not reachable by entities
if( !bsp.RemoveOutside( mapFile, AREACONTENTS_SOLID, entityClassNames ) )
{
bsp.LeakFile( name );
delete mapFile;
common->Printf( "%s has no outside", name.c_str() );
return false;
}
// gravitational subdivision
GravitationalSubdivision( bsp );
// merge portals where possible
bsp.MergePortals( AREACONTENTS_SOLID );
// melt portal windings
bsp.MeltPortals( AREACONTENTS_SOLID );
if( aasSettings->writeBrushMap )
{
WriteLedgeMap( fileName, "_" + aasSettings->fileExtension + "_ledge" );
}
// ledge subdivisions
LedgeSubdivision( bsp );
// merge leaf nodes
MergeLeafNodes( bsp );
// merge portals where possible
bsp.MergePortals( AREACONTENTS_SOLID );
// melt portal windings
bsp.MeltPortals( AREACONTENTS_SOLID );
// store the file from the bsp tree
StoreFile( bsp );
file->settings = *aasSettings;
// calculate reachability
reach.Build( mapFile, file );
// build clusters
cluster.Build( file );
// optimize the file
if( !aasSettings->noOptimize )
{
file->Optimize();
}
// write the file
name.SetFileExtension( aasSettings->fileExtension );
file->Write( name, mapFile->GetGeometryCRC() );
// delete the map file
delete mapFile;
common->Printf( "%6d seconds to create AAS\n", ( Sys_Milliseconds() - startTime ) / 1000 );
return true;
}
/*
============
idAASBuild::BuildReachability
============
*/
bool idAASBuild::BuildReachability( const idStr& fileName, const idAASSettings* settings )
{
int startTime;
idMapFile* mapFile;
idStr name;
idAASReach reach;
idAASCluster cluster;
startTime = Sys_Milliseconds();
aasSettings = settings;
name = fileName;
name.SetFileExtension( "map" );
mapFile = new idMapFile;
if( !mapFile->Parse( name ) )
{
delete mapFile;
common->Error( "Couldn't load map file: '%s'", name.c_str() );
return false;
}
file = new idAASFileLocal();
name.SetFileExtension( aasSettings->fileExtension );
if( !file->Load( name, 0 ) )
{
delete mapFile;
common->Error( "Couldn't load AAS file: '%s'", name.c_str() );
return false;
}
file->settings = *aasSettings;
// calculate reachability
reach.Build( mapFile, file );
// build clusters
cluster.Build( file );
// write the file
file->Write( name, mapFile->GetGeometryCRC() );
// delete the map file
delete mapFile;
common->Printf( "%6d seconds to calculate reachability\n", ( Sys_Milliseconds() - startTime ) / 1000 );
return true;
}
/*
============
ParseOptions
============
*/
int ParseOptions( const idCmdArgs& args, idAASSettings& settings )
{
int i;
idStr str;
for( i = 1; i < args.Argc(); i++ )
{
str = args.Argv( i );
str.StripLeading( '-' );
if( str.Icmp( "usePatches" ) == 0 )
{
settings.usePatches = true;
common->Printf( "usePatches = true\n" );
}
else if( str.Icmp( "writeBrushMap" ) == 0 )
{
settings.writeBrushMap = true;
common->Printf( "writeBrushMap = true\n" );
}
else if( str.Icmp( "playerFlood" ) == 0 )
{
settings.playerFlood = true;
common->Printf( "playerFlood = true\n" );
}
else if( str.Icmp( "noOptimize" ) == 0 )
{
settings.noOptimize = true;
common->Printf( "noOptimize = true\n" );
}
}
return args.Argc() - 1;
}
/*
============
RunAAS_f
============
*/
void RunAAS_f( const idCmdArgs& args )
{
int i;
idAASBuild aas;
idAASSettings settings;
idStr mapName;
if( args.Argc() <= 1 )
{
common->Printf( "runAAS [options] <mapfile>\n"
"options:\n"
" -usePatches = use bezier patches for collision detection.\n"
" -writeBrushMap = write a brush map with the AAS geometry.\n"
" -playerFlood = use player spawn points as valid AAS positions.\n" );
return;
}
common->ClearWarnings( "compiling AAS" );
common->SetRefreshOnPrint( true );
// get the aas settings definitions
const idDict* dict = gameEdit->FindEntityDefDict( "aas_types", false );
if( !dict )
{
common->Error( "Unable to find entityDef for 'aas_types'" );
}
const idKeyValue* kv = dict->MatchPrefix( "type" );
while( kv != NULL )
{
const idDict* settingsDict = gameEdit->FindEntityDefDict( kv->GetValue(), false );
if( !settingsDict )
{
common->Warning( "Unable to find '%s' in def/aas.def", kv->GetValue().c_str() );
}
else
{
settings.FromDict( kv->GetValue(), settingsDict );
i = ParseOptions( args, settings );
mapName = args.Argv( i );
mapName.BackSlashesToSlashes();
if( mapName.Icmpn( "maps/", 4 ) != 0 )
{
mapName = "maps/" + mapName;
}
aas.Build( mapName, &settings );
}
kv = dict->MatchPrefix( "type", kv );
if( kv )
{
common->Printf( "=======================================================\n" );
}
}
common->SetRefreshOnPrint( false );
common->PrintWarnings();
}
/*
============
RunAASDir_f
============
*/
void RunAASDir_f( const idCmdArgs& args )
{
int i;
idAASBuild aas;
idAASSettings settings;
idFileList* mapFiles;
if( args.Argc() <= 1 )
{
common->Printf( "runAASDir <folder>\n" );
return;
}
common->ClearWarnings( "compiling AAS" );
common->SetRefreshOnPrint( true );
// get the aas settings definitions
const idDict* dict = gameEdit->FindEntityDefDict( "aas_types", false );
if( !dict )
{
common->Error( "Unable to find entityDef for 'aas_types'" );
}
// scan for .map files
mapFiles = fileSystem->ListFiles( idStr( "maps/" ) + args.Argv( 1 ), ".map" );
// create AAS files for all the .map files
for( i = 0; i < mapFiles->GetNumFiles(); i++ )
{
if( i )
{
common->Printf( "=======================================================\n" );
}
const idKeyValue* kv = dict->MatchPrefix( "type" );
while( kv != NULL )
{
const idDict* settingsDict = gameEdit->FindEntityDefDict( kv->GetValue(), false );
if( !settingsDict )
{
common->Warning( "Unable to find '%s' in def/aas.def", kv->GetValue().c_str() );
}
else
{
settings.FromDict( kv->GetValue(), settingsDict );
aas.Build( idStr( "maps/" ) + args.Argv( 1 ) + "/" + mapFiles->GetFile( i ), &settings );
}
kv = dict->MatchPrefix( "type", kv );
if( kv )
{
common->Printf( "=======================================================\n" );
}
}
}
fileSystem->FreeFileList( mapFiles );
common->SetRefreshOnPrint( false );
common->PrintWarnings();
}
/*
============
RunReach_f
============
*/
void RunReach_f( const idCmdArgs& args )
{
int i;
idAASBuild aas;
idAASSettings settings;
if( args.Argc() <= 1 )
{
common->Printf( "runReach [options] <mapfile>\n" );
return;
}
common->ClearWarnings( "calculating AAS reachability" );
common->SetRefreshOnPrint( true );
// get the aas settings definitions
const idDict* dict = gameEdit->FindEntityDefDict( "aas_types", false );
if( !dict )
{
common->Error( "Unable to find entityDef for 'aas_types'" );
}
const idKeyValue* kv = dict->MatchPrefix( "type" );
while( kv != NULL )
{
const idDict* settingsDict = gameEdit->FindEntityDefDict( kv->GetValue(), false );
if( !settingsDict )
{
common->Warning( "Unable to find '%s' in def/aas.def", kv->GetValue().c_str() );
}
else
{
settings.FromDict( kv->GetValue(), settingsDict );
i = ParseOptions( args, settings );
aas.BuildReachability( idStr( "maps/" ) + args.Argv( i ), &settings );
}
kv = dict->MatchPrefix( "type", kv );
if( kv )
{
common->Printf( "=======================================================\n" );
}
}
common->SetRefreshOnPrint( false );
common->PrintWarnings();
}