summaryrefslogtreecommitdiff
path: root/src/qcommon/files.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/qcommon/files.cpp')
-rw-r--r--src/qcommon/files.cpp3986
1 files changed, 3986 insertions, 0 deletions
diff --git a/src/qcommon/files.cpp b/src/qcommon/files.cpp
new file mode 100644
index 0000000..68b72b3
--- /dev/null
+++ b/src/qcommon/files.cpp
@@ -0,0 +1,3986 @@
+/*
+ Copyright (C) 2016 Victor Roemer (wtfbbqhax), <victor@badsec.org>.
+ Copyright (C) 2000-2013 Darklegion Development
+ Copyright (C) 1999-2005 Id Software, Inc.
+ Copyright (C) 2015-2019 GrangerHub
+
+ This file is part of Tremulous.
+
+ Tremulous 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 3 of the License,
+ or (at your option) any later version.
+
+ Tremulous 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 Tremulous; if not, see <https://www.gnu.org/licenses/>
+
+*/
+
+#include "files.h"
+
+#ifdef _WIN32
+#include <io.h>
+#include <windows.h>
+#endif
+
+#include <cctype>
+#include <cstdarg>
+#include <cstdint>
+#include <cstdio>
+#include <cstdlib>
+#include <cstring>
+#include <string>
+
+#include "cmd.h"
+#include "cvar.h"
+#include "md4.h"
+#include "q_platform.h"
+#include "q_shared.h"
+#include "qcommon.h"
+#include "unzip.h"
+#include "vm.h"
+
+#ifndef DEDICATED
+#include "client/cl_rest.h"
+#endif
+#include "sys/sys_shared.h"
+
+using namespace std;
+
+#define MAX_ZPATH 256
+#define MAX_SEARCH_PATHS 4096
+#define MAX_FILEHASH_SIZE 1024
+
+static bool FS_IsDemoExt(const char *filename);
+static bool FS_IsExt(const char *filename, const char *ext, int namelen);
+
+struct fileInPack_t {
+ char* name;
+ unsigned long pos; // file info position in zip
+ unsigned long len; // uncompressed file size
+ fileInPack_t* next;
+};
+
+struct pack_t {
+ char pakPathname[MAX_OSPATH]; // /tremulous/baseq3
+ char pakFilename[MAX_OSPATH]; // /tremulous/base/pak0.pk3
+ char pakBasename[MAX_OSPATH]; // pak0
+ char pakGamename[MAX_OSPATH]; // base
+ unzFile handle; // handle to zip file
+ int checksum; // regular checksum
+ int pure_checksum; // checksum for pure
+ int numfiles; // number of files in pk3
+ int referenced; // referenced file flags
+ int hashSize; // hash table size (power of 2)
+ fileInPack_t **hashTable; // hash table
+ fileInPack_t *buildBuffer; // buffer with the filenames etc.
+ // some multiprotocol stuff
+ bool onlyPrimary;
+ bool onlyAlternate;
+ pack_t *primaryVersion;
+
+ // member functions
+ inline fileInPack_t* find(string filename);
+ inline bool is_pure();
+};
+
+struct directory_t {
+ char path[MAX_OSPATH];
+ char fullpath[MAX_OSPATH]; // /tremulous/base
+ char gamedir[MAX_OSPATH]; // base
+};
+
+struct searchpath_t {
+ pack_t *pack; // only one of pack / dir will be non nullptr
+ directory_t *dir;
+ searchpath_t *next;
+};
+
+static char fs_gamedir[MAX_OSPATH]; // this will be a single file name with no separators
+static cvar_t *fs_debug;
+static cvar_t *fs_homepath;
+
+static cvar_t *fs_basepath;
+static cvar_t *fs_basegame;
+#ifdef __APPLE__
+static cvar_t *fs_apppath; // Also search the .app bundle for .pk3 files
+#endif
+static cvar_t *fs_gamedirvar;
+
+static searchpath_t *fs_searchpaths;
+static int fs_readCount; // total bytes read
+static int fs_loadCount; // total files read
+static int fs_loadStack; // total files in memory
+static int fs_packFiles = 0; // total number of files in packs
+
+static int fs_checksumFeed;
+
+union qfile_gut {
+ FILE *o;
+ unzFile z;
+};
+
+struct qfile_ut {
+ qfile_gut file;
+ bool unique;
+};
+
+struct fileHandleData_t {
+ qfile_ut handleFiles;
+ bool handleSync;
+ int fileSize;
+ int zipFilePos;
+ int zipFileLen;
+ bool zipFile;
+ char name[MAX_ZPATH];
+
+ void close();
+};
+
+static fileHandleData_t fsh[MAX_FILE_HANDLES];
+
+// TTimo - https://zerowing.idsoftware.com/bugzilla/show_bug.cgi?id=540
+// wether we did a reorder on the current search path when joining the server
+
+static bool fs_reordered;
+
+// never load anything from pk3 files that are not present at the server when pure
+
+static int fs_numServerPaks = 0;
+static int fs_serverPaks[MAX_SEARCH_PATHS]; // checksums
+static char *fs_serverPakNames[MAX_SEARCH_PATHS]; // pk3 names
+
+// only used for autodownload, to make sure the client has at least
+// all the pk3 files that are referenced at the server side
+
+static int fs_numServerReferencedPaks;
+static int fs_serverReferencedPaks[MAX_SEARCH_PATHS]; // checksums
+static char *fs_serverReferencedPakNames[MAX_SEARCH_PATHS]; // pk3 names
+
+// last valid game folder used
+char lastValidBase[MAX_OSPATH];
+char lastValidGame[MAX_OSPATH];
+
+#ifdef FS_MISSING
+FILE *missingFiles = nullptr;
+#endif
+
+/*
+==============
+FS_Initialized
+==============
+*/
+
+bool FS_Initialized(void) { return fs_searchpaths ? true : false; }
+
+/*
+=================
+pack_t::is_pure()
+
+FIXME: also use hashed file names
+=================
+*/
+inline bool pack_t::is_pure()
+{
+ if (fs_numServerPaks)
+ {
+ for (int i = 0; i < fs_numServerPaks; i++)
+ if (checksum == fs_serverPaks[i])
+ return true;
+
+ return false;
+ }
+ return true;
+}
+
+/*
+=================
+FS_LoadStack
+return load stack
+=================
+*/
+int FS_LoadStack(void) { return fs_loadStack; }
+
+inline fileInPack_t* pack_t::find(string filename)
+{
+ long hash = 0;
+ auto fn = filename.c_str();
+ for (long i = 0; fn[i]; i++)
+ {
+ long c = tolower(fn[i]);
+ if (c == '.')
+ break; // FIXME probably a bad idea
+
+ if (c == '\\')
+ c = '/';
+ hash += c * (i + 119);
+ }
+ hash = (hash ^ (hash >> 10) ^ (hash >> 20));
+ hash &= (hashSize - 1);
+
+ for (auto file = hashTable[hash]; file; file = file->next)
+ {
+ if (FS_FilenameCompare(file->name, fn) == false)
+ return file;
+ }
+ return nullptr;
+}
+/*
+================
+return a hash value for the filename
+================
+*/
+static long FS_HashFileName(const char *fname, int hashSize)
+{
+ int i;
+ long hash;
+ char letter;
+
+ hash = 0;
+ i = 0;
+ while (fname[i] != '\0')
+ {
+ letter = tolower(fname[i]);
+ if (letter == '.') break; // don't include extension
+ if (letter == '\\') letter = '/'; // damn path names
+ if (letter == PATH_SEP) letter = '/'; // damn path names
+ hash += (long)(letter) * (i + 119);
+ i++;
+ }
+ hash = (hash ^ (hash >> 10) ^ (hash >> 20));
+ hash &= (hashSize - 1);
+ return hash;
+}
+
+static fileHandle_t FS_HandleForFile(void)
+{
+ for (int i = 1; i < MAX_FILE_HANDLES; i++)
+ {
+ if (fsh[i].handleFiles.file.o == nullptr)
+ {
+ return i;
+ }
+ }
+ Com_Error(ERR_DROP, "FS_HandleForFile: none free");
+ return 0;
+}
+
+static FILE *FS_FileForHandle(fileHandle_t f)
+{
+ if (f < 1 || f >= MAX_FILE_HANDLES)
+ {
+ Com_Error(ERR_DROP, "FS_FileForHandle: out of range");
+ }
+
+ if (fsh[f].zipFile == true)
+ {
+ Com_Error(ERR_DROP, "FS_FileForHandle: can't get FILE on zip file");
+ }
+
+ if (!fsh[f].handleFiles.file.o)
+ {
+ Com_Error(ERR_DROP, "FS_FileForHandle: nullptr");
+ }
+
+ return fsh[f].handleFiles.file.o;
+}
+
+void FS_ForceFlush(fileHandle_t f)
+{
+ FILE *file = FS_FileForHandle(f);
+ setvbuf(file, nullptr, _IONBF, 0);
+}
+
+/*
+================
+FS_fplength
+================
+*/
+
+long FS_fplength(FILE *h)
+{
+ long pos = ftell(h);
+ fseek(h, 0, SEEK_END);
+
+ long end = ftell(h);
+ fseek(h, pos, SEEK_SET);
+
+ return end;
+}
+
+/*
+================
+FS_filelength
+
+If this is called on a non-unique FILE (from a pak file),
+it will return the size of the pak file, not the expected
+size of the file.
+================
+*/
+long FS_filelength(fileHandle_t f)
+{
+ FILE *h = FS_FileForHandle(f);
+ if (h == nullptr) return -1;
+
+ return FS_fplength(h);
+}
+
+/*
+====================
+FS_ReplaceSeparators
+
+Fix things up differently for win/unix/mac
+====================
+*/
+void FS_ReplaceSeparators(char *path)
+{
+ bool lastCharWasSep = false;
+
+ for (char *s = path; *s; s++)
+ {
+ if (*s == '/' || *s == '\\')
+ {
+ if (!lastCharWasSep)
+ {
+ *s = PATH_SEP;
+ lastCharWasSep = true;
+ }
+ else
+ {
+ memmove(s, s + 1, strlen(s));
+ }
+ }
+ else
+ {
+ lastCharWasSep = false;
+ }
+ }
+}
+
+/*
+===================
+FS_BuildOSPath
+
+Qpath may have either forward or backwards slashes
+===================
+*/
+char *FS_BuildOSPath(const char *base, const char *game, const char *qpath)
+{
+ char temp[MAX_OSPATH];
+ // "FIXME FS_BuildOSPath() returns static buffer with function scope"
+
+ // This code will alternate between 2 different buffers-
+ // XXX 3 or more calls to FS_BuildOSPath in a row are not safe.
+ static char ospath[2][MAX_OSPATH];
+ static bool toggle;
+
+ toggle = !toggle; // flip-flop to allow two returns without clash
+
+ if (!game || !game[0])
+ {
+ game = fs_gamedir;
+ }
+
+ Com_sprintf(temp, sizeof(temp), "/%s/%s", game, qpath);
+ FS_ReplaceSeparators(temp);
+ Com_sprintf(ospath[toggle], sizeof(ospath[0]), "%s%s", base, temp);
+
+ Com_DPrintf(S_COLOR_GREEN "%s: returning " S_COLOR_RED "%s\n",
+ __FUNCTION__, ospath[toggle]);
+
+ return ospath[toggle];
+}
+
+/*
+============
+FS_OpenWithDefault
+
+Wrapper for Sys_OpenWithDefault()
+============
+*/
+static bool FS_OpenWithDefault( const char *path )
+{
+ if( Sys_OpenWithDefault( path ) )
+ {
+ // minimize the client's window
+ Cmd_ExecuteString( "minimize" );
+ return true;
+ }
+
+ return false;
+}
+
+/*
+============
+FS_BrowseHomepath
+
+Opens the homepath in the default file manager
+============
+*/
+bool FS_BrowseHomepath( void )
+{
+ const char *homePath = Sys_DefaultHomePath( );
+
+ if (!homePath || !homePath[0])
+ {
+ homePath = fs_basepath->string;
+ }
+
+ if( FS_OpenWithDefault( homePath ) )
+ return true;
+
+ Com_Printf( S_COLOR_RED "FS_BrowseHomepath: failed to open the homepath with the default file manager.\n" S_COLOR_WHITE );
+ return false;
+}
+
+/*
+============
+FS_OpenBaseGamePath
+
+Opens the given path for the
+base game in the default file manager
+============
+*/
+bool FS_OpenBaseGamePath( const char *baseGamePath )
+{
+ const char *homePath = Sys_DefaultHomePath( );
+ const char *path;
+
+ if (!homePath || !homePath[0])
+ {
+ homePath = fs_basepath->string;
+ }
+
+ path = FS_BuildOSPath( homePath, fs_basegame->string, baseGamePath);
+
+ if( FS_OpenWithDefault( path ) )
+ return true;
+
+ Com_Printf( S_COLOR_RED "FS_BrowseHomepath: failed to open the homepath with the default file manager.\n" S_COLOR_WHITE );
+ return false;
+}
+
+/*
+============
+FS_CreatePath
+
+Creates any directories needed to store the given filename
+============
+*/
+bool FS_CreatePath(const char *OSPath)
+{
+ // make absolutely sure that it can't back up the path
+ // FIXME: is c: allowed???
+ if (strstr(OSPath, "..") || strstr(OSPath, "::"))
+ {
+ Com_Printf("WARNING: refusing to create relative path \"%s\"\n", OSPath);
+ return true;
+ }
+
+ char path[MAX_OSPATH];
+ Q_strncpyz(path, OSPath, sizeof(path));
+ FS_ReplaceSeparators(path);
+
+ // Skip creation of the root directory as it will always be there
+ char *ofs = strchr(path, PATH_SEP);
+ if (ofs != nullptr)
+ {
+ ofs++;
+ }
+
+ for (; ofs != nullptr && *ofs; ofs++)
+ {
+ if (*ofs == PATH_SEP)
+ {
+ // create the directory
+ *ofs = 0;
+ if (!Sys_Mkdir(path))
+ {
+ Com_Error(ERR_FATAL, "FS_CreatePath: failed to create path \"%s\"", path);
+ }
+ *ofs = PATH_SEP;
+ }
+ }
+
+ return false;
+}
+
+/*
+=================
+FS_CheckFilenameIsMutable
+
+ERR_FATAL if trying to maniuplate a file with the platform library, QVM, or pk3 extension
+=================
+ */
+static void FS_CheckFilenameIsMutable(const char *filename, const char *function)
+{
+ // Check if the filename ends with the library, QVM, or pk3 extension
+ if (Sys_DllExtension(filename) || COM_CompareExtension(filename, ".qvm") ||
+ COM_CompareExtension(filename, ".lua") || COM_CompareExtension(filename, ".pk3"))
+ {
+ Com_Error(ERR_FATAL, "%s: Not allowed to manipulate '%s' due to %s extension",
+ function, filename, COM_GetExtension(filename));
+ }
+}
+
+/*
+===========
+FS_Remove
+
+===========
+*/
+void FS_Remove(const char *osPath)
+{
+ FS_CheckFilenameIsMutable(osPath, __FUNCTION__);
+// RB begin
+#if defined(_WIN32)
+ ::DeleteFile(osPath);
+#else
+ remove(osPath);
+#endif
+}
+
+/*
+===========
+FS_HomeRemove
+
+===========
+*/
+void FS_HomeRemove(const char *homePath)
+{
+ FS_CheckFilenameIsMutable(homePath, __FUNCTION__);
+ FS_Remove(FS_BuildOSPath(fs_homepath->string, fs_gamedir, homePath));
+}
+
+#if 0
+bool FS_RemoveDir(const char* relativePath)
+{
+ bool success = true;
+ success = Sys_Rmdir(FS_BuildOSPath(fs_homepath->string, fs_gamedir, relativePath));
+ return success;
+}
+#endif
+
+/*
+================
+FS_FileInPathExists
+
+Tests if path and file exists
+================
+*/
+bool FS_FileInPathExists(const char *testpath)
+{
+ FILE *filep;
+
+ filep = Sys_FOpen(testpath, "rb");
+
+ if (filep)
+ {
+ fclose(filep);
+ return true;
+ }
+
+ return false;
+}
+
+/*
+================
+FS_FileExists
+
+Tests if the file exists in the current gamedir, this DOES NOT
+search the paths. This is to determine if opening a file to write
+(which always goes into the current gamedir) will cause any overwrites.
+NOTE TTimo: this goes with FS_FOpenFileWrite for opening the file afterwards
+================
+*/
+bool FS_FileExists(const char *file)
+{
+ return FS_FileInPathExists(FS_BuildOSPath(fs_homepath->string, fs_gamedir, file));
+}
+
+/*
+================
+FS_SV_FileExists
+
+Tests if the file exists
+================
+*/
+bool FS_SV_FileExists(const char *file)
+{
+ char *testpath = FS_BuildOSPath(fs_homepath->string, file, "");
+ testpath[strlen(testpath) - 1] = '\0';
+
+ return FS_FileInPathExists(testpath);
+}
+
+/*
+===========
+FS_SV_FOpenFileWrite
+
+===========
+*/
+fileHandle_t FS_SV_FOpenFileWrite(const char *filename)
+{
+ if (!fs_searchpaths)
+ {
+ Com_Error(ERR_FATAL, "Filesystem call made without initialization");
+ }
+
+ char *ospath = FS_BuildOSPath(fs_homepath->string, filename, "");
+ ospath[strlen(ospath) - 1] = '\0';
+
+ fileHandle_t f = FS_HandleForFile();
+ fsh[f].zipFile = false;
+
+ Com_DPrintf("FS_SV_FOpenFileWrite: %s\n", ospath);
+
+ FS_CheckFilenameIsMutable(ospath, __FUNCTION__);
+
+ if (FS_CreatePath(ospath))
+ {
+ return 0;
+ }
+
+ Com_DPrintf("writing to: %s\n", ospath);
+ fsh[f].handleFiles.file.o = Sys_FOpen(ospath, "wb");
+
+ Q_strncpyz(fsh[f].name, filename, sizeof(fsh[f].name));
+
+ fsh[f].handleSync = false;
+ if (!fsh[f].handleFiles.file.o)
+ {
+ f = 0;
+ }
+
+ return f;
+}
+
+/*
+===========
+FS_SV_FOpenFileRead
+
+Search for a file somewhere below the home path then base path
+in that order
+===========
+*/
+long FS_SV_FOpenFileRead(const char *filename, fileHandle_t *fp)
+{
+ if (!fs_searchpaths)
+ {
+ Com_Error(ERR_FATAL, "Filesystem call made without initialization");
+ }
+
+ fileHandle_t f = FS_HandleForFile();
+ fsh[f].zipFile = false;
+
+ Q_strncpyz(fsh[f].name, filename, sizeof(fsh[f].name));
+
+ // don't let sound stutter
+ S_ClearSoundBuffer();
+
+ // search homepath
+ char *ospath = FS_BuildOSPath(fs_homepath->string, filename, "");
+ // remove trailing slash
+ ospath[strlen(ospath) - 1] = '\0';
+
+ Com_DPrintf("FS_SV_FOpenFileRead (fs_homepath): %s\n", ospath);
+
+ fsh[f].handleFiles.file.o = Sys_FOpen(ospath, "rb");
+ fsh[f].handleSync = false;
+ if (!fsh[f].handleFiles.file.o)
+ {
+ // If fs_homepath == fs_basepath, don't bother
+ if (Q_stricmp(fs_homepath->string, fs_basepath->string))
+ {
+ // search basepath
+ ospath = FS_BuildOSPath(fs_basepath->string, filename, "");
+ ospath[strlen(ospath) - 1] = '\0';
+
+ Com_DPrintf("FS_SV_FOpenFileRead (fs_basepath): %s\n", ospath);
+
+ fsh[f].handleFiles.file.o = Sys_FOpen(ospath, "rb");
+ fsh[f].handleSync = false;
+ }
+
+ if (!fsh[f].handleFiles.file.o)
+ {
+ f = 0;
+ }
+ }
+
+ *fp = f;
+ if (f)
+ {
+ return FS_filelength(f);
+ }
+
+ return -1;
+}
+
+/*
+===========
+FS_SV_Rename
+
+===========
+*/
+void FS_SV_Rename(const char *from, const char *to, bool safe)
+{
+ if (!fs_searchpaths)
+ {
+ Com_Error(ERR_FATAL, "Filesystem call made without initialization");
+ }
+
+ // don't let sound stutter
+ S_ClearSoundBuffer();
+
+ char *from_ospath = FS_BuildOSPath(fs_homepath->string, from, "");
+ char *to_ospath = FS_BuildOSPath(fs_homepath->string, to, "");
+
+ from_ospath[strlen(from_ospath) - 1] = '\0';
+ to_ospath[strlen(to_ospath) - 1] = '\0';
+
+ Com_DPrintf("FS_SV_Rename: (%s) %s --> %s\n", safe ? "safe" : "unsafe", from_ospath, to_ospath);
+
+ if (safe)
+ {
+ FS_CheckFilenameIsMutable(to_ospath, __FUNCTION__);
+ }
+
+ rename(from_ospath, to_ospath);
+}
+
+/*
+===========
+FS_Rename
+
+===========
+*/
+void FS_Rename(const char *from, const char *to)
+{
+ if (!fs_searchpaths)
+ {
+ Com_Error(ERR_FATAL, "Filesystem call made without initialization");
+ }
+
+ // don't let sound stutter
+ S_ClearSoundBuffer();
+
+ char *from_ospath = FS_BuildOSPath(fs_homepath->string, fs_gamedir, from);
+ char *to_ospath = FS_BuildOSPath(fs_homepath->string, fs_gamedir, to);
+
+ Com_DPrintf("FS_Rename: %s --> %s\n", from_ospath, to_ospath);
+
+ FS_CheckFilenameIsMutable(to_ospath, __FUNCTION__);
+
+ rename(from_ospath, to_ospath);
+}
+
+/*
+==============
+FS_FCloseFile
+
+If the FILE pointer is an open pak file, leave it open.
+
+For some reason, other dll's can't just cal fclose()
+on files returned by FS_FOpenFile...
+==============
+*/
+void fileHandleData_t::close()
+{
+ if (zipFile == true)
+ {
+ unzCloseCurrentFile(handleFiles.file.z);
+
+ if (handleFiles.unique)
+ unzClose(handleFiles.file.z);
+ }
+ // we didn't find it as a pak, so close it as a unique file
+ else if (handleFiles.file.o)
+ {
+ ::fclose(handleFiles.file.o);
+ }
+
+ ::memset(this, 0, sizeof(*this));
+}
+
+void FS_FCloseFile(fileHandle_t f)
+{
+ if (!fs_searchpaths)
+ Com_Error(ERR_FATAL, "Filesystem call made without initialization");
+
+ fsh[f].close();
+
+ ::memset(&fsh[f], 0, sizeof(fsh[f]));
+}
+
+/*
+===========
+FS_FOpenFileWrite
+
+===========
+*/
+fileHandle_t FS_FOpenFileWrite(const char *filename)
+{
+
+ if (!fs_searchpaths)
+ {
+ Com_Error(ERR_FATAL, "Filesystem call made without initialization");
+ }
+
+ fileHandle_t f = FS_HandleForFile();
+ fsh[f].zipFile = false;
+
+ char *ospath = FS_BuildOSPath(fs_homepath->string, fs_gamedir, filename);
+
+ Com_DPrintf("FS_FOpenFileWrite: %s\n", ospath);
+
+ FS_CheckFilenameIsMutable(ospath, __FUNCTION__);
+
+ if (FS_CreatePath(ospath))
+ {
+ return 0;
+ }
+
+ // enabling the following line causes a recursive function call loop
+ // when running with +set logfile 1 +set developer 1
+ // Com_DPrintf( "writing to: %s\n", ospath );
+ fsh[f].handleFiles.file.o = Sys_FOpen(ospath, "wb");
+
+ Q_strncpyz(fsh[f].name, filename, sizeof(fsh[f].name));
+
+ fsh[f].handleSync = false;
+ if (!fsh[f].handleFiles.file.o)
+ {
+ f = 0;
+ }
+ return f;
+}
+
+/*
+===========
+FS_FOpenFileAppend
+
+===========
+*/
+fileHandle_t FS_FOpenFileAppend(const char *filename)
+{
+ char *ospath;
+ fileHandle_t f;
+
+ if (!fs_searchpaths)
+ {
+ Com_Error(ERR_FATAL, "Filesystem call made without initialization");
+ }
+
+ f = FS_HandleForFile();
+ fsh[f].zipFile = false;
+
+ Q_strncpyz(fsh[f].name, filename, sizeof(fsh[f].name));
+
+ // don't let sound stutter
+ S_ClearSoundBuffer();
+
+ ospath = FS_BuildOSPath(fs_homepath->string, fs_gamedir, filename);
+
+ Com_DPrintf("FS_FOpenFileAppend: %s\n", ospath);
+
+ FS_CheckFilenameIsMutable(ospath, __FUNCTION__);
+
+ if (FS_CreatePath(ospath))
+ {
+ return 0;
+ }
+
+ fsh[f].handleFiles.file.o = Sys_FOpen(ospath, "ab");
+ fsh[f].handleSync = false;
+ if (!fsh[f].handleFiles.file.o)
+ {
+ f = 0;
+ }
+ return f;
+}
+
+/*
+===========
+FS_FCreateOpenPipeFile
+
+===========
+*/
+fileHandle_t FS_FCreateOpenPipeFile(const char *filename)
+{
+ if (!fs_searchpaths)
+ {
+ Com_Error(ERR_FATAL, "Filesystem call made without initialization");
+ }
+
+ fileHandle_t f = FS_HandleForFile();
+ fsh[f].zipFile = false;
+
+ Q_strncpyz(fsh[f].name, filename, sizeof(fsh[f].name));
+
+ // don't let sound stutter
+ S_ClearSoundBuffer();
+
+ char *ospath = FS_BuildOSPath(fs_homepath->string, fs_gamedir, filename);
+
+ Com_DPrintf("FS_FCreateOpenPipeFile: %s\n", ospath);
+
+ FS_CheckFilenameIsMutable(ospath, __FUNCTION__);
+
+ FILE *fifo = Sys_Mkfifo(ospath);
+ if (fifo)
+ {
+ fsh[f].handleFiles.file.o = fifo;
+ fsh[f].handleSync = false;
+ }
+ else
+ {
+ Com_Printf(S_COLOR_YELLOW
+ "WARNING: Could not create new com_pipefile at %s. "
+ "com_pipefile will not be used.\n",
+ ospath);
+ f = 0;
+ }
+
+ return f;
+}
+
+/*
+===========
+FS_FilenameCompare
+
+Ignore case and seprator char distinctions
+===========
+*/
+bool FS_FilenameCompare(const char *s1, const char *s2)
+{
+ int c1, c2;
+
+ do
+ {
+ c1 = *s1++;
+ c2 = *s2++;
+
+ if (c1 >= 'a' && c1 <= 'z')
+ {
+ c1 -= ('a' - 'A');
+ }
+ if (c2 >= 'a' && c2 <= 'z')
+ {
+ c2 -= ('a' - 'A');
+ }
+
+ if (c1 == '\\' || c1 == ':')
+ {
+ c1 = '/';
+ }
+ if (c2 == '\\' || c2 == ':')
+ {
+ c2 = '/';
+ }
+
+ if (c1 != c2)
+ {
+ return true; // strings not equal
+ }
+ } while (c1);
+
+ return false; // strings are equal
+}
+
+/*
+===========
+FS_IsExt
+
+Return true if ext matches file extension filename
+===========
+*/
+static bool FS_IsExt(const char *filename, const char *ext, int namelen)
+{
+ int extlen = strlen(ext);
+
+ if (extlen > namelen) return false;
+
+ filename += namelen - extlen;
+
+ return !Q_stricmp(filename, ext);
+}
+
+/*
+===========
+FS_IsDemoExt
+
+Return true if filename has a demo extension
+===========
+*/
+
+static bool FS_IsDemoExt(const char *filename)
+{
+ const char *ext_test = strrchr(filename, '.');
+ if (ext_test && !Q_stricmpn(ext_test + 1, DEMOEXT, ARRAY_LEN(DEMOEXT) - 1))
+ {
+ int protocol = atoi(ext_test + ARRAY_LEN(DEMOEXT));
+ if (protocol == PROTOCOL_VERSION) return true;
+
+ for (int i = 0; demo_protocols[i]; i++)
+ if (demo_protocols[i] == protocol) return true;
+ }
+
+ return false;
+}
+
+/*
+===========
+FS_FOpenFileReadDir
+
+Tries opening file "filename" in searchpath "search"
+Returns filesize and an open FILE pointer.
+===========
+*/
+long FS_FOpenFileReadDir(
+ const char *filename, void *_search, fileHandle_t *file, bool uniqueFILE, bool unpure)
+{
+ pack_t *pak;
+ directory_t *dir;
+ char *netpath;
+ FILE *filep;
+ int len;
+
+ searchpath_t *search = static_cast<searchpath_t *>(_search);
+
+ if (filename == nullptr)
+ Com_Error(ERR_FATAL, "FS_FOpenFileRead: nullptr 'filename' parameter passed");
+
+ // qpaths are not supposed to have a leading slash
+ if (filename[0] == '/' || filename[0] == '\\') filename++;
+
+ // make absolutely sure that it can't back up the path.
+ // The searchpaths do guarantee that something will always
+ // be prepended, so we don't need to worry about "c:" or "//limbo"
+ if (strstr(filename, "..") || strstr(filename, "::"))
+ {
+ if (file == nullptr) return false;
+
+ *file = 0;
+ return -1;
+ }
+
+ if (file == nullptr)
+ {
+ // just wants to see if file is there
+
+ if ( fs_debug->integer )
+ {
+ Com_Printf(S_COLOR_GREEN "Searching for: " S_COLOR_RED "%s\n", filename);
+ }
+
+ // is the element a pak file?
+ if (search->pack)
+ {
+ auto pakfile = search->pack->find(filename);
+ if (pakfile)
+ {
+ // found it!
+ if (!pakfile->len)
+ {
+ // FIXME: It's not nice, but legacy code depends on
+ // positive value if file exists no matter what size
+ return 1;
+ }
+
+ return pakfile->len;
+ }
+ }
+ else if (search->dir)
+ {
+ dir = search->dir;
+
+ netpath = FS_BuildOSPath(dir->path, dir->gamedir, filename);
+ filep = Sys_FOpen(netpath, "rb");
+
+ if (filep)
+ {
+ len = FS_fplength(filep);
+ fclose(filep);
+
+ if (len)
+ return len;
+ else
+ return 1;
+ }
+ }
+
+ return 0;
+ }
+
+ *file = FS_HandleForFile();
+ fsh[*file].handleFiles.unique = uniqueFILE;
+
+ // is the element a pak file?
+ if (search->pack)
+ {
+ pak = search->pack;
+ auto pakfile = pak->find(filename);
+ if (pakfile )
+ {
+ if ( fs_debug->integer == 2 )
+ Com_Printf(S_COLOR_GREEN "#2 Searching for: " S_COLOR_RED "%s\n", filename);
+
+ // disregard if it doesn't match one of the allowed pure pak files
+ if (!unpure && !pak->is_pure())
+ {
+ if ( fs_debug->integer == 2 )
+ Com_Printf(S_COLOR_GREEN "Ugh-oh %s found in unpure pk3\n", filename);
+
+ *file = 0;
+ return -1;
+ }
+
+ len = strlen(filename);
+
+ if (!(pak->referenced & FS_GENERAL_REF))
+ {
+ if ( !FS_IsExt(filename, ".shader", len)
+ && !FS_IsExt(filename, ".mtr", len)
+ && !FS_IsExt(filename, ".txt", len)
+ && !FS_IsExt(filename, ".cfg", len)
+ && !FS_IsExt(filename, ".config", len)
+ && !FS_IsExt(filename, ".arena", len)
+ && !FS_IsExt(filename, ".menu", len)
+ && !strstr(filename, "levelshots") )
+ {
+ pak->referenced |= FS_GENERAL_REF;
+ }
+ }
+
+ if (strstr(filename, "cgame.qvm"))
+ pak->referenced |= FS_CGAME_REF;
+
+ if (strstr(filename, "ui.qvm"))
+ pak->referenced |= FS_UI_REF;
+
+ if (uniqueFILE)
+ {
+ fsh[*file].handleFiles.file.z = unzOpen(pak->pakFilename);
+ if ( !fsh[*file].handleFiles.file.z )
+ Com_Error(ERR_FATAL, "Couldn't open %s", pak->pakFilename);
+ }
+ else
+ {
+ fsh[*file].handleFiles.file.z = pak->handle;
+ }
+
+ Q_strncpyz(fsh[*file].name, filename, sizeof(fsh[*file].name));
+ fsh[*file].zipFile = true;
+
+ // set the file position in the zip file (also sets the current file info)
+ unzSetOffset(fsh[*file].handleFiles.file.z, pakfile->pos);
+
+ // open the file in the zip
+ unzOpenCurrentFile(fsh[*file].handleFiles.file.z);
+ fsh[*file].zipFilePos = pakfile->pos;
+ fsh[*file].zipFileLen = pakfile->len;
+
+ if ( fs_debug->integer )
+ {
+ Com_Printf( "FS_FOpenFileRead: %s (found in '%s')\n",
+ filename, pak->pakFilename);
+ }
+
+ return pakfile->len;
+ }
+ }
+ else if (search->dir)
+ {
+ // check a file in the directory tree
+
+ // if we are running restricted, the only files we
+ // will allow to come from the directory are .cfg files
+ len = strlen(filename);
+ // FIXME TTimo I'm not sure about the fs_numServerPaks test
+ // if you are using FS_ReadFile to find out if a file exists,
+ // this test can make the search fail although the file is in the directory
+ // I had the problem on https://zerowing.idsoftware.com/bugzilla/show_bug.cgi?id=8
+ // turned out I used FS_FileExists instead
+ if (!unpure && fs_numServerPaks)
+ {
+ if (!FS_IsExt(filename, ".cfg", len) && // for config files
+ !FS_IsExt(filename, ".lua", len) && // lua
+ !FS_IsExt(filename, ".menu", len) && // menu files
+ !FS_IsExt(filename, ".game", len) && // menu files
+ !FS_IsExt(filename, ".dat", len) && // for journal files
+ !FS_IsDemoExt(filename)) // demos
+ {
+ *file = 0;
+ return -1;
+ }
+ }
+
+ dir = search->dir;
+
+ netpath = FS_BuildOSPath(dir->path, dir->gamedir, filename);
+ filep = Sys_FOpen(netpath, "rb");
+
+ if (filep == nullptr)
+ {
+ *file = 0;
+ return -1;
+ }
+
+ Q_strncpyz(fsh[*file].name, filename, sizeof(fsh[*file].name));
+ fsh[*file].zipFile = false;
+
+ if (fs_debug->integer)
+ {
+ Com_Printf("FS_FOpenFileRead: %s (found in '%s%c%s')\n",
+ filename, dir->path, PATH_SEP, dir->gamedir);
+ }
+
+ fsh[*file].handleFiles.file.o = filep;
+ return FS_fplength(filep);
+ }
+
+ *file = 0;
+ return -1;
+}
+
+/*
+===========
+FS_FOpenFileRead
+
+Finds the file in the search path.
+Returns filesize and an open FILE pointer.
+Used for streaming data out of either a
+separate file or a ZIP file.
+===========
+*/
+long FS_FOpenFileRead(const char *filename, fileHandle_t *file, bool uniqueFILE)
+{
+ searchpath_t *search;
+ long len;
+
+ if (!fs_searchpaths) Com_Error(ERR_FATAL, "Filesystem call made without initialization");
+
+ bool isLocalConfig = !strcmp(filename, "autoexec.cfg") || !strcmp(filename, Q3CONFIG_CFG);
+ for (search = fs_searchpaths; search; search = search->next)
+ {
+ // autoexec.cfg and q3config.cfg can only be loaded outside of pk3 files.
+ if (isLocalConfig && search->pack) continue;
+
+ len = FS_FOpenFileReadDir(filename, search, file, uniqueFILE, false);
+
+ if (file == nullptr)
+ {
+ if (len > 0) return len;
+ }
+ else
+ {
+ if (len >= 0 && *file) return len;
+ }
+ }
+
+#ifdef FS_MISSING
+ if (missingFiles) fprintf(missingFiles, "%s\n", filename);
+#endif
+
+ if (file)
+ {
+ *file = 0;
+ return -1;
+ }
+ else
+ {
+ // When file is nullptr, we're querying the existance of the file
+ // If we've got here, it doesn't exist
+ return 0;
+ }
+}
+
+/*
+=================
+FS_FindVM
+
+Find a suitable VM file in search path order.
+
+In each searchpath try:
+ - open DLL file if DLL loading enabled
+ - open QVM file
+
+Enable search for DLL by setting enableDll to FSVM_ENABLEDLL
+
+write found DLL or QVM to "found" and return VMI_NATIVE if DLL, VMI_COMPILED if QVM
+Return the searchpath in "startSearch".
+=================
+*/
+
+int FS_FindVM(void **startSearch, char *found, int foundlen, const char *name, int enableDll)
+{
+ searchpath_t *search, *lastSearch;
+ directory_t *dir;
+ pack_t *pack;
+ char dllName[MAX_OSPATH], qvmName[MAX_OSPATH];
+ char *netpath;
+
+ if (!fs_searchpaths) Com_Error(ERR_FATAL, "Filesystem call made without initialization");
+
+ if (enableDll) Com_sprintf(dllName, sizeof(dllName), "%s" DLL_EXT, name);
+
+ Com_sprintf(qvmName, sizeof(qvmName), "vm/%s.qvm", name);
+
+ lastSearch = static_cast<searchpath_t *>(*startSearch);
+ if (*startSearch == nullptr)
+ search = fs_searchpaths;
+ else
+ search = lastSearch->next;
+
+ while (search)
+ {
+ if (search->dir && (!fs_numServerPaks || !strcmp(name, "game")))
+ {
+ dir = search->dir;
+
+ if (enableDll)
+ {
+ netpath = FS_BuildOSPath(dir->path, dir->gamedir, dllName);
+
+ if (FS_FileInPathExists(netpath))
+ {
+ Q_strncpyz(found, netpath, foundlen);
+ *startSearch = search;
+
+ return VMI_NATIVE;
+ }
+ }
+
+ if (FS_FOpenFileReadDir(qvmName, search, nullptr, false, false) > 0)
+ {
+ *startSearch = search;
+ return VMI_COMPILED;
+ }
+ }
+ else if (search->pack)
+ {
+ pack = search->pack;
+
+ if (lastSearch && lastSearch->pack)
+ {
+ // make sure we only try loading one VM file per game dir
+ // i.e. if VM from pak7.pk3 fails we won't try one from pak6.pk3
+
+ if (!FS_FilenameCompare(lastSearch->pack->pakPathname, pack->pakPathname))
+ {
+ search = search->next;
+ continue;
+ }
+ }
+
+ if (FS_FOpenFileReadDir(qvmName, search, nullptr, false, false) > 0)
+ {
+ *startSearch = search;
+
+ return VMI_COMPILED;
+ }
+ }
+
+ search = search->next;
+ }
+
+ return -1;
+}
+
+int FS_Read(void *buffer, int len, fileHandle_t f)
+{
+ int block, remaining;
+ int read;
+ byte *buf;
+ int tries;
+
+ if (!fs_searchpaths)
+ {
+ Com_Error(ERR_FATAL, "Filesystem call made without initialization");
+ }
+
+ if (!f)
+ {
+ return 0;
+ }
+
+ buf = (byte *)buffer;
+ fs_readCount += len;
+
+ if (fsh[f].zipFile == false)
+ {
+ remaining = len;
+ tries = 0;
+ while (remaining)
+ {
+ block = remaining;
+ read = fread(buf, 1, block, fsh[f].handleFiles.file.o);
+ if (read == 0)
+ {
+ // we might have been trying to read from a CD, which
+ // sometimes returns a 0 read on windows
+ if (!tries)
+ {
+ tries = 1;
+ }
+ else
+ {
+ return len - remaining; // Com_Error (ERR_FATAL, "FS_Read: 0 bytes read");
+ }
+ }
+
+ if (read == -1)
+ {
+ Com_Error(ERR_FATAL, "FS_Read: -1 bytes read");
+ }
+
+ remaining -= read;
+ buf += read;
+ }
+ return len;
+ }
+ else
+ {
+ return unzReadCurrentFile(fsh[f].handleFiles.file.z, buffer, len);
+ }
+}
+
+/*
+=================
+FS_Write
+
+Properly handles partial writes
+=================
+*/
+int FS_Write(const void *buffer, int len, fileHandle_t h)
+{
+ if (!fs_searchpaths)
+ {
+ Com_Error(ERR_FATAL, "Filesystem call made without initialization");
+ }
+
+ if (!h)
+ {
+ return 0;
+ }
+
+ FILE *f = FS_FileForHandle(h);
+ byte *buf = (byte *)buffer;
+
+ int remaining = len;
+ int tries = 0;
+ while (remaining)
+ {
+ int block = remaining;
+ int written = fwrite(buf, 1, block, f);
+ if (written == 0)
+ {
+ if (!tries)
+ {
+ tries = 1;
+ }
+ else
+ {
+ Com_Printf("FS_Write: 0 bytes written\n");
+ return 0;
+ }
+ }
+
+ if (written == -1)
+ {
+ Com_Printf("FS_Write: -1 bytes written\n");
+ return 0;
+ }
+
+ remaining -= written;
+ buf += written;
+ }
+
+ if (fsh[h].handleSync)
+ {
+ fflush(f);
+ }
+ return len;
+}
+
+void QDECL FS_Printf(fileHandle_t h, const char *fmt, ...)
+{
+ va_list argptr;
+ char msg[MAXPRINTMSG];
+
+ va_start(argptr, fmt);
+ Q_vsnprintf(msg, sizeof(msg), fmt, argptr);
+ va_end(argptr);
+
+ FS_Write(msg, strlen(msg), h);
+}
+
+#define PK3_SEEK_BUFFER_SIZE 65536
+
+/*
+=================
+FS_Seek
+
+=================
+*/
+int FS_Seek(fileHandle_t f, long offset, enum FS_Origin origin)
+{
+ if (!fs_searchpaths)
+ {
+ Com_Error(ERR_FATAL, "Filesystem call made without initialization");
+ return -1;
+ }
+
+ if (fsh[f].zipFile == true)
+ {
+ // FIXME: this is really, really crappy
+ //(but better than what was here before)
+ byte buffer[PK3_SEEK_BUFFER_SIZE];
+ int remainder;
+ int currentPosition = FS_FTell(f);
+
+ // change negative offsets into FS_SEEK_SET
+ if (offset < 0)
+ {
+ switch (origin)
+ {
+ case FS_SEEK_END:
+ remainder = fsh[f].zipFileLen + offset;
+ break;
+
+ case FS_SEEK_CUR:
+ remainder = currentPosition + offset;
+ break;
+
+ case FS_SEEK_SET:
+ default:
+ remainder = 0;
+ break;
+ }
+
+ if (remainder < 0)
+ {
+ remainder = 0;
+ }
+
+ origin = FS_SEEK_SET;
+ }
+ else
+ {
+ if (origin == FS_SEEK_END)
+ {
+ remainder = fsh[f].zipFileLen - currentPosition + offset;
+ }
+ else
+ {
+ remainder = offset;
+ }
+ }
+
+ switch (origin)
+ {
+ case FS_SEEK_SET:
+ if (remainder == currentPosition)
+ {
+ return offset;
+ }
+ unzSetOffset(fsh[f].handleFiles.file.z, fsh[f].zipFilePos);
+ unzOpenCurrentFile(fsh[f].handleFiles.file.z);
+ // fallthrough
+
+ case FS_SEEK_END: // fall through
+ case FS_SEEK_CUR:
+ while (remainder > PK3_SEEK_BUFFER_SIZE)
+ {
+ FS_Read(buffer, PK3_SEEK_BUFFER_SIZE, f);
+ remainder -= PK3_SEEK_BUFFER_SIZE;
+ }
+ FS_Read(buffer, remainder, f);
+ return offset;
+
+ default:
+ Com_Error(ERR_FATAL, "Bad origin in FS_Seek");
+ return -1;
+ }
+ }
+ else
+ {
+ FILE *file;
+ file = FS_FileForHandle(f);
+ int _origin;
+ switch (origin)
+ {
+ case FS_SEEK_CUR:
+ _origin = SEEK_CUR;
+ break;
+ case FS_SEEK_END:
+ _origin = SEEK_END;
+ break;
+ case FS_SEEK_SET:
+ _origin = SEEK_SET;
+ break;
+ default:
+ Com_Error(ERR_FATAL, "Bad origin in FS_Seek");
+ break;
+ }
+
+ return fseek(file, offset, _origin);
+ }
+}
+
+/*
+======================================================================================
+
+CONVENIENCE FUNCTIONS FOR ENTIRE FILES
+
+======================================================================================
+*/
+
+int FS_FileIsInPAK_A(bool alternate, const char *filename, int *pChecksum)
+{
+ if (!fs_searchpaths)
+ Com_Error(ERR_FATAL, "Filesystem call made without initialization");
+
+ if (!filename)
+ Com_Error(ERR_FATAL, "FS_FOpenFileRead: nullptr 'filename' parameter passed");
+
+ // qpaths are not supposed to have a leading slash
+ if (filename[0] == '/' || filename[0] == '\\')
+ filename++;
+
+ // make absolutely sure that it can't back up the path.
+ // The searchpaths do guarantee that something will always
+ // be prepended, so we don't need to worry about "c:" or "//limbo"
+ if (strstr(filename, "..") || strstr(filename, "::"))
+ return -1;
+
+ //
+ // search through the path, one element at a time
+ //
+ for (auto search = fs_searchpaths; search; search = search->next)
+ {
+ if (!search->pack)
+ continue;
+
+ // disregard if it doesn't match one of the allowed pure pak files
+ if (!search->pack->is_pure())
+ continue;
+
+ if ((alternate && search->pack->onlyPrimary) ||
+ (!alternate && search->pack->onlyAlternate))
+ continue;
+
+ auto found = search->pack->find(filename);
+ if (found)
+ {
+ if (pChecksum)
+ *pChecksum = search->pack->pure_checksum;
+
+ return 1;
+ }
+ }
+ return -1;
+}
+
+int FS_FileIsInPAK(const char *filename, int *pChecksum)
+{
+ return FS_FileIsInPAK_A(false, filename, pChecksum);
+}
+/*
+============
+FS_ReadFileDir
+
+Filename are relative to the quake search path
+a null buffer will just return the file length without loading
+If searchPath is non-nullptr search only in that specific search path
+============
+*/
+long FS_ReadFileDir(const char *qpath, void *searchPath, bool unpure, void **buffer)
+{
+ fileHandle_t h;
+ byte *buf;
+ bool isConfig;
+ long len;
+
+ if (!fs_searchpaths)
+ {
+ Com_Error(ERR_FATAL, "Filesystem call made without initialization");
+ }
+
+ if (!qpath || !qpath[0])
+ {
+ Com_Error(ERR_FATAL, "FS_ReadFile with empty name");
+ }
+
+ buf = nullptr; // quiet compiler warning
+
+ // if this is a .cfg file and we are playing back a journal, read
+ // it from the journal file
+ if (strstr(qpath, ".cfg"))
+ {
+ isConfig = true;
+ if (com_journal && com_journal->integer == 2)
+ {
+ Com_DPrintf("Loading %s from journal file.\n", qpath);
+
+ int r = FS_Read(&len, sizeof(len), com_journalDataFile);
+ if (r != sizeof(len))
+ {
+ if (buffer != nullptr) *buffer = nullptr;
+ return -1;
+ }
+
+ // if the file didn't exist when the journal was created
+ if (!len)
+ {
+ if (buffer == nullptr)
+ {
+ return 1; // hack for old journal files
+ }
+ *buffer = nullptr;
+ return -1;
+ }
+
+ if (buffer == nullptr)
+ {
+ return len;
+ }
+
+ buf = static_cast<byte *>(Hunk_AllocateTempMemory(len + 1));
+ *buffer = buf;
+
+ r = FS_Read(buf, len, com_journalDataFile);
+ if (r != len)
+ {
+ Com_Error(ERR_FATAL, "Read from journalDataFile failed");
+ }
+
+ fs_loadCount++;
+ fs_loadStack++;
+
+ // guarantee that it will have a trailing 0 for string operations
+ buf[len] = 0;
+
+ return len;
+ }
+ }
+ else
+ {
+ isConfig = false;
+ }
+
+ searchpath_t *search = static_cast<searchpath_t *>(searchPath);
+ if (search == nullptr)
+ {
+ // look for it in the filesystem or pack files
+ len = FS_FOpenFileRead(qpath, &h, false);
+ }
+ else
+ {
+ // look for it in a specific search path only
+ len = FS_FOpenFileReadDir(qpath, search, &h, false, unpure);
+ }
+
+ if (h == 0)
+ {
+ if (buffer)
+ {
+ *buffer = nullptr;
+ }
+ // if we are journalling and it is a config file, write a zero to the journal file
+ if (isConfig && com_journal && com_journal->integer == 1)
+ {
+ Com_DPrintf("Writing zero for %s to journal file.\n", qpath);
+ len = 0;
+ FS_Write(&len, sizeof(len), com_journalDataFile);
+ FS_Flush(com_journalDataFile);
+ }
+ return -1;
+ }
+
+ if (!buffer)
+ {
+ if (isConfig && com_journal && com_journal->integer == 1)
+ {
+ Com_DPrintf("Writing len for %s to journal file.\n", qpath);
+ FS_Write(&len, sizeof(len), com_journalDataFile);
+ FS_Flush(com_journalDataFile);
+ }
+ FS_FCloseFile(h);
+ return len;
+ }
+
+ fs_loadCount++;
+ fs_loadStack++;
+
+ buf = static_cast<byte *>(Hunk_AllocateTempMemory(len + 1));
+ *buffer = buf;
+
+ FS_Read(buf, len, h);
+
+ // guarantee that it will have a trailing 0 for string operations
+ buf[len] = 0;
+ FS_FCloseFile(h);
+
+ // if we are journalling and it is a config file, write it to the journal file
+ if (isConfig && com_journal && com_journal->integer == 1)
+ {
+ Com_DPrintf("Writing %s to journal file.\n", qpath);
+ FS_Write(&len, sizeof(len), com_journalDataFile);
+ FS_Write(buf, len, com_journalDataFile);
+ FS_Flush(com_journalDataFile);
+ }
+ return len;
+}
+
+/*
+============
+FS_ReadFile
+
+Filename are relative to the quake search path
+a null buffer will just return the file length without loading
+============
+*/
+long FS_ReadFile(const char *qpath, void **buffer)
+{
+ return FS_ReadFileDir(qpath, nullptr, false, buffer);
+}
+/*
+=============
+FS_FreeFile
+=============
+*/
+void FS_FreeFile(void *buffer)
+{
+ if (!fs_searchpaths)
+ {
+ Com_Error(ERR_FATAL, "Filesystem call made without initialization");
+ }
+ if (!buffer)
+ {
+ Com_Error(ERR_FATAL, "FS_FreeFile( nullptr )");
+ }
+
+ fs_loadStack--;
+ Hunk_FreeTempMemory(buffer);
+
+ // if all of our temp files are free, clear all of our space
+ if (fs_loadStack == 0)
+ {
+ Hunk_ClearTempMemory();
+ }
+}
+
+/*
+============
+FS_WriteFile
+
+Filename are relative to the quake search path
+============
+*/
+void FS_WriteFile(const char *qpath, const void *buffer, int size)
+{
+ if (!fs_searchpaths)
+ {
+ Com_Error(ERR_FATAL, "Filesystem call made without initialization");
+ }
+
+ if (!qpath || !buffer)
+ {
+ Com_Error(ERR_FATAL, "FS_WriteFile: nullptr parameter");
+ }
+
+ fileHandle_t f = FS_FOpenFileWrite(qpath);
+ if (!f)
+ {
+ Com_Printf("Failed to open %s\n", qpath);
+ return;
+ }
+
+ FS_Write(buffer, size, f);
+ FS_FCloseFile(f);
+}
+
+/*
+==========================================================================
+
+ZIP FILE LOADING
+
+==========================================================================
+*/
+
+/*
+=================
+FS_LoadZipFile
+
+Creates a new pack_t in the search chain for the contents of a zip file.
+=================
+*/
+static pack_t *FS_LoadZipFile(const char *zipfile, const char *basename)
+{
+ int fs_numHeaderLongs = 0;
+ unsigned long len = 0;
+ char filename[MAX_ZPATH];
+
+ auto z = unzOpen(zipfile);
+
+ unz_global_info gi;
+ int err = unzGetGlobalInfo(z, &gi);
+ if (err) return nullptr;
+
+ err = unzGoToFirstFile(z);
+ if (err) return nullptr;
+
+ for (uLong i = 0; i < gi.number_entry; i++)
+ {
+ unz_file_info fi;
+ err = unzGetCurrentFileInfo(
+ z, &fi, filename, sizeof(filename), nullptr, 0, nullptr, 0);
+
+ if (err) break;
+
+ len += strlen(filename) + 1;
+ unzGoToNextFile(z);
+ }
+
+ fileInPack_t *buildBuffer =
+ static_cast<fileInPack_t *>(Z_Malloc((gi.number_entry * sizeof(fileInPack_t)) + len));
+
+ char *namePtr = ((char *)buildBuffer) + gi.number_entry * sizeof(fileInPack_t);
+
+ int *fs_headerLongs = static_cast<int *>(Z_Malloc((gi.number_entry + 1) * sizeof(int)));
+
+ fs_headerLongs[fs_numHeaderLongs] = LittleLong(fs_checksumFeed);
+ fs_numHeaderLongs++;
+
+ // get the hash table size from the number of files in the zip
+ // because lots of custom pk3 files have less than 32 or 64 files
+ uLong hashsiz;
+ for (hashsiz = 1; hashsiz <= MAX_FILEHASH_SIZE; hashsiz <<= 1)
+ {
+ if (hashsiz > gi.number_entry) break;
+ }
+
+ pack_t *pack = static_cast<pack_t *>(Z_Malloc(sizeof(pack_t) + hashsiz * sizeof(fileInPack_t *)));
+
+ pack->hashSize = hashsiz;
+ pack->hashTable = (fileInPack_t **)(((char *)pack) + sizeof(pack_t));
+
+ for (int i = 0; i < pack->hashSize; i++)
+ {
+ pack->hashTable[i] = nullptr;
+ }
+
+ Q_strncpyz(pack->pakFilename, zipfile, sizeof(pack->pakFilename));
+ Q_strncpyz(pack->pakBasename, basename, sizeof(pack->pakBasename));
+
+ // strip .pk3 if needed
+ if ( strlen(pack->pakBasename) > 4 &&
+ !Q_stricmp(pack->pakBasename + strlen(pack->pakBasename) - 4, ".pk3"))
+ {
+ pack->pakBasename[strlen(pack->pakBasename) - 4] = '\0';
+ }
+
+ pack->handle = z;
+ pack->numfiles = gi.number_entry;
+ unzGoToFirstFile(z);
+ for (uLong i = 0; i < gi.number_entry; i++)
+ {
+ unz_file_info fi;
+ err = unzGetCurrentFileInfo(
+ z, &fi, filename, sizeof(filename), nullptr, 0, nullptr, 0);
+
+ if (err) break;
+
+ if (fi.uncompressed_size)
+ {
+ fs_headerLongs[fs_numHeaderLongs] = LittleLong(fi.crc);
+ fs_numHeaderLongs++;
+ }
+
+ Q_strlwr(filename);
+ long hash = FS_HashFileName(filename, pack->hashSize);
+
+ buildBuffer[i].name = namePtr;
+ strcpy(buildBuffer[i].name, filename);
+ namePtr += strlen(filename) + 1;
+
+ // store the file position in the zip
+ buildBuffer[i].pos = unzGetOffset(z);
+ buildBuffer[i].len = fi.uncompressed_size;
+ buildBuffer[i].next = pack->hashTable[hash];
+
+ pack->hashTable[hash] = &buildBuffer[i];
+ unzGoToNextFile(z);
+ }
+
+ pack->checksum =
+ Com_BlockChecksum(&fs_headerLongs[1], sizeof(*fs_headerLongs) * (fs_numHeaderLongs - 1));
+ pack->pure_checksum =
+ Com_BlockChecksum(fs_headerLongs, sizeof(*fs_headerLongs) * fs_numHeaderLongs);
+ pack->checksum = LittleLong(pack->checksum);
+ pack->pure_checksum = LittleLong(pack->pure_checksum);
+
+ Z_Free(fs_headerLongs);
+
+ pack->buildBuffer = buildBuffer;
+ return pack;
+}
+
+/*
+=================
+FS_FreePak
+
+Frees a pak structure and releases all associated resources
+=================
+*/
+
+static void FS_FreePak(pack_t *thepak)
+{
+ unzClose(thepak->handle);
+ Z_Free(thepak->buildBuffer);
+ Z_Free(thepak);
+}
+
+/*
+=================
+FS_GetZipChecksum
+
+Compares whether the given pak file matches a referenced checksum
+=================
+*/
+bool FS_CompareZipChecksum(const char *zipfile)
+{
+ pack_t *thepak = FS_LoadZipFile(zipfile, "");
+
+ if (!thepak) return false;
+
+ int checksum = thepak->checksum;
+ FS_FreePak(thepak);
+
+ for (int i = 0; i < fs_numServerReferencedPaks; i++)
+ {
+ if (checksum == fs_serverReferencedPaks[i]) return true;
+ }
+
+ return false;
+}
+
+/*
+=================================================================================
+
+DIRECTORY SCANNING FUNCTIONS
+
+=================================================================================
+*/
+
+#define MAX_FOUND_FILES 0x1000
+
+static int FS_ReturnPath(const char *zname, char *zpath, int *depth)
+{
+ int newdep = 0;
+ int len = 0;
+ int at = 0;
+ zpath[0] = 0;
+
+ while (zname[at] != 0)
+ {
+ if (zname[at] == '/' || zname[at] == '\\')
+ {
+ len = at;
+ newdep++;
+ }
+ at++;
+ }
+ strcpy(zpath, zname);
+ zpath[len] = 0;
+ *depth = newdep;
+
+ return len;
+}
+
+/*
+==================
+FS_AddFileToList
+==================
+*/
+static int FS_AddFileToList(char *name, char *list[MAX_FOUND_FILES], int nfiles)
+{
+ if (nfiles == MAX_FOUND_FILES - 1)
+ {
+ return nfiles;
+ }
+
+ for (int i = 0; i < nfiles; i++)
+ {
+ if (!Q_stricmp(name, list[i]))
+ {
+ return nfiles; // allready in list
+ }
+ }
+ list[nfiles] = CopyString(name);
+ nfiles++;
+
+ return nfiles;
+}
+
+/*
+===============
+FS_ListFilteredFiles
+
+Returns a uniqued list of files that match the given criteria
+from all search paths
+===============
+*/
+char **FS_ListFilteredFiles(const char *path, const char *extension, const char *filter,
+ int *numfiles, bool allowNonPureFilesOnDisk)
+{
+ int nfiles;
+ char **listCopy;
+ char *list[MAX_FOUND_FILES];
+ searchpath_t *search;
+ int i;
+ int pathLength;
+ int extensionLength;
+ int length, pathDepth, temp;
+ pack_t *pak;
+ fileInPack_t *buildBuffer;
+ char zpath[MAX_ZPATH];
+
+ if (!fs_searchpaths)
+ {
+ Com_Error(ERR_FATAL, "Filesystem call made without initialization");
+ }
+
+ if (!path)
+ {
+ *numfiles = 0;
+ return nullptr;
+ }
+ if (!extension)
+ {
+ extension = "";
+ }
+
+ pathLength = strlen(path);
+ if (path[pathLength - 1] == '\\' || path[pathLength - 1] == '/')
+ {
+ pathLength--;
+ }
+ extensionLength = strlen(extension);
+ nfiles = 0;
+ FS_ReturnPath(path, zpath, &pathDepth);
+
+ //
+ // search through the path, one element at a time, adding to list
+ //
+ for (search = fs_searchpaths; search; search = search->next)
+ {
+ // is the element a pak file?
+ if (search->pack)
+ {
+ // ZOID: If we are pure, don't search for files on paks that
+ // aren't on the pure list
+ if (!search->pack->is_pure())
+ {
+ continue;
+ }
+
+ // look through all the pak file elements
+ pak = search->pack;
+ buildBuffer = pak->buildBuffer;
+ for (i = 0; i < pak->numfiles; i++)
+ {
+ char *name;
+ int zpathLen, depth;
+
+ // check for directory match
+ name = buildBuffer[i].name;
+ //
+ if (filter)
+ {
+ // case insensitive
+ if (!Com_FilterPath(filter, name, false)) continue;
+ // unique the match
+ nfiles = FS_AddFileToList(name, list, nfiles);
+ }
+ else
+ {
+ zpathLen = FS_ReturnPath(name, zpath, &depth);
+
+ if ((depth - pathDepth) > 2 || pathLength > zpathLen ||
+ Q_stricmpn(name, path, pathLength))
+ {
+ continue;
+ }
+
+ // check for extension match
+ length = strlen(name);
+ if (length < extensionLength)
+ {
+ continue;
+ }
+
+ if (Q_stricmp(name + length - extensionLength, extension))
+ {
+ continue;
+ }
+ // unique the match
+
+ temp = pathLength;
+ if (pathLength)
+ {
+ temp++; // include the '/'
+ }
+ nfiles = FS_AddFileToList(name + temp, list, nfiles);
+ }
+ }
+ }
+ else if (search->dir)
+ { // scan for files in the filesystem
+
+ // don't scan directories for files if we are pure or restricted
+ if (fs_numServerPaks && !allowNonPureFilesOnDisk)
+ {
+ continue;
+ }
+ else
+ {
+ int numSysFiles;
+ char *netpath = FS_BuildOSPath(search->dir->path, search->dir->gamedir, path);
+ char **sysFiles = Sys_ListFiles(netpath, extension, filter, &numSysFiles, false);
+ for (i = 0; i < numSysFiles; i++)
+ {
+ // unique the match
+ char *name = sysFiles[i];
+ nfiles = FS_AddFileToList(name, list, nfiles);
+ }
+ Sys_FreeFileList(sysFiles);
+ }
+ }
+ }
+
+ // return a copy of the list
+ *numfiles = nfiles;
+
+ if (!nfiles)
+ {
+ return nullptr;
+ }
+
+ listCopy = static_cast<char **>(Z_Malloc((nfiles + 1) * sizeof(*listCopy)));
+ for (i = 0; i < nfiles; i++)
+ {
+ listCopy[i] = list[i];
+ }
+ listCopy[i] = nullptr;
+
+ return listCopy;
+}
+
+/*
+=================
+FS_ListFiles
+=================
+*/
+char **FS_ListFiles(const char *path, const char *extension, int *numfiles)
+{
+ return FS_ListFilteredFiles(path, extension, nullptr, numfiles, false);
+}
+
+/*
+=================
+FS_FreeFileList
+=================
+*/
+void FS_FreeFileList(char **list)
+{
+ if (!fs_searchpaths)
+ {
+ Com_Error(ERR_FATAL, "Filesystem call made without initialization");
+ }
+
+ if (!list)
+ {
+ return;
+ }
+
+ for (int i = 0; list[i]; i++)
+ {
+ Z_Free(list[i]);
+ }
+
+ Z_Free(list);
+}
+
+/*
+================
+FS_GetFileList
+================
+*/
+int FS_GetFileList(const char *path, const char *extension, char *listbuf, int bufsize)
+{
+ int nFiles = 0;
+ int nTotal = 0;
+ *listbuf = 0;
+
+ if (Q_stricmp(path, "$modlist") == 0)
+ {
+ return FS_GetModList(listbuf, bufsize);
+ }
+
+ char **pFiles = FS_ListFiles(path, extension, &nFiles);
+
+ for (int i = 0; i < nFiles; i++)
+ {
+ int nLen = strlen(pFiles[i]) + 1;
+ if (nTotal + nLen + 1 < bufsize)
+ {
+ strcpy(listbuf, pFiles[i]);
+ listbuf += nLen;
+ nTotal += nLen;
+ }
+ else
+ {
+ nFiles = i;
+ break;
+ }
+ }
+
+ FS_FreeFileList(pFiles);
+
+ return nFiles;
+}
+
+/*
+================
+FS_GetFilteredFiles
+================
+*/
+int FS_GetFilteredFiles(
+ const char *path, const char *extension, const char *filter, char *listbuf, int bufsize)
+{
+ int nFiles = 0;
+ int nTotal = 0;
+ *listbuf = 0;
+
+ char **pFiles = FS_ListFilteredFiles(path, extension, filter, &nFiles, false);
+
+ for (int i = 0; i < nFiles; i++)
+ {
+ int nLen = strlen(pFiles[i]) + 1;
+ if (nTotal + nLen + 1 < bufsize)
+ {
+ strcpy(listbuf, pFiles[i]);
+ listbuf += nLen;
+ nTotal += nLen;
+ }
+ else
+ {
+ nFiles = i;
+ break;
+ }
+ }
+
+ FS_FreeFileList(pFiles);
+
+ return nFiles;
+}
+
+/*
+=======================
+Sys_ConcatenateFileLists
+
+mkv: Naive implementation. Concatenates three lists into a
+ new list, and frees the old lists from the heap.
+bk001129 - from cvs1.17 (mkv)
+
+FIXME TTimo those two should move to common.c next to Sys_ListFiles
+=======================
+ */
+static unsigned int Sys_CountFileList(char **list)
+{
+ unsigned int i = 0;
+ if (list)
+ {
+ while (*list)
+ {
+ list++;
+ i++;
+ }
+ }
+ return i;
+}
+
+static char **Sys_ConcatenateFileLists(char **list0, char **list1)
+{
+ char **cat, **dst;
+
+ int totalLength = 0;
+ totalLength += Sys_CountFileList(list0);
+ totalLength += Sys_CountFileList(list1);
+
+ /* Create new list. */
+ dst = cat = static_cast<char **>(Z_Malloc((totalLength + 1) * sizeof(char *)));
+
+ /* Copy over lists. */
+ if (list0)
+ {
+ for (char **src = list0; *src; src++, dst++) *dst = *src;
+ }
+
+ if (list1)
+ {
+ for (char **src = list1; *src; src++, dst++) *dst = *src;
+ }
+
+ // Terminate the list
+ *dst = nullptr;
+
+ // Free our old lists.
+ // NOTE: not freeing their content, it's been merged in dst and still being used
+ if (list0) Z_Free(list0);
+ if (list1) Z_Free(list1);
+
+ return cat;
+}
+
+/*
+================
+FS_GetModList
+
+Returns a list of mod directory names
+A mod directory is a peer to baseq3 with a pk3 or pk3dir in it
+================
+*/
+int FS_GetModList( char *listbuf, int bufsize )
+{
+ char * start = listbuf;
+ *listbuf = '\0';
+
+ // paths to search for mods
+ const char * const paths[] = {
+ fs_basepath->string,
+ fs_homepath->string
+ };
+
+ char **pFiles = nullptr;
+ for (int i = 0; i < ARRAY_LEN(paths); i++)
+ {
+ int dummy;
+ char **pFiles0 = Sys_ListFiles(paths[i], NULL, NULL, &dummy, true);
+ pFiles = Sys_ConcatenateFileLists(pFiles, pFiles0);
+ }
+
+ int nMods = 0;
+ int nTotal = 0;
+ for (int i = 0; i < Sys_CountFileList(pFiles); i++)
+ {
+ const char* name = pFiles[i];
+
+ if ( name[0] == '.' )
+ continue;
+
+ // In order to be a valid mod the directory must contain at least one
+ // .pk3 or .pk3dir we didn't keep the information when we merged the
+ // directory names, as to what OS Path it was found under so we will
+ // try each of them here.
+ int nPaks = 0;
+ int nPakDirs = 0;
+ for (int j = 0; j < ARRAY_LEN(paths); j++)
+ {
+ const char* path = FS_BuildOSPath(paths[j], name, "");
+ int nDirs = 0;
+
+ char **pPaks = Sys_ListFiles(path, ".pk3", NULL, &nPaks, false);
+ char **pDirs = Sys_ListFiles(path, "/", NULL, &nDirs, false);
+ for (int k = 0; k < nDirs; k++)
+ {
+ // we only want to count directories ending with ".pk3dir"
+ if (FS_IsExt(pDirs[k], ".pk3dir", strlen(pDirs[k])))
+ nPakDirs++;
+ }
+
+ // we only use Sys_ListFiles to check whether files are present
+ Sys_FreeFileList(pPaks);
+ Sys_FreeFileList(pDirs);
+
+ if (nPaks > 0 || nPakDirs > 0)
+ break;
+ }
+
+ if (nPaks > 0 || nPakDirs > 0)
+ {
+ size_t nLen = strlen(name) + 1;
+
+ if (nTotal + nLen + 1 < bufsize)
+ {
+ strcpy(listbuf, name);
+ listbuf += nLen;
+ nTotal += nLen;
+ nMods++;
+ }
+ else
+ {
+ Com_Printf(S_COLOR_RED "Warning: Too many mods!\n");
+ break;
+ }
+ }
+ }
+
+ Sys_FreeFileList( pFiles );
+ return nMods;
+}
+
+//============================================================================
+
+/*
+================
+FS_Dir_f
+================
+*/
+void FS_Dir_f(void)
+{
+ const char *path;
+ const char *extension;
+
+ if (Cmd_Argc() < 2 || Cmd_Argc() > 3)
+ {
+ Com_Printf("usage: dir <directory> [extension]\n");
+ return;
+ }
+
+ if (Cmd_Argc() == 2)
+ {
+ path = Cmd_Argv(1);
+ extension = "";
+ }
+ else
+ {
+ path = Cmd_Argv(1);
+ extension = Cmd_Argv(2);
+ }
+
+ Com_Printf("Directory of %s %s\n", path, extension);
+ Com_Printf("---------------\n");
+
+ int ndirs;
+ char **dirnames = FS_ListFiles(path, extension, &ndirs);
+
+ for (int i = 0; i < ndirs; i++)
+ {
+ Com_Printf("%s\n", dirnames[i]);
+ }
+ FS_FreeFileList(dirnames);
+}
+
+/*
+===========
+FS_ConvertPath
+===========
+*/
+void FS_ConvertPath(char *s)
+{
+ while (*s)
+ {
+ if (*s == '\\' || *s == ':')
+ {
+ *s = '/';
+ }
+ s++;
+ }
+}
+
+/*
+===========
+FS_PathCmp
+
+Ignore case and seprator char distinctions
+===========
+*/
+int FS_PathCmp(const char *s1, const char *s2)
+{
+ int c1, c2;
+
+ do
+ {
+ c1 = *s1++;
+ c2 = *s2++;
+
+ if (c1 >= 'a' && c1 <= 'z')
+ {
+ c1 -= ('a' - 'A');
+ }
+ if (c2 >= 'a' && c2 <= 'z')
+ {
+ c2 -= ('a' - 'A');
+ }
+
+ if (c1 == '\\' || c1 == ':')
+ {
+ c1 = '/';
+ }
+ if (c2 == '\\' || c2 == ':')
+ {
+ c2 = '/';
+ }
+
+ if (c1 < c2)
+ {
+ return -1; // strings not equal
+ }
+ else if (c1 > c2)
+ {
+ return 1;
+ }
+ } while (c1);
+
+ return 0; // strings are equal
+}
+
+/*
+================
+FS_SortFileList
+================
+*/
+void FS_SortFileList(char **filelist, int numfiles)
+{
+ char **sortedlist = static_cast<char **>(Z_Malloc((numfiles + 1) * sizeof(*sortedlist)));
+ sortedlist[0] = nullptr;
+
+ int numsortedfiles = 0;
+ for (int i = 0; i < numfiles; i++)
+ {
+ int j;
+ for (j = 0; j < numsortedfiles; j++)
+ {
+ if (FS_PathCmp(filelist[i], sortedlist[j]) < 0) break;
+ }
+
+ int k;
+ for (k = numsortedfiles; k > j; k--)
+ {
+ sortedlist[k] = sortedlist[k - 1];
+ }
+
+ sortedlist[j] = filelist[i];
+ numsortedfiles++;
+ }
+ ::memcpy(filelist, sortedlist, numfiles * sizeof(*filelist));
+ Z_Free(sortedlist);
+}
+
+/*
+================
+FS_NewDir_f
+================
+*/
+void FS_NewDir_f(void)
+{
+ if (Cmd_Argc() < 2)
+ {
+ Com_Printf("usage: fdir <filter>\n");
+ Com_Printf("example: fdir *q3dm*.bsp\n");
+ return;
+ }
+
+ Com_Printf("---------------\n");
+
+ const char *filter = Cmd_Argv(1);
+
+ int ndirs;
+ char **dirnames = FS_ListFilteredFiles("", "", filter, &ndirs, false);
+
+ FS_SortFileList(dirnames, ndirs);
+ for (int i = 0; i < ndirs; i++)
+ {
+ FS_ConvertPath(dirnames[i]);
+ Com_Printf("%s\n", dirnames[i]);
+ }
+ Com_Printf("%d files listed\n", ndirs);
+ FS_FreeFileList(dirnames);
+}
+
+/*
+============
+FS_Path_f
+
+============
+*/
+void FS_Path_f(void)
+{
+ Com_Printf("We are looking in the current search path:\n");
+
+ for (auto s = fs_searchpaths; s; s = s->next)
+ {
+ if (s->pack)
+ {
+ Com_Printf("%s (%i files%s)\n",
+ s->pack->pakFilename,
+ s->pack->numfiles,
+ s->pack->onlyPrimary ? ", not for 1.1"
+ : s->pack->onlyAlternate ? ", only for 1.1" : "");
+
+ if (s->pack->primaryVersion)
+ Com_Printf(" (the 1.1 version of %s)\n",
+ s->pack->primaryVersion->pakFilename);
+
+ if (fs_numServerPaks)
+ {
+ if (!s->pack->is_pure())
+ {
+ Com_Printf(" not on the pure list\n");
+ }
+ else
+ {
+ Com_Printf(" on the pure list\n");
+ }
+ }
+ }
+ else
+ {
+ Com_Printf("%s%c%s\n", s->dir->path, PATH_SEP, s->dir->gamedir);
+ }
+ }
+
+ Com_Printf("\n");
+ for (int i = 1; i < MAX_FILE_HANDLES; i++)
+ {
+ if (fsh[i].handleFiles.file.o)
+ {
+ Com_Printf("handle %i: %s\n", i, fsh[i].name);
+ }
+ }
+}
+
+/*
+============
+FS_TouchFile_f
+============
+*/
+void FS_TouchFile_f(void)
+{
+ if (Cmd_Argc() != 2)
+ {
+ Com_Printf("Usage: touchFile <file>\n");
+ return;
+ }
+
+ fileHandle_t f;
+ FS_FOpenFileRead(Cmd_Argv(1), &f, false);
+ if (f)
+ {
+ FS_FCloseFile(f);
+ }
+}
+
+/*
+============
+FS_Which
+============
+*/
+
+bool FS_Which(const char *filename, void *searchPath)
+{
+ searchpath_t *search = static_cast<searchpath_t *>(searchPath);
+
+ if (FS_FOpenFileReadDir(filename, search, nullptr, false, false) > 0)
+ {
+ if (search->pack)
+ {
+ Com_Printf("File \"%s\" found in \"%s\"\n", filename, search->pack->pakFilename);
+ return true;
+ }
+ else if (search->dir)
+ {
+ Com_Printf("File \"%s\" found at \"%s\"\n", filename, search->dir->fullpath);
+ return true;
+ }
+ }
+
+ return false;
+}
+
+/*
+============
+FS_Which_f
+============
+*/
+void FS_Which_f(void)
+{
+ const char *filename = Cmd_Argv(1);
+
+ if (!filename[0])
+ {
+ Com_Printf("Usage: which <file>\n");
+ return;
+ }
+
+ // qpaths are not supposed to have a leading slash
+ if (filename[0] == '/' || filename[0] == '\\') filename++;
+
+ // just wants to see if file is there
+ for (searchpath_t *search = fs_searchpaths; search; search = search->next)
+ if (FS_Which(filename, search)) return;
+
+ Com_Printf("File not found: \"%s\"\n", filename);
+}
+
+//===========================================================================
+
+static int QDECL paksort(const void *a, const void *b)
+{
+ char *aa = *(char **)a;
+ char *bb = *(char **)b;
+
+ return FS_PathCmp(aa, bb);
+}
+
+/*
+================
+FS_AddGameDirectory
+
+Sets fs_gamedir, adds the directory to the head of the path,
+then loads the zip headers
+================
+*/
+void FS_AddGameDirectory(const char *path, const char *dir)
+{
+ pack_t *pak;
+ char curpath[MAX_OSPATH + 1];
+ int numfiles;
+ char **pakfiles;
+ int pakfilesi;
+ char **pakfilestmp;
+ int numdirs;
+ char **pakdirs;
+ int pakdirsi;
+ char **pakdirstmp;
+
+ int lengths[10][2];
+
+ // Unique
+ for (searchpath_t *sp = fs_searchpaths; sp; sp = sp->next)
+ {
+ if (sp->dir && !Q_stricmp(sp->dir->path, path) && !Q_stricmp(sp->dir->gamedir, dir))
+ return; // we've already got this one
+ }
+
+ Q_strncpyz(fs_gamedir, dir, sizeof(fs_gamedir));
+
+ // find all pak files in this directory
+ Q_strncpyz(curpath, FS_BuildOSPath(path, dir, ""), sizeof(curpath));
+ curpath[strlen(curpath) - 1] = '\0'; // strip the trailing slash
+
+ // Get .pk3 files
+ pakfiles = Sys_ListFiles(curpath, ".pk3", nullptr, &numfiles, false);
+
+ qsort(pakfiles, numfiles, sizeof(char *), paksort);
+
+ if (fs_numServerPaks)
+ {
+ numdirs = 0;
+ pakdirs = nullptr;
+ }
+ else
+ {
+ // Get top level directories (we'll filter them later since the Sys_ListFiles filtering is
+ // terrible)
+ pakdirs = Sys_ListFiles(curpath, "/", nullptr, &numdirs, false);
+ qsort(pakdirs, numdirs, sizeof(char *), paksort);
+ }
+
+ char prefixBuf[MAX_STRING_CHARS];
+ Q_strncpyz(prefixBuf, Cvar_VariableString("fs_pk3PrefixPairs"), sizeof(prefixBuf));
+ int numPairs = 0;
+
+ char *p = prefixBuf;
+ if (!p[0]) p = nullptr;
+
+ const char *prefixes[10][2];
+ while (p)
+ {
+ prefixes[numPairs][0] = p;
+ p = strchr(p, '&');
+ if (!p)
+ {
+ Com_Printf(S_COLOR_YELLOW "WARNING: fs_pk3PrefixPairs ends with an incomplete pair\n");
+ break;
+ }
+ lengths[numPairs][0] = (int)(p - prefixes[numPairs][0]);
+ *p++ = '\0';
+ prefixes[numPairs][1] = p;
+ p = strchr(p, '|');
+ if (p)
+ {
+ lengths[numPairs][1] = (int)(p - prefixes[numPairs][1]);
+ *p++ = '\0';
+ }
+ else
+ {
+ lengths[numPairs][1] = (int)strlen(prefixes[numPairs][1]);
+ }
+ if (lengths[numPairs][0] == 0 && lengths[numPairs][1] == 0)
+ {
+ Com_Printf(S_COLOR_YELLOW
+ "WARNING: fs_pk3PrefixPairs contains a null-null pair, "
+ "skipping this pair\n");
+ continue;
+ }
+ if (lengths[numPairs][0] != 0 && lengths[numPairs][1] != 0 &&
+ !Q_stricmpn(prefixes[numPairs][0], prefixes[numPairs][1],
+ MIN(lengths[numPairs][0], lengths[numPairs][1])))
+ {
+ Com_Printf(S_COLOR_YELLOW
+ "WARNING: in fs_pk3PrefixPairs, one of '%s' and '%s' is a real prefix "
+ "of the other, skipping this pair\n",
+ prefixes[numPairs][0], prefixes[numPairs][1]);
+
+ continue;
+ }
+ ++numPairs;
+ }
+ searchpath_t *otherSearchpaths = fs_searchpaths;
+
+ pakfilesi = 0;
+ pakdirsi = 0;
+
+ while ((pakfilesi < numfiles) || (pakdirsi < numdirs))
+ {
+ bool pakwhich;
+ // Check if a pakfile or pakdir comes next
+ if (pakfilesi >= numfiles)
+ {
+ pakwhich = false; // We've used all the pak files, it must be a pak directory.
+ }
+ else if (pakdirsi >= numdirs)
+ {
+ pakwhich = true; // We've used all the pak directories, it must be a pak file.
+ }
+ else
+ {
+ // Could be either, compare to see which name comes first
+ // Need tmp variables for appropriate indirection for paksort()
+ pakfilestmp = &pakfiles[pakfilesi];
+ pakdirstmp = &pakdirs[pakdirsi];
+ pakwhich = (paksort(pakfilestmp, pakdirstmp) < 0);
+ }
+
+ if (pakwhich)
+ {
+ // The next .pk3 file is before the next .pk3dir
+ char *pakfile = FS_BuildOSPath(path, dir, pakfiles[pakfilesi]);
+ if ((pak = FS_LoadZipFile(pakfile, pakfiles[pakfilesi])) == 0)
+ {
+ // This isn't a .pk3! Next!
+ pakfilesi++;
+ continue;
+ }
+
+ Q_strncpyz(pak->pakPathname, curpath, sizeof(pak->pakPathname));
+ // store the game name for downloading
+ Q_strncpyz(pak->pakGamename, dir, sizeof(pak->pakGamename));
+
+ fs_packFiles += pak->numfiles;
+
+ searchpath_t *search = static_cast<searchpath_t *>(Z_Malloc(sizeof(searchpath_t)));
+ search->pack = pak;
+ search->next = fs_searchpaths;
+ fs_searchpaths = search;
+
+ pak->onlyPrimary = false;
+ pak->onlyAlternate = false;
+ for (int i = 0; i < numPairs; ++i)
+ {
+ if (lengths[i][0] && !Q_stricmpn(pak->pakBasename, prefixes[i][0], lengths[i][0]))
+ {
+ pak->onlyPrimary = true;
+ break;
+ }
+ else if (lengths[i][1] &&
+ !Q_stricmpn(pak->pakBasename, prefixes[i][1], lengths[i][1]))
+ {
+ pak->onlyAlternate = true;
+ break;
+ }
+ }
+
+ pak->primaryVersion = nullptr;
+ pakfilesi++;
+ }
+ else
+ {
+ // The next .pk3dir is before the next .pk3 file
+ // But wait, this could be any directory, we're filtering to only ending with ".pk3dir"
+ // here.
+ int len = strlen(pakdirs[pakdirsi]);
+ if (!FS_IsExt(pakdirs[pakdirsi], ".pk3dir", len))
+ {
+ // This isn't a .pk3dir! Next!
+ pakdirsi++;
+ continue;
+ }
+
+ char *pakfile = FS_BuildOSPath(path, dir, pakdirs[pakdirsi]);
+
+ // add the directory to the search path
+ searchpath_t *search = static_cast<searchpath_t *>(Z_Malloc(sizeof(searchpath_t)));
+ search->dir = static_cast<directory_t *>(Z_Malloc(sizeof(*search->dir)));
+
+ Q_strncpyz(search->dir->path, curpath, sizeof(search->dir->path)); // c:\quake3\baseq3
+ Q_strncpyz(search->dir->fullpath, pakfile,
+ sizeof(search->dir->fullpath)); // c:\quake3\baseq3\mypak.pk3dir
+ Q_strncpyz(search->dir->gamedir, pakdirs[pakdirsi],
+ sizeof(search->dir->gamedir)); // mypak.pk3dir
+
+ search->next = fs_searchpaths;
+ fs_searchpaths = search;
+
+ pakdirsi++;
+ }
+ }
+
+ // done
+ Sys_FreeFileList(pakfiles);
+ Sys_FreeFileList(pakdirs);
+
+ if (numPairs > 0)
+ {
+ int bnlengths[2];
+ for (searchpath_t *search = fs_searchpaths; search != otherSearchpaths;
+ search = search->next)
+ {
+ if (!(search->pack && search->pack->onlyPrimary))
+ {
+ continue;
+ }
+
+ bnlengths[0] = (int)strlen(search->pack->pakBasename);
+ for (searchpath_t *srch = fs_searchpaths; srch != otherSearchpaths; srch = srch->next)
+ {
+ if (!(srch->pack && srch->pack->onlyAlternate))
+ {
+ continue;
+ }
+
+ bnlengths[1] = (int)strlen(srch->pack->pakBasename);
+ for (int i = 0; i < numPairs; ++i)
+ {
+ if (lengths[i][0] && lengths[i][1] && bnlengths[0] >= lengths[i][0] &&
+ bnlengths[1] >= lengths[i][1] &&
+ !Q_stricmp(search->pack->pakBasename + lengths[i][0],
+ srch->pack->pakBasename + lengths[i][1]))
+ {
+ srch->pack->primaryVersion = search->pack;
+ break;
+ }
+ }
+ }
+ }
+ }
+
+ //
+ // add the directory to the search path
+ //
+ searchpath_t *search = static_cast<searchpath_t *>(Z_Malloc(sizeof(searchpath_t)));
+ search->dir = static_cast<directory_t *>(Z_Malloc(sizeof(*search->dir)));
+
+ Q_strncpyz(search->dir->path, path, sizeof(search->dir->path));
+ Q_strncpyz(search->dir->fullpath, curpath, sizeof(search->dir->fullpath));
+ Q_strncpyz(search->dir->gamedir, dir, sizeof(search->dir->gamedir));
+
+ search->next = fs_searchpaths;
+ fs_searchpaths = search;
+}
+
+/*
+================
+FS_CheckDirTraversal
+
+Check whether the string contains stuff like "../" to prevent directory traversal bugs
+and return true if it does.
+================
+*/
+
+bool FS_CheckDirTraversal(const char *checkdir)
+{
+ if (strstr(checkdir, "../") || strstr(checkdir, "..\\")) return true;
+
+ return false;
+}
+
+/*
+================
+FS_ComparePaks
+
+----------------
+dlstring == true
+
+Returns a list of pak files that we should download from the server. They all get stored
+in the current gamedir and an FS_Restart will be fired up after we download them all.
+
+The string is the format:
+
+@remotename@localname [repeat]
+
+static int fs_numServerReferencedPaks;
+static int fs_serverReferencedPaks[MAX_SEARCH_PATHS];
+static char *fs_serverReferencedPakNames[MAX_SEARCH_PATHS];
+
+----------------
+dlstring == false
+
+we are not interested in a download string format, we want something human-readable
+(this is used for diagnostics while connecting to a pure server)
+
+================
+*/
+bool FS_ComparePaks(char *neededpaks, int len, bool dlstring)
+{
+ if (!fs_numServerReferencedPaks)
+ {
+ return false; // Server didn't send any pack information along
+ }
+
+ char *origpos = neededpaks;
+ *neededpaks = '\0';
+
+ bool havepak = false;
+ for (int i = 0; i < fs_numServerReferencedPaks; i++)
+ {
+ // Ok, see if we have this pak file
+ havepak = false;
+
+ // Make sure the server cannot make us write to non-quake3 directories.
+ if (FS_CheckDirTraversal(fs_serverReferencedPakNames[i]))
+ {
+ Com_Printf("WARNING: Invalid download name %s\n", fs_serverReferencedPakNames[i]);
+ continue;
+ }
+
+ for (searchpath_t *sp = fs_searchpaths; sp; sp = sp->next)
+ {
+ if (sp->pack && sp->pack->checksum == fs_serverReferencedPaks[i])
+ {
+ havepak = true; // This is it!
+ break;
+ }
+ }
+
+ if (!havepak && fs_serverReferencedPakNames[i] && *fs_serverReferencedPakNames[i])
+ {
+ // Don't got it
+ if (dlstring)
+ {
+ // We need this to make sure we won't hit the end of the buffer or the server could
+ // overwrite non-pk3 files on clients by writing so much crap into neededpaks that
+ // Q_strcat cuts off the .pk3 extension.
+
+ origpos += strlen(origpos);
+
+ // Remote name
+ Q_strcat(neededpaks, len, "@");
+ Q_strcat(neededpaks, len, fs_serverReferencedPakNames[i]);
+ Q_strcat(neededpaks, len, ".pk3");
+
+ // Local name
+ Q_strcat(neededpaks, len, "@");
+ // Do we have one with the same name?
+ if (FS_SV_FileExists(va("%s.pk3", fs_serverReferencedPakNames[i])))
+ {
+ char st[MAX_ZPATH];
+ // We already have one called this, we need to download it to another name
+ // Make something up with the checksum in it
+ Com_sprintf(st, sizeof(st), "%s.%08x.pk3", fs_serverReferencedPakNames[i],
+ fs_serverReferencedPaks[i]);
+ Q_strcat(neededpaks, len, st);
+ }
+ else
+ {
+ Q_strcat(neededpaks, len, fs_serverReferencedPakNames[i]);
+ Q_strcat(neededpaks, len, ".pk3");
+ }
+
+ // Find out whether it might have overflowed the buffer and don't add this file to
+ // the
+ // list if that is the case.
+ if (strlen(origpos) + (origpos - neededpaks) >= (len - 1))
+ {
+ *origpos = '\0';
+ break;
+ }
+ }
+ else
+ {
+ Q_strcat(neededpaks, len, fs_serverReferencedPakNames[i]);
+ Q_strcat(neededpaks, len, ".pk3");
+ // Do we have one with the same name?
+ if (FS_SV_FileExists(va("%s.pk3", fs_serverReferencedPakNames[i])))
+ {
+ Q_strcat(neededpaks, len, " (local file exists with wrong checksum)");
+ }
+ Q_strcat(neededpaks, len, "\n");
+ }
+ }
+ }
+
+ if (*neededpaks)
+ {
+ return true;
+ }
+
+ return false; // We have them all
+}
+
+/*
+================
+FS_Shutdown
+
+Frees all resources.
+================
+*/
+
+void FS_Shutdown(bool closemfp)
+{
+ for (int i = 0; i < MAX_FILE_HANDLES; i++)
+ {
+ if (fsh[i].fileSize) FS_FCloseFile(i);
+ }
+
+ searchpath_t *next;
+ // free everything
+ for (auto p = fs_searchpaths; p; p = next)
+ {
+ next = p->next;
+ if (p->pack) FS_FreePak(p->pack);
+ if (p->dir) Z_Free(p->dir);
+ Z_Free(p);
+ }
+
+ // Any FS_ calls will now be an error until reinitialized
+ fs_searchpaths = nullptr;
+
+ Cmd_RemoveCommand("path");
+ Cmd_RemoveCommand("dir");
+ Cmd_RemoveCommand("fdir");
+ Cmd_RemoveCommand("touchFile");
+ Cmd_RemoveCommand("which");
+
+#ifdef FS_MISSING
+ if (closemfp)
+ {
+ fclose(missingFiles);
+ }
+#endif
+}
+
+/*
+================
+FS_ReorderPurePaks
+NOTE TTimo: the reordering that happens here is not reflected in the cvars (\cvarlist *pak*)
+ this can lead to misleading situations, see
+https://zerowing.idsoftware.com/bugzilla/show_bug.cgi?id=540
+================
+*/
+static void FS_ReorderPurePaks(void)
+{
+ // do this before fs_numServerPaks check?
+ fs_reordered = false;
+
+ // only relevant when connected to pure server
+ if (!fs_numServerPaks) return;
+
+ // we insert in order at the beginning of the list
+ auto p_insert_index = &fs_searchpaths;
+ for (int i = 0; i < fs_numServerPaks; i++)
+ {
+ // track the pointer-to-current-item
+ auto p_previous = p_insert_index;
+ for (auto s = *p_insert_index; s; s = s->next)
+ {
+ // the part of the list before p_insert_index has been sorted already
+ if (s->pack && fs_serverPaks[i] == s->pack->checksum)
+ {
+ fs_reordered = true;
+
+ // move this element to the insert list
+ *p_previous = s->next;
+ s->next = *p_insert_index;
+ *p_insert_index = s;
+
+ // increment insert list
+ p_insert_index = &s->next;
+
+ // iterate to next server pack
+ break;
+ }
+ p_previous = &s->next;
+ }
+ }
+}
+
+/*
+================
+FS_Startup
+================
+*/
+static void FS_Startup(const char *gameName)
+{
+ Com_Printf("----- FS_Startup -----\n");
+ fs_packFiles = 0;
+
+ fs_debug = Cvar_Get("fs_debug", "0", 0);
+ fs_basepath = Cvar_Get("fs_basepath", Sys_DefaultInstallPath(), CVAR_INIT | CVAR_PROTECTED);
+ fs_basegame = Cvar_Get("fs_basegame", BASEGAME, CVAR_INIT);
+
+ const char *homePath = Sys_DefaultHomePath();
+ if (!homePath || !homePath[0])
+ {
+ homePath = fs_basepath->string;
+ }
+
+ fs_homepath = Cvar_Get("fs_homepath", homePath, CVAR_INIT | CVAR_PROTECTED);
+ fs_gamedirvar = Cvar_Get("fs_game", BASEGAME, CVAR_INIT | CVAR_SYSTEMINFO);
+
+#ifdef DEDICATED
+ // add search path elements in reverse priority order
+ if (fs_basepath->string[0])
+ FS_AddGameDirectory(fs_basepath->string, gameName);
+
+ // NOTE: same filtering below for mods and basegame
+ if (fs_homepath->string[0] && Q_stricmp(fs_homepath->string,fs_basepath->string))
+ {
+ FS_CreatePath(fs_homepath->string);
+ FS_AddGameDirectory(fs_homepath->string, gameName);
+ }
+
+ // check for additional base game so mods can be based upon other mods
+ if ( fs_basegame->string[0] && Q_stricmp( fs_basegame->string, gameName ) )
+ {
+ if (fs_basepath->string[0])
+ FS_AddGameDirectory(fs_basepath->string, fs_basegame->string);
+ if (fs_homepath->string[0] && Q_stricmp(fs_homepath->string,fs_basepath->string))
+ FS_AddGameDirectory(fs_homepath->string, fs_basegame->string);
+ }
+
+ // check for additional game folder for mods
+ if ( fs_gamedirvar->string[0] && Q_stricmp( fs_gamedirvar->string, gameName ) )
+ {
+ if (fs_basepath->string[0])
+ FS_AddGameDirectory(fs_basepath->string, fs_gamedirvar->string);
+ if (fs_homepath->string[0] && Q_stricmp(fs_homepath->string,fs_basepath->string))
+ FS_AddGameDirectory(fs_homepath->string, fs_gamedirvar->string);
+ }
+
+#else
+
+ // add search path elements in reverse priority order
+ if (fs_basepath->string[0])
+ {
+ FS_AddGameDirectory(fs_basepath->string, "base");
+ }
+
+#ifdef __APPLE__
+ // Make MacOSX also include the base path included with the .app bundle
+ fs_apppath = Cvar_Get("fs_apppath", Sys_DefaultAppPath(), CVAR_INIT | CVAR_PROTECTED);
+ if (fs_apppath->string[0])
+ {
+ FS_AddGameDirectory(fs_apppath->string, "base");
+ }
+#endif
+
+ // NOTE: same filtering below for mods and basegame
+ if (fs_homepath->string[0] && Q_stricmp(fs_homepath->string, fs_basepath->string))
+ {
+ FS_CreatePath(fs_homepath->string);
+ FS_AddGameDirectory(fs_homepath->string, "base");
+ }
+
+ // check for additional base game so mods can be based upon other mods
+ if (fs_basegame->string[0] && Q_stricmp(fs_basegame->string, gameName))
+ {
+ if (fs_basepath->string[0])
+ FS_AddGameDirectory(fs_basepath->string, fs_basegame->string);
+ if (fs_homepath->string[0] && Q_stricmp(fs_homepath->string, fs_basepath->string))
+ FS_AddGameDirectory(fs_homepath->string, fs_basegame->string);
+ }
+
+ // check for additional game folder for mods
+ if (fs_gamedirvar->string[0] && Q_stricmp(fs_gamedirvar->string, gameName))
+ {
+ if (fs_basepath->string[0])
+ FS_AddGameDirectory(fs_basepath->string, fs_gamedirvar->string);
+ if (fs_homepath->string[0] && Q_stricmp(fs_homepath->string, fs_basepath->string))
+ FS_AddGameDirectory(fs_homepath->string, fs_gamedirvar->string);
+ }
+
+ // NOTE: same filtering below for mods and basegame
+ if (fs_homepath->string[0] && Q_stricmp(fs_homepath->string, fs_basepath->string))
+ {
+ FS_CreatePath(fs_homepath->string);
+ FS_AddGameDirectory(fs_homepath->string, gameName);
+ }
+
+ // add search path elements in reverse priority order
+ if (fs_basepath->string[0])
+ {
+ FS_AddGameDirectory(fs_basepath->string, gameName);
+ }
+
+#ifdef __APPLE__
+ // Make MacOSX also include the base path included with the .app bundle
+ fs_apppath = Cvar_Get("fs_apppath", Sys_DefaultAppPath(), CVAR_INIT | CVAR_PROTECTED);
+ if (fs_apppath->string[0])
+ {
+ FS_AddGameDirectory(fs_apppath->string, gameName);
+ }
+#endif
+#endif
+
+ // add our commands
+ Cmd_AddCommand("path", FS_Path_f);
+ Cmd_AddCommand("dir", FS_Dir_f);
+ Cmd_AddCommand("fdir", FS_NewDir_f);
+ Cmd_AddCommand("touchFile", FS_TouchFile_f);
+ Cmd_AddCommand("which", FS_Which_f);
+
+ // reorder the pure pk3 files according to server order
+ FS_ReorderPurePaks();
+
+ // print the current search paths
+ FS_Path_f();
+
+ // We just loaded, it's not modified
+ fs_gamedirvar->modified = false;
+
+ Com_Printf("----------------------\n");
+
+#ifdef FS_MISSING
+ if (missingFiles == nullptr)
+ {
+ missingFiles = Sys_FOpen("\\missing.txt", "ab");
+ }
+#endif
+
+ Com_Printf("%d files in pk3 files\n", fs_packFiles);
+}
+
+/*
+=====================
+FS_LoadedPakChecksums
+
+Returns a space separated string containing the checksums of all loaded pk3 files.
+Servers with sv_pure set will get this string and pass it to clients.
+=====================
+*/
+const char *FS_LoadedPakChecksums(bool alternate)
+{
+ static char info[BIG_INFO_STRING];
+ info[0] = 0;
+
+ for (auto search = fs_searchpaths; search; search = search->next)
+ {
+ // is the element a pak file?
+ if (!search->pack) continue;
+ if ((alternate && search->pack->onlyPrimary) || (!alternate && search->pack->onlyAlternate))
+ continue;
+ Q_strcat(info, sizeof(info), va("%i ", search->pack->checksum));
+ }
+
+ return info;
+}
+
+/*
+=====================
+FS_LoadedPakNames
+
+Returns a space separated string containing the names of all loaded pk3 files.
+Servers with sv_pure set will get this string and pass it to clients.
+=====================
+*/
+const char *FS_LoadedPakNames(bool alternate)
+{
+ static char info[BIG_INFO_STRING];
+ info[0] = 0;
+
+ for (auto search = fs_searchpaths; search; search = search->next)
+ {
+ // is the element a pak file?
+ if (!search->pack) continue;
+ if ((alternate && search->pack->onlyPrimary) || (!alternate && search->pack->onlyAlternate))
+ continue;
+ if (info[0]) Q_strcat(info, sizeof(info), " ");
+ Q_strcat(info, sizeof(info), search->pack->pakBasename);
+ }
+
+ return info;
+}
+
+/*
+=====================
+FS_LoadedPakPureChecksums
+
+Returns a space separated string containing the pure checksums of all loaded pk3 files.
+Servers with sv_pure use these checksums to compare with the checksums the clients send
+back to the server.
+=====================
+*/
+const char *FS_LoadedPakPureChecksums(bool alternate)
+{
+ static char info[BIG_INFO_STRING];
+ info[0] = 0;
+
+ for (auto search = fs_searchpaths; search; search = search->next)
+ {
+ // is the element a pak file?
+ if (!search->pack) continue;
+ if ((alternate && search->pack->onlyPrimary) || (!alternate && search->pack->onlyAlternate))
+ continue;
+ Q_strcat(info, sizeof(info), va("%i ", search->pack->pure_checksum));
+ }
+
+ return info;
+}
+
+/*
+=====================
+FS_ReferencedPakChecksums
+
+Returns a space separated string containing the checksums of all referenced pk3 files.
+The server will send this to the clients so they can check which files should be auto-downloaded.
+=====================
+*/
+const char *FS_ReferencedPakChecksums(bool alternate)
+{
+ static char info[BIG_INFO_STRING];
+ info[0] = 0;
+
+ for (auto search = fs_searchpaths; search; search = search->next)
+ {
+ // is the element a pak file?
+ if (search->pack)
+ {
+ if ((alternate and search->pack->onlyPrimary) or
+ (!alternate and search->pack->onlyAlternate))
+ continue;
+
+ if (search->pack->referenced or
+ (search->pack->primaryVersion and search->pack->primaryVersion->referenced) or
+ (*fs_gamedirvar->string and Q_stricmp(fs_gamedirvar->string, BASEGAME) and
+ Q_stricmp(search->pack->pakGamename, fs_gamedirvar->string) == 0))
+ {
+ Q_strcat(info, sizeof(info), va("%i ", search->pack->checksum));
+ }
+ }
+ }
+
+ return info;
+}
+
+/*
+=====================
+FS_ReferencedPakPureChecksums
+
+Returns a space separated string containing the pure checksums of all referenced pk3 files.
+Servers with sv_pure set will get this string back from clients for pure validation
+
+The string has a specific order, "cgame ui @ ref1 ref2 ref3 ..."
+=====================
+*/
+const char *FS_ReferencedPakPureChecksums(void)
+{
+ static char info[BIG_INFO_STRING];
+ info[0] = 0;
+
+ int checksum = fs_checksumFeed;
+ int numPaks = 0;
+ for (int nFlags = FS_CGAME_REF; nFlags; nFlags = nFlags >> 1)
+ {
+ if (nFlags & FS_GENERAL_REF)
+ {
+ // add a delimter between must haves and general refs
+ // Q_strcat(info, sizeof(info), "@ ");
+ info[strlen(info) + 1] = '\0';
+ info[strlen(info) + 2] = '\0';
+ info[strlen(info)] = '@';
+ info[strlen(info)] = ' ';
+ }
+ for (auto search = fs_searchpaths; search; search = search->next)
+ {
+ // is the element a pak file and has it been referenced based on flag?
+ if (search->pack && (search->pack->referenced & nFlags))
+ {
+ Q_strcat(info, sizeof(info), va("%i ", search->pack->pure_checksum));
+ if (nFlags & (FS_CGAME_REF | FS_UI_REF)) break;
+
+ checksum ^= search->pack->pure_checksum;
+ numPaks++;
+ }
+ }
+ }
+ // last checksum is the encoded number of referenced pk3s
+ checksum ^= numPaks;
+ Q_strcat(info, sizeof(info), va("%i ", checksum));
+
+ return info;
+}
+
+/*
+=====================
+FS_ReferencedPakNames
+
+Returns a space separated string containing the names of all referenced pk3 files.
+The server will send this to the clients so they can check which files should be auto-downloaded.
+=====================
+*/
+const char *FS_ReferencedPakNames(bool alternate)
+{
+ static char info[BIG_INFO_STRING];
+ info[0] = 0;
+
+ // we want to return ALL pk3's from the fs_game path
+ // and referenced one's from base
+ for (auto search = fs_searchpaths; search; search = search->next)
+ {
+ // is the element a pak file?
+ if (search->pack)
+ {
+ if ((alternate && search->pack->onlyPrimary) ||
+ (!alternate && search->pack->onlyAlternate))
+ continue;
+
+ if (search->pack->referenced ||
+ (search->pack->primaryVersion && search->pack->primaryVersion->referenced) ||
+ (fs_gamedirvar->string[0] && Q_stricmp(fs_gamedirvar->string, BASEGAME) &&
+ !Q_stricmp(search->pack->pakGamename, fs_gamedirvar->string)))
+ {
+ if (*info) Q_strcat(info, sizeof(info), " ");
+
+ Q_strcat(info, sizeof(info), search->pack->pakGamename);
+ Q_strcat(info, sizeof(info), "/");
+ Q_strcat(info, sizeof(info), search->pack->pakBasename);
+ }
+ }
+ }
+
+ return info;
+}
+
+/*
+=====================
+FS_ClearPakReferences
+=====================
+*/
+void FS_ClearPakReferences(int flags)
+{
+ if (!flags) flags = -1;
+
+ for (auto search = fs_searchpaths; search; search = search->next)
+ {
+ // is the element a pak file and has it been referenced?
+ if (search->pack) search->pack->referenced &= ~flags;
+ }
+}
+
+/*
+=====================
+FS_PureServerSetLoadedPaks
+
+If the string is empty, all data sources will be allowed.
+If not empty, only pk3 files that match one of the space
+separated checksums will be checked for files, with the
+exception of .cfg and .dat files.
+=====================
+*/
+void FS_PureServerSetLoadedPaks(const char *pakSums, const char *pakNames)
+{
+ Cmd_TokenizeString(pakSums);
+
+ int c = Cmd_Argc();
+ if (c > MAX_SEARCH_PATHS) c = MAX_SEARCH_PATHS;
+
+ fs_numServerPaks = c;
+
+ for (int i = 0; i < c; i++) fs_serverPaks[i] = atoi(Cmd_Argv(i));
+
+ if (fs_numServerPaks)
+ {
+ Com_DPrintf("Connected to a pure server.\n");
+ }
+ else if (fs_reordered)
+ {
+ // force a restart to make sure the search order will be correct
+ Com_DPrintf("FS search reorder is required\n");
+ FS_Restart(fs_checksumFeed);
+ return;
+ }
+
+ for (int i = 0; i < c; i++)
+ {
+ if (fs_serverPakNames[i]) Z_Free(fs_serverPakNames[i]);
+ fs_serverPakNames[i] = nullptr;
+ }
+
+ if (pakNames && pakNames[0])
+ {
+ Cmd_TokenizeString(pakNames);
+
+ int d = Cmd_Argc();
+ if (d > MAX_SEARCH_PATHS) d = MAX_SEARCH_PATHS;
+
+ for (int i = 0; i < d; i++) fs_serverPakNames[i] = CopyString(Cmd_Argv(i));
+ }
+}
+
+/*
+=====================
+FS_PureServerSetReferencedPaks
+
+The checksums and names of the pk3 files referenced at the server
+are sent to the client and stored here. The client will use these
+checksums to see if any pk3 files need to be auto-downloaded.
+=====================
+*/
+void FS_PureServerSetReferencedPaks(const char *pakSums, const char *pakNames)
+{
+ Cmd_TokenizeString(pakSums);
+
+ unsigned c = Cmd_Argc();
+ if (c > MAX_SEARCH_PATHS) c = MAX_SEARCH_PATHS;
+
+ for (unsigned i = 0; i < c; i++) fs_serverReferencedPaks[i] = atoi(Cmd_Argv(i));
+
+ for (unsigned i = 0; i < ARRAY_LEN(fs_serverReferencedPakNames); i++)
+ {
+ if (fs_serverReferencedPakNames[i]) Z_Free(fs_serverReferencedPakNames[i]);
+ fs_serverReferencedPakNames[i] = nullptr;
+ }
+
+ unsigned d = 0;
+ if (pakNames && *pakNames)
+ {
+ Cmd_TokenizeString(pakNames);
+
+ d = Cmd_Argc();
+ if (d > c) d = c;
+
+ for (unsigned i = 0; i < d; i++) fs_serverReferencedPakNames[i] = CopyString(Cmd_Argv(i));
+ }
+
+ // ensure that there are as many checksums as there are pak names.
+ if (d < c) c = d;
+
+ fs_numServerReferencedPaks = c;
+}
+
+/*
+================
+FS_InitFilesystem
+
+Called only at inital startup, not when the filesystem
+is resetting due to a game change
+================
+*/
+void FS_InitFilesystem(void)
+{
+ // allow command line parms to override our defaults
+ // we have to specially handle this, because normal command
+ // line variable sets don't happen until after the filesystem
+ // has already been initialized
+ Com_StartupVariable("fs_basepath");
+ Com_StartupVariable("fs_homepath");
+ Com_StartupVariable("fs_game");
+ Com_StartupVariable("fs_pk3PrefixPairs");
+
+ if (!FS_FilenameCompare(Cvar_VariableString("fs_game"), BASEGAME)) Cvar_Set("fs_game", "");
+
+ // try to start up normally
+ FS_Startup(BASEGAME);
+
+ // if we can't find default.cfg, assume that the paths are
+ // busted and error out now, rather than getting an unreadable
+ // graphics screen when the font fails to load
+ if (FS_ReadFile("default.cfg", nullptr) <= 0)
+ {
+ Com_Error(ERR_FATAL, "Couldn't load default.cfg");
+ }
+
+ Q_strncpyz(lastValidBase, fs_basegame->string, sizeof(lastValidBase));
+ Q_strncpyz(lastValidGame, fs_gamedirvar->string, sizeof(lastValidGame));
+}
+
+/*
+================
+FS_Restart
+================
+*/
+void FS_Restart(int checksumFeed)
+{
+ // free anything we currently have loaded
+ FS_Shutdown(false);
+
+ // set the checksum feed
+ fs_checksumFeed = checksumFeed;
+
+ // clear pak references
+ FS_ClearPakReferences(0);
+
+ // try to start up normally
+ FS_Startup(BASEGAME);
+
+ // if we can't find default.cfg, assume that the paths are
+ // busted and error out now, rather than getting an unreadable
+ // graphics screen when the font fails to load
+ if (FS_ReadFile("default.cfg", nullptr) <= 0)
+ {
+ // this might happen when connecting to a pure server not using BASEGAME/pak0.pk3
+ // (for instance a TA demo server)
+ if (lastValidBase[0])
+ {
+ FS_PureServerSetLoadedPaks("", "");
+ Cvar_Set("fs_basegame", lastValidBase);
+ Cvar_Set("fs_game", lastValidGame);
+ lastValidBase[0] = lastValidGame[0] = '\0';
+ FS_Restart(checksumFeed);
+ Com_Error(ERR_DROP, "Invalid game folder");
+ return;
+ }
+ Com_Error(ERR_FATAL, "Couldn't load default.cfg");
+ }
+
+ if (Q_stricmp(fs_gamedirvar->string, lastValidGame))
+ {
+ // skip the autogen.cfg if "safe" is on the command line
+ if (!Com_SafeMode())
+ {
+ Cbuf_AddText("exec " Q3CONFIG_CFG "\n");
+ }
+ }
+
+ Q_strncpyz(lastValidBase, fs_basegame->string, sizeof(lastValidBase));
+ Q_strncpyz(lastValidGame, fs_gamedirvar->string, sizeof(lastValidGame));
+}
+
+/*
+=================
+FS_ConditionalRestart
+
+Restart if necessary
+Return true if restarting due to game directory changed, false otherwise
+=================
+*/
+bool FS_ConditionalRestart(int checksumFeed, bool disconnect)
+{
+ if (fs_gamedirvar->modified)
+ {
+ if (FS_FilenameCompare(lastValidGame, fs_gamedirvar->string) &&
+ (*lastValidGame || FS_FilenameCompare(fs_gamedirvar->string, BASEGAME)) &&
+ (*fs_gamedirvar->string || FS_FilenameCompare(lastValidGame, BASEGAME)))
+ {
+ Com_GameRestart(checksumFeed, disconnect);
+ return true;
+ }
+ fs_gamedirvar->modified = false;
+ }
+
+ if (checksumFeed != fs_checksumFeed)
+ FS_Restart(checksumFeed);
+
+ else if (fs_numServerPaks && !fs_reordered)
+ FS_ReorderPurePaks();
+
+ return false;
+}
+
+/*
+========================================================================================
+
+Handle based file calls for virtual machines
+
+========================================================================================
+*/
+
+int FS_FOpenFileByMode(const char *qpath, fileHandle_t *f, enum FS_Mode mode)
+{
+ int r;
+ bool sync = false;
+
+ switch (mode)
+ {
+ case FS_READ:
+ r = FS_FOpenFileRead(qpath, f, true);
+ break;
+
+ case FS_WRITE:
+ *f = FS_FOpenFileWrite(qpath);
+ r = 0;
+ if (*f == 0) r = -1;
+ break;
+
+ case FS_APPEND_SYNC:
+ sync = true;
+ // fall through
+
+ case FS_APPEND:
+ *f = FS_FOpenFileAppend(qpath);
+ r = 0;
+ if (*f == 0) r = -1;
+ break;
+
+ default:
+ Com_Error(ERR_FATAL, "FS_FOpenFileByMode: bad mode");
+ return -1;
+ }
+
+ if (!f) return r;
+
+ if (*f)
+ {
+ fsh[*f].fileSize = r;
+ }
+ fsh[*f].handleSync = sync;
+
+ return r;
+}
+
+int FS_FTell(fileHandle_t f)
+{
+ if (fsh[f].zipFile == true) return unztell(fsh[f].handleFiles.file.z);
+ return ftell(fsh[f].handleFiles.file.o);
+}
+
+void FS_Flush(fileHandle_t f)
+{
+ fflush(fsh[f].handleFiles.file.o);
+}
+
+void FS_FilenameCompletion(const char *dir, const char *ext, bool stripExt,
+ void (*callback)(const char *s), bool allowNonPureFilesOnDisk)
+{
+ int nfiles;
+ char filename[MAX_STRING_CHARS];
+ char **filenames = FS_ListFilteredFiles(dir, ext, nullptr, &nfiles, allowNonPureFilesOnDisk);
+
+ FS_SortFileList(filenames, nfiles);
+
+ for (int i = 0; i < nfiles; i++)
+ {
+ FS_ConvertPath(filenames[i]);
+ Q_strncpyz(filename, filenames[i], MAX_STRING_CHARS);
+ if (stripExt) COM_StripExtension(filename, filename, sizeof(filename));
+ callback(filename);
+ }
+ FS_FreeFileList(filenames);
+}
+
+const char *FS_GetCurrentGameDir(void)
+{
+ if (fs_gamedirvar->string[0]) return fs_gamedirvar->string;
+ return BASEGAME;
+}