/*
===========================================================================
Wolfenstein: Enemy Territory GPL Source Code
Copyright (C) 1999-2010 id Software LLC, a ZeniMax Media company.
This file is part of the Wolfenstein: Enemy Territory GPL Source Code (Wolf ET Source Code).
Wolf ET 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.
Wolf ET 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 Wolf ET Source Code. If not, see .
In addition, the Wolf: ET 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 Wolf ET 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.
===========================================================================
*/
/*
* name: g_save.c
*
*/
#ifdef SAVEGAME_SUPPORT
#include "../game/g_local.h"
#include "../game/q_shared.h"
#include "../game/botlib.h" //bot lib interface
#include "../game/be_aas.h"
#include "../game/be_ea.h"
#include "../game/be_ai_gen.h"
#include "../game/be_ai_goal.h"
#include "../game/be_ai_move.h"
#include "../botai/botai.h" //bot ai interface
#include "../botai/ai_main.h"
#include "../botai/chars.h"
#include "../botai/ai_team.h"
#include "../botai/ ai_dmq3.h"
#include "../game/be_ai_chat.h"
extern bot_waypoint_t botai_waypoints[MAX_BOTAIWAYPOINTS];
extern bot_waypoint_t *botai_freewaypoints;
/*
Wolf savegame system.
Using the "checkpoint" system, we only need to save at specific locations, but various entities
may have changed behind us, so therefore we need to save as much as possible, but without going
overboard.
For now, everything is saved from the entity, client and cast_state structures, except the fields
defined in the ignoreField structures below. Any pointer fields need to be specified in the
saveField structures below.
!! NOTE: when working on Wolf patches, make sure you only add fields to the very end of the three
main structures saved here (entity, client, cast_state). If any fields are inserted in the middle
of these structures, savegames will become corrupted, and there is no way of checking for this,
so it'll just crash.
*/
//vmCvar_t musicCvar;
//char musicString[MAX_STRING_CHARS];
static int ver;
typedef enum {
F_NONE,
F_STRING,
F_ENTITY, // index on disk, pointer in memory
F_ITEM, // index on disk, pointer in memory
F_CLIENT, // index on disk, pointer in memory
F_FUNCTION,
F_SPLINE, // index on disk, pointer in memory
F_AIFUNCTION,
F_BOTWAYPOINT, // index on disk, pointer in memory
F_BOTSCRIPTDATA, // index on disk, pointer in memory
} saveFieldtype_t;
typedef struct {
int ofs;
saveFieldtype_t type;
} saveField_t;
//.......................................................................................
// these are the fields that cannot be saved directly, so they need to be converted
static saveField_t gentityFields[] = {
{FOFS( client ), F_CLIENT},
{FOFS( classname ), F_STRING},
{FOFS( model ), F_STRING},
{FOFS( model2 ), F_STRING},
{FOFS( parent ), F_ENTITY},
{FOFS( nextTrain ), F_ENTITY},
{FOFS( prevTrain ), F_ENTITY},
{FOFS( message ), F_STRING},
{FOFS( target ), F_STRING},
{FOFS( targetname ), F_STRING},
{FOFS( team ), F_STRING},
{FOFS( targetShaderName ), F_STRING},
{FOFS( targetShaderNewName ), F_STRING},
{FOFS( target_ent ), F_ENTITY},
{FOFS( think ), F_FUNCTION},
{FOFS( reached ), F_FUNCTION},
{FOFS( blocked ), F_FUNCTION},
{FOFS( touch ), F_FUNCTION},
{FOFS( use ), F_FUNCTION},
{FOFS( pain ), F_FUNCTION},
{FOFS( die ), F_FUNCTION},
{FOFS( chain ), F_ENTITY},
{FOFS( enemy ), F_ENTITY},
{FOFS( activator ), F_ENTITY},
{FOFS( teamchain ), F_ENTITY},
{FOFS( teammaster ), F_ENTITY},
{FOFS( item ), F_ITEM},
{FOFS( aiName ), F_STRING},
{FOFS( AIScript_AlertEntity ), F_FUNCTION},
{FOFS( aiSkin ), F_STRING},
{FOFS( dl_stylestring ), F_STRING},
{FOFS( dl_shader ), F_STRING},
{FOFS( melee ), F_ENTITY},
{FOFS( spawnitem ), F_STRING},
{FOFS( track ), F_STRING},
{FOFS( scriptName ), F_STRING},
{FOFS( tagName ), F_STRING},
{FOFS( tagParent ), F_ENTITY},
{FOFS( constages ), F_STRING},
{FOFS( desstages ), F_STRING},
{FOFS( botNextStaticEntity ), F_ENTITY},
{FOFS( backspline ), F_SPLINE},
{0, 0}
};
static saveField_t gclientFields[] = {
{CFOFS( hook ), F_ENTITY},
{CFOFS( persistantPowerup ), F_ENTITY},
{CFOFS( cameraPortal ), F_ENTITY},
{CFOFS( touchingTOI ), F_ENTITY},
{CFOFS( landmineSpotted ), F_ENTITY},
{CFOFS( tempHead ), F_ENTITY},
{CFOFS( tempLeg ), F_ENTITY},
{0, 0}
};
static saveField_t botwaypointFields[] = {
{BWPOFS( next ), F_BOTWAYPOINT},
{BWPOFS( prev ), F_BOTWAYPOINT},
{0, 0}
};
static saveField_t botstateFields[] = {
{BFOFS( ainode ), F_AIFUNCTION},
{BFOFS( ainodeText ), F_STRING},
{BFOFS( checkpoints ), F_BOTWAYPOINT}, // for waypoints we need to store the botai_waypoints seperately
{BFOFS( patrolpoints ), F_BOTWAYPOINT},
{BFOFS( curpatrolpoint ), F_BOTWAYPOINT},
{BFOFS( script.data ), F_BOTSCRIPTDATA},
{BFOFS( indicator ), F_ENTITY},
{0, 0}
};
//.......................................................................................
// this is where we define fields or sections of structures that we should totally ignore
typedef struct {
int ofs;
int len;
} ignoreField_t;
static ignoreField_t gentityIgnoreFields[] = {
// don't process events that have already occured before the game was saved
//{FOFS(s.events[0]), sizeof(int) * MAX_EVENTS},
//{FOFS(s.eventParms[0]), sizeof(int) * MAX_EVENTS},
//{FOFS(s.eventSequence), sizeof(int)},
{FOFS( numScriptEvents ), sizeof( int )},
{FOFS( scriptEvents ), sizeof( g_script_event_t * ) }, // gets created upon parsing the script file, this is static while playing
{0, 0}
};
static ignoreField_t gclientIgnoreFields[] = {
// don't process events that have already occured before the game was saved
//{CFOFS(ps.events[0]), sizeof(int) * MAX_EVENTS},
//{CFOFS(ps.eventParms[0]), sizeof(int) * MAX_EVENTS},
//{CFOFS(ps.eventSequence), sizeof(int)},
//{CFOFS(ps.oldEventSequence),sizeof(int)},
{0, 0}
};
static ignoreField_t botstateIgnoreFields[] = {
{BFOFS( objectiveMatrix ), sizeof( botMatrix_t * )}, // these are based on the playerclass stored in the botstate and further static
{BFOFS( behaviourMatrix ), sizeof( botMatrix_t * )},
{BFOFS( engagementMatrix ), sizeof( botMatrix_t * )},
{BFOFS( character ), sizeof( int )}, // these are all returned from the botlib
{BFOFS( gs ), sizeof( int )},
{BFOFS( ws ), sizeof( int )},
{BFOFS( cs ), sizeof( int )},
{BFOFS( ms ), sizeof( int )},
{BFOFS( attribs ), sizeof( float ) * BOT_MAX_ATTRIBUTES},
{0, 0}
};
//.......................................................................................
// persistant data is optionally carried across level changes
// !! WARNING: cannot save pointer or string variables
typedef struct {
int ofs;
int len;
} persField_t;
static persField_t gentityPersFields[] = {
{FOFS( health ), sizeof( int )},
{0, 0}
};
static persField_t gclientPersFields[] = {
{CFOFS( ps.weapon ), sizeof( int )},
{CFOFS( ps.ammo[0] ), sizeof( int ) * MAX_WEAPONS},
{CFOFS( ps.ammoclip[0] ), sizeof( int ) * MAX_WEAPONS}, //----(SA) added for ammo in clip
{CFOFS( ps.persistant[0] ), sizeof( int ) * MAX_PERSISTANT},
{CFOFS( ps.stats[0] ), sizeof( int ) * MAX_STATS},
{CFOFS( ps.weapons[0] ), sizeof( int ) * MAX_WEAPONS / ( sizeof( int ) * 8 )}, // //----(SA) added. weapons owned got moved outside stats[]
{CFOFS( ps.powerups[0] ), sizeof( int ) * MAX_POWERUPS},
{CFOFS( ps.holding ), sizeof( int )}, //----(SA) added
{0, 0}
};
//.......................................................................................
// this stores all functions in the game code
typedef struct {
char *funcStr;
byte *funcPtr;
} funcList_t;
//-----------------
// MSVC likes to needlessly(?) warn about these defines, so disable certain warnings temporarily
#ifdef _WIN32
#pragma warning( push )
#pragma warning( disable : 4054 )
#endif
//-----------------
#include "g_func_decs.h" // declare all game functions
funcList_t gameFuncList[] = {
#include "g_funcs.h"
};
#include "../botai/ai_func_decs.h" // declare all ai functions
funcList_t aiFuncList[] = {
#include "../botai/ai_funcs.h"
};
//-----------------
#ifdef _WIN32
#pragma warning( pop ) // return previous warning state
#endif
//-----------------
//=========================================================
/*
===============
G_SaveWriteError
===============
*/
void G_SaveWriteError( void ) {
// TTimo
#ifdef __linux__
G_Error( "Unable to save game.\n\nPlease check that you have at least 5mb free of disk space in your home directory." );
#else
G_Error( "Insufficient free disk space.\n\nPlease free at least 5mb of free space on game drive." );
#endif
}
static int saveByteCount;
/*
===============
G_SaveWrite
FS_Write doesnt always accurately return how many bytes were written, so tally them up, and
check them before we rename to the real file
===============
*/
int G_SaveWrite( const void *buffer, int len, fileHandle_t f ) {
saveByteCount += len;
return trap_FS_Write( buffer, len, f );
}
//=========================================================
funcList_t *G_FindFuncAtAddress( byte *adr ) {
int i;
for ( i = 0; gameFuncList[i].funcStr; i++ ) {
if ( gameFuncList[i].funcPtr == adr ) {
return &gameFuncList[i];
}
}
return NULL;
}
byte *G_FindFuncByName( char *name ) {
int i;
for ( i = 0; gameFuncList[i].funcStr; i++ ) {
if ( !strcmp( name, gameFuncList[i].funcStr ) ) {
return gameFuncList[i].funcPtr;
}
}
return NULL;
}
funcList_t *AI_FindFuncAtAddress( byte *adr ) {
int i;
for ( i = 0; aiFuncList[i].funcStr; i++ ) {
if ( aiFuncList[i].funcPtr == adr ) {
return &aiFuncList[i];
}
}
return NULL;
}
byte *AI_FindFuncByName( char *name ) {
int i;
for ( i = 0; aiFuncList[i].funcStr; i++ ) {
if ( !strcmp( name, aiFuncList[i].funcStr ) ) {
return aiFuncList[i].funcPtr;
}
}
return NULL;
}
void WriteField1( saveField_t *field, byte *base ) {
void *p;
int len;
int index;
funcList_t *func;
p = ( void * )( base + field->ofs );
switch ( field->type )
{
case F_STRING:
if ( *(char **)p ) {
len = strlen( *(char **)p ) + 1;
} else {
len = 0;
}
*(int *)p = len;
break;
case F_ENTITY:
if ( *(gentity_t **)p == NULL ) {
index = -1;
} else {
index = *(gentity_t **)p - g_entities;
}
if ( index >= MAX_GENTITIES || index < -1 ) {
G_Error( "WriteField1: entity out of range (%i)", index );
}
*(int *)p = index;
break;
case F_CLIENT:
if ( *(gclient_t **)p == NULL ) {
index = -1;
} else {
index = *(gclient_t **)p - level.clients;
}
if ( index >= MAX_CLIENTS || index < -1 ) {
G_Error( "WriteField1: client out of range (%i)", index );
}
*(int *)p = index;
break;
case F_ITEM:
if ( *(gitem_t **)p == NULL ) {
index = -1;
} else {
index = *(gitem_t **)p - bg_itemlist;
}
*(int *)p = index;
break;
// match this with a function address in the function list, which is built using the
// "extractfuncs.bat" in the utils folder. We then save the string equivalent
// of the function. This effectively gives us cross-version save games.
case F_FUNCTION:
if ( *(byte **)p == NULL ) {
len = 0;
} else {
func = G_FindFuncAtAddress( *(byte **)p );
if ( !func ) {
G_Error( "WriteField1: unknown function, cannot save game" );
}
len = strlen( func->funcStr ) + 1;
}
*(int *)p = len;
break;
case F_SPLINE:
if ( *(splinePath_t **)p == NULL ) {
index = -1;
} else {
index = *(splinePath_t **)p - splinePaths;
}
*(int *)p = index;
break;
// match this with a function address in the function list, which is built using the
// "extractfuncs.bat" in the utils folder. We then save the string equivalent
// of the function. This effectively gives us cross-version save games.
case F_AIFUNCTION:
if ( *(byte **)p == NULL ) {
len = 0;
} else {
func = AI_FindFuncAtAddress( *(byte **)p );
if ( !func ) {
G_Error( "WriteField1: unknown ai function, cannot save game" );
}
len = strlen( func->funcStr ) + 1;
}
*(int *)p = len;
break;
case F_BOTWAYPOINT:
if ( *(bot_waypoint_t **)p == NULL ) {
index = -1;
} else {
index = *(bot_waypoint_t **)p - botai_waypoints;
}
*(int *)p = index;
break;
case F_BOTSCRIPTDATA:
if ( *(bot_script_data_t **)p == NULL ) {
index = -1;
} else {
index = Bot_FindSriptGlobalData( *(bot_script_data_t **)p );
if ( index < 0 ) {
G_Error( "WriteField1: unknown bot script data, cannot save game" );
}
}
*(int *)p = index;
break;
default:
G_Error( "WriteField1: unknown field type" );
}
}
void WriteField2( fileHandle_t f, saveField_t *field, byte *base ) {
int len;
void *p;
funcList_t *func;
p = ( void * )( base + field->ofs );
switch ( field->type )
{
case F_STRING:
if ( *(char **)p ) {
len = strlen( *(char **)p ) + 1;
if ( !G_SaveWrite( *(char **)p, len, f ) ) {
G_SaveWriteError();
}
}
break;
case F_FUNCTION:
if ( *(byte **)p ) {
func = G_FindFuncAtAddress( *(byte **)p );
if ( !func ) {
G_Error( "WriteField2: unknown function, cannot save game" );
}
len = strlen( func->funcStr ) + 1;
if ( !G_SaveWrite( func->funcStr, len, f ) ) {
G_SaveWriteError();
}
}
break;
case F_AIFUNCTION:
if ( *(byte **)p ) {
func = AI_FindFuncAtAddress( *(byte **)p );
if ( !func ) {
G_Error( "WriteField2: unknown function, cannot save game" );
}
len = strlen( func->funcStr ) + 1;
if ( !G_SaveWrite( func->funcStr, len, f ) ) {
G_SaveWriteError();
}
}
break;
default:
break;
}
}
void ReadField( fileHandle_t f, saveField_t *field, byte *base ) {
void *p;
int len;
int index;
char funcStr[512];
p = ( void * )( base + field->ofs );
switch ( field->type )
{
case F_STRING:
len = *(int *)p;
if ( !len ) {
*(char **)p = NULL;
} else
{
*(char **)p = G_Alloc( len );
trap_FS_Read( *(char **)p, len, f );
}
break;
case F_ENTITY:
index = *(int *)p;
if ( index >= MAX_GENTITIES || index < -1 ) {
G_Error( "ReadField: entity out of range (%i)", index );
}
if ( index == -1 ) {
*(gentity_t **)p = NULL;
} else {
*(gentity_t **)p = &g_entities[index];
}
break;
case F_CLIENT:
index = *(int *)p;
if ( index >= MAX_CLIENTS || index < -1 ) {
G_Error( "ReadField: client out of range (%i)", index );
}
if ( index == -1 ) {
*(gclient_t **)p = NULL;
} else {
*(gclient_t **)p = &level.clients[index];
}
break;
case F_ITEM:
index = *(int *)p;
if ( index == -1 ) {
*(gitem_t **)p = NULL;
} else {
*(gitem_t **)p = &bg_itemlist[index];
}
break;
//relative to code segment
case F_FUNCTION:
len = *(int *)p;
if ( !len ) {
*(byte **)p = NULL;
} else
{
//funcStr = G_Alloc (len);
if ( len > sizeof( funcStr ) ) {
G_Error( "ReadField: function name is greater than buffer (%i chars)", sizeof( funcStr ) );
}
trap_FS_Read( funcStr, len, f );
if ( ( *(byte **)p = G_FindFuncByName( funcStr ) ) == NULL ) {
G_Error( "ReadField: unknown function '%s'\ncannot load game", funcStr );
}
}
break;
case F_SPLINE:
index = *(int *)p;
if ( index == -1 ) {
*(splinePath_t **)p = NULL;
} else {
*(splinePath_t **)p = &splinePaths[index];
}
break;
//relative to code segment
case F_AIFUNCTION:
len = *(int *)p;
if ( !len ) {
*(byte **)p = NULL;
} else
{
//funcStr = G_Alloc (len);
if ( len > sizeof( funcStr ) ) {
G_Error( "ReadField: function name is greater than buffer (%i chars)", sizeof( funcStr ) );
}
trap_FS_Read( funcStr, len, f );
if ( ( *(byte **)p = AI_FindFuncByName( funcStr ) ) == NULL ) {
G_Error( "ReadField: unknown function '%s'\ncannot load game", funcStr );
}
}
break;
case F_BOTWAYPOINT:
index = *(int *)p;
if ( index == -1 ) {
*(bot_waypoint_t **)p = NULL;
} else {
*(bot_waypoint_t **)p = &botai_waypoints[index];
}
break;
case F_BOTSCRIPTDATA:
index = *(int *)p;
if ( index == -1 ) {
*(bot_script_data_t **)p = NULL;
} else {
*(bot_script_data_t **)p = botCharacterScriptData[index].data;
}
break;
default:
G_Error( "ReadField: unknown field type" );
}
}
//=========================================================
#define SAVE_ENCODE_COUNT_BYTES 1
/*
===============
G_Save_Encode
returns the number of bytes written to "out"
===============
*/
int G_Save_Encode( byte *raw, byte *out, int rawsize, int outsize ) {
int rawcount, oldrawcount, outcount;
int mode;
byte count; //DAJ was int but caused endian bugs
rawcount = 0;
outcount = 0;
while ( rawcount < rawsize ) {
oldrawcount = rawcount;
// is this a non-zero?
if ( raw[rawcount] ) {
mode = 1;
} else {
mode = 0;
}
// calc the count
count = 0;
while ( rawcount < rawsize && ( raw[rawcount] != 0 ) == mode && count < ( ( ( 1 << ( SAVE_ENCODE_COUNT_BYTES * 8 - 1 ) ) - 1 ) ) ) {
rawcount++;
count++;
}
// write the count, followed by data if required
memcpy( out + outcount, &count, SAVE_ENCODE_COUNT_BYTES );
// switch the sign bit if zeros
if ( !mode ) {
out[outcount + SAVE_ENCODE_COUNT_BYTES - 1] |= ( 1 << 7 );
outcount += SAVE_ENCODE_COUNT_BYTES;
} else {
outcount += SAVE_ENCODE_COUNT_BYTES;
// write the data
memcpy( out + outcount, raw + oldrawcount, count );
outcount += count;
}
}
return outcount;
}
/*
===============
G_Save_Decode
===============
*/
void G_Save_Decode( byte *in, int insize, byte *out, int outsize ) {
int incount, outcount;
byte count; //DAJ was in but caused endian bugs
//
incount = 0;
outcount = 0;
while ( incount < insize ) {
// read the count
count = 0;
memcpy( &count, in + incount, SAVE_ENCODE_COUNT_BYTES );
incount += SAVE_ENCODE_COUNT_BYTES;
// if it's negative, zero it out
if ( count & ( 1 << ( ( SAVE_ENCODE_COUNT_BYTES * 8 ) - 1 ) ) ) {
count &= ~( 1 << ( ( SAVE_ENCODE_COUNT_BYTES * 8 ) - 1 ) );
memset( out + outcount, 0, count );
outcount += count;
} else {
// copy the data from "in"
memcpy( out + outcount, in + incount, count );
outcount += count;
incount += count;
}
}
}
//=========================================================
byte clientBuf[ 2 * sizeof( gentity_t ) ];
/*
===============
WriteClient
===============
*/
void WriteClient( fileHandle_t f, gclient_t *cl ) {
saveField_t *field;
gclient_t temp;
int length;
// copy the structure across, then process the fields
temp = *cl;
// first, kill all events (assume they have been processed)
memset( temp.ps.events, 0, sizeof( temp.ps.events ) );
memset( temp.ps.eventParms, 0, sizeof( temp.ps.eventParms ) );
temp.ps.eventSequence = 0;
temp.ps.oldEventSequence = 0;
temp.ps.entityEventSequence = 0;
// change the pointers to lengths or indexes
for ( field = gclientFields ; field->type ; field++ )
{
WriteField1( field, (byte *)&temp );
}
// write the block
//if (!G_SaveWrite (&temp, sizeof(temp), f)) G_SaveWriteError();
length = G_Save_Encode( (byte *)&temp, clientBuf, sizeof( temp ), sizeof( clientBuf ) );
if ( !G_SaveWrite( &length, sizeof( length ), f ) ) {
G_SaveWriteError();
}
if ( !G_SaveWrite( &clientBuf, length, f ) ) {
G_SaveWriteError();
}
// now write any allocated data following the edict
for ( field = gclientFields ; field->type ; field++ )
{
WriteField2( f, field, (byte *)cl );
}
}
/*
===============
ReadClient
===============
*/
void ReadClient( fileHandle_t f, gclient_t *client, int size ) {
saveField_t *field;
ignoreField_t *ifield;
gclient_t temp;
gentity_t *ent;
int decodedSize;
// read the encoded chunk
trap_FS_Read( &decodedSize, sizeof( int ), f );
if ( decodedSize > sizeof( clientBuf ) ) {
G_Error( "G_LoadGame: encoded chunk is greater than buffer" );
}
trap_FS_Read( clientBuf, decodedSize, f ); \
// decode it
G_Save_Decode( clientBuf, decodedSize, (byte *)&temp, sizeof( temp ) );
// convert any feilds back to the correct data
for ( field = gclientFields ; field->type ; field++ )
{
ReadField( f, field, (byte *)&temp );
}
// backup any fields that we don't want to read in
for ( ifield = gclientIgnoreFields ; ifield->len ; ifield++ )
{
memcpy( ( (byte *)&temp ) + ifield->ofs, ( (byte *)client ) + ifield->ofs, ifield->len );
}
// now copy the temp structure into the existing structure
memcpy( client, &temp, size );
// make sure they face the right way
//client->ps.pm_flags |= PMF_RESPAWNED;
// don't allow full run speed for a bit
//if (client->ps.clientNum == 0) { // only set this for the player
client->ps.pm_flags |= PMF_TIME_LOAD;
client->ps.pm_time = 1000;
//}
ent = &g_entities[client->ps.clientNum];
if ( ent->r.svFlags & SVF_BOT ) {
if ( trap_BotAllocateClient( client->ps.clientNum ) < 0 ) {
G_Error( "G_LoadGame: can't allocate required client for bot" );
}
}
// make sure they face the right way
trap_GetUsercmd( ent->client - level.clients, &ent->client->pers.cmd );
SetClientViewAngle( ent, ent->client->ps.viewangles );
// dead characters should stay on last frame after a loadgame
if ( client->ps.eFlags & EF_DEAD ) {
client->ps.eFlags |= EF_FORCE_END_FRAME;
}
// RF, disabled, not required now with screen fading, causes characters to possibly spawn events
// before they are known in the cgame
// run a client frame to drop exactly to the floor,
// initialize animations and other things
//trap_GetUsercmd( ent-g_entities, &ent->client->pers.cmd );
//ent->client->ps.commandTime = ent->client->pers.cmd.serverTime - 100;
//ClientThink( ent-g_entities );
ClientUserinfoChanged( ent->client - level.clients );
// tell the client to reset it's cgame stuff
if ( !( ent->r.svFlags & SVF_BOT ) ) {
vmCvar_t cvar;
// tell it which weapon to use after spawning in
trap_Cvar_Register( &cvar, "cg_loadWeaponSelect", "0", CVAR_ROM );
trap_Cvar_Set( "cg_loadWeaponSelect", va( "%i", client->ps.weapon ) );
//
trap_SendServerCommand( client->ps.clientNum, "map_restart\n" );
}
}
//=========================================================
byte entityBuf[ 2 * sizeof( gentity_t ) ];
/*
===============
WriteEntity
===============
*/
void WriteEntity( fileHandle_t f, gentity_t *ent ) {
saveField_t *field;
gentity_t temp;
int length;
// copy the structure across, then process the fields
temp = *ent;
// first, kill all events (assume they have been processed)
memset( temp.s.events, 0, sizeof( temp.s.events ) );
memset( temp.s.eventParms, 0, sizeof( temp.s.eventParms ) );
temp.s.eventSequence = 0;
// change the pointers to lengths or indexes
for ( field = gentityFields ; field->type ; field++ )
{
WriteField1( field, (byte *)&temp );
}
// write the block
//if (!G_SaveWrite (&temp, sizeof(temp), f)) G_SaveWriteError();
length = G_Save_Encode( (byte *)&temp, entityBuf, sizeof( temp ), sizeof( entityBuf ) );
if ( !G_SaveWrite( &length, sizeof( length ), f ) ) {
G_SaveWriteError();
}
if ( !G_SaveWrite( &entityBuf, length, f ) ) {
G_SaveWriteError();
}
// now write any allocated data following the edict
for ( field = gentityFields ; field->type ; field++ )
{
WriteField2( f, field, (byte *)ent );
}
}
/*
===============
ReadEntity
===============
*/
void ReadEntity( fileHandle_t f, gentity_t *ent, int size ) {
saveField_t *field;
ignoreField_t *ifield;
gentity_t temp, backup, backup2;
// vmCvar_t cvar;
int decodedSize;
backup = *ent;
// read the encoded chunk
trap_FS_Read( &decodedSize, sizeof( int ), f );
if ( decodedSize > sizeof( entityBuf ) ) {
G_Error( "G_LoadGame: encoded chunk is greater than buffer" );
}
trap_FS_Read( entityBuf, decodedSize, f );
// decode it
G_Save_Decode( entityBuf, decodedSize, (byte *)&temp, sizeof( temp ) );
// convert any fields back to the correct data
for ( field = gentityFields ; field->type ; field++ )
{
ReadField( f, field, (byte *)&temp );
}
// backup any fields that we don't want to read in
for ( ifield = gentityIgnoreFields ; ifield->len ; ifield++ )
{
memcpy( ( (byte *)&temp ) + ifield->ofs, ( (byte *)ent ) + ifield->ofs, ifield->len );
}
// kill all events (assume they have been processed)
if ( !temp.freeAfterEvent ) {
temp.s.event = 0;
memset( temp.s.events, 0, sizeof( temp.s.events ) );
memset( temp.s.eventParms, 0, sizeof( temp.s.eventParms ) );
temp.s.eventSequence = 0;
temp.eventTime = 0;
}
// now copy the temp structure into the existing structure
memcpy( ent, &temp, size );
// notify server of changes in position/orientation
if ( ent->r.linked /*&& (!(ent->r.svFlags & SVF_BOT) || !ent->aiInactive)*/ ) {
trap_LinkEntity( ent );
} else {
trap_UnlinkEntity( ent );
}
// if this is a mover, check areaportals
if ( ent->s.eType == ET_MOVER && ent->moverState != backup.moverState ) {
if ( ent->teammaster == ent || !ent->teammaster ) {
if ( ent->moverState == MOVER_POS1ROTATE || ent->moverState == MOVER_POS1 ) {
// closed areaportal
trap_AdjustAreaPortalState( ent, qfalse );
} else { // must be open
// portals are always opened before the mover starts to open, so we must move
// it back to the start position, link, set portals, then move it back
backup2 = *ent;
*ent = backup;
// link it at original position
trap_LinkEntity( ent );
// set portals
trap_AdjustAreaPortalState( ent, qtrue );
// put it back
*ent = backup2;
trap_LinkEntity( ent );
}
}
}
// check for blocking AAS at save time
if ( ent->AASblocking ) {
G_SetAASBlockingEntity( ent, AAS_AREA_DISABLED );
}
// check for this being a tagconnect entity
if ( ent->tagName && ent->tagParent ) { // the parent might not be there yet
G_ProcessTagConnect( ent, qfalse );
}
}
//=========================================================
/*
==============
WriteTime
==============
*/
void WriteTime( fileHandle_t f ) {
qtime_t tm;
// just save it all so it can be interpreted as desired
trap_RealTime( &tm );
G_SaveWrite( &tm.tm_sec, sizeof( tm.tm_sec ), f ); /* seconds after the minute - [0,59] */
G_SaveWrite( &tm.tm_min, sizeof( tm.tm_min ), f ); /* minutes after the hour - [0,59] */
G_SaveWrite( &tm.tm_hour, sizeof( tm.tm_hour ), f ); /* hours since midnight - [0,23] */
G_SaveWrite( &tm.tm_mday, sizeof( tm.tm_mday ), f ); /* day of the month - [1,31] */
G_SaveWrite( &tm.tm_mon, sizeof( tm.tm_mon ), f ); /* months since January - [0,11] */
G_SaveWrite( &tm.tm_year, sizeof( tm.tm_year ), f ); /* years since 1900 */
G_SaveWrite( &tm.tm_wday, sizeof( tm.tm_wday ), f ); /* days since Sunday - [0,6] */
G_SaveWrite( &tm.tm_yday, sizeof( tm.tm_yday ), f ); /* days since January 1 - [0,365] */
G_SaveWrite( &tm.tm_isdst, sizeof( tm.tm_isdst ),f ); /* daylight savings time flag */
}
/*
==============
ReadTime
==============
*/
void ReadTime( fileHandle_t f, qtime_t *tm ) {
trap_FS_Read( &tm->tm_sec, sizeof( tm->tm_sec ), f );
trap_FS_Read( &tm->tm_min, sizeof( tm->tm_min ), f );
trap_FS_Read( &tm->tm_hour, sizeof( tm->tm_hour ), f );
trap_FS_Read( &tm->tm_mday, sizeof( tm->tm_mday ), f );
trap_FS_Read( &tm->tm_mon, sizeof( tm->tm_mon ), f );
trap_FS_Read( &tm->tm_year, sizeof( tm->tm_year ), f );
trap_FS_Read( &tm->tm_wday, sizeof( tm->tm_wday ), f );
trap_FS_Read( &tm->tm_yday, sizeof( tm->tm_yday ), f );
trap_FS_Read( &tm->tm_isdst, sizeof( tm->tm_isdst ), f );
}
/*
==============
G_Save_TimeStr
==============
*/
char *G_Save_TimeStr( void ) {
qtime_t tm;
//
trap_RealTime( &tm );
//
return va( "%2i:%s%i:%s%i %s",
( 1 + ( tm.tm_hour + 11 ) % 12 ), // 12 hour format
( tm.tm_min > 9 ? "" : "0" ), // minute padding
tm.tm_min,
( tm.tm_sec > 9 ? "" : "0" ), // second padding
tm.tm_sec,
( tm.tm_hour < 12 ? "am" : "pm" ) );
}
static char *monthStr[12] =
{
"JAN", "FEB", "MAR", "APR", "MAY", "JUN", "JUL", "AUG", "SEP", "OCT", "NOV", "DEC"
};
/*
==============
G_Save_DateStr
==============
*/
char *G_Save_DateStr( void ) {
qtime_t tm;
//
trap_RealTime( &tm );
//
return va( "%s %i, %i",
monthStr[tm.tm_mon],
tm.tm_mday,
1900 + tm.tm_year );
}
//=========================================================
byte botwaypointBuf[ 2 * sizeof( bot_waypoint_t ) ];
/*
===============
WriteBotWaypoint
===============
*/
void WriteBotWaypoint( fileHandle_t f, bot_waypoint_t *wp ) {
saveField_t *field;
bot_waypoint_t temp;
int length;
// copy the structure across, then process the fields
temp = *wp;
// change the pointers to lengths or indexes
for ( field = botwaypointFields ; field->type ; field++ )
{
WriteField1( field, (byte *)&temp );
}
// write the block
length = G_Save_Encode( (byte *)&temp, botwaypointBuf, sizeof( temp ), sizeof( botwaypointBuf ) );
if ( !G_SaveWrite( &length, sizeof( length ), f ) ) {
G_SaveWriteError();
}
if ( !G_SaveWrite( &botwaypointBuf, length, f ) ) {
G_SaveWriteError();
}
// now write any allocated data following the edict
for ( field = botwaypointFields ; field->type ; field++ )
{
WriteField2( f, field, (byte *)wp );
}
}
/*
===============
ReadBotWaypoint
===============
*/
void ReadBotWaypoint( fileHandle_t f, bot_waypoint_t *wp, int size ) {
saveField_t *field;
bot_waypoint_t temp;
int decodedSize;
// read the encoded chunk
trap_FS_Read( &decodedSize, sizeof( int ), f );
if ( decodedSize > sizeof( botwaypointBuf ) ) {
G_Error( "G_LoadGame: encoded chunk is greater than buffer" );
}
trap_FS_Read( botwaypointBuf, decodedSize, f ); \
// decode it
G_Save_Decode( botwaypointBuf, decodedSize, (byte *)&temp, sizeof( temp ) );
// convert any feilds back to the correct data
for ( field = botwaypointFields ; field->type ; field++ )
{
ReadField( f, field, (byte *)&temp );
}
// now copy the temp structure into the existing structure
memcpy( wp, &temp, size );
}
//=========================================================
byte botstateBuf[ 2 * sizeof( bot_state_t ) ];
/*
===============
WriteBotState
===============
*/
void WriteBotState( fileHandle_t f, bot_state_t *bs ) {
saveField_t *field;
bot_state_t temp;
int length;
// copy the structure across, then process the fields
temp = *bs;
// change the pointers to lengths or indexes
for ( field = botstateFields ; field->type ; field++ )
{
WriteField1( field, (byte *)&temp );
}
// write the block
length = G_Save_Encode( (byte *)&temp, botstateBuf, sizeof( temp ), sizeof( botstateBuf ) );
if ( !G_SaveWrite( &length, sizeof( length ), f ) ) {
G_SaveWriteError();
}
if ( !G_SaveWrite( &botstateBuf, length, f ) ) {
G_SaveWriteError();
}
// now write any allocated data following the edict
for ( field = botstateFields ; field->type ; field++ )
{
WriteField2( f, field, (byte *)bs );
}
}
/*
===============
ReadBotState
===============
*/
void ReadBotState( fileHandle_t f, bot_state_t *bs, int size ) {
saveField_t *field;
ignoreField_t *ifield;
bot_state_t temp;
int decodedSize;
char filename[MAX_QPATH], name[MAX_QPATH], gender[MAX_QPATH];
int errnum;
// read the encoded chunk
trap_FS_Read( &decodedSize, sizeof( int ), f );
if ( decodedSize > sizeof( botstateBuf ) ) {
G_Error( "G_LoadGame: encoded chunk is greater than buffer" );
}
trap_FS_Read( botstateBuf, decodedSize, f ); \
// decode it
G_Save_Decode( botstateBuf, decodedSize, (byte *)&temp, sizeof( temp ) );
// convert any feilds back to the correct data
for ( field = botstateFields ; field->type ; field++ )
{
ReadField( f, field, (byte *)&temp );
}
// backup any fields that we don't want to read in
for ( ifield = botstateIgnoreFields ; ifield->len ; ifield++ )
{
memcpy( ( (byte *)&temp ) + ifield->ofs, ( (byte *)bs ) + ifield->ofs, ifield->len );
}
// now copy the temp structure into the existing structure
memcpy( bs, &temp, size );
// initialize character and set botstates in engine
//load the bot character
bs->character = trap_BotLoadCharacter( bs->settings.characterfile, bs->settings.skill );
if ( !bs->character ) {
G_Error( S_COLOR_RED, "Fatal: couldn't load skill %d from %s\n", bs->settings.skill, bs->settings.characterfile );
}
//allocate a goal state
bs->gs = trap_BotAllocGoalState( bs->client );
//load the item weights
trap_Characteristic_String( bs->character, CHARACTERISTIC_ITEMWEIGHTS, filename, MAX_QPATH );
errnum = trap_BotLoadItemWeights( bs->gs, filename );
if ( errnum != BLERR_NOERROR ) {
trap_BotFreeGoalState( bs->gs );
G_Error( S_COLOR_RED, "Fatal: ReadBotState failed\n" );
}
//allocate a weapon state
bs->ws = trap_BotAllocWeaponState();
//load the weapon weights
trap_Characteristic_String( bs->character, CHARACTERISTIC_WEAPONWEIGHTS, filename, MAX_QPATH );
errnum = trap_BotLoadWeaponWeights( bs->ws, filename );
if ( errnum != BLERR_NOERROR ) {
trap_BotFreeGoalState( bs->gs );
trap_BotFreeWeaponState( bs->ws );
G_Error( S_COLOR_RED, "Fatal: ReadBotState failed\n" );
}
//allocate a chat state
bs->cs = trap_BotAllocChatState();
//load the chat file
trap_Characteristic_String( bs->character, CHARACTERISTIC_CHAT_FILE, filename, MAX_QPATH );
trap_Characteristic_String( bs->character, CHARACTERISTIC_CHAT_NAME, name, MAX_QPATH );
errnum = trap_BotLoadChatFile( bs->cs, filename, name );
if ( errnum != BLERR_NOERROR ) {
trap_BotFreeChatState( bs->cs );
trap_BotFreeGoalState( bs->gs );
trap_BotFreeWeaponState( bs->ws );
G_Error( S_COLOR_RED, "Fatal: ReadBotState failed\n" );
}
//get the gender characteristic
trap_Characteristic_String( bs->character, CHARACTERISTIC_GENDER, gender, MAX_QPATH );
//set the chat gender
if ( *gender == 'f' || *gender == 'F' ) {
trap_BotSetChatGender( bs->cs, CHAT_GENDERFEMALE );
} else if ( *gender == 'm' || *gender == 'M' ) {
trap_BotSetChatGender( bs->cs, CHAT_GENDERMALE );
} else { trap_BotSetChatGender( bs->cs, CHAT_GENDERLESS );}
bs->ms = trap_BotAllocMoveState();
bs->walker = trap_Characteristic_BFloat( bs->character, CHARACTERISTIC_WALKER, 0, 1 );
BotGetInitialAttributes( bs );
trap_EA_ResetInput( bs->client, NULL );
BotAI_SetNumBots( BotAI_GetNumBots() + 1 );
}
//=========================================================
static char infoString[SAVE_INFOSTRING_LENGTH];
/*
===============
G_SaveGame
returns qtrue if successful
TODO: have G_SaveWrite return the number of byte's written, so if it doesn't
succeed, we can abort the save, and not save the file. This means we should
save to a temporary name, then copy it across to the real name after success,
so full disks don't result in lost saved games.
===============
*/
qboolean G_SaveGame( char *username ) {
char filename[MAX_QPATH];
char mapstr[MAX_QPATH];
//char leveltime[MAX_QPATH];
//char healthstr[MAX_QPATH];
vmCvar_t mapname;
fileHandle_t f;
int i, len;
gentity_t *ent;
gclient_t *cl;
bot_waypoint_t *wp;
bot_state_t *bs;
//int playtime, minutes;
char cl_profileStr[256];
long hash;
char *ch;
char userinfo[MAX_INFO_STRING];
//if (reloading)
// return qtrue; // actually this should be qtrue, but we should make it silent during reloading
if ( g_entities[0].health <= 0 ) { // no save when dead
return qtrue;
}
if ( g_gametype.integer != GT_SINGLE_PLAYER ) { // don't allow saves in MP
return qtrue;
}
G_DPrintf( "G_SaveGame '%s'\n", username );
if ( !username ) {
username = "current";
}
// validate the filename
for ( i = 0; i < strlen( username ); i++ ) {
if ( !Q_isforfilename( username[i] ) && username[i] != '\\' ) { // (allow '\\' so games can be saved in subdirs)
G_Printf( "G_SaveGame: '%s'. Invalid character (%c) in filename. Must use alphanumeric characters only.\n", username, username[i] );
return qtrue;
}
}
saveByteCount = 0;
trap_Cvar_VariableStringBuffer( "cl_profile", cl_profileStr, 256 );
// generate hash for profile
for ( hash = 0, ch = (char *)cl_profileStr; *ch != '\0'; ch++ ) {
hash += (long)( tolower( *ch ) ) * ( ( ch - cl_profileStr ) + 119 );
}
// open the file
Com_sprintf( filename, MAX_QPATH, "profiles/%s/save/temp.sav", cl_profileStr );
if ( trap_FS_FOpenFile( filename, &f, FS_WRITE ) < 0 ) {
G_Error( "G_SaveGame: cannot open file for saving\n" );
}
// write the version
i = SAVE_VERSION;
ver = SAVE_VERSION;
if ( !G_SaveWrite( &i, sizeof( i ), f ) ) {
G_SaveWriteError();
}
// write the mapname
trap_Cvar_Register( &mapname, "mapname", "", CVAR_SERVERINFO | CVAR_ROM );
Com_sprintf( mapstr, MAX_QPATH, mapname.string );
if ( !G_SaveWrite( mapstr, MAX_QPATH, f ) ) {
G_SaveWriteError();
}
// write out the level time
if ( !G_SaveWrite( &level.time, sizeof( level.time ), f ) ) {
G_SaveWriteError();
}
// write out current time/date info
WriteTime( f );
// write profile hash
if ( !G_SaveWrite( &hash, sizeof( hash ), f ) ) {
G_SaveWriteError();
}
// create and write the info string
// (SA) I made another cvar so there's no confusion
/*Q_strncpyz( mapstr, mapname.string, sizeof(mapstr) );
for (i=0; i 9 ? "" : "0"), // minute padding
minutes,
((playtime/1000)%60 > 9 ? "" : "0"), // second padding
((playtime/1000)%60) ),
healthstr,
g_entities[0].health );
// write it out
// length
i = strlen( infoString );
if (!G_SaveWrite (&i, sizeof(i), f)) {
G_SaveWriteError();
}
// string
if (!G_SaveWrite (infoString, strlen(infoString), f)) {
G_SaveWriteError();
}*/
//----(SA) added
//----(SA) end
// write music
/* trap_Cvar_Register( &musicCvar, "s_currentMusic", "", CVAR_ROM );
if (!G_SaveWrite (musicCvar.string, MAX_QPATH, f)) {
G_SaveWriteError();
}*/
// trap_Cvar_VariableStringBuffer( "sg_fog", infoString, sizeof(infoString) );
trap_GetConfigstring( CS_FOGVARS, infoString, sizeof( infoString ) );
i = strlen( infoString );
if ( !G_SaveWrite( &i, sizeof( i ), f ) ) {
G_SaveWriteError();
}
// if there's fog info to save
if ( i ) {
if ( !G_SaveWrite( infoString, strlen( infoString ), f ) ) {
G_SaveWriteError();
}
}
if ( ver >= 31 ) {
trap_GetConfigstring( CS_GLOBALFOGVARS, infoString, sizeof( infoString ) );
i = strlen( infoString );
if ( !G_SaveWrite( &i, sizeof( i ), f ) ) {
G_SaveWriteError();
}
// if there's fog info to save
if ( i ) {
if ( !G_SaveWrite( infoString, strlen( infoString ), f ) ) {
G_SaveWriteError();
}
}
}
// write out the entity structures
i = sizeof( gentity_t );
if ( !G_SaveWrite( &i, sizeof( i ), f ) ) {
G_SaveWriteError();
}
for ( i = 0 ; i < level.num_entities ; i++ )
{
ent = &g_entities[i];
if ( !ent->inuse || ent->s.number == ENTITYNUM_WORLD ) {
continue;
}
if ( !G_SaveWrite( &i, sizeof( i ), f ) ) {
G_SaveWriteError();
}
WriteEntity( f, ent );
}
i = -1;
if ( !G_SaveWrite( &i, sizeof( i ), f ) ) {
G_SaveWriteError();
}
// write out the client structures
i = sizeof( gclient_t );
if ( !G_SaveWrite( &i, sizeof( i ), f ) ) {
G_SaveWriteError();
}
for ( i = 0 ; i < MAX_CLIENTS ; i++ )
{
cl = &level.clients[i];
if ( cl->pers.connected != CON_CONNECTED ) {
continue;
}
if ( !G_SaveWrite( &i, sizeof( i ), f ) ) {
G_SaveWriteError();
}
// store userinfo if bot
if ( g_entities[i].r.svFlags & SVF_BOT ) {
trap_GetUserinfo( i, userinfo, sizeof( userinfo ) );
len = strlen( userinfo );
if ( !G_SaveWrite( &len, sizeof( len ), f ) ) {
G_SaveWriteError();
}
if ( len ) {
if ( !G_SaveWrite( userinfo, strlen( userinfo ), f ) ) {
G_SaveWriteError();
}
}
}
WriteClient( f, cl );
}
i = -1;
if ( !G_SaveWrite( &i, sizeof( i ), f ) ) {
G_SaveWriteError();
}
// write out the botwaypoint linked list
i = sizeof( bot_waypoint_t );
if ( !G_SaveWrite( &i, sizeof( i ), f ) ) {
G_SaveWriteError();
}
for ( i = 0 ; i < MAX_BOTAIWAYPOINTS ; i++ )
{
wp = &botai_waypoints[i];
WriteBotWaypoint( f, wp );
}
if ( botai_freewaypoints ) {
i = botai_freewaypoints - botai_waypoints;
} else {
i = -1;
}
if ( !G_SaveWrite( &i, sizeof( i ), f ) ) {
G_SaveWriteError();
}
// write out the botstate structures
i = sizeof( bot_state_t );
if ( !G_SaveWrite( &i, sizeof( i ), f ) ) {
G_SaveWriteError();
}
for ( i = 0 ; i < MAX_CLIENTS ; i++ )
{
bs = &botstates[i];
if ( !bs || !bs->inuse ) {
continue;
}
// if the parent client isn't connected
if ( level.clients[i].pers.connected != CON_CONNECTED ) {
continue;
}
if ( !G_SaveWrite( &i, sizeof( i ), f ) ) {
G_SaveWriteError();
}
WriteBotState( f, bs );
}
i = -1;
if ( !G_SaveWrite( &i, sizeof( i ), f ) ) {
G_SaveWriteError();
}
trap_FS_FCloseFile( f );
// check the byte count
if ( ( len = trap_FS_FOpenFile( filename, &f, FS_READ ) ) != saveByteCount ) {
trap_FS_FCloseFile( f );
G_SaveWriteError();
return qfalse;
}
trap_FS_FCloseFile( f );
// now rename the file to the actual file
Com_sprintf( mapstr, MAX_QPATH, "profiles/%s/save/%s.sav", cl_profileStr, username );
trap_FS_Rename( filename, mapstr );
// double check that it saved ok
if ( ( len = trap_FS_FOpenFile( mapstr, &f, FS_READ ) ) != saveByteCount ) {
trap_FS_FCloseFile( f );
G_SaveWriteError();
return qfalse;
}
trap_FS_FCloseFile( f );
#ifdef __MACOS__
trap_FS_CopyFile( mapstr, va( "profiles/%s/save/current.sav", cl_profileStr );
#endif
return qtrue;
}
/*
===============
G_LoadGame
Always loads in "current.svg". So if loading a specific savegame, first copy it to that.
===============
*/
void G_LoadGame( void ) {
char filename[MAX_QPATH];
char mapname[MAX_QPATH];
fileHandle_t f;
int i, leveltime, size, last, len;
gentity_t *ent;
gclient_t *cl;
bot_waypoint_t *wp;
bot_state_t *bs;
qtime_t tm;
qboolean serverEntityUpdate = qfalse;
char *p;
int k;
char cl_profileStr[256];
long hash, fhash;
char *ch;
char userinfo[MAX_INFO_STRING];
if ( g_gametype.integer != GT_SINGLE_PLAYER ) { // don't allow loads in MP
return;
}
if ( saveGamePending ) {
return;
}
G_DPrintf( "G_LoadGame '%s'\n", filename );
trap_Cvar_VariableStringBuffer( "cl_profile", cl_profileStr, 256 );
// generate hash for profile
for ( hash = 0, ch = (char *)cl_profileStr; *ch != '\0'; ch++ ) {
hash += (long)( tolower( *ch ) ) * ( ( ch - cl_profileStr ) + 119 );
}
// enforce the "current" savegame, since that is used for all loads
Com_sprintf( filename, sizeof( filename ), "profiles/%s/save/current.sav", cl_profileStr );
// open the file
if ( trap_FS_FOpenFile( filename, &f, FS_READ ) < 0 ) {
G_Error( "G_LoadGame: savegame '%s' not found\n", filename );
}
// read the version
trap_FS_Read( &i, sizeof( i ), f );
if ( i != SAVE_VERSION ) {
trap_FS_FCloseFile( f );
G_Error( "G_LoadGame: savegame '%s' is wrong version (%i, should be %i)\n", filename, i, SAVE_VERSION );
}
ver = i;
// read the mapname (this is only used in the sever exe, so just discard it)
trap_FS_Read( mapname, MAX_QPATH, f );
// read the level time
trap_FS_Read( &i, sizeof( i ), f );
leveltime = i;
// NOTE: do not change the above order without also changing the server code
// read current time/date info
ReadTime( f, &tm );
// read profile hash
trap_FS_Read( &fhash, sizeof( fhash ), f );
if ( hash != fhash ) {
trap_FS_FCloseFile( f );
G_Error( "G_LoadGame: savegame '%s' is for another profile\n", filename, i, SAVE_VERSION );
}
// read music
/*trap_FS_Read(musicString, MAX_QPATH, f);
if(strlen(musicString)) {
trap_Cvar_Register( &musicCvar, "s_currentMusic", "", CVAR_ROM ); // get current music
if(Q_stricmp(musicString, musicCvar.string)) { // it's different than what's playing, so fade out and queue up
// trap_SendServerCommand(-1, "mu_fade 0 1000\n");
// trap_SetConfigstring( CS_MUSIC_QUEUE, musicString);
trap_SendServerCommand(-1, va("mu_start %s 1000\n", musicString) ); // (SA) trying this instead
}
}*/
// get length
trap_FS_Read( &i, sizeof( i ), f );
// get fog string
if ( i ) {
trap_FS_Read( infoString, i, f );
infoString[i] = 0;
} else {
Q_strncpyz( infoString, "none", sizeof( infoString ) );
}
// set the configstring so the 'savegame current' has good fog
if ( !Q_stricmp( infoString, "none" ) ) { // no fog
trap_Cvar_Set( "r_savegameFogColor", "none" );
} else {
// send it off to get set on the client
for ( p = &infoString[0],k = 0; *p; p++ ) {
if ( *p == ' ' ) {
k++;
}
if ( k == 6 ) { // the last parameter
infoString[p - infoString + 1] = '0';
infoString[p - infoString + 2] = 0;
break;
}
}
trap_Cvar_Set( "r_savegameFogColor", infoString );
}
if ( ver >= 31 ) {
// get length
trap_FS_Read( &i, sizeof( i ), f );
// get fog string
if ( i ) {
trap_FS_Read( infoString, i, f );
infoString[i] = 0;
} else {
Q_strncpyz( infoString, "none", sizeof( infoString ) );
}
// set the configstring so the 'savegame current' has good fog
if ( !Q_stricmp( infoString, "none" ) ) { // no fog
trap_Cvar_Set( "r_savegameGlobalFog", "none" );
} else {
// send it off to get set on the client
for ( p = &infoString[0],k = 0; *p; p++ ) {
if ( *p == ' ' ) {
k++;
}
if ( k == 6 ) { // the last parameter
infoString[p - infoString + 1] = '0';
infoString[p - infoString + 2] = 0;
break;
}
}
trap_Cvar_Set( "r_savegameGlobalFog", infoString );
}
}
// reset all AAS blocking entities
trap_AAS_SetAASBlockingEntity( vec3_origin, vec3_origin, -1 );
// read the entity structures
trap_FS_Read( &i, sizeof( i ), f );
size = i;
last = 0;
while ( 1 )
{
trap_FS_Read( &i, sizeof( i ), f );
ent = &g_entities[i];
// free any bot data if present
if ( i < 0 ) {
break;
}
if ( i >= MAX_GENTITIES ) {
trap_FS_FCloseFile( f );
G_Error( "G_LoadGame: entitynum out of range (%i, MAX = %i)\n", i, MAX_GENTITIES );
}
// if( i < MAX_CLIENTS && ent->inuse && ent->r.svFlags & SVF_BOT ) {
// BotAIShutdownClient(i);
// trap_DropClient( i, "" );
// }
if ( i >= level.num_entities ) { // notify server
level.num_entities = i + 1;
serverEntityUpdate = qtrue;
}
//ent = &g_entities[i];
ReadEntity( f, ent, size );
// free all entities that we skipped
for ( ; last < i; last++ ) {
if ( g_entities[last].inuse && i != ENTITYNUM_WORLD ) {
if ( last < MAX_CLIENTS ) {
// free any bot data if present
// if( g_entities[last].r.svFlags & SVF_BOT ) {
// BotAIShutdownClient(last);
// }
trap_DropClient( last, "", 0 );
} else {
G_FreeEntity( &g_entities[last] );
}
}
}
last = i + 1;
}
// clear all remaining entities
for ( ent = &g_entities[last] ; last < MAX_GENTITIES ; last++, ent++ ) {
memset( ent, 0, sizeof( *ent ) );
ent->classname = "freed";
ent->freetime = level.time;
ent->inuse = qfalse;
}
// read the client structures
trap_FS_Read( &i, sizeof( i ), f );
size = i;
while ( 1 )
{
trap_FS_Read( &i, sizeof( i ), f );
if ( i < 0 ) {
break;
}
if ( i > MAX_CLIENTS ) {
trap_FS_FCloseFile( f );
G_Error( "G_LoadGame: clientnum out of range\n" );
}
cl = &level.clients[i];
// load userinfo if bot
if ( g_entities[i].r.svFlags & SVF_BOT ) {
// get length
trap_FS_Read( &len, sizeof( len ), f );
// get fog string
if ( len ) {
trap_FS_Read( userinfo, len, f );
}
userinfo[len] = 0;
trap_SetUserinfo( i, userinfo );
}
/*if (cl->pers.connected == CON_DISCONNECTED) {
trap_FS_FCloseFile( f );
G_Error( "G_LoadGame: client mis-match in savegame" );
}*/
ReadClient( f, cl, size );
}
BotAI_SetNumBots( 0 );
// inform server of entity count if it has increased
if ( serverEntityUpdate ) {
// let the server system know that there are more entities
trap_LocateGameData( level.gentities, level.num_entities, sizeof( gentity_t ),
&level.clients[0].ps, sizeof( level.clients[0] ) );
}
// read the botwaypoint linked list
trap_FS_Read( &i, sizeof( i ), f );
size = i;
for ( i = 0; i < MAX_BOTAIWAYPOINTS; i++ ) {
wp = &botai_waypoints[i];
ReadBotWaypoint( f, wp, size );
}
trap_FS_Read( &i, sizeof( i ), f );
botai_freewaypoints = &botai_waypoints[i];
// read the botstate structures
trap_FS_Read( &i, sizeof( i ), f );
size = i;
while ( 1 )
{
trap_FS_Read( &i, sizeof( i ), f );
if ( i < 0 ) {
break;
}
if ( i > MAX_CLIENTS ) {
trap_FS_FCloseFile( f );
G_Error( "G_LoadGame: botstatenum out of range\n" );
}
bs = &botstates[i];
ReadBotState( f, bs, size );
}
// read music
/* trap_FS_Read(musicString, MAX_QPATH, f);
if(strlen(musicString)) {
trap_Cvar_Register( &musicCvar, "s_currentMusic", "", CVAR_ROM ); // get current music
if(Q_stricmp(musicString, musicCvar.string)) { // it's different than what's playing, so fade out and queue up
trap_SendServerCommand(-1, "mu_fade 0 1000\n");
trap_SetConfigstring( CS_MUSIC_QUEUE, musicString);
}
}*/
//----(SA) end moved
trap_FS_FCloseFile( f );
level.lastLoadTime = leveltime;
/*
// always save to the "current" savegame
last = level.time;
level.time = leveltime; // use the value we just for the save time
G_SaveGame(NULL);
// additionally update the last game that was loaded
trap_Cvar_VariableStringBuffer( "savegame_filename", mapname, sizeof(mapname) );
if (strlen( mapname ) > 0 && !strstr( mapname, "autosave" )) {
// clear it out so we dont lose it after a map_restart
trap_Cvar_Set( "savegame_filename", "" );
if (strstr(mapname, ".svg")) mapname[strstr(mapname, ".sav") - mapname] = '\0';
if (strstr(mapname, "/")) {
G_SaveGame( strstr(mapname, "/") + 1 );
} else {
G_SaveGame( mapname );
}
}
// restore the correct level.time
level.time = last;
*/
}
//=========================================================
/*
===============
PersWriteClient
===============
*/
void PersWriteClient( fileHandle_t f, gclient_t *cl ) {
persField_t *field;
// save the fields
for ( field = gclientPersFields ; field->len ; field++ )
{ // write the block
G_SaveWrite( ( void * )( (byte *)cl + field->ofs ), field->len, f );
}
}
/*
===============
PersReadClient
===============
*/
void PersReadClient( fileHandle_t f, gclient_t *cl ) {
persField_t *field;
// read the fields
for ( field = gclientPersFields ; field->len ; field++ )
{ // read the block
trap_FS_Read( ( void * )( (byte *)cl + field->ofs ), field->len, f );
}
}
//=========================================================
/*
===============
PersWriteEntity
===============
*/
void PersWriteEntity( fileHandle_t f, gentity_t *ent ) {
persField_t *field;
// save the fields
for ( field = gentityPersFields ; field->len ; field++ )
{ // write the block
G_SaveWrite( ( void * )( (byte *)ent + field->ofs ), field->len, f );
}
}
/*
===============
PersReadEntity
===============
*/
void PersReadEntity( fileHandle_t f, gentity_t *cl ) {
persField_t *field;
// read the fields
for ( field = gentityPersFields ; field->len ; field++ )
{ // read the block
trap_FS_Read( ( void * )( (byte *)cl + field->ofs ), field->len, f );
}
}
//=========================================================
/*
===============
G_SavePersistant
returns qtrue if successful
NOTE: only saves the local player's data, doesn't support AI characters
TODO: have G_SaveWrite return the number of byte's written, so if it doesn't
succeed, we can abort the save, and not save the file. This means we should
save to a temporary name, then copy it across to the real name after success,
so full disks don't result in lost saved games.
===============
*/
qboolean G_SavePersistant( char *nextmap ) {
char filename[MAX_QPATH];
fileHandle_t f;
int persid;
char cl_profileStr[256];
saveByteCount = 0;
trap_Cvar_VariableStringBuffer( "cl_profile", cl_profileStr, 256 );
// open the file
Com_sprintf( filename, MAX_QPATH, "profiles/%s/save/temp.psw", cl_profileStr );
if ( trap_FS_FOpenFile( filename, &f, FS_WRITE ) < 0 ) {
G_Error( "G_SavePersistant: cannot open '%s' for saving\n", filename );
}
// write the mapname
G_SaveWrite( nextmap, MAX_QPATH, f );
// save out the pers id
persid = trap_Milliseconds() + ( rand() & 0xffff );
G_SaveWrite( &persid, sizeof( persid ), f );
trap_Cvar_Set( "persid", va( "%i", persid ) );
// write out the entity structure
PersWriteEntity( f, &g_entities[0] );
// write out the client structure
PersWriteClient( f, &level.clients[0] );
trap_FS_FCloseFile( f );
// now check that it is the correct size
Com_sprintf( filename, MAX_QPATH, "profiles/%s/save/temp.psw", cl_profileStr );
if ( trap_FS_FOpenFile( filename, &f, FS_READ ) < saveByteCount ) {
trap_FS_FCloseFile( f );
G_SaveWriteError();
return qfalse;
}
trap_FS_FCloseFile( f );
// rename it to the real file
trap_FS_Rename( va( "profiles/%s/save/temp.psw", cl_profileStr ), va( "profiles/%s/save/current.psw", cl_profileStr ) );
// now check that it is the correct size
Com_sprintf( filename, MAX_QPATH, "profiles/%s/save/current.psw", cl_profileStr );
if ( trap_FS_FOpenFile( filename, &f, FS_READ ) < saveByteCount ) {
trap_FS_FCloseFile( f );
G_SaveWriteError();
return qfalse;
}
trap_FS_FCloseFile( f );
return qtrue;
}
/*
===============
G_LoadPersistant
===============
*/
void G_LoadPersistant( void ) {
fileHandle_t f;
char filename[MAX_QPATH];
char mapstr[MAX_QPATH];
vmCvar_t cvar_mapname;
int persid;
char cl_profileStr[256];
trap_Cvar_VariableStringBuffer( "cl_profile", cl_profileStr, 256 );
Com_sprintf( filename, MAX_QPATH, "profiles/%s/save/current.psw", cl_profileStr );
// open the file
if ( trap_FS_FOpenFile( filename, &f, FS_READ ) < 0 ) {
// not here, we shall assume they didn't want one
return;
}
// read the mapname, if it's not the same, then ignore the file
trap_FS_Read( mapstr, MAX_QPATH, f );
trap_Cvar_Register( &cvar_mapname, "mapname", "", CVAR_SERVERINFO | CVAR_ROM );
if ( Q_stricmp( cvar_mapname.string, mapstr ) ) {
trap_FS_FCloseFile( f );
return;
}
// check the pers id
trap_FS_Read( &persid, sizeof( persid ), f );
if ( persid != trap_Cvar_VariableIntegerValue( "persid" ) ) {
trap_FS_FCloseFile( f );
return;
}
// read the entity structure
PersReadEntity( f, &g_entities[0] );
// read the client structure
PersReadClient( f, &level.clients[0] );
trap_FS_FCloseFile( f );
// clear out the persid, since the persistent data has been read
trap_Cvar_Set( "persid", "0" );
}
#endif // SAVEGAME_SUPPORT