/* dir.c - directory operations Copyright (C) 2022 Alibek Omarov, Velaron 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 3 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. */ #include #include #include #if XASH_POSIX #include #endif #include #include #include "port.h" #include "filesystem_internal.h" #include "crtlib.h" #include "xash3d_mathlib.h" #include "common/com_strings.h" static void FS_Close_DIR( searchpath_t *search ) { } static void FS_PrintInfo_DIR( searchpath_t *search, char *dst, size_t size ) { Q_strncpy( dst, search->filename, size ); } static int FS_FindFile_DIR( searchpath_t *search, const char *path ) { char netpath[MAX_SYSPATH]; Q_snprintf( netpath, sizeof( netpath ), "%s%s", search->filename, path ); if( FS_SysFileExists( netpath, !( search->flags & FS_CUSTOM_PATH ) ) ) return 0; return -1; } static void FS_Search_DIR( searchpath_t *search, stringlist_t *list, const char *pattern, int caseinsensitive ) { string netpath, temp; stringlist_t dirlist; const char *slash, *backslash, *colon, *separator; int basepathlength, dirlistindex, resultlistindex; char *basepath; slash = Q_strrchr( pattern, '/' ); backslash = Q_strrchr( pattern, '\\' ); colon = Q_strrchr( pattern, ':' ); separator = Q_max( slash, backslash ); separator = Q_max( separator, colon ); basepathlength = separator ? (separator + 1 - pattern) : 0; basepath = Mem_Calloc( fs_mempool, basepathlength + 1 ); if( basepathlength ) memcpy( basepath, pattern, basepathlength ); basepath[basepathlength] = '\0'; Q_snprintf( netpath, sizeof( netpath ), "%s%s", search->filename, basepath ); stringlistinit( &dirlist ); listdirectory( &dirlist, netpath, caseinsensitive ); Q_strncpy( temp, basepath, sizeof( temp ) ); for( dirlistindex = 0; dirlistindex < dirlist.numstrings; dirlistindex++ ) { Q_strncpy( &temp[basepathlength], dirlist.strings[dirlistindex], sizeof( temp ) - basepathlength ); if( matchpattern( temp, (char *)pattern, true ) ) { for( resultlistindex = 0; resultlistindex < list->numstrings; resultlistindex++ ) { if( !Q_strcmp( list->strings[resultlistindex], temp ) ) break; } if( resultlistindex == list->numstrings ) stringlistappend( list, temp ); } } stringlistfreecontents( &dirlist ); Mem_Free( basepath ); } static int FS_FileTime_DIR( searchpath_t *search, const char *filename ) { char path[MAX_SYSPATH]; Q_snprintf( path, sizeof( path ), "%s%s", search->filename, filename ); return FS_SysFileTime( path ); } static file_t *FS_OpenFile_DIR( searchpath_t *search, const char *filename, const char *mode, int pack_ind ) { char path[MAX_SYSPATH]; Q_snprintf( path, sizeof( path ), "%s%s", search->filename, filename ); return FS_SysOpen( path, mode ); } void FS_InitDirectorySearchpath( searchpath_t *search, const char *path, int flags ) { memset( search, 0, sizeof( searchpath_t )); Q_strncpy( search->filename, path, sizeof( search->filename )); search->type = SEARCHPATH_PLAIN; search->flags = flags; search->printinfo = FS_PrintInfo_DIR; search->close = FS_Close_DIR; search->openfile = FS_OpenFile_DIR; search->filetime = FS_FileTime_DIR; search->findfile = FS_FindFile_DIR; search->search = FS_Search_DIR; } qboolean FS_AddDir_Fullpath( const char *path, qboolean *already_loaded, int flags ) { searchpath_t *search; for( search = fs_searchpaths; search; search = search->next ) { if( search->type == SEARCHPATH_PLAIN && !Q_stricmp( search->filename, path )) { if( already_loaded ) *already_loaded = true; return true; } } if( already_loaded ) *already_loaded = false; search = (searchpath_t *)Mem_Calloc( fs_mempool, sizeof( searchpath_t )); FS_InitDirectorySearchpath( search, path, flags ); search->next = fs_searchpaths; fs_searchpaths = search; Con_Printf( "Adding directory: %s\n", path ); return true; }