ref: 21bc4c4ab90df0455eaeba6a4c959005493f0a76
parent: 7f27444e2e93702fbd23dedc0fb7e7b4f3e13e50
author: Simon Howard <fraggle@gmail.com>
date: Sun Aug 25 01:44:45 EDT 2013
First step towards portable Hexen savegames. Perform endianness conversions for ACS vars and add functions to read/write player_t structures. Subversion-branch: /branches/v2-branch Subversion-revision: 2625
--- a/src/hexen/sv_save.c
+++ b/src/hexen/sv_save.c
@@ -37,8 +37,8 @@
#define MOBJ_NULL -1
#define MOBJ_XX_PLAYER -2
#define GET_BYTE (*SavePtr.b++)
-#define GET_WORD (*SavePtr.w++)
-#define GET_LONG (*SavePtr.l++)
+#define GET_WORD SHORT(*SavePtr.w++)
+#define GET_LONG LONG(*SavePtr.l++)
#define MAX_MAPS 99
#define BASE_SLOT 6
#define REBORN_SLOT 7
@@ -142,6 +142,7 @@
static void StreamOutByte(byte val);
static void StreamOutWord(unsigned short val);
static void StreamOutLong(unsigned int val);
+static void StreamOutPtr(void *ptr);
// EXTERNAL DATA DECLARATIONS ----------------------------------------------
@@ -259,6 +260,527 @@
// CODE --------------------------------------------------------------------
+// Autogenerated functions for reading/writing structs:
+
+//
+// acsstore_t
+//
+
+static void StreamIn_acsstore_t(acsstore_t *str)
+{
+ int i;
+
+ // int map;
+ str->map = GET_LONG;
+
+ // int script;
+ str->script = GET_LONG;
+
+ // byte args[4];
+ for (i=0; i<4; ++i)
+ {
+ str->args[i] = GET_BYTE;
+ }
+}
+
+static void StreamOut_acsstore_t(acsstore_t *str)
+{
+ int i;
+
+ // int map;
+ StreamOutLong(str->map);
+
+ // int script;
+ StreamOutLong(str->script);
+
+ // byte args[4];
+ for (i=0; i<4; ++i)
+ {
+ StreamOutByte(str->args[i]);
+ }
+}
+
+
+//
+// ticcmd_t
+// (this is based on the Vanilla definition of the struct)
+//
+
+static void StreamIn_ticcmd_t(ticcmd_t *str)
+{
+ // char forwardmove;
+ str->forwardmove = GET_BYTE;
+
+ // char sidemove;
+ str->sidemove = GET_BYTE;
+
+ // short angleturn;
+ str->angleturn = GET_WORD;
+
+ // short consistancy;
+ str->consistancy = GET_WORD;
+
+ // byte chatchar;
+ str->chatchar = GET_BYTE;
+
+ // byte buttons;
+ str->buttons = GET_BYTE;
+
+ // byte lookfly;
+ str->lookfly = GET_BYTE;
+
+ // byte arti;
+ str->arti = GET_BYTE;
+}
+
+static void StreamOut_ticcmd_t(ticcmd_t *str)
+{
+ // char forwardmove;
+ StreamOutByte(str->forwardmove);
+
+ // char sidemove;
+ StreamOutByte(str->sidemove);
+
+ // short angleturn;
+ StreamOutWord(str->angleturn);
+
+ // short consistancy;
+ StreamOutWord(str->consistancy);
+
+ // byte chatchar;
+ StreamOutByte(str->chatchar);
+
+ // byte buttons;
+ StreamOutByte(str->buttons);
+
+ // byte lookfly;
+ StreamOutByte(str->lookfly);
+
+ // byte arti;
+ StreamOutByte(str->arti);
+}
+
+
+
+//
+// inventory_t
+//
+
+static void StreamIn_inventory_t(inventory_t *str)
+{
+ // int type;
+ str->type = GET_LONG;
+
+ // int count;
+ str->count = GET_LONG;
+}
+
+static void StreamOut_inventory_t(inventory_t *str)
+{
+ // int type;
+ StreamOutLong(str->type);
+
+ // int count;
+ StreamOutLong(str->count);
+}
+
+
+//
+// pspdef_t
+//
+
+static void StreamIn_pspdef_t(pspdef_t *str)
+{
+ int state_num;
+
+ // state_t *state;
+
+ // This is a pointer; it is stored as an index into the states table.
+
+ state_num = GET_LONG;
+
+ if (state_num != 0)
+ {
+ str->state = states + state_num;
+ }
+ else
+ {
+ str->state = NULL;
+ }
+
+ // int tics;
+ str->tics = GET_LONG;
+
+ // fixed_t sx, sy;
+ str->sx = GET_LONG;
+ str->sy = GET_LONG;
+}
+
+static void StreamOut_pspdef_t(pspdef_t *str)
+{
+ // state_t *state;
+ // This is a pointer; store the index in the states table,
+ // rather than the pointer itself.
+ if (str->state != NULL)
+ {
+ StreamOutLong(str->state - states);
+ }
+ else
+ {
+ StreamOutLong(0);
+ }
+
+ // int tics;
+ StreamOutLong(str->tics);
+
+ // fixed_t sx, sy;
+ StreamOutLong(str->sx);
+ StreamOutLong(str->sy);
+}
+
+
+//
+// player_t
+//
+
+static void StreamIn_player_t(player_t *str)
+{
+ int i;
+
+ // mobj_t *mo;
+ // Pointer value is reset on load.
+ GET_LONG;
+ str->mo = NULL;
+
+ // playerstate_t playerstate;
+ str->playerstate = GET_LONG;
+
+ // ticcmd_t cmd;
+ StreamIn_ticcmd_t(&str->cmd);
+
+ // pclass_t class;
+ str->class = GET_LONG;
+
+ // fixed_t viewz;
+ str->viewz = GET_LONG;
+
+ // fixed_t viewheight;
+ str->viewheight = GET_LONG;
+
+ // fixed_t deltaviewheight;
+ str->deltaviewheight = GET_LONG;
+
+ // fixed_t bob;
+ str->bob = GET_LONG;
+
+ // int flyheight;
+ str->flyheight = GET_LONG;
+
+ // int lookdir;
+ str->lookdir = GET_LONG;
+
+ // boolean centering;
+ str->centering = GET_LONG;
+
+ // int health;
+ str->health = GET_LONG;
+
+ // int armorpoints[NUMARMOR];
+ for (i=0; i<NUMARMOR; ++i)
+ {
+ str->armorpoints[i] = GET_LONG;
+ }
+
+ // inventory_t inventory[NUMINVENTORYSLOTS];
+ for (i=0; i<NUMINVENTORYSLOTS; ++i)
+ {
+ StreamIn_inventory_t(&str->inventory[i]);
+ }
+
+ // artitype_t readyArtifact;
+ str->readyArtifact = GET_LONG;
+
+ // int artifactCount;
+ str->artifactCount = GET_LONG;
+
+ // int inventorySlotNum;
+ str->inventorySlotNum = GET_LONG;
+
+ // int powers[NUMPOWERS];
+ for (i=0; i<NUMPOWERS; ++i)
+ {
+ str->powers[i] = GET_LONG;
+ }
+
+ // int keys;
+ str->keys = GET_LONG;
+
+ // int pieces;
+ str->pieces = GET_LONG;
+
+ // signed int frags[MAXPLAYERS];
+ for (i=0; i<MAXPLAYERS; ++i)
+ {
+ str->frags[i] = GET_LONG;
+ }
+
+ // weapontype_t readyweapon;
+ str->readyweapon = GET_LONG;
+
+ // weapontype_t pendingweapon;
+ str->pendingweapon = GET_LONG;
+
+ // boolean weaponowned[NUMWEAPONS];
+ for (i=0; i<NUMWEAPONS; ++i)
+ {
+ str->weaponowned[i] = GET_LONG;
+ }
+
+ // int mana[NUMMANA];
+ for (i=0; i<NUMMANA; ++i)
+ {
+ str->mana[i] = GET_LONG;
+ }
+
+ // int attackdown, usedown;
+ str->attackdown = GET_LONG;
+ str->usedown = GET_LONG;
+
+ // int cheats;
+ str->cheats = GET_LONG;
+
+ // int refire;
+ str->refire = GET_LONG;
+
+ // int killcount, itemcount, secretcount;
+ str->killcount = GET_LONG;
+ str->itemcount = GET_LONG;
+ str->secretcount = GET_LONG;
+
+ // char message[80];
+ for (i=0; i<80; ++i)
+ {
+ str->message[i] = GET_BYTE;
+ }
+
+ // int messageTics;
+ str->messageTics = GET_LONG;
+
+ // short ultimateMessage;
+ str->ultimateMessage = GET_WORD;
+
+ // short yellowMessage;
+ str->yellowMessage = GET_WORD;
+
+ // int damagecount, bonuscount;
+ str->damagecount = GET_LONG;
+ str->bonuscount = GET_LONG;
+
+ // int poisoncount;
+ str->poisoncount = GET_LONG;
+
+ // mobj_t *poisoner;
+ // Pointer value is reset.
+ GET_LONG;
+ str->poisoner = NULL;
+
+ // mobj_t *attacker;
+ // Pointer value is reset.
+ GET_LONG;
+ str->attacker = NULL;
+
+ // int extralight;
+ str->extralight = GET_LONG;
+
+ // int fixedcolormap;
+ str->fixedcolormap = GET_LONG;
+
+ // int colormap;
+ str->colormap = GET_LONG;
+
+ // pspdef_t psprites[NUMPSPRITES];
+ for (i=0; i<NUMPSPRITES; ++i)
+ {
+ StreamIn_pspdef_t(&str->psprites[i]);
+ }
+
+ // int morphTics;
+ str->morphTics = GET_LONG;
+
+ // unsigned int jumpTics;
+ str->jumpTics = GET_LONG;
+
+ // unsigned int worldTimer;
+ str->worldTimer = GET_LONG;
+}
+
+static void StreamOut_player_t(player_t *str)
+{
+ int i;
+
+ // mobj_t *mo;
+ StreamOutPtr(str->mo);
+
+ // playerstate_t playerstate;
+ StreamOutLong(str->playerstate);
+
+ // ticcmd_t cmd;
+ StreamOut_ticcmd_t(&str->cmd);
+
+ // pclass_t class;
+ StreamOutLong(str->class);
+
+ // fixed_t viewz;
+ StreamOutLong(str->viewz);
+
+ // fixed_t viewheight;
+ StreamOutLong(str->viewheight);
+
+ // fixed_t deltaviewheight;
+ StreamOutLong(str->deltaviewheight);
+
+ // fixed_t bob;
+ StreamOutLong(str->bob);
+
+ // int flyheight;
+ StreamOutLong(str->flyheight);
+
+ // int lookdir;
+ StreamOutLong(str->lookdir);
+
+ // boolean centering;
+ StreamOutLong(str->centering);
+
+ // int health;
+ StreamOutLong(str->health);
+
+ // int armorpoints[NUMARMOR];
+ for (i=0; i<NUMARMOR; ++i)
+ {
+ StreamOutLong(str->armorpoints[i]);
+ }
+
+ // inventory_t inventory[NUMINVENTORYSLOTS];
+ for (i=0; i<NUMINVENTORYSLOTS; ++i)
+ {
+ StreamOut_inventory_t(&str->inventory[i]);
+ }
+
+ // artitype_t readyArtifact;
+ StreamOutLong(str->readyArtifact);
+
+ // int artifactCount;
+ StreamOutLong(str->artifactCount);
+
+ // int inventorySlotNum;
+ StreamOutLong(str->inventorySlotNum);
+
+ // int powers[NUMPOWERS];
+ for (i=0; i<NUMPOWERS; ++i)
+ {
+ StreamOutLong(str->powers[i]);
+ }
+
+ // int keys;
+ StreamOutLong(str->keys);
+
+ // int pieces;
+ StreamOutLong(str->pieces);
+
+ // signed int frags[MAXPLAYERS];
+ for (i=0; i<MAXPLAYERS; ++i)
+ {
+ StreamOutLong(str->frags[i]);
+ }
+
+ // weapontype_t readyweapon;
+ StreamOutLong(str->readyweapon);
+
+ // weapontype_t pendingweapon;
+ StreamOutLong(str->pendingweapon);
+
+ // boolean weaponowned[NUMWEAPONS];
+ for (i=0; i<NUMWEAPONS; ++i)
+ {
+ StreamOutLong(str->weaponowned[i]);
+ }
+
+ // int mana[NUMMANA];
+ for (i=0; i<NUMMANA; ++i)
+ {
+ StreamOutLong(str->mana[i]);
+ }
+
+ // int attackdown, usedown;
+ StreamOutLong(str->attackdown);
+ StreamOutLong(str->usedown);
+
+ // int cheats;
+ StreamOutLong(str->cheats);
+
+ // int refire;
+ StreamOutLong(str->refire);
+
+ // int killcount, itemcount, secretcount;
+ StreamOutLong(str->killcount);
+ StreamOutLong(str->itemcount);
+ StreamOutLong(str->secretcount);
+
+ // char message[80];
+ for (i=0; i<80; ++i)
+ {
+ StreamOutByte(str->message[i]);
+ }
+
+ // int messageTics;
+ StreamOutLong(str->messageTics);
+
+ // short ultimateMessage;
+ StreamOutWord(str->ultimateMessage);
+
+ // short yellowMessage;
+ StreamOutWord(str->yellowMessage);
+
+ // int damagecount, bonuscount;
+ StreamOutLong(str->damagecount);
+ StreamOutLong(str->bonuscount);
+
+ // int poisoncount;
+ StreamOutLong(str->poisoncount);
+
+ // mobj_t *poisoner;
+ StreamOutPtr(str->poisoner);
+
+ // mobj_t *attacker;
+ StreamOutPtr(str->attacker);
+
+ // int extralight;
+ StreamOutLong(str->extralight);
+
+ // int fixedcolormap;
+ StreamOutLong(str->fixedcolormap);
+
+ // int colormap;
+ StreamOutLong(str->colormap);
+
+ // pspdef_t psprites[NUMPSPRITES];
+ for (i=0; i<NUMPSPRITES; ++i)
+ {
+ StreamOut_pspdef_t(&str->psprites[i]);
+ }
+
+ // int morphTics;
+ StreamOutLong(str->morphTics);
+
+ // unsigned int jumpTics;
+ StreamOutLong(str->jumpTics);
+
+ // unsigned int worldTimer;
+ StreamOutLong(str->worldTimer);
+}
+
+
+
//==========================================================================
//
// SV_SaveGame
@@ -269,6 +791,7 @@
{
char fileName[100];
char versionText[HXS_VERSION_TEXT_LENGTH];
+ unsigned int i;
// Open the output file
sprintf(fileName, "%shex6.hxs", SavePath);
@@ -290,9 +813,16 @@
StreamOutByte(gameskill);
// Write global script info
- StreamOutBuffer(WorldVars, sizeof(WorldVars));
- StreamOutBuffer(ACSStore, sizeof(ACSStore));
+ for (i = 0; i < MAX_ACS_WORLD_VARS; ++i)
+ {
+ StreamOutLong(WorldVars[i]);
+ }
+ for (i = 0; i < MAX_ACS_STORE + 1; ++i)
+ {
+ StreamOut_acsstore_t(&ACSStore[i]);
+ }
+
ArchivePlayers();
// Place a termination marker
@@ -394,11 +924,17 @@
gameskill = GET_BYTE;
// Read global script info
- memcpy(WorldVars, SavePtr.b, sizeof(WorldVars));
- SavePtr.b += sizeof(WorldVars);
- memcpy(ACSStore, SavePtr.b, sizeof(ACSStore));
- SavePtr.b += sizeof(ACSStore);
+ for (i = 0; i < MAX_ACS_WORLD_VARS; ++i)
+ {
+ WorldVars[i] = GET_LONG;
+ }
+
+ for (i = 0; i < MAX_ACS_STORE + 1; ++i)
+ {
+ StreamIn_acsstore_t(&ACSStore[i]);
+ }
+
// Read the player structures
UnarchivePlayers();
@@ -727,8 +1263,6 @@
static void ArchivePlayers(void)
{
int i;
- int j;
- player_t tempPlayer;
StreamOutLong(ASEG_PLAYERS);
for (i = 0; i < MAXPLAYERS; i++)
@@ -742,16 +1276,7 @@
continue;
}
StreamOutByte(PlayerClass[i]);
- tempPlayer = players[i];
- for (j = 0; j < NUMPSPRITES; j++)
- {
- if (tempPlayer.psprites[j].state)
- {
- tempPlayer.psprites[j].state =
- (state_t *) (tempPlayer.psprites[j].state - states);
- }
- }
- StreamOutBuffer(&tempPlayer, sizeof(player_t));
+ StreamOut_player_t(&players[i]);
}
}
@@ -763,7 +1288,7 @@
static void UnarchivePlayers(void)
{
- int i, j;
+ int i;
AssertSegment(ASEG_PLAYERS);
for (i = 0; i < MAXPLAYERS; i++)
@@ -777,20 +1302,8 @@
continue;
}
PlayerClass[i] = GET_BYTE;
- memcpy(&players[i], SavePtr.b, sizeof(player_t));
- SavePtr.b += sizeof(player_t);
- players[i].mo = NULL; // Will be set when unarc thinker
+ StreamIn_player_t(&players[i]);
P_ClearMessage(&players[i]);
- players[i].attacker = NULL;
- players[i].poisoner = NULL;
- for (j = 0; j < NUMPSPRITES; j++)
- {
- if (players[i].psprites[j].state)
- {
- players[i].psprites[j].state =
- &states[(int) players[i].psprites[j].state];
- }
- }
}
}
@@ -1361,7 +1874,11 @@
StreamOutWord(ACSInfo[i].state);
StreamOutWord(ACSInfo[i].waitValue);
}
- StreamOutBuffer(MapVars, sizeof(MapVars));
+
+ for (i = 0; i< MAX_ACS_MAP_VARS; ++i)
+ {
+ StreamOutLong(MapVars[i]);
+ }
}
//==========================================================================
@@ -1380,8 +1897,11 @@
ACSInfo[i].state = GET_WORD;
ACSInfo[i].waitValue = GET_WORD;
}
- memcpy(MapVars, SavePtr.b, sizeof(MapVars));
- SavePtr.b += sizeof(MapVars);
+
+ for (i = 0; i < MAX_ACS_MAP_VARS; ++i)
+ {
+ MapVars[i] = GET_LONG;
+ }
}
//==========================================================================
@@ -1753,6 +2273,7 @@
static void StreamOutWord(unsigned short val)
{
+ val = SHORT(val);
fwrite(&val, sizeof(unsigned short), 1, SavingFP);
}
@@ -1764,5 +2285,26 @@
static void StreamOutLong(unsigned int val)
{
+ val = LONG(val);
fwrite(&val, sizeof(int), 1, SavingFP);
}
+
+//==========================================================================
+//
+// StreamOutPtr
+//
+//==========================================================================
+
+static void StreamOutPtr(void *val)
+{
+ long ptr;
+
+ // Write a pointer value. In Vanilla Hexen pointers are 32-bit but
+ // nowadays they might be larger. Whatever value we write here isn't
+ // going to be much use when we reload the game.
+
+ ptr = (long) val;
+ StreamOutLong((unsigned int) (ptr & 0xffffffff));
+}
+
+