643 lines
15 KiB
C
643 lines
15 KiB
C
/* -------------------------------------------------------------------------------
|
|
|
|
Copyright (C) 1999-2007 id Software, Inc. and contributors.
|
|
For a list of contributors, see the accompanying CONTRIBUTORS file.
|
|
|
|
This file is part of GtkRadiant.
|
|
|
|
GtkRadiant is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
GtkRadiant 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 GtkRadiant; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
----------------------------------------------------------------------------------
|
|
|
|
This code has been altered significantly from its original form, to support
|
|
several games based on the Quake III Arena engine, in the form of "Q3Map2."
|
|
|
|
------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
/* marker */
|
|
#define PATH_INIT_C
|
|
|
|
|
|
|
|
/* dependencies */
|
|
#include "vmap.h"
|
|
|
|
|
|
|
|
/* path support */
|
|
#define MAX_BASE_PATHS 10
|
|
#define MAX_GAME_PATHS 10
|
|
#define MAX_PAK_PATHS 200
|
|
|
|
char *homePath;
|
|
char installPath[ MAX_OS_PATH ];
|
|
|
|
int numBasePaths;
|
|
char *basePaths[ MAX_BASE_PATHS ];
|
|
int numGamePaths;
|
|
char *gamePaths[ MAX_GAME_PATHS ];
|
|
int numPakPaths;
|
|
char *pakPaths[ MAX_PAK_PATHS ];
|
|
char *homeBasePath = NULL;
|
|
|
|
|
|
/*
|
|
some of this code is based off the original q3map port from loki
|
|
and finds various paths. moved here from bsp.c for clarity.
|
|
*/
|
|
|
|
/*
|
|
PathLokiGetHomeDir()
|
|
gets the user's home dir (for ~/.q3a)
|
|
*/
|
|
|
|
char *LokiGetHomeDir( void ){
|
|
#ifndef Q_UNIX
|
|
return NULL;
|
|
#else
|
|
static char buf[ 4096 ];
|
|
struct passwd pw, *pwp;
|
|
char *home;
|
|
static char homeBuf[MAX_OS_PATH];
|
|
|
|
|
|
/* get the home environment variable */
|
|
home = getenv( "HOME" );
|
|
|
|
/* look up home dir in password database */
|
|
if(!home)
|
|
{
|
|
if ( getpwuid_r( getuid(), &pw, buf, sizeof( buf ), &pwp ) == 0 ) {
|
|
return pw.pw_dir;
|
|
}
|
|
}
|
|
|
|
snprintf( homeBuf, sizeof( homeBuf ), "%s/.", home );
|
|
|
|
/* return it */
|
|
return homeBuf;
|
|
#endif
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
PathLokiInitPaths()
|
|
initializes some paths on linux/os x
|
|
*/
|
|
|
|
void LokiInitPaths( char *argv0 ){
|
|
char *home;
|
|
|
|
if ( !homePath ) {
|
|
/* get home dir */
|
|
home = LokiGetHomeDir();
|
|
if ( home == NULL ) {
|
|
home = ".";
|
|
}
|
|
|
|
/* set home path */
|
|
homePath = home;
|
|
}
|
|
else{
|
|
home = homePath;
|
|
}
|
|
|
|
#ifndef Q_UNIX
|
|
/* this is kinda crap, but hey */
|
|
strcpy( installPath, "../" );
|
|
#else
|
|
|
|
char temp[ MAX_OS_PATH ];
|
|
char *path;
|
|
char *last;
|
|
qboolean found;
|
|
|
|
|
|
path = getenv( "PATH" );
|
|
|
|
/* do some path divining */
|
|
Q_strncpyz( temp, argv0, sizeof( temp ) );
|
|
if ( strrchr( temp, '/' ) ) {
|
|
argv0 = strrchr( argv0, '/' ) + 1;
|
|
}
|
|
else if ( path ) {
|
|
|
|
/*
|
|
This code has a special behavior when q3map2 is a symbolic link.
|
|
|
|
For each dir in ${PATH} (example: "/usr/bin", "/usr/local/bin" if ${PATH} == "/usr/bin:/usr/local/bin"),
|
|
it looks for "${dir}/q3map2" (file exists and is executable),
|
|
then it uses "dirname(realpath("${dir}/q3map2"))/../" as installPath.
|
|
|
|
So, if "/usr/bin/q3map2" is a symbolic link to "/opt/radiant/tools/q3map2",
|
|
it will find the installPath "/usr/share/radiant/",
|
|
so q3map2 will look for "/opt/radiant/baseq3" to find paks.
|
|
|
|
More precisely, it looks for "${dir}/${argv[0]}",
|
|
so if "/usr/bin/q3map2" is a symbolic link to "/opt/radiant/tools/q3map2",
|
|
and if "/opt/radiant/tools/q3ma2" is a symbolic link to "/opt/radiant/tools/q3map2.x86_64",
|
|
it will use "dirname("/opt/radiant/tools/q3map2.x86_64")/../" as path,
|
|
so it will use "/opt/radiant/" as installPath, which will be expanded later to "/opt/radiant/baseq3" to find paks.
|
|
*/
|
|
|
|
found = qfalse;
|
|
last = path;
|
|
|
|
/* go through each : segment of path */
|
|
while ( last[ 0 ] != '\0' && found == qfalse )
|
|
{
|
|
/* null out temp */
|
|
temp[ 0 ] = '\0';
|
|
|
|
/* find next chunk */
|
|
last = strchr( path, ':' );
|
|
if ( last == NULL ) {
|
|
last = path + strlen( path );
|
|
}
|
|
|
|
/* found home dir candidate */
|
|
if ( *path == '~' ) {
|
|
Q_strncpyz( temp, home, sizeof( temp ) );
|
|
path++;
|
|
}
|
|
|
|
/* concatenate */
|
|
if ( last > ( path + 1 ) ) {
|
|
// +1 hack: Q_strncat calls Q_strncpyz that expects a len including '\0'
|
|
// so that extraneous char will be rewritten by '\0', so it's ok.
|
|
// Also, in this case this extraneous char is always ':' or '\0', so it's ok.
|
|
Q_strncat( temp, sizeof( temp ), path, ( last - path + 1) );
|
|
Q_strcat( temp, sizeof( temp ), "/" );
|
|
}
|
|
Q_strcat( temp, sizeof( temp ), argv0 );
|
|
|
|
/* verify the path */
|
|
if ( access( temp, X_OK ) == 0 ) {
|
|
found = qtrue;
|
|
}
|
|
path = last + 1;
|
|
}
|
|
}
|
|
|
|
/* flake */
|
|
if ( realpath( temp, installPath ) ) {
|
|
/*
|
|
if "q3map2" is "/opt/radiant/tools/q3map2",
|
|
installPath is "/opt/radiant"
|
|
*/
|
|
*( strrchr( installPath, '/' ) ) = '\0';
|
|
*( strrchr( installPath, '/' ) ) = '\0';
|
|
}
|
|
#endif
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
CleanPath() - ydnar
|
|
cleans a dos path \ -> /
|
|
*/
|
|
|
|
void CleanPath( char *path ){
|
|
while ( *path )
|
|
{
|
|
if ( *path == '\\' ) {
|
|
*path = '/';
|
|
}
|
|
path++;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
GetGame() - ydnar
|
|
gets the game_t based on a -game argument
|
|
returns NULL if no match found
|
|
*/
|
|
|
|
game_t *GetGame( char *arg ){
|
|
int i;
|
|
|
|
|
|
/* dummy check */
|
|
if ( arg == NULL || arg[ 0 ] == '\0' ) {
|
|
return NULL;
|
|
}
|
|
|
|
/* joke */
|
|
if ( !Q_stricmp( arg, "quake1" ) ||
|
|
!Q_stricmp( arg, "quake2" ) ||
|
|
!Q_stricmp( arg, "unreal" ) ||
|
|
!Q_stricmp( arg, "ut2k3" ) ||
|
|
!Q_stricmp( arg, "dn3d" ) ||
|
|
!Q_stricmp( arg, "dnf" ) ||
|
|
!Q_stricmp( arg, "hl" ) ) {
|
|
Sys_Printf( "April fools, silly rabbit!\n" );
|
|
exit( 0 );
|
|
}
|
|
|
|
/* test it */
|
|
i = 0;
|
|
while ( games[ i ].arg != NULL )
|
|
{
|
|
if ( Q_stricmp( arg, games[ i ].arg ) == 0 ) {
|
|
return &games[ i ];
|
|
}
|
|
i++;
|
|
}
|
|
|
|
/* no matching game */
|
|
return NULL;
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
AddBasePath() - ydnar
|
|
adds a base path to the list
|
|
*/
|
|
|
|
void AddBasePath( char *path ){
|
|
/* dummy check */
|
|
if ( path == NULL || path[ 0 ] == '\0' || numBasePaths >= MAX_BASE_PATHS ) {
|
|
return;
|
|
}
|
|
|
|
/* add it to the list */
|
|
basePaths[ numBasePaths ] = safe_malloc( strlen( path ) + 1 );
|
|
strcpy( basePaths[ numBasePaths ], path );
|
|
CleanPath( basePaths[ numBasePaths ] );
|
|
numBasePaths++;
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
AddHomeBasePath() - ydnar
|
|
adds a base path to the beginning of the list, prefixed by ~/
|
|
*/
|
|
|
|
void AddHomeBasePath( char *path ){
|
|
int i;
|
|
char temp[ MAX_OS_PATH ];
|
|
int homePathLen;
|
|
|
|
if ( !homePath ) {
|
|
return;
|
|
}
|
|
|
|
/* dummy check */
|
|
if ( path == NULL || path[ 0 ] == '\0' ) {
|
|
return;
|
|
}
|
|
|
|
/* strip leading dot, if homePath does not end in /. */
|
|
homePathLen = strlen( homePath );
|
|
if ( !strcmp( path, "." ) ) {
|
|
/* -fs_homebase . means that -fs_home is to be used as is */
|
|
strcpy( temp, homePath );
|
|
}
|
|
else if ( homePathLen >= 2 && !strcmp( homePath + homePathLen - 2, "/." ) ) {
|
|
/* remove trailing /. of homePath */
|
|
homePathLen -= 2;
|
|
|
|
/* concatenate home dir and path */
|
|
sprintf( temp, "%.*s/%s", homePathLen, homePath, path );
|
|
}
|
|
else
|
|
{
|
|
/* remove leading . of path */
|
|
if ( path[0] == '.' ) {
|
|
++path;
|
|
}
|
|
|
|
/* concatenate home dir and path */
|
|
sprintf( temp, "%s/%s", homePath, path );
|
|
}
|
|
|
|
/* make a hole */
|
|
for ( i = ( MAX_BASE_PATHS - 2 ); i >= 0; i-- )
|
|
basePaths[ i + 1 ] = basePaths[ i ];
|
|
|
|
/* add it to the list */
|
|
basePaths[ 0 ] = safe_malloc( strlen( temp ) + 1 );
|
|
strcpy( basePaths[ 0 ], temp );
|
|
CleanPath( basePaths[ 0 ] );
|
|
numBasePaths++;
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
AddGamePath() - ydnar
|
|
adds a game path to the list
|
|
*/
|
|
|
|
void AddGamePath( char *path ){
|
|
int i;
|
|
|
|
/* dummy check */
|
|
if ( path == NULL || path[ 0 ] == '\0' || numGamePaths >= MAX_GAME_PATHS ) {
|
|
return;
|
|
}
|
|
|
|
/* add it to the list */
|
|
gamePaths[ numGamePaths ] = safe_malloc( strlen( path ) + 1 );
|
|
strcpy( gamePaths[ numGamePaths ], path );
|
|
CleanPath( gamePaths[ numGamePaths ] );
|
|
numGamePaths++;
|
|
|
|
/* don't add it if it's already there */
|
|
for ( i = 0; i < numGamePaths - 1; i++ )
|
|
{
|
|
if ( strcmp( gamePaths[i], gamePaths[numGamePaths - 1] ) == 0 ) {
|
|
free( gamePaths[numGamePaths - 1] );
|
|
gamePaths[numGamePaths - 1] = NULL;
|
|
numGamePaths--;
|
|
break;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
|
|
/*
|
|
AddPakPath()
|
|
adds a pak path to the list
|
|
*/
|
|
|
|
void AddPakPath( char *path ){
|
|
/* dummy check */
|
|
if ( path == NULL || path[ 0 ] == '\0' || numPakPaths >= MAX_PAK_PATHS ) {
|
|
return;
|
|
}
|
|
|
|
/* add it to the list */
|
|
pakPaths[ numPakPaths ] = safe_malloc( strlen( path ) + 1 );
|
|
strcpy( pakPaths[ numPakPaths ], path );
|
|
CleanPath( pakPaths[ numPakPaths ] );
|
|
numPakPaths++;
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
InitPaths() - ydnar
|
|
cleaned up some of the path initialization code from bsp.c
|
|
will remove any arguments it uses
|
|
*/
|
|
|
|
void InitPaths( int *argc, char **argv ){
|
|
int i, j, k, len, len2;
|
|
char temp[ MAX_OS_PATH ];
|
|
|
|
int noBasePath = 0;
|
|
int noHomePath = 0;
|
|
int noMagicPath = 0;
|
|
|
|
/* note it */
|
|
Sys_FPrintf( SYS_VRB, "--- InitPaths ---\n" );
|
|
|
|
/* get the install path for backup */
|
|
LokiInitPaths( argv[ 0 ] );
|
|
|
|
/* set game to default (q3a) */
|
|
game = &games[ 0 ];
|
|
numBasePaths = 0;
|
|
numGamePaths = 0;
|
|
|
|
/* parse through the arguments and extract those relevant to paths */
|
|
for ( i = 0; i < *argc; i++ )
|
|
{
|
|
/* check for null */
|
|
if ( argv[ i ] == NULL ) {
|
|
continue;
|
|
}
|
|
|
|
/* -game */
|
|
/*if ( strcmp( argv[ i ], "-game" ) == 0 ) {
|
|
if ( ++i >= *argc ) {
|
|
Error( "Out of arguments: No game specified after %s", argv[ i - 1 ] );
|
|
}
|
|
argv[ i - 1 ] = NULL;
|
|
game = GetGame( argv[ i ] );
|
|
if ( game == NULL ) {
|
|
game = &games[ 0 ];
|
|
}
|
|
argv[ i ] = NULL;
|
|
}*/
|
|
|
|
/* -fs_forbiddenpath */
|
|
else if ( strcmp( argv[ i ], "-fs_forbiddenpath" ) == 0 ) {
|
|
if ( ++i >= *argc ) {
|
|
Error( "Out of arguments: No path specified after %s.", argv[ i - 1 ] );
|
|
}
|
|
argv[ i - 1 ] = NULL;
|
|
if ( g_numForbiddenDirs < VFS_MAXDIRS ) {
|
|
strncpy( g_strForbiddenDirs[g_numForbiddenDirs], argv[i], PATH_MAX );
|
|
g_strForbiddenDirs[g_numForbiddenDirs][PATH_MAX] = 0;
|
|
++g_numForbiddenDirs;
|
|
}
|
|
argv[ i ] = NULL;
|
|
}
|
|
|
|
/* -fs_nobasepath */
|
|
else if ( strcmp( argv[ i ], "-fs_nobasepath" ) == 0 ) {
|
|
noBasePath = 1;
|
|
// we don't want any basepath, neither guessed ones
|
|
noMagicPath = 1;
|
|
argv[ i ] = NULL;
|
|
}
|
|
|
|
/* -fs_nomagicpath */
|
|
else if ( strcmp( argv[ i ], "-fs_nomagicpath") == 0) {
|
|
noMagicPath = 1;
|
|
argv[ i ] = NULL;
|
|
}
|
|
|
|
/* -fs_basepath */
|
|
else if ( strcmp( argv[ i ], "-fs_basepath" ) == 0 ) {
|
|
if ( ++i >= *argc ) {
|
|
Error( "Out of arguments: No path specified after %s.", argv[ i - 1 ] );
|
|
}
|
|
argv[ i - 1 ] = NULL;
|
|
AddBasePath( argv[ i ] );
|
|
argv[ i ] = NULL;
|
|
}
|
|
|
|
/* -fs_game */
|
|
else if ( strcmp( argv[ i ], "-fs_game" ) == 0 || strcmp( argv[ i ], "-game" ) == 0 ) {
|
|
if ( ++i >= *argc ) {
|
|
Error( "Out of arguments: No path specified after %s.", argv[ i - 1 ] );
|
|
}
|
|
argv[ i - 1 ] = NULL;
|
|
AddGamePath( argv[ i ] );
|
|
argv[ i ] = NULL;
|
|
}
|
|
|
|
/* -fs_home */
|
|
else if ( strcmp( argv[ i ], "-fs_home" ) == 0 ) {
|
|
if ( ++i >= *argc ) {
|
|
Error( "Out of arguments: No path specified after %s.", argv[ i - 1 ] );
|
|
}
|
|
argv[ i - 1 ] = NULL;
|
|
homePath = argv[i];
|
|
argv[ i ] = NULL;
|
|
}
|
|
|
|
/* -fs_nohomepath */
|
|
else if ( strcmp( argv[ i ], "-fs_nohomepath" ) == 0 ) {
|
|
noHomePath = 1;
|
|
argv[ i ] = NULL;
|
|
}
|
|
|
|
/* -fs_homebase */
|
|
else if ( strcmp( argv[ i ], "-fs_homebase" ) == 0 ) {
|
|
if ( ++i >= *argc ) {
|
|
Error( "Out of arguments: No path specified after %s.", argv[ i - 1 ] );
|
|
}
|
|
argv[ i - 1 ] = NULL;
|
|
homeBasePath = argv[i];
|
|
argv[ i ] = NULL;
|
|
}
|
|
|
|
/* -fs_homepath - sets both of them */
|
|
else if ( strcmp( argv[ i ], "-fs_homepath" ) == 0 ) {
|
|
if ( ++i >= *argc ) {
|
|
Error( "Out of arguments: No path specified after %s.", argv[ i - 1 ] );
|
|
}
|
|
argv[ i - 1 ] = NULL;
|
|
homePath = argv[i];
|
|
homeBasePath = ".";
|
|
argv[ i ] = NULL;
|
|
}
|
|
|
|
/* -fs_pakpath */
|
|
else if ( strcmp( argv[ i ], "-fs_pakpath" ) == 0 ) {
|
|
if ( ++i >= *argc ) {
|
|
Error( "Out of arguments: No path specified after %s.", argv[ i - 1 ] );
|
|
}
|
|
argv[ i - 1 ] = NULL;
|
|
AddPakPath( argv[ i ] );
|
|
argv[ i ] = NULL;
|
|
}
|
|
}
|
|
|
|
/* remove processed arguments */
|
|
for ( i = 0, j = 0, k = 0; i < *argc && j < *argc; i++, j++ )
|
|
{
|
|
for ( ; j < *argc && argv[ j ] == NULL; j++ ) ;
|
|
argv[ i ] = argv[ j ];
|
|
if ( argv[ i ] != NULL ) {
|
|
k++;
|
|
}
|
|
}
|
|
*argc = k;
|
|
|
|
/* add standard game path */
|
|
AddGamePath( game->gamePath );
|
|
|
|
/* if there is no base path set, figure it out unless fs_nomagicpath is set */
|
|
if ( numBasePaths == 0 && noBasePath == 0 && noMagicPath == 0 ) {
|
|
/* this is another crappy replacement for SetQdirFromPath() */
|
|
len2 = strlen( game->magic );
|
|
for ( i = 0; i < *argc && numBasePaths == 0; i++ )
|
|
{
|
|
/* extract the arg */
|
|
strcpy( temp, argv[ i ] );
|
|
CleanPath( temp );
|
|
len = strlen( temp );
|
|
Sys_FPrintf( SYS_VRB, "Searching for \"%s\" in \"%s\" (%d)...\n", game->magic, temp, i );
|
|
|
|
/* this is slow, but only done once */
|
|
for ( j = 0; j < ( len - len2 ); j++ )
|
|
{
|
|
/* check for the game's magic word */
|
|
if ( Q_strncasecmp( &temp[ j ], game->magic, len2 ) == 0 ) {
|
|
/* now find the next slash and nuke everything after it */
|
|
while ( temp[ ++j ] != '/' && temp[ j ] != '\0' ) ;
|
|
temp[ j ] = '\0';
|
|
|
|
/* add this as a base path */
|
|
AddBasePath( temp );
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* add install path */
|
|
if ( numBasePaths == 0 ) {
|
|
AddBasePath( installPath );
|
|
}
|
|
|
|
/* check again */
|
|
if ( numBasePaths == 0 ) {
|
|
Error( "Failed to find a valid base path." );
|
|
}
|
|
}
|
|
|
|
if ( noBasePath == 1 ) {
|
|
numBasePaths = 0;
|
|
}
|
|
|
|
if ( noHomePath == 0 ) {
|
|
/* this only affects unix */
|
|
if ( homeBasePath ) {
|
|
AddHomeBasePath( homeBasePath );
|
|
}
|
|
else{
|
|
AddHomeBasePath( game->homeBasePath );
|
|
}
|
|
}
|
|
|
|
/* initialize vfs paths */
|
|
if ( numBasePaths > MAX_BASE_PATHS ) {
|
|
numBasePaths = MAX_BASE_PATHS;
|
|
}
|
|
if ( numGamePaths > MAX_GAME_PATHS ) {
|
|
numGamePaths = MAX_GAME_PATHS;
|
|
}
|
|
|
|
/* walk the list of game paths */
|
|
for ( j = 0; j < numGamePaths; j++ )
|
|
{
|
|
/* walk the list of base paths */
|
|
for ( i = 0; i < numBasePaths; i++ )
|
|
{
|
|
/* create a full path and initialize it */
|
|
sprintf( temp, "%s/%s/", basePaths[ i ], gamePaths[ j ] );
|
|
vfsInitDirectory( temp );
|
|
}
|
|
}
|
|
|
|
/* initialize vfs paths */
|
|
if ( numPakPaths > MAX_PAK_PATHS ) {
|
|
numPakPaths = MAX_PAK_PATHS;
|
|
}
|
|
|
|
/* walk the list of pak paths */
|
|
for ( i = 0; i < numPakPaths; i++ )
|
|
{
|
|
/* initialize this pak path */
|
|
vfsInitDirectory( pakPaths[ i ] );
|
|
}
|
|
|
|
/* done */
|
|
Sys_Printf( "\n" );
|
|
}
|