929 lines
23 KiB
C++
929 lines
23 KiB
C++
|
// Filename:- g_savegame.cpp
|
||
|
//
|
||
|
|
||
|
#include "g_local.h"
|
||
|
#include "fields.h"
|
||
|
#include "objectives.h"
|
||
|
#include "../cgame/cg_camera.h"
|
||
|
|
||
|
|
||
|
extern void OBJ_LoadTacticalInfo(void);
|
||
|
|
||
|
extern int Q3_VariableSave( void );
|
||
|
extern int Q3_VariableLoad( void );
|
||
|
|
||
|
extern void G_LoadSave_WriteMiscData(void);
|
||
|
extern void G_LoadSave_ReadMiscData(void);
|
||
|
|
||
|
static qboolean gbSaveGameWasOldFormat;
|
||
|
|
||
|
field_t savefields_gEntity[] =
|
||
|
{
|
||
|
{strFOFS(client), F_GCLIENT},
|
||
|
{strFOFS(owner), F_GENTITY},
|
||
|
{strFOFS(classname), F_STRING},
|
||
|
{strFOFS(model), F_STRING},
|
||
|
{strFOFS(model2), F_STRING},
|
||
|
// {strFOFS(model3), F_STRING}, - MCG
|
||
|
{strFOFS(nextTrain), F_GENTITY},
|
||
|
{strFOFS(prevTrain), F_GENTITY},
|
||
|
{strFOFS(message), F_STRING},
|
||
|
{strFOFS(target), F_STRING},
|
||
|
{strFOFS(target2), F_STRING},
|
||
|
{strFOFS(target3), F_STRING},
|
||
|
{strFOFS(target4), F_STRING},
|
||
|
{strFOFS(targetname), F_STRING},
|
||
|
{strFOFS(team), F_STRING},
|
||
|
{strFOFS(roff), F_STRING},
|
||
|
// {strFOFS(target_ent), F_GENTITY}, - MCG
|
||
|
{strFOFS(chain), F_GENTITY},
|
||
|
{strFOFS(enemy), F_GENTITY},
|
||
|
{strFOFS(activator), F_GENTITY},
|
||
|
{strFOFS(teamchain), F_GENTITY},
|
||
|
{strFOFS(teammaster), F_GENTITY},
|
||
|
{strFOFS(item), F_ITEM},
|
||
|
{strFOFS(NPC_type), F_STRING},
|
||
|
{strFOFS(closetarget), F_STRING},
|
||
|
{strFOFS(opentarget), F_STRING},
|
||
|
{strFOFS(paintarget), F_STRING},
|
||
|
{strFOFS(NPC_targetname), F_STRING},
|
||
|
{strFOFS(NPC_target), F_STRING},
|
||
|
{strFOFS(ownername), F_STRING},
|
||
|
{strFOFS(lastEnemy), F_GENTITY},
|
||
|
{strFOFS(behaviorSet), F_BEHAVIORSET},
|
||
|
{strFOFS(script_targetname),F_STRING},
|
||
|
{strFOFS(sequencer), F_NULL}, // CSequencer *sequencer;
|
||
|
{strFOFS(taskManager), F_NULL}, // CTaskManager *taskManager;
|
||
|
{strFOFS(NPC), F_BOOLPTR},
|
||
|
{strFOFS(soundSet), F_STRING},
|
||
|
{strFOFS(cameraGroup), F_STRING},
|
||
|
{strFOFS(infoString), F_STRING},
|
||
|
{strFOFS(parms), F_BOOLPTR},
|
||
|
{strFOFS(fullName), F_STRING},
|
||
|
// {strFOFS(timers), F_BOOLPTR}, // handled directly
|
||
|
|
||
|
{NULL, 0, F_IGNORE}
|
||
|
};
|
||
|
|
||
|
field_t savefields_gNPC[] =
|
||
|
{
|
||
|
// {strNPCOFS(pendingEnemy), F_GENTITY},
|
||
|
{strNPCOFS(touchedByPlayer), F_GENTITY},
|
||
|
{strNPCOFS(aimingBeam), F_GENTITY},
|
||
|
{strNPCOFS(eventOwner), F_GENTITY},
|
||
|
{strNPCOFS(coverTarg), F_GENTITY},
|
||
|
{strNPCOFS(tempGoal), F_GENTITY},
|
||
|
{strNPCOFS(goalEntity), F_GENTITY},
|
||
|
{strNPCOFS(lastGoalEntity), F_GENTITY},
|
||
|
{strNPCOFS(eventualGoal), F_GENTITY},
|
||
|
{strNPCOFS(hidingGoal), F_GENTITY},
|
||
|
{strNPCOFS(captureGoal), F_GENTITY},
|
||
|
{strNPCOFS(sayTarg), F_GENTITY},
|
||
|
{strNPCOFS(defendEnt), F_GENTITY},
|
||
|
{strNPCOFS(greetEnt), F_GENTITY},
|
||
|
|
||
|
{NULL, 0, F_IGNORE}
|
||
|
};
|
||
|
|
||
|
field_t savefields_LevelLocals[] =
|
||
|
{
|
||
|
{strLLOFS(locationHead), F_GENTITY},
|
||
|
// {strLLOFS(bodyQue), F_BODYQUEUE}, - MCG
|
||
|
{strLLOFS(alertEvents), F_ALERTEVENT},
|
||
|
|
||
|
{NULL, 0, F_IGNORE}
|
||
|
};
|
||
|
|
||
|
|
||
|
/*
|
||
|
struct gclient_s {
|
||
|
// ps MUST be the first element, because the server expects it
|
||
|
ok playerState_t ps; // communicated by server to clients
|
||
|
|
||
|
// private to game
|
||
|
ok clientPersistant_t pers;
|
||
|
ok clientSession_t sess;
|
||
|
|
||
|
ok usercmd_t usercmd; // most recent usercmd
|
||
|
|
||
|
//Client info - updated when ClientInfoChanged is called, instead of using configstrings
|
||
|
ok clientInfo_t clientInfo;
|
||
|
ok renderInfo_t renderInfo;
|
||
|
};
|
||
|
*/
|
||
|
// I'll keep a blank one for now in case I need to add anything...
|
||
|
//
|
||
|
field_t savefields_gClient[] =
|
||
|
{
|
||
|
{strCLOFS(squadname), F_STRING},
|
||
|
{strCLOFS(team_leader), F_GENTITY},
|
||
|
{strCLOFS(leader), F_GENTITY},
|
||
|
{strCLOFS(follower), F_GENTITY},
|
||
|
{strCLOFS(formationGoal), F_GENTITY},
|
||
|
{strCLOFS(clientInfo.customBasicSoundDir),F_STRING},
|
||
|
{strCLOFS(clientInfo.customCombatSoundDir),F_STRING},
|
||
|
{strCLOFS(clientInfo.customExtraSoundDir),F_STRING},
|
||
|
{strCLOFS(clientInfo.customScavSoundDir),F_STRING},
|
||
|
|
||
|
{NULL, 0, F_IGNORE}
|
||
|
};
|
||
|
|
||
|
|
||
|
list<string> strList;
|
||
|
|
||
|
|
||
|
/////////// char * /////////////
|
||
|
//
|
||
|
//
|
||
|
int GetStringNum(const char *psString)
|
||
|
{
|
||
|
assert( psString != (char *)0xcdcdcdcd );
|
||
|
|
||
|
// NULL ptrs I'll write out as a strlen of -1...
|
||
|
//
|
||
|
if (!psString)
|
||
|
{
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
strList.push_back( psString );
|
||
|
return strlen(psString) + 1; // this gives us the chunk length for the reader later
|
||
|
}
|
||
|
|
||
|
char *GetStringPtr(int iStrlen, char *psOriginal/*may be NULL*/)
|
||
|
{
|
||
|
if (iStrlen != -1)
|
||
|
{
|
||
|
static char sString[768]; // arb, inc if nec.
|
||
|
|
||
|
memset(sString,0, sizeof(sString));
|
||
|
|
||
|
assert(iStrlen+1<=sizeof(sString));
|
||
|
|
||
|
gi.ReadFromSaveGame('STRG', sString, iStrlen);
|
||
|
|
||
|
// now we should keep the original string if that's the same, else make a new G_Alloc ptr...
|
||
|
//
|
||
|
if (psOriginal && !strcmp(sString,psOriginal))
|
||
|
{
|
||
|
return psOriginal; // keep original ptr (which may be something horrible/immediate like [client->squadname = "string"])
|
||
|
}
|
||
|
#ifdef _DEBUG
|
||
|
// OutputDebugString(va("LOADSAVE INFO: String ptr mismatch, disk str = \"%s\", original = \"%s\", G_NewString() needed\n",sString,psOriginal?psOriginal:"NULL"));
|
||
|
#endif
|
||
|
|
||
|
// Note, I can't do this, because of things like ...ent->classname = "freed"... in G_FreeEntity(), so you don't want to
|
||
|
// try an overwrite stuff like that, even if you *can* fit (eg) "NPC" inside it...
|
||
|
//
|
||
|
// // hmmm, if the strings, don't match, can we at least the new string inside the old one?...
|
||
|
// //
|
||
|
// if (psOriginal && (strlen(sString)<=strlen(psOriginal)))
|
||
|
// {
|
||
|
// strcpy(psOriginal,sString);
|
||
|
// #ifdef _DEBUG
|
||
|
// OutputDebugString(va("... but I fitted disk string (strlen %d) into original (strlen %d)\n",strlen(sString),strlen(psOriginal)));
|
||
|
// #endif
|
||
|
// return psOriginal;
|
||
|
// }
|
||
|
return G_NewString(sString);
|
||
|
}
|
||
|
|
||
|
return NULL;
|
||
|
}
|
||
|
//
|
||
|
//
|
||
|
////////////////////////////////
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
/////////// gentity_t * ////////
|
||
|
//
|
||
|
//
|
||
|
int GetGEntityNum(gentity_t* ent)
|
||
|
{
|
||
|
assert( ent != (gentity_t *) 0xcdcdcdcd);
|
||
|
|
||
|
if (ent == NULL)
|
||
|
{
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
// note that I now validate the return value (to avoid triggering asserts on re-load) because of the
|
||
|
// way that the level_locals_t alertEvents struct contains a count of which ones are valid, so I'm guessing
|
||
|
// that some of them aren't (valid)...
|
||
|
//
|
||
|
int iReturnIndex = ent - g_entities;
|
||
|
|
||
|
if (iReturnIndex < 0 || iReturnIndex >= MAX_GENTITIES)
|
||
|
{
|
||
|
iReturnIndex = -1; // will get a NULL ptr on reload
|
||
|
}
|
||
|
return iReturnIndex;
|
||
|
}
|
||
|
|
||
|
gentity_t *GetGEntityPtr(int iEntNum)
|
||
|
{
|
||
|
if (iEntNum == -1)
|
||
|
{
|
||
|
return NULL;
|
||
|
}
|
||
|
assert(iEntNum >= 0);
|
||
|
assert(iEntNum < MAX_GENTITIES);
|
||
|
return (g_entities + iEntNum);
|
||
|
}
|
||
|
//
|
||
|
//
|
||
|
////////////////////////////////
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
/////////// gclient_t * ////////
|
||
|
//
|
||
|
//
|
||
|
int GetGClientNum(gclient_t *c)
|
||
|
{
|
||
|
assert(c != (gclient_t *)0xcdcdcdcd);
|
||
|
|
||
|
if (c == NULL)
|
||
|
{
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
return (c - level.clients);
|
||
|
}
|
||
|
|
||
|
gclient_t *GetGClientPtr(int c)
|
||
|
{
|
||
|
if (c == -1)
|
||
|
{
|
||
|
return NULL;
|
||
|
}
|
||
|
if (c == -2)
|
||
|
{
|
||
|
return (gclient_t *) -2; // preserve this value so that I know to load in one of Mike's private NPC clients later
|
||
|
}
|
||
|
|
||
|
assert(c >= 0);
|
||
|
assert(c < level.maxclients);
|
||
|
return (level.clients + c);
|
||
|
}
|
||
|
//
|
||
|
//
|
||
|
////////////////////////////////
|
||
|
|
||
|
|
||
|
/////////// gitem_t * //////////
|
||
|
//
|
||
|
//
|
||
|
int GetGItemNum (gitem_t *pItem)
|
||
|
{
|
||
|
assert(pItem != (gitem_t*) 0xcdcdcdcd);
|
||
|
|
||
|
if (pItem == NULL)
|
||
|
{
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
return pItem - bg_itemlist;
|
||
|
}
|
||
|
|
||
|
gitem_t *GetGItemPtr(int iItem)
|
||
|
{
|
||
|
if (iItem == -1)
|
||
|
{
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
assert(iItem >= 0);
|
||
|
assert(iItem < bg_numItems);
|
||
|
return &bg_itemlist[iItem];
|
||
|
}
|
||
|
//
|
||
|
//
|
||
|
////////////////////////////////
|
||
|
|
||
|
|
||
|
void EnumerateField(field_t *pField, byte *pbBase)
|
||
|
{
|
||
|
void *pv = (void *)(pbBase + pField->iOffset);
|
||
|
|
||
|
switch (pField->eFieldType)
|
||
|
{
|
||
|
case F_STRING:
|
||
|
*(int *)pv = GetStringNum(*(char **)pv);
|
||
|
break;
|
||
|
|
||
|
case F_GENTITY:
|
||
|
*(int *)pv = GetGEntityNum(*(gentity_t **)pv);
|
||
|
break;
|
||
|
|
||
|
case F_GCLIENT:
|
||
|
{
|
||
|
// unfortunately, I now need to see if this is a 'real' client (and therefore resolve to an enum), or
|
||
|
// whether it's one of Mike G's private clients that needs saving here (thanks Mike...)
|
||
|
//
|
||
|
gentity_t *ent = (gentity_t *) pbBase;
|
||
|
|
||
|
if (ent->NPC == NULL)
|
||
|
{
|
||
|
// regular client...
|
||
|
//
|
||
|
*(int *)pv = GetGClientNum(*(gclient_t **)pv);
|
||
|
break;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// this must be one of Mike's, so mark it as special...
|
||
|
//
|
||
|
*(int *)pv = -2; // yeuch, but distinguishes it from a valid 0 index, or -1 for client==NULL
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case F_ITEM:
|
||
|
*(int *)pv = GetGItemNum(*(gitem_t **)pv);
|
||
|
break;
|
||
|
|
||
|
case F_BEHAVIORSET:
|
||
|
{
|
||
|
const char **p = (const char **) pv;
|
||
|
for (int i=0; i<NUM_BSETS; i++)
|
||
|
{
|
||
|
pv = &p[i]; // since you can't ++ a void ptr
|
||
|
*(int *)pv = GetStringNum(*(char **)pv);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
/*MCG
|
||
|
case F_BODYQUEUE:
|
||
|
{
|
||
|
gentity_t **p = (gentity_t **) pv;
|
||
|
for (int i=0; i<BODY_QUEUE_SIZE; i++)
|
||
|
{
|
||
|
pv = &p[i]; // since you can't ++ a void ptr
|
||
|
*(int *)pv = GetGEntityNum(*(gentity_t **)pv);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
*/
|
||
|
|
||
|
case F_ALERTEVENT: // convert all gentity_t ptrs in an alertEvent array into indexes...
|
||
|
{
|
||
|
alertEvent_t* p = (alertEvent_t *) pv;
|
||
|
|
||
|
for (int i=0; i<MAX_ALERT_EVENTS; i++)
|
||
|
{
|
||
|
p[i].owner = (gentity_t *) GetGEntityNum(p[i].owner);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case F_BOOLPTR:
|
||
|
*(qboolean *)pv = !!(*(int *)pv);
|
||
|
break;
|
||
|
|
||
|
// These are pointers that are always recreated
|
||
|
case F_NULL:
|
||
|
*(void **)pv = NULL;
|
||
|
break;
|
||
|
|
||
|
case F_IGNORE:
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
G_Error ("EnumerateField: unknown field type");
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void EnumerateFields(field_t *pFields, byte *pbData, unsigned long ulChid, int iLen)
|
||
|
{
|
||
|
assert(strList.empty());
|
||
|
|
||
|
// enumerate all the fields...
|
||
|
//
|
||
|
if (pFields)
|
||
|
{
|
||
|
for (field_t *pField = pFields; pField->psName; pField++)
|
||
|
{
|
||
|
assert(pField->iOffset < iLen);
|
||
|
EnumerateField(pField, pbData);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// save out raw data...
|
||
|
//
|
||
|
gi.AppendToSaveGame(ulChid, pbData, iLen);
|
||
|
|
||
|
// save out any associated strings..
|
||
|
//
|
||
|
list<string>::iterator it = strList.begin();
|
||
|
for (unsigned int i=0; i<strList.size(); i++, ++it)
|
||
|
{
|
||
|
gi.AppendToSaveGame('STRG', (void *)(*it).c_str(), (*it).length() + 1);
|
||
|
}
|
||
|
|
||
|
strList.clear(); // make sure everything is cleaned up nicely
|
||
|
}
|
||
|
|
||
|
|
||
|
void EvaluateField(field_t *pField, byte *pbBase, byte *pbOriginalRefData/* may be NULL*/)
|
||
|
{
|
||
|
void *pv = (void *)(pbBase + pField->iOffset);
|
||
|
void *pvOriginal = (void *)(pbOriginalRefData + pField->iOffset);
|
||
|
|
||
|
switch (pField->eFieldType)
|
||
|
{
|
||
|
case F_STRING:
|
||
|
*(char **)pv = GetStringPtr(*(int *)pv, pbOriginalRefData?*(char**)pvOriginal:NULL);
|
||
|
break;
|
||
|
|
||
|
case F_GENTITY:
|
||
|
*(gentity_t **)pv = GetGEntityPtr(*(int *)pv);
|
||
|
break;
|
||
|
|
||
|
case F_GCLIENT:
|
||
|
*(gclient_t **)pv = GetGClientPtr(*(int *)pv);
|
||
|
break;
|
||
|
|
||
|
case F_ITEM:
|
||
|
*(gitem_t **)pv = GetGItemPtr(*(int *)pv);
|
||
|
break;
|
||
|
|
||
|
case F_BEHAVIORSET:
|
||
|
{
|
||
|
char **p = (char **) pv;
|
||
|
char **pO= (char **) pvOriginal;
|
||
|
for (int i=0; i<NUM_BSETS; i++, p++, pO++)
|
||
|
{
|
||
|
*p = GetStringPtr(*(int *)p, pbOriginalRefData?*(char **)pO:NULL);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
/*MCG
|
||
|
case F_BODYQUEUE:
|
||
|
{
|
||
|
gentity_t **p = (gentity_t **) pv;
|
||
|
for (int i=0; i<BODY_QUEUE_SIZE; i++, p++)
|
||
|
{
|
||
|
*p = GetGEntityPtr(*(int *)p);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
*/
|
||
|
|
||
|
case F_ALERTEVENT:
|
||
|
{
|
||
|
alertEvent_t* p = (alertEvent_t *) pv;
|
||
|
|
||
|
for (int i=0; i<MAX_ALERT_EVENTS; i++)
|
||
|
{
|
||
|
p[i].owner = GetGEntityPtr((int)(p[i].owner));
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
// // These fields are patched in when their relevant owners are loaded
|
||
|
case F_BOOLPTR:
|
||
|
case F_NULL:
|
||
|
break;
|
||
|
|
||
|
case F_IGNORE:
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
G_Error ("EvaluateField: unknown field type");
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void EvaluateFields(field_t *pFields, byte *pbData, byte *pbOriginalRefData, unsigned long ulChid, int iSize, qboolean bOkToSizeMisMatch)
|
||
|
{
|
||
|
// if (gi.ReadFromSaveGameOptional('TEST', pbData, iSize))
|
||
|
// {
|
||
|
// // load some new stuff that wasn't in the original file
|
||
|
// }
|
||
|
int iReadSize = gi.ReadFromSaveGame(ulChid, pbData, bOkToSizeMisMatch?0:iSize);
|
||
|
|
||
|
if (iReadSize != iSize)
|
||
|
{
|
||
|
// at the moment we know that it's only ever the GCLI chunk here,
|
||
|
// if this changes then check the 'ulChid' header and act accordingly.
|
||
|
//
|
||
|
// (we know struct size has INCREASED, so this is ok/safe...
|
||
|
//
|
||
|
assert(ulChid == 'GCLI');
|
||
|
assert(iSize>iReadSize); // :-)
|
||
|
memset(&pbData[iReadSize], 0, iSize-iReadSize); // zero out new objectives that weren't in old-format save file
|
||
|
gbSaveGameWasOldFormat = qtrue;
|
||
|
}
|
||
|
|
||
|
if (pFields)
|
||
|
{
|
||
|
for (field_t *pField = pFields; pField->psName; pField++)
|
||
|
{
|
||
|
EvaluateField(pField, pbData, pbOriginalRefData);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
==============
|
||
|
WriteLevelLocals
|
||
|
|
||
|
All pointer variables (except function pointers) must be handled specially.
|
||
|
==============
|
||
|
*/
|
||
|
void WriteLevelLocals ()
|
||
|
{
|
||
|
level_locals_t temp = level; // copy out all data into a temp space
|
||
|
|
||
|
EnumerateFields(savefields_LevelLocals, (byte *)&temp, 'LVLC', LLOFS(LEVEL_LOCALS_T_SAVESTOP)); // sizeof(temp));
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
==============
|
||
|
ReadLevelLocals
|
||
|
|
||
|
All pointer variables (except function pointers) must be handled specially.
|
||
|
==============
|
||
|
*/
|
||
|
void ReadLevelLocals ()
|
||
|
{
|
||
|
// preserve client ptr either side of the load, because clients are already saved/loaded through Read/Writegame...
|
||
|
//
|
||
|
gclient_t *pClients = level.clients; // save clients
|
||
|
|
||
|
level_locals_t temp = level; // struct copy
|
||
|
EvaluateFields(savefields_LevelLocals, (byte *)&temp, (byte *)&level, 'LVLC', LLOFS(LEVEL_LOCALS_T_SAVESTOP),qfalse); // sizeof(level_locals_t));
|
||
|
level = temp; // struct copy
|
||
|
|
||
|
level.clients = pClients; // restore clients
|
||
|
}
|
||
|
|
||
|
void WriteGEntities(qboolean qbAutosave)
|
||
|
{
|
||
|
int iCount = 0;
|
||
|
|
||
|
for (int i=0; i<(qbAutosave?1:globals.num_entities); i++)
|
||
|
{
|
||
|
gentity_t* ent = &g_entities[i];
|
||
|
|
||
|
if ( ent->inuse )
|
||
|
{
|
||
|
iCount++;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
gi.AppendToSaveGame('NMED', &iCount, sizeof(iCount));
|
||
|
|
||
|
for (i=0; i<(qbAutosave?1:globals.num_entities); i++)
|
||
|
{
|
||
|
gentity_t* ent = &g_entities[i];
|
||
|
|
||
|
if ( ent->inuse)
|
||
|
{
|
||
|
gi.AppendToSaveGame('EDNM', (void *)&i, sizeof(i));
|
||
|
|
||
|
qboolean qbLinked = ent->linked;
|
||
|
gi.unlinkentity( ent );
|
||
|
gentity_t tempEnt = *ent; // make local copy
|
||
|
tempEnt.linked = qbLinked;
|
||
|
|
||
|
if (qbLinked)
|
||
|
{
|
||
|
gi.linkentity( ent );
|
||
|
}
|
||
|
|
||
|
EnumerateFields(savefields_gEntity, (byte *)&tempEnt, 'GENT', sizeof(tempEnt));
|
||
|
|
||
|
// now for any fiddly bits that would be rather awkward to build into the enumerator...
|
||
|
//
|
||
|
if (tempEnt.NPC)
|
||
|
{
|
||
|
gNPC_t npc = *ent->NPC; // NOT *tempEnt.NPC; !! :-)
|
||
|
|
||
|
EnumerateFields(savefields_gNPC, (byte *)&npc, 'GNPC', sizeof(npc));
|
||
|
}
|
||
|
|
||
|
if (tempEnt.client == (gclient_t *)-2) // I know, I know...
|
||
|
{
|
||
|
gclient_t client = *ent->client; // NOT *tempEnt.client!!
|
||
|
EnumerateFields(savefields_gClient, (byte *)&client, 'GCLI', sizeof(client));
|
||
|
}
|
||
|
|
||
|
if (tempEnt.parms)
|
||
|
{
|
||
|
gi.AppendToSaveGame('PARM', ent->parms, sizeof(*ent->parms));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//Write out all entity timers
|
||
|
TIMER_Save();//WriteEntityTimers();
|
||
|
|
||
|
if (!qbAutosave)
|
||
|
{
|
||
|
//Save out ICARUS information
|
||
|
iICARUS->Save();
|
||
|
|
||
|
// this marker needs to be here, it lets me know if Icarus doesn't load everything back later,
|
||
|
// which has happened, and doesn't always show up onscreen until certain game situations.
|
||
|
// This saves time debugging, and makes things easier to track.
|
||
|
//
|
||
|
static int iBlah = 1234;
|
||
|
gi.AppendToSaveGame('ICOK', &iBlah, sizeof(iBlah));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void ReadGEntities(qboolean qbAutosave)
|
||
|
{
|
||
|
int iCount;
|
||
|
|
||
|
gi.ReadFromSaveGame('NMED', (void *)&iCount, sizeof(iCount));
|
||
|
|
||
|
int iPreviousEntRead = -1;
|
||
|
for (int i=0; i<iCount; i++)
|
||
|
{
|
||
|
int iEntIndex;
|
||
|
gi.ReadFromSaveGame('EDNM', (void *)&iEntIndex, sizeof(iEntIndex));
|
||
|
|
||
|
if (iEntIndex >= globals.num_entities)
|
||
|
{
|
||
|
globals.num_entities = iEntIndex + 1;
|
||
|
}
|
||
|
|
||
|
if (iPreviousEntRead != iEntIndex-1)
|
||
|
{
|
||
|
for (int j=iPreviousEntRead+1; j!=iEntIndex; j++)
|
||
|
{
|
||
|
if ( g_entities[j].inuse ) // not actually necessary
|
||
|
{
|
||
|
G_FreeEntity(&g_entities[j]);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
iPreviousEntRead = iEntIndex;
|
||
|
|
||
|
// slightly naff syntax here, but makes a few ops clearer later...
|
||
|
//
|
||
|
gentity_t entity;
|
||
|
gentity_t* pEntOriginal = &g_entities[iEntIndex];
|
||
|
gentity_t* pEnt = &entity;
|
||
|
|
||
|
EvaluateFields(savefields_gEntity, (byte *)pEnt, (byte *)pEntOriginal, 'GENT', sizeof(*pEnt),qfalse);
|
||
|
|
||
|
// now for any fiddly bits...
|
||
|
//
|
||
|
if (pEnt->NPC) // will be qtrue/qfalse
|
||
|
{
|
||
|
gNPC_t tempNPC;
|
||
|
|
||
|
EvaluateFields(savefields_gNPC, (byte *)&tempNPC,(byte *)pEntOriginal->NPC, 'GNPC', sizeof (*pEnt->NPC),qfalse);
|
||
|
|
||
|
// so can we pinch the original's one or do we have to malloc a new one?...
|
||
|
//
|
||
|
if (pEntOriginal->NPC)
|
||
|
{
|
||
|
// pinch this G_Alloc handle...
|
||
|
//
|
||
|
pEnt->NPC = pEntOriginal->NPC;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// original didn't have one (hmmm...), so make a new one...
|
||
|
//
|
||
|
//assert(0); // I want to know about this, though not in release
|
||
|
pEnt->NPC = (gNPC_t *) G_Alloc(sizeof(*pEnt->NPC));
|
||
|
}
|
||
|
|
||
|
// copy over the one we've just loaded...
|
||
|
//
|
||
|
*pEnt->NPC = tempNPC; // struct copy
|
||
|
|
||
|
}
|
||
|
|
||
|
if (pEnt->client == (gclient_t*) -2) // one of Mike G's NPC clients?
|
||
|
{
|
||
|
gclient_t tempGClient;
|
||
|
|
||
|
EvaluateFields(savefields_gClient, (byte *)&tempGClient, (byte *)pEntOriginal->client, 'GCLI', sizeof(*pEnt->client),qtrue);
|
||
|
|
||
|
// can we pinch the original's client handle or do we have to malloc a new one?...
|
||
|
//
|
||
|
if (pEntOriginal->client)
|
||
|
{
|
||
|
// pinch this G_Alloc handle...
|
||
|
//
|
||
|
pEnt->client = pEntOriginal->client;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// original didn't have one (hmmm...) so make a new one...
|
||
|
//
|
||
|
pEnt->client = (gclient_t *) G_Alloc(sizeof(*pEnt->client));
|
||
|
}
|
||
|
|
||
|
// copy over the one we've just loaded....
|
||
|
//
|
||
|
*pEnt->client = tempGClient; // struct copy
|
||
|
|
||
|
// Code so that original EF save games have the correct weapons
|
||
|
// (weapon enum was changed - WP_PROTON_GUN was added into the middle of it)
|
||
|
if (gbSaveGameWasOldFormat==qtrue)
|
||
|
{
|
||
|
if (pEnt->client->ps.weapon >= WP_PROTON_GUN)
|
||
|
{
|
||
|
pEnt->client->ps.weapon++;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Some Icarus thing... (probably)
|
||
|
//
|
||
|
if (pEnt->parms) // will be qtrue/qfalse
|
||
|
{
|
||
|
parms_t tempParms;
|
||
|
|
||
|
gi.ReadFromSaveGame('PARM', &tempParms, sizeof(tempParms));
|
||
|
|
||
|
// so can we pinch the original's one or do we have to malloc a new one?...
|
||
|
//
|
||
|
if (pEntOriginal->parms)
|
||
|
{
|
||
|
// pinch this G_Alloc handle...
|
||
|
//
|
||
|
pEnt->parms = pEntOriginal->parms;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// original didn't have one, so make a new one...
|
||
|
//
|
||
|
pEnt->parms = (parms_t *) G_Alloc(sizeof(*pEnt->parms));
|
||
|
}
|
||
|
|
||
|
// copy over the one we've just loaded...
|
||
|
//
|
||
|
*pEnt->parms = tempParms; // struct copy
|
||
|
}
|
||
|
|
||
|
gi.unlinkentity (pEntOriginal);
|
||
|
|
||
|
ICARUS_FreeEnt( pEntOriginal );
|
||
|
*pEntOriginal = *pEnt; // struct copy
|
||
|
|
||
|
///evil hack because the sytem stores sfx_t handles directly instead of the set, we have to reget the set's sfx_t
|
||
|
if (pEntOriginal->s.eType == ET_MOVER && pEntOriginal->s.loopSound>0)
|
||
|
{
|
||
|
extern int BMS_MID; //from g_mover
|
||
|
pEntOriginal->s.loopSound = CAS_GetBModelSound( pEntOriginal->soundSet, BMS_MID );
|
||
|
if (pEntOriginal->s.loopSound == -1)
|
||
|
{
|
||
|
pEntOriginal->s.loopSound = 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
qboolean qbLinked = pEntOriginal->linked;
|
||
|
pEntOriginal->linked = qfalse;
|
||
|
|
||
|
if (qbLinked)
|
||
|
{
|
||
|
gi.linkentity (pEntOriginal);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//Read in all the entity timers
|
||
|
TIMER_Load();//ReadEntityTimers();
|
||
|
|
||
|
if (!qbAutosave)
|
||
|
{
|
||
|
// now zap any g_ents that were inuse when the level was loaded, but are no longer in use in the saved version
|
||
|
// that we've just loaded...
|
||
|
//
|
||
|
for (i=iPreviousEntRead+1; i<globals.num_entities; i++)
|
||
|
{
|
||
|
if ( g_entities[i].inuse ) // not actually necessary
|
||
|
{
|
||
|
G_FreeEntity(&g_entities[i]);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//Load ICARUS information
|
||
|
ICARUS_EntList.clear();
|
||
|
iICARUS->Load();
|
||
|
|
||
|
// check that Icarus has loaded everything it saved out by having a marker chunk after it...
|
||
|
//
|
||
|
static int iBlah = 1234;
|
||
|
gi.ReadFromSaveGame('ICOK', &iBlah, sizeof(iBlah));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void WriteLevel(qboolean qbAutosave)
|
||
|
{
|
||
|
if (!qbAutosave) //-always save the client
|
||
|
{
|
||
|
// write out one client - us!
|
||
|
//
|
||
|
assert(level.maxclients == 1); // I'll need to know if this changes, otherwise I'll need to change the way ReadGame works
|
||
|
gclient_t client = level.clients[0];
|
||
|
EnumerateFields(savefields_gClient, (byte *)&client, 'GCLI', sizeof(client));
|
||
|
}
|
||
|
|
||
|
OBJ_SaveObjectiveData();
|
||
|
|
||
|
/////////////
|
||
|
|
||
|
WriteLevelLocals(); // level_locals_t level
|
||
|
WriteGEntities(qbAutosave);
|
||
|
Q3_VariableSave();
|
||
|
G_LoadSave_WriteMiscData();
|
||
|
|
||
|
|
||
|
|
||
|
// (Do NOT put any write-code below this line)
|
||
|
//
|
||
|
// put out an end-marker so that the load code can check everything was read in...
|
||
|
//
|
||
|
static int iDONE = 1234;
|
||
|
gi.AppendToSaveGame('DONE', &iDONE, sizeof(iDONE));
|
||
|
}
|
||
|
|
||
|
void ReadLevel(qboolean qbAutosave, qboolean qbLoadTransition)
|
||
|
{
|
||
|
gbSaveGameWasOldFormat = qfalse;
|
||
|
|
||
|
if ( qbLoadTransition )
|
||
|
{
|
||
|
//loadtransitions do not need to read the objectives and client data from the level they're going to
|
||
|
//In a loadtransition, client data is carried over on the server and will be stomped later anyway.
|
||
|
//The objective info (in client->sess data), however, is read in from G_ReadSessionData which is called before this func,
|
||
|
//we do NOT want to stomp that session data when doing a load transition
|
||
|
|
||
|
//However, we should still save this info out because these savegames may need to be
|
||
|
//loaded normally later- perhaps if you die and need to respawn, perhaps as some kind
|
||
|
//of emergency savegame for resuming, etc.
|
||
|
|
||
|
//SO: We read it in, but throw it away.
|
||
|
|
||
|
//Read & throw away gclient info
|
||
|
gclient_t junkClient;
|
||
|
EvaluateFields(savefields_gClient, (byte *)&junkClient, (byte *)&level.clients[0], 'GCLI', sizeof(*level.clients),qtrue);
|
||
|
|
||
|
//Read & throw away objective info
|
||
|
objectives_t junkObj[MAX_MISSION_OBJ];
|
||
|
gi.ReadFromSaveGame('OBJT', (void *) &junkObj, 0);
|
||
|
|
||
|
//We still want to load in the tactical_info, though.
|
||
|
OBJ_LoadTacticalInfo();
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if (!qbAutosave )//always load the client unless it's an autosave
|
||
|
{
|
||
|
assert(level.maxclients == 1); // I'll need to know if this changes, otherwise I'll need to change the way things work
|
||
|
|
||
|
gclient_t GClient;
|
||
|
EvaluateFields(savefields_gClient, (byte *)&GClient, (byte *)&level.clients[0], 'GCLI', sizeof(*level.clients),qtrue);
|
||
|
level.clients[0] = GClient; // struct copy
|
||
|
}
|
||
|
|
||
|
OBJ_LoadObjectiveData();//loads mission objectives AND tactical info
|
||
|
}
|
||
|
|
||
|
/////////////
|
||
|
|
||
|
ReadLevelLocals(); // level_locals_t level
|
||
|
ReadGEntities(qbAutosave);
|
||
|
Q3_VariableLoad();
|
||
|
G_LoadSave_ReadMiscData();
|
||
|
|
||
|
|
||
|
|
||
|
// (Do NOT put any read-code below this line)
|
||
|
//
|
||
|
// check that the whole file content was loaded by specifically requesting an end-marker...
|
||
|
//
|
||
|
static int iDONE = 1234;
|
||
|
gi.ReadFromSaveGame('DONE', &iDONE, sizeof(iDONE));
|
||
|
|
||
|
|
||
|
// now do something here based on 'gbSaveGameWasOldFormat' if you want...
|
||
|
//
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
qboolean GameAllowedToSaveHere(void)
|
||
|
{
|
||
|
return !in_camera;
|
||
|
}
|
||
|
|
||
|
//////////////////// eof /////////////////////
|
||
|
|