2006-04-13 20:47:06 +00:00
|
|
|
//-------------------------------------------------------------------------
|
|
|
|
/*
|
2010-05-25 10:56:00 +00:00
|
|
|
Copyright (C) 2010 EDuke32 developers and contributors
|
2006-04-13 20:47:06 +00:00
|
|
|
|
2010-05-25 10:56:00 +00:00
|
|
|
This file is part of EDuke32.
|
2006-04-13 20:47:06 +00:00
|
|
|
|
|
|
|
EDuke32 is free software; you can redistribute it and/or
|
|
|
|
modify it under the terms of the GNU General Public License version 2
|
|
|
|
as published by the Free Software Foundation.
|
|
|
|
|
|
|
|
This program 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 this program; if not, write to the Free Software
|
2014-07-20 08:55:56 +00:00
|
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
2006-04-13 20:47:06 +00:00
|
|
|
*/
|
|
|
|
//-------------------------------------------------------------------------
|
|
|
|
|
|
|
|
#include "duke3d.h"
|
2010-08-02 08:13:51 +00:00
|
|
|
#include "premap.h"
|
2010-06-23 04:20:46 +00:00
|
|
|
#include "prlights.h"
|
2011-07-29 22:07:49 +00:00
|
|
|
#include "savegame.h"
|
2013-05-20 19:31:42 +00:00
|
|
|
#ifdef LUNATIC
|
|
|
|
# include "lunatic_game.h"
|
|
|
|
static int32_t g_savedOK;
|
|
|
|
const char *g_failedVarname;
|
|
|
|
#endif
|
2006-04-13 20:47:06 +00:00
|
|
|
|
2017-12-18 11:24:53 +00:00
|
|
|
static OutputFileCounter savecounter;
|
2008-08-16 22:15:46 +00:00
|
|
|
|
2011-07-29 22:07:49 +00:00
|
|
|
// For storing pointers in files.
|
|
|
|
// back_p==0: ptr -> "small int"
|
|
|
|
// back_p==1: "small int" -> ptr
|
|
|
|
//
|
|
|
|
// mode: see enum in savegame.h
|
2015-03-24 00:40:12 +00:00
|
|
|
void G_Util_PtrToIdx(void *ptr, int32_t const count, const void *base, int32_t const mode)
|
2011-07-29 22:07:49 +00:00
|
|
|
{
|
2012-11-15 14:28:36 +00:00
|
|
|
intptr_t *iptr = (intptr_t *)ptr;
|
2015-02-11 05:22:23 +00:00
|
|
|
intptr_t const ibase = (intptr_t)base;
|
|
|
|
int32_t const onlynon0_p = mode&P2I_ONLYNON0_BIT;
|
2011-07-29 22:07:49 +00:00
|
|
|
|
|
|
|
// TODO: convert to proper offsets/indices for (a step towards) cross-
|
|
|
|
// compatibility between 32- and 64-bit systems in the netplay.
|
|
|
|
// REMEMBER to bump BYTEVERSION then.
|
|
|
|
|
2015-02-11 05:22:23 +00:00
|
|
|
// WARNING: C std doesn't say that bit pattern of NULL is necessarily 0!
|
|
|
|
if ((mode & P2I_BACK_BIT) == 0)
|
|
|
|
{
|
2016-08-27 01:41:21 +00:00
|
|
|
for (bssize_t i = 0; i < count; i++)
|
2015-02-11 05:22:23 +00:00
|
|
|
if (!onlynon0_p || iptr[i])
|
2011-07-29 22:07:49 +00:00
|
|
|
iptr[i] -= ibase;
|
2015-02-11 05:22:23 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-08-27 01:41:21 +00:00
|
|
|
for (bssize_t i = 0; i < count; i++)
|
2015-02-11 05:22:23 +00:00
|
|
|
if (!onlynon0_p || iptr[i])
|
2011-07-29 22:07:49 +00:00
|
|
|
iptr[i] += ibase;
|
2015-02-11 05:22:23 +00:00
|
|
|
}
|
2011-07-29 22:07:49 +00:00
|
|
|
}
|
|
|
|
|
2015-03-24 00:40:12 +00:00
|
|
|
void G_Util_PtrToIdx2(void *ptr, int32_t const count, size_t const stride, const void *base, int32_t const mode)
|
2012-10-30 15:54:35 +00:00
|
|
|
{
|
2012-11-11 17:57:01 +00:00
|
|
|
uint8_t *iptr = (uint8_t *)ptr;
|
2015-02-11 05:22:23 +00:00
|
|
|
intptr_t const ibase = (intptr_t)base;
|
|
|
|
int32_t const onlynon0_p = mode&P2I_ONLYNON0_BIT;
|
2012-10-30 15:54:35 +00:00
|
|
|
|
2015-02-11 05:22:23 +00:00
|
|
|
if ((mode & P2I_BACK_BIT) == 0)
|
2012-10-30 15:54:35 +00:00
|
|
|
{
|
2016-08-27 01:41:21 +00:00
|
|
|
for (bssize_t i = 0; i < count; ++i)
|
2012-10-30 15:54:35 +00:00
|
|
|
{
|
2015-02-11 05:22:23 +00:00
|
|
|
if (!onlynon0_p || *(intptr_t *)iptr)
|
2012-10-30 15:54:35 +00:00
|
|
|
*(intptr_t *)iptr -= ibase;
|
2015-02-11 05:22:23 +00:00
|
|
|
|
|
|
|
iptr += stride;
|
2012-10-30 15:54:35 +00:00
|
|
|
}
|
2015-02-11 05:22:23 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-08-27 01:41:21 +00:00
|
|
|
for (bssize_t i = 0; i < count; ++i)
|
2015-02-11 05:22:23 +00:00
|
|
|
{
|
|
|
|
if (!onlynon0_p || *(intptr_t *)iptr)
|
|
|
|
*(intptr_t *)iptr += ibase;
|
2012-10-30 15:54:35 +00:00
|
|
|
|
2015-02-11 05:22:23 +00:00
|
|
|
iptr += stride;
|
|
|
|
}
|
2012-10-30 15:54:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-07-29 22:07:28 +00:00
|
|
|
// TODO: sync with TROR special interpolations? (e.g. upper floor of subway)
|
|
|
|
void G_ResetInterpolations(void)
|
|
|
|
{
|
|
|
|
int32_t k, i;
|
|
|
|
|
2016-08-27 01:42:01 +00:00
|
|
|
g_interpolationCnt = 0;
|
2011-07-29 22:07:28 +00:00
|
|
|
|
|
|
|
k = headspritestat[STAT_EFFECTOR];
|
|
|
|
while (k >= 0)
|
|
|
|
{
|
|
|
|
switch (sprite[k].lotag)
|
|
|
|
{
|
2012-08-26 22:16:08 +00:00
|
|
|
case SE_31_FLOOR_RISE_FALL:
|
2011-07-29 22:07:28 +00:00
|
|
|
G_SetInterpolation(§or[sprite[k].sectnum].floorz);
|
|
|
|
break;
|
2012-08-26 22:16:08 +00:00
|
|
|
case SE_32_CEILING_RISE_FALL:
|
2011-07-29 22:07:28 +00:00
|
|
|
G_SetInterpolation(§or[sprite[k].sectnum].ceilingz);
|
|
|
|
break;
|
2012-08-26 22:16:08 +00:00
|
|
|
case SE_17_WARP_ELEVATOR:
|
|
|
|
case SE_25_PISTON:
|
2011-07-29 22:07:28 +00:00
|
|
|
G_SetInterpolation(§or[sprite[k].sectnum].floorz);
|
|
|
|
G_SetInterpolation(§or[sprite[k].sectnum].ceilingz);
|
|
|
|
break;
|
2012-09-08 22:18:44 +00:00
|
|
|
case SE_0_ROTATING_SECTOR:
|
|
|
|
case SE_5:
|
|
|
|
case SE_6_SUBWAY:
|
|
|
|
case SE_11_SWINGING_DOOR:
|
|
|
|
case SE_14_SUBWAY_CAR:
|
|
|
|
case SE_15_SLIDING_DOOR:
|
2012-08-26 22:16:08 +00:00
|
|
|
case SE_16_REACTOR:
|
|
|
|
case SE_26:
|
|
|
|
case SE_30_TWO_WAY_TRAIN:
|
2011-07-29 22:07:28 +00:00
|
|
|
Sect_SetInterpolation(sprite[k].sectnum);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
k = nextspritestat[k];
|
|
|
|
}
|
|
|
|
|
2016-08-27 01:42:01 +00:00
|
|
|
for (i=g_interpolationCnt-1; i>=0; i--) bakipos[i] = *curipos[i];
|
|
|
|
for (i = g_animateCnt-1; i>=0; i--)
|
|
|
|
G_SetInterpolation(g_animatePtr[i]);
|
2011-07-29 22:07:28 +00:00
|
|
|
}
|
|
|
|
|
2017-12-18 11:24:53 +00:00
|
|
|
savebrief_t g_lastautosave, g_lastusersave, g_freshload;
|
|
|
|
int32_t g_lastAutoSaveArbitraryID = -1;
|
|
|
|
bool g_saveRequested;
|
|
|
|
savebrief_t * g_quickload;
|
2011-12-25 15:34:06 +00:00
|
|
|
|
2017-12-18 11:24:53 +00:00
|
|
|
menusave_t * g_menusaves;
|
2018-10-25 23:31:45 +00:00
|
|
|
uint16_t g_nummenusaves;
|
2013-08-04 20:37:45 +00:00
|
|
|
|
2018-03-08 03:55:21 +00:00
|
|
|
static menusave_t * g_internalsaves;
|
2018-10-25 23:31:45 +00:00
|
|
|
static uint16_t g_numinternalsaves;
|
2018-03-08 03:55:21 +00:00
|
|
|
|
2017-12-18 11:24:53 +00:00
|
|
|
static void ReadSaveGameHeaders_CACHE1D(CACHE1D_FIND_REC *f)
|
|
|
|
{
|
|
|
|
savehead_t h;
|
2011-12-25 15:34:06 +00:00
|
|
|
|
2017-12-18 11:24:53 +00:00
|
|
|
for (; f != nullptr; f = f->next)
|
2011-12-25 15:34:06 +00:00
|
|
|
{
|
2017-12-18 11:24:53 +00:00
|
|
|
char const * fn = f->name;
|
|
|
|
int32_t fil = kopen4loadfrommod(fn, 0);
|
2011-12-25 15:34:06 +00:00
|
|
|
if (fil == -1)
|
|
|
|
continue;
|
|
|
|
|
2018-03-08 03:55:21 +00:00
|
|
|
menusave_t & msv = g_internalsaves[g_numinternalsaves];
|
2017-12-18 11:24:53 +00:00
|
|
|
|
|
|
|
int32_t k = sv_loadheader(fil, 0, &h);
|
2011-12-25 15:34:06 +00:00
|
|
|
if (k)
|
|
|
|
{
|
2018-03-08 03:55:21 +00:00
|
|
|
if (k < 0)
|
|
|
|
msv.isUnreadable = 1;
|
|
|
|
msv.isOldVer = 1;
|
2011-12-25 15:34:06 +00:00
|
|
|
}
|
2017-12-18 11:24:53 +00:00
|
|
|
else
|
|
|
|
msv.isOldVer = 0;
|
2011-12-25 15:34:06 +00:00
|
|
|
|
2018-03-08 03:55:17 +00:00
|
|
|
msv.isAutoSave = h.isAutoSave();
|
|
|
|
|
2018-03-08 03:55:21 +00:00
|
|
|
strncpy(msv.brief.path, fn, ARRAY_SIZE(msv.brief.path));
|
|
|
|
++g_numinternalsaves;
|
|
|
|
|
2017-12-18 11:24:53 +00:00
|
|
|
if (k >= 0 && h.savename[0] != '\0')
|
|
|
|
{
|
|
|
|
memcpy(msv.brief.name, h.savename, ARRAY_SIZE(msv.brief.name));
|
|
|
|
}
|
2018-03-08 03:55:21 +00:00
|
|
|
else
|
|
|
|
msv.isUnreadable = 1;
|
2011-12-25 15:34:06 +00:00
|
|
|
|
|
|
|
kclose(fil);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
static int countcache1dfind(CACHE1D_FIND_REC *f)
|
2011-12-25 15:34:06 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
int x = 0;
|
2017-12-18 11:24:53 +00:00
|
|
|
for (; f != nullptr; f = f->next)
|
|
|
|
++x;
|
|
|
|
return x;
|
|
|
|
}
|
|
|
|
|
2018-03-08 03:55:36 +00:00
|
|
|
static void ReadSaveGameHeaders_Internal(void)
|
2017-12-18 11:24:53 +00:00
|
|
|
{
|
|
|
|
static char const DefaultPath[] = "/", SavePattern[] = "*.esv";
|
|
|
|
|
|
|
|
CACHE1D_FIND_REC *findfiles_default = klistpath(DefaultPath, SavePattern, CACHE1D_FIND_FILE);
|
2011-12-25 15:34:06 +00:00
|
|
|
|
2017-12-18 11:24:53 +00:00
|
|
|
// potentially overallocating but programmatically simple
|
2018-10-25 23:31:45 +00:00
|
|
|
int const numfiles = countcache1dfind(findfiles_default);
|
2018-03-08 03:55:21 +00:00
|
|
|
size_t const internalsavesize = sizeof(menusave_t) * numfiles;
|
2018-07-26 16:34:03 +00:00
|
|
|
|
2018-03-08 03:55:21 +00:00
|
|
|
g_internalsaves = (menusave_t *)Xrealloc(g_internalsaves, internalsavesize);
|
2018-07-26 16:34:03 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
for (int x = 0; x < numfiles; ++x)
|
2018-08-02 01:21:19 +00:00
|
|
|
g_internalsaves[x].clear();
|
2014-05-17 12:36:40 +00:00
|
|
|
|
2018-03-08 03:55:21 +00:00
|
|
|
g_numinternalsaves = 0;
|
2017-12-18 11:24:53 +00:00
|
|
|
ReadSaveGameHeaders_CACHE1D(findfiles_default);
|
|
|
|
klistfree(findfiles_default);
|
2018-03-08 03:55:21 +00:00
|
|
|
|
|
|
|
g_nummenusaves = 0;
|
2018-10-25 23:31:45 +00:00
|
|
|
for (int x = g_numinternalsaves-1; x >= 0; --x)
|
2018-03-08 03:55:21 +00:00
|
|
|
{
|
|
|
|
menusave_t & msv = g_internalsaves[x];
|
|
|
|
if (!msv.isUnreadable)
|
|
|
|
{
|
|
|
|
++g_nummenusaves;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
size_t const menusavesize = sizeof(menusave_t) * g_nummenusaves;
|
2018-07-26 16:34:03 +00:00
|
|
|
|
2018-03-08 03:55:21 +00:00
|
|
|
g_menusaves = (menusave_t *)Xrealloc(g_menusaves, menusavesize);
|
2018-07-26 16:34:03 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
for (int x = 0; x < g_nummenusaves; ++x)
|
2018-08-02 01:21:19 +00:00
|
|
|
g_menusaves[x].clear();
|
2018-07-26 16:34:03 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
for (int x = g_numinternalsaves-1, y = 0; x >= 0; --x)
|
2018-03-08 03:55:21 +00:00
|
|
|
{
|
|
|
|
menusave_t & msv = g_internalsaves[x];
|
|
|
|
if (!msv.isUnreadable)
|
|
|
|
{
|
|
|
|
g_menusaves[y++] = msv;
|
|
|
|
}
|
|
|
|
}
|
2018-03-08 03:55:32 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
for (int x = g_numinternalsaves-1; x >= 0; --x)
|
2018-03-08 03:55:32 +00:00
|
|
|
{
|
|
|
|
char const * const path = g_internalsaves[x].brief.path;
|
2018-10-25 23:31:45 +00:00
|
|
|
int const pathlen = Bstrlen(path);
|
2018-03-08 03:55:32 +00:00
|
|
|
if (pathlen < 12)
|
|
|
|
continue;
|
|
|
|
char const * const fn = path + (pathlen-12);
|
|
|
|
if (fn[0] == 's' && fn[1] == 'a' && fn[2] == 'v' && fn[3] == 'e' &&
|
|
|
|
isdigit(fn[4]) && isdigit(fn[5]) && isdigit(fn[6]) && isdigit(fn[7]))
|
|
|
|
{
|
|
|
|
char number[5];
|
|
|
|
memcpy(number, fn+4, 4);
|
|
|
|
number[4] = '\0';
|
2018-10-25 23:31:45 +00:00
|
|
|
savecounter.count = Batoi(number)+1;
|
2018-03-08 03:55:32 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2017-12-18 11:24:53 +00:00
|
|
|
}
|
|
|
|
|
2018-03-08 03:55:36 +00:00
|
|
|
void ReadSaveGameHeaders(void)
|
|
|
|
{
|
|
|
|
ReadSaveGameHeaders_Internal();
|
|
|
|
|
|
|
|
if (!ud.autosavedeletion)
|
|
|
|
return;
|
|
|
|
|
|
|
|
bool didDelete = false;
|
2018-10-25 23:31:45 +00:00
|
|
|
int numautosaves = 0;
|
|
|
|
for (int x = 0; x < g_nummenusaves; ++x)
|
2018-03-08 03:55:36 +00:00
|
|
|
{
|
|
|
|
menusave_t & msv = g_menusaves[x];
|
|
|
|
if (!msv.isAutoSave)
|
|
|
|
continue;
|
|
|
|
if (numautosaves >= ud.maxautosaves)
|
|
|
|
{
|
|
|
|
G_DeleteSave(msv.brief);
|
|
|
|
didDelete = true;
|
|
|
|
}
|
|
|
|
++numautosaves;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (didDelete)
|
|
|
|
ReadSaveGameHeaders_Internal();
|
|
|
|
}
|
|
|
|
|
2017-12-18 11:24:53 +00:00
|
|
|
int32_t G_LoadSaveHeaderNew(char const *fn, savehead_t *saveh)
|
|
|
|
{
|
|
|
|
int32_t fil = kopen4loadfrommod(fn, 0);
|
2011-12-25 15:34:06 +00:00
|
|
|
if (fil == -1)
|
|
|
|
return -1;
|
|
|
|
|
2017-12-18 11:24:53 +00:00
|
|
|
int32_t i = sv_loadheader(fil, 0, saveh);
|
2017-12-01 06:19:04 +00:00
|
|
|
if (i < 0)
|
2011-12-25 15:34:06 +00:00
|
|
|
goto corrupt;
|
|
|
|
|
2017-12-18 11:24:53 +00:00
|
|
|
int32_t screenshotofs;
|
2011-12-25 15:34:06 +00:00
|
|
|
if (kread(fil, &screenshotofs, 4) != 4)
|
|
|
|
goto corrupt;
|
|
|
|
|
|
|
|
walock[TILE_LOADSHOT] = 255;
|
|
|
|
if (waloff[TILE_LOADSHOT] == 0)
|
2018-04-12 21:03:30 +00:00
|
|
|
cacheAllocateBlock(&waloff[TILE_LOADSHOT], 320*200, &walock[TILE_LOADSHOT]);
|
2014-09-30 04:14:21 +00:00
|
|
|
tilesiz[TILE_LOADSHOT].x = 200;
|
|
|
|
tilesiz[TILE_LOADSHOT].y = 320;
|
2011-12-25 15:34:06 +00:00
|
|
|
if (screenshotofs)
|
|
|
|
{
|
2018-03-23 21:41:02 +00:00
|
|
|
if (kdfread_LZ4((char *)waloff[TILE_LOADSHOT], 320, 200, fil) != 200)
|
2011-12-25 15:34:52 +00:00
|
|
|
{
|
2018-03-08 03:55:32 +00:00
|
|
|
OSD_Printf("G_LoadSaveHeaderNew(): failed reading screenshot in \"%s\"\n", fn);
|
2011-12-25 15:34:06 +00:00
|
|
|
goto corrupt;
|
2011-12-25 15:34:52 +00:00
|
|
|
}
|
2011-12-25 15:34:06 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Bmemset((char *)waloff[TILE_LOADSHOT], 0, 320*200);
|
|
|
|
}
|
2018-04-12 21:03:47 +00:00
|
|
|
tileInvalidate(TILE_LOADSHOT, 0, 255);
|
2011-12-25 15:34:06 +00:00
|
|
|
|
|
|
|
kclose(fil);
|
|
|
|
return 0;
|
2006-12-10 06:49:01 +00:00
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
corrupt:
|
|
|
|
kclose(fil);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void sv_postudload();
|
2006-04-13 20:47:06 +00:00
|
|
|
|
2018-01-29 11:13:49 +00:00
|
|
|
// hack
|
|
|
|
static int different_user_map;
|
|
|
|
|
2013-05-19 19:29:18 +00:00
|
|
|
// XXX: keyboard input 'blocked' after load fail? (at least ESC?)
|
2017-12-18 11:24:53 +00:00
|
|
|
int32_t G_LoadPlayer(savebrief_t & sv)
|
2011-12-25 15:34:06 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
int const fil = kopen4loadfrommod(sv.path, 0);
|
2011-12-25 15:34:06 +00:00
|
|
|
|
|
|
|
if (fil == -1)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
ready2send = 0;
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
savehead_t h;
|
|
|
|
int status = sv_loadheader(fil, 0, &h);
|
|
|
|
|
|
|
|
if (status < 0 || h.numplayers != ud.multimode)
|
2011-12-25 15:34:06 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
if (status == -4 || status == -3 || status == 1)
|
2011-12-25 15:34:06 +00:00
|
|
|
P_DoQuote(QUOTE_SAVE_BAD_VERSION, g_player[myconnectindex].ps);
|
2018-10-25 23:31:45 +00:00
|
|
|
else if (h.numplayers != ud.multimode)
|
2011-12-25 15:34:06 +00:00
|
|
|
P_DoQuote(QUOTE_SAVE_BAD_PLAYERS, g_player[myconnectindex].ps);
|
|
|
|
|
|
|
|
kclose(fil);
|
|
|
|
ototalclock = totalclock;
|
|
|
|
ready2send = 1;
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2017-12-02 09:24:55 +00:00
|
|
|
VM_OnEvent(EVENT_PRELOADGAME, g_player[screenpeek].ps->i, screenpeek);
|
2017-06-24 09:21:07 +00:00
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
// some setup first
|
|
|
|
ud.multimode = h.numplayers;
|
|
|
|
|
|
|
|
if (numplayers > 1)
|
|
|
|
{
|
|
|
|
pub = NUMPAGES;
|
|
|
|
pus = NUMPAGES;
|
|
|
|
G_UpdateScreenArea();
|
|
|
|
G_DrawBackground();
|
2017-06-19 23:07:18 +00:00
|
|
|
menutext_center(100, "Loading...");
|
2018-04-12 21:02:51 +00:00
|
|
|
videoNextPage();
|
2011-12-25 15:34:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Net_WaitForServer();
|
|
|
|
|
|
|
|
FX_StopAllSounds();
|
|
|
|
S_ClearSoundLocks();
|
|
|
|
|
|
|
|
// non-"m_" fields will be loaded from svgm_udnetw
|
|
|
|
ud.m_volume_number = h.volnum;
|
|
|
|
ud.m_level_number = h.levnum;
|
|
|
|
ud.m_player_skill = h.skill;
|
|
|
|
|
2015-04-20 20:46:42 +00:00
|
|
|
// NOTE: Bmemcpy needed for SAVEGAME_MUSIC.
|
|
|
|
EDUKE32_STATIC_ASSERT(sizeof(boardfilename) == sizeof(h.boardfn));
|
2018-10-25 23:31:45 +00:00
|
|
|
different_user_map = Bstrcmp(boardfilename, h.boardfn);
|
2015-04-20 20:46:42 +00:00
|
|
|
Bmemcpy(boardfilename, h.boardfn, sizeof(boardfilename));
|
2014-09-07 18:10:15 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
int const mapIdx = h.volnum*MAXLEVELS + h.levnum;
|
2011-12-25 15:34:06 +00:00
|
|
|
|
|
|
|
if (boardfilename[0])
|
|
|
|
Bstrcpy(currentboardfilename, boardfilename);
|
2016-08-27 01:42:01 +00:00
|
|
|
else if (g_mapInfo[mapIdx].filename)
|
|
|
|
Bstrcpy(currentboardfilename, g_mapInfo[mapIdx].filename);
|
2011-12-25 15:34:06 +00:00
|
|
|
|
|
|
|
if (currentboardfilename[0])
|
|
|
|
{
|
2018-04-12 21:03:30 +00:00
|
|
|
artSetupMapArt(currentboardfilename);
|
2013-05-06 19:43:38 +00:00
|
|
|
append_ext_UNSAFE(currentboardfilename, ".mhk");
|
2018-04-12 21:03:47 +00:00
|
|
|
engineLoadMHK(currentboardfilename);
|
2011-12-25 15:34:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Bmemcpy(currentboardfilename, boardfilename, BMAX_PATH);
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
if (status == 2)
|
2014-09-30 04:03:30 +00:00
|
|
|
G_NewGame_EnterLevel();
|
2018-10-25 23:31:45 +00:00
|
|
|
else if ((status = sv_loadsnapshot(fil, 0, &h))) // read the rest...
|
2014-09-30 04:03:30 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
// in theory, we could load into an initial dump first and trivially
|
|
|
|
// recover if things go wrong...
|
|
|
|
Bsprintf(tempbuf, "Loading save game file \"%s\" failed (code %d), cannot recover.", sv.path, status);
|
|
|
|
G_GameExit(tempbuf);
|
2011-12-25 15:34:06 +00:00
|
|
|
}
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
sv_postudload(); // ud.m_XXX = ud.XXX
|
2017-12-02 09:24:55 +00:00
|
|
|
VM_OnEvent(EVENT_LOADGAME, g_player[screenpeek].ps->i, screenpeek);
|
2017-12-17 02:03:55 +00:00
|
|
|
kclose(fil);
|
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-04-19 22:42:21 +00:00
|
|
|
////////// TIMER SAVING/RESTORING //////////
|
|
|
|
|
|
|
|
static struct {
|
|
|
|
int32_t totalclock, totalclocklock; // engine
|
|
|
|
int32_t ototalclock, lockclock; // game
|
|
|
|
} g_timers;
|
|
|
|
|
|
|
|
static void G_SaveTimers(void)
|
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
g_timers.totalclock = totalclock;
|
2014-04-19 22:42:21 +00:00
|
|
|
g_timers.totalclocklock = totalclocklock;
|
2018-10-25 23:31:45 +00:00
|
|
|
g_timers.ototalclock = ototalclock;
|
|
|
|
g_timers.lockclock = lockclock;
|
2014-04-19 22:42:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void G_RestoreTimers(void)
|
|
|
|
{
|
2018-04-12 21:02:51 +00:00
|
|
|
timerUpdate();
|
2014-04-19 22:42:21 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
totalclock = g_timers.totalclock;
|
2014-04-19 22:42:21 +00:00
|
|
|
totalclocklock = g_timers.totalclocklock;
|
2018-10-25 23:31:45 +00:00
|
|
|
ototalclock = g_timers.ototalclock;
|
|
|
|
lockclock = g_timers.lockclock;
|
2014-04-19 22:42:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//////////
|
2011-12-25 15:34:06 +00:00
|
|
|
|
2015-02-11 05:22:23 +00:00
|
|
|
#ifdef __ANDROID__
|
|
|
|
static void G_SavePalette(void)
|
|
|
|
{
|
|
|
|
int32_t pfil;
|
|
|
|
|
|
|
|
if ((pfil = kopen4load("palette.dat", 0)) != -1)
|
|
|
|
{
|
|
|
|
int len = kfilelength(pfil);
|
|
|
|
|
|
|
|
FILE *fil = fopen("palette.dat", "rb");
|
|
|
|
|
|
|
|
if (!fil)
|
|
|
|
{
|
|
|
|
fil = fopen("palette.dat", "wb");
|
|
|
|
|
|
|
|
if (fil)
|
|
|
|
{
|
|
|
|
char *buf = (char *) Xaligned_alloc(16, len);
|
|
|
|
|
|
|
|
kread(pfil, buf, len);
|
|
|
|
fwrite(buf, len, 1, fil);
|
|
|
|
fclose(fil);
|
2018-10-25 23:31:45 +00:00
|
|
|
|
|
|
|
Bfree(buf);
|
2015-02-11 05:22:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else fclose(fil);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-03-08 03:55:25 +00:00
|
|
|
void G_DeleteSave(savebrief_t const & sv)
|
|
|
|
{
|
|
|
|
if (!sv.isValid())
|
|
|
|
return;
|
|
|
|
|
|
|
|
char temp[BMAX_PATH];
|
|
|
|
|
|
|
|
if (G_ModDirSnprintf(temp, sizeof(temp), "%s", sv.path))
|
|
|
|
{
|
|
|
|
OSD_Printf("G_SavePlayer: file name \"%s\" too long\n", sv.path);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
unlink(temp);
|
|
|
|
}
|
|
|
|
|
2018-03-08 03:55:41 +00:00
|
|
|
void G_DeleteOldSaves(void)
|
|
|
|
{
|
|
|
|
ReadSaveGameHeaders();
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
for (int x = 0; x < g_numinternalsaves; ++x)
|
2018-03-08 03:55:41 +00:00
|
|
|
{
|
|
|
|
menusave_t const & msv = g_internalsaves[x];
|
|
|
|
if (msv.isOldVer || msv.isUnreadable)
|
|
|
|
G_DeleteSave(msv.brief);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
uint16_t G_CountOldSaves(void)
|
2018-03-08 05:15:11 +00:00
|
|
|
{
|
|
|
|
ReadSaveGameHeaders();
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
int bad = 0;
|
|
|
|
for (int x = 0; x < g_numinternalsaves; ++x)
|
2018-03-08 05:15:11 +00:00
|
|
|
{
|
|
|
|
menusave_t const & msv = g_internalsaves[x];
|
|
|
|
if (msv.isOldVer || msv.isUnreadable)
|
|
|
|
++bad;
|
|
|
|
}
|
|
|
|
|
|
|
|
return bad;
|
|
|
|
}
|
|
|
|
|
2018-03-08 03:55:17 +00:00
|
|
|
int32_t G_SavePlayer(savebrief_t & sv, bool isAutoSave)
|
2011-12-25 15:34:06 +00:00
|
|
|
{
|
2015-02-11 05:22:23 +00:00
|
|
|
#ifdef __ANDROID__
|
|
|
|
G_SavePalette();
|
|
|
|
#endif
|
|
|
|
|
2014-04-19 22:42:21 +00:00
|
|
|
G_SaveTimers();
|
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
Net_WaitForServer();
|
|
|
|
ready2send = 0;
|
|
|
|
|
2017-12-18 11:24:53 +00:00
|
|
|
char temp[BMAX_PATH];
|
2011-12-25 15:34:06 +00:00
|
|
|
|
2017-12-18 11:24:53 +00:00
|
|
|
errno = 0;
|
|
|
|
FILE *fil;
|
|
|
|
|
|
|
|
if (sv.isValid())
|
|
|
|
{
|
|
|
|
if (G_ModDirSnprintf(temp, sizeof(temp), "%s", sv.path))
|
2012-09-05 17:25:37 +00:00
|
|
|
{
|
2017-12-18 11:24:53 +00:00
|
|
|
OSD_Printf("G_SavePlayer: file name \"%s\" too long\n", sv.path);
|
|
|
|
goto saveproblem;
|
2012-09-05 17:25:37 +00:00
|
|
|
}
|
2011-12-25 15:34:06 +00:00
|
|
|
fil = fopen(temp, "wb");
|
2017-12-18 11:24:53 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
static char const SaveName[] = "save0000.esv";
|
2018-10-25 23:31:45 +00:00
|
|
|
int const len = G_ModDirSnprintfLite(temp, ARRAY_SIZE(temp), SaveName);
|
|
|
|
if (len >= ARRAY_SSIZE(temp)-1)
|
2012-03-01 23:35:53 +00:00
|
|
|
{
|
2017-12-18 11:24:53 +00:00
|
|
|
OSD_Printf("G_SavePlayer: could not form automatic save path\n");
|
|
|
|
goto saveproblem;
|
2012-03-01 23:35:53 +00:00
|
|
|
}
|
2017-12-18 11:24:53 +00:00
|
|
|
char * zeros = temp + (len-8);
|
|
|
|
fil = savecounter.opennextfile(temp, zeros);
|
|
|
|
savecounter.count++;
|
|
|
|
// don't copy the mod dir into sv.path
|
2018-10-25 23:31:45 +00:00
|
|
|
Bstrcpy(sv.path, temp + (len-(ARRAY_SIZE(SaveName)-1)));
|
2017-12-18 11:24:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!fil)
|
|
|
|
{
|
|
|
|
OSD_Printf("G_SavePlayer: failed opening \"%s\" for writing: %s\n",
|
|
|
|
temp, strerror(errno));
|
|
|
|
goto saveproblem;
|
2011-12-25 15:34:06 +00:00
|
|
|
}
|
|
|
|
|
2018-01-29 11:13:30 +00:00
|
|
|
// temporary hack
|
|
|
|
ud.user_map = G_HaveUserMap();
|
|
|
|
|
2012-01-05 21:58:23 +00:00
|
|
|
#ifdef POLYMER
|
2018-04-12 21:03:12 +00:00
|
|
|
if (videoGetRenderMode() == REND_POLYMER)
|
2012-01-05 21:58:23 +00:00
|
|
|
polymer_resetlights();
|
|
|
|
#endif
|
|
|
|
|
2017-12-02 09:24:55 +00:00
|
|
|
VM_OnEvent(EVENT_SAVEGAME, g_player[screenpeek].ps->i, screenpeek);
|
2012-05-25 05:14:19 +00:00
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
// SAVE!
|
2018-03-08 03:55:17 +00:00
|
|
|
sv_saveandmakesnapshot(fil, sv.name, 0, 0, 0, 0, isAutoSave);
|
2011-12-25 15:34:06 +00:00
|
|
|
|
|
|
|
fclose(fil);
|
|
|
|
|
|
|
|
if (!g_netServer && ud.multimode < 2)
|
|
|
|
{
|
2013-05-20 19:31:42 +00:00
|
|
|
#ifdef LUNATIC
|
|
|
|
if (!g_savedOK)
|
2016-08-27 01:40:35 +00:00
|
|
|
Bstrcpy(apStrings[QUOTE_RESERVED4], "^10Failed Saving Game");
|
2013-05-20 19:31:42 +00:00
|
|
|
else
|
|
|
|
#endif
|
2016-08-27 01:40:35 +00:00
|
|
|
Bstrcpy(apStrings[QUOTE_RESERVED4], "Game Saved");
|
2011-12-25 15:34:06 +00:00
|
|
|
P_DoQuote(QUOTE_RESERVED4, g_player[myconnectindex].ps);
|
|
|
|
}
|
|
|
|
|
|
|
|
ready2send = 1;
|
|
|
|
Net_WaitForServer();
|
|
|
|
|
2014-04-19 22:42:21 +00:00
|
|
|
G_RestoreTimers();
|
2011-12-25 15:34:06 +00:00
|
|
|
ototalclock = totalclock;
|
|
|
|
|
2017-12-02 09:24:55 +00:00
|
|
|
VM_OnEvent(EVENT_POSTSAVEGAME, g_player[screenpeek].ps->i, screenpeek);
|
2017-06-24 09:21:07 +00:00
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
return 0;
|
2017-12-18 11:24:53 +00:00
|
|
|
|
|
|
|
saveproblem:
|
|
|
|
ready2send = 1;
|
|
|
|
Net_WaitForServer();
|
|
|
|
|
|
|
|
G_RestoreTimers();
|
|
|
|
ototalclock = totalclock;
|
|
|
|
|
|
|
|
return -1;
|
2011-12-25 15:34:06 +00:00
|
|
|
}
|
2006-04-13 20:47:06 +00:00
|
|
|
|
2018-03-11 03:47:11 +00:00
|
|
|
int32_t G_LoadPlayerMaybeMulti(savebrief_t & sv)
|
2012-09-05 17:25:43 +00:00
|
|
|
{
|
|
|
|
if (g_netServer || ud.multimode > 1)
|
|
|
|
{
|
2016-08-27 01:40:35 +00:00
|
|
|
Bstrcpy(apStrings[QUOTE_RESERVED4], "Multiplayer Loading Not Yet Supported");
|
2012-09-05 17:25:43 +00:00
|
|
|
P_DoQuote(QUOTE_RESERVED4, g_player[myconnectindex].ps);
|
|
|
|
|
|
|
|
// g_player[myconnectindex].ps->gm = MODE_GAME;
|
2018-03-11 03:47:11 +00:00
|
|
|
return 127;
|
2012-09-05 17:25:43 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-12-18 11:24:53 +00:00
|
|
|
int32_t c = G_LoadPlayer(sv);
|
2012-09-05 17:25:43 +00:00
|
|
|
if (c == 0)
|
|
|
|
g_player[myconnectindex].ps->gm = MODE_GAME;
|
2018-03-11 03:47:11 +00:00
|
|
|
return c;
|
2012-09-05 17:25:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-08 03:55:17 +00:00
|
|
|
void G_SavePlayerMaybeMulti(savebrief_t & sv, bool isAutoSave)
|
2012-09-05 17:25:43 +00:00
|
|
|
{
|
2013-04-03 17:26:49 +00:00
|
|
|
CONFIG_WriteSetup(2);
|
2013-03-19 06:23:12 +00:00
|
|
|
|
2012-09-05 17:25:43 +00:00
|
|
|
if (g_netServer || ud.multimode > 1)
|
|
|
|
{
|
2016-08-27 01:40:35 +00:00
|
|
|
Bstrcpy(apStrings[QUOTE_RESERVED4], "Multiplayer Saving Not Yet Supported");
|
2012-09-05 17:25:43 +00:00
|
|
|
P_DoQuote(QUOTE_RESERVED4, g_player[myconnectindex].ps);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-03-08 03:55:17 +00:00
|
|
|
G_SavePlayer(sv, isAutoSave);
|
2012-09-05 17:25:43 +00:00
|
|
|
}
|
|
|
|
}
|
2010-01-24 23:33:17 +00:00
|
|
|
|
|
|
|
////////// GENERIC SAVING/LOADING SYSTEM //////////
|
|
|
|
|
|
|
|
typedef struct dataspec_
|
|
|
|
{
|
|
|
|
uint32_t flags;
|
2016-01-11 05:05:38 +00:00
|
|
|
void * const ptr;
|
2010-01-24 23:33:17 +00:00
|
|
|
uint32_t size;
|
|
|
|
intptr_t cnt;
|
|
|
|
} dataspec_t;
|
|
|
|
|
2016-01-11 05:05:38 +00:00
|
|
|
typedef struct dataspec_gv_
|
|
|
|
{
|
|
|
|
uint32_t flags;
|
|
|
|
void * ptr;
|
|
|
|
uint32_t size;
|
|
|
|
intptr_t cnt;
|
|
|
|
} dataspec_gv_t;
|
|
|
|
|
2010-01-24 23:33:17 +00:00
|
|
|
#define SV_DEFAULTCOMPRTHRES 8
|
|
|
|
static uint8_t savegame_diffcompress; // 0:none, 1:Ken's LZW in cache1d.c
|
|
|
|
static uint8_t savegame_comprthres;
|
|
|
|
|
|
|
|
|
|
|
|
#define DS_DYNAMIC 1 // dereference .ptr one more time
|
|
|
|
#define DS_STRING 2
|
|
|
|
#define DS_CMP 4
|
|
|
|
// 8
|
|
|
|
#define DS_CNT(x) ((sizeof(x))<<3) // .cnt is pointer to...
|
|
|
|
#define DS_CNT16 16
|
|
|
|
#define DS_CNT32 32
|
|
|
|
#define DS_CNTMASK (8|DS_CNT16|DS_CNT32|64)
|
|
|
|
// 64
|
|
|
|
#define DS_LOADFN 128 // .ptr is function that is run when loading
|
|
|
|
#define DS_SAVEFN 256 // .ptr is function that is run when saving
|
2011-02-21 23:08:21 +00:00
|
|
|
#define DS_NOCHK 1024 // don't check for diffs (and don't write out in dump) since assumed constant throughout demo
|
2011-05-12 23:31:13 +00:00
|
|
|
#define DS_PROTECTFN 512
|
2010-01-24 23:33:17 +00:00
|
|
|
#define DS_END (0x70000000)
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
static int32_t ds_getcnt(const dataspec_t *spec)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
int cnt = -1;
|
2015-02-11 05:22:23 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
switch (spec->flags & DS_CNTMASK)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
case 0: cnt = spec->cnt; break;
|
|
|
|
case DS_CNT16: cnt = *((int16_t *)spec->cnt); break;
|
|
|
|
case DS_CNT32: cnt = *((int32_t *)spec->cnt); break;
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
2015-02-11 05:22:23 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
return cnt;
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
static inline void ds_get(const dataspec_t *spec, void **ptr, int32_t *cnt)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
*cnt = ds_getcnt(spec);
|
|
|
|
*ptr = (spec->flags & DS_DYNAMIC) ? *((void **)spec->ptr) : spec->ptr;
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// write state to file and/or to dump
|
|
|
|
static uint8_t *writespecdata(const dataspec_t *spec, FILE *fil, uint8_t *dump)
|
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
for (; spec->flags != DS_END; spec++)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
if (spec->flags & (DS_SAVEFN|DS_LOADFN))
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
if (spec->flags & DS_SAVEFN)
|
|
|
|
(*(void (*)(void))spec->ptr)();
|
2010-01-24 23:33:17 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
if (!fil && (spec->flags & (DS_NOCHK|DS_CMP|DS_STRING)))
|
2010-01-24 23:33:17 +00:00
|
|
|
continue;
|
2018-10-25 23:31:45 +00:00
|
|
|
else if (spec->flags & DS_STRING)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
fwrite(spec->ptr, Bstrlen((const char *)spec->ptr), 1, fil); // not null-terminated!
|
2010-01-24 23:33:17 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
void * ptr;
|
|
|
|
int32_t cnt;
|
|
|
|
|
|
|
|
ds_get(spec, &ptr, &cnt);
|
|
|
|
|
|
|
|
if (cnt < 0)
|
|
|
|
{
|
|
|
|
OSD_Printf("wsd: cnt=%d, f=0x%x.\n", cnt, spec->flags);
|
|
|
|
continue;
|
|
|
|
}
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2015-03-29 02:40:27 +00:00
|
|
|
if (!ptr || !cnt)
|
|
|
|
continue;
|
|
|
|
|
2010-01-24 23:33:17 +00:00
|
|
|
if (fil)
|
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
if ((spec->flags & DS_CMP) || ((spec->flags & DS_CNTMASK) == 0 && spec->size * cnt <= savegame_comprthres))
|
|
|
|
fwrite(ptr, spec->size, cnt, fil);
|
2010-01-24 23:33:17 +00:00
|
|
|
else
|
2018-10-25 23:31:45 +00:00
|
|
|
dfwrite_LZ4((void *)ptr, spec->size, cnt, fil);
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
if (dump && (spec->flags & (DS_NOCHK|DS_CMP)) == 0)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
Bmemcpy(dump, ptr, spec->size * cnt);
|
|
|
|
dump += spec->size * cnt;
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return dump;
|
|
|
|
}
|
|
|
|
|
|
|
|
// let havedump=dumpvar&&*dumpvar
|
|
|
|
// (fil>=0 && havedump): first restore dump from file, then restore state from dump
|
|
|
|
// (fil<0 && havedump): only restore state from dump
|
|
|
|
// (fil>=0 && !havedump): only restore state from file
|
|
|
|
static int32_t readspecdata(const dataspec_t *spec, int32_t fil, uint8_t **dumpvar)
|
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
uint8_t * dump = dumpvar ? *dumpvar : NULL;
|
|
|
|
auto const sptr = spec;
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
for (; spec->flags != DS_END; spec++)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
if (fil < 0 && spec->flags & (DS_NOCHK|DS_STRING|DS_CMP)) // we're updating
|
2010-01-24 23:33:17 +00:00
|
|
|
continue;
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
if (spec->flags & (DS_LOADFN|DS_SAVEFN))
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
if (spec->flags & DS_LOADFN)
|
|
|
|
(*(void (*)())spec->ptr)();
|
2010-01-24 23:33:17 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
if (spec->flags & (DS_STRING|DS_CMP)) // DS_STRING and DS_CMP is for static data only
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
static char cmpstrbuf[32];
|
|
|
|
int const siz = (spec->flags & DS_STRING) ? Bstrlen((const char *)spec->ptr) : spec->size * spec->cnt;
|
|
|
|
int const ksiz = kread(fil, cmpstrbuf, siz);
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
if (ksiz != siz || Bmemcmp(spec->ptr, cmpstrbuf, siz))
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
OSD_Printf("rsd: spec=%s, idx=%d:\n", (char *)sptr->ptr, (int32_t)(spec-sptr));
|
|
|
|
|
|
|
|
if (ksiz!=siz)
|
|
|
|
OSD_Printf(" kread returned %d, expected %d.\n", ksiz, siz);
|
2010-01-24 23:33:17 +00:00
|
|
|
else
|
2011-12-25 15:34:06 +00:00
|
|
|
OSD_Printf(" sp->ptr and cmpstrbuf not identical!\n");
|
2018-10-25 23:31:45 +00:00
|
|
|
|
2010-01-24 23:33:17 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
void * ptr;
|
|
|
|
int32_t cnt;
|
|
|
|
|
|
|
|
ds_get(spec, &ptr, &cnt);
|
|
|
|
|
|
|
|
if (cnt < 0)
|
|
|
|
{
|
|
|
|
OSD_Printf("rsd: cnt<0... wtf?\n");
|
|
|
|
return -1;
|
|
|
|
}
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2015-03-29 02:40:27 +00:00
|
|
|
if (!ptr || !cnt)
|
|
|
|
continue;
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
if (fil >= 0)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
auto const mem = (dump && (spec->flags & DS_NOCHK) == 0) ? dump : (uint8_t *)ptr;
|
|
|
|
bool const comp = !((spec->flags & DS_CNTMASK) == 0 && spec->size * cnt <= savegame_comprthres);
|
|
|
|
int const siz = comp ? cnt : cnt * spec->size;
|
|
|
|
int const ksiz = comp ? kdfread_LZ4(mem, spec->size, siz, fil) : kread(fil, mem, siz);
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
if (ksiz != siz)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
OSD_Printf("rsd: spec=%s, idx=%d, mem=%p\n", (char *)sptr->ptr, (int32_t)(spec - sptr), mem);
|
2011-12-25 15:34:06 +00:00
|
|
|
OSD_Printf(" (%s): read %d, expected %d!\n",
|
2018-10-25 23:31:45 +00:00
|
|
|
((spec->flags & DS_CNTMASK) == 0 && spec->size * cnt <= savegame_comprthres) ? "uncompressed" : "compressed", ksiz, siz);
|
2010-02-23 18:13:46 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
if (ksiz == -1)
|
2011-12-21 18:42:12 +00:00
|
|
|
OSD_Printf(" read: %s\n", strerror(errno));
|
2018-10-25 23:31:45 +00:00
|
|
|
|
2010-01-24 23:33:17 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
if (dump && (spec->flags & DS_NOCHK) == 0)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
Bmemcpy(ptr, dump, spec->size * cnt);
|
|
|
|
dump += spec->size * cnt;
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dumpvar)
|
|
|
|
*dumpvar = dump;
|
2018-10-25 23:31:45 +00:00
|
|
|
|
2010-01-24 23:33:17 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define UINT(bits) uint##bits##_t
|
|
|
|
#define BYTES(bits) (bits>>3)
|
2016-01-11 05:05:38 +00:00
|
|
|
#define VAL(bits,p) (*(UINT(bits) const *)(p))
|
|
|
|
#define WVAL(bits,p) (*(UINT(bits) *)(p))
|
2010-01-24 23:33:17 +00:00
|
|
|
|
|
|
|
static void docmpsd(const void *ptr, void *dump, uint32_t size, uint32_t cnt, uint8_t **diffvar)
|
|
|
|
{
|
|
|
|
uint8_t *retdiff = *diffvar;
|
|
|
|
|
|
|
|
// Hail to the C preprocessor, baby!
|
2018-10-25 23:31:45 +00:00
|
|
|
#define CPSINGLEVAL(Datbits) \
|
|
|
|
if (VAL(Datbits, ptr) != VAL(Datbits, dump)) \
|
|
|
|
{ \
|
|
|
|
WVAL(Datbits, retdiff) = WVAL(Datbits, dump) = VAL(Datbits, ptr); \
|
|
|
|
*diffvar = retdiff + BYTES(Datbits); \
|
|
|
|
}
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
if (cnt == 1)
|
2010-01-24 23:33:17 +00:00
|
|
|
switch (size)
|
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
case 8: CPSINGLEVAL(64); return;
|
|
|
|
case 4: CPSINGLEVAL(32); return;
|
|
|
|
case 2: CPSINGLEVAL(16); return;
|
|
|
|
case 1: CPSINGLEVAL(8); return;
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
#define CPELTS(Idxbits, Datbits) \
|
|
|
|
do \
|
|
|
|
{ \
|
|
|
|
for (int i = 0; i < nelts; i++) \
|
|
|
|
{ \
|
|
|
|
if (*p != *op) \
|
|
|
|
{ \
|
|
|
|
*op = *p; \
|
|
|
|
WVAL(Idxbits, retdiff) = i; \
|
|
|
|
retdiff += BYTES(Idxbits); \
|
|
|
|
WVAL(Datbits, retdiff) = *p; \
|
|
|
|
retdiff += BYTES(Datbits); \
|
|
|
|
} \
|
|
|
|
p++; \
|
|
|
|
op++; \
|
|
|
|
} \
|
|
|
|
WVAL(Idxbits, retdiff) = -1; \
|
|
|
|
retdiff += BYTES(Idxbits); \
|
2010-01-24 23:33:17 +00:00
|
|
|
} while (0)
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
#define CPDATA(Datbits) \
|
|
|
|
do \
|
|
|
|
{ \
|
|
|
|
auto p = (UINT(Datbits) const *)ptr; \
|
|
|
|
auto op = (UINT(Datbits) *)dump; \
|
|
|
|
int nelts = tabledivide32_noinline(size * cnt, BYTES(Datbits)); \
|
|
|
|
if (nelts > 65536) \
|
|
|
|
CPELTS(32, Datbits); \
|
|
|
|
else if (nelts > 256) \
|
|
|
|
CPELTS(16, Datbits); \
|
|
|
|
else \
|
|
|
|
CPELTS(8, Datbits); \
|
2010-01-24 23:33:17 +00:00
|
|
|
} while (0)
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
if (size == 8)
|
2010-01-24 23:33:17 +00:00
|
|
|
CPDATA(64);
|
2018-10-25 23:31:45 +00:00
|
|
|
else if ((size & 3) == 0)
|
2010-01-24 23:33:17 +00:00
|
|
|
CPDATA(32);
|
2018-10-25 23:31:45 +00:00
|
|
|
else if ((size & 1) == 0)
|
2010-01-24 23:33:17 +00:00
|
|
|
CPDATA(16);
|
|
|
|
else
|
|
|
|
CPDATA(8);
|
|
|
|
|
|
|
|
*diffvar = retdiff;
|
|
|
|
|
2010-02-23 18:13:46 +00:00
|
|
|
#undef CPELTS
|
|
|
|
#undef CPSINGLEVAL
|
|
|
|
#undef CPDATA
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// get the number of elements to be monitored for changes
|
|
|
|
static int32_t getnumvar(const dataspec_t *spec)
|
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
int n = 0;
|
|
|
|
for (; spec->flags != DS_END; spec++)
|
|
|
|
if (spec->flags & (DS_STRING|DS_CMP|DS_NOCHK|DS_SAVEFN|DS_LOADFN))
|
|
|
|
++n;
|
2010-01-24 23:33:17 +00:00
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
|
|
|
// update dump at *dumpvar with new state and write diff to *diffvar
|
|
|
|
static void cmpspecdata(const dataspec_t *spec, uint8_t **dumpvar, uint8_t **diffvar)
|
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
uint8_t * dump = *dumpvar;
|
|
|
|
uint8_t * diff = *diffvar;
|
|
|
|
int nbytes = (getnumvar(spec) + 7) >> 3;
|
|
|
|
int const slen = Bstrlen((const char *)spec->ptr);
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
Bmemcpy(diff, spec->ptr, slen);
|
|
|
|
diff += slen;
|
2010-01-24 23:33:17 +00:00
|
|
|
|
|
|
|
while (nbytes--)
|
|
|
|
*(diff++) = 0; // the bitmap of indices which elements of spec have changed go here
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
int eltnum = 0;
|
|
|
|
|
|
|
|
for (spec++; spec->flags!=DS_END; spec++)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
if ((spec->flags&(DS_NOCHK|DS_STRING|DS_CMP)))
|
2010-01-24 23:33:17 +00:00
|
|
|
continue;
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
if (spec->flags&(DS_LOADFN|DS_SAVEFN))
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
if ((spec->flags&(DS_PROTECTFN))==0)
|
|
|
|
(*(void (*)())spec->ptr)();
|
2010-01-24 23:33:17 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
void * ptr;
|
|
|
|
int32_t cnt;
|
|
|
|
|
|
|
|
ds_get(spec, &ptr, &cnt);
|
|
|
|
|
|
|
|
if (cnt < 0)
|
|
|
|
{
|
|
|
|
OSD_Printf("csd: cnt=%d, f=0x%x\n", cnt, spec->flags);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t * const tmptr = diff;
|
|
|
|
|
|
|
|
docmpsd(ptr, dump, spec->size, cnt, &diff);
|
2010-01-24 23:33:17 +00:00
|
|
|
|
|
|
|
if (diff != tmptr)
|
2018-10-25 23:31:45 +00:00
|
|
|
(*diffvar + slen)[eltnum>>3] |= 1<<(eltnum&7);
|
|
|
|
|
|
|
|
dump += spec->size*cnt;
|
2010-01-24 23:33:17 +00:00
|
|
|
eltnum++;
|
|
|
|
}
|
|
|
|
|
|
|
|
*diffvar = diff;
|
|
|
|
*dumpvar = dump;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define VALOFS(bits,p,ofs) (*(((UINT(bits) *)(p)) + (ofs)))
|
|
|
|
|
|
|
|
// apply diff to dump, not to state! state is restored from dump afterwards.
|
|
|
|
static int32_t applydiff(const dataspec_t *spec, uint8_t **dumpvar, uint8_t **diffvar)
|
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
uint8_t * dump = *dumpvar;
|
|
|
|
uint8_t * diff = *diffvar;
|
|
|
|
int const nbytes = (getnumvar(spec)+7)>>3;
|
|
|
|
int const slen = Bstrlen((const char *)spec->ptr);
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
if (Bmemcmp(diff, spec->ptr, slen)) // check STRING magic (sync check)
|
2010-01-24 23:33:17 +00:00
|
|
|
return 1;
|
2011-12-21 18:40:47 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
diff += slen+nbytes;
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
int eltnum = -1;
|
|
|
|
for (spec++; spec->flags != DS_END; spec++)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
if ((spec->flags & (DS_NOCHK|DS_STRING|DS_CMP|DS_LOADFN|DS_SAVEFN)))
|
2010-01-24 23:33:17 +00:00
|
|
|
continue;
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
int const cnt = ds_getcnt(spec);
|
2010-01-24 23:33:17 +00:00
|
|
|
if (cnt < 0) return 1;
|
|
|
|
|
|
|
|
eltnum++;
|
2018-10-25 23:31:45 +00:00
|
|
|
if (((*diffvar+slen)[eltnum>>3] & (1<<(eltnum&7))) == 0)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
dump += spec->size * cnt;
|
2010-01-24 23:33:17 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// ----------
|
2018-10-25 23:31:45 +00:00
|
|
|
#define CPSINGLEVAL(Datbits) \
|
|
|
|
WVAL(Datbits, dump) = VAL(Datbits, diff); \
|
|
|
|
diff += BYTES(Datbits); \
|
|
|
|
dump += BYTES(Datbits)
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
if (cnt == 1)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
switch (spec->size)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
case 8: CPSINGLEVAL(64); continue;
|
|
|
|
case 4: CPSINGLEVAL(32); continue;
|
|
|
|
case 2: CPSINGLEVAL(16); continue;
|
|
|
|
case 1: CPSINGLEVAL(8); continue;
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
#define CPELTS(Idxbits, Datbits) \
|
|
|
|
do \
|
|
|
|
{ \
|
|
|
|
UINT(Idxbits) idx; \
|
|
|
|
goto readidx_##Idxbits##_##Datbits; \
|
|
|
|
do \
|
|
|
|
{ \
|
|
|
|
VALOFS(Datbits, dump, idx) = VAL(Datbits, diff); \
|
|
|
|
diff += BYTES(Datbits); \
|
|
|
|
readidx_##Idxbits##_##Datbits: \
|
|
|
|
idx = VAL(Idxbits, diff); \
|
|
|
|
diff += BYTES(Idxbits); \
|
|
|
|
} while ((int##Idxbits##_t)idx != -1); \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define CPDATA(Datbits) \
|
|
|
|
do \
|
|
|
|
{ \
|
|
|
|
int const elts = tabledivide32_noinline(spec->size * cnt, BYTES(Datbits)); \
|
|
|
|
if (elts > 65536) \
|
|
|
|
CPELTS(32, Datbits); \
|
|
|
|
else if (elts > 256) \
|
|
|
|
CPELTS(16, Datbits); \
|
|
|
|
else \
|
|
|
|
CPELTS(8, Datbits); \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
if (spec->size == 8)
|
2010-01-24 23:33:17 +00:00
|
|
|
CPDATA(64);
|
2018-10-25 23:31:45 +00:00
|
|
|
else if ((spec->size & 3) == 0)
|
2010-01-24 23:33:17 +00:00
|
|
|
CPDATA(32);
|
2018-10-25 23:31:45 +00:00
|
|
|
else if ((spec->size & 1) == 0)
|
2010-01-24 23:33:17 +00:00
|
|
|
CPDATA(16);
|
|
|
|
else
|
|
|
|
CPDATA(8);
|
2018-10-25 23:31:45 +00:00
|
|
|
dump += spec->size * cnt;
|
2010-01-24 23:33:17 +00:00
|
|
|
// ----------
|
|
|
|
|
2010-02-23 18:13:46 +00:00
|
|
|
#undef CPELTS
|
|
|
|
#undef CPSINGLEVAL
|
|
|
|
#undef CPDATA
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
*diffvar = diff;
|
|
|
|
*dumpvar = dump;
|
2010-01-24 23:33:17 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
#undef VAL
|
|
|
|
#undef VALOFS
|
|
|
|
#undef BYTES
|
|
|
|
#undef UINT
|
|
|
|
|
|
|
|
// calculate size needed for dump
|
|
|
|
static uint32_t calcsz(const dataspec_t *spec)
|
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
uint32_t dasiz = 0;
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
for (; spec->flags != DS_END; spec++)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
|
|
|
// DS_STRINGs are used as sync checks in the diffs but not in the dump
|
2018-10-25 23:31:45 +00:00
|
|
|
if ((spec->flags & (DS_CMP|DS_NOCHK|DS_SAVEFN|DS_LOADFN|DS_STRING)))
|
2010-01-24 23:33:17 +00:00
|
|
|
continue;
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
int const cnt = ds_getcnt(spec);
|
|
|
|
|
|
|
|
if (cnt <= 0)
|
|
|
|
continue;
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
dasiz += cnt * spec->size;
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return dasiz;
|
|
|
|
}
|
|
|
|
|
2011-03-04 08:50:58 +00:00
|
|
|
#ifdef USE_OPENGL
|
2010-01-25 01:11:34 +00:00
|
|
|
static void sv_prespriteextsave();
|
|
|
|
static void sv_postspriteext();
|
2011-02-27 22:11:39 +00:00
|
|
|
#endif
|
2018-09-28 04:27:48 +00:00
|
|
|
#if defined LUNATIC
|
2013-06-07 10:18:17 +00:00
|
|
|
// Recreate Lua state.
|
|
|
|
// XXX: It may matter a great deal when this is run from if the Lua code refers
|
|
|
|
// to C-side data at file scope. Such usage is strongly discouraged though.
|
|
|
|
static void sv_create_lua_state(void)
|
|
|
|
{
|
|
|
|
El_CreateGameState();
|
|
|
|
G_PostCreateGameState();
|
|
|
|
}
|
2013-02-03 12:48:17 +00:00
|
|
|
#endif
|
2010-01-25 01:11:34 +00:00
|
|
|
static void sv_postactordata();
|
|
|
|
static void sv_preanimateptrsave();
|
|
|
|
static void sv_postanimateptr();
|
|
|
|
static void sv_prequote();
|
|
|
|
static void sv_quotesave();
|
|
|
|
static void sv_quoteload();
|
|
|
|
static void sv_prequoteredef();
|
|
|
|
static void sv_quoteredefsave();
|
|
|
|
static void sv_quoteredefload();
|
|
|
|
static void sv_postquoteredef();
|
|
|
|
static void sv_restsave();
|
|
|
|
static void sv_restload();
|
2015-03-24 00:40:55 +00:00
|
|
|
static void sv_preprojectilesave();
|
|
|
|
static void sv_postprojectilesave();
|
|
|
|
static void sv_preprojectileload();
|
|
|
|
static void sv_postprojectileload();
|
|
|
|
|
2018-09-28 04:27:48 +00:00
|
|
|
static projectile_t *savegame_projectiledata;
|
2018-10-25 23:31:45 +00:00
|
|
|
static uint8_t savegame_projectiles[MAXTILES >> 3];
|
|
|
|
static int32_t savegame_projectilecnt = 0;
|
2010-01-25 01:11:34 +00:00
|
|
|
|
|
|
|
#define SVARDATALEN \
|
|
|
|
((sizeof(g_player[0].user_name)+sizeof(g_player[0].pcolor)+sizeof(g_player[0].pteam) \
|
2012-12-09 13:24:32 +00:00
|
|
|
+sizeof(g_player[0].frags)+sizeof(DukePlayer_t))*MAXPLAYERS)
|
2010-01-25 01:11:34 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
static uint8_t savegame_quotedef[MAXQUOTES >> 3];
|
|
|
|
static char (*savegame_quotes)[MAXQUOTELEN];
|
|
|
|
static char (*savegame_quoteredefs)[MAXQUOTELEN];
|
2010-01-25 01:11:34 +00:00
|
|
|
static uint8_t savegame_restdata[SVARDATALEN];
|
|
|
|
|
2016-01-11 05:05:38 +00:00
|
|
|
static char svgm_udnetw_string [] = "blK:udnt";
|
2010-01-25 01:11:34 +00:00
|
|
|
static const dataspec_t svgm_udnetw[] =
|
|
|
|
{
|
2016-01-11 05:05:38 +00:00
|
|
|
{ DS_STRING, (void *)svgm_udnetw_string, 0, 1 },
|
2010-01-25 01:11:34 +00:00
|
|
|
{ 0, &ud.multimode, sizeof(ud.multimode), 1 },
|
2016-08-27 01:42:01 +00:00
|
|
|
{ 0, &g_playerSpawnCnt, sizeof(g_playerSpawnCnt), 1 },
|
2010-01-25 01:11:34 +00:00
|
|
|
{ 0, &g_playerSpawnPoints, sizeof(g_playerSpawnPoints), 1 },
|
|
|
|
|
|
|
|
{ DS_NOCHK, &ud.volume_number, sizeof(ud.volume_number), 1 },
|
|
|
|
{ DS_NOCHK, &ud.level_number, sizeof(ud.level_number), 1 },
|
2017-12-01 06:19:15 +00:00
|
|
|
{ DS_NOCHK, &ud.user_map, sizeof(ud.user_map), 1 },
|
2010-01-25 01:11:34 +00:00
|
|
|
{ DS_NOCHK, &ud.player_skill, sizeof(ud.player_skill), 1 },
|
2017-12-01 06:19:15 +00:00
|
|
|
{ DS_NOCHK, &ud.music_episode, sizeof(ud.music_episode), 1 },
|
|
|
|
{ DS_NOCHK, &ud.music_level, sizeof(ud.music_level), 1 },
|
2010-01-25 01:11:34 +00:00
|
|
|
|
|
|
|
{ DS_NOCHK, &ud.from_bonus, sizeof(ud.from_bonus), 1 },
|
|
|
|
{ DS_NOCHK, &ud.secretlevel, sizeof(ud.secretlevel), 1 },
|
|
|
|
{ DS_NOCHK, &ud.respawn_monsters, sizeof(ud.respawn_monsters), 1 },
|
|
|
|
{ DS_NOCHK, &ud.respawn_items, sizeof(ud.respawn_items), 1 },
|
|
|
|
{ DS_NOCHK, &ud.respawn_inventory, sizeof(ud.respawn_inventory), 1 },
|
|
|
|
{ 0, &ud.god, sizeof(ud.god), 1 },
|
|
|
|
{ 0, &ud.auto_run, sizeof(ud.auto_run), 1 },
|
|
|
|
// { DS_NOCHK, &ud.crosshair, sizeof(ud.crosshair), 1 },
|
|
|
|
{ DS_NOCHK, &ud.monsters_off, sizeof(ud.monsters_off), 1 },
|
|
|
|
{ DS_NOCHK, &ud.last_level, sizeof(ud.last_level), 1 },
|
|
|
|
{ 0, &ud.eog, sizeof(ud.eog), 1 },
|
|
|
|
{ DS_NOCHK, &ud.coop, sizeof(ud.coop), 1 },
|
|
|
|
{ DS_NOCHK, &ud.marker, sizeof(ud.marker), 1 },
|
|
|
|
{ DS_NOCHK, &ud.ffire, sizeof(ud.ffire), 1 },
|
|
|
|
{ DS_NOCHK, &ud.noexits, sizeof(ud.noexits), 1 },
|
|
|
|
{ DS_NOCHK, &ud.playerai, sizeof(ud.playerai), 1 },
|
2010-02-13 21:46:42 +00:00
|
|
|
{ 0, &ud.pause_on, sizeof(ud.pause_on), 1 },
|
2010-01-25 01:11:34 +00:00
|
|
|
{ DS_NOCHK, ¤tboardfilename[0], BMAX_PATH, 1 },
|
2011-12-25 15:34:06 +00:00
|
|
|
// { DS_LOADFN, (void *)&sv_postudload, 0, 1 },
|
2010-01-25 01:11:34 +00:00
|
|
|
{ 0, connectpoint2, sizeof(connectpoint2), 1 },
|
|
|
|
{ 0, &randomseed, sizeof(randomseed), 1 },
|
|
|
|
{ 0, &g_globalRandom, sizeof(g_globalRandom), 1 },
|
2013-07-04 19:38:46 +00:00
|
|
|
#ifdef LUNATIC
|
|
|
|
// Save game tic count for Lunatic because it is exposed to userland. See
|
|
|
|
// test/helixspawner.lua for an example.
|
|
|
|
{ 0, &g_moveThingsCount, sizeof(g_moveThingsCount), 1 },
|
|
|
|
#endif
|
2010-01-25 01:11:34 +00:00
|
|
|
// { 0, &lockclock_dummy, sizeof(lockclock), 1 },
|
|
|
|
{ DS_END, 0, 0, 0 }
|
|
|
|
};
|
|
|
|
|
Inreased debugging level for catching oob accesses to 'main' arrays.
If enabled, this makes the following arrays be allocated statically:
spriteext, spritesmooth, sector, wall, sprite, tsprite, while
necessarily disabling the clipshape feature (because it relies on
setting sector/wall to different malloc'd block temporarily).
To compile, pass DEBUGANYWAY=1 in addition to RELEASE=0 to 'make',
and it's really only useful with CC=clang, of course.
git-svn-id: https://svn.eduke32.com/eduke32@2270 1a8010ca-5511-0410-912e-c29ae57300e0
2012-01-19 23:17:34 +00:00
|
|
|
#if !defined DEBUG_MAIN_ARRAYS
|
|
|
|
# define DS_MAINAR DS_DYNAMIC
|
|
|
|
#else
|
|
|
|
# define DS_MAINAR 0
|
|
|
|
#endif
|
|
|
|
|
2016-01-11 05:05:38 +00:00
|
|
|
static char svgm_secwsp_string [] = "blK:swsp";
|
2010-01-25 01:11:34 +00:00
|
|
|
static const dataspec_t svgm_secwsp[] =
|
|
|
|
{
|
2016-01-11 05:05:38 +00:00
|
|
|
{ DS_STRING, (void *)svgm_secwsp_string, 0, 1 },
|
2010-01-25 01:11:34 +00:00
|
|
|
{ DS_NOCHK, &numwalls, sizeof(numwalls), 1 },
|
Inreased debugging level for catching oob accesses to 'main' arrays.
If enabled, this makes the following arrays be allocated statically:
spriteext, spritesmooth, sector, wall, sprite, tsprite, while
necessarily disabling the clipshape feature (because it relies on
setting sector/wall to different malloc'd block temporarily).
To compile, pass DEBUGANYWAY=1 in addition to RELEASE=0 to 'make',
and it's really only useful with CC=clang, of course.
git-svn-id: https://svn.eduke32.com/eduke32@2270 1a8010ca-5511-0410-912e-c29ae57300e0
2012-01-19 23:17:34 +00:00
|
|
|
{ DS_MAINAR|DS_CNT(numwalls), &wall, sizeof(walltype), (intptr_t)&numwalls },
|
2010-01-25 01:11:34 +00:00
|
|
|
{ DS_NOCHK, &numsectors, sizeof(numsectors), 1 },
|
Inreased debugging level for catching oob accesses to 'main' arrays.
If enabled, this makes the following arrays be allocated statically:
spriteext, spritesmooth, sector, wall, sprite, tsprite, while
necessarily disabling the clipshape feature (because it relies on
setting sector/wall to different malloc'd block temporarily).
To compile, pass DEBUGANYWAY=1 in addition to RELEASE=0 to 'make',
and it's really only useful with CC=clang, of course.
git-svn-id: https://svn.eduke32.com/eduke32@2270 1a8010ca-5511-0410-912e-c29ae57300e0
2012-01-19 23:17:34 +00:00
|
|
|
{ DS_MAINAR|DS_CNT(numsectors), §or, sizeof(sectortype), (intptr_t)&numsectors },
|
|
|
|
{ DS_MAINAR, &sprite, sizeof(spritetype), MAXSPRITES },
|
2011-05-12 23:31:13 +00:00
|
|
|
#ifdef YAX_ENABLE
|
|
|
|
{ DS_NOCHK, &numyaxbunches, sizeof(numyaxbunches), 1 },
|
2013-04-09 17:35:11 +00:00
|
|
|
# if !defined NEW_MAP_FORMAT
|
2011-05-12 23:31:13 +00:00
|
|
|
{ DS_CNT(numsectors), yax_bunchnum, sizeof(yax_bunchnum[0]), (intptr_t)&numsectors },
|
|
|
|
{ DS_CNT(numwalls), yax_nextwall, sizeof(yax_nextwall[0]), (intptr_t)&numwalls },
|
2013-04-09 17:35:11 +00:00
|
|
|
# endif
|
2011-05-12 23:31:13 +00:00
|
|
|
{ DS_LOADFN|DS_PROTECTFN, (void *)&sv_postyaxload, 0, 1 },
|
|
|
|
#endif
|
2012-03-14 22:31:49 +00:00
|
|
|
{ 0, &Numsprites, sizeof(Numsprites), 1 },
|
2012-03-13 20:07:17 +00:00
|
|
|
{ 0, &tailspritefree, sizeof(tailspritefree), 1 },
|
2010-01-25 01:11:34 +00:00
|
|
|
{ 0, &headspritesect[0], sizeof(headspritesect[0]), MAXSECTORS+1 },
|
|
|
|
{ 0, &prevspritesect[0], sizeof(prevspritesect[0]), MAXSPRITES },
|
|
|
|
{ 0, &nextspritesect[0], sizeof(nextspritesect[0]), MAXSPRITES },
|
|
|
|
{ 0, &headspritestat[0], sizeof(headspritestat[0]), MAXSTATUS+1 },
|
|
|
|
{ 0, &prevspritestat[0], sizeof(prevspritestat[0]), MAXSPRITES },
|
|
|
|
{ 0, &nextspritestat[0], sizeof(nextspritestat[0]), MAXSPRITES },
|
2011-03-04 08:50:58 +00:00
|
|
|
#ifdef USE_OPENGL
|
2010-01-25 01:11:34 +00:00
|
|
|
{ DS_SAVEFN, (void *)&sv_prespriteextsave, 0, 1 },
|
|
|
|
#endif
|
Inreased debugging level for catching oob accesses to 'main' arrays.
If enabled, this makes the following arrays be allocated statically:
spriteext, spritesmooth, sector, wall, sprite, tsprite, while
necessarily disabling the clipshape feature (because it relies on
setting sector/wall to different malloc'd block temporarily).
To compile, pass DEBUGANYWAY=1 in addition to RELEASE=0 to 'make',
and it's really only useful with CC=clang, of course.
git-svn-id: https://svn.eduke32.com/eduke32@2270 1a8010ca-5511-0410-912e-c29ae57300e0
2012-01-19 23:17:34 +00:00
|
|
|
{ DS_MAINAR, &spriteext, sizeof(spriteext_t), MAXSPRITES },
|
2017-06-25 11:23:53 +00:00
|
|
|
#ifndef NEW_MAP_FORMAT
|
|
|
|
{ DS_MAINAR, &wallext, sizeof(wallext_t), MAXWALLS },
|
|
|
|
#endif
|
2011-03-04 08:50:58 +00:00
|
|
|
#ifdef USE_OPENGL
|
2010-01-25 01:11:34 +00:00
|
|
|
{ DS_SAVEFN|DS_LOADFN, (void *)&sv_postspriteext, 0, 1 },
|
|
|
|
#endif
|
2016-08-27 01:42:01 +00:00
|
|
|
{ DS_NOCHK, &g_cyclerCnt, sizeof(g_cyclerCnt), 1 },
|
|
|
|
{ DS_CNT(g_cyclerCnt), &g_cyclers[0][0], sizeof(g_cyclers[0]), (intptr_t)&g_cyclerCnt },
|
|
|
|
{ DS_NOCHK, &g_animWallCnt, sizeof(g_animWallCnt), 1 },
|
|
|
|
{ DS_CNT(g_animWallCnt), &animwall, sizeof(animwall[0]), (intptr_t)&g_animWallCnt },
|
2010-01-25 01:11:34 +00:00
|
|
|
{ DS_NOCHK, &g_mirrorCount, sizeof(g_mirrorCount), 1 },
|
2014-03-22 09:25:15 +00:00
|
|
|
{ DS_NOCHK, &g_mirrorWall[0], sizeof(g_mirrorWall[0]), ARRAY_SIZE(g_mirrorWall) },
|
|
|
|
{ DS_NOCHK, &g_mirrorSector[0], sizeof(g_mirrorSector[0]), ARRAY_SIZE(g_mirrorSector) },
|
2010-01-25 01:11:34 +00:00
|
|
|
// projectiles
|
|
|
|
{ 0, &SpriteProjectile[0], sizeof(projectile_t), MAXSPRITES },
|
|
|
|
{ 0, &everyothertime, sizeof(everyothertime), 1 },
|
|
|
|
{ DS_END, 0, 0, 0 }
|
|
|
|
};
|
|
|
|
|
2016-01-11 05:05:38 +00:00
|
|
|
static char svgm_script_string [] = "blK:scri";
|
2010-01-25 01:11:34 +00:00
|
|
|
static const dataspec_t svgm_script[] =
|
|
|
|
{
|
2016-01-11 05:05:38 +00:00
|
|
|
{ DS_STRING, (void *)svgm_script_string, 0, 1 },
|
2015-03-24 00:40:55 +00:00
|
|
|
{ DS_SAVEFN, (void *) &sv_preprojectilesave, 0, 1 },
|
2018-10-03 00:43:31 +00:00
|
|
|
{ 0, savegame_projectiles, sizeof(uint8_t), MAXTILES >> 3 },
|
2015-03-24 00:40:55 +00:00
|
|
|
{ DS_LOADFN, (void *) &sv_preprojectileload, 0, 1 },
|
2018-10-25 23:31:45 +00:00
|
|
|
{ DS_DYNAMIC|DS_CNT(savegame_projectilecnt), &savegame_projectiledata, sizeof(projectile_t), (intptr_t)&savegame_projectilecnt },
|
2015-03-24 00:40:55 +00:00
|
|
|
{ DS_SAVEFN, (void *) &sv_postprojectilesave, 0, 1 },
|
|
|
|
{ DS_LOADFN, (void *) &sv_postprojectileload, 0, 1 },
|
2010-08-02 08:13:51 +00:00
|
|
|
{ 0, &actor[0], sizeof(actor_t), MAXSPRITES },
|
2010-01-25 01:11:34 +00:00
|
|
|
{ DS_SAVEFN|DS_LOADFN, (void *)&sv_postactordata, 0, 1 },
|
2013-06-07 10:18:17 +00:00
|
|
|
#if defined LUNATIC
|
|
|
|
{ DS_LOADFN|DS_NOCHK, (void *)&sv_create_lua_state, 0, 1 },
|
|
|
|
#endif
|
2010-01-25 01:11:34 +00:00
|
|
|
{ DS_END, 0, 0, 0 }
|
|
|
|
};
|
|
|
|
|
2016-01-11 05:05:38 +00:00
|
|
|
static char svgm_anmisc_string [] = "blK:anms";
|
|
|
|
static char svgm_end_string [] = "savegame_end";
|
|
|
|
|
2010-01-25 01:11:34 +00:00
|
|
|
static const dataspec_t svgm_anmisc[] =
|
|
|
|
{
|
2016-01-11 05:05:38 +00:00
|
|
|
{ DS_STRING, (void *)svgm_anmisc_string, 0, 1 },
|
2016-08-27 01:42:01 +00:00
|
|
|
{ 0, &g_animateCnt, sizeof(g_animateCnt), 1 },
|
|
|
|
{ 0, &g_animateSect[0], sizeof(g_animateSect[0]), MAXANIMATES },
|
|
|
|
{ 0, &g_animateGoal[0], sizeof(g_animateGoal[0]), MAXANIMATES },
|
|
|
|
{ 0, &g_animateVel[0], sizeof(g_animateVel[0]), MAXANIMATES },
|
2010-01-25 01:11:34 +00:00
|
|
|
{ DS_SAVEFN, (void *)&sv_preanimateptrsave, 0, 1 },
|
2016-08-27 01:42:01 +00:00
|
|
|
{ 0, &g_animatePtr[0], sizeof(g_animatePtr[0]), MAXANIMATES },
|
2010-01-25 01:11:34 +00:00
|
|
|
{ DS_SAVEFN|DS_LOADFN , (void *)&sv_postanimateptr, 0, 1 },
|
2013-12-26 19:45:12 +00:00
|
|
|
{ 0, &g_curViewscreen, sizeof(g_curViewscreen), 1 },
|
2016-01-21 19:35:30 +00:00
|
|
|
{ 0, &g_origins[0], sizeof(g_origins[0]), ARRAY_SIZE(g_origins) },
|
2010-02-13 21:46:42 +00:00
|
|
|
{ 0, &g_spriteDeleteQueuePos, sizeof(g_spriteDeleteQueuePos), 1 },
|
2016-08-27 01:40:35 +00:00
|
|
|
{ DS_NOCHK, &g_deleteQueueSize, sizeof(g_deleteQueueSize), 1 },
|
|
|
|
{ DS_CNT(g_deleteQueueSize), &SpriteDeletionQueue[0], sizeof(int16_t), (intptr_t)&g_deleteQueueSize },
|
2010-01-25 01:11:34 +00:00
|
|
|
{ 0, &show2dsector[0], sizeof(uint8_t), MAXSECTORS>>3 },
|
2016-08-27 01:42:01 +00:00
|
|
|
{ DS_NOCHK, &g_cloudCnt, sizeof(g_cloudCnt), 1 },
|
|
|
|
{ 0, &g_cloudSect[0], sizeof(g_cloudSect), 1 },
|
|
|
|
{ 0, &g_cloudX, sizeof(g_cloudX), 1 },
|
|
|
|
{ 0, &g_cloudY, sizeof(g_cloudY), 1 },
|
Clean up parallaxed sky functionality, part 2.
- Rename sky_t members: yscale -> horizfrac, bits -> lognumtiles.
- Add default sky (8 tiles, horizfrac=32768 (i.e. 1/2 the scene horiz), offsets
all zero) and CLOUDYOCEAN sky (8 tiles, horizfrac=65536, offsets all zero)
to multipsky[].
- Get rid of "psky_t g_psky", merely maintaining a g_pskyidx instead. Set it up
at map load time so as to keep the behavior of the legacy per-map psky:
the last sector index with a matching psky ceiling wins.
- In mapstate_t, save g_pskyidx too, not (former) pskybits and pskyoffs[].
- Make on-map-load global psky setup consistent for the game and editor by
factoring it out into common.c: G_SetupGlobalPsky().
- Remove a couple of useless initializations, add some static assertions.
This commit is more likely to introduce subtle differences in behavior.
Specifically, getpsky() now always returns the default sky properties instead of
the global sky ones (but with all-zero offsets) when no match for a suiting
multi-psky is found. This is only likely to affect the yscale/horizfrac of
non-multi-pskies when a global non-default multi-psky has been set up.
Bump BYTEVERSION again.
git-svn-id: https://svn.eduke32.com/eduke32@3976 1a8010ca-5511-0410-912e-c29ae57300e0
2013-08-04 20:37:48 +00:00
|
|
|
{ 0, &g_pskyidx, sizeof(g_pskyidx), 1 }, // DS_NOCHK?
|
2010-01-25 01:11:34 +00:00
|
|
|
{ 0, &g_earthquakeTime, sizeof(g_earthquakeTime), 1 },
|
|
|
|
|
|
|
|
{ DS_SAVEFN|DS_LOADFN|DS_NOCHK, (void *)sv_prequote, 0, 1 },
|
|
|
|
{ DS_SAVEFN, (void *)&sv_quotesave, 0, 1 },
|
|
|
|
{ DS_NOCHK, &savegame_quotedef, sizeof(savegame_quotedef), 1 }, // quotes can change during runtime, but new quote numbers cannot be allocated
|
|
|
|
{ DS_DYNAMIC, &savegame_quotes, MAXQUOTELEN, MAXQUOTES },
|
|
|
|
{ DS_LOADFN, (void *)&sv_quoteload, 0, 1 },
|
|
|
|
|
|
|
|
{ DS_NOCHK|DS_SAVEFN|DS_LOADFN, (void *)&sv_prequoteredef, 0, 1 },
|
|
|
|
{ DS_NOCHK|DS_SAVEFN, (void *)&sv_quoteredefsave, 0, 1 }, // quote redefinitions replace quotes at runtime, but cannot be changed after CON compilation
|
2016-08-27 01:40:35 +00:00
|
|
|
{ DS_NOCHK|DS_DYNAMIC|DS_CNT(g_numXStrings), &savegame_quoteredefs, MAXQUOTELEN, (intptr_t)&g_numXStrings },
|
2010-01-25 01:11:34 +00:00
|
|
|
{ DS_NOCHK|DS_LOADFN, (void *)&sv_quoteredefload, 0, 1 },
|
|
|
|
{ DS_NOCHK|DS_SAVEFN|DS_LOADFN, (void *)&sv_postquoteredef, 0, 1 },
|
2013-06-13 17:11:09 +00:00
|
|
|
#ifdef LUNATIC
|
|
|
|
{ 0, g_playerWeapon, sizeof(weapondata_t), MAXPLAYERS*MAX_WEAPONS },
|
|
|
|
#endif
|
2010-01-25 01:11:34 +00:00
|
|
|
{ DS_SAVEFN, (void *)&sv_restsave, 0, 1 },
|
|
|
|
{ 0, savegame_restdata, 1, sizeof(savegame_restdata) }, // sz/cnt swapped for kdfread
|
|
|
|
{ DS_LOADFN, (void *)&sv_restload, 0, 1 },
|
|
|
|
|
2016-01-11 05:05:38 +00:00
|
|
|
{ DS_STRING, (void *)svgm_end_string, 0, 1 },
|
2010-01-25 01:11:34 +00:00
|
|
|
{ DS_END, 0, 0, 0 }
|
|
|
|
};
|
|
|
|
|
2013-05-19 19:29:18 +00:00
|
|
|
#if !defined LUNATIC
|
2016-01-11 05:05:38 +00:00
|
|
|
static dataspec_gv_t *svgm_vars=NULL;
|
2013-05-19 19:29:18 +00:00
|
|
|
#endif
|
2011-12-25 15:34:06 +00:00
|
|
|
static uint8_t *dosaveplayer2(FILE *fil, uint8_t *mem);
|
|
|
|
static int32_t doloadplayer2(int32_t fil, uint8_t **memptr);
|
|
|
|
static void postloadplayer(int32_t savegamep);
|
2010-01-24 23:33:17 +00:00
|
|
|
|
|
|
|
// SVGM snapshot system
|
|
|
|
static uint32_t svsnapsiz;
|
2018-10-25 23:31:45 +00:00
|
|
|
static uint8_t *svsnapshot;
|
|
|
|
static uint8_t *svinitsnap;
|
2010-01-24 23:33:17 +00:00
|
|
|
static uint32_t svdiffsiz;
|
|
|
|
static uint8_t *svdiff;
|
|
|
|
|
|
|
|
#include "gamedef.h"
|
|
|
|
|
2013-05-19 19:29:18 +00:00
|
|
|
#if !defined LUNATIC
|
2018-03-07 04:21:31 +00:00
|
|
|
#define SV_SKIPMASK (/*GAMEVAR_SYSTEM|*/ GAMEVAR_READONLY | GAMEVAR_PTR_MASK | /*GAMEVAR_NORESET |*/ GAMEVAR_SPECIAL)
|
2016-01-11 05:06:10 +00:00
|
|
|
|
|
|
|
static char svgm_vars_string [] = "blK:vars";
|
2010-01-24 23:33:17 +00:00
|
|
|
// setup gamevar data spec for snapshotting and diffing... gamevars must be loaded when called
|
|
|
|
static void sv_makevarspec()
|
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
int vcnt = 0;
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
for (int i = 0; i < g_gameVarCount; i++)
|
|
|
|
vcnt += (aGameVars[i].flags & SV_SKIPMASK) ? 0 : 1;
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
for (int i=0; i<g_gameArrayCount; i++)
|
|
|
|
vcnt += !(aGameArrays[i].flags & (GAMEARRAY_SYSTEM|GAMEARRAY_READONLY)); // SYSTEM_GAMEARRAY
|
2013-12-06 18:56:39 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
svgm_vars = (dataspec_gv_t *)Xrealloc(svgm_vars, (vcnt + 2) * sizeof(dataspec_gv_t));
|
2013-05-24 13:54:32 +00:00
|
|
|
|
2010-01-24 23:33:17 +00:00
|
|
|
svgm_vars[0].flags = DS_STRING;
|
2018-10-25 23:31:45 +00:00
|
|
|
svgm_vars[0].ptr = svgm_vars_string;
|
|
|
|
svgm_vars[0].cnt = 1;
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
vcnt = 1;
|
|
|
|
|
|
|
|
for (int i = 0; i < g_gameVarCount; i++)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
if (aGameVars[i].flags & SV_SKIPMASK)
|
2010-01-24 23:33:17 +00:00
|
|
|
continue;
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
unsigned const per = aGameVars[i].flags & GAMEVAR_USER_MASK;
|
|
|
|
|
|
|
|
svgm_vars[vcnt].flags = 0;
|
|
|
|
svgm_vars[vcnt].ptr = (per == 0) ? &aGameVars[i].global : aGameVars[i].pValues;
|
|
|
|
svgm_vars[vcnt].size = sizeof(intptr_t);
|
|
|
|
svgm_vars[vcnt].cnt = (per == 0) ? 1 : (per == GAMEVAR_PERPLAYER ? MAXPLAYERS : MAXSPRITES);
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
++vcnt;
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
for (int i = 0; i < g_gameArrayCount; i++)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2013-12-06 18:56:39 +00:00
|
|
|
// We must not update read-only SYSTEM_GAMEARRAY gamearrays: besides
|
|
|
|
// being questionable by itself, sizeof(...) may be e.g. 4 whereas the
|
|
|
|
// actual element type is int16_t (such as tilesizx[]/tilesizy[]).
|
2017-07-08 19:41:43 +00:00
|
|
|
if (aGameArrays[i].flags & (GAMEARRAY_SYSTEM|GAMEARRAY_READONLY))
|
2013-12-06 18:56:39 +00:00
|
|
|
continue;
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
auto const pValues = aGameArrays[i].pValues;
|
|
|
|
|
|
|
|
svgm_vars[vcnt].flags = 0;
|
|
|
|
svgm_vars[vcnt].ptr = pValues;
|
2017-07-08 19:41:43 +00:00
|
|
|
|
|
|
|
if (aGameArrays[i].flags & GAMEARRAY_BITMAP)
|
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
svgm_vars[vcnt].size = (aGameArrays[i].size + 7) >> 3;
|
|
|
|
svgm_vars[vcnt].cnt = 1; // assumed constant throughout demo, i.e. no RESIZEARRAY
|
2017-07-08 19:41:43 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
svgm_vars[vcnt].size = pValues == NULL ? 0 : sizeof(aGameArrays[0].pValues[0]);
|
|
|
|
svgm_vars[vcnt].cnt = aGameArrays[i].size; // assumed constant throughout demo, i.e. no RESIZEARRAY
|
2017-07-08 19:41:43 +00:00
|
|
|
}
|
2018-10-25 23:31:45 +00:00
|
|
|
|
|
|
|
++vcnt;
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
svgm_vars[vcnt].flags = DS_END;
|
|
|
|
svgm_vars[vcnt].ptr = NULL;
|
|
|
|
svgm_vars[vcnt].size = 0;
|
|
|
|
svgm_vars[vcnt].cnt = 0;
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
2013-05-19 19:29:18 +00:00
|
|
|
#endif
|
2010-01-24 23:33:17 +00:00
|
|
|
|
|
|
|
void sv_freemem()
|
|
|
|
{
|
2015-02-11 05:22:23 +00:00
|
|
|
DO_FREE_AND_NULL(svsnapshot);
|
|
|
|
DO_FREE_AND_NULL(svinitsnap);
|
|
|
|
DO_FREE_AND_NULL(svdiff);
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
|
2014-05-30 00:02:19 +00:00
|
|
|
static void SV_AllocSnap(int32_t allocinit)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
|
|
|
sv_freemem();
|
|
|
|
|
2014-05-30 00:02:19 +00:00
|
|
|
svsnapshot = (uint8_t *)Xmalloc(svsnapsiz);
|
2010-01-24 23:33:17 +00:00
|
|
|
if (allocinit)
|
2014-05-30 00:02:19 +00:00
|
|
|
svinitsnap = (uint8_t *)Xmalloc(svsnapsiz);
|
2010-01-24 23:33:17 +00:00
|
|
|
svdiffsiz = svsnapsiz; // theoretically it's less than could be needed in the worst case, but practically it's overkill
|
2014-05-30 00:02:19 +00:00
|
|
|
svdiff = (uint8_t *)Xmalloc(svdiffsiz);
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
// make snapshot only if spot < 0 (demo)
|
2018-03-08 03:55:17 +00:00
|
|
|
int32_t sv_saveandmakesnapshot(FILE *fil, char const *name, int8_t spot, int8_t recdiffsp, int8_t diffcompress, int8_t synccompress, bool isAutoSave)
|
2011-12-25 15:34:06 +00:00
|
|
|
{
|
|
|
|
savehead_t h;
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
// set a few savegame system globals
|
2010-01-24 23:33:17 +00:00
|
|
|
savegame_comprthres = SV_DEFAULTCOMPRTHRES;
|
2011-12-25 15:34:06 +00:00
|
|
|
savegame_diffcompress = diffcompress;
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
// calculate total snapshot size
|
2013-05-19 19:29:18 +00:00
|
|
|
#if !defined LUNATIC
|
2010-01-24 23:33:17 +00:00
|
|
|
sv_makevarspec();
|
2016-01-11 05:05:38 +00:00
|
|
|
svsnapsiz = calcsz((const dataspec_t *)svgm_vars);
|
2013-05-19 19:29:18 +00:00
|
|
|
#else
|
|
|
|
svsnapsiz = 0;
|
|
|
|
#endif
|
2010-01-24 23:33:17 +00:00
|
|
|
svsnapsiz += calcsz(svgm_udnetw) + calcsz(svgm_secwsp) + calcsz(svgm_script) + calcsz(svgm_anmisc);
|
|
|
|
|
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
// create header
|
2018-03-08 03:54:58 +00:00
|
|
|
Bmemcpy(h.headerstr, "E32SAVEGAME", 11);
|
2011-12-25 15:34:06 +00:00
|
|
|
h.majorver = SV_MAJOR_VER;
|
|
|
|
h.minorver = SV_MINOR_VER;
|
2018-10-25 23:31:45 +00:00
|
|
|
h.ptrsize = sizeof(intptr_t);
|
|
|
|
|
2018-03-08 03:55:17 +00:00
|
|
|
if (isAutoSave)
|
2018-10-25 23:31:45 +00:00
|
|
|
h.ptrsize |= 1u << 7u;
|
|
|
|
|
|
|
|
h.bytever = BYTEVERSION;
|
|
|
|
h.userbytever = ud.userbytever;
|
|
|
|
h.scriptcrc = g_scriptcrc;
|
|
|
|
h.comprthres = savegame_comprthres;
|
|
|
|
h.recdiffsp = recdiffsp;
|
2011-12-25 15:34:06 +00:00
|
|
|
h.diffcompress = savegame_diffcompress;
|
|
|
|
h.synccompress = synccompress;
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
h.reccnt = 0;
|
2011-12-25 15:34:06 +00:00
|
|
|
h.snapsiz = svsnapsiz;
|
|
|
|
|
|
|
|
// the following is kinda redundant, but we save it here to be able to quickly fetch
|
|
|
|
// it in a savegame header read
|
2018-10-25 23:31:45 +00:00
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
h.numplayers = ud.multimode;
|
2018-10-25 23:31:45 +00:00
|
|
|
h.volnum = ud.volume_number;
|
|
|
|
h.levnum = ud.level_number;
|
|
|
|
h.skill = ud.player_skill;
|
2014-09-07 18:10:15 +00:00
|
|
|
|
2015-04-20 20:46:42 +00:00
|
|
|
const uint32_t BSZ = sizeof(h.boardfn);
|
|
|
|
EDUKE32_STATIC_ASSERT(BSZ == sizeof(currentboardfilename));
|
|
|
|
Bstrncpy(h.boardfn, currentboardfilename, BSZ);
|
|
|
|
|
2017-12-18 11:24:53 +00:00
|
|
|
if (spot >= 0)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2011-12-25 15:34:06 +00:00
|
|
|
// savegame
|
2017-12-18 11:24:53 +00:00
|
|
|
Bstrncpyz(h.savename, name, sizeof(h.savename));
|
2015-02-11 05:22:23 +00:00
|
|
|
#ifdef __ANDROID__
|
2016-08-27 01:42:01 +00:00
|
|
|
Bstrncpyz(h.volname, g_volumeNames[ud.volume_number], sizeof(h.volname));
|
|
|
|
Bstrncpyz(h.skillname, g_skillNames[ud.player_skill], sizeof(h.skillname));
|
2015-02-11 05:22:23 +00:00
|
|
|
#endif
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
2011-12-25 15:34:06 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
// demo
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
const time_t t = time(NULL);
|
|
|
|
struct tm * st;
|
2011-12-25 15:34:06 +00:00
|
|
|
|
2013-11-04 18:21:09 +00:00
|
|
|
Bstrncpyz(h.savename, "EDuke32 demo", sizeof(h.savename));
|
2011-12-25 15:34:06 +00:00
|
|
|
if (t>=0 && (st = localtime(&t)))
|
2013-11-04 18:21:09 +00:00
|
|
|
Bsnprintf(h.savename, sizeof(h.savename), "Demo %04d%02d%02d %s",
|
|
|
|
st->tm_year+1900, st->tm_mon+1, st->tm_mday, s_buildRev);
|
2011-12-25 15:34:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// write header
|
|
|
|
fwrite(&h, sizeof(savehead_t), 1, fil);
|
|
|
|
|
|
|
|
// for savegames, the file offset after the screenshot goes here;
|
|
|
|
// for demos, we keep it 0 to signify that we didn't save one
|
|
|
|
fwrite("\0\0\0\0", 4, 1, fil);
|
|
|
|
if (spot >= 0 && waloff[TILE_SAVESHOT])
|
|
|
|
{
|
|
|
|
int32_t ofs;
|
|
|
|
|
|
|
|
// write the screenshot compressed
|
2018-03-23 21:41:02 +00:00
|
|
|
dfwrite_LZ4((char *)waloff[TILE_SAVESHOT], 320, 200, fil);
|
2011-12-25 15:34:06 +00:00
|
|
|
|
|
|
|
// write the current file offset right after the header
|
|
|
|
ofs = ftell(fil);
|
|
|
|
fseek(fil, sizeof(savehead_t), SEEK_SET);
|
|
|
|
fwrite(&ofs, 4, 1, fil);
|
|
|
|
fseek(fil, ofs, SEEK_SET);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef DEBUGGINGAIDS
|
|
|
|
OSD_Printf("sv_saveandmakesnapshot: snapshot size: %d bytes.\n", svsnapsiz);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (spot >= 0)
|
|
|
|
{
|
|
|
|
// savegame
|
|
|
|
dosaveplayer2(fil, NULL);
|
2013-05-20 19:31:42 +00:00
|
|
|
#ifdef LUNATIC
|
|
|
|
if (!g_savedOK)
|
|
|
|
{
|
2013-06-30 20:38:35 +00:00
|
|
|
OSD_Printf("sv_saveandmakesnapshot: failed serializing Lunatic gamevar \"%s\".\n",
|
2013-05-20 19:31:42 +00:00
|
|
|
g_failedVarname);
|
|
|
|
g_failedVarname = NULL;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
#endif
|
2011-12-25 15:34:06 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// demo
|
2014-05-30 00:02:19 +00:00
|
|
|
SV_AllocSnap(0);
|
2011-12-25 15:34:06 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
uint8_t * const p = dosaveplayer2(fil, svsnapshot);
|
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
if (p != svsnapshot+svsnapsiz)
|
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
OSD_Printf("sv_saveandmakesnapshot: ptr-(snapshot end)=%d!\n", (int32_t)(p - (svsnapshot + svsnapsiz)));
|
2011-12-25 15:34:06 +00:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
2010-01-24 23:33:17 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
// if file is not an EDuke32 savegame/demo, h->headerstr will be all zeros
|
|
|
|
int32_t sv_loadheader(int32_t fil, int32_t spot, savehead_t *h)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2011-12-25 15:34:06 +00:00
|
|
|
int32_t havedemo = (spot < 0);
|
|
|
|
|
|
|
|
if (kread(fil, h, sizeof(savehead_t)) != sizeof(savehead_t))
|
|
|
|
{
|
|
|
|
OSD_Printf("%s %d header corrupt.\n", havedemo ? "Demo":"Savegame", havedemo ? -spot : spot);
|
|
|
|
Bmemset(h->headerstr, 0, sizeof(h->headerstr));
|
|
|
|
return -1;
|
|
|
|
}
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2018-03-08 03:54:58 +00:00
|
|
|
if (Bmemcmp(h->headerstr, "E32SAVEGAME", 11)
|
|
|
|
#if 1
|
|
|
|
&& Bmemcmp(h->headerstr, "EDuke32SAVE", 11)
|
|
|
|
#endif
|
|
|
|
)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2018-03-08 03:54:58 +00:00
|
|
|
char headerCstr[sizeof(h->headerstr) + 1];
|
|
|
|
Bmemcpy(headerCstr, h->headerstr, sizeof(h->headerstr));
|
|
|
|
headerCstr[sizeof(h->headerstr)] = '\0';
|
|
|
|
OSD_Printf("%s %d header reads \"%s\", expected \"E32SAVEGAME\".\n",
|
|
|
|
havedemo ? "Demo":"Savegame", havedemo ? -spot : spot, headerCstr);
|
2011-12-25 15:34:06 +00:00
|
|
|
Bmemset(h->headerstr, 0, sizeof(h->headerstr));
|
2017-12-01 06:19:04 +00:00
|
|
|
return -2;
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
|
2018-09-01 19:36:10 +00:00
|
|
|
if (h->majorver != SV_MAJOR_VER || h->minorver != SV_MINOR_VER || h->bytever != BYTEVERSION || h->userbytever != ud.userbytever || (apScript != NULL && h->scriptcrc != g_scriptcrc))
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2018-08-09 16:06:49 +00:00
|
|
|
#ifndef DEBUGGINGAIDS
|
2011-12-25 15:34:06 +00:00
|
|
|
if (havedemo)
|
2018-08-09 16:06:49 +00:00
|
|
|
#endif
|
|
|
|
OSD_Printf("Incompatible file version. Expected %d.%d.%d.%d.%0x, found %d.%d.%d.%d.%0x\n", SV_MAJOR_VER, SV_MINOR_VER, BYTEVERSION,
|
|
|
|
ud.userbytever, g_scriptcrc, h->majorver, h->minorver, h->bytever, h->userbytever, h->scriptcrc);
|
2017-12-01 06:19:04 +00:00
|
|
|
|
|
|
|
if (h->majorver == SV_MAJOR_VER && h->minorver == SV_MINOR_VER)
|
|
|
|
{
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Bmemset(h->headerstr, 0, sizeof(h->headerstr));
|
|
|
|
return -3;
|
|
|
|
}
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
2011-12-25 15:34:06 +00:00
|
|
|
|
2018-03-08 03:55:02 +00:00
|
|
|
if (h->getPtrSize() != sizeof(intptr_t))
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2018-08-09 16:06:49 +00:00
|
|
|
#ifndef DEBUGGINGAIDS
|
2011-12-25 15:34:06 +00:00
|
|
|
if (havedemo)
|
2018-08-09 16:06:49 +00:00
|
|
|
#endif
|
|
|
|
OSD_Printf("File incompatible. Expected pointer size %d, found %d\n",
|
2018-03-08 03:55:02 +00:00
|
|
|
(int32_t)sizeof(intptr_t), h->getPtrSize());
|
2017-12-01 06:19:04 +00:00
|
|
|
|
|
|
|
Bmemset(h->headerstr, 0, sizeof(h->headerstr));
|
|
|
|
return -4;
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
int32_t sv_loadsnapshot(int32_t fil, int32_t spot, savehead_t *h)
|
|
|
|
{
|
|
|
|
uint8_t *p;
|
|
|
|
int32_t i;
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
if (spot < 0)
|
|
|
|
{
|
|
|
|
// demo
|
|
|
|
i = sv_loadheader(fil, spot, h);
|
|
|
|
if (i)
|
|
|
|
return i;
|
|
|
|
|
|
|
|
// Used to be in doloadplayer2(), now redundant for savegames since
|
|
|
|
// we checked before. Multiplayer demos need still to be taken care of.
|
|
|
|
if (h->numplayers != numplayers)
|
|
|
|
return 9;
|
|
|
|
}
|
|
|
|
// else (if savegame), we just read the header and are now at offset sizeof(savehead_t)
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
#ifdef DEBUGGINGAIDS
|
|
|
|
OSD_Printf("sv_loadsnapshot: snapshot size: %d bytes.\n", h->snapsiz);
|
|
|
|
#endif
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
if (kread(fil, &i, 4) != 4)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2011-12-25 15:34:06 +00:00
|
|
|
OSD_Printf("sv_snapshot: couldn't read 4 bytes after header.\n");
|
|
|
|
return 7;
|
|
|
|
}
|
|
|
|
if (i > 0)
|
|
|
|
{
|
|
|
|
if (klseek(fil, i, SEEK_SET) != i)
|
|
|
|
{
|
|
|
|
OSD_Printf("sv_snapshot: failed skipping over the screenshot.\n");
|
|
|
|
return 8;
|
|
|
|
}
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
|
2011-12-26 14:37:34 +00:00
|
|
|
savegame_comprthres = h->comprthres;
|
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
if (spot >= 0)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2011-12-25 15:34:06 +00:00
|
|
|
// savegame
|
|
|
|
i = doloadplayer2(fil, NULL);
|
|
|
|
if (i)
|
|
|
|
{
|
|
|
|
OSD_Printf("sv_loadsnapshot: doloadplayer2() returned %d.\n", i);
|
|
|
|
return 5;
|
|
|
|
}
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
2011-12-25 15:34:06 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
// demo
|
|
|
|
savegame_diffcompress = h->diffcompress;
|
|
|
|
|
|
|
|
svsnapsiz = h->snapsiz;
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2014-05-30 00:02:19 +00:00
|
|
|
SV_AllocSnap(1);
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
p = svsnapshot;
|
|
|
|
i = doloadplayer2(fil, &p);
|
|
|
|
if (i)
|
|
|
|
{
|
|
|
|
OSD_Printf("sv_loadsnapshot: doloadplayer2() returned %d.\n", i);
|
|
|
|
sv_freemem();
|
|
|
|
return 5;
|
|
|
|
}
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
if (p != svsnapshot+svsnapsiz)
|
|
|
|
{
|
|
|
|
OSD_Printf("sv_loadsnapshot: internal error: p-(snapshot end)=%d!\n",
|
|
|
|
(int32_t)(p-(svsnapshot+svsnapsiz)));
|
|
|
|
sv_freemem();
|
|
|
|
return 6;
|
|
|
|
}
|
|
|
|
|
|
|
|
Bmemcpy(svinitsnap, svsnapshot, svsnapsiz);
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
postloadplayer((spot >= 0));
|
|
|
|
|
2010-01-24 23:33:17 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
|
2010-01-24 23:33:17 +00:00
|
|
|
uint32_t sv_writediff(FILE *fil)
|
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
uint8_t *p = svsnapshot;
|
|
|
|
uint8_t *d = svdiff;
|
2010-01-24 23:33:17 +00:00
|
|
|
|
|
|
|
cmpspecdata(svgm_udnetw, &p, &d);
|
|
|
|
cmpspecdata(svgm_secwsp, &p, &d);
|
|
|
|
cmpspecdata(svgm_script, &p, &d);
|
|
|
|
cmpspecdata(svgm_anmisc, &p, &d);
|
2013-05-19 19:29:18 +00:00
|
|
|
#if !defined LUNATIC
|
2016-01-11 05:05:38 +00:00
|
|
|
cmpspecdata((const dataspec_t *)svgm_vars, &p, &d);
|
2013-05-19 19:29:18 +00:00
|
|
|
#endif
|
2010-01-24 23:33:17 +00:00
|
|
|
|
|
|
|
if (p != svsnapshot+svsnapsiz)
|
|
|
|
OSD_Printf("sv_writediff: dump+siz=%p, p=%p!\n", svsnapshot+svsnapsiz, p);
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
uint32_t const diffsiz = d - svdiff;
|
2010-01-24 23:33:17 +00:00
|
|
|
|
|
|
|
fwrite("dIfF",4,1,fil);
|
|
|
|
fwrite(&diffsiz, sizeof(diffsiz), 1, fil);
|
2018-10-25 23:31:45 +00:00
|
|
|
|
2010-01-24 23:33:17 +00:00
|
|
|
if (savegame_diffcompress)
|
2018-03-23 21:41:02 +00:00
|
|
|
dfwrite_LZ4(svdiff, 1, diffsiz, fil); // cnt and sz swapped
|
2010-01-24 23:33:17 +00:00
|
|
|
else
|
|
|
|
fwrite(svdiff, 1, diffsiz, fil);
|
|
|
|
|
|
|
|
return diffsiz;
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t sv_readdiff(int32_t fil)
|
|
|
|
{
|
2010-02-23 18:13:46 +00:00
|
|
|
int32_t diffsiz;
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
if (kread(fil, &diffsiz, sizeof(uint32_t)) != sizeof(uint32_t))
|
2010-02-23 18:13:46 +00:00
|
|
|
return -1;
|
2018-10-25 23:31:45 +00:00
|
|
|
|
2010-01-24 23:33:17 +00:00
|
|
|
if (savegame_diffcompress)
|
|
|
|
{
|
2018-03-23 21:41:02 +00:00
|
|
|
if (kdfread_LZ4(svdiff, 1, diffsiz, fil) != diffsiz) // cnt and sz swapped
|
2010-01-24 23:33:17 +00:00
|
|
|
return -2;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (kread(fil, svdiff, diffsiz) != diffsiz)
|
|
|
|
return -2;
|
|
|
|
}
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
uint8_t *p = svsnapshot;
|
|
|
|
uint8_t *d = svdiff;
|
|
|
|
|
2010-02-23 18:13:46 +00:00
|
|
|
if (applydiff(svgm_udnetw, &p, &d)) return -3;
|
|
|
|
if (applydiff(svgm_secwsp, &p, &d)) return -4;
|
|
|
|
if (applydiff(svgm_script, &p, &d)) return -5;
|
|
|
|
if (applydiff(svgm_anmisc, &p, &d)) return -6;
|
2013-05-19 19:29:18 +00:00
|
|
|
#if !defined LUNATIC
|
2016-01-11 05:05:38 +00:00
|
|
|
if (applydiff((const dataspec_t *)svgm_vars, &p, &d)) return -7;
|
2013-05-19 19:29:18 +00:00
|
|
|
#endif
|
2010-02-23 18:13:46 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
int i = 0;
|
|
|
|
|
2010-02-23 18:13:46 +00:00
|
|
|
if (p!=svsnapshot+svsnapsiz)
|
|
|
|
i|=1;
|
|
|
|
if (d!=svdiff+diffsiz)
|
|
|
|
i|=2;
|
|
|
|
if (i)
|
|
|
|
OSD_Printf("sv_readdiff: p=%p, svsnapshot+svsnapsiz=%p; d=%p, svdiff+diffsiz=%p",
|
|
|
|
p, svsnapshot+svsnapsiz, d, svdiff+diffsiz);
|
|
|
|
return i;
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// SVGM data description
|
|
|
|
static void sv_postudload()
|
|
|
|
{
|
2011-12-25 15:34:06 +00:00
|
|
|
// Bmemcpy(&boardfilename[0], ¤tboardfilename[0], BMAX_PATH); // DON'T do this in demos!
|
|
|
|
#if 1
|
2018-10-25 23:31:45 +00:00
|
|
|
ud.m_level_number = ud.level_number;
|
|
|
|
ud.m_volume_number = ud.volume_number;
|
|
|
|
ud.m_player_skill = ud.player_skill;
|
|
|
|
ud.m_respawn_monsters = ud.respawn_monsters;
|
|
|
|
ud.m_respawn_items = ud.respawn_items;
|
2010-01-24 23:33:17 +00:00
|
|
|
ud.m_respawn_inventory = ud.respawn_inventory;
|
2018-10-25 23:31:45 +00:00
|
|
|
ud.m_monsters_off = ud.monsters_off;
|
|
|
|
ud.m_coop = ud.coop;
|
|
|
|
ud.m_marker = ud.marker;
|
|
|
|
ud.m_ffire = ud.ffire;
|
|
|
|
ud.m_noexits = ud.noexits;
|
2010-02-13 21:46:42 +00:00
|
|
|
#endif
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
//static int32_t lockclock_dummy;
|
|
|
|
|
2011-03-04 08:50:58 +00:00
|
|
|
#ifdef USE_OPENGL
|
2010-01-24 23:33:17 +00:00
|
|
|
static void sv_prespriteextsave()
|
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
for (int i=0; i<MAXSPRITES; i++)
|
2010-01-24 23:33:17 +00:00
|
|
|
if (spriteext[i].mdanimtims)
|
|
|
|
{
|
|
|
|
spriteext[i].mdanimtims -= mdtims;
|
|
|
|
if (spriteext[i].mdanimtims==0)
|
|
|
|
spriteext[i].mdanimtims++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
static void sv_postspriteext()
|
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
for (int i=0; i<MAXSPRITES; i++)
|
2010-01-24 23:33:17 +00:00
|
|
|
if (spriteext[i].mdanimtims)
|
|
|
|
spriteext[i].mdanimtims += mdtims;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2011-05-12 23:31:13 +00:00
|
|
|
#ifdef YAX_ENABLE
|
2012-03-11 17:37:08 +00:00
|
|
|
void sv_postyaxload(void)
|
2011-05-12 23:31:13 +00:00
|
|
|
{
|
|
|
|
yax_update(numyaxbunches>0 ? 2 : 1);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2010-01-24 23:33:17 +00:00
|
|
|
static void sv_postactordata()
|
|
|
|
{
|
2017-06-23 03:59:39 +00:00
|
|
|
#ifdef POLYMER
|
2018-10-25 23:31:45 +00:00
|
|
|
for (int i = 0; i < MAXSPRITES; i++)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2011-06-19 18:30:32 +00:00
|
|
|
actor[i].lightptr = NULL;
|
|
|
|
actor[i].lightId = -1;
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
2017-06-23 03:59:39 +00:00
|
|
|
#endif
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void sv_preanimateptrsave()
|
|
|
|
{
|
2016-08-27 01:42:01 +00:00
|
|
|
G_Util_PtrToIdx(g_animatePtr, g_animateCnt, sector, P2I_FWD);
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
static void sv_postanimateptr()
|
|
|
|
{
|
2016-08-27 01:42:01 +00:00
|
|
|
G_Util_PtrToIdx(g_animatePtr, g_animateCnt, sector, P2I_BACK);
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
static void sv_prequote()
|
|
|
|
{
|
|
|
|
if (!savegame_quotes)
|
2014-07-06 00:10:46 +00:00
|
|
|
{
|
|
|
|
void *ptr = Xcalloc(MAXQUOTES, MAXQUOTELEN);
|
|
|
|
savegame_quotes = (char(*)[MAXQUOTELEN])ptr;
|
|
|
|
}
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
static void sv_quotesave()
|
|
|
|
{
|
|
|
|
Bmemset(savegame_quotedef, 0, sizeof(savegame_quotedef));
|
2018-10-25 23:31:45 +00:00
|
|
|
for (int i = 0; i < MAXQUOTES; i++)
|
2016-08-27 01:40:35 +00:00
|
|
|
if (apStrings[i])
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
|
|
|
savegame_quotedef[i>>3] |= 1<<(i&7);
|
2016-08-27 01:40:35 +00:00
|
|
|
Bmemcpy(savegame_quotes[i], apStrings[i], MAXQUOTELEN);
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
static void sv_quoteload()
|
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
for (int i = 0; i < MAXQUOTES; i++)
|
|
|
|
if (savegame_quotedef[i>>3] & (1<<(i&7)))
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2013-01-01 15:24:25 +00:00
|
|
|
C_AllocQuote(i);
|
2016-08-27 01:40:35 +00:00
|
|
|
Bmemcpy(apStrings[i], savegame_quotes[i], MAXQUOTELEN);
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
}
|
2015-03-24 00:40:55 +00:00
|
|
|
|
|
|
|
static void sv_preprojectilesave()
|
|
|
|
{
|
2018-09-28 04:27:48 +00:00
|
|
|
savegame_projectilecnt = 0;
|
2018-09-29 01:54:52 +00:00
|
|
|
Bmemset(savegame_projectiles, 0, sizeof(savegame_projectiles));
|
2018-09-28 04:27:48 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
for (int i = 0; i < MAXTILES; i++)
|
2018-09-28 04:27:48 +00:00
|
|
|
if (g_tile[i].proj)
|
|
|
|
savegame_projectilecnt++;
|
|
|
|
|
2018-09-29 01:54:52 +00:00
|
|
|
if (savegame_projectilecnt > 0)
|
2018-09-28 04:27:48 +00:00
|
|
|
savegame_projectiledata = (projectile_t *) Xrealloc(savegame_projectiledata, sizeof(projectile_t) * savegame_projectilecnt);
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
for (int i = 0, cnt = 0; i < MAXTILES; i++)
|
2015-03-24 00:40:55 +00:00
|
|
|
{
|
|
|
|
if (g_tile[i].proj)
|
|
|
|
{
|
2018-09-28 04:27:48 +00:00
|
|
|
savegame_projectiles[i>>3] |= 1<<(i&7);
|
|
|
|
Bmemcpy(&savegame_projectiledata[cnt++], g_tile[i].proj, sizeof(projectile_t));
|
2015-03-24 00:40:55 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void sv_postprojectilesave()
|
|
|
|
{
|
2018-09-29 01:54:52 +00:00
|
|
|
DO_FREE_AND_NULL(savegame_projectiledata);
|
2015-03-24 00:40:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void sv_preprojectileload()
|
|
|
|
{
|
2018-09-29 01:54:52 +00:00
|
|
|
savegame_projectilecnt = 0;
|
2015-05-16 11:56:49 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
for (int i = 0; i < MAXTILES; i++)
|
2018-09-28 04:27:48 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
if (savegame_projectiles[i>>3] & (1<<(i&7)))
|
2018-09-29 01:54:52 +00:00
|
|
|
savegame_projectilecnt++;
|
2018-09-28 04:27:48 +00:00
|
|
|
}
|
|
|
|
|
2018-09-29 01:54:52 +00:00
|
|
|
if (savegame_projectilecnt > 0)
|
|
|
|
savegame_projectiledata = (projectile_t *) Xrealloc(savegame_projectiledata, sizeof(projectile_t) * savegame_projectilecnt);
|
2015-03-24 00:40:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void sv_postprojectileload()
|
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
for (int i = 0, cnt = 0; i < MAXTILES; i++)
|
2015-03-24 00:40:55 +00:00
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
if (savegame_projectiles[i>>3] & (1<<(i&7)))
|
2015-03-24 00:40:55 +00:00
|
|
|
{
|
2015-05-16 11:56:49 +00:00
|
|
|
C_AllocProjectile(i);
|
2018-09-28 04:27:48 +00:00
|
|
|
Bmemcpy(g_tile[i].proj, &savegame_projectiledata[cnt++], sizeof(projectile_t));
|
2015-03-24 00:40:55 +00:00
|
|
|
}
|
|
|
|
}
|
2018-09-29 01:54:52 +00:00
|
|
|
|
|
|
|
DO_FREE_AND_NULL(savegame_projectiledata);
|
2015-03-24 00:40:55 +00:00
|
|
|
}
|
|
|
|
|
2010-01-24 23:33:17 +00:00
|
|
|
static void sv_prequoteredef()
|
|
|
|
{
|
|
|
|
// "+1" needed for dfwrite which doesn't handle the src==NULL && cnt==0 case
|
2016-08-27 01:40:35 +00:00
|
|
|
void *ptr = Xcalloc(g_numXStrings+1, MAXQUOTELEN);
|
2018-10-25 23:31:45 +00:00
|
|
|
savegame_quoteredefs = (decltype(savegame_quoteredefs))ptr;
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
static void sv_quoteredefsave()
|
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
for (int i = 0; i < g_numXStrings; i++)
|
2016-08-27 01:40:35 +00:00
|
|
|
if (apXStrings[i])
|
|
|
|
Bmemcpy(savegame_quoteredefs[i], apXStrings[i], MAXQUOTELEN);
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
static void sv_quoteredefload()
|
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
for (int i = 0; i < g_numXStrings; i++)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2016-08-27 01:40:35 +00:00
|
|
|
if (!apXStrings[i])
|
|
|
|
apXStrings[i] = (char *)Xcalloc(1,MAXQUOTELEN);
|
|
|
|
Bmemcpy(apXStrings[i], savegame_quoteredefs[i], MAXQUOTELEN);
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
static void sv_postquoteredef()
|
|
|
|
{
|
|
|
|
Bfree(savegame_quoteredefs), savegame_quoteredefs=NULL;
|
|
|
|
}
|
|
|
|
static void sv_restsave()
|
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
uint8_t * mem = savegame_restdata;
|
2010-01-24 23:33:17 +00:00
|
|
|
DukePlayer_t dummy_ps;
|
|
|
|
|
|
|
|
Bmemset(&dummy_ps, 0, sizeof(DukePlayer_t));
|
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
#define CPDAT(ptr,sz) do { Bmemcpy(mem, ptr, sz), mem+=sz ; } while (0)
|
|
|
|
for (int i = 0; i < MAXPLAYERS; i++)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
|
|
|
CPDAT(g_player[i].user_name, 32);
|
|
|
|
CPDAT(&g_player[i].pcolor, sizeof(g_player[0].pcolor));
|
|
|
|
CPDAT(&g_player[i].pteam, sizeof(g_player[0].pteam));
|
|
|
|
CPDAT(&g_player[i].frags[0], sizeof(g_player[0].frags));
|
2018-10-25 23:31:45 +00:00
|
|
|
CPDAT(g_player[i].ps ? g_player[i].ps : &dummy_ps, sizeof(DukePlayer_t));
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
2011-12-25 15:34:06 +00:00
|
|
|
|
2018-10-25 23:31:45 +00:00
|
|
|
Bassert((savegame_restdata + SVARDATALEN) - mem == 0);
|
2010-02-23 18:13:46 +00:00
|
|
|
#undef CPDAT
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
static void sv_restload()
|
|
|
|
{
|
2018-10-25 23:31:45 +00:00
|
|
|
uint8_t * mem = savegame_restdata;
|
2010-01-24 23:33:17 +00:00
|
|
|
DukePlayer_t dummy_ps;
|
|
|
|
|
2010-02-23 18:13:46 +00:00
|
|
|
#define CPDAT(ptr,sz) Bmemcpy(ptr, mem, sz), mem+=sz
|
2018-10-25 23:31:45 +00:00
|
|
|
for (int i = 0; i < MAXPLAYERS; i++)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
|
|
|
CPDAT(g_player[i].user_name, 32);
|
|
|
|
CPDAT(&g_player[i].pcolor, sizeof(g_player[0].pcolor));
|
|
|
|
CPDAT(&g_player[i].pteam, sizeof(g_player[0].pteam));
|
|
|
|
CPDAT(&g_player[i].frags[0], sizeof(g_player[0].frags));
|
2018-10-25 23:31:45 +00:00
|
|
|
CPDAT(g_player[i].ps ? g_player[i].ps : &dummy_ps, sizeof(DukePlayer_t));
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
2010-02-23 18:13:46 +00:00
|
|
|
#undef CPDAT
|
2018-02-25 01:18:29 +00:00
|
|
|
|
2018-10-07 05:19:50 +00:00
|
|
|
if (g_player[myconnectindex].ps)
|
|
|
|
g_player[myconnectindex].ps->auto_aim = ud.config.AutoAim;
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
|
2011-12-25 15:32:00 +00:00
|
|
|
#ifdef DEBUGGINGAIDS
|
2011-12-28 20:33:54 +00:00
|
|
|
# define PRINTSIZE(name) do { if (mem) OSD_Printf(name ": %d\n", (int32_t)(mem-tmem)); \
|
2018-04-12 21:02:51 +00:00
|
|
|
OSD_Printf(name ": %d ms\n", timerGetTicks()-t); t=timerGetTicks(); tmem=mem; } while (0)
|
2011-12-25 15:32:00 +00:00
|
|
|
#else
|
2011-12-28 20:33:54 +00:00
|
|
|
# define PRINTSIZE(name) do { } while (0)
|
2011-12-25 15:32:00 +00:00
|
|
|
#endif
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2013-05-20 19:31:42 +00:00
|
|
|
#ifdef LUNATIC
|
2013-10-27 21:12:22 +00:00
|
|
|
// <levelnum>: if we're not serializing for a mapstate, -1
|
|
|
|
// otherwise, the linearized level number
|
|
|
|
LUNATIC_CB const char *(*El_SerializeGamevars)(int32_t *slenptr, int32_t levelnum);
|
2013-05-20 19:31:42 +00:00
|
|
|
#endif
|
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
static uint8_t *dosaveplayer2(FILE *fil, uint8_t *mem)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2011-12-28 20:33:54 +00:00
|
|
|
#ifdef DEBUGGINGAIDS
|
2010-01-24 23:33:17 +00:00
|
|
|
uint8_t *tmem = mem;
|
2018-04-12 21:02:51 +00:00
|
|
|
int32_t t=timerGetTicks();
|
2011-12-28 20:33:54 +00:00
|
|
|
#endif
|
2010-01-24 23:33:17 +00:00
|
|
|
mem=writespecdata(svgm_udnetw, fil, mem); // user settings, players & net
|
2011-12-25 15:32:00 +00:00
|
|
|
PRINTSIZE("ud");
|
2010-01-24 23:33:17 +00:00
|
|
|
mem=writespecdata(svgm_secwsp, fil, mem); // sector, wall, sprite
|
2011-12-25 15:32:00 +00:00
|
|
|
PRINTSIZE("sws");
|
2013-06-20 18:31:47 +00:00
|
|
|
#ifdef LUNATIC
|
2013-05-20 19:31:42 +00:00
|
|
|
{
|
2013-06-20 18:31:47 +00:00
|
|
|
// Serialize Lunatic gamevars. When loading, the restoration code must
|
|
|
|
// be present before Lua state creation in svgm_script, so save it
|
|
|
|
// right before, too.
|
2013-05-20 19:31:42 +00:00
|
|
|
int32_t slen, slen_ext;
|
2013-10-27 21:12:22 +00:00
|
|
|
const char *svcode = El_SerializeGamevars(&slen, -1);
|
2013-05-20 19:31:42 +00:00
|
|
|
|
|
|
|
if (slen < 0)
|
|
|
|
{
|
|
|
|
// Serialization failed.
|
|
|
|
g_savedOK = 0;
|
|
|
|
g_failedVarname = svcode;
|
|
|
|
return mem;
|
|
|
|
}
|
|
|
|
|
|
|
|
fwrite("\0\1LunaGVAR\3\4", 12, 1, fil);
|
|
|
|
slen_ext = B_LITTLE32(slen);
|
|
|
|
fwrite(&slen_ext, sizeof(slen_ext), 1, fil);
|
2018-03-23 21:41:02 +00:00
|
|
|
dfwrite_LZ4(svcode, 1, slen, fil); // cnt and sz swapped
|
2013-05-20 19:31:42 +00:00
|
|
|
|
|
|
|
g_savedOK = 1;
|
|
|
|
}
|
2013-05-19 19:29:18 +00:00
|
|
|
#endif
|
2013-06-20 18:31:47 +00:00
|
|
|
mem=writespecdata(svgm_script, fil, mem); // script
|
|
|
|
PRINTSIZE("script");
|
|
|
|
mem=writespecdata(svgm_anmisc, fil, mem); // animates, quotes & misc.
|
|
|
|
PRINTSIZE("animisc");
|
|
|
|
|
|
|
|
#if !defined LUNATIC
|
2016-08-27 01:40:35 +00:00
|
|
|
Gv_WriteSave(fil); // gamevars
|
2016-01-11 05:05:38 +00:00
|
|
|
mem=writespecdata((const dataspec_t *)svgm_vars, 0, mem);
|
2013-06-20 18:31:47 +00:00
|
|
|
PRINTSIZE("vars");
|
|
|
|
#endif
|
2010-01-24 23:33:17 +00:00
|
|
|
|
|
|
|
return mem;
|
|
|
|
}
|
|
|
|
|
2013-06-20 18:31:47 +00:00
|
|
|
#ifdef LUNATIC
|
|
|
|
char *g_elSavecode = NULL;
|
|
|
|
|
|
|
|
static int32_t El_ReadSaveCode(int32_t fil)
|
|
|
|
{
|
|
|
|
// Read Lua code to restore gamevar values from the savegame.
|
|
|
|
// It will be run from Lua with its state creation later on.
|
|
|
|
|
|
|
|
char header[12];
|
|
|
|
int32_t slen;
|
|
|
|
|
|
|
|
if (kread(fil, header, 12) != 12)
|
|
|
|
{
|
|
|
|
OSD_Printf("doloadplayer2: failed reading Lunatic gamevar header.\n");
|
|
|
|
return -100;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Bmemcmp(header, "\0\1LunaGVAR\3\4", 12))
|
|
|
|
{
|
|
|
|
OSD_Printf("doloadplayer2: Lunatic gamevar header doesn't match.\n");
|
|
|
|
return -101;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (kread(fil, &slen, sizeof(slen)) != sizeof(slen))
|
|
|
|
{
|
|
|
|
OSD_Printf("doloadplayer2: failed reading Lunatic gamevar string size.\n");
|
|
|
|
return -102;
|
|
|
|
}
|
|
|
|
|
|
|
|
slen = B_LITTLE32(slen);
|
|
|
|
if (slen < 0)
|
|
|
|
{
|
|
|
|
OSD_Printf("doloadplayer2: invalid Lunatic gamevar string size %d.\n", slen);
|
|
|
|
return -103;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (slen > 0)
|
|
|
|
{
|
2014-05-30 00:02:19 +00:00
|
|
|
char *svcode = (char *)Xmalloc(slen+1);
|
2013-06-20 18:31:47 +00:00
|
|
|
|
2018-03-23 21:41:02 +00:00
|
|
|
if (kdfread_LZ4(svcode, 1, slen, fil) != slen) // cnt and sz swapped
|
2013-06-20 18:31:47 +00:00
|
|
|
{
|
|
|
|
OSD_Printf("doloadplayer2: failed reading Lunatic gamevar restoration code.\n");
|
|
|
|
Bfree(svcode);
|
|
|
|
return -104;
|
|
|
|
}
|
|
|
|
|
|
|
|
svcode[slen] = 0;
|
|
|
|
g_elSavecode = svcode;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void El_FreeSaveCode(void)
|
|
|
|
{
|
|
|
|
// Free Lunatic gamevar savegame restoration Lua code.
|
|
|
|
Bfree(g_elSavecode);
|
|
|
|
g_elSavecode = NULL;
|
|
|
|
}
|
|
|
|
#endif
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
static int32_t doloadplayer2(int32_t fil, uint8_t **memptr)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2011-12-28 20:33:54 +00:00
|
|
|
uint8_t *mem = memptr ? *memptr : NULL;
|
|
|
|
#ifdef DEBUGGINGAIDS
|
|
|
|
uint8_t *tmem=mem;
|
2018-04-12 21:02:51 +00:00
|
|
|
int32_t t=timerGetTicks();
|
2011-12-28 20:33:54 +00:00
|
|
|
#endif
|
2011-12-25 15:34:06 +00:00
|
|
|
if (readspecdata(svgm_udnetw, fil, &mem)) return -2;
|
2011-12-25 15:32:00 +00:00
|
|
|
PRINTSIZE("ud");
|
2010-01-24 23:33:17 +00:00
|
|
|
if (readspecdata(svgm_secwsp, fil, &mem)) return -4;
|
2011-12-25 15:32:00 +00:00
|
|
|
PRINTSIZE("sws");
|
2013-06-20 18:31:47 +00:00
|
|
|
#ifdef LUNATIC
|
|
|
|
{
|
|
|
|
int32_t ret = El_ReadSaveCode(fil);
|
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
#endif
|
2010-01-24 23:33:17 +00:00
|
|
|
if (readspecdata(svgm_script, fil, &mem)) return -5;
|
2011-12-25 15:32:00 +00:00
|
|
|
PRINTSIZE("script");
|
2010-01-24 23:33:17 +00:00
|
|
|
if (readspecdata(svgm_anmisc, fil, &mem)) return -6;
|
2011-12-25 15:32:00 +00:00
|
|
|
PRINTSIZE("animisc");
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2013-05-19 19:29:18 +00:00
|
|
|
#if !defined LUNATIC
|
2017-10-31 02:09:25 +00:00
|
|
|
int i;
|
|
|
|
|
|
|
|
if ((i = Gv_ReadSave(fil))) return i;
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2011-12-21 18:42:12 +00:00
|
|
|
if (mem)
|
2011-12-25 15:32:00 +00:00
|
|
|
{
|
2011-12-25 15:34:06 +00:00
|
|
|
int32_t i;
|
|
|
|
|
2011-12-25 15:32:00 +00:00
|
|
|
sv_makevarspec();
|
2011-12-21 18:42:12 +00:00
|
|
|
for (i=1; svgm_vars[i].flags!=DS_END; i++)
|
|
|
|
{
|
|
|
|
Bmemcpy(mem, svgm_vars[i].ptr, svgm_vars[i].size*svgm_vars[i].cnt); // careful! works because there are no DS_DYNAMIC's!
|
|
|
|
mem += svgm_vars[i].size*svgm_vars[i].cnt;
|
|
|
|
}
|
2011-12-25 15:32:00 +00:00
|
|
|
}
|
2011-12-28 20:33:54 +00:00
|
|
|
PRINTSIZE("vars");
|
2013-05-19 19:29:18 +00:00
|
|
|
#endif
|
2010-01-24 23:33:17 +00:00
|
|
|
|
|
|
|
if (memptr)
|
|
|
|
*memptr = mem;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t sv_updatestate(int32_t frominit)
|
|
|
|
{
|
|
|
|
uint8_t *p = svsnapshot, *pbeg=p;
|
|
|
|
|
|
|
|
if (frominit)
|
|
|
|
Bmemcpy(svsnapshot, svinitsnap, svsnapsiz);
|
|
|
|
|
|
|
|
if (readspecdata(svgm_udnetw, -1, &p)) return -2;
|
|
|
|
if (readspecdata(svgm_secwsp, -1, &p)) return -4;
|
|
|
|
if (readspecdata(svgm_script, -1, &p)) return -5;
|
|
|
|
if (readspecdata(svgm_anmisc, -1, &p)) return -6;
|
|
|
|
|
2013-05-19 19:29:18 +00:00
|
|
|
#if !defined LUNATIC
|
2016-01-11 05:05:38 +00:00
|
|
|
if (readspecdata((const dataspec_t *)svgm_vars, -1, &p)) return -8;
|
2013-05-19 19:29:18 +00:00
|
|
|
#endif
|
2010-01-24 23:33:17 +00:00
|
|
|
|
|
|
|
if (p != pbeg+svsnapsiz)
|
2011-12-25 15:32:00 +00:00
|
|
|
{
|
2010-01-24 23:33:17 +00:00
|
|
|
OSD_Printf("sv_updatestate: ptr-(snapshot end)=%d\n", (int32_t)(p-(pbeg+svsnapsiz)));
|
2011-12-25 15:32:00 +00:00
|
|
|
return -9;
|
|
|
|
}
|
2010-01-24 23:33:17 +00:00
|
|
|
|
|
|
|
if (frominit)
|
2011-12-25 15:34:06 +00:00
|
|
|
postloadplayer(0);
|
|
|
|
#ifdef POLYMER
|
2018-04-12 21:03:12 +00:00
|
|
|
if (videoGetRenderMode() == REND_POLYMER)
|
2011-12-25 15:34:06 +00:00
|
|
|
polymer_resetlights(); // must do it after polymer_loadboard() !!!
|
|
|
|
#endif
|
2010-01-24 23:33:17 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
static void postloadplayer(int32_t savegamep)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
|
|
|
int32_t i;
|
2011-12-25 15:34:06 +00:00
|
|
|
|
2010-01-24 23:33:17 +00:00
|
|
|
//1
|
|
|
|
if (g_player[myconnectindex].ps->over_shoulder_on != 0)
|
|
|
|
{
|
2013-01-19 18:28:32 +00:00
|
|
|
CAMERADIST = 0;
|
|
|
|
CAMERACLOCK = 0;
|
2010-01-24 23:33:17 +00:00
|
|
|
g_player[myconnectindex].ps->over_shoulder_on = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
//2
|
2011-12-25 15:34:06 +00:00
|
|
|
screenpeek = myconnectindex;
|
|
|
|
|
|
|
|
//2.5
|
|
|
|
if (savegamep)
|
|
|
|
{
|
2018-01-29 11:13:49 +00:00
|
|
|
int32_t musicIdx = (ud.music_episode*MAXLEVELS) + ud.music_level;
|
2014-09-07 18:10:15 +00:00
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
Bmemset(gotpic, 0, sizeof(gotpic));
|
|
|
|
S_ClearSoundLocks();
|
|
|
|
G_CacheMapData();
|
|
|
|
|
2018-01-29 11:13:49 +00:00
|
|
|
if (boardfilename[0] != 0 && ud.level_number == 7 && ud.volume_number == 0 && ud.music_level == 7 && ud.music_episode == 0)
|
2011-12-25 15:34:06 +00:00
|
|
|
{
|
|
|
|
char levname[BMAX_PATH];
|
|
|
|
G_SetupFilenameBasedMusic(levname, boardfilename, ud.level_number);
|
|
|
|
}
|
|
|
|
|
2018-01-29 11:13:49 +00:00
|
|
|
if (g_mapInfo[musicIdx].musicfn != NULL && (musicIdx != g_musicIndex || different_user_map))
|
2011-12-25 15:34:06 +00:00
|
|
|
{
|
2018-01-29 11:13:49 +00:00
|
|
|
ud.music_episode = g_musicIndex / MAXLEVELS;
|
|
|
|
ud.music_level = g_musicIndex % MAXLEVELS;
|
|
|
|
S_PlayLevelMusicOrNothing(musicIdx);
|
|
|
|
}
|
2018-01-29 11:14:08 +00:00
|
|
|
else
|
|
|
|
S_ContinueLevelMusic();
|
2011-12-25 15:34:06 +00:00
|
|
|
|
2018-01-29 11:13:49 +00:00
|
|
|
if (ud.config.MusicToggle)
|
2011-12-25 15:34:06 +00:00
|
|
|
S_PauseMusic(0);
|
|
|
|
|
|
|
|
g_player[myconnectindex].ps->gm = MODE_GAME;
|
|
|
|
ud.recstat = 0;
|
|
|
|
|
|
|
|
if (g_player[myconnectindex].ps->jetpack_on)
|
|
|
|
A_PlaySound(DUKE_JETPACK_IDLE, g_player[myconnectindex].ps->i);
|
|
|
|
}
|
|
|
|
|
2010-01-24 23:33:17 +00:00
|
|
|
//3
|
|
|
|
P_UpdateScreenPal(g_player[myconnectindex].ps);
|
2012-02-02 17:34:16 +00:00
|
|
|
g_restorePalette = -1;
|
2011-12-25 15:34:06 +00:00
|
|
|
|
|
|
|
//3.5
|
|
|
|
if (savegamep)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2012-11-13 10:50:06 +00:00
|
|
|
for (SPRITES_OF(STAT_FX, i))
|
2012-10-14 22:13:49 +00:00
|
|
|
if (sprite[i].picnum == MUSICANDSFX)
|
|
|
|
{
|
2016-08-27 01:40:35 +00:00
|
|
|
T2(i) = ud.config.SoundToggle;
|
|
|
|
T1(i) = 0;
|
2012-10-14 22:13:49 +00:00
|
|
|
}
|
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
G_UpdateScreenArea();
|
|
|
|
FX_SetReverb(0);
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
2011-12-25 15:34:06 +00:00
|
|
|
|
|
|
|
//4
|
|
|
|
if (savegamep)
|
2010-01-24 23:33:17 +00:00
|
|
|
{
|
2012-01-05 20:48:37 +00:00
|
|
|
if (ud.lockout)
|
2011-12-25 15:34:06 +00:00
|
|
|
{
|
2016-08-27 01:42:01 +00:00
|
|
|
for (i=0; i<g_animWallCnt; i++)
|
2012-01-28 14:38:23 +00:00
|
|
|
switch (DYNAMICTILEMAP(wall[animwall[i].wallnum].picnum))
|
2011-12-25 15:34:06 +00:00
|
|
|
{
|
|
|
|
case FEMPIC1__STATIC:
|
|
|
|
wall[animwall[i].wallnum].picnum = BLANKSCREEN;
|
|
|
|
break;
|
|
|
|
case FEMPIC2__STATIC:
|
|
|
|
case FEMPIC3__STATIC:
|
|
|
|
wall[animwall[i].wallnum].picnum = SCREENBREAK6;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2012-01-05 20:48:37 +00:00
|
|
|
#if 0
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for (i=0; i<g_numAnimWalls; i++)
|
|
|
|
if (wall[animwall[i].wallnum].extra >= 0)
|
|
|
|
wall[animwall[i].wallnum].picnum = wall[animwall[i].wallnum].extra;
|
|
|
|
}
|
|
|
|
#endif
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//5
|
2011-07-29 22:07:28 +00:00
|
|
|
G_ResetInterpolations();
|
2010-01-24 23:33:17 +00:00
|
|
|
|
|
|
|
//6
|
|
|
|
g_showShareware = 0;
|
2011-12-25 15:34:06 +00:00
|
|
|
if (savegamep)
|
|
|
|
everyothertime = 0;
|
2011-12-21 18:42:12 +00:00
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
//7
|
2010-01-24 23:33:17 +00:00
|
|
|
for (i=0; i<MAXPLAYERS; i++)
|
2011-12-21 18:42:12 +00:00
|
|
|
g_player[i].playerquitflag = 1;
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2011-12-25 15:34:06 +00:00
|
|
|
// ----------
|
|
|
|
|
|
|
|
//7.5
|
|
|
|
if (savegamep)
|
|
|
|
{
|
|
|
|
ready2send = 1;
|
|
|
|
G_ClearFIFO();
|
|
|
|
Net_WaitForServer();
|
|
|
|
}
|
|
|
|
|
|
|
|
//8
|
|
|
|
// if (savegamep) ?
|
2013-07-04 19:38:46 +00:00
|
|
|
#ifdef LUNATIC
|
|
|
|
G_ResetTimers(1);
|
|
|
|
#else
|
|
|
|
G_ResetTimers(0);
|
|
|
|
#endif
|
2010-01-24 23:33:17 +00:00
|
|
|
|
|
|
|
#ifdef POLYMER
|
2011-12-25 15:34:06 +00:00
|
|
|
//9
|
2018-04-12 21:03:12 +00:00
|
|
|
if (videoGetRenderMode() == REND_POLYMER)
|
2010-01-25 01:11:34 +00:00
|
|
|
polymer_loadboard();
|
2010-01-24 23:33:17 +00:00
|
|
|
|
2018-04-12 21:03:12 +00:00
|
|
|
// this light pointer nulling needs to be outside the videoGetRenderMode check
|
2011-12-25 15:34:06 +00:00
|
|
|
// because we might be loading the savegame using another renderer but
|
|
|
|
// change to Polymer later
|
|
|
|
for (i=0; i<MAXSPRITES; i++)
|
|
|
|
{
|
|
|
|
actor[i].lightptr = NULL;
|
|
|
|
actor[i].lightId = -1;
|
|
|
|
}
|
2017-06-23 03:59:39 +00:00
|
|
|
#endif
|
2010-01-24 23:33:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
////////// END GENERIC SAVING/LOADING SYSTEM //////////
|
2015-02-11 05:22:23 +00:00
|
|
|
|