yquake2remaster/src/server/sv_save.c

528 lines
11 KiB
C
Raw Normal View History

/*
* Copyright (C) 1997-2001 Id Software, Inc.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or (at
* your option) any later version.
*
* 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
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
* 02111-1307, USA.
*
* =======================================================================
*
* Serverside savegame code.
*
* =======================================================================
2012-04-29 13:57:33 +00:00
*/
#include "header/server.h"
/*
* Delete save/<XXX>/
*/
void
2012-07-21 08:06:07 +00:00
SV_WipeSavegame(char *savename)
{
2012-07-21 08:06:07 +00:00
char name[MAX_OSPATH];
char *s;
2012-07-21 08:06:07 +00:00
Com_DPrintf("SV_WipeSaveGame(%s)\n", savename);
2012-07-21 08:06:07 +00:00
Com_sprintf(name, sizeof(name), "%s/save/%s/server.ssv",
FS_Gamedir(), savename);
Sys_Remove(name);
2012-07-21 08:06:07 +00:00
Com_sprintf(name, sizeof(name), "%s/save/%s/game.ssv",
FS_Gamedir(), savename);
Sys_Remove(name);
2012-07-21 08:06:07 +00:00
Com_sprintf(name, sizeof(name), "%s/save/%s/*.sav", FS_Gamedir(), savename);
s = Sys_FindFirst(name, 0, 0);
while (s)
{
Sys_Remove(s);
2012-07-21 08:06:07 +00:00
s = Sys_FindNext(0, 0);
}
Sys_FindClose();
2012-07-21 08:06:07 +00:00
Com_sprintf(name, sizeof(name), "%s/save/%s/*.sv2", FS_Gamedir(), savename);
s = Sys_FindFirst(name, 0, 0);
2012-07-21 08:06:07 +00:00
while (s)
{
Sys_Remove(s);
2012-07-21 08:06:07 +00:00
s = Sys_FindNext(0, 0);
}
Sys_FindClose();
}
2024-01-01 15:36:53 +00:00
static void
2012-07-21 08:06:07 +00:00
CopyFile(char *src, char *dst)
{
2012-07-21 08:06:07 +00:00
FILE *f1, *f2;
size_t l;
2012-07-21 08:06:07 +00:00
byte buffer[65536];
2012-07-21 08:06:07 +00:00
Com_DPrintf("CopyFile (%s, %s)\n", src, dst);
f1 = Q_fopen(src, "rb");
2012-07-21 08:06:07 +00:00
if (!f1)
{
return;
}
f2 = Q_fopen(dst, "wb");
2012-07-21 08:06:07 +00:00
if (!f2)
{
2012-07-21 08:06:07 +00:00
fclose(f1);
return;
}
2012-07-21 08:06:07 +00:00
while (1)
{
2012-07-21 08:06:07 +00:00
l = fread(buffer, 1, sizeof(buffer), f1);
2012-07-21 08:06:07 +00:00
if (!l)
{
break;
}
2012-07-21 08:06:07 +00:00
fwrite(buffer, 1, l, f2);
}
2012-07-21 08:06:07 +00:00
fclose(f1);
fclose(f2);
}
void
2012-07-21 08:06:07 +00:00
SV_CopySaveGame(char *src, char *dst)
{
2012-07-21 08:06:07 +00:00
char name[MAX_OSPATH], name2[MAX_OSPATH];
size_t l, len;
2012-07-21 08:06:07 +00:00
char *found;
2012-07-21 08:06:07 +00:00
Com_DPrintf("SV_CopySaveGame(%s, %s)\n", src, dst);
2012-07-21 08:06:07 +00:00
SV_WipeSavegame(dst);
/* copy the savegame over */
2012-07-21 08:06:07 +00:00
Com_sprintf(name, sizeof(name), "%s/save/%s/server.ssv", FS_Gamedir(), src);
Com_sprintf(name2, sizeof(name2), "%s/save/%s/server.ssv", FS_Gamedir(), dst);
FS_CreatePath(name2);
CopyFile(name, name2);
2012-07-21 08:06:07 +00:00
Com_sprintf(name, sizeof(name), "%s/save/%s/game.ssv", FS_Gamedir(), src);
Com_sprintf(name2, sizeof(name2), "%s/save/%s/game.ssv", FS_Gamedir(), dst);
CopyFile(name, name2);
2012-07-21 08:06:07 +00:00
Com_sprintf(name, sizeof(name), "%s/save/%s/", FS_Gamedir(), src);
len = strlen(name);
Com_sprintf(name, sizeof(name), "%s/save/%s/*.sav", FS_Gamedir(), src);
found = Sys_FindFirst(name, 0, 0);
2012-07-21 08:06:07 +00:00
while (found)
{
2012-07-21 08:06:07 +00:00
strcpy(name + len, found + len);
2012-07-21 08:06:07 +00:00
Com_sprintf(name2, sizeof(name2), "%s/save/%s/%s",
FS_Gamedir(), dst, found + len);
CopyFile(name, name2);
/* change sav to sv2 */
2012-07-21 08:06:07 +00:00
l = strlen(name);
strcpy(name + l - 3, "sv2");
l = strlen(name2);
strcpy(name2 + l - 3, "sv2");
CopyFile(name, name2);
2012-07-21 08:06:07 +00:00
found = Sys_FindNext(0, 0);
}
Sys_FindClose();
}
void
2012-07-21 08:06:07 +00:00
SV_WriteLevelFile(void)
{
2012-07-21 08:06:07 +00:00
char name[MAX_OSPATH];
char workdir[MAX_OSPATH];
2012-07-21 08:06:07 +00:00
FILE *f;
2012-07-21 08:06:07 +00:00
Com_DPrintf("SV_WriteLevelFile()\n");
2012-07-21 08:06:07 +00:00
Com_sprintf(name, sizeof(name), "%s/save/current/%s.sv2",
FS_Gamedir(), sv.name);
f = Q_fopen(name, "wb");
2012-07-21 08:06:07 +00:00
if (!f)
{
2012-07-21 08:06:07 +00:00
Com_Printf("Failed to open %s\n", name);
return;
}
2012-07-21 08:06:07 +00:00
fwrite(sv.configstrings, sizeof(sv.configstrings), 1, f);
CM_WritePortalState(f);
fclose(f);
Com_sprintf(name, sizeof(name), "%s/save/current", FS_Gamedir());
Sys_GetWorkDir(workdir, sizeof(workdir));
Sys_Mkdir(name);
if (!Sys_SetWorkDir(name))
{
Com_Printf("Couldn't change to %s\n", name);
Sys_SetWorkDir(workdir);
return;
}
Com_sprintf(name, sizeof(name), "%s.sav", sv.name);
2012-07-21 08:06:07 +00:00
ge->WriteLevel(name);
Sys_SetWorkDir(workdir);
}
void
2012-07-21 08:06:07 +00:00
SV_ReadLevelFile(void)
{
2012-07-21 08:06:07 +00:00
char name[MAX_OSPATH];
char workdir[MAX_OSPATH];
fileHandle_t f;
2012-07-21 08:06:07 +00:00
Com_DPrintf("SV_ReadLevelFile()\n");
2012-07-21 08:06:07 +00:00
Com_sprintf(name, sizeof(name), "save/current/%s.sv2", sv.name);
FS_FOpenFile(name, &f, true);
2012-07-21 08:06:07 +00:00
if (!f)
{
2012-07-21 08:06:07 +00:00
Com_Printf("Failed to open %s\n", name);
return;
}
2012-07-21 08:06:07 +00:00
FS_Read(sv.configstrings, sizeof(sv.configstrings), f);
CM_ReadPortalState(f);
FS_FCloseFile(f);
Com_sprintf(name, sizeof(name), "%s/save/current", FS_Gamedir());
Sys_GetWorkDir(workdir, sizeof(workdir));
if (!Sys_SetWorkDir(name))
{
Com_Printf("Couldn't change to %s\n", name);
Sys_SetWorkDir(workdir);
return;
}
Com_sprintf(name, sizeof(name), "%s.sav", sv.name);
2012-07-21 08:06:07 +00:00
ge->ReadLevel(name);
Sys_SetWorkDir(workdir);
}
void
2012-07-21 08:06:07 +00:00
SV_WriteServerFile(qboolean autosave)
{
2012-07-21 08:06:07 +00:00
FILE *f;
cvar_t *var;
char name[MAX_OSPATH], string[128];
char workdir[MAX_OSPATH];
2012-07-21 08:06:07 +00:00
char comment[32];
time_t aclock;
2012-07-21 08:06:07 +00:00
struct tm *newtime;
2012-07-21 08:06:07 +00:00
Com_DPrintf("SV_WriteServerFile(%s)\n", autosave ? "true" : "false");
2012-07-21 08:06:07 +00:00
Com_sprintf(name, sizeof(name), "%s/save/current/server.ssv", FS_Gamedir());
f = Q_fopen(name, "wb");
2012-07-21 08:06:07 +00:00
if (!f)
{
2012-07-21 08:06:07 +00:00
Com_Printf("Couldn't write %s\n", name);
return;
}
/* write the comment field */
2012-07-21 08:06:07 +00:00
memset(comment, 0, sizeof(comment));
2012-07-21 08:06:07 +00:00
if (!autosave)
{
2012-07-21 08:06:07 +00:00
time(&aclock);
newtime = localtime(&aclock);
Com_sprintf(comment, sizeof(comment), "%2i:%i%i %2i/%2i ",
newtime->tm_hour, newtime->tm_min / 10,
newtime->tm_min % 10, newtime->tm_mon + 1,
newtime->tm_mday);
Q_strlcat(comment, sv.configstrings[CS_NAME], sizeof(comment));
}
else
{
/* autosaved */
2012-07-21 08:06:07 +00:00
Com_sprintf(comment, sizeof(comment), "ENTERING %s",
sv.configstrings[CS_NAME]);
}
2012-07-21 08:06:07 +00:00
fwrite(comment, 1, sizeof(comment), f);
/* write the mapcmd */
2012-07-21 08:06:07 +00:00
fwrite(svs.mapcmd, 1, sizeof(svs.mapcmd), f);
2012-04-29 13:57:33 +00:00
/* write all CVAR_LATCH cvars
2012-07-21 08:06:07 +00:00
these will be things like coop,
skill, deathmatch, etc */
for (var = cvar_vars; var; var = var->next)
{
char cvarname[LATCH_CVAR_SAVELENGTH] = {0};
2012-07-21 08:06:07 +00:00
if (!(var->flags & CVAR_LATCH))
{
continue;
}
if ((strlen(var->name) >= sizeof(cvarname) - 1) ||
2012-07-21 08:06:07 +00:00
(strlen(var->string) >= sizeof(string) - 1))
{
2012-07-21 08:06:07 +00:00
Com_Printf("Cvar too long: %s = %s\n", var->name, var->string);
continue;
}
2012-07-21 08:06:07 +00:00
memset(string, 0, sizeof(string));
strcpy(cvarname, var->name);
2012-07-21 08:06:07 +00:00
strcpy(string, var->string);
fwrite(cvarname, 1, sizeof(cvarname), f);
2012-07-21 08:06:07 +00:00
fwrite(string, 1, sizeof(string), f);
}
2012-07-21 08:06:07 +00:00
fclose(f);
/* write game state */
Com_sprintf(name, sizeof(name), "%s/save/current", FS_Gamedir());
Sys_GetWorkDir(workdir, sizeof(workdir));
Sys_Mkdir(name);
if (!Sys_SetWorkDir(name))
{
Com_Printf("Couldn't change to %s\n", name);
Sys_SetWorkDir(workdir);
return;
}
ge->WriteGame("game.ssv", autosave);
Sys_SetWorkDir(workdir);
}
2024-01-01 15:36:53 +00:00
static void
2012-07-21 08:06:07 +00:00
SV_ReadServerFile(void)
{
fileHandle_t f;
2012-07-21 08:06:07 +00:00
char name[MAX_OSPATH], string[128];
char workdir[MAX_OSPATH];
2012-07-21 08:06:07 +00:00
char comment[32];
char mapcmd[MAX_SAVE_TOKEN_CHARS];
2012-07-21 08:06:07 +00:00
Com_DPrintf("SV_ReadServerFile()\n");
2012-07-21 08:06:07 +00:00
Com_sprintf(name, sizeof(name), "save/current/server.ssv");
FS_FOpenFile(name, &f, true);
2012-07-21 08:06:07 +00:00
if (!f)
{
2012-07-21 08:06:07 +00:00
Com_Printf("Couldn't read %s\n", name);
return;
}
/* read the comment field */
2012-07-21 08:06:07 +00:00
FS_Read(comment, sizeof(comment), f);
/* read the mapcmd */
2012-07-21 08:06:07 +00:00
FS_Read(mapcmd, sizeof(mapcmd), f);
2012-04-29 13:57:33 +00:00
/* read all CVAR_LATCH cvars
these will be things like
2012-07-21 08:06:07 +00:00
coop, skill, deathmatch, etc */
while (1)
{
char cvarname[LATCH_CVAR_SAVELENGTH] = {0};
if (!FS_FRead(cvarname, 1, sizeof(cvarname), f))
{
break;
}
2012-07-21 08:06:07 +00:00
FS_Read(string, sizeof(string), f);
Com_DPrintf("Set %s = %s\n", cvarname, string);
Cvar_ForceSet(cvarname, string);
}
2012-07-21 08:06:07 +00:00
FS_FCloseFile(f);
/* start a new game fresh with new cvars */
SV_InitGame();
2012-07-21 08:06:07 +00:00
strcpy(svs.mapcmd, mapcmd);
/* read game state */
Com_sprintf(name, sizeof(name), "%s/save/current", FS_Gamedir());
Sys_GetWorkDir(workdir, sizeof(workdir));
if (!Sys_SetWorkDir(name))
{
Com_Printf("Couldn't change to %s\n", name);
Sys_SetWorkDir(workdir);
return;
}
ge->ReadGame("game.ssv");
/* While loading a savegame the global edict arrays is free()ed
and newly malloc()ed to reset all entity states. When the game
puts the first client into the server it sends it's entity
state to us, so as long as there's only one client (the game
is running in single player mode) everything's okay. But when
there're more clients (the game is running in coop mode) the
entity states if all clients >1 are dangeling. hack around
that by reconnecting them here. */
cvar_t *coop = Cvar_Get("coop", "0", CVAR_LATCH);
if (coop->value)
{
for (int i = 0; i < maxclients->value; i++)
{
edict_t *ent = EDICT_NUM(i + 1);
svs.clients[i].edict = ent;
}
}
Sys_SetWorkDir(workdir);
2012-04-29 13:57:33 +00:00
}
void
2012-07-21 08:06:07 +00:00
SV_Loadgame_f(void)
{
2012-07-21 08:06:07 +00:00
char name[MAX_OSPATH];
FILE *f;
char *dir;
qboolean isautosave;
2012-07-21 08:06:07 +00:00
if (Cmd_Argc() != 2)
{
2012-07-21 08:06:07 +00:00
Com_Printf("USAGE: loadgame <directory>\n");
return;
}
2012-07-21 08:06:07 +00:00
Com_Printf("Loading game...\n");
2012-07-21 08:06:07 +00:00
dir = Cmd_Argv(1);
2012-07-21 08:06:07 +00:00
if (strstr(dir, "..") || strstr(dir, "/") || strstr(dir, "\\"))
{
2012-07-21 08:06:07 +00:00
Com_Printf("Bad savedir.\n");
}
/* make sure the server.ssv file exists */
2012-07-21 08:06:07 +00:00
Com_sprintf(name, sizeof(name), "%s/save/%s/server.ssv",
FS_Gamedir(), Cmd_Argv(1));
f = Q_fopen(name, "rb");
2012-07-21 08:06:07 +00:00
if (!f)
{
2012-07-21 08:06:07 +00:00
Com_Printf("No such savegame: %s\n", name);
return;
}
Com_Printf("Savegame: %s\n", Cmd_Argv(1));
if (strcmp(Cmd_Argv(1), "save0") == 0 ||
strcmp(Cmd_Argv(1), "current") == 0)
{
isautosave = true;
}
else
{
isautosave = false;
}
2012-07-21 08:06:07 +00:00
fclose(f);
2012-07-21 08:06:07 +00:00
SV_CopySaveGame(Cmd_Argv(1), "current");
SV_ReadServerFile();
/* go to the map */
sv.state = ss_dead; /* don't save current level when changing */
SV_Map(false, svs.mapcmd, true, isautosave);
}
void
2012-07-21 08:06:07 +00:00
SV_Savegame_f(void)
{
2012-07-21 08:06:07 +00:00
char *dir;
2012-07-21 08:06:07 +00:00
if (sv.state != ss_game)
{
2012-07-21 08:06:07 +00:00
Com_Printf("You must be in a game to save.\n");
return;
}
2012-07-21 08:06:07 +00:00
if (Cmd_Argc() != 2)
{
2012-07-21 08:06:07 +00:00
Com_Printf("USAGE: savegame <directory>\n");
return;
}
2012-07-21 08:06:07 +00:00
if (Cvar_VariableValue("deathmatch"))
{
2012-07-21 08:06:07 +00:00
Com_Printf("Can't savegame in a deathmatch\n");
return;
}
2012-07-21 08:06:07 +00:00
if (!strcmp(Cmd_Argv(1), "current"))
{
2012-07-21 08:06:07 +00:00
Com_Printf("Can't save to 'current'\n");
return;
}
2012-07-21 08:06:07 +00:00
if ((maxclients->value == 1) &&
(svs.clients[0].edict->client->ps.stats[STAT_HEALTH] <= 0))
{
2012-07-21 08:06:07 +00:00
Com_Printf("\nCan't savegame while dead!\n");
return;
}
2012-07-21 08:06:07 +00:00
dir = Cmd_Argv(1);
2012-07-21 08:06:07 +00:00
if (strstr(dir, "..") || strstr(dir, "/") || strstr(dir, "\\"))
{
2012-07-21 08:06:07 +00:00
Com_Printf("Bad savedir.\n");
}
2012-07-21 08:06:07 +00:00
Com_Printf("Saving game...\n");
2012-04-29 13:57:33 +00:00
/* archive current level, including all client edicts.
when the level is reloaded, they will be shells awaiting
a connecting client */
SV_WriteLevelFile();
/* save server state */
2012-07-21 08:06:07 +00:00
SV_WriteServerFile(false);
/* copy it off */
2012-07-21 08:06:07 +00:00
SV_CopySaveGame("current", dir);
2012-07-21 08:06:07 +00:00
Com_Printf("Done.\n");
}
2012-07-21 08:06:07 +00:00