2022-02-27 07:53:44 -05:00
# include <STDInclude.hpp>
2023-01-03 07:16:44 -05:00
# include <Utils/Compression.hpp>
2022-12-26 07:07:24 -05:00
# include "Console.hpp"
2023-01-03 07:16:44 -05:00
# include "FastFiles.hpp"
2022-11-26 12:38:34 -05:00
# include <version.hpp>
2022-11-16 12:25:21 -05:00
# include "AssetInterfaces/ILocalizeEntry.hpp"
2017-01-19 16:23:59 -05:00
namespace Components
{
std : : string ZoneBuilder : : TraceZone ;
std : : vector < std : : pair < Game : : XAssetType , std : : string > > ZoneBuilder : : TraceAssets ;
2017-05-27 09:38:12 -04:00
bool ZoneBuilder : : MainThreadInterrupted ;
DWORD ZoneBuilder : : InterruptingThreadId ;
2022-12-18 16:47:59 -05:00
volatile bool ZoneBuilder : : CommandThreadTerminate = false ;
2017-05-27 09:38:12 -04:00
std : : thread ZoneBuilder : : CommandThread ;
2018-12-17 08:29:18 -05:00
ZoneBuilder : : Zone : : Zone ( const std : : string & name ) : indexStart ( 0 ) , externalSize ( 0 ) ,
2017-03-26 09:47:53 -04:00
// Reserve 100MB by default.
// That's totally fine, as the dedi doesn't load images and therefore doesn't need much memory.
// That way we can be sure it won't need to reallocate memory.
// Side note: if you need a fastfile larger than 100MB, you're doing it wrong-
// Well, decompressed maps can get way larger than 100MB, so let's increase that.
buffer ( 0xC800000 ) ,
2023-02-15 06:58:31 -05:00
zoneName ( name ) ,
dataMap ( " zone_source/ " + name + " .csv " ) ,
branding { nullptr } ,
2023-02-17 06:26:40 -05:00
assetDepth ( 0 ) ,
iw4ofApi ( getIW4OfApiParams ( ) )
2022-07-06 11:48:40 -04:00
{
}
2017-01-19 16:23:59 -05:00
ZoneBuilder : : Zone : : ~ Zone ( )
{
# ifdef DEBUG
for ( auto & subAsset : this - > loadedSubAssets )
{
bool found = false ;
std : : string name = Game : : DB_GetXAssetName ( & subAsset ) ;
for ( auto & alias : this - > aliasList )
{
if ( subAsset . type = = alias . first . type & & name = = Game : : DB_GetXAssetName ( & alias . first ) )
{
found = true ;
break ;
}
}
if ( ! found )
{
2022-12-17 10:52:00 -05:00
Logger : : Print ( " Asset {} of type {} was loaded, but not written! \n " , name , Game : : DB_GetXAssetTypeName ( subAsset . type ) ) ;
2017-01-19 16:23:59 -05:00
}
}
for ( auto & alias : this - > aliasList )
{
bool found = false ;
std : : string name = Game : : DB_GetXAssetName ( & alias . first ) ;
for ( auto & subAsset : this - > loadedSubAssets )
{
if ( subAsset . type = = alias . first . type & & name = = Game : : DB_GetXAssetName ( & subAsset ) )
{
found = true ;
break ;
}
}
if ( ! found )
{
2022-12-17 10:52:00 -05:00
Logger : : Error ( Game : : ERR_FATAL , " Asset {} of type {} was written, but not loaded! \n " , name , Game : : DB_GetXAssetTypeName ( alias . first . type ) ) ;
2017-01-19 16:23:59 -05:00
}
}
# endif
// Unload our fastfiles
2022-12-17 10:52:00 -05:00
Game : : XZoneInfo info { } ;
2017-01-19 16:23:59 -05:00
info . name = nullptr ;
info . allocFlags = 0 ;
info . freeFlags = 0x20 ;
2022-12-17 10:52:00 -05:00
2017-01-19 16:23:59 -05:00
Game : : DB_LoadXAssets ( & info , 1 , true ) ;
AssetHandler : : ClearTemporaryAssets ( ) ;
}
Utils : : Stream * ZoneBuilder : : Zone : : getBuffer ( )
{
return & this - > buffer ;
}
Utils : : Memory : : Allocator * ZoneBuilder : : Zone : : getAllocator ( )
{
return & this - > memAllocator ;
}
2023-02-15 06:58:31 -05:00
iw4of : : api * ZoneBuilder : : Zone : : getIW4OfApi ( )
{
return & iw4ofApi ;
}
2017-01-19 16:23:59 -05:00
void ZoneBuilder : : Zone : : Zone : : build ( )
{
2022-06-30 15:37:47 -04:00
if ( ! this - > dataMap . isValid ( ) )
2017-04-04 16:35:30 -04:00
{
2022-06-12 17:07:53 -04:00
Logger : : Print ( " Unable to load CSV for '{}'! \n " , this - > zoneName ) ;
2017-04-04 16:35:30 -04:00
return ;
}
2017-01-19 16:23:59 -05:00
this - > loadFastFiles ( ) ;
Logger : : Print ( " Linking assets... \n " ) ;
if ( ! this - > loadAssets ( ) ) return ;
this - > addBranding ( ) ;
Logger : : Print ( " Saving... \n " ) ;
this - > saveData ( ) ;
2022-06-30 15:37:47 -04:00
if ( this - > buffer . hasBlock ( ) )
2017-03-20 16:07:55 -04:00
{
2022-06-12 17:07:53 -04:00
Logger : : Error ( Game : : ERR_FATAL , " Non-popped blocks left! \n " ) ;
2017-03-20 16:07:55 -04:00
}
2017-01-19 16:23:59 -05:00
Logger : : Print ( " Compressing... \n " ) ;
this - > writeZone ( ) ;
}
2022-12-18 16:47:59 -05:00
void ZoneBuilder : : Zone : : loadFastFiles ( ) const
2017-01-19 16:23:59 -05:00
{
Logger : : Print ( " Loading required FastFiles... \n " ) ;
2022-07-06 11:48:40 -04:00
for ( std : : size_t i = 0 ; i < this - > dataMap . getRows ( ) ; + + i )
2017-01-19 16:23:59 -05:00
{
if ( this - > dataMap . getElementAt ( i , 0 ) = = " require " )
{
std : : string fastfile = this - > dataMap . getElementAt ( i , 1 ) ;
if ( ! Game : : DB_IsZoneLoaded ( fastfile . data ( ) ) )
{
Game : : XZoneInfo info ;
info . name = fastfile . data ( ) ;
info . allocFlags = 0x20 ;
info . freeFlags = 0 ;
Game : : DB_LoadXAssets ( & info , 1 , true ) ;
}
else
{
2022-06-12 17:07:53 -04:00
Logger : : Print ( " Zone '{}' already loaded \n " , fastfile ) ;
2017-01-19 16:23:59 -05:00
}
}
}
}
bool ZoneBuilder : : Zone : : loadAssets ( )
{
2022-07-06 11:48:40 -04:00
for ( std : : size_t i = 0 ; i < this - > dataMap . getRows ( ) ; + + i )
2017-01-19 16:23:59 -05:00
{
2022-11-16 12:25:21 -05:00
if ( this - > dataMap . getElementAt ( i , 0 ) = = " require " s )
2017-01-19 16:23:59 -05:00
{
2022-11-16 12:25:21 -05:00
continue ;
}
if ( this - > dataMap . getElementAt ( i , 0 ) = = " localize " s )
{
const auto filename = this - > dataMap . getElementAt ( i , 1 ) ;
2022-12-03 19:15:46 -05:00
if ( FileSystem : : File file = std : : format ( " localizedstrings/{}.str " , filename ) )
2017-01-19 16:23:59 -05:00
{
2022-11-16 12:25:21 -05:00
Assets : : ILocalizeEntry : : ParseLocalizedStringsFile ( this , filename , file . getName ( ) ) ;
continue ;
2017-01-19 16:23:59 -05:00
}
2022-12-03 19:15:46 -05:00
if ( FileSystem : : File file = std : : format ( " localizedstrings/{}.json " , filename ) )
{
Assets : : ILocalizeEntry : : ParseLocalizedStringsJson ( this , file ) ;
continue ;
}
2022-11-16 12:25:21 -05:00
}
2017-01-19 16:23:59 -05:00
2022-11-16 12:25:21 -05:00
if ( this - > dataMap . getColumns ( i ) > 2 )
{
auto oldName = this - > dataMap . getElementAt ( i , 1 ) ;
auto newName = this - > dataMap . getElementAt ( i , 2 ) ;
auto typeName = this - > dataMap . getElementAt ( i , 0 ) ;
auto type = Game : : DB_GetXAssetNameType ( typeName . data ( ) ) ;
if ( type < Game : : XAssetType : : ASSET_TYPE_COUNT & & type > = 0 )
{
this - > renameAsset ( type , oldName , newName ) ;
}
else
2017-01-19 16:23:59 -05:00
{
2022-11-16 12:25:21 -05:00
Logger : : Error ( Game : : ERR_FATAL , " Unable to rename '{}' to '{}' as the asset type '{}' is invalid! " , oldName , newName , typeName ) ;
2017-01-19 16:23:59 -05:00
}
}
2022-11-16 12:25:21 -05:00
if ( ! this - > loadAssetByName ( this - > dataMap . getElementAt ( i , 0 ) , this - > dataMap . getElementAt ( i , 1 ) , false ) )
{
return false ;
}
2017-01-19 16:23:59 -05:00
}
return true ;
}
bool ZoneBuilder : : Zone : : loadAsset ( Game : : XAssetType type , void * data , bool isSubAsset )
{
Game : : XAsset asset { type , { data } } ;
const char * name = Game : : DB_GetXAssetName ( & asset ) ;
2017-03-20 05:38:30 -04:00
if ( name ) return this - > loadAssetByName ( type , std : : string ( name ) , isSubAsset ) ;
2017-01-19 16:23:59 -05:00
else return false ;
}
2018-12-17 08:29:18 -05:00
bool ZoneBuilder : : Zone : : loadAssetByName ( Game : : XAssetType type , const std : : string & name , bool isSubAsset )
2017-01-19 16:23:59 -05:00
{
2017-03-20 05:38:30 -04:00
return this - > loadAssetByName ( Game : : DB_GetXAssetTypeName ( type ) , name , isSubAsset ) ;
2017-01-19 16:23:59 -05:00
}
2018-12-17 08:29:18 -05:00
bool ZoneBuilder : : Zone : : loadAssetByName ( const std : : string & typeName , std : : string name , bool isSubAsset )
2017-01-19 16:23:59 -05:00
{
Game : : XAssetType type = Game : : DB_GetXAssetNameType ( typeName . data ( ) ) ;
2022-11-16 12:25:21 -05:00
if ( name . find ( ' ' , 0 ) ! = std : : string : : npos )
2022-06-12 17:07:53 -04:00
{
2022-11-16 12:25:21 -05:00
Logger : : Warning ( Game : : CON_CHANNEL_DONT_FILTER , " Asset with name '{}' contains spaces. Check your zone source file to ensure this is correct! \n " , name ) ;
2022-06-12 17:07:53 -04:00
}
2019-01-16 12:15:13 -05:00
2017-01-19 16:23:59 -05:00
// Sanitize name for empty assets
if ( name [ 0 ] = = ' , ' ) name . erase ( name . begin ( ) ) ;
2022-12-14 03:40:15 -05:00
// Fix forward slashes for FXEffectDef (and probably other assets)
std : : replace ( name . begin ( ) , name . end ( ) , ' \\ ' , ' / ' ) ;
2017-01-19 16:23:59 -05:00
if ( this - > findAsset ( type , name ) ! = - 1 | | this - > findSubAsset ( type , name ) . data ) return true ;
if ( type = = Game : : XAssetType : : ASSET_TYPE_INVALID | | type > = Game : : XAssetType : : ASSET_TYPE_COUNT )
{
2022-06-12 17:07:53 -04:00
Logger : : Error ( Game : : ERR_FATAL , " Invalid asset type '{}' \n " , typeName ) ;
2017-01-19 16:23:59 -05:00
return false ;
}
Game : : XAssetHeader assetHeader = AssetHandler : : FindAssetForZone ( type , name , this , isSubAsset ) ;
2021-04-05 17:19:40 -04:00
2017-01-19 16:23:59 -05:00
if ( ! assetHeader . data )
2022-10-16 11:17:42 -04:00
{
2022-06-12 17:07:53 -04:00
Logger : : Error ( Game : : ERR_FATAL , " Missing asset '{}' of type '{}' \n " , name , Game : : DB_GetXAssetTypeName ( type ) ) ;
2017-01-19 16:23:59 -05:00
return false ;
}
Game : : XAsset asset ;
asset . type = type ;
asset . header = assetHeader ;
2022-12-14 03:40:15 -05:00
// Handle script strings
AssetHandler : : ZoneMark ( asset , this ) ;
2017-01-19 16:23:59 -05:00
if ( isSubAsset )
{
this - > loadedSubAssets . push_back ( asset ) ;
}
else
{
this - > loadedAssets . push_back ( asset ) ;
}
return true ;
}
int ZoneBuilder : : Zone : : findAsset ( Game : : XAssetType type , std : : string name )
{
if ( name [ 0 ] = = ' , ' ) name . erase ( name . begin ( ) ) ;
for ( unsigned int i = 0 ; i < this - > loadedAssets . size ( ) ; + + i )
{
Game : : XAsset * asset = & this - > loadedAssets [ i ] ;
if ( asset - > type ! = type ) continue ;
2022-11-28 13:06:42 -05:00
const auto * assetName = Game : : DB_GetXAssetName ( asset ) ;
if ( ! assetName ) return - 1 ;
if ( assetName [ 0 ] = = ' , ' & & assetName [ 1 ] ! = ' \0 ' ) + + assetName ;
2017-01-19 16:23:59 -05:00
2022-06-30 15:37:47 -04:00
if ( this - > getAssetName ( type , assetName ) = = name )
2017-03-25 08:42:32 -04:00
{
return i ;
}
2017-01-19 16:23:59 -05:00
if ( name = = assetName )
{
return i ;
}
}
return - 1 ;
}
Game : : XAssetHeader ZoneBuilder : : Zone : : findSubAsset ( Game : : XAssetType type , std : : string name )
{
if ( name [ 0 ] = = ' , ' ) name . erase ( name . begin ( ) ) ;
for ( unsigned int i = 0 ; i < this - > loadedSubAssets . size ( ) ; + + i )
{
Game : : XAsset * asset = & this - > loadedSubAssets [ i ] ;
if ( asset - > type ! = type ) continue ;
const char * assetName = Game : : DB_GetXAssetName ( asset ) ;
if ( assetName [ 0 ] = = ' , ' ) + + assetName ;
if ( name = = assetName )
{
return asset - > header ;
}
}
2017-01-20 16:41:03 -05:00
return { nullptr } ;
2017-01-19 16:23:59 -05:00
}
Game : : XAsset * ZoneBuilder : : Zone : : getAsset ( int index )
{
if ( static_cast < uint32_t > ( index ) < this - > loadedAssets . size ( ) )
{
return & this - > loadedAssets [ index ] ;
}
return nullptr ;
}
uint32_t ZoneBuilder : : Zone : : getAssetTableOffset ( int index )
{
Utils : : Stream : : Offset offset ;
offset . block = Game : : XFILE_BLOCK_VIRTUAL ;
offset . offset = ( this - > indexStart + ( index * sizeof ( Game : : XAsset ) ) + 4 ) ;
return offset . getPackedOffset ( ) ;
}
bool ZoneBuilder : : Zone : : hasAlias ( Game : : XAsset asset )
{
return this - > getAlias ( asset ) ! = 0 ;
}
Game : : XAssetHeader ZoneBuilder : : Zone : : saveSubAsset ( Game : : XAssetType type , void * ptr )
{
Game : : XAssetHeader header { ptr } ;
Game : : XAsset asset { type , header } ;
std : : string name = Game : : DB_GetXAssetName ( & asset ) ;
int assetIndex = this - > findAsset ( type , name ) ;
if ( assetIndex = = - 1 ) // nested asset
{
// already written. find alias and store in ptr
2022-06-30 15:37:47 -04:00
if ( this - > hasAlias ( asset ) )
2017-01-19 16:23:59 -05:00
{
header . data = reinterpret_cast < void * > ( this - > getAlias ( asset ) ) ;
}
else
{
asset . header = this - > findSubAsset ( type , name ) ;
if ( ! asset . header . data )
{
2022-06-12 17:07:53 -04:00
Logger : : Error ( Game : : ERR_FATAL , " Missing required asset '{}' ({}). Export failed! " , name , Game : : DB_GetXAssetTypeName ( type ) ) ;
2017-01-19 16:23:59 -05:00
}
// we alias the next 4 (aligned) bytes of the stream b/c DB_InsertPointer gives us a nice pointer to use as the alias
// otherwise it would be a fuckfest trying to figure out where the alias is in the stream
this - > buffer . pushBlock ( Game : : XFILE_BLOCK_VIRTUAL ) ;
this - > buffer . align ( Utils : : Stream : : ALIGN_4 ) ;
this - > storeAlias ( asset ) ;
this - > buffer . increaseBlockSize ( 4 ) ;
this - > buffer . popBlock ( ) ;
this - > buffer . pushBlock ( Game : : XFILE_BLOCK_TEMP ) ;
this - > buffer . align ( Utils : : Stream : : ALIGN_4 ) ;
AssetHandler : : ZoneSave ( asset , this ) ;
this - > buffer . popBlock ( ) ;
header . data = reinterpret_cast < void * > ( - 2 ) ; // DB_InsertPointer marker
}
}
else
{
// asset was written normally. not sure this is even possible but its here
header . data = reinterpret_cast < void * > ( this - > getAssetTableOffset ( assetIndex ) ) ;
}
return header ;
}
void ZoneBuilder : : Zone : : writeZone ( )
{
FILETIME fileTime ;
GetSystemTimeAsFileTime ( & fileTime ) ;
Game : : XFileHeader header =
{
2021-07-10 05:30:35 -04:00
# ifndef GENERATE_IW4X_SPECIFIC_ZONES
2017-01-19 16:23:59 -05:00
XFILE_MAGIC_UNSIGNED ,
# else
XFILE_HEADER_IW4X | ( static_cast < unsigned __int64 > ( XFILE_VERSION_IW4X ) < < 32 ) ,
# endif
XFILE_VERSION ,
Game : : XFileLanguage : : XLANG_NONE ,
fileTime . dwHighDateTime ,
fileTime . dwLowDateTime
} ;
std : : string outBuffer ;
outBuffer . append ( reinterpret_cast < char * > ( & header ) , sizeof ( header ) ) ;
std : : string zoneBuffer = this - > buffer . toBuffer ( ) ;
2021-07-10 05:30:35 -04:00
# ifdef GENERATE_IW4X_SPECIFIC_ZONES
2017-01-19 16:23:59 -05:00
// Insert a random byte, this will destroy the whole alignment and result in a crash, if not handled
zoneBuffer . insert ( zoneBuffer . begin ( ) , static_cast < char > ( Utils : : Cryptography : : Rand : : GenerateInt ( ) ) ) ;
2017-04-27 16:03:05 -04:00
char lastByte = 0 ;
for ( unsigned int i = 0 ; i < zoneBuffer . size ( ) ; + + i )
{
char oldLastByte = lastByte ;
lastByte = zoneBuffer [ i ] ;
Utils : : RotLeft ( zoneBuffer [ i ] , 6 ) ;
zoneBuffer [ i ] ^ = - 1 ;
Utils : : RotRight ( zoneBuffer [ i ] , 4 ) ;
zoneBuffer [ i ] ^ = oldLastByte ;
}
2017-01-19 16:23:59 -05:00
# endif
2022-06-12 17:07:53 -04:00
Utils : : IO : : WriteFile ( " uncompressed " , zoneBuffer ) ;
2022-12-02 14:18:04 -05:00
const auto _0 = gsl : : finally ( [ ]
{
Utils : : IO : : RemoveFile ( " uncompressed " ) ;
} ) ;
2019-01-15 21:25:40 -05:00
2017-07-03 10:32:10 -04:00
zoneBuffer = Utils : : Compression : : ZLib : : Compress ( zoneBuffer ) ;
2017-01-19 16:23:59 -05:00
outBuffer . append ( zoneBuffer ) ;
std : : string outFile = " zone/ " + this - > zoneName + " .ff " ;
Utils : : IO : : WriteFile ( outFile , outBuffer ) ;
Logger : : Print ( " done. \n " ) ;
2022-11-16 12:25:21 -05:00
Logger : : Print ( " Zone '{}' written with {} assets and {} script strings \n " , outFile , ( this - > aliasList . size ( ) + this - > loadedAssets . size ( ) ) , this - > scriptStrings . size ( ) ) ;
2017-01-19 16:23:59 -05:00
}
void ZoneBuilder : : Zone : : saveData ( )
{
// Add header
Game : : ZoneHeader zoneHeader = { 0 } ;
zoneHeader . assetList . assetCount = this - > loadedAssets . size ( ) ;
Utils : : Stream : : ClearPointer ( & zoneHeader . assetList . assets ) ;
// Increment ScriptStrings count (for empty script string) if available
2022-12-14 03:40:15 -05:00
zoneHeader . assetList . stringList . count = this - > scriptStrings . size ( ) + 1 ;
Utils : : Stream : : ClearPointer ( & zoneHeader . assetList . stringList . strings ) ;
2017-01-19 16:23:59 -05:00
// Write header
this - > buffer . save ( & zoneHeader , sizeof ( Game : : ZoneHeader ) ) ;
this - > buffer . pushBlock ( Game : : XFILE_BLOCK_VIRTUAL ) ; // Push main stream onto the stream stack
// Write ScriptStrings, if available
2022-12-14 03:40:15 -05:00
this - > buffer . saveNull ( 4 ) ;
// Empty script string?
// This actually represents a NULL string, but as scriptString.
// So scriptString loading for NULL scriptStrings from fastfile results in a NULL scriptString.
// That's the reason why the count is incremented by 1, if scriptStrings are available.
// Write ScriptString pointer table
for ( std : : size_t i = 0 ; i < this - > scriptStrings . size ( ) ; + + i )
2017-01-19 16:23:59 -05:00
{
2022-12-14 03:40:15 -05:00
this - > buffer . saveMax ( 4 ) ;
}
2017-01-19 16:23:59 -05:00
2022-12-14 03:40:15 -05:00
// Write ScriptStrings
for ( auto ScriptString : this - > scriptStrings )
{
this - > buffer . saveString ( ScriptString . data ( ) ) ;
2017-01-19 16:23:59 -05:00
}
// Align buffer (4 bytes) to get correct offsets for pointers
this - > buffer . align ( Utils : : Stream : : ALIGN_4 ) ;
this - > indexStart = this - > buffer . getBlockSize ( Game : : XFILE_BLOCK_VIRTUAL ) ; // Mark AssetTable offset
// AssetTable
for ( auto asset : this - > loadedAssets )
{
2017-01-20 16:41:03 -05:00
Game : : XAsset entry = { asset . type , nullptr } ;
2017-01-19 16:23:59 -05:00
Utils : : Stream : : ClearPointer ( & entry . header . data ) ;
this - > buffer . save ( & entry ) ;
}
// Assets
for ( auto asset : this - > loadedAssets )
{
this - > buffer . pushBlock ( Game : : XFILE_BLOCK_TEMP ) ;
this - > buffer . align ( Utils : : Stream : : ALIGN_4 ) ;
this - > store ( asset . header ) ;
AssetHandler : : ZoneSave ( asset , this ) ;
this - > buffer . popBlock ( ) ;
}
// Adapt header
this - > buffer . enterCriticalSection ( ) ;
Game : : XFile * header = reinterpret_cast < Game : : XFile * > ( this - > buffer . data ( ) ) ;
header - > size = this - > buffer . length ( ) - sizeof ( Game : : XFile ) ; // Write correct data size
header - > externalSize = this - > externalSize ; // This actually stores how much external data has to be loaded. It's used to calculate the loadscreen progress
// Write stream sizes
for ( int i = 0 ; i < Game : : MAX_XFILE_COUNT ; + + i )
{
header - > blockSize [ i ] = this - > buffer . getBlockSize ( static_cast < Game : : XFILE_BLOCK_TYPES > ( i ) ) ;
}
this - > buffer . leaveCriticalSection ( ) ;
this - > buffer . popBlock ( ) ;
}
// Add branding asset
void ZoneBuilder : : Zone : : addBranding ( )
{
2023-02-17 06:26:40 -05:00
const auto now = std : : chrono : : system_clock : : now ( ) ;
2023-02-17 07:43:52 -05:00
auto zoneBranding = std : : format ( " Built using the IW4x ZoneBuilder! {:%d-%m-%Y %H:%M:%OS} " , now ) ;
2023-02-17 06:26:40 -05:00
auto brandingLen = zoneBranding . size ( ) ; // + 1 is added by the save code
2017-01-19 16:23:59 -05:00
2023-02-17 07:43:52 -05:00
this - > branding = { this - > zoneName . data ( ) , 0 , static_cast < int > ( brandingLen ) , getAllocator ( ) - > duplicateString ( zoneBranding ) } ;
2023-02-17 06:26:40 -05:00
if ( this - > findAsset ( Game : : ASSET_TYPE_RAWFILE , this - > branding . name ) ! = - 1 )
2017-01-19 16:23:59 -05:00
{
2022-06-12 17:07:53 -04:00
Logger : : Error ( Game : : ERR_FATAL , " Unable to add branding. Asset '{}' already exists! " , this - > branding . name ) ;
2017-01-19 16:23:59 -05:00
}
Game : : XAssetHeader header = { & this - > branding } ;
2023-02-17 06:26:40 -05:00
Game : : XAsset brandingAsset = { Game : : ASSET_TYPE_RAWFILE , header } ;
2017-01-19 16:23:59 -05:00
this - > loadedAssets . push_back ( brandingAsset ) ;
}
// Check if the given pointer has already been mapped
bool ZoneBuilder : : Zone : : hasPointer ( const void * pointer )
{
2022-05-31 12:38:09 -04:00
return this - > pointerMap . contains ( pointer ) ;
2017-01-19 16:23:59 -05:00
}
// Get stored offset for given file pointer
unsigned int ZoneBuilder : : Zone : : safeGetPointer ( const void * pointer )
{
if ( this - > hasPointer ( pointer ) )
{
return this - > pointerMap [ pointer ] ;
}
return NULL ;
}
void ZoneBuilder : : Zone : : storePointer ( const void * pointer )
{
this - > pointerMap [ pointer ] = this - > buffer . getPackedOffset ( ) ;
}
void ZoneBuilder : : Zone : : storeAlias ( Game : : XAsset asset )
{
if ( ! this - > hasAlias ( asset ) )
{
this - > aliasList . push_back ( { asset , this - > buffer . getPackedOffset ( ) } ) ;
}
}
unsigned int ZoneBuilder : : Zone : : getAlias ( Game : : XAsset asset )
{
std : : string name = Game : : DB_GetXAssetName ( & asset ) ;
for ( auto & entry : this - > aliasList )
{
if ( asset . type = = entry . first . type & & name = = Game : : DB_GetXAssetName ( & entry . first ) )
{
return entry . second ;
}
}
return 0 ;
}
2018-12-17 08:29:18 -05:00
int ZoneBuilder : : Zone : : addScriptString ( const std : : string & str )
2017-01-19 16:23:59 -05:00
{
2022-11-27 14:20:07 -05:00
return this - > addScriptString ( static_cast < std : : uint16_t > ( Game : : SL_GetString ( str . data ( ) , 0 ) ) ) ;
2017-01-19 16:23:59 -05:00
}
// Mark a scriptString for writing and map it.
int ZoneBuilder : : Zone : : addScriptString ( unsigned short gameIndex )
{
// Handle NULL scriptStrings
// Might optimize that later
if ( ! gameIndex )
{
if ( this - > scriptStrings . empty ( ) )
{
this - > scriptStrings . push_back ( " " ) ;
}
return 0 ;
}
std : : string str = Game : : SL_ConvertToString ( gameIndex ) ;
int prev = this - > findScriptString ( str ) ;
if ( prev > 0 )
{
this - > scriptStringMap [ gameIndex ] = prev ;
return prev ;
}
this - > scriptStrings . push_back ( str ) ;
this - > scriptStringMap [ gameIndex ] = this - > scriptStrings . size ( ) ;
return this - > scriptStrings . size ( ) ;
}
// Find a local scriptString
2018-12-17 08:29:18 -05:00
int ZoneBuilder : : Zone : : findScriptString ( const std : : string & str )
2017-01-19 16:23:59 -05:00
{
for ( unsigned int i = 0 ; i < this - > scriptStrings . size ( ) ; + + i )
{
if ( this - > scriptStrings [ i ] = = str )
{
return ( i + 1 ) ;
}
}
return - 1 ;
}
2022-11-16 12:25:21 -05:00
void ZoneBuilder : : Zone : : addRawAsset ( Game : : XAssetType type , void * ptr )
{
this - > loadedAssets . push_back ( { type , { ptr } } ) ;
}
2017-01-19 16:23:59 -05:00
// Remap a scriptString to it's corresponding value in the local scriptString table.
2022-12-14 03:40:15 -05:00
void ZoneBuilder : : Zone : : mapScriptString ( unsigned short & gameIndex )
2017-01-19 16:23:59 -05:00
{
2022-12-14 03:40:15 -05:00
gameIndex = 0xFFFF & this - > scriptStringMap [ gameIndex ] ;
2017-01-19 16:23:59 -05:00
}
// Store a new name for a given asset
2018-12-17 08:29:18 -05:00
void ZoneBuilder : : Zone : : renameAsset ( Game : : XAssetType type , const std : : string & asset , const std : : string & newName )
2017-01-19 16:23:59 -05:00
{
if ( type < Game : : XAssetType : : ASSET_TYPE_COUNT & & type > = 0 )
{
this - > renameMap [ type ] [ asset ] = newName ;
}
else
{
2022-06-12 17:07:53 -04:00
Logger : : Error ( Game : : ERR_FATAL , " Unable to rename '{}' to '{}' as the asset type is invalid! " , asset , newName ) ;
2017-01-19 16:23:59 -05:00
}
}
// Return the new name for a given asset
2018-12-17 08:29:18 -05:00
std : : string ZoneBuilder : : Zone : : getAssetName ( Game : : XAssetType type , const std : : string & asset )
2017-01-19 16:23:59 -05:00
{
if ( type < Game : : XAssetType : : ASSET_TYPE_COUNT & & type > = 0 )
{
2022-05-31 12:38:09 -04:00
if ( this - > renameMap [ type ] . contains ( asset ) )
2017-01-19 16:23:59 -05:00
{
return this - > renameMap [ type ] [ asset ] ;
}
}
else
{
2022-06-12 17:07:53 -04:00
Logger : : Error ( Game : : ERR_FATAL , " Unable to get name for '{}' as the asset type is invalid! " , asset ) ;
2017-01-19 16:23:59 -05:00
}
return asset ;
}
void ZoneBuilder : : Zone : : store ( Game : : XAssetHeader header )
{
if ( ! this - > hasPointer ( header . data ) ) // We should never have to restore a pointer, so this expression should always resolve into false
{
this - > storePointer ( header . data ) ;
}
}
void ZoneBuilder : : Zone : : incrementExternalSize ( unsigned int size )
{
this - > externalSize + = size ;
}
bool ZoneBuilder : : IsEnabled ( )
{
2017-06-13 09:35:12 -04:00
static std : : optional < bool > flag ;
2017-01-19 16:23:59 -05:00
2017-06-13 09:35:12 -04:00
if ( ! flag . has_value ( ) )
2017-01-19 16:23:59 -05:00
{
2017-06-13 09:35:12 -04:00
flag . emplace ( Flags : : HasFlag ( " zonebuilder " ) ) ;
2017-01-19 16:23:59 -05:00
}
2017-06-13 09:35:12 -04:00
return ( flag . value ( ) & & ! Dedicated : : IsEnabled ( ) ) ;
2017-01-19 16:23:59 -05:00
}
2018-12-17 08:29:18 -05:00
void ZoneBuilder : : BeginAssetTrace ( const std : : string & zone )
2017-01-19 16:23:59 -05:00
{
ZoneBuilder : : TraceZone = zone ;
}
std : : vector < std : : pair < Game : : XAssetType , std : : string > > ZoneBuilder : : EndAssetTrace ( )
{
ZoneBuilder : : TraceZone . clear ( ) ;
std : : vector < std : : pair < Game : : XAssetType , std : : string > > AssetTrace ;
Utils : : Merge ( & AssetTrace , ZoneBuilder : : TraceAssets ) ;
ZoneBuilder : : TraceAssets . clear ( ) ;
return AssetTrace ;
}
2018-12-17 08:29:18 -05:00
Game : : XAssetHeader ZoneBuilder : : GetEmptyAssetIfCommon ( Game : : XAssetType type , const std : : string & name , ZoneBuilder : : Zone * builder )
2017-01-19 16:23:59 -05:00
{
2017-01-20 16:41:03 -05:00
Game : : XAssetHeader header = { nullptr } ;
2017-01-19 16:23:59 -05:00
if ( type > = 0 & & type < Game : : XAssetType : : ASSET_TYPE_COUNT )
{
2017-05-01 07:08:34 -04:00
int zoneIndex = Game : : DB_GetZoneIndex ( " common_mp " ) ;
if ( zoneIndex > 0 )
2017-01-19 16:23:59 -05:00
{
2022-06-12 17:07:53 -04:00
Game : : XAssetEntry * entry = Game : : DB_FindXAssetEntry ( type , name . data ( ) ) ;
2017-01-19 16:23:59 -05:00
2022-06-12 17:07:53 -04:00
if ( entry & & entry - > zoneIndex = = zoneIndex )
{
// Allocate an empty asset (filled with zeros)
header . data = builder - > getAllocator ( ) - > allocate ( Game : : DB_GetXAssetSizeHandlers [ type ] ( ) ) ;
2017-01-19 16:23:59 -05:00
2022-06-12 17:07:53 -04:00
// Set the name to the original name, so it can be stored
Game : : DB_SetXAssetNameHandlers [ type ] ( & header , name . data ( ) ) ;
AssetHandler : : StoreTemporaryAsset ( type , header ) ;
2019-01-15 19:40:57 -05:00
2022-06-12 17:07:53 -04:00
// Set the name to the empty name
Game : : DB_SetXAssetNameHandlers [ type ] ( & header , builder - > getAllocator ( ) - > duplicateString ( " , " + name ) ) ;
}
2017-01-19 16:23:59 -05:00
}
}
return header ;
}
2023-02-17 06:26:40 -05:00
iw4of : : params_t ZoneBuilder : : Zone : : getIW4OfApiParams ( )
2023-02-15 06:58:31 -05:00
{
2023-02-22 04:03:35 -05:00
iw4of : : params_t params { } ;
params . write_only_once = true ;
2023-02-15 06:58:31 -05:00
params . find_other_asset = [ this ] ( int type , const std : : string & name ) - > void *
{
return AssetHandler : : FindAssetForZone ( static_cast < Game : : XAssetType > ( type ) , name , this ) . data ;
} ;
params . fs_read_file = [ ] ( const std : : string & filename ) - > std : : string
{
auto file = FileSystem : : File ( filename ) ;
if ( file . exists ( ) )
{
return file . getBuffer ( ) ;
}
return { } ;
} ;
params . store_in_string_table = [ ] ( const std : : string & text ) - > unsigned int
{
return Game : : SL_GetString ( text . data ( ) , 0 ) ;
} ;
params . print = [ ] ( iw4of : : params_t : : print_type t , const std : : string & message ) - > void
{
switch ( t )
{
case iw4of : : params_t : : P_ERR :
2023-02-17 06:26:40 -05:00
Logger : : Error ( Game : : ERR_FATAL , " {} " , message ) ;
2023-02-15 06:58:31 -05:00
break ;
case iw4of : : params_t : : P_WARN :
Logger : : Print ( " {} " , message ) ;
break ;
}
} ;
2023-02-17 06:26:40 -05:00
if ( * Game : : fs_basepath & & * Game : : fs_gameDirVar )
{
params . work_directory = std : : format ( " {}/{} " , ( * Game : : fs_basepath ) - > current . string , ( * Game : : fs_gameDirVar ) - > current . string ) ;
}
else
{
Logger : : Error ( Game : : ERR_FATAL , " Missing FS Game directory or basepath directory! " ) ;
}
2023-02-15 06:58:31 -05:00
2023-02-17 06:26:40 -05:00
return params ;
2023-02-15 06:58:31 -05:00
}
2017-01-19 16:23:59 -05:00
int ZoneBuilder : : StoreTexture ( Game : : GfxImageLoadDef * * loadDef , Game : : GfxImage * image )
{
size_t size = 16 + ( * loadDef ) - > resourceSize ;
2017-06-04 05:01:04 -04:00
void * data = Utils : : Memory : : GetAllocator ( ) - > allocate ( size ) ;
2017-01-19 16:23:59 -05:00
std : : memcpy ( data , * loadDef , size ) ;
2018-05-09 08:33:52 -04:00
image - > texture . loadDef = reinterpret_cast < Game : : GfxImageLoadDef * > ( data ) ;
2017-01-19 16:23:59 -05:00
return 0 ;
}
void ZoneBuilder : : ReleaseTexture ( Game : : XAssetHeader header )
{
2018-05-09 08:33:52 -04:00
if ( header . image & & header . image - > texture . loadDef )
2017-01-19 16:23:59 -05:00
{
2018-05-09 08:33:52 -04:00
Utils : : Memory : : GetAllocator ( ) - > free ( header . image - > texture . loadDef ) ;
2017-01-19 16:23:59 -05:00
}
}
2017-05-26 15:41:40 -04:00
void ZoneBuilder : : AssumeMainThreadRole ( )
{
2017-05-27 09:38:12 -04:00
ZoneBuilder : : MainThreadInterrupted = true ;
ZoneBuilder : : InterruptingThreadId = GetCurrentThreadId ( ) ;
2017-05-26 15:41:40 -04:00
}
void ZoneBuilder : : ResetThreadRole ( )
{
2017-05-27 09:38:12 -04:00
ZoneBuilder : : MainThreadInterrupted = false ;
ZoneBuilder : : InterruptingThreadId = 0x0 ;
2017-05-26 15:41:40 -04:00
}
bool ZoneBuilder : : IsThreadMainThreadHook ( )
{
// this is the thread that is interrupting so let it act as the main thread
2017-05-27 09:38:12 -04:00
if ( ZoneBuilder : : MainThreadInterrupted & & GetCurrentThreadId ( ) = = ZoneBuilder : : InterruptingThreadId )
2017-05-26 15:41:40 -04:00
{
return true ;
}
// block the main thread from doing anything "main thread" specific while
// the other thread is interrupting
//while (ZoneBuilder::mainThreadInterrupted) std::this_thread::sleep_for(100ms);
// normal functionality
return GetCurrentThreadId ( ) = = Utils : : Hook : : Get < DWORD > ( 0x1CDE7FC ) ;
}
2022-12-18 16:47:59 -05:00
static Game : : XZoneInfo baseZones_old [ ] =
{
2018-05-09 08:33:52 -04:00
{ " code_pre_gfx_mp " , Game : : DB_ZONE_CODE , 0 } ,
{ " localized_code_pre_gfx_mp " , Game : : DB_ZONE_CODE_LOC , 0 } ,
{ " code_post_gfx_mp " , Game : : DB_ZONE_CODE , 0 } ,
{ " localized_code_post_gfx_mp " , Game : : DB_ZONE_CODE_LOC , 0 } ,
{ " common_mp " , Game : : DB_ZONE_COMMON , 0 } ,
{ " localized_common_mp " , Game : : DB_ZONE_COMMON_LOC , 0 } ,
{ " ui_mp " , Game : : DB_ZONE_GAME , 0 } ,
{ " localized_ui_mp " , Game : : DB_ZONE_GAME , 0 }
2017-05-21 17:00:39 -04:00
} ;
2017-05-26 15:41:40 -04:00
2022-12-18 16:47:59 -05:00
static Game : : XZoneInfo baseZones [ ] =
{
2018-05-09 08:33:52 -04:00
{ " defaults " , Game : : DB_ZONE_CODE , 0 } ,
{ " techsets " , Game : : DB_ZONE_CODE , 0 } ,
{ " common_mp " , Game : : DB_ZONE_COMMON , 0 } ,
{ " localized_common_mp " , Game : : DB_ZONE_COMMON_LOC , 0 } ,
{ " ui_mp " , Game : : DB_ZONE_GAME , 0 } ,
{ " localized_ui_mp " , Game : : DB_ZONE_GAME , 0 }
2017-05-21 17:00:39 -04:00
} ;
2022-06-12 18:02:20 -04:00
void ZoneBuilder : : Com_Quitf_t ( )
{
ExitProcess ( 0 ) ;
}
2022-12-18 16:47:59 -05:00
void ZoneBuilder : : CommandThreadCallback ( )
{
Com_InitThreadData ( ) ;
while ( ! ZoneBuilder : : CommandThreadTerminate )
{
ZoneBuilder : : AssumeMainThreadRole ( ) ;
Utils : : Hook : : Call < void ( int , int ) > ( 0x4E2C80 ) ( 0 , 0 ) ; // Cbuf_Execute
ZoneBuilder : : ResetThreadRole ( ) ;
std : : this_thread : : sleep_for ( 1 ms ) ;
}
}
BOOL APIENTRY ZoneBuilder : : EntryPoint ( HINSTANCE /*hInstance*/ , HINSTANCE /*hPrevInstance*/ , LPSTR /*lpCmdLine*/ , int /*nShowCmd*/ )
2017-05-22 21:34:01 -04:00
{
2022-12-18 16:47:59 -05:00
Utils : : Hook : : Call < void ( ) > ( 0x42F0A0 ) ( ) ; // Com_InitCriticalSections
Utils : : Hook : : Call < void ( ) > ( 0x4301B0 ) ( ) ; // Com_InitMainThread
Utils : : Hook : : Call < void ( int ) > ( 0x406D10 ) ( 0 ) ; // Win_InitLocalization
Utils : : Hook : : Call < void ( ) > ( 0x4FF220 ) ( ) ; // Com_InitParse
Utils : : Hook : : Call < void ( ) > ( 0x4D8220 ) ( ) ; // Dvar_Init
Utils : : Hook : : Call < void ( ) > ( 0x4D2280 ) ( ) ; // SL_Init
Utils : : Hook : : Call < void ( ) > ( 0x48F660 ) ( ) ; // Cmd_Init
Utils : : Hook : : Call < void ( ) > ( 0x4D9210 ) ( ) ; // Cbuf_Init
Utils : : Hook : : Call < void ( ) > ( 0x47F390 ) ( ) ; // Swap_Init
Utils : : Hook : : Call < void ( ) > ( 0x60AD10 ) ( ) ; // Com_InitDvars
Utils : : Hook : : Call < void ( ) > ( 0x420830 ) ( ) ; // Com_InitHunkMemory
Utils : : Hook : : Call < void ( ) > ( 0x4A62A0 ) ( ) ; // LargeLocalInit
Utils : : Hook : : Call < void ( ) > ( 0x4DCC10 ) ( ) ; // Sys_InitCmdEvents
Utils : : Hook : : Call < void ( ) > ( 0x64A020 ) ( ) ; // PMem_Init
2017-05-23 10:42:04 -04:00
if ( ! Flags : : HasFlag ( " stdout " ) )
{
2017-05-27 11:17:12 -04:00
Console : : ShowAsyncConsole ( ) ;
2017-05-23 13:53:53 -04:00
Utils : : Hook : : Call < void ( ) > ( 0x43D140 ) ( ) ; // Com_EventLoop
2017-05-23 10:42:04 -04:00
}
2022-12-18 16:47:59 -05:00
2017-05-22 21:34:01 -04:00
Utils : : Hook : : Call < void ( unsigned int ) > ( 0x502580 ) ( static_cast < unsigned int > ( __rdtsc ( ) ) ) ; // Netchan_Init
2022-12-18 16:47:59 -05:00
Utils : : Hook : : Call < void ( ) > ( 0x429080 ) ( ) ; // FS_InitFileSystem
Utils : : Hook : : Call < void ( ) > ( 0x4BFBE0 ) ( ) ; // Con_InitChannels
Utils : : Hook : : Call < void ( ) > ( 0x4E0FB0 ) ( ) ; // DB_InitThread
Utils : : Hook : : Call < void ( ) > ( 0x5196C0 ) ( ) ; // R_RegisterDvars
2017-05-22 21:34:01 -04:00
Game : : NET_Init ( ) ;
2022-12-18 16:47:59 -05:00
Utils : : Hook : : Call < void ( ) > ( 0x4F5090 ) ( ) ; // SND_InitDriver
Utils : : Hook : : Call < void ( ) > ( 0x46A630 ) ( ) ; // SND_Init
2017-05-22 21:34:01 -04:00
Utils : : Hook : : Call < void ( ) > ( 0x43D140 ) ( ) ; // Com_EventLoop
2017-05-21 17:00:39 -04:00
2022-12-18 16:47:59 -05:00
ZoneBuilder : : CommandThread = Utils : : Thread : : CreateNamedThread ( " Command Thread " , ZoneBuilder : : CommandThreadCallback ) ;
ZoneBuilder : : CommandThread . detach ( ) ;
2017-05-26 15:41:40 -04:00
2022-06-12 18:02:20 -04:00
Command : : Add ( " quit " , ZoneBuilder : : Com_Quitf_t ) ;
2017-05-23 10:42:04 -04:00
2017-05-21 17:00:39 -04:00
// now load default assets and shaders
2017-05-26 15:41:40 -04:00
if ( FastFiles : : Exists ( " defaults " ) & & FastFiles : : Exists ( " techsets " ) )
2017-05-21 17:00:39 -04:00
{
2017-05-22 21:34:01 -04:00
Game : : DB_LoadXAssets ( baseZones , ARRAYSIZE ( baseZones ) , 0 ) ;
2017-05-21 17:00:39 -04:00
}
else
{
2022-12-18 16:47:59 -05:00
Logger : : Warning ( Game : : CON_CHANNEL_DONT_FILTER , " Missing new init zones (defaults.ff & techsets.ff). You will need to load fastfiles to manually obtain techsets. \n " ) ;
2017-05-22 21:34:01 -04:00
Game : : DB_LoadXAssets ( baseZones_old , ARRAYSIZE ( baseZones_old ) , 0 ) ;
2017-05-21 17:00:39 -04:00
}
2017-05-27 09:38:12 -04:00
Logger : : Print ( " Waiting for fastiles to load... \n " ) ;
2017-05-22 21:34:01 -04:00
while ( ! Game : : Sys_IsDatabaseReady ( ) )
{
Utils : : Hook : : Call < void ( ) > ( 0x43D140 ) ( ) ; // Com_EventLoop
std : : this_thread : : sleep_for ( 100 ms ) ;
}
2017-05-21 17:00:39 -04:00
Logger : : Print ( " Done! \n " ) ;
// defaults need to load before we do this
2017-05-22 21:34:01 -04:00
Utils : : Hook : : Call < void ( ) > ( 0x4E1F30 ) ( ) ; // G_SetupWeaponDef
2022-06-12 17:07:53 -04:00
Utils : : Hook : : Call < void ( ) > ( 0x4454C0 ) ( ) ; // Item_SetupKeywordHash (for loading menus)
Utils : : Hook : : Call < void ( ) > ( 0x501BC0 ) ( ) ; // Menu_SetupKeywordHash (for loading menus)
Utils : : Hook : : Call < void ( ) > ( 0x4A1280 ) ( ) ; // something related to uiInfoArray
2017-05-22 21:34:01 -04:00
2017-05-27 08:22:26 -04:00
Utils : : Hook : : Call < void ( const char * ) > ( 0x464A90 ) ( GetCommandLineA ( ) ) ; // Com_ParseCommandLine
Utils : : Hook : : Call < void ( ) > ( 0x60C3D0 ) ( ) ; // Com_AddStartupCommands
2017-05-23 10:42:04 -04:00
// so for this i'm going to say that we just run the commands (after + signs)
// and don't spawn into a shell because why not?
if ( Flags : : HasFlag ( " stdout " ) )
{
Utils : : Hook : : Call < void ( ) > ( 0x440EC0 ) ( ) ; // DB_Update
Utils : : Hook : : Call < void ( ) > ( 0x43D140 ) ( ) ; // Com_EventLoop
Utils : : Hook : : Call < void ( int , int ) > ( 0x4E2C80 ) ( 0 , 0 ) ; // Cbuf_Execute
return 0 ;
}
2017-05-22 21:34:01 -04:00
2017-05-23 13:53:53 -04:00
Logger : : Print ( " -------------------------------------------------------------------------------- \n " ) ;
2023-04-19 13:53:29 -04:00
Logger : : Print ( " IW4x ZoneBuilder - {} \n " , VERSION ) ;
2017-05-23 13:53:53 -04:00
Logger : : Print ( " Commands: \n " ) ;
Logger : : Print ( " \t -buildzone [zone]: builds a zone from a csv located in zone_source \n " ) ;
Logger : : Print ( " \t -buildall: builds all zones in zone_source \n " ) ;
Logger : : Print ( " \t -verifyzone [zone]: loads and verifies the specified zone \n " ) ;
Logger : : Print ( " \t -listassets [assettype]: lists all loaded assets of the specified type \n " ) ;
Logger : : Print ( " \t -quit: quits the program \n " ) ;
Logger : : Print ( " -------------------------------------------------------------------------------- \n " ) ;
2017-05-22 21:34:01 -04:00
// now run main loop until quit
2017-05-26 15:41:40 -04:00
unsigned int frames = 0 ;
2017-05-25 07:34:01 -04:00
while ( true )
2017-05-21 17:00:39 -04:00
{
2017-05-26 15:41:40 -04:00
if ( frames % 100 = = 0 )
{
Utils : : Hook : : Call < void ( ) > ( 0x440EC0 ) ( ) ; // DB_Update
Utils : : Hook : : Call < void ( ) > ( 0x43EBB0 ) ( ) ; // check for quit
}
2017-05-22 21:34:01 -04:00
Utils : : Hook : : Call < void ( ) > ( 0x43D140 ) ( ) ; // Com_EventLoop
2017-05-27 09:38:12 -04:00
std : : this_thread : : sleep_for ( 1 ms ) ;
2017-05-26 15:41:40 -04:00
frames + + ;
2017-05-21 17:00:39 -04:00
}
2017-05-23 10:42:04 -04:00
return 0 ;
2017-05-21 17:00:39 -04:00
}
2022-06-12 17:07:53 -04:00
void ZoneBuilder : : HandleError ( Game : : errorParm_t code , const char * fmt , . . . )
2017-05-22 21:55:03 -04:00
{
2022-11-16 12:25:21 -05:00
char buffer [ 0x1000 ] { } ;
va_list ap ;
va_start ( ap , fmt ) ;
vsnprintf_s ( buffer , _TRUNCATE , fmt , ap ) ;
va_end ( ap ) ;
2017-05-25 15:12:20 -04:00
if ( ! Flags : : HasFlag ( " stdout " ) )
{
MessageBoxA ( nullptr , buffer , " Error! " , MB_OK | MB_ICONERROR ) ;
}
else
{
perror ( buffer ) ;
fflush ( stderr ) ;
}
2022-06-12 17:07:53 -04:00
if ( code = = Game : : ERR_FATAL )
{
ExitProcess ( 1 ) ;
}
2017-05-22 21:55:03 -04:00
}
2017-05-26 15:41:40 -04:00
__declspec ( naked ) void ZoneBuilder : : SoftErrorAssetOverflow ( )
{
// fuck with the stack to return to before DB_AddXAsset
__asm
{
add esp , 12 // exit from DB_AllocXAssetEntry
pop edi
pop esi
pop ebp
pop ebx
add esp , 14 h
mov eax , [ esp + 8 ]
retn
}
}
2018-12-17 08:29:18 -05:00
std : : string ZoneBuilder : : FindMaterialByTechnique ( const std : : string & techniqueName )
2017-05-26 15:41:40 -04:00
{
static bool replacementFound = false ;
replacementFound = false ; // the above one only runs the first time
const char * ret = " default " ;
Game : : DB_EnumXAssetEntries ( Game : : XAssetType : : ASSET_TYPE_MATERIAL , [ techniqueName , & ret ] ( Game : : XAssetEntry * entry )
{
if ( ! replacementFound )
{
Game : : XAssetHeader header = entry - > asset . header ;
std : : string name = techniqueName ;
if ( name [ 0 ] = = ' , ' ) name = name . substr ( 1 ) ;
if ( name = = header . material - > techniqueSet - > name )
{
2018-05-09 08:33:52 -04:00
ret = header . material - > info . name ;
2017-05-26 15:41:40 -04:00
replacementFound = true ;
}
}
2022-01-19 14:32:49 -05:00
} , false ) ;
2017-05-26 15:41:40 -04:00
if ( replacementFound ) return ret ;
2017-05-27 09:38:12 -04:00
return " " ;
2017-05-26 15:41:40 -04:00
}
2022-12-14 03:40:15 -05:00
void ZoneBuilder : : ReallocateLoadedSounds ( void * & data , [[maybe_unused]] void * a2 )
{
assert ( data ) ;
auto * sound = Utils : : Hook : : Get < Game : : MssSound * > ( 0x112AE04 ) ;
2022-12-18 16:47:59 -05:00
const auto length = sound - > info . data_len ;
const auto allocatedSpace = Game : : Z_Malloc ( static_cast < int > ( length ) ) ;
2022-12-14 03:40:15 -05:00
memcpy_s ( allocatedSpace , length , data , length ) ;
data = allocatedSpace ;
2022-12-17 10:52:00 -05:00
sound - > data = static_cast < char * > ( allocatedSpace ) ;
2022-12-14 03:40:15 -05:00
sound - > info . data_ptr = allocatedSpace ;
}
2022-12-24 15:41:23 -05:00
Game : : Sys_File ZoneBuilder : : Sys_CreateFile_Stub ( const char * dir , const char * filename )
{
auto file = Game : : Sys_CreateFile ( dir , filename ) ;
if ( file . handle = = INVALID_HANDLE_VALUE )
{
file = Game : : Sys_CreateFile ( " zone \\ zonebuilder \\ " , filename ) ;
}
return file ;
}
2017-01-19 16:23:59 -05:00
ZoneBuilder : : ZoneBuilder ( )
{
2017-01-26 16:56:46 -05:00
// ReSharper disable CppStaticAssertFailure
2017-01-19 16:23:59 -05:00
AssertSize ( Game : : XFileHeader , 21 ) ;
AssertSize ( Game : : XFile , 40 ) ;
static_assert ( Game : : MAX_XFILE_COUNT = = 8 , " XFile block enum is invalid! " ) ;
if ( ZoneBuilder : : IsEnabled ( ) )
{
// Prevent loading textures (preserves loaddef)
//Utils::Hook::Set<BYTE>(Game::Load_Texture, 0xC3);
2022-12-24 15:41:23 -05:00
Utils : : Hook ( 0x5BC832 , Sys_CreateFile_Stub , HOOK_CALL ) . install ( ) - > quick ( ) ;
2017-01-19 16:23:59 -05:00
// Store the loaddef
Utils : : Hook ( Game : : Load_Texture , StoreTexture , HOOK_JUMP ) . install ( ) - > quick ( ) ;
// Release the loaddef
Game : : DB_ReleaseXAssetHandlers [ Game : : XAssetType : : ASSET_TYPE_IMAGE ] = ZoneBuilder : : ReleaseTexture ;
2022-12-18 16:47:59 -05:00
// r_loadForrenderer = 0
2017-01-19 16:23:59 -05:00
Utils : : Hook : : Set < BYTE > ( 0x519DDF , 0 ) ;
2022-12-18 16:47:59 -05:00
// r_delayloadimage ret
2017-01-19 16:23:59 -05:00
Utils : : Hook : : Set < BYTE > ( 0x51F450 , 0xC3 ) ;
// r_registerDvars hack
Utils : : Hook : : Set < BYTE > ( 0x51B1CD , 0xC3 ) ;
// Prevent destroying textures
Utils : : Hook : : Set < BYTE > ( 0x51F03D , 0xEB ) ;
// Don't create default assets
Utils : : Hook : : Set < BYTE > ( 0x407BAA , 0xEB ) ;
// Don't mark clip maps as 'in use'
Utils : : Hook : : Nop ( 0x405E07 , 7 ) ;
// Don't mark assets
//Utils::Hook::Nop(0x5BB632, 5);
2022-12-18 16:47:59 -05:00
// Load sounds
2022-12-14 03:40:15 -05:00
Utils : : Hook ( 0x492EFC , ReallocateLoadedSounds , HOOK_CALL ) . install ( ) - > quick ( ) ;
2017-01-19 16:23:59 -05:00
// Don't display errors when assets are missing (we might manually build those)
Utils : : Hook : : Nop ( 0x5BB3F2 , 5 ) ;
Utils : : Hook : : Nop ( 0x5BB422 , 5 ) ;
Utils : : Hook : : Nop ( 0x5BB43A , 5 ) ;
2017-04-07 16:07:07 -04:00
// Don't read stats
Utils : : Hook ( 0x4875E1 , 0x487717 , HOOK_JUMP ) . install ( ) - > quick ( ) ;
2017-05-26 15:41:40 -04:00
// patch g_copyInfo because we're using so many more assets than originally intended
int newLimit = 0x2000 ;
2017-06-08 14:32:41 -04:00
int * g_copyInfo_new = Utils : : Memory : : GetAllocator ( ) - > allocateArray < int > ( newLimit ) ;
2017-05-26 15:41:40 -04:00
Utils : : Hook : : Set < int * > ( 0x494083 , g_copyInfo_new ) ; // DB_DelayLoadImages
Utils : : Hook : : Set < int * > ( 0x5BB9CC , g_copyInfo_new ) ; // DB_AddXAsset
Utils : : Hook : : Set < int * > ( 0x5BC723 , g_copyInfo_new ) ; // DB_PostLoadXZone
Utils : : Hook : : Set < int * > ( 0x5BC759 , g_copyInfo_new ) ;
Utils : : Hook : : Set < int > ( 0x5BB9AD , newLimit ) ; // limit check
2017-01-19 16:23:59 -05:00
// hunk size (was 300 MiB)
Utils : : Hook : : Set < DWORD > ( 0x64A029 , 0x38400000 ) ; // 900 MiB
Utils : : Hook : : Set < DWORD > ( 0x64A057 , 0x38400000 ) ;
2022-12-18 16:47:59 -05:00
// change FS_GameDirDomainFunc
2018-05-09 06:04:20 -04:00
Utils : : Hook : : Set < int ( * ) ( Game : : dvar_t * , Game : : DvarValue ) > ( 0x643203 , [ ] ( Game : : dvar_t * dvar , Game : : DvarValue value )
2017-05-25 07:34:01 -04:00
{
2018-05-09 06:04:20 -04:00
int result = Utils : : Hook : : Call < int ( Game : : dvar_t * , Game : : DvarValue ) > ( 0x642FC0 ) ( dvar , value ) ;
2017-05-25 07:34:01 -04:00
2022-06-30 15:37:47 -04:00
if ( result )
2017-05-25 07:34:01 -04:00
{
2022-12-18 16:47:59 -05:00
if ( std : : strcmp ( value . string , dvar - > current . string ) ! = 0 )
2017-05-25 07:34:01 -04:00
{
dvar - > current . string = value . string ;
Game : : FS_Restart ( 0 , 0 ) ;
}
}
return result ;
} ) ;
2017-05-22 21:34:01 -04:00
// set new entry point
Utils : : Hook ( 0x4513DA , ZoneBuilder : : EntryPoint , HOOK_JUMP ) . install ( ) - > quick ( ) ;
2017-05-21 17:00:39 -04:00
2022-06-12 18:02:20 -04:00
// set quit handler
Utils : : Hook ( 0x4D4000 , ZoneBuilder : : Com_Quitf_t , HOOK_JUMP ) . install ( ) - > quick ( ) ;
2022-06-12 17:07:53 -04:00
// handle Com_error Calls
Utils : : Hook ( Game : : Com_Error , ZoneBuilder : : HandleError , HOOK_JUMP ) . install ( ) - > quick ( ) ;
2017-05-22 21:55:03 -04:00
2022-12-18 16:47:59 -05:00
// Sys_IsMainThread hook
2017-05-26 15:41:40 -04:00
Utils : : Hook ( 0x4C37D0 , ZoneBuilder : : IsThreadMainThreadHook , HOOK_JUMP ) . install ( ) - > quick ( ) ;
2017-05-27 11:17:12 -04:00
// Don't exec startup config in fs_restart
Utils : : Hook : : Set < BYTE > ( 0x461B48 , 0xEB ) ;
2017-05-26 15:41:40 -04:00
// remove overriding asset messages
Utils : : Hook : : Nop ( 0x5BC74E , 5 ) ;
// don't remap techsets
Utils : : Hook : : Nop ( 0x5BC791 , 5 ) ;
2018-12-17 08:29:18 -05:00
AssetHandler : : OnLoad ( [ ] ( Game : : XAssetType type , Game : : XAssetHeader /*asset*/ , const std : : string & name , bool * /*restrict*/ )
2017-01-19 16:23:59 -05:00
{
if ( ! ZoneBuilder : : TraceZone . empty ( ) & & ZoneBuilder : : TraceZone = = FastFiles : : Current ( ) )
{
2022-12-18 16:47:59 -05:00
ZoneBuilder : : TraceAssets . emplace_back ( std : : make_pair ( type , name ) ) ;
# ifdef _DEBUG
OutputDebugStringA ( Utils : : String : : Format ( " %s \n " , name ) ) ;
# endif
2017-01-19 16:23:59 -05:00
}
} ) ;
Command : : Add ( " verifyzone " , [ ] ( Command : : Params * params )
{
2022-03-17 14:50:20 -04:00
if ( params - > size ( ) < 2 ) return ;
2017-01-19 16:23:59 -05:00
std : : string zone = params - > get ( 1 ) ;
ZoneBuilder : : BeginAssetTrace ( zone ) ;
Game : : XZoneInfo info ;
info . name = zone . data ( ) ;
2018-05-09 08:33:52 -04:00
info . allocFlags = Game : : DB_ZONE_MOD ;
2017-01-19 16:23:59 -05:00
info . freeFlags = 0 ;
2022-06-12 17:07:53 -04:00
Logger : : Print ( " Loading zone '{}'... \n " , zone ) ;
2017-01-19 16:23:59 -05:00
Game : : DB_LoadXAssets ( & info , 1 , true ) ;
AssetHandler : : FindOriginalAsset ( Game : : XAssetType : : ASSET_TYPE_RAWFILE , zone . data ( ) ) ; // Lock until zone is loaded
auto assets = ZoneBuilder : : EndAssetTrace ( ) ;
2022-06-12 17:07:53 -04:00
Logger : : Print ( " Unloading zone '{}'... \n " , zone ) ;
2018-05-09 08:33:52 -04:00
info . freeFlags = Game : : DB_ZONE_MOD ;
2017-01-19 16:23:59 -05:00
info . allocFlags = 0 ;
info . name = nullptr ;
Game : : DB_LoadXAssets ( & info , 1 , true ) ;
AssetHandler : : FindOriginalAsset ( Game : : XAssetType : : ASSET_TYPE_RAWFILE , " default " ) ; // Lock until zone is unloaded
2022-06-12 17:07:53 -04:00
Logger : : Print ( " Zone '{}' loaded with {} assets: \n " , zone , assets . size ( ) ) ;
2017-01-19 16:23:59 -05:00
int count = 0 ;
for ( auto i = assets . begin ( ) ; i ! = assets . end ( ) ; + + i , + + count )
{
2022-06-12 17:07:53 -04:00
Logger : : Print ( " {}: {}: {} \n " , count , Game : : DB_GetXAssetTypeName ( i - > first ) , i - > second ) ;
2017-01-19 16:23:59 -05:00
}
Logger : : Print ( " \n " ) ;
} ) ;
Command : : Add ( " buildzone " , [ ] ( Command : : Params * params )
{
2022-03-17 14:50:20 -04:00
if ( params - > size ( ) < 2 ) return ;
2017-01-19 16:23:59 -05:00
std : : string zoneName = params - > get ( 1 ) ;
2022-06-12 17:07:53 -04:00
Logger : : Print ( " Building zone '{}'... \n " , zoneName ) ;
2017-01-19 16:23:59 -05:00
Zone ( zoneName ) . build ( ) ;
} ) ;
2022-12-11 12:54:24 -05:00
Command : : Add ( " buildall " , [ ] ( [[maybe_unused]] Command : : Params * params )
2017-01-19 16:23:59 -05:00
{
2023-01-05 04:59:09 -05:00
auto path = std : : format ( " {} \\ zone_source " , ( * Game : : fs_basepath ) - > current . string ) ;
2022-12-11 12:54:24 -05:00
auto zoneSources = FileSystem : : GetSysFileList ( path , " csv " , false ) ;
2017-01-19 16:23:59 -05:00
for ( auto source : zoneSources )
{
if ( Utils : : String : : EndsWith ( source , " .csv " ) )
{
source = source . substr ( 0 , source . find ( " .csv " ) ) ;
}
2022-12-11 12:54:24 -05:00
Command : : Execute ( std : : format ( " buildzone {} " , source ) , true ) ;
2017-01-19 16:23:59 -05:00
}
} ) ;
2017-05-26 15:41:40 -04:00
static std : : set < std : : string > curTechsets_list ;
static std : : set < std : : string > techsets_list ;
2018-12-17 08:29:18 -05:00
AssetHandler : : OnLoad ( [ ] ( Game : : XAssetType type , Game : : XAssetHeader , const std : : string & name , bool * )
2017-03-26 09:47:53 -04:00
{
2017-05-26 15:41:40 -04:00
if ( type = = Game : : ASSET_TYPE_TECHNIQUE_SET )
{
if ( name [ 0 ] = = ' , ' ) return ; // skip techsets from common_mp
if ( techsets_list . find ( name ) = = techsets_list . end ( ) )
{
curTechsets_list . emplace ( name ) ;
techsets_list . emplace ( name ) ;
}
}
2017-03-26 09:47:53 -04:00
} ) ;
2022-12-14 03:40:15 -05:00
AssetHandler : : OnLoad ( [ ] ( Game : : XAssetType type , Game : : XAssetHeader asset , [[maybe_unused]] const std::string& name, [[maybe_unused]] bool * restrict )
{
if ( type ! = Game : : ASSET_TYPE_SOUND )
{
return ;
}
auto sound = asset . sound ;
for ( size_t i = 0 ; i < sound - > count ; i + + )
{
auto thisSound = sound - > head [ i ] ;
if ( thisSound . soundFile - > type = = Game : : SAT_LOADED )
{
if ( thisSound . soundFile - > u . loadSnd - > sound . data = = nullptr )
{
// ouch
// This should never happen and will cause a memory leak
// Let's change it to a streamed sound instead
thisSound . soundFile - > type = Game : : SAT_STREAMED ;
auto virtualPath = std : : filesystem : : path ( thisSound . soundFile - > u . loadSnd - > name ) ;
thisSound . soundFile - > u . streamSnd . filename . info . raw . name = Utils : : Memory : : DuplicateString ( virtualPath . filename ( ) . string ( ) ) ;
auto dir = virtualPath . remove_filename ( ) . string ( ) ;
dir = dir . substr ( 0 , dir . size ( ) - 1 ) ; // remove /
thisSound . soundFile - > u . streamSnd . filename . info . raw . dir = Utils : : Memory : : DuplicateString ( dir ) ;
}
}
}
} ) ;
2017-05-26 15:41:40 -04:00
Command : : Add ( " buildtechsets " , [ ] ( Command : : Params * )
2017-03-26 09:47:53 -04:00
{
2017-05-26 15:41:40 -04:00
Utils : : IO : : CreateDir ( " zone_source/techsets " ) ;
Utils : : IO : : CreateDir ( " zone/techsets " ) ;
std : : string csvStr ;
2023-01-06 07:51:41 -05:00
const auto dir = std : : format ( " zone/{} " , Game : : Win_GetLanguage ( ) ) ;
auto fileList = Utils : : IO : : ListFiles ( dir , false ) ;
2023-03-17 08:46:29 -04:00
for ( const auto & entry : fileList )
2017-03-26 09:47:53 -04:00
{
2023-03-17 08:46:29 -04:00
auto zone = entry . path ( ) . string ( ) ;
2017-05-29 13:52:37 -04:00
Utils : : String : : Replace ( zone , Utils : : String : : VA ( " zone/%s/ " , Game : : Win_GetLanguage ( ) ) , " " ) ;
2017-05-26 15:41:40 -04:00
Utils : : String : : Replace ( zone , " .ff " , " " ) ;
if ( Utils : : IO : : FileExists ( " zone/techsets/ " + zone + " _techsets.ff " ) )
{
2022-06-12 17:07:53 -04:00
Logger : : Print ( " Skipping previously generated zone {} \n " , zone ) ;
2017-05-26 15:41:40 -04:00
continue ;
}
if ( zone . find ( " _load " ) ! = std : : string : : npos )
{
2022-06-12 17:07:53 -04:00
Logger : : Print ( " Skipping loadscreen zone {} \n " , zone ) ;
2017-05-26 15:41:40 -04:00
continue ;
}
if ( Game : : DB_IsZoneLoaded ( zone . c_str ( ) ) | | ! FastFiles : : Exists ( zone ) )
{
continue ;
}
if ( zone [ 0 ] = = ' . ' ) continue ; // fucking mac dotfiles
curTechsets_list . clear ( ) ; // clear from last run
// load the zone
Game : : XZoneInfo info ;
info . name = zone . c_str ( ) ;
2018-05-09 08:33:52 -04:00
info . allocFlags = Game : : DB_ZONE_MOD ;
2017-05-26 15:41:40 -04:00
info . freeFlags = 0x0 ;
Game : : DB_LoadXAssets ( & info , 1 , 0 ) ;
while ( ! Game : : Sys_IsDatabaseReady ( ) ) std : : this_thread : : sleep_for ( 100 ms ) ; // wait till its fully loaded
2023-01-06 07:51:41 -05:00
if ( curTechsets_list . empty ( ) )
2017-05-26 15:41:40 -04:00
{
2022-06-12 17:07:53 -04:00
Logger : : Print ( " Skipping empty zone {} \n " , zone ) ;
2017-05-26 15:41:40 -04:00
// unload zone
info . name = nullptr ;
info . allocFlags = 0x0 ;
2018-05-09 08:33:52 -04:00
info . freeFlags = Game : : DB_ZONE_MOD ;
2017-05-26 15:41:40 -04:00
Game : : DB_LoadXAssets ( & info , 1 , true ) ;
continue ;
}
// ok so we're just gonna use the materials because they will use the techsets
csvStr . clear ( ) ;
for ( auto tech : curTechsets_list )
{
std : : string mat = ZoneBuilder : : FindMaterialByTechnique ( tech ) ;
if ( mat . length ( ) = = 0 )
{
csvStr . append ( " techset, " + tech + " \n " ) ;
}
else
{
csvStr . append ( " material, " + mat + " \n " ) ;
}
}
// save csv
2023-01-06 07:51:41 -05:00
Utils : : IO : : WriteFile ( " zone_source/techsets/ " + zone + " _techsets.csv " , csvStr ) ;
2017-05-26 15:41:40 -04:00
// build the techset zone
std : : string zoneName = " techsets/ " + zone + " _techsets " ;
2022-06-12 17:07:53 -04:00
Logger : : Print ( " Building zone '{}'... \n " , zoneName ) ;
2017-05-26 15:41:40 -04:00
Zone ( zoneName ) . build ( ) ;
// unload original zone
info . name = nullptr ;
info . allocFlags = 0x0 ;
2018-05-09 08:33:52 -04:00
info . freeFlags = Game : : DB_ZONE_MOD ;
2017-05-26 15:41:40 -04:00
Game : : DB_LoadXAssets ( & info , 1 , true ) ;
while ( ! Game : : Sys_IsDatabaseReady ( ) ) std : : this_thread : : sleep_for ( 10 ms ) ; // wait till its fully loaded
2017-03-26 09:47:53 -04:00
}
2017-05-26 15:41:40 -04:00
curTechsets_list . clear ( ) ;
techsets_list . clear ( ) ;
Game : : DB_EnumXAssets ( Game : : ASSET_TYPE_TECHNIQUE_SET , [ ] ( Game : : XAssetHeader header , void * )
{
curTechsets_list . emplace ( header . techniqueSet - > name ) ;
techsets_list . emplace ( header . techniqueSet - > name ) ;
} , nullptr , false ) ;
// HACK: set language to 'techsets' to load from that dir
2022-01-17 12:21:51 -05:00
const char * language = Utils : : Hook : : Get < const char * > ( 0x649E740 ) ;
2020-12-09 14:13:34 -05:00
Utils : : Hook : : Set < const char * > ( 0x649E740 , " techsets " ) ;
2017-05-26 15:41:40 -04:00
// load generated techset fastfiles
2023-01-06 07:51:41 -05:00
auto list = Utils : : IO : : ListFiles ( " zone/techsets " , false ) ;
2017-05-26 15:41:40 -04:00
int i = 0 ;
int subCount = 0 ;
2023-03-17 08:46:29 -04:00
for ( const auto & entry : list )
2017-05-26 15:41:40 -04:00
{
2023-03-17 08:46:29 -04:00
auto it = entry . path ( ) . string ( ) ;
2017-05-26 15:41:40 -04:00
Utils : : String : : Replace ( it , " zone/techsets/ " , " " ) ;
Utils : : String : : Replace ( it , " .ff " , " " ) ;
if ( it . find ( " _techsets " ) = = std : : string : : npos ) continue ; // skip files we didn't generate for this
if ( ! Game : : DB_IsZoneLoaded ( it . data ( ) ) )
{
Game : : XZoneInfo info ;
info . name = it . data ( ) ;
2018-05-09 08:33:52 -04:00
info . allocFlags = Game : : DB_ZONE_MOD ;
2017-05-26 15:41:40 -04:00
info . freeFlags = 0 ;
Game : : DB_LoadXAssets ( & info , 1 , 0 ) ;
while ( ! Game : : Sys_IsDatabaseReady ( ) ) std : : this_thread : : sleep_for ( 10 ms ) ; // wait till its fully loaded
}
else
{
2022-06-12 17:07:53 -04:00
Logger : : Print ( " Zone '{}' already loaded \n " , it ) ;
2017-05-26 15:41:40 -04:00
}
if ( i = = 20 ) // cap at 20 just to be safe
{
// create csv with the techsets in it
csvStr . clear ( ) ;
for ( auto tech : curTechsets_list )
{
std : : string mat = ZoneBuilder : : FindMaterialByTechnique ( tech ) ;
if ( mat . length ( ) = = 0 )
{
csvStr . append ( " techset, " + tech + " \n " ) ;
}
else
{
csvStr . append ( " material, " + mat + " \n " ) ;
}
}
std : : string tempZoneFile = Utils : : String : : VA ( " zone_source/techsets/techsets%d.csv " , subCount ) ;
std : : string tempZone = Utils : : String : : VA ( " techsets/techsets%d " , subCount ) ;
2023-01-06 07:51:41 -05:00
Utils : : IO : : WriteFile ( tempZoneFile , csvStr ) ;
2017-05-26 15:41:40 -04:00
2022-06-12 17:07:53 -04:00
Logger : : Print ( " Building zone '{}'... \n " , tempZone ) ;
2017-05-26 15:41:40 -04:00
Zone ( tempZone ) . build ( ) ;
// unload all zones
Game : : XZoneInfo info ;
info . name = nullptr ;
info . allocFlags = 0x0 ;
2018-05-09 08:33:52 -04:00
info . freeFlags = Game : : DB_ZONE_MOD ;
2017-05-26 15:41:40 -04:00
Game : : DB_LoadXAssets ( & info , 1 , true ) ;
2020-12-09 14:13:34 -05:00
Utils : : Hook : : Set < const char * > ( 0x649E740 , " techsets " ) ;
2017-05-26 15:41:40 -04:00
i = 0 ;
subCount + + ;
curTechsets_list . clear ( ) ;
techsets_list . clear ( ) ;
}
i + + ;
}
// last iteration
if ( i ! = 0 )
{
// create csv with the techsets in it
csvStr . clear ( ) ;
for ( auto tech : curTechsets_list )
{
std : : string mat = ZoneBuilder : : FindMaterialByTechnique ( tech ) ;
if ( mat . length ( ) = = 0 )
{
2022-06-12 17:07:53 -04:00
Logger : : Print ( " Couldn't find a material for techset {}. Sort Keys will be incorrect. \n " , tech ) ;
2017-05-26 15:41:40 -04:00
csvStr . append ( " techset, " + tech + " \n " ) ;
}
else
{
csvStr . append ( " material, " + mat + " \n " ) ;
}
}
std : : string tempZoneFile = Utils : : String : : VA ( " zone_source/techsets/techsets%d.csv " , subCount ) ;
std : : string tempZone = Utils : : String : : VA ( " techsets/techsets%d " , subCount ) ;
2023-01-06 07:51:41 -05:00
Utils : : IO : : WriteFile ( tempZoneFile , csvStr ) ;
2017-05-26 15:41:40 -04:00
2022-06-12 17:07:53 -04:00
Logger : : Print ( " Building zone '{}'... \n " , tempZone ) ;
2017-05-26 15:41:40 -04:00
Zone ( tempZone ) . build ( ) ;
// unload all zones
Game : : XZoneInfo info ;
info . name = nullptr ;
info . allocFlags = 0x0 ;
2018-05-09 08:33:52 -04:00
info . freeFlags = Game : : DB_ZONE_MOD ;
2017-05-26 15:41:40 -04:00
Game : : DB_LoadXAssets ( & info , 1 , true ) ;
subCount + + ;
}
// build final techsets fastfile
if ( subCount > 24 )
{
2022-06-12 17:07:53 -04:00
Logger : : Error ( Game : : ERR_DROP , " How did you have 576 fastfiles? \n " ) ;
2017-05-26 15:41:40 -04:00
}
curTechsets_list . clear ( ) ;
techsets_list . clear ( ) ;
for ( int j = 0 ; j < subCount ; + + j )
{
Game : : XZoneInfo info ;
info . name = Utils : : String : : VA ( " techsets%d " , j ) ;
2018-05-09 08:33:52 -04:00
info . allocFlags = Game : : DB_ZONE_MOD ;
2017-05-26 15:41:40 -04:00
info . freeFlags = 0 ;
Game : : DB_LoadXAssets ( & info , 1 , 0 ) ;
while ( ! Game : : Sys_IsDatabaseReady ( ) ) std : : this_thread : : sleep_for ( 10 ms ) ; // wait till its fully loaded
}
// create csv with the techsets in it
csvStr . clear ( ) ;
2023-03-17 08:46:29 -04:00
for ( const auto & tech : curTechsets_list )
2017-05-26 15:41:40 -04:00
{
2023-01-06 07:51:41 -05:00
auto mat = ZoneBuilder : : FindMaterialByTechnique ( tech ) ;
2017-05-26 15:41:40 -04:00
if ( mat . length ( ) = = 0 )
{
csvStr . append ( " techset, " + tech + " \n " ) ;
}
else
{
csvStr . append ( " material, " + mat + " \n " ) ;
}
}
2023-01-06 07:51:41 -05:00
Utils : : IO : : WriteFile ( " zone_source/techsets/techsets.csv " , csvStr ) ;
2017-05-26 15:41:40 -04:00
// set language back
2022-01-17 12:21:51 -05:00
Utils : : Hook : : Set < const char * > ( 0x649E740 , language ) ;
2017-05-26 15:41:40 -04:00
Logger : : Print ( " Building zone 'techsets/techsets'... \n " ) ;
Zone ( " techsets/techsets " ) . build ( ) ;
2017-03-26 09:47:53 -04:00
} ) ;
2017-01-19 16:23:59 -05:00
Command : : Add ( " listassets " , [ ] ( Command : : Params * params )
{
2022-03-17 14:50:20 -04:00
if ( params - > size ( ) < 2 ) return ;
2017-01-19 16:23:59 -05:00
Game : : XAssetType type = Game : : DB_GetXAssetNameType ( params - > get ( 1 ) ) ;
if ( type ! = Game : : XAssetType : : ASSET_TYPE_INVALID )
{
Game : : DB_EnumXAssets ( type , [ ] ( Game : : XAssetHeader header , void * data )
{
Game : : XAsset asset = { * reinterpret_cast < Game : : XAssetType * > ( data ) , header } ;
2022-06-12 17:07:53 -04:00
Logger : : Print ( " {} \n " , Game : : DB_GetXAssetName ( & asset ) ) ;
2017-01-19 16:23:59 -05:00
} , & type , false ) ;
}
} ) ;
2017-05-26 15:41:40 -04:00
Command : : Add ( " loadtempzone " , [ ] ( Command : : Params * params )
{
2022-03-17 14:50:20 -04:00
if ( params - > size ( ) < 2 ) return ;
2017-05-26 15:41:40 -04:00
if ( FastFiles : : Exists ( params - > get ( 1 ) ) )
{
Game : : XZoneInfo info ;
info . name = params - > get ( 1 ) ;
info . allocFlags = 0x80 ;
info . freeFlags = 0x0 ;
Game : : DB_LoadXAssets ( & info , 1 , 0 ) ;
}
} ) ;
Command : : Add ( " unloadtempzones " , [ ] ( Command : : Params * )
{
Game : : XZoneInfo info ;
info . name = nullptr ;
info . allocFlags = 0x0 ;
info . freeFlags = 0x80 ;
Game : : DB_LoadXAssets ( & info , 1 , true ) ;
AssetHandler : : FindOriginalAsset ( Game : : XAssetType : : ASSET_TYPE_RAWFILE , " default " ) ; // Lock until zone is unloaded
} ) ;
2017-01-19 16:23:59 -05:00
Command : : Add ( " materialInfoDump " , [ ] ( Command : : Params * )
{
Game : : DB_EnumXAssets ( Game : : ASSET_TYPE_MATERIAL , [ ] ( Game : : XAssetHeader header , void * )
{
2022-06-12 17:07:53 -04:00
Logger : : Print ( " {}: {:#X} {:#X} {:#X} \n " ,
header . material - > info . name , header . material - > info . sortKey & 0xFF , header . material - > info . gameFlags & 0xFF , header . material - > stateFlags & 0xFF ) ;
2017-01-19 16:23:59 -05:00
} , nullptr , false ) ;
} ) ;
2017-04-21 14:49:06 -04:00
Command : : Add ( " iwiDump " , [ ] ( Command : : Params * params )
{
2022-03-17 14:50:20 -04:00
if ( params - > size ( ) < 2 ) return ;
2017-04-21 14:49:06 -04:00
2023-01-05 04:59:09 -05:00
auto path = std : : format ( " {} \\ mods \\ {} \\ images " , ( * Game : : fs_basepath ) - > current . string , params - > get ( 1 ) ) ;
2022-12-11 12:54:24 -05:00
auto images = FileSystem : : GetSysFileList ( path , " iwi " , false ) ;
2017-04-21 14:49:06 -04:00
2022-12-11 12:54:24 -05:00
for ( auto i = images . begin ( ) ; i ! = images . end ( ) ; )
2017-04-21 14:49:06 -04:00
{
2022-12-11 12:54:24 -05:00
* i = std : : format ( " images/{} " , * i ) ;
2017-04-21 14:49:06 -04:00
2022-06-30 15:37:47 -04:00
if ( FileSystem : : File ( * i ) . exists ( ) )
2017-04-21 14:49:06 -04:00
{
i = images . erase ( i ) ;
continue ;
}
+ + i ;
}
Logger : : Print ( " ------------------- BEGIN IWI DUMP ------------------- \n " ) ;
2022-07-29 15:54:18 -04:00
Logger : : Print ( " {} \n " , nlohmann : : json ( images ) . dump ( ) ) ;
2017-04-21 14:49:06 -04:00
Logger : : Print ( " ------------------- END IWI DUMP ------------------- \n " ) ;
} ) ;
2017-01-19 16:23:59 -05:00
}
}
ZoneBuilder : : ~ ZoneBuilder ( )
{
2022-12-18 16:47:59 -05:00
ZoneBuilder : : CommandThreadTerminate = true ;
2022-06-12 18:02:20 -04:00
if ( ZoneBuilder : : CommandThread . joinable ( ) )
2017-05-27 09:38:12 -04:00
{
ZoneBuilder : : CommandThread . join ( ) ;
}
2017-01-19 16:23:59 -05:00
}
2017-04-27 16:03:05 -04:00
# if defined(DEBUG) || defined(FORCE_UNIT_TESTS)
bool ZoneBuilder : : unitTest ( )
{
printf ( " Testing circular bit shifting (left)... " ) ;
unsigned int integer = 0x80000000 ;
Utils : : RotLeft ( integer , 1 ) ;
2022-06-30 15:37:47 -04:00
if ( integer ! = 1 )
2017-04-27 16:03:05 -04:00
{
printf ( " Error \n " ) ;
printf ( " Bit shifting failed: %X \n " , integer ) ;
return false ;
}
printf ( " Success \n " ) ;
printf ( " Testing circular bit shifting (right)... " ) ;
unsigned char byte = 0b00000011 ;
Utils : : RotRight ( byte , 2 ) ;
if ( byte ! = 0b11000000 )
{
printf ( " Error \n " ) ;
printf ( " Bit shifting failed %X \n " , byte & 0xFF ) ;
return false ;
}
printf ( " Success \n " ) ;
return true ;
}
# endif
2017-01-19 16:23:59 -05:00
}