yquake2remaster/src/server/sv_save.c

448 lines
8.8 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"
2012-07-21 08:06:07 +00:00
void CM_ReadPortalState(fileHandle_t f);
/*
* 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);
2012-07-21 08:06:07 +00:00
remove(name);
2012-07-21 08:06:07 +00:00
Com_sprintf(name, sizeof(name), "%s/save/%s/game.ssv",
FS_Gamedir(), savename);
2012-07-21 08:06:07 +00:00
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)
{
2012-07-21 08:06:07 +00:00
remove(s);
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)
{
2012-07-21 08:06:07 +00:00
remove(s);
s = Sys_FindNext(0, 0);
}
Sys_FindClose();
}
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];
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);
2012-07-21 08:06:07 +00:00
Com_sprintf(name, sizeof(name), "%s/save/current/%s.sav",
FS_Gamedir(), sv.name);
ge->WriteLevel(name);
}
void
2012-07-21 08:06:07 +00:00
SV_ReadLevelFile(void)
{
2012-07-21 08:06:07 +00:00
char name[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);
2012-07-21 08:06:07 +00:00
Com_sprintf(name, sizeof(name), "%s/save/current/%s.sav",
FS_Gamedir(), sv.name);
ge->ReadLevel(name);
}
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 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 */
2012-07-21 08:06:07 +00:00
Com_sprintf(name, sizeof(name), "%s/save/current/game.ssv", FS_Gamedir());
ge->WriteGame(name, autosave);
}
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 comment[32];
char mapcmd[MAX_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 */
2012-07-21 08:06:07 +00:00
Com_sprintf(name, sizeof(name), "%s/save/current/game.ssv", FS_Gamedir());
ge->ReadGame(name);
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;
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;
}
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 */
2012-07-21 08:06:07 +00:00
SV_Map(false, svs.mapcmd, true);
}
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