You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
204 lines
4.3 KiB
204 lines
4.3 KiB
2 years ago
|
/*
|
||
|
xar.c -- Xash ARchives (XAR)
|
||
|
Copyright (C) 2023 Alibek Omarov
|
||
|
|
||
|
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 "port.h"
|
||
|
#include "build.h"
|
||
|
#include <stdlib.h>
|
||
|
#include <string.h>
|
||
|
#include <time.h>
|
||
|
#include <errno.h>
|
||
|
#include "filesystem.h"
|
||
|
#if XASH_POSIX
|
||
|
#include <sys/stat.h>
|
||
|
#include <dlfcn.h>
|
||
|
#define LoadLibrary( x ) dlopen( x, RTLD_NOW )
|
||
|
#define GetProcAddress( x, y ) dlsym( x, y )
|
||
|
#define FreeLibrary( x ) dlclose( x )
|
||
|
#elif XASH_WIN32
|
||
|
#include <windows.h>
|
||
|
#endif
|
||
|
|
||
|
static void *g_hModule;
|
||
|
static FSAPI g_pfnGetFSAPI;
|
||
|
static fs_api_t g_fs;
|
||
|
static fs_globals_t *g_nullglobals;
|
||
|
|
||
|
static qboolean LoadFilesystem( void )
|
||
|
{
|
||
|
#if 0
|
||
|
string cwd;
|
||
|
|
||
|
if( getcwd( cwd, sizeof( cwd )) == NULL )
|
||
|
{
|
||
|
printf( "getcwd() failed: %s\n", strerror( errno ));
|
||
|
return false;
|
||
|
}
|
||
|
#endif
|
||
|
g_hModule = LoadLibrary( "filesystem_stdio." OS_LIB_EXT );
|
||
|
if( !g_hModule )
|
||
|
return false;
|
||
|
|
||
|
g_pfnGetFSAPI = (void*)GetProcAddress( g_hModule, GET_FS_API );
|
||
|
if( !g_pfnGetFSAPI )
|
||
|
return false;
|
||
|
|
||
|
if( !g_pfnGetFSAPI( FS_API_VERSION, &g_fs, &g_nullglobals, NULL ))
|
||
|
return false;
|
||
|
|
||
|
// g_fs.InitStdio( true, cwd, );
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
static void FS_CreatePath( char *path )
|
||
|
{
|
||
|
char *ofs, save;
|
||
|
|
||
|
for( ofs = path + 1; *ofs; ofs++ )
|
||
|
{
|
||
|
if( *ofs == '/' || *ofs == '\\' )
|
||
|
{
|
||
|
// create the directory
|
||
|
save = *ofs;
|
||
|
*ofs = 0;
|
||
|
_mkdir( path );
|
||
|
*ofs = save;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void usage( const char *arg0 )
|
||
|
{
|
||
|
printf( "%s: <action> [option...] <file>\n", arg0 );
|
||
|
puts( "XAR is a simple frontend to Xash3D FWGS's filesystem_stdio library" );
|
||
|
puts( "that allows interacting with archive types supported by it" );
|
||
|
puts( "Options:" );
|
||
|
puts( "\tx\t\teXtract the archive" );
|
||
|
puts( "\tt\t\tlisT the archive" );
|
||
|
// TODO: make an interface for modifying
|
||
|
// puts( "\tc\t\tCreate the archive" );
|
||
|
// puts( "\tu\t\tUpdate the archive" );
|
||
|
// puts( "\tr\t\tRemove from the archive" );
|
||
|
|
||
|
puts( "Extract and list options:" );
|
||
|
puts( "\t-wads\tauto-mount WADs inside archives" );
|
||
|
exit( 1 );
|
||
|
}
|
||
|
|
||
|
int main( int argc, char **argv )
|
||
|
{
|
||
|
const char *filename;
|
||
|
search_t *search;
|
||
|
char action;
|
||
|
int i, flags = FS_NOWRITE_PATH | FS_SKIP_ARCHIVED_WADS;
|
||
|
|
||
|
if( argc < 3 )
|
||
|
usage( argv[0] );
|
||
|
|
||
|
if( !LoadFilesystem())
|
||
|
{
|
||
|
puts( "Can't load filesystem_stdio!" );
|
||
|
return 2;
|
||
|
}
|
||
|
|
||
|
action = argv[1][0];
|
||
|
if( action != 'x' && action != 't' )
|
||
|
{
|
||
|
printf( "Unknown action: %c\n", action );
|
||
|
usage( argv[0] );
|
||
|
}
|
||
|
|
||
|
for( i = 2; i < argc - 1; i++ )
|
||
|
{
|
||
|
if( !strcmp( argv[i], "-wads" ))
|
||
|
ClearBits( flags, FS_SKIP_ARCHIVED_WADS );
|
||
|
else
|
||
|
{
|
||
|
printf( "Unknown option: %s\n", argv[i] );
|
||
|
usage( argv[0] );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
filename = argv[argc-1];
|
||
|
|
||
|
if( g_fs.MountArchive_Fullpath( filename, flags ) == NULL )
|
||
|
{
|
||
|
printf( "Can't mount %s\n", filename );
|
||
|
return 3;
|
||
|
}
|
||
|
|
||
|
// suboptimal, but that's what is available with current FS API
|
||
|
search = g_fs.Search( "*", false, false );
|
||
|
if( !search )
|
||
|
{
|
||
|
printf( "Can't find any files in %s\n", filename );
|
||
|
return 4;
|
||
|
}
|
||
|
|
||
|
if( action == 't' )
|
||
|
{
|
||
|
puts( "File list:" );
|
||
|
for( i = 0; i < search->numfilenames; i++ )
|
||
|
{
|
||
|
printf( "\t%s\n", search->filenames[i] );
|
||
|
}
|
||
|
}
|
||
|
else if( action == 'x' )
|
||
|
{
|
||
|
for( i = 0; i < search->numfilenames; i++ )
|
||
|
{
|
||
|
struct stat st;
|
||
|
char *path = search->filenames[i];
|
||
|
file_t *from;
|
||
|
char buffer[4096];
|
||
|
FILE *to;
|
||
|
|
||
|
if(( from = g_fs.Open( path, "rb", false )) == NULL )
|
||
|
{
|
||
|
printf( "Can't open %s in archive, skipping...\n", path );
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
FS_CreatePath( path );
|
||
|
if( stat( path, &st ) == 0 )
|
||
|
{
|
||
|
printf( "Will not overwrite existing %s file\n", path );
|
||
|
g_fs.Close( from );
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
if(( to = fopen( search->filenames[i], "wb" )) == NULL )
|
||
|
{
|
||
|
printf( "fopen() failed: %s\n", strerror( errno ));
|
||
|
g_fs.Close( from );
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
printf( "Unpacking %s...\n", path );
|
||
|
while( !g_fs.Eof( from ))
|
||
|
{
|
||
|
size_t len = g_fs.Read( from, buffer, sizeof( buffer ));
|
||
|
fwrite( buffer, 1, len, to );
|
||
|
}
|
||
|
fclose( to );
|
||
|
g_fs.Close( from );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
free( search );
|
||
|
|
||
|
return 0;
|
||
|
}
|