2005-09-27 04:05:32 +00:00
// log.c: handles console logging functions and cvars
# include "quakedef.h"
2006-04-12 00:17:02 +00:00
// cvar callbacks
2015-04-21 04:12:00 +00:00
static void QDECL Log_Dir_Callback ( struct cvar_s * var , char * oldvalue ) ;
static void QDECL Log_Name_Callback ( struct cvar_s * var , char * oldvalue ) ;
2006-04-12 00:17:02 +00:00
2005-09-27 04:05:32 +00:00
// cvars
# define CONLOGGROUP "Console logging"
2010-07-11 02:22:39 +00:00
cvar_t log_enable [ LOG_TYPES ] = { CVARF ( " log_enable " , " 0 " , CVAR_NOTFROMSERVER ) ,
2014-06-24 03:02:32 +00:00
CVARF ( " log_enable_players " , " 0 " , CVAR_NOTFROMSERVER ) ,
CVARF ( " log_enable_rcon " , " 1 " , CVAR_NOTFROMSERVER )
} ;
2010-07-11 02:22:39 +00:00
cvar_t log_name [ LOG_TYPES ] = { CVARFC ( " log_name " , " " , CVAR_NOTFROMSERVER , Log_Name_Callback ) ,
2014-09-02 02:44:43 +00:00
CVARFC ( " log_name_players " , " players " , CVAR_NOTFROMSERVER , Log_Name_Callback ) ,
CVARFC ( " log_name_rcon " , " rcon " , CVAR_NOTFROMSERVER , Log_Name_Callback ) } ;
2017-09-21 00:52:52 +00:00
cvar_t log_dir_var = CVARFC ( " log_dir " , " " , CVAR_NOTFROMSERVER , Log_Dir_Callback ) ;
2013-10-08 14:28:11 +00:00
cvar_t log_readable = CVARFD ( " log_readable " , " 7 " , CVAR_NOTFROMSERVER , " Bitfield describing what to convert/strip. If 0, exact byte representation will be used. \n &1: Dequakify text. \n &2: Strip special markup. \n &4: Strip ansi control codes. " ) ;
2016-12-07 18:45:25 +00:00
cvar_t log_developer = CVARFD ( " log_developer " , " 0 " , CVAR_NOTFROMSERVER , " Enables logging of console prints when set to 1. Otherwise unimportant messages will not fill up your log files. " ) ;
2010-07-11 02:22:39 +00:00
cvar_t log_rotate_files = CVARF ( " log_rotate_files " , " 0 " , CVAR_NOTFROMSERVER ) ;
cvar_t log_rotate_size = CVARF ( " log_rotate_size " , " 131072 " , CVAR_NOTFROMSERVER ) ;
2013-10-08 14:28:11 +00:00
cvar_t log_timestamps = CVARF ( " log_timestamps " , " 1 " , CVAR_NOTFROMSERVER ) ;
# ifdef _WIN32
cvar_t log_dosformat = CVARF ( " log_dosformat " , " 1 " , CVAR_NOTFROMSERVER ) ;
# else
2010-07-11 02:22:39 +00:00
cvar_t log_dosformat = CVARF ( " log_dosformat " , " 0 " , CVAR_NOTFROMSERVER ) ;
2013-10-08 14:28:11 +00:00
# endif
qboolean log_newline [ LOG_TYPES ] ;
2005-09-27 04:05:32 +00:00
2018-11-19 06:37:25 +00:00
# ifdef IPLOG
cvar_t iplog_autodump = CVARFD ( " ipautodump " , " 1 " , CVAR_NOTFROMSERVER , " Enables dumping the 'iplog.txt' file, which contains a log of usernames seen for a given IP, which is useful for detecting fake-nicks. " ) ;
# endif
2017-09-21 00:52:52 +00:00
static char log_dir [ MAX_OSPATH ] ;
static enum fs_relative log_root = FS_GAMEONLY ;
2005-09-27 04:05:32 +00:00
2006-04-12 00:17:02 +00:00
// Log_Dir_Callback: called when a log_dir is changed
2015-04-21 04:12:00 +00:00
static void QDECL Log_Dir_Callback ( struct cvar_s * var , char * oldvalue )
2006-04-12 00:17:02 +00:00
{
char * t = var - > string ;
2014-01-17 08:31:16 +00:00
char * e = t + ( * t ? strlen ( t ) : 0 ) ;
2006-04-12 00:17:02 +00:00
2014-01-17 08:31:16 +00:00
// sanity check for directory. // is equivelent to /../ on some systems, so make sure that can't be used either. : is for drives on windows or amiga, or alternative thingies on windows, so block thoses completely.
if ( strstr ( t , " .. " ) | | strstr ( t , " : " ) | | * t = = ' / ' | | * t = = ' \\ ' | | * e = = ' / ' | | * e = = ' \\ ' | | strstr ( t , " // " ) | | strstr ( t , " \\ \\ " ) )
2006-04-12 00:17:02 +00:00
{
2007-09-23 15:28:06 +00:00
Con_Printf ( CON_NOTICE " %s forced to default due to invalid characters. \n " , var - > name ) ;
2006-04-12 00:17:02 +00:00
// recursion is avoided by assuming the default value is sane
Cvar_ForceSet ( var , var - > defaultstr ) ;
}
2017-09-21 00:52:52 +00:00
if ( ! strncmp ( var - > string , " ./ " , 2 ) | | ! strncmp ( var - > string , " . \\ " , 2 ) )
{
strcpy ( log_dir , var - > string + 2 ) ;
log_root = FS_ROOT ;
}
else
{
strcpy ( log_dir , var - > string ) ;
log_root = FS_GAMEONLY ;
}
2006-04-12 00:17:02 +00:00
}
2007-07-27 21:24:31 +00:00
// Log_Name_Callback: called when a log_name is changed
2015-04-21 04:12:00 +00:00
static void QDECL Log_Name_Callback ( struct cvar_s * var , char * oldvalue )
2006-04-12 00:17:02 +00:00
{
char * t = var - > string ;
// sanity check for directory
if ( strstr ( t , " .. " ) | | strstr ( t , " : " ) | | strstr ( t , " / " ) | | strstr ( t , " \\ " ) )
{
2007-09-23 15:28:06 +00:00
Con_Printf ( CON_NOTICE " %s forced to default due to invalid characters. \n " , var - > name ) ;
2006-04-12 00:17:02 +00:00
// recursion is avoided by assuming the default value is sane
Cvar_ForceSet ( var , var - > defaultstr ) ;
}
}
2005-09-27 04:05:32 +00:00
// Con_Log: log string to console log
2017-08-14 16:38:44 +00:00
void Log_String ( logtype_t lognum , const char * s )
2005-09-27 04:05:32 +00:00
{
2007-07-27 21:24:31 +00:00
vfsfile_t * fi ;
2005-09-27 04:05:32 +00:00
char * f ; // filename
char * t ;
2013-10-08 14:28:11 +00:00
char utf8 [ 2048 ] ;
2005-09-27 04:05:32 +00:00
int i ;
2014-12-06 02:19:34 +00:00
char fbase [ MAX_QPATH ] ;
------------------------------------------------------------------------
r4169 | acceptthis | 2013-01-17 08:55:12 +0000 (Thu, 17 Jan 2013) | 31 lines
removed MAX_VISEDICTS limit.
PEXT2_REPLACEMENTDELTAS tweaked, now has 4 million entity limit. still not enabled by default.
TE_BEAM now maps to a separate TEQW_BEAM to avoid conflicts with QW.
added android multitouch emulation for windows/rawinput (in_simulatemultitouch).
split topcolor/bottomcolor from scoreboard, for dp's colormap|1024 feature.
now using utf-8 for windows consoles.
qcc warnings/errors now give clickable console links for quick+easy editing.
disabled menutint when the currently active item changes contrast or gamma (for OneManClan).
Added support for drawfont/drawfontscale.
tweaked the qcvm a little to reduce the number of pointers.
.doll file loading. still experimental and will likely crash. requires csqc active, even if its a dummy progs. this will be fixed in time. Still other things that need cleaning up.
windows: gl_font "?" shows the standard windows font-selection dialog, and can be used to select windows fonts. not all work. and you probably don't want to use windings.
fixed splitscreen support when playing mvds. added mini-scoreboards to splitscreen.
editor/debugger now shows asm if there's no linenumber info. also, pressing f1 for help shows the shortcuts.
Added support for .framegroups files for psk(psa) and iqm formats.
True support for ezquake's colour codes. Mutually exclusive with background colours.
path command output slightly more readable.
added support for digest_hex (MD4, SHA1, CRC16).
skingroups now colourmap correctly.
Fix terrain colour hints, and litdata from the wrong bsp.
fix ftp dual-homed issue. support epsv command, and enable ipv6 (eprt still not supported).
remove d3d11 compilation from the makefile. the required headers are not provided by mingw, and are not available to the build bot, so don't bother.
fix v *= v.x and similar opcodes.
fteqcc: fixed support for áéÃóú type chars in names. utf-8 files now properly supported (even with the utf-8 bom/identifier). utf-16 also supported.
fteqcc: fixed '#if 1 == 3 && 4' parsing.
fteqcc: -Werror acts on the warning, rather than as a separate error. Line numbers are thus more readable.
fteqcc: copyright message now includes compile date instead.
fteqccgui: the treeview control is now coloured depending on whether there were warnings/errors in the last compile.
fteqccgui: the output window is now focused and scrolls down as compilation progresses.
pr_dumpplatform command dumps out some pragmas to convert more serious warnings to errors. This is to avoid the infamous 'fteqcc sucks cos my code sucks' issue.
rewrote prespawn/modelist/soundlist code. server tracks progress now.
------------------------------------------------------------------------
git-svn-id: https://svn.code.sf.net/p/fteqw/code/trunk@4167 fc73d0e0-1445-4013-8a0c-d673dee63da5
2013-03-12 22:29:40 +00:00
char fname [ MAX_QPATH ] ;
2013-10-08 14:28:11 +00:00
conchar_t cline [ 2048 ] , * c ;
2016-01-18 05:22:07 +00:00
unsigned int u , flags ;
2005-09-27 04:05:32 +00:00
2007-07-27 21:24:31 +00:00
f = NULL ;
switch ( lognum )
{
case LOG_CONSOLE :
f = " qconsole " ;
break ;
case LOG_PLAYER :
f = " players " ;
break ;
2014-06-24 03:02:32 +00:00
case LOG_RCON :
f = " rcon " ;
break ;
2007-07-27 21:24:31 +00:00
default :
2014-08-25 07:35:41 +00:00
return ;
2007-07-27 21:24:31 +00:00
}
2014-08-25 07:35:41 +00:00
if ( ! log_enable [ lognum ] . value )
return ;
2007-07-27 21:24:31 +00:00
if ( log_name [ lognum ] . string [ 0 ] )
f = log_name [ lognum ] . string ;
if ( ! f )
return ;
2005-09-27 04:05:32 +00:00
2013-10-08 14:28:11 +00:00
COM_ParseFunString ( CON_WHITEMASK , s , cline , sizeof ( cline ) , ! ( log_readable . ival & 2 ) ) ;
t = utf8 ;
2016-01-18 05:22:07 +00:00
for ( c = cline ; * c ; )
2005-09-27 04:05:32 +00:00
{
2016-01-18 05:22:07 +00:00
c = Font_Decode ( c , & flags , & u ) ;
if ( ( flags & CON_HIDDEN ) & & ( log_readable . ival & 2 ) )
2013-10-08 14:28:11 +00:00
continue ;
if ( log_readable . ival & 1 )
2016-01-18 05:22:07 +00:00
u = COM_DeQuake ( u ) ;
2005-09-27 04:05:32 +00:00
2013-10-08 14:28:11 +00:00
//at the start of a new line, we might want a timestamp (so timestamps are correct for the first char of the line, instead of the preceeding \n)
if ( log_newline [ lognum ] )
2005-09-27 04:05:32 +00:00
{
2013-10-08 14:28:11 +00:00
if ( log_timestamps . ival )
2005-09-27 04:05:32 +00:00
{
2013-10-08 14:28:11 +00:00
time_t unixtime = time ( NULL ) ;
2014-05-30 03:57:30 +00:00
int bufferspace = utf8 + sizeof ( utf8 ) - 1 - t ;
if ( bufferspace > 0 )
{
strftime ( t , bufferspace , " %Y-%m-%d %H:%M:%S " , localtime ( & unixtime ) ) ;
t + = strlen ( t ) ;
}
2005-09-27 04:05:32 +00:00
}
2013-10-08 14:28:11 +00:00
log_newline [ lognum ] = false ;
2005-09-27 04:05:32 +00:00
}
2013-10-08 14:28:11 +00:00
//make sure control codes are stripped. no exploiting xterm bugs please.
if ( ( log_readable . ival & 4 ) & & ( ( u < 32 & & u ! = ' \t ' & & u ! = ' \n ' ) | | u = = 127 | | ( u > = 128 & & u < 128 + 32 ) ) ) //\r is stripped too
u = ' ? ' ;
//if dos format logs, we insert a \r before every \n (also flag next char as the start of a new line)
if ( u = = ' \n ' )
2005-09-27 04:05:32 +00:00
{
2013-10-08 14:28:11 +00:00
log_newline [ lognum ] = true ;
if ( log_dosformat . ival )
t + = utf8_encode ( t , ' \r ' , utf8 + sizeof ( utf8 ) - 1 - t ) ;
2005-09-27 04:05:32 +00:00
}
2013-10-08 14:28:11 +00:00
t + = utf8_encode ( t , u , utf8 + sizeof ( utf8 ) - 1 - t ) ;
2005-09-27 04:05:32 +00:00
}
* t = 0 ;
2017-09-21 00:52:52 +00:00
if ( * log_dir )
Q_snprintfz ( fbase , sizeof ( fname ) - 4 , " %s/%s " , log_dir , f ) ;
2014-01-17 08:31:16 +00:00
else
2014-12-06 02:19:34 +00:00
Q_snprintfz ( fbase , sizeof ( fname ) - 4 , " %s " , f ) ;
Q_snprintfz ( fname , sizeof ( fname ) , " %s.log " , fbase ) ;
2005-09-27 04:05:32 +00:00
// file rotation
2006-03-04 20:43:48 +00:00
if ( log_rotate_size . value > = 4096 & & log_rotate_files . value > = 1 )
2005-09-27 04:05:32 +00:00
{
int x ;
2005-12-21 03:07:33 +00:00
vfsfile_t * fi ;
2005-09-27 04:05:32 +00:00
// check file size, use x as temp
2017-09-21 00:52:52 +00:00
if ( ( fi = FS_OpenVFS ( fname , " rb " , log_root ) ) )
2005-09-27 04:05:32 +00:00
{
2005-12-21 03:07:33 +00:00
x = VFS_GETLEN ( fi ) ;
VFS_CLOSE ( fi ) ;
2013-10-08 14:28:11 +00:00
x + = strlen ( utf8 ) ; // add string size to file size to never go over
2005-09-27 04:05:32 +00:00
}
else
x = 0 ;
if ( x > ( int ) log_rotate_size . value )
{
2016-08-25 00:12:14 +00:00
char newf [ MAX_QPATH ] ;
char oldf [ MAX_QPATH ] ;
2005-09-27 04:05:32 +00:00
i = log_rotate_files . value ;
2006-03-04 20:43:48 +00:00
2005-09-27 04:05:32 +00:00
// unlink file at the top of the chain
2014-12-06 02:19:34 +00:00
Q_snprintfz ( oldf , sizeof ( oldf ) , " %s.%i.log " , fbase , i ) ;
2017-09-21 00:52:52 +00:00
FS_Remove ( oldf , log_root ) ;
2005-09-27 04:05:32 +00:00
// rename files through chain
for ( x = i - 1 ; x > 0 ; x - - )
{
strcpy ( newf , oldf ) ;
2014-12-06 02:19:34 +00:00
Q_snprintfz ( oldf , sizeof ( oldf ) , " %s.%i.log " , fbase , x ) ;
2005-09-27 04:05:32 +00:00
// check if file exists, otherwise skip
2017-09-21 00:52:52 +00:00
if ( ( fi = FS_OpenVFS ( oldf , " rb " , log_root ) ) )
2005-12-21 03:07:33 +00:00
VFS_CLOSE ( fi ) ;
2005-09-27 04:05:32 +00:00
else
continue ; // skip nonexistant files
2017-09-21 00:52:52 +00:00
if ( ! FS_Rename ( oldf , newf , log_root ) )
2005-09-27 04:05:32 +00:00
{
// rename failed, disable log and bug out
2007-07-27 21:24:31 +00:00
Cvar_ForceSet ( & log_enable [ lognum ] , " 0 " ) ;
2005-09-27 04:05:32 +00:00
Con_Printf ( " Unable to rotate log files. Logging disabled. \n " ) ;
return ;
}
}
// TODO: option to compress file somewhere in here?
2013-10-08 14:28:11 +00:00
// rename our base file, which had better exist...
2017-09-21 00:52:52 +00:00
if ( ! FS_Rename ( fname , oldf , log_root ) )
2005-09-27 04:05:32 +00:00
{
// rename failed, disable log and bug out
2007-07-27 21:24:31 +00:00
Cvar_ForceSet ( & log_enable [ lognum ] , " 0 " ) ;
2005-09-27 04:05:32 +00:00
Con_Printf ( " Unable to rename base log file. Logging disabled. \n " ) ;
return ;
}
}
}
2017-09-21 00:52:52 +00:00
FS_CreatePath ( fname , log_root ) ;
if ( ( fi = FS_OpenVFS ( fname , " ab " , log_root ) ) )
2007-07-27 21:24:31 +00:00
{
2013-10-08 14:28:11 +00:00
VFS_WRITE ( fi , utf8 , strlen ( utf8 ) ) ;
2007-07-27 21:24:31 +00:00
VFS_CLOSE ( fi ) ;
}
else
2005-12-05 16:47:29 +00:00
{
// write failed, bug out
2007-07-27 21:24:31 +00:00
Cvar_ForceSet ( & log_enable [ lognum ] , " 0 " ) ;
2005-12-05 16:47:29 +00:00
Con_Printf ( " Unable to write to log file. Logging disabled. \n " ) ;
return ;
}
2005-09-27 04:05:32 +00:00
}
2017-08-14 16:38:44 +00:00
void Con_Log ( const char * s )
2007-07-27 21:24:31 +00:00
{
Log_String ( LOG_CONSOLE , s ) ;
}
2007-08-07 14:41:23 +00:00
# ifndef CLIENTONLY
2007-07-27 21:24:31 +00:00
//still to add stuff at:
//connects
//disconnects
//kicked
void SV_LogPlayer ( client_t * cl , char * msg )
{
char line [ 2048 ] ;
2008-06-08 14:37:57 +00:00
char remote_adr [ MAX_ADR_SIZE ] ;
char realip_adr [ MAX_ADR_SIZE ] ;
2008-12-06 02:08:57 +00:00
if ( cl - > protocol = = SCP_BAD )
return ; //don't log botclients
2018-07-05 16:21:44 +00:00
Q_snprintfz ( line , sizeof ( line ) - 1 ,
" %s \\ %s \\ %i \\ %s \\ %s \\ %i \\ guid \\ %s " ,
2011-05-15 13:23:13 +00:00
msg , cl - > name , cl - > userid ,
2013-05-03 04:28:08 +00:00
NET_BaseAdrToString ( remote_adr , sizeof ( remote_adr ) , & cl - > netchan . remote_address ) , ( cl - > realip_status > 0 ? NET_BaseAdrToString ( realip_adr , sizeof ( realip_adr ) , & cl - > realip ) : " ?? " ) ,
2018-07-05 16:21:44 +00:00
cl - > netchan . remote_address . port , cl - > guid ) ;
InfoBuf_ToString ( & cl - > userinfo , line + strlen ( line ) , sizeof ( line ) - 1 - strlen ( line ) , NULL , NULL , NULL , NULL , NULL ) ;
Q_strncatz ( line , " \n " , sizeof ( line ) ) ;
2007-07-27 21:24:31 +00:00
Log_String ( LOG_PLAYER , line ) ;
}
2007-08-07 14:41:23 +00:00
# endif
2007-07-27 21:24:31 +00:00
void Log_Logfile_f ( void )
{
if ( log_enable [ LOG_CONSOLE ] . value )
{
Cvar_SetValue ( & log_enable [ LOG_CONSOLE ] , 0 ) ;
Con_Printf ( " Logging disabled. \n " ) ;
}
else
{
2017-09-21 00:52:52 +00:00
char * f ;
char syspath [ MAX_OSPATH ] ;
2007-07-27 21:24:31 +00:00
f = " qconsole " ;
if ( log_name [ LOG_CONSOLE ] . string [ 0 ] )
f = log_name [ LOG_CONSOLE ] . string ;
2017-09-21 00:52:52 +00:00
if ( * log_dir )
f = va ( " %s/%s.log " , log_dir , f ) ;
else
f = va ( " %s.log " , f ) ;
if ( FS_NativePath ( f , log_root , syspath , sizeof ( syspath ) ) )
Con_Printf ( " %s " , va ( " Logging to %s \n " , syspath ) ) ;
else
Con_Printf ( " %s " , va ( " Logging to %s \n " , f ) ) ;
2007-07-27 21:24:31 +00:00
Cvar_SetValue ( & log_enable [ LOG_CONSOLE ] , 1 ) ;
}
}
/*
void SV_Fraglogfile_f ( void )
{
2016-08-25 00:12:14 +00:00
char name [ MAX_QPATH ] ;
2007-07-27 21:24:31 +00:00
int i ;
if ( sv_fraglogfile )
{
2013-11-29 14:36:47 +00:00
Con_TPrintf ( " Frag file logging off. \n " ) ;
2007-07-27 21:24:31 +00:00
VFS_CLOSE ( sv_fraglogfile ) ;
sv_fraglogfile = NULL ;
return ;
}
// find an unused name
for ( i = 0 ; i < 1000 ; i + + )
{
sprintf ( name , " frag_%i.log " , i ) ;
sv_fraglogfile = FS_OpenVFS ( name , " rb " , FS_GAME ) ;
if ( ! sv_fraglogfile )
{ // can't read it, so create this one
sv_fraglogfile = FS_OpenVFS ( name , " wb " , FS_GAME ) ;
if ( ! sv_fraglogfile )
i = 1000 ; // give error
break ;
}
VFS_CLOSE ( sv_fraglogfile ) ;
}
if ( i = = 1000 )
{
2013-11-29 14:36:47 +00:00
Con_TPrintf ( " Can't open any logfiles. \n " ) ;
2007-07-27 21:24:31 +00:00
sv_fraglogfile = NULL ;
return ;
}
2013-11-29 14:36:47 +00:00
Con_TPrintf ( " Logging frags to %s. \n " , name ) ;
2007-07-27 21:24:31 +00:00
}
*/
2018-11-19 06:37:25 +00:00
# ifdef IPLOG
2017-02-11 16:14:06 +00:00
/*for fuck sake, why can people still not write simple files. proquake is writing binary files as text ones. this function is to try to deal with that fuckup*/
static size_t IPLog_Read_Fucked ( qbyte * file , size_t * offset , size_t totalsize , qbyte * out , size_t outsize )
{
size_t read = 0 ;
while ( outsize - - > 0 & & * offset < totalsize )
{
if ( file [ * offset ] = = ' \r ' & & * offset + 1 < totalsize & & file [ * offset + 1 ] = = ' \n ' )
{
out [ read ] = ' \n ' ;
* offset + = 2 ;
read + = 1 ;
}
else
{
out [ read ] = file [ * offset ] ;
* offset + = 1 ;
read + = 1 ;
}
}
return read ;
}
2018-11-19 06:37:25 +00:00
/*need to make sure any 13 bytes are followed by 10s so that we don't bug out when read back in *sigh* */
2017-02-11 16:14:06 +00:00
static size_t IPLog_Write_Fucked ( vfsfile_t * file , qbyte * out , size_t outsize )
{
qbyte tmp [ 64 ] ;
size_t write = 0 ;
size_t block = 0 ;
while ( outsize - - > 0 )
{
if ( block > = sizeof ( tmp ) - 4 )
{
VFS_WRITE ( file , tmp , block ) ;
write + = block ;
block = 0 ;
}
if ( * out = = ' \n ' )
tmp [ block + + ] = ' \r ' ;
tmp [ block + + ] = * out + + ;
}
if ( block )
{
VFS_WRITE ( file , tmp , block ) ;
write + = block ;
}
return write ;
}
2017-02-11 17:17:54 +00:00
qboolean IPLog_Merge_File ( const char * fname )
2017-01-24 10:27:39 +00:00
{
char ip [ MAX_ADR_SIZE ] ;
char name [ 256 ] ;
char line [ 1024 ] ;
vfsfile_t * f ;
if ( ! * fname )
fname = " iplog.txt " ;
2017-02-11 16:14:06 +00:00
f = FS_OpenVFS ( fname , " rb " , FS_PUBBASEGAMEONLY ) ;
if ( ! f )
f = FS_OpenVFS ( fname , " rb " , FS_GAME ) ;
2017-01-24 10:27:39 +00:00
if ( ! f )
return false ;
2017-02-11 16:14:06 +00:00
if ( ! Q_strcasecmp ( COM_FileExtension ( fname , name , sizeof ( name ) ) , " dat " ) )
2017-01-24 10:27:39 +00:00
{ //we don't write this format because of it being limited to ipv4, as well as player name lengths
2017-02-11 16:14:06 +00:00
size_t l = VFS_GETLEN ( f ) , offset = 0 ;
qbyte * ffs = malloc ( l + 1 ) ;
VFS_READ ( f , ffs , l ) ;
ffs [ l ] = 0 ;
while ( IPLog_Read_Fucked ( ffs , & offset , l , line , 20 ) = = 20 )
{ //yes, these addresses are weird.
Q_snprintfz ( ip , sizeof ( ip ) , " %i.%i.%i.xxx " , ( qbyte ) line [ 2 ] , ( qbyte ) line [ 1 ] , ( qbyte ) line [ 0 ] ) ;
2017-01-24 10:27:39 +00:00
memcpy ( name , line + 4 , 20 - 4 ) ;
name [ 20 - 4 ] = 0 ;
IPLog_Add ( ip , name ) ;
}
2017-02-11 16:14:06 +00:00
free ( ffs ) ;
2017-01-24 10:27:39 +00:00
}
else
{
while ( VFS_GETS ( f , line , sizeof ( line ) - 1 ) )
{
//whether the name contains quotes or what is an awkward one.
//we always write quotes (including string markup to avoid issues)
//dp doesn't, and our parser is lazy, so its possible we'll get gibberish that way
if ( COM_ParseOut ( COM_ParseOut ( line , ip , sizeof ( ip ) ) , name , sizeof ( name ) ) )
IPLog_Add ( ip , name ) ;
}
}
VFS_CLOSE ( f ) ;
return true ;
}
2017-02-11 16:14:06 +00:00
struct iplog_entry
{
netadr_t adr ;
netadr_t mask ;
char name [ 1 ] ;
} * * iplog_entries ;
size_t iplog_num , iplog_max ;
2017-01-24 10:27:39 +00:00
void IPLog_Add ( const char * ipstr , const char * name )
{
2017-02-11 16:14:06 +00:00
size_t i ;
netadr_t a , m ;
while ( * ipstr = = ' ' | | * ipstr = = ' \t ' )
ipstr + + ;
2017-01-24 10:27:39 +00:00
if ( * ipstr ! = ' [ ' & & * ipstr < ' 0 ' & & * ipstr > ' 9 ' )
return ;
2017-02-11 16:14:06 +00:00
if ( * ipstr = = ' [ ' )
ipstr + + ;
//some names are dodgy.
if ( ! * name
//|| !Q_strcasecmp(name, /*nq default*/"player") || !Q_strcasecmp(name, /*qw default*/"unnamed")
| | ! strcmp ( name , /*nq fallback*/ " unconnected " ) | | ! strncmp ( name , " BOT: " , 4 ) )
return ;
memset ( & a , 0 , sizeof ( a ) ) ;
memset ( & m , 0 , sizeof ( m ) ) ;
if ( ! NET_StringToAdrMasked ( ipstr , false , & a , & m ) )
return ;
2017-01-24 10:27:39 +00:00
//might be x.y.z.w:port
//might be x.y.z.FUCKED
//might be x.y.z.0/24
//might be [::]:port
//might be [::]/bits
//or other ways to express an ip address
2017-02-11 16:14:06 +00:00
//FIXME: ignore private addresses?
//check for dupes
for ( i = 0 ; i < iplog_num ; i + + )
{
if ( ! memcmp ( & a , & iplog_entries [ i ] - > adr , sizeof ( netadr_t ) ) & & ! memcmp ( & m , & iplog_entries [ i ] - > mask , sizeof ( netadr_t ) ) & & ! Q_strcasecmp ( name , iplog_entries [ i ] - > name ) )
return ;
}
//looks like its new...
if ( iplog_num = = iplog_max )
Z_ReallocElements ( ( void * * ) & iplog_entries , & iplog_max , iplog_max + 64 , sizeof ( * iplog_entries ) ) ;
iplog_entries [ iplog_num ] = BZ_Malloc ( sizeof ( struct iplog_entry ) + strlen ( name ) ) ;
iplog_entries [ iplog_num ] - > adr = a ;
iplog_entries [ iplog_num ] - > mask = m ;
strcpy ( iplog_entries [ iplog_num ] - > name , name ) ;
iplog_num + + ;
2017-01-24 10:27:39 +00:00
}
2017-02-11 16:14:06 +00:00
static void IPLog_Identify ( netadr_t * adr , netadr_t * mask , char * fmt , . . . )
{
va_list argptr ;
qboolean found = false ;
char line [ 256 ] ;
size_t i ;
va_start ( argptr , fmt ) ;
vsnprintf ( line , sizeof ( line ) , fmt , argptr ) ;
va_end ( argptr ) ;
Con_Printf ( " %s: " , line ) ;
for ( i = 0 ; i < iplog_num ; i + + )
{
if ( NET_CompareAdrMasked ( adr , & iplog_entries [ i ] - > adr , mask ? mask : & iplog_entries [ i ] - > mask ) )
{
if ( found )
Con_Printf ( " , " ) ;
found = true ;
Con_Printf ( " %s " , iplog_entries [ i ] - > name ) ;
}
}
if ( ! found )
Con_Printf ( " <no matches> " ) ;
Con_Printf ( " \n " ) ;
}
# include "cl_ignore.h"
2017-01-24 10:27:39 +00:00
static void IPLog_Identify_f ( void )
{
2017-02-11 16:14:06 +00:00
const char * nameorip = Cmd_Argv ( 1 ) ;
netadr_t adr , mask ;
char clean [ 256 ] ;
2017-02-19 00:15:42 +00:00
char * endofnum ;
strtoul ( nameorip , & endofnum , 10 ) ;
if ( * endofnum & & NET_StringToAdrMasked ( nameorip , false , & adr , & mask ) )
{ //if not a single number, try to parse as an ip
2017-02-11 16:14:06 +00:00
//treading carefully here, to avoid dns name lookups weirding everything out.
IPLog_Identify ( & adr , & mask , " Identity of %s " , NET_AdrToStringMasked ( clean , sizeof ( clean ) , & adr , & mask ) ) ;
}
# ifndef CLIENTONLY
else if ( sv . active )
{ //if server is active, walk players to see if there's a name match to get their address and guess an address mask
client_t * cl ;
int clnum = - 1 ;
while ( ( cl = SV_GetClientForString ( nameorip , & clnum ) ) )
{
if ( cl - > realip_status )
{
IPLog_Identify ( & cl - > realip , NULL , " Identity of %s (real) [%s] " , cl - > name , NET_AdrToString ( clean , sizeof ( clean ) , & cl - > realip ) ) ;
IPLog_Identify ( & cl - > netchan . remote_address , NULL , " Identity of %s (proxy) [%s] " , cl - > name , NET_AdrToString ( clean , sizeof ( clean ) , & cl - > realip ) ) ;
}
else
IPLog_Identify ( & cl - > netchan . remote_address , NULL , " Identity of %s [%s] " , cl - > name , NET_AdrToString ( clean , sizeof ( clean ) , & cl - > realip ) ) ;
}
}
# endif
# ifndef SERVERONLY
else if ( cls . state > = ca_connected )
{ //else if client is active, walk players to see if there's a name match, to get their address+mask if known via nq hacks
int slot ;
netadr_t adr ;
if ( ( slot = Player_StringtoSlot ( nameorip ) ) < 0 )
Con_Printf ( " %s: no player with userid %s \n " , Cmd_Argv ( 0 ) , nameorip ) ;
else if ( ! * cl . players [ slot ] . ip )
Con_Printf ( " %s: ip address of %s is not known \n " , Cmd_Argv ( 0 ) , cl . players [ slot ] . name ) ;
else
{
added r_meshpitch cvar that allows for fixing the unfixable mesh pitch bug from vanilla... needs a better name... do note that this will break pretty much any mod, so this is really only for TCs designed to use it. Its likely that I missed places.
nqsv: added support for spectators with nq clients. the angles are a bit rough, but hey. need to do something about frags so nq clients know who's a spectator. use 'cmd observe' to get an nq client to spectate on an fte server (then attack/jump behave the same as in qw clients).
nqsv: rewrote EF_MUZZLEFLASH handling, so svc_muzzleflash is now translated properly to EF_MUZZLEFLASH, and vice versa. No more missing muzzleflashes!
added screenshot_cubemap, so you can actually pre-generate cubemaps with fte (which can be used for reflections or whatever).
misc fixes (server crash, a couple of other less important ones).
external files based on a model's name will now obey r_replacemodels properly, instead of needing to use foo.mdl_0.skin for foo.md3.
identify <playernum> should now use the correct masked ip, instead of abrubtly failing (reported by kt)
vid_toggle console command should now obey vid_width and vid_height when switching to fullscreen, but only if vid_fullscreen is actually set, which should make it seem better behaved (reported by kt).
qcc: cleaned up sym->symboldata[sym->ofs] to be more consistent at all stages.
qcc: typedef float vec4[4]; now works to define a float array with 4 elements (however, it will be passed by-value rather than by-reference).
qcc: cleaned up optional vs __out ordering issues.
qccgui: shift+f3 searches backwards
git-svn-id: https://svn.code.sf.net/p/fteqw/code/trunk@5064 fc73d0e0-1445-4013-8a0c-d673dee63da5
2017-02-27 09:34:35 +00:00
if ( NET_StringToAdrMasked ( cl . players [ slot ] . ip , false , & adr , & mask ) )
IPLog_Identify ( & adr , & mask , " Identity of %s [%s] " , cl . players [ slot ] . name , cl . players [ slot ] . ip ) ;
else
Con_Printf ( " ip address of %s not known, cannot identify \n " , cl . players [ slot ] . name ) ;
2017-02-11 16:14:06 +00:00
}
}
# endif
else
Con_Printf ( " %s: not connected, nor raw address \n " , Cmd_Argv ( 0 ) ) ;
2017-01-24 10:27:39 +00:00
}
2017-07-28 01:49:25 +00:00
static int IPLog_Dump ( const char * fname )
2017-01-24 10:27:39 +00:00
{
2017-02-11 16:14:06 +00:00
size_t i ;
vfsfile_t * f ;
qbyte line [ 20 ] ;
2017-01-24 10:27:39 +00:00
if ( ! * fname )
fname = " iplog.txt " ;
2017-02-11 16:14:06 +00:00
2017-07-28 01:49:25 +00:00
if ( ! iplog_num & & ! COM_FCheckExists ( fname ) )
return 2 ; //no entries, nothing to overwrite
2017-02-11 16:14:06 +00:00
f = FS_OpenVFS ( fname , " wb " , FS_PUBBASEGAMEONLY ) ;
if ( ! f )
return false ;
if ( ! Q_strcasecmp ( COM_FileExtension ( fname , line , sizeof ( line ) ) , " dat " ) )
{
for ( i = 0 ; i < iplog_num ; i + + )
{
//this shitty format supports only ipv4.
if ( iplog_entries [ i ] - > adr . type ! = NA_IP )
continue ;
line [ 0 ] = iplog_entries [ i ] - > adr . address . ip [ 2 ] ;
line [ 1 ] = iplog_entries [ i ] - > adr . address . ip [ 1 ] ;
line [ 2 ] = iplog_entries [ i ] - > adr . address . ip [ 0 ] ;
line [ 3 ] = 0 ;
strncpy ( line + 4 , iplog_entries [ i ] - > name , sizeof ( line ) - 4 ) ;
IPLog_Write_Fucked ( f , line , sizeof ( line ) ) ; //convert \n to \r\n, to avoid fucking up any formatting with binary data (inside the address part, so *.13.10.* won't corrupt the file)
}
}
else
2017-01-24 10:27:39 +00:00
{
2017-02-11 16:14:06 +00:00
VFS_PRINTF ( f , " //generated by " FULLENGINENAME " \n " ) ;
for ( i = 0 ; i < iplog_num ; i + + )
{
char ip [ 512 ] ;
char buf [ 1024 ] ;
char buf2 [ 1024 ] ;
VFS_PRINTF ( f , log_dosformat . value ? " %s %s \r \n " : " %s %s \n " , COM_QuotedString ( NET_AdrToStringMasked ( ip , sizeof ( ip ) , & iplog_entries [ i ] - > adr , & iplog_entries [ i ] - > mask ) , buf2 , sizeof ( buf2 ) , false ) , COM_QuotedString ( iplog_entries [ i ] - > name , buf , sizeof ( buf ) , false ) ) ;
}
2017-01-24 10:27:39 +00:00
}
VFS_CLOSE ( f ) ;
2017-02-11 16:14:06 +00:00
return true ;
}
static void IPLog_Dump_f ( void )
{
char native [ MAX_OSPATH ] ;
const char * fname = Cmd_Argv ( 1 ) ;
if ( FS_NativePath ( fname , FS_GAMEONLY , native , sizeof ( native ) ) )
Q_strncpyz ( native , fname , sizeof ( native ) ) ;
IPLog_Merge_File ( fname ) ; //merge from the existing file, so that we're hopefully more robust if multiple processes are poking the same file.
2017-07-28 01:49:25 +00:00
switch ( IPLog_Dump ( fname ) )
2017-02-11 16:14:06 +00:00
{
2017-07-28 01:49:25 +00:00
case 0 :
Con_Printf ( " unable to write %s \n " , fname ) ;
break ;
default :
case 1 :
2017-02-11 16:14:06 +00:00
Con_Printf ( " wrote %s \n " , native ) ;
2017-07-28 01:49:25 +00:00
break ;
case 2 :
Con_Printf ( " nothing to write \n " ) ;
break ;
2017-02-11 16:14:06 +00:00
}
2017-01-24 10:27:39 +00:00
}
static void IPLog_Merge_f ( void )
{
const char * fname = Cmd_Argv ( 1 ) ;
if ( ! IPLog_Merge_File ( fname ) )
Con_Printf ( " unable to read %s \n " , fname ) ;
}
2018-11-19 06:37:25 +00:00
# endif
2017-09-20 11:27:13 +00:00
# ifndef SERVERONLY
struct certlog_s
{
link_t l ;
char * hostname ;
size_t certsize ;
qbyte cert [ 1 ] ;
} ;
static link_t certlog ;
static struct certlog_s * CertLog_Find ( const char * hostname )
{
struct certlog_s * l ;
for ( l = ( struct certlog_s * ) certlog . next ; l ! = ( struct certlog_s * ) & certlog ; l = ( struct certlog_s * ) l - > l . next )
{
if ( ! strcmp ( l - > hostname , hostname ) )
return l ;
}
return NULL ;
}
static void CertLog_Update ( const char * hostname , const void * cert , size_t certsize )
{
struct certlog_s * l = CertLog_Find ( hostname ) ;
if ( l )
{
RemoveLink ( & l - > l ) ;
Z_Free ( l ) ;
}
l = Z_Malloc ( sizeof ( * l ) + certsize + strlen ( hostname ) ) ;
l - > certsize = certsize ;
l - > hostname = l - > cert + l - > certsize ;
memcpy ( l - > cert , cert , certsize ) ;
strcpy ( l - > hostname , hostname ) ;
InsertLinkAfter ( & l - > l , & certlog ) ;
}
static void CertLog_Write ( void )
{
struct certlog_s * l ;
vfsfile_t * f = NULL ; //FS_OpenVFS("knowncerts.txt", "wb", FS_ROOT);
if ( f )
{
VFS_PRINTF ( f , " version 1.0 \n " ) ;
for ( l = ( struct certlog_s * ) certlog . next ; l ! = ( struct certlog_s * ) & certlog ; l = ( struct certlog_s * ) l - > l . next )
{
char certhex [ 32768 ] ;
size_t i ;
const char * hex = " 0123456789abcdef " ;
for ( i = 0 ; i < l - > certsize ; i + + )
{
certhex [ i * 2 + 0 ] = hex [ l - > cert [ i ] > > 4 ] ;
certhex [ i * 2 + 1 ] = hex [ l - > cert [ i ] & 0xf ] ;
}
certhex [ i * 2 ] = 0 ;
VFS_PRINTF ( f , " %s \" %s \" \n " , l - > hostname , certhex ) ;
}
}
}
static void CertLog_Purge ( void )
{
while ( certlog . next ! = & certlog )
{
struct certlog_s * l = ( struct certlog_s * ) certlog . next ;
RemoveLink ( & l - > l ) ;
Z_Free ( l ) ;
}
}
static int hexdecode ( char c )
{
if ( c > = ' 0 ' & & c < = ' 9 ' )
return c - ' 0 ' ;
else if ( c > = ' a ' & & c < = ' f ' )
return c - ' a ' + 10 ;
else if ( c > = ' A ' & & c < = ' F ' )
return c - ' A ' + 10 ;
return 0 ;
}
static void CertLog_Import ( const char * filename )
{
char addressstring [ 512 ] ;
char certhex [ 32768 ] ;
char certdata [ 16384 ] ;
char line [ 65536 ] , * l ;
size_t i , certsize ;
vfsfile_t * f = FS_OpenVFS ( filename , " rb " , FS_ROOT ) ;
//CertLog_Purge();
VFS_GETS ( f , line , sizeof ( line ) ) ;
if ( strncmp ( line , " version 1. " , 10 ) )
return ; //unsupported...
while ( VFS_GETS ( f , line , sizeof ( line ) ) )
{
l = line ;
l = COM_ParseOut ( l , addressstring , sizeof ( addressstring ) ) ;
l = COM_ParseOut ( l , certhex , sizeof ( certhex ) ) ;
certsize = 0 ;
for ( i = 0 ; certsize < sizeof ( certdata ) ; i + + )
{
if ( ! certhex [ ( i < < 1 ) + 0 ] | | ! certhex [ ( i < < 1 ) + 1 ] )
break ;
certdata [ certsize + + ] = ( hexdecode ( certhex [ ( i < < 1 ) + 0 ] ) < < 4 ) | hexdecode ( certhex [ ( i < < 1 ) + 1 ] ) ;
}
CertLog_Update ( addressstring , certdata , certsize ) ;
}
}
static void CertLog_UntrustAll_f ( void )
{
CertLog_Purge ( ) ;
}
static void CertLog_Import_f ( void )
{
const char * fname = Cmd_Argv ( 1 ) ;
if ( ! * fname )
fname = " knowncerts.txt " ;
CertLog_Import ( fname ) ;
}
struct certprompt_s
{
char * hostname ;
size_t certsize ;
qbyte cert [ 1 ] ;
} ;
static struct certprompt_s * certlog_curprompt ;
static void CertLog_Add_Prompted ( void * vctx , int button )
{
struct certprompt_s * ctx = vctx ;
if ( button = = 0 ) //button_yes / button_left
{
CertLog_Update ( ctx - > hostname , ctx - > cert , ctx - > certsize ) ;
CertLog_Write ( ) ;
}
else
CL_Disconnect ( ) ;
certlog_curprompt = NULL ;
}
qboolean CertLog_ConnectOkay ( const char * hostname , void * cert , size_t certsize )
{
struct certlog_s * l = CertLog_Find ( hostname ) ;
if ( certlog_curprompt )
return false ;
if ( ! l | | l - > certsize ! = certsize | | memcmp ( l - > cert , cert , certsize ) )
{
if ( qrenderer )
{
struct certprompt_s * ctx = certlog_curprompt = Z_Malloc ( sizeof ( * ctx ) + certsize + strlen ( hostname ) ) ;
ctx - > hostname = ctx - > cert + certsize ;
ctx - > certsize = certsize ;
memcpy ( ctx - > cert , cert , certsize ) ;
strcpy ( ctx - > hostname , hostname ) ;
2017-10-12 12:02:25 +00:00
//FIXME: display some sort of fingerprint
2017-09-20 11:27:13 +00:00
if ( ! l )
2017-10-12 12:02:25 +00:00
M_Menu_Prompt ( CertLog_Add_Prompted , ctx , va ( " %s \n Server certificate is \n self-signed " , hostname ) , " Trust " , NULL , " Disconnect " ) ;
2017-09-20 11:27:13 +00:00
else
2017-10-12 12:02:25 +00:00
M_Menu_Prompt ( CertLog_Add_Prompted , ctx , va ( " %s \n ^1Server certificate HAS CHANGED \n Zomg \n ^bFlee in Terror " , hostname ) , " ReTrust " , NULL , " Disconnect " ) ;
2017-09-20 11:27:13 +00:00
}
return false ; //can't connect yet...
}
return true ;
}
# endif
2017-02-11 16:14:06 +00:00
void Log_ShutDown ( void )
{
2018-11-19 06:37:25 +00:00
# ifdef IPLOG
if ( iplog_autodump . ival )
IPLog_Dump ( " iplog.txt " ) ;
2017-02-11 16:14:06 +00:00
// IPLog_Dump("iplog.dat");
while ( iplog_num > 0 )
{
iplog_num - - ;
BZ_Free ( iplog_entries [ iplog_num ] ) ;
}
BZ_Free ( iplog_entries ) ;
iplog_entries = NULL ;
iplog_max = iplog_num = 0 ;
2018-11-19 06:37:25 +00:00
# endif
2017-02-11 16:14:06 +00:00
}
2007-07-27 21:24:31 +00:00
2005-09-27 04:05:32 +00:00
void Log_Init ( void )
{
2007-07-27 21:24:31 +00:00
int i ;
2005-09-27 04:05:32 +00:00
// register cvars
2007-07-27 21:24:31 +00:00
for ( i = 0 ; i < LOG_TYPES ; i + + )
{
Cvar_Register ( & log_enable [ i ] , CONLOGGROUP ) ;
Cvar_Register ( & log_name [ i ] , CONLOGGROUP ) ;
2013-10-08 14:28:11 +00:00
log_newline [ i ] = true ;
2007-07-27 21:24:31 +00:00
}
2017-09-21 00:52:52 +00:00
Cvar_Register ( & log_dir_var , CONLOGGROUP ) ;
2005-09-27 04:05:32 +00:00
Cvar_Register ( & log_readable , CONLOGGROUP ) ;
Cvar_Register ( & log_developer , CONLOGGROUP ) ;
Cvar_Register ( & log_rotate_size , CONLOGGROUP ) ;
Cvar_Register ( & log_rotate_files , CONLOGGROUP ) ;
Cvar_Register ( & log_dosformat , CONLOGGROUP ) ;
2013-10-08 14:28:11 +00:00
Cvar_Register ( & log_timestamps , CONLOGGROUP ) ;
2005-09-27 04:05:32 +00:00
2007-07-27 21:24:31 +00:00
Cmd_AddCommand ( " logfile " , Log_Logfile_f ) ;
2018-11-19 06:37:25 +00:00
# ifdef IPLOG
Cmd_AddCommandD ( " identify " , IPLog_Identify_f , " Looks up a player's ip to see if they're using a different name " ) ;
2017-01-24 10:27:39 +00:00
Cmd_AddCommand ( " ipmerge " , IPLog_Merge_f ) ;
Cmd_AddCommand ( " ipdump " , IPLog_Dump_f ) ;
2018-11-19 06:37:25 +00:00
Cvar_Register ( & iplog_autodump , CONLOGGROUP ) ;
# endif
2017-01-24 10:27:39 +00:00
2005-09-27 04:05:32 +00:00
// cmd line options, debug options
# ifdef CRAZYDEBUGGING
Android: fat presses, vibrator, onscreen keyboard, keep-screen-on, console scaling, touch-based console scrolling, additional bindables.
Some memory leaks fixed.
latency with the nq protocol over loopback is much reduced.
Terrain: now mostly a property of a (q1 for now) bsp map, file format changed, glsl now built in, terrain editor builtin improved/changed, holes supported.
git-svn-id: https://svn.code.sf.net/p/fteqw/code/trunk@4067 fc73d0e0-1445-4013-8a0c-d673dee63da5
2012-07-14 16:25:18 +00:00
Cvar_ForceSet ( & log_enable [ LOG_CONSOLE ] , " 1 " ) ;
2005-09-27 04:05:32 +00:00
TRACE ( ( " dbg: Con_Init: log_enable forced \n " ) ) ;
# endif
if ( COM_CheckParm ( " -condebug " ) )
2007-07-27 21:24:31 +00:00
Cvar_ForceSet ( & log_enable [ LOG_CONSOLE ] , " 1 " ) ;
2017-09-20 11:27:13 +00:00
# ifndef SERVERONLY
ClearLink ( & certlog ) ;
Cmd_AddCommand ( " dtls_untrustall " , CertLog_UntrustAll_f ) ;
Cmd_AddCommand ( " dtls_importtrust " , CertLog_Import_f ) ;
# endif
2006-03-04 20:43:48 +00:00
}