2018-04-13 19:23:45 +03:00
/*
host . c - dedicated and normal host
Copyright ( C ) 2007 Uncle Mike
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 .
*/
2018-04-13 21:28:46 +03:00
# ifdef XASH_SDL
# include <SDL.h>
# endif // XASH_SDL
# include <stdarg.h> // va_args
# include <errno.h> // errno
# include <string.h> // strerror
# ifndef _WIN32
# include <unistd.h> // fork
# include <sys/types.h>
# include <sys/stat.h>
# include <fcntl.h>
# endif
# ifdef __EMSCRIPTEN__
# include <emscripten/emscripten.h>
# endif
# include <errno.h>
2018-04-13 19:23:45 +03:00
# include "common.h"
2018-06-19 16:59:53 +03:00
# include "base_cmd.h"
# include "client.h"
2018-04-13 19:23:45 +03:00
# include "netchan.h"
# include "protocol.h"
# include "mod_local.h"
# include "mathlib.h"
# include "input.h"
2018-04-13 19:26:48 +03:00
# include "enginefeatures.h"
2018-04-13 19:23:45 +03:00
# include "render_api.h" // decallist_t
2018-04-13 21:28:46 +03:00
2018-04-13 19:23:45 +03:00
typedef void ( * pfnChangeGame ) ( const char * progname ) ;
pfnChangeGame pChangeGame = NULL ;
2018-04-13 21:28:46 +03:00
host_parm_t host ; // host parms
2018-04-13 19:23:45 +03:00
sysinfo_t SI ;
CVAR_DEFINE ( host_developer , " developer " , " 0 " , 0 , " engine is in development-mode " ) ;
2018-04-26 03:09:36 +03:00
CVAR_DEFINE_AUTO ( sys_ticrate , " 100 " , 0 , " framerate in dedicated mode " ) ;
2018-04-13 19:23:45 +03:00
convar_t * host_gameloaded ;
convar_t * host_clientloaded ;
convar_t * host_limitlocal ;
convar_t * host_maxfps ;
convar_t * host_framerate ;
convar_t * con_gamemaps ;
convar_t * build , * ver ;
2018-12-05 23:57:05 +07:00
int Host_CompareFileTime ( int ft1 , int ft2 )
2018-04-13 19:23:45 +03:00
{
if ( ft1 < ft2 )
{
return - 1 ;
}
else if ( ft1 > ft2 )
{
return 1 ;
}
return 0 ;
}
void Host_ShutdownServer ( void )
{
SV_Shutdown ( " Server was killed \n " ) ;
}
/*
= = = = = = = = = = = = = = = =
Host_PrintEngineFeatures
= = = = = = = = = = = = = = = =
*/
void Host_PrintEngineFeatures ( void )
{
if ( FBitSet ( host . features , ENGINE_WRITE_LARGE_COORD ) )
2018-10-04 13:08:48 +07:00
Con_Reportf ( " ^3EXT:^7 big world support enabled \n " ) ;
2018-04-13 19:23:45 +03:00
if ( FBitSet ( host . features , ENGINE_LOAD_DELUXEDATA ) )
2018-10-04 13:08:48 +07:00
Con_Reportf ( " ^3EXT:^7 deluxemap support enabled \n " ) ;
2018-04-13 19:23:45 +03:00
if ( FBitSet ( host . features , ENGINE_PHYSICS_PUSHER_EXT ) )
2018-10-04 13:08:48 +07:00
Con_Reportf ( " ^3EXT:^7 Improved MOVETYPE_PUSH is used \n " ) ;
2018-04-13 19:23:45 +03:00
if ( FBitSet ( host . features , ENGINE_LARGE_LIGHTMAPS ) )
2018-10-04 13:08:48 +07:00
Con_Reportf ( " ^3EXT:^7 Large lightmaps enabled \n " ) ;
2018-04-13 19:23:45 +03:00
if ( FBitSet ( host . features , ENGINE_COMPENSATE_QUAKE_BUG ) )
2018-10-04 13:08:48 +07:00
Con_Reportf ( " ^3EXT:^7 Compensate quake bug enabled \n " ) ;
2018-04-13 19:23:45 +03:00
}
2018-06-19 16:59:53 +03:00
/*
= = = = = = = = = = = = = =
Host_IsQuakeCompatible
= = = = = = = = = = = = = =
*/
qboolean Host_IsQuakeCompatible ( void )
{
// feature set
if ( FBitSet ( host . features , ENGINE_QUAKE_COMPATIBLE ) )
return true ;
# ifndef XASH_DEDICATED
// quake demo playing
if ( cls . demoplayback = = DEMO_QUAKE1 )
return true ;
# endif // XASH_DEDICATED
return false ;
}
2018-04-13 19:23:45 +03:00
/*
= = = = = = = = = = = = = = = =
Host_EndGame
= = = = = = = = = = = = = = = =
*/
void Host_EndGame ( qboolean abort , const char * message , . . . )
{
va_list argptr ;
static char string [ MAX_SYSPATH ] ;
va_start ( argptr , message ) ;
Q_vsnprintf ( string , sizeof ( string ) , message , argptr ) ;
va_end ( argptr ) ;
2018-10-04 13:08:48 +07:00
Con_Printf ( " Host_EndGame: %s \n " , string ) ;
2018-04-13 19:23:45 +03:00
SV_Shutdown ( " \n " ) ;
2018-04-18 18:32:30 +03:00
# ifndef XASH_DEDICATED
2018-04-13 19:23:45 +03:00
CL_Disconnect ( ) ;
// recreate world if needs
CL_ClearEdicts ( ) ;
2018-04-18 18:32:30 +03:00
# endif
2018-04-13 19:23:45 +03:00
// release all models
Mod_FreeAll ( ) ;
if ( abort ) Host_AbortCurrentFrame ( ) ;
}
/*
= = = = = = = = = = = = = = = =
Host_AbortCurrentFrame
aborts the current host frame and goes on with the next one
= = = = = = = = = = = = = = = =
*/
void Host_AbortCurrentFrame ( void )
{
longjmp ( host . abortframe , 1 ) ;
}
/*
= = = = = = = = = = = = = = = = = =
Host_CheckSleep
= = = = = = = = = = = = = = = = = =
*/
void Host_CheckSleep ( void )
{
2018-05-01 17:27:14 +03:00
if ( Host_IsDedicated ( ) )
2018-04-13 19:23:45 +03:00
{
// let the dedicated server some sleep
2018-04-26 03:09:36 +03:00
Sys_Sleep ( 1 ) ;
2018-04-13 19:23:45 +03:00
}
else
{
if ( host . status = = HOST_NOFOCUS )
{
if ( SV_Active ( ) & & CL_IsInGame ( ) )
Sys_Sleep ( 1 ) ; // listenserver
else Sys_Sleep ( 20 ) ; // sleep 20 ms otherwise
}
else if ( host . status = = HOST_SLEEP )
{
// completely sleep in minimized state
Sys_Sleep ( 20 ) ;
}
}
}
void Host_NewInstance ( const char * name , const char * finalmsg )
{
if ( ! pChangeGame ) return ;
host . change_game = true ;
Q_strncpy ( host . finalmsg , finalmsg , sizeof ( host . finalmsg ) ) ;
pChangeGame ( name ) ; // call from hl.exe
}
/*
= = = = = = = = = = = = = = = = =
Host_ChangeGame_f
Change game modification
= = = = = = = = = = = = = = = = =
*/
void Host_ChangeGame_f ( void )
{
int i ;
if ( Cmd_Argc ( ) ! = 2 )
{
Con_Printf ( S_USAGE " game <directory> \n " ) ;
return ;
}
// validate gamedir
for ( i = 0 ; i < SI . numgames ; i + + )
{
if ( ! Q_stricmp ( SI . games [ i ] - > gamefolder , Cmd_Argv ( 1 ) ) )
break ;
}
if ( i = = SI . numgames )
{
Con_Printf ( " %s not exist \n " , Cmd_Argv ( 1 ) ) ;
}
else if ( ! Q_stricmp ( GI - > gamefolder , Cmd_Argv ( 1 ) ) )
{
Con_Printf ( " %s already active \n " , Cmd_Argv ( 1 ) ) ;
}
else
{
const char * arg1 = va ( " %s%s " , ( host . type = = HOST_NORMAL ) ? " " : " # " , Cmd_Argv ( 1 ) ) ;
const char * arg2 = va ( " change game to '%s' " , SI . games [ i ] - > title ) ;
Host_NewInstance ( arg1 , arg2 ) ;
}
}
/*
= = = = = = = = = = = = = = =
Host_Exec_f
= = = = = = = = = = = = = = =
*/
void Host_Exec_f ( void )
{
string cfgpath ;
char * f , * txt ;
size_t len ;
if ( Cmd_Argc ( ) ! = 2 )
{
Con_Printf ( S_USAGE " exec <filename> \n " ) ;
return ;
}
if ( ! Q_stricmp ( " game.cfg " , Cmd_Argv ( 1 ) ) )
{
// don't execute game.cfg in singleplayer
if ( SV_GetMaxClients ( ) = = 1 )
return ;
}
Q_strncpy ( cfgpath , Cmd_Argv ( 1 ) , sizeof ( cfgpath ) ) ;
COM_DefaultExtension ( cfgpath , " .cfg " ) ; // append as default
f = FS_LoadFile ( cfgpath , & len , false ) ;
if ( ! f )
{
2018-10-04 13:08:48 +07:00
Con_Reportf ( " couldn't exec %s \n " , Cmd_Argv ( 1 ) ) ;
2018-04-13 19:23:45 +03:00
return ;
}
if ( ! Q_stricmp ( " config.cfg " , Cmd_Argv ( 1 ) ) )
host . config_executed = true ;
// adds \n\0 at end of the file
2018-06-09 01:28:35 +03:00
txt = Z_Calloc ( len + 2 ) ;
2018-04-13 19:23:45 +03:00
memcpy ( txt , f , len ) ;
Q_strncat ( txt , " \n " , len + 2 ) ;
Mem_Free ( f ) ;
if ( ! host . apply_game_config )
2018-10-04 13:08:48 +07:00
Con_Printf ( " execing %s \n " , Cmd_Argv ( 1 ) ) ;
2018-04-13 19:23:45 +03:00
Cbuf_InsertText ( txt ) ;
Mem_Free ( txt ) ;
}
/*
= = = = = = = = = = = = = = =
Host_MemStats_f
= = = = = = = = = = = = = = =
*/
void Host_MemStats_f ( void )
{
switch ( Cmd_Argc ( ) )
{
case 1 :
Mem_PrintList ( 1 < < 30 ) ;
Mem_PrintStats ( ) ;
break ;
case 2 :
Mem_PrintList ( Q_atoi ( Cmd_Argv ( 1 ) ) * 1024 ) ;
Mem_PrintStats ( ) ;
break ;
default :
Con_Printf ( S_USAGE " memlist <all> \n " ) ;
break ;
}
}
void Host_Minimize_f ( void )
{
2018-04-13 21:28:46 +03:00
# ifdef XASH_SDL
if ( host . hWnd ) SDL_MinimizeWindow ( host . hWnd ) ;
# endif
2018-04-13 19:23:45 +03:00
}
/*
= = = = = = = = = = = = = = = = =
Host_IsLocalGame
singleplayer game detect
= = = = = = = = = = = = = = = = =
*/
qboolean Host_IsLocalGame ( void )
{
if ( SV_Active ( ) )
{
return ( SV_GetMaxClients ( ) = = 1 ) ? true : false ;
}
else
{
return ( CL_GetMaxClients ( ) = = 1 ) ? true : false ;
}
}
qboolean Host_IsLocalClient ( void )
{
// only the local client have the active server
if ( CL_Initialized ( ) & & SV_Initialized ( ) )
return true ;
return false ;
}
/*
= = = = = = = = = = = = = = = = =
Host_RegisterDecal
= = = = = = = = = = = = = = = = =
*/
qboolean Host_RegisterDecal ( const char * name , int * count )
{
char shortname [ MAX_QPATH ] ;
int i ;
if ( ! COM_CheckString ( name ) )
return 0 ;
COM_FileBase ( name , shortname ) ;
for ( i = 1 ; i < MAX_DECALS & & host . draw_decals [ i ] [ 0 ] ; i + + )
{
if ( ! Q_stricmp ( host . draw_decals [ i ] , shortname ) )
return true ;
}
if ( i = = MAX_DECALS )
{
2018-10-04 13:08:48 +07:00
Con_DPrintf ( S_ERROR " MAX_DECALS limit exceeded (%d) \n " , MAX_DECALS ) ;
2018-04-13 19:23:45 +03:00
return false ;
}
// register new decal
Q_strncpy ( host . draw_decals [ i ] , shortname , sizeof ( host . draw_decals [ i ] ) ) ;
* count + = 1 ;
return true ;
}
/*
= = = = = = = = = = = = = = = = =
Host_InitDecals
= = = = = = = = = = = = = = = = =
*/
void Host_InitDecals ( void )
{
int i , num_decals = 0 ;
search_t * t ;
2018-06-12 12:14:56 +03:00
// NOTE: only once resource without which engine can't continue work
if ( ! FS_FileExists ( " gfx/conchars " , false ) )
Sys_Error ( " W_LoadWadFile: couldn't load gfx.wad \n " ) ;
2018-04-13 19:23:45 +03:00
memset ( host . draw_decals , 0 , sizeof ( host . draw_decals ) ) ;
// lookup all the decals in decals.wad (basedir, gamedir, falldir)
t = FS_Search ( " decals.wad/*.* " , true , false ) ;
for ( i = 0 ; t & & i < t - > numfilenames ; i + + )
{
if ( ! Host_RegisterDecal ( t - > filenames [ i ] , & num_decals ) )
break ;
}
if ( t ) Mem_Free ( t ) ;
2018-04-19 20:11:24 +00:00
Con_Reportf ( " InitDecals: %i decals \n " , num_decals ) ;
2018-04-13 19:23:45 +03:00
}
/*
= = = = = = = = = = = = = = = = = = =
Host_GetCommands
Add them exactly as if they had been typed at the console
= = = = = = = = = = = = = = = = = = =
*/
void Host_GetCommands ( void )
{
char * cmd ;
2018-04-14 04:08:28 +03:00
while ( ( cmd = Sys_Input ( ) ) )
{
Cbuf_AddText ( cmd ) ;
Cbuf_Execute ( ) ;
}
2018-04-13 19:23:45 +03:00
}
/*
= = = = = = = = = = = = = = = = = = =
Host_CalcFPS
compute actual FPS for various modes
= = = = = = = = = = = = = = = = = = =
*/
double Host_CalcFPS ( void )
{
double fps = 0.0 ;
// NOTE: we should play demos with same fps as it was recorded
2018-04-18 18:32:30 +03:00
# ifndef XASH_DEDICATED
2018-04-13 19:23:45 +03:00
if ( CL_IsPlaybackDemo ( ) | | CL_IsRecordDemo ( ) )
2018-04-26 03:09:36 +03:00
{
2018-04-13 19:23:45 +03:00
fps = CL_GetDemoFramerate ( ) ;
2018-04-26 03:09:36 +03:00
}
2018-04-13 19:23:45 +03:00
else if ( Host_IsLocalGame ( ) )
2018-04-26 03:09:36 +03:00
{
2018-04-13 19:23:45 +03:00
fps = host_maxfps - > value ;
2018-04-26 03:09:36 +03:00
}
2018-04-26 03:23:00 +03:00
else
# endif
if ( Host_IsDedicated ( ) )
2018-04-26 03:09:36 +03:00
{
fps = sys_ticrate . value ;
}
2018-04-13 19:23:45 +03:00
else
{
fps = host_maxfps - > value ;
fps = bound ( MIN_FPS , fps , MAX_FPS ) ;
}
2018-04-18 18:32:30 +03:00
# ifndef XASH_DEDICATED
2018-04-13 19:23:45 +03:00
// probably left part of this condition is redundant :-)
if ( host . type ! = HOST_DEDICATED & & Host_IsLocalGame ( ) & & ! CL_IsTimeDemo ( ) )
{
// ajdust fps for vertical synchronization
2018-10-27 23:31:55 +03:00
if ( CVAR_TO_BOOL ( gl_vsync ) )
2018-04-13 19:23:45 +03:00
{
if ( vid_displayfrequency - > value ! = 0.0f )
fps = vid_displayfrequency - > value ;
else fps = 60.0 ; // default
}
}
2018-04-18 18:32:30 +03:00
# endif
2018-04-13 19:23:45 +03:00
return fps ;
}
/*
= = = = = = = = = = = = = = = = = = =
Host_FilterTime
Returns false if the time is too short to run a frame
= = = = = = = = = = = = = = = = = = =
*/
qboolean Host_FilterTime ( float time )
{
static double oldtime ;
double fps ;
host . realtime + = time ;
fps = Host_CalcFPS ( ) ;
// clamp the fps in multiplayer games
if ( fps ! = 0.0 )
{
// limit fps to withing tolerable range
fps = bound ( MIN_FPS , fps , MAX_FPS ) ;
2018-05-01 17:27:14 +03:00
if ( Host_IsDedicated ( ) )
2018-04-26 03:09:36 +03:00
{
if ( ( host . realtime - oldtime ) < ( 1.0 / ( fps + 1.0 ) ) )
return false ;
}
else
{
if ( ( host . realtime - oldtime ) < ( 1.0 / fps ) )
return false ;
}
2018-04-13 19:23:45 +03:00
}
host . frametime = host . realtime - oldtime ;
host . realframetime = bound ( MIN_FRAMETIME , host . frametime , MAX_FRAMETIME ) ;
oldtime = host . realtime ;
// NOTE: allow only in singleplayer while demos are not active
if ( host_framerate - > value > 0.0f & & Host_IsLocalGame ( ) & & ! CL_IsPlaybackDemo ( ) & & ! CL_IsRecordDemo ( ) )
host . frametime = bound ( MIN_FRAMETIME , host_framerate - > value , MAX_FRAMETIME ) ;
else host . frametime = bound ( MIN_FRAMETIME , host . frametime , MAX_FRAMETIME ) ;
return true ;
}
/*
= = = = = = = = = = = = = = = = =
Host_Frame
= = = = = = = = = = = = = = = = =
*/
void Host_Frame ( float time )
{
Host_CheckSleep ( ) ;
// decide the simulation time
if ( ! Host_FilterTime ( time ) )
return ;
Host_InputFrame ( ) ; // input frame
Host_ClientBegin ( ) ; // begin client
Host_GetCommands ( ) ; // dedicated in
Host_ServerFrame ( ) ; // server frame
Host_ClientFrame ( ) ; // client frame
host . framecount + + ;
}
/*
= = = = = = = = = = = = = = = = =
Host_Error
= = = = = = = = = = = = = = = = =
*/
void Host_Error ( const char * error , . . . )
{
static char hosterror1 [ MAX_SYSPATH ] ;
static char hosterror2 [ MAX_SYSPATH ] ;
static qboolean recursive = false ;
va_list argptr ;
if ( host . mouse_visible & & ! CL_IsInMenu ( ) )
{
// hide VGUI mouse
2018-04-13 21:28:46 +03:00
# ifdef XASH_SDL
SDL_ShowCursor ( 0 ) ;
# endif
2018-04-13 19:23:45 +03:00
host . mouse_visible = false ;
}
va_start ( argptr , error ) ;
Q_vsprintf ( hosterror1 , error , argptr ) ;
va_end ( argptr ) ;
CL_WriteMessageHistory ( ) ; // before Q_error call
if ( host . framecount < 3 )
{
Sys_Error ( " Host_InitError: %s " , hosterror1 ) ;
}
else if ( host . framecount = = host . errorframe )
{
Sys_Error ( " Host_MultiError: %s " , hosterror2 ) ;
return ;
}
else
{
if ( host . allow_console )
{
UI_SetActiveMenu ( false ) ;
Key_SetKeyDest ( key_console ) ;
Con_Printf ( " Host_Error: %s " , hosterror1 ) ;
}
else MSGBOX2 ( hosterror1 ) ;
}
// host is shutting down. don't invoke infinite loop
if ( host . status = = HOST_SHUTDOWN ) return ;
if ( recursive )
{
Con_Printf ( " Host_RecursiveError: %s " , hosterror2 ) ;
2018-04-23 21:49:37 +03:00
Sys_Error ( " %s " , hosterror1 ) ;
2018-04-13 19:23:45 +03:00
return ; // don't multiple executes
}
recursive = true ;
Q_strncpy ( hosterror2 , hosterror1 , MAX_SYSPATH ) ;
host . errorframe = host . framecount ; // to avoid multply calls per frame
Q_sprintf ( host . finalmsg , " Server crashed: %s " , hosterror1 ) ;
// clearing cmd buffer to prevent execute any commands
COM_InitHostState ( ) ;
Cbuf_Clear ( ) ;
Host_ShutdownServer ( ) ;
CL_Drop ( ) ; // drop clients
// recreate world if needs
CL_ClearEdicts ( ) ;
// release all models
Mod_FreeAll ( ) ;
recursive = false ;
Host_AbortCurrentFrame ( ) ;
}
void Host_Error_f ( void )
{
const char * error = Cmd_Argv ( 1 ) ;
if ( ! * error ) error = " Invoked host error " ;
Host_Error ( " %s \n " , error ) ;
}
void Sys_Error_f ( void )
{
const char * error = Cmd_Argv ( 1 ) ;
if ( ! * error ) error = " Invoked sys error " ;
Sys_Error ( " %s \n " , error ) ;
}
/*
= = = = = = = = = = = = = = = = =
Host_Crash_f
= = = = = = = = = = = = = = = = =
*/
static void Host_Crash_f ( void )
{
* ( int * ) 0 = 0xffffffff ;
}
/*
= = = = = = = = = = = = = = = = =
Host_InitCommon
= = = = = = = = = = = = = = = = =
*/
2018-04-13 21:28:46 +03:00
void Host_InitCommon ( int argc , char * * argv , const char * progname , qboolean bChangeGame )
2018-04-13 19:23:45 +03:00
{
char dev_level [ 4 ] ;
int developer = 0 ;
2018-04-13 21:28:46 +03:00
const char * baseDir ;
2018-04-26 14:11:01 +00:00
char ticrate [ 16 ] ;
2018-04-13 19:23:45 +03:00
2018-04-13 21:28:46 +03:00
// some commands may turn engine into infinite loop,
// e.g. xash.exe +game xash -game xash
// so we clear all cmd_args, but leave dbg states as well
Sys_ParseCommandLine ( argc , argv ) ;
2018-04-13 19:23:45 +03:00
2018-04-13 21:28:46 +03:00
if ( ! Sys_CheckParm ( " -noch " ) )
Sys_SetupCrashHandler ( ) ;
2018-04-13 19:23:45 +03:00
2018-04-13 21:28:46 +03:00
// to be accessed later
if ( ( host . daemonized = Sys_CheckParm ( " -daemonize " ) ) )
{
# if defined(_POSIX_VERSION) && !defined(XASH_MOBILE_PLATFORM)
pid_t daemon ;
2018-04-13 19:23:45 +03:00
2018-04-13 21:28:46 +03:00
daemon = fork ( ) ;
2018-04-13 19:23:45 +03:00
2018-04-13 21:28:46 +03:00
if ( daemon < 0 )
{
Host_Error ( " fork() failed: %s \n " , strerror ( errno ) ) ;
}
2018-04-13 19:23:45 +03:00
2018-04-13 21:28:46 +03:00
if ( daemon > 0 )
2018-04-13 19:23:45 +03:00
{
2018-04-13 21:28:46 +03:00
// parent
2018-11-16 15:25:04 +03:00
Con_Reportf ( " Child pid: %i \n " , daemon ) ;
2018-04-13 21:28:46 +03:00
exit ( 0 ) ;
2018-04-13 19:23:45 +03:00
}
else
{
2018-04-13 21:28:46 +03:00
// don't be closed by parent
if ( setsid ( ) < 0 )
2018-04-13 19:23:45 +03:00
{
2018-04-13 21:28:46 +03:00
Host_Error ( " setsid() failed: %s \n " , strerror ( errno ) ) ;
2018-04-13 19:23:45 +03:00
}
2018-04-13 21:28:46 +03:00
// set permissions
umask ( 0 ) ;
// engine will still use stdin/stdout,
// so just redirect them to /dev/null
close ( STDIN_FILENO ) ;
close ( STDOUT_FILENO ) ;
close ( STDERR_FILENO ) ;
open ( " /dev/null " , O_RDONLY ) ; // becomes stdin
open ( " /dev/null " , O_RDWR ) ; // stdout
open ( " /dev/null " , O_RDWR ) ; // stderr
// fallthrough
2018-04-13 19:23:45 +03:00
}
2018-04-13 21:28:46 +03:00
# elif defined(XASH_MOBILE_PLATFORM)
Sys_Error ( " Can't run in background on mobile platforms! " ) ;
# else
Sys_Error ( " Daemonize not supported on this platform! " ) ;
# endif
2018-04-13 19:23:45 +03:00
}
2018-04-13 21:28:46 +03:00
if ( ( baseDir = getenv ( " XASH3D_BASEDIR " ) ) )
{
Q_strncpy ( host . rootdir , baseDir , sizeof ( host . rootdir ) ) ;
}
else
{
# if TARGET_OS_IOS
const char * IOS_GetDocsDir ( ) ;
Q_strncpy ( host . rootdir , IOS_GetDocsDir ( ) , sizeof ( host . rootdir ) ) ;
# elif defined(XASH_SDL)
2018-11-16 15:25:04 +03:00
char * szBasePath ;
if ( ! ( szBasePath = SDL_GetBasePath ( ) ) )
2018-04-13 21:28:46 +03:00
Sys_Error ( " couldn't determine current directory: %s " , SDL_GetError ( ) ) ;
2018-11-16 15:25:04 +03:00
Q_strncpy ( host . rootdir , szBasePath , sizeof ( host . rootdir ) ) ;
SDL_free ( szBasePath ) ;
2018-04-13 21:28:46 +03:00
# else
if ( ! getcwd ( host . rootdir , sizeof ( host . rootdir ) ) )
{
Sys_Error ( " couldn't determine current directory: %s " , strerror ( errno ) ) ;
host . rootdir [ 0 ] = 0 ;
}
# endif
}
if ( host . rootdir [ Q_strlen ( host . rootdir ) - 1 ] = = ' / ' )
host . rootdir [ Q_strlen ( host . rootdir ) - 1 ] = 0 ;
2018-07-10 23:03:27 +03:00
// get readonly root. The order is: check for arg, then env.
// if still not got it, rodir is disabled.
host . rodir [ 0 ] = 0 ;
if ( ! Sys_GetParmFromCmdLine ( " -rodir " , host . rodir ) )
{
char * roDir ;
if ( ( roDir = getenv ( " XASH3D_RODIR " ) ) )
Q_strncpy ( host . rodir , roDir , sizeof ( host . rodir ) ) ;
}
if ( host . rodir [ 0 ] & & host . rodir [ Q_strlen ( host . rodir ) - 1 ] = = ' / ' )
host . rodir [ Q_strlen ( host . rodir ) - 1 ] = 0 ;
2018-04-13 21:28:46 +03:00
host . enabledll = ! Sys_CheckParm ( " -nodll " ) ;
# ifdef DLL_LOADER
if ( host . enabledll )
Setup_LDT_Keeper ( ) ; // Must call before creating any thread
# endif
host . change_game = bChangeGame ;
host . config_executed = false ;
host . status = HOST_INIT ; // initialzation started
Memory_Init ( ) ; // init memory subsystem
2018-04-13 19:23:45 +03:00
host . mempool = Mem_AllocPool ( " Zone Engine " ) ;
2018-04-21 08:06:55 +00:00
// HACKHACK: Quake console is always allowed
2018-10-28 00:39:29 +03:00
// TODO: determine if we are running QWrap more reliable
2018-10-27 23:31:55 +03:00
if ( Sys_CheckParm ( " -console " ) | | ! Q_stricmp ( SI . exeName , " quake " ) )
2018-04-13 19:23:45 +03:00
host . allow_console = true ;
if ( Sys_CheckParm ( " -dev " ) )
{
host . allow_console = true ;
developer = DEV_NORMAL ;
if ( Sys_GetParmFromCmdLine ( " -dev " , dev_level ) )
{
if ( Q_isdigit ( dev_level ) )
developer = bound ( DEV_NONE , abs ( Q_atoi ( dev_level ) ) , DEV_EXTENDED ) ;
}
}
host . con_showalways = true ;
2018-04-13 21:28:46 +03:00
# ifdef XASH_DEDICATED
host . type = HOST_DEDICATED ; // predict state
# else
if ( Sys_CheckParm ( " -dedicated " ) | | progname [ 0 ] = = ' # ' )
{
host . type = HOST_DEDICATED ;
}
else
2018-04-13 19:23:45 +03:00
{
2018-04-13 21:28:46 +03:00
host . type = HOST_NORMAL ;
2018-04-13 19:23:45 +03:00
}
2018-04-13 21:28:46 +03:00
# endif
2018-04-13 19:23:45 +03:00
2018-04-13 21:28:46 +03:00
# ifdef XASH_SDL
// should work even if it failed
SDL_Init ( SDL_INIT_TIMER ) ;
2018-04-13 19:23:45 +03:00
2018-04-13 21:28:46 +03:00
if ( SDL_Init ( SDL_INIT_VIDEO | SDL_INIT_EVENTS ) )
2018-04-13 19:23:45 +03:00
{
2018-04-13 21:28:46 +03:00
Sys_Warn ( " SDL_Init failed: %s " , SDL_GetError ( ) ) ;
2018-04-13 19:23:45 +03:00
host . type = HOST_DEDICATED ;
}
2018-04-13 21:28:46 +03:00
SDL_SetHint ( SDL_HINT_ACCELEROMETER_AS_JOYSTICK , " 0 " ) ;
# if defined XASH_GLES && !defined __EMSCRIPTEN__ && !TARGET_OS_IOS
SDL_SetHint ( SDL_HINT_OPENGL_ES_DRIVER , " 1 " ) ;
# endif
2018-04-21 00:37:11 +03:00
SDL_StopTextInput ( ) ;
2018-04-13 21:28:46 +03:00
# endif
2018-04-13 19:23:45 +03:00
2018-04-13 21:28:46 +03:00
if ( ! host . rootdir [ 0 ] | | SetCurrentDirectory ( host . rootdir ) ! = 0 )
2018-11-16 15:25:04 +03:00
Con_Reportf ( " %s is working directory now \n " , host . rootdir ) ;
2018-04-13 21:28:46 +03:00
else
Sys_Error ( " Changing working directory to %s failed. \n " , host . rootdir ) ;
2018-04-13 19:23:45 +03:00
2018-04-13 21:28:46 +03:00
Sys_InitLog ( ) ;
2018-04-13 19:23:45 +03:00
2018-04-13 21:28:46 +03:00
// set default gamedir
if ( progname [ 0 ] = = ' # ' )
progname + + ;
Q_strncpy ( SI . exeName , progname , sizeof ( SI . exeName ) ) ;
2018-04-17 03:56:39 +03:00
Q_strncpy ( SI . basedirName , progname , sizeof ( SI . exeName ) ) ;
2018-04-13 19:23:45 +03:00
2018-04-13 21:28:46 +03:00
if ( Host_IsDedicated ( ) )
{
Sys_MergeCommandLine ( ) ;
2018-04-13 19:23:45 +03:00
host . allow_console = true ;
}
else
{
// don't show console as default
if ( developer < = DEV_NORMAL )
host . con_showalways = false ;
}
// member console allowing
host . allow_console_init = host . allow_console ;
2018-04-14 04:08:28 +03:00
# ifdef _WIN32
Wcon_CreateConsole ( ) ; // system console used by dedicated server or show fatal errors
# endif
2018-04-26 03:23:00 +03:00
// timeBeginPeriod( 1 ); // a1ba: Do we need this?
2018-04-13 19:23:45 +03:00
// NOTE: this message couldn't be passed into game console but it doesn't matter
2018-10-04 13:08:48 +07:00
Con_Reportf ( " Sys_LoadLibrary: Loading xash.dll - ok \n " ) ;
2018-04-13 19:23:45 +03:00
// get default screen res
VID_InitDefaultResolution ( ) ;
// init host state machine
COM_InitHostState ( ) ;
2018-06-01 21:28:25 +03:00
// init hashed commands
BaseCmd_Init ( ) ;
2018-04-13 19:23:45 +03:00
// startup cmds and cvars subsystem
Cmd_Init ( ) ;
Cvar_Init ( ) ;
// share developer level across all dlls
Q_snprintf ( dev_level , sizeof ( dev_level ) , " %i " , developer ) ;
Cvar_DirectSet ( & host_developer , dev_level ) ;
2018-04-26 03:09:36 +03:00
Cvar_RegisterVariable ( & sys_ticrate ) ;
if ( Sys_GetParmFromCmdLine ( " -sys_ticrate " , ticrate ) )
{
2018-04-26 14:11:01 +00:00
double fps = bound ( MIN_FPS , atof ( ticrate ) , MAX_FPS ) ;
2018-04-26 03:09:36 +03:00
Cvar_SetValue ( " sys_ticrate " , fps ) ;
}
2018-04-13 19:23:45 +03:00
Con_Init ( ) ; // early console running to catch all the messages
Cmd_AddCommand ( " exec " , Host_Exec_f , " execute a script file " ) ;
Cmd_AddCommand ( " memlist " , Host_MemStats_f , " prints memory pool information " ) ;
FS_Init ( ) ;
Image_Init ( ) ;
Sound_Init ( ) ;
FS_LoadGameInfo ( NULL ) ;
Q_strncpy ( host . gamefolder , GI - > gamefolder , sizeof ( host . gamefolder ) ) ;
2018-04-21 11:16:33 +03:00
// DEPRECATED: by FWGS fork
#if 0
2018-04-13 19:23:45 +03:00
if ( GI - > secure )
{
// clear all developer levels when game is protected
Cvar_DirectSet ( & host_developer , " 0 " ) ;
host . allow_console_init = false ;
host . con_showalways = false ;
host . allow_console = false ;
}
2018-04-21 11:16:33 +03:00
# endif
2018-04-13 19:23:45 +03:00
HPAK_Init ( ) ;
IN_Init ( ) ;
Key_Init ( ) ;
}
void Host_FreeCommon ( void )
{
Image_Shutdown ( ) ;
Sound_Shutdown ( ) ;
Netchan_Shutdown ( ) ;
HPAK_FlushHostQueue ( ) ;
FS_Shutdown ( ) ;
}
/*
= = = = = = = = = = = = = = = = =
Host_Main
= = = = = = = = = = = = = = = = =
*/
2018-04-13 21:28:46 +03:00
int EXPORT Host_Main ( int argc , char * * argv , const char * progname , int bChangeGame , pfnChangeGame func )
2018-04-13 19:23:45 +03:00
{
static double oldtime , newtime ;
pChangeGame = func ; // may be NULL
2018-04-13 21:28:46 +03:00
Host_InitCommon ( argc , argv , progname , bChangeGame ) ;
2018-04-13 19:23:45 +03:00
// init commands and vars
if ( host_developer . value > = DEV_EXTENDED )
{
Cmd_AddCommand ( " sys_error " , Sys_Error_f , " just throw a fatal error to test shutdown procedures " ) ;
Cmd_AddCommand ( " host_error " , Host_Error_f , " just throw a host error to test shutdown procedures " ) ;
Cmd_AddCommand ( " crash " , Host_Crash_f , " a way to force a bus error for development reasons " ) ;
}
2018-04-26 03:09:36 +03:00
host_maxfps = Cvar_Get ( " fps_max " , " 72 " , FCVAR_ARCHIVE , " host fps upper limit " ) ;
2018-04-13 19:23:45 +03:00
host_framerate = Cvar_Get ( " host_framerate " , " 0 " , 0 , " locks frame timing to this value in seconds " ) ;
host_gameloaded = Cvar_Get ( " host_gameloaded " , " 0 " , FCVAR_READ_ONLY , " inidcates a loaded game.dll " ) ;
host_clientloaded = Cvar_Get ( " host_clientloaded " , " 0 " , FCVAR_READ_ONLY , " inidcates a loaded client.dll " ) ;
host_limitlocal = Cvar_Get ( " host_limitlocal " , " 0 " , 0 , " apply cl_cmdrate and rate to loopback connection " ) ;
con_gamemaps = Cvar_Get ( " con_mapfilter " , " 1 " , FCVAR_ARCHIVE , " when true show only maps in game folder " ) ;
2018-10-04 14:27:14 +07:00
2018-10-04 13:08:48 +07:00
build = Cvar_Get ( " buildnum " , va ( " %i " , Q_buildnum ( ) ) , FCVAR_READ_ONLY , " returns a current build number " ) ;
2018-10-04 14:27:14 +07:00
2018-04-23 21:36:33 +03:00
ver = Cvar_Get ( " ver " , va ( " %i/%s (hw build %i) " , PROTOCOL_VERSION , XASH_VERSION , Q_buildnum ( ) ) , FCVAR_READ_ONLY , " shows an engine version " ) ;
2018-04-13 19:23:45 +03:00
2018-11-29 01:54:25 +03:00
Cvar_Get ( " host_ver " , va ( " %i %s %s %s %s " , Q_buildnum ( ) , XASH_VERSION , Q_buildos ( ) , Q_buildarch ( ) , Q_buildcommit ( ) ) , FCVAR_READ_ONLY , " detailed info about this build " ) ;
2018-04-13 19:23:45 +03:00
Mod_Init ( ) ;
NET_Init ( ) ;
2018-06-01 20:44:16 +03:00
NET_InitMasters ( ) ;
2018-04-13 19:23:45 +03:00
Netchan_Init ( ) ;
// allow to change game from the console
if ( pChangeGame ! = NULL )
{
Cmd_AddCommand ( " game " , Host_ChangeGame_f , " change game " ) ;
Cvar_Get ( " host_allow_changegame " , " 1 " , FCVAR_READ_ONLY , " allows to change games " ) ;
}
else
{
Cvar_Get ( " host_allow_changegame " , " 0 " , FCVAR_READ_ONLY , " allows to change games " ) ;
}
SV_Init ( ) ;
CL_Init ( ) ;
2018-04-17 03:56:39 +03:00
ID_Init ( ) ;
2018-05-01 17:27:14 +03:00
if ( Host_IsDedicated ( ) )
2018-04-13 19:23:45 +03:00
{
2018-04-14 04:08:28 +03:00
# ifdef _WIN32
Wcon_InitConsoleCommands ( ) ;
# endif
2018-04-13 19:23:45 +03:00
Cmd_AddCommand ( " quit " , Sys_Quit , " quit the game " ) ;
Cmd_AddCommand ( " exit " , Sys_Quit , " quit the game " ) ;
}
else Cmd_AddCommand ( " minimize " , Host_Minimize_f , " minimize main window to tray " ) ;
host . errorframe = 0 ;
// post initializations
switch ( host . type )
{
case HOST_NORMAL :
2018-04-14 04:08:28 +03:00
# ifdef _WIN32
Wcon_ShowConsole ( false ) ; // hide console
# endif
2018-04-13 19:23:45 +03:00
// execute startup config and cmdline
Cbuf_AddText ( va ( " exec %s.rc \n " , SI . rcName ) ) ;
Cbuf_Execute ( ) ;
if ( ! host . config_executed )
{
Cbuf_AddText ( " exec config.cfg \n " ) ;
Cbuf_Execute ( ) ;
}
break ;
case HOST_DEDICATED :
// allways parse commandline in dedicated-mode
host . stuffcmds_pending = true ;
break ;
}
host . change_game = false ; // done
Cmd_RemoveCommand ( " setgl " ) ;
Cbuf_ExecStuffCmds ( ) ; // execute stuffcmds (commandline)
SCR_CheckStartupVids ( ) ; // must be last
oldtime = Sys_DoubleTime ( ) - 0.1 ;
2018-05-01 17:27:14 +03:00
if ( Host_IsDedicated ( ) & & GameState - > nextstate = = STATE_RUNFRAME )
2018-04-13 19:23:45 +03:00
Con_Printf ( " type 'map <mapname>' to run server... (TAB-autocomplete is working too) \n " ) ;
// main window message loop
while ( ! host . crashed )
{
newtime = Sys_DoubleTime ( ) ;
COM_Frame ( newtime - oldtime ) ;
oldtime = newtime ;
}
// never reached
return 0 ;
}
/*
= = = = = = = = = = = = = = = = =
Host_Shutdown
= = = = = = = = = = = = = = = = =
*/
void EXPORT Host_Shutdown ( void )
{
if ( host . shutdown_issued ) return ;
host . shutdown_issued = true ;
if ( host . status ! = HOST_ERR_FATAL ) host . status = HOST_SHUTDOWN ; // prepare host to normal shutdown
if ( ! host . change_game ) Q_strncpy ( host . finalmsg , " Server shutdown " , sizeof ( host . finalmsg ) ) ;
2018-04-18 18:32:30 +03:00
# ifndef XASH_DEDICATED
2018-04-13 19:23:45 +03:00
if ( host . type = = HOST_NORMAL )
Host_WriteConfig ( ) ;
2018-04-18 18:32:30 +03:00
# endif
2018-04-13 19:23:45 +03:00
2018-04-20 08:41:02 +00:00
SV_Shutdown ( " Server shutdown \n " ) ;
2018-04-13 19:23:45 +03:00
CL_Shutdown ( ) ;
Mod_Shutdown ( ) ;
NET_Shutdown ( ) ;
Host_FreeCommon ( ) ;
2018-04-14 04:08:28 +03:00
# ifdef _WIN32
Wcon_DestroyConsole ( ) ;
# endif
2018-04-13 19:23:45 +03:00
// must be last, console uses this
Mem_FreePool ( & host . mempool ) ;
2018-04-13 21:28:46 +03:00
// restore filter
Sys_RestoreCrashHandler ( ) ;
2018-05-03 23:51:23 +07:00
Sys_CloseLog ( ) ;
2018-04-13 19:23:45 +03:00
}