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.
3572 lines
86 KiB
3572 lines
86 KiB
/* |
|
sv_client.c - client interactions |
|
Copyright (C) 2008 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. |
|
*/ |
|
|
|
#include "common.h" |
|
#include "const.h" |
|
#include "server.h" |
|
#include "net_encode.h" |
|
#include "net_api.h" |
|
|
|
const char *clc_strings[clc_lastmsg+1] = |
|
{ |
|
"clc_bad", |
|
"clc_nop", |
|
"clc_move", |
|
"clc_stringcmd", |
|
"clc_delta", |
|
"clc_resourcelist", |
|
"clc_unused6", |
|
"clc_fileconsistency", |
|
"clc_voicedata", |
|
"clc_cvarvalue", |
|
"clc_cvarvalue2", |
|
}; |
|
|
|
typedef struct ucmd_s |
|
{ |
|
const char *name; |
|
qboolean (*func)( sv_client_t *cl ); |
|
} ucmd_t; |
|
|
|
static int g_userid = 1; |
|
|
|
static void SV_UserinfoChanged( sv_client_t *cl ); |
|
static void SV_ExecuteClientCommand( sv_client_t *cl, const char *s ); |
|
|
|
/* |
|
================= |
|
SV_GetPlayerCount |
|
|
|
================= |
|
*/ |
|
void SV_GetPlayerCount( int *players, int *bots ) |
|
{ |
|
int i; |
|
|
|
*players = 0; |
|
*bots = 0; |
|
|
|
if( !svs.clients ) |
|
return; |
|
|
|
for( i = 0; i < svs.maxclients; i++ ) |
|
{ |
|
if( svs.clients[i].state >= cs_connected ) |
|
{ |
|
if( FBitSet( svs.clients[i].flags, FCL_FAKECLIENT )) |
|
(*bots)++; |
|
else |
|
(*players)++; |
|
} |
|
|
|
} |
|
} |
|
|
|
/* |
|
================= |
|
SV_GetChallenge |
|
|
|
Returns a challenge number that can be used |
|
in a subsequent client_connect command. |
|
We do this to prevent denial of service attacks that |
|
flood the server with invalid connection IPs. With a |
|
challenge, they must give a valid IP address. |
|
================= |
|
*/ |
|
static void SV_GetChallenge( netadr_t from ) |
|
{ |
|
int i, oldest = 0; |
|
double oldestTime; |
|
|
|
oldestTime = 0x7fffffff; |
|
|
|
// see if we already have a challenge for this ip |
|
for( i = 0; i < MAX_CHALLENGES; i++ ) |
|
{ |
|
if( !svs.challenges[i].connected && NET_CompareAdr( from, svs.challenges[i].adr )) |
|
break; |
|
|
|
if( svs.challenges[i].time < oldestTime ) |
|
{ |
|
oldestTime = svs.challenges[i].time; |
|
oldest = i; |
|
} |
|
} |
|
|
|
if( i == MAX_CHALLENGES ) |
|
{ |
|
// this is the first time this client has asked for a challenge |
|
svs.challenges[oldest].challenge = (COM_RandomLong( 0, 0xFFFF ) << 16) | COM_RandomLong( 0, 0xFFFF ); |
|
svs.challenges[oldest].adr = from; |
|
svs.challenges[oldest].time = host.realtime; |
|
svs.challenges[oldest].connected = false; |
|
i = oldest; |
|
} |
|
|
|
// send it back |
|
Netchan_OutOfBandPrint( NS_SERVER, svs.challenges[i].adr, "challenge %i", svs.challenges[i].challenge ); |
|
} |
|
|
|
static int SV_GetFragmentSize( void *pcl, fragsize_t mode ) |
|
{ |
|
sv_client_t *cl = (sv_client_t*)pcl; |
|
int cl_frag_size; |
|
|
|
if( Netchan_IsLocal( &cl->netchan )) |
|
return FRAGMENT_LOCAL_SIZE; |
|
|
|
if( mode == FRAGSIZE_UNRELIABLE ) |
|
{ |
|
// allow setting unreliable limit with "setinfo cl_urmax" |
|
cl_frag_size = Q_atoi( Info_ValueForKey( cl->userinfo, "cl_urmax" )); |
|
if( cl_frag_size == 0 ) |
|
return NET_MAX_MESSAGE; |
|
return bound( FRAGMENT_MAX_SIZE, cl_frag_size, NET_MAX_MESSAGE ); |
|
} |
|
|
|
cl_frag_size = Q_atoi( Info_ValueForKey( cl->userinfo, "cl_dlmax" )); |
|
cl_frag_size = bound( FRAGMENT_MIN_SIZE, cl_frag_size, FRAGMENT_MAX_SIZE ); |
|
|
|
if( mode != FRAGSIZE_FRAG ) |
|
{ |
|
if( cl->extensions & NET_EXT_SPLITSIZE ) |
|
return cl_frag_size; |
|
else |
|
return 0; // original engine behaviour |
|
} |
|
|
|
// get in-game fragmentation size |
|
if( cl->state == cs_spawned ) |
|
{ |
|
// allow setting in-game fragsize with "setinfo cl_frmax" |
|
int frmax = Q_atoi( Info_ValueForKey( cl->userinfo, "cl_frmax" )); |
|
|
|
if( frmax < FRAGMENT_MIN_SIZE || frmax > FRAGMENT_MAX_SIZE ) |
|
cl_frag_size /= 2; // add window for unreliable |
|
else |
|
cl_frag_size = frmax; |
|
} |
|
|
|
return cl_frag_size - HEADER_BYTES; |
|
} |
|
|
|
/* |
|
================ |
|
SV_RejectConnection |
|
|
|
Rejects connection request and sends back a message |
|
================ |
|
*/ |
|
void SV_RejectConnection( netadr_t from, const char *fmt, ... ) |
|
{ |
|
char text[1024]; |
|
va_list argptr; |
|
|
|
va_start( argptr, fmt ); |
|
Q_vsnprintf( text, sizeof( text ), fmt, argptr ); |
|
va_end( argptr ); |
|
|
|
Con_Reportf( "%s connection refused. Reason: %s\n", NET_AdrToString( from ), text ); |
|
Netchan_OutOfBandPrint( NS_SERVER, from, "errormsg\n^1Server was reject the connection:^7 %s", text ); |
|
Netchan_OutOfBandPrint( NS_SERVER, from, "print\n^1Server was reject the connection:^7 %s", text ); |
|
Netchan_OutOfBandPrint( NS_SERVER, from, "disconnect\n" ); |
|
} |
|
|
|
/* |
|
================ |
|
SV_FailDownload |
|
|
|
for some reasons file can't be downloaded |
|
tell the client about this problem |
|
================ |
|
*/ |
|
static void SV_FailDownload( sv_client_t *cl, const char *filename ) |
|
{ |
|
if( !COM_CheckString( filename )) |
|
return; |
|
|
|
MSG_BeginServerCmd( &cl->netchan.message, svc_filetxferfailed ); |
|
MSG_WriteString( &cl->netchan.message, filename ); |
|
} |
|
|
|
/* |
|
================ |
|
SV_CheckChallenge |
|
|
|
Make sure connecting client is not spoofing |
|
================ |
|
*/ |
|
static int SV_CheckChallenge( netadr_t from, int challenge ) |
|
{ |
|
int i; |
|
|
|
// see if the challenge is valid |
|
// don't care if it is a local address. |
|
if( NET_IsLocalAddress( from )) |
|
return 1; |
|
|
|
for( i = 0; i < MAX_CHALLENGES; i++ ) |
|
{ |
|
if( NET_CompareAdr( from, svs.challenges[i].adr )) |
|
{ |
|
if( challenge == svs.challenges[i].challenge ) |
|
break; // valid challenge |
|
#if 0 |
|
// g-cont. this breaks multiple connections from single machine |
|
SV_RejectConnection( from, "bad challenge %i\n", challenge ); |
|
return 0; |
|
#endif |
|
} |
|
} |
|
|
|
if( i == MAX_CHALLENGES ) |
|
{ |
|
SV_RejectConnection( from, "no challenge for your address\n" ); |
|
return 0; |
|
} |
|
svs.challenges[i].connected = true; |
|
|
|
return 1; |
|
} |
|
|
|
/* |
|
================ |
|
SV_CheckIPRestrictions |
|
|
|
Determine if client is outside appropriate address range |
|
================ |
|
*/ |
|
static int SV_CheckIPRestrictions( netadr_t from ) |
|
{ |
|
if( sv_lan.value ) |
|
{ |
|
if( !NET_CompareClassBAdr( from, net_local ) && !NET_IsReservedAdr( from )) |
|
return 0; |
|
} |
|
return 1; |
|
} |
|
|
|
/* |
|
================ |
|
SV_FindEmptySlot |
|
|
|
Get slot # and set client_t pointer for player, if possible |
|
We don't do this search on a "reconnect, we just reuse the slot |
|
================ |
|
*/ |
|
static int SV_FindEmptySlot( netadr_t from, int *pslot, sv_client_t **ppClient ) |
|
{ |
|
sv_client_t *cl; |
|
int i; |
|
|
|
for( i = 0, cl = svs.clients; i < svs.maxclients; i++, cl++ ) |
|
{ |
|
if( cl->state == cs_free ) |
|
{ |
|
*ppClient = cl; |
|
*pslot = i; |
|
return 1; |
|
} |
|
} |
|
|
|
SV_RejectConnection( from, "server is full\n" ); |
|
return 0; |
|
} |
|
|
|
/* |
|
================== |
|
SV_ConnectClient |
|
|
|
A connection request that did not come from the master |
|
================== |
|
*/ |
|
static void SV_ConnectClient( netadr_t from ) |
|
{ |
|
char userinfo[MAX_INFO_STRING]; |
|
char protinfo[MAX_INFO_STRING]; |
|
sv_client_t *cl, *newcl = NULL; |
|
qboolean reconnect = false; |
|
int nClientSlot = 0; |
|
int qport, version; |
|
int i, count = 0; |
|
int challenge; |
|
const char *s; |
|
int extensions; |
|
|
|
if( Cmd_Argc() < 5 ) |
|
{ |
|
SV_RejectConnection( from, "insufficient connection info\n" ); |
|
return; |
|
} |
|
|
|
version = Q_atoi( Cmd_Argv( 1 )); |
|
|
|
if( version != PROTOCOL_VERSION ) |
|
{ |
|
SV_RejectConnection( from, "unsupported protocol (%i should be %i)\n", version, PROTOCOL_VERSION ); |
|
return; |
|
} |
|
|
|
challenge = Q_atoi( Cmd_Argv( 2 )); // get challenge |
|
|
|
// see if the challenge is valid (local clients don't need to challenge) |
|
if( !SV_CheckChallenge( from, challenge )) |
|
return; |
|
|
|
s = Cmd_Argv( 3 ); // protocol info |
|
|
|
if( !Info_IsValid( s )) |
|
{ |
|
SV_RejectConnection( from, "invalid protinfo in connect command\n" ); |
|
return; |
|
} |
|
|
|
Q_strncpy( protinfo, s, sizeof( protinfo )); |
|
|
|
if( !SV_ProcessUserAgent( from, protinfo ) ) |
|
{ |
|
return; |
|
} |
|
|
|
// extract qport from protocol info |
|
qport = Q_atoi( Info_ValueForKey( protinfo, "qport" )); |
|
|
|
s = Info_ValueForKey( protinfo, "uuid" ); |
|
if( Q_strlen( s ) != 32 ) |
|
{ |
|
SV_RejectConnection( from, "invalid authentication certificate length\n" ); |
|
return; |
|
} |
|
|
|
extensions = Q_atoi( Info_ValueForKey( protinfo, "ext" ) ); |
|
|
|
// LAN servers restrict to class b IP addresses |
|
if( !SV_CheckIPRestrictions( from )) |
|
{ |
|
SV_RejectConnection( from, "LAN servers are restricted to local clients (class C)\n" ); |
|
return; |
|
} |
|
|
|
s = Cmd_Argv( 4 ); // user info |
|
|
|
if( Q_strlen( s ) > MAX_INFO_STRING || !Info_IsValid( s )) |
|
{ |
|
SV_RejectConnection( from, "invalid userinfo in connect command\n" ); |
|
return; |
|
} |
|
|
|
Q_strncpy( userinfo, s, sizeof( userinfo )); |
|
|
|
// check connection password (don't verify local client) |
|
if( !NET_IsLocalAddress( from ) && sv_password.string[0] && Q_stricmp( sv_password.string, Info_ValueForKey( userinfo, "password" ))) |
|
{ |
|
SV_RejectConnection( from, "invalid password\n" ); |
|
return; |
|
} |
|
|
|
// if there is already a slot for this ip, reuse it |
|
for( i = 0, cl = svs.clients; i < svs.maxclients; i++, cl++ ) |
|
{ |
|
if( cl->state == cs_free || cl->state == cs_zombie ) |
|
continue; |
|
|
|
if( NET_CompareBaseAdr( from, cl->netchan.remote_address ) && ( cl->netchan.qport == qport || from.port == cl->netchan.remote_address.port )) |
|
{ |
|
reconnect = true; |
|
newcl = cl; |
|
break; |
|
} |
|
} |
|
|
|
// A reconnecting client will re-use the slot found above when checking for reconnection. |
|
// the slot will be wiped clean. |
|
if( !reconnect ) |
|
{ |
|
// connect the client if there are empty slots. |
|
if( !SV_FindEmptySlot( from, &nClientSlot, &newcl )) |
|
return; |
|
} |
|
else |
|
{ |
|
Con_Reportf( S_NOTE "%s:reconnect\n", NET_AdrToString( from )); |
|
} |
|
|
|
// find a client slot |
|
ASSERT( newcl != NULL ); |
|
|
|
// build a new connection |
|
// accept the new client |
|
|
|
sv.current_client = newcl; |
|
newcl->edict = EDICT_NUM( (newcl - svs.clients) + 1 ); |
|
newcl->challenge = challenge; // save challenge for checksumming |
|
if( newcl->frames ) Mem_Free( newcl->frames ); |
|
newcl->frames = (client_frame_t *)Z_Calloc( sizeof( client_frame_t ) * SV_UPDATE_BACKUP ); |
|
newcl->userid = g_userid++; // create unique userid |
|
newcl->state = cs_connected; |
|
newcl->extensions = extensions & (NET_EXT_SPLITSIZE); |
|
Q_strncpy( newcl->useragent, protinfo, MAX_INFO_STRING ); |
|
|
|
// reset viewentities (from previous level) |
|
memset( newcl->viewentity, 0, sizeof( newcl->viewentity )); |
|
newcl->num_viewents = 0; |
|
// HACKHACK: can hear all players by default to avoid issues |
|
// with server.dll without voice game manager |
|
newcl->listeners = -1; |
|
|
|
// initailize netchan |
|
Netchan_Setup( NS_SERVER, &newcl->netchan, from, qport, newcl, SV_GetFragmentSize ); |
|
MSG_Init( &newcl->datagram, "Datagram", newcl->datagram_buf, sizeof( newcl->datagram_buf )); // datagram buf |
|
|
|
Q_strncpy( newcl->hashedcdkey, Info_ValueForKey( protinfo, "uuid" ), 32 ); |
|
newcl->hashedcdkey[32] = '\0'; |
|
|
|
// build protinfo answer |
|
protinfo[0] = '\0'; |
|
Info_SetValueForKeyf( protinfo, "ext", sizeof( protinfo ), "%d", newcl->extensions ); |
|
|
|
// send the connect packet to the client |
|
Netchan_OutOfBandPrint( NS_SERVER, from, "client_connect %s", protinfo ); |
|
|
|
newcl->upstate = us_inactive; |
|
newcl->connection_started = host.realtime; |
|
newcl->cl_updaterate = 0.05; // 20 fps as default |
|
newcl->delta_sequence = -1; |
|
newcl->flags = 0; |
|
|
|
|
|
|
|
// reset any remaining events |
|
memset( &newcl->events, 0, sizeof( newcl->events )); |
|
|
|
// parse some info from the info strings (this can override cl_updaterate) |
|
Q_strncpy( newcl->userinfo, userinfo, sizeof( newcl->userinfo )); |
|
SV_UserinfoChanged( newcl ); |
|
SV_ClearResourceLists( newcl ); |
|
#if 0 |
|
memset( &newcl->resourcesneeded, 0, sizeof( resource_t )); |
|
memset( &newcl->resourcesonhand, 0, sizeof( resource_t )); |
|
newcl->resourcesneeded.pNext = newcl->resourcesneeded.pPrev = &newcl->resourcesneeded; |
|
newcl->resourcesonhand.pNext = newcl->resourcesonhand.pPrev = &newcl->resourcesonhand; |
|
#endif |
|
newcl->next_messagetime = host.realtime + newcl->cl_updaterate; |
|
newcl->next_sendinfotime = 0.0; |
|
newcl->ignored_ents = 0; |
|
newcl->chokecount = 0; |
|
|
|
// reset stats |
|
newcl->next_checkpingtime = -1.0; |
|
newcl->packet_loss = 0.0f; |
|
|
|
// if this was the first client on the server, or the last client |
|
// the server can hold, send a heartbeat to the master. |
|
for( i = 0, cl = svs.clients; i < svs.maxclients; i++, cl++ ) |
|
if( cl->state >= cs_connected ) count++; |
|
|
|
Log_Printf( "\"%s<%i><%i><>\" connected, address \"%s\"\n", newcl->name, newcl->userid, i, NET_AdrToString( newcl->netchan.remote_address )); |
|
|
|
if( count == 1 || count == svs.maxclients ) |
|
NET_MasterClear(); |
|
} |
|
|
|
/* |
|
================== |
|
SV_FakeConnect |
|
|
|
A connection request that came from the game module |
|
================== |
|
*/ |
|
edict_t *GAME_EXPORT SV_FakeConnect( const char *netname ) |
|
{ |
|
char userinfo[MAX_INFO_STRING]; |
|
int i, count = 0; |
|
sv_client_t *cl; |
|
|
|
if( !COM_CheckString( netname )) |
|
netname = "Bot"; |
|
|
|
// find a client slot |
|
for( i = 0, cl = svs.clients; i < svs.maxclients; i++, cl++ ) |
|
{ |
|
if( cl->state == cs_free ) |
|
break; |
|
} |
|
|
|
if( i == svs.maxclients ) |
|
return NULL; // server is full |
|
|
|
userinfo[0] = '\0'; |
|
|
|
// setup fake client params |
|
Info_SetValueForKey( userinfo, "name", netname, MAX_INFO_STRING ); |
|
Info_SetValueForKey( userinfo, "model", "gordon", MAX_INFO_STRING ); |
|
Info_SetValueForKey( userinfo, "topcolor", "1", MAX_INFO_STRING ); |
|
Info_SetValueForKey( userinfo, "bottomcolor", "1", MAX_INFO_STRING ); |
|
|
|
// build a new connection |
|
// accept the new client |
|
sv.current_client = cl; |
|
|
|
if( cl->frames ) Mem_Free( cl->frames ); // fakeclients doesn't have frames |
|
memset( cl, 0, sizeof( sv_client_t )); |
|
|
|
cl->edict = EDICT_NUM( (cl - svs.clients) + 1 ); |
|
cl->userid = g_userid++; // create unique userid |
|
SetBits( cl->flags, FCL_FAKECLIENT ); |
|
|
|
// parse some info from the info strings |
|
Q_strncpy( cl->userinfo, userinfo, sizeof( cl->userinfo )); |
|
SV_UserinfoChanged( cl ); |
|
SetBits( cl->flags, FCL_RESEND_USERINFO ); |
|
cl->next_sendinfotime = 0.0; |
|
|
|
// if this was the first client on the server, or the last client |
|
// the server can hold, send a heartbeat to the master. |
|
for( i = 0, cl = svs.clients; i < svs.maxclients; i++, cl++ ) |
|
if( cl->state >= cs_connected ) count++; |
|
cl = sv.current_client; |
|
|
|
Log_Printf( "\"%s<%i><%i><>\" connected, address \"local\"\n", cl->name, cl->userid, i ); |
|
|
|
SetBits( cl->edict->v.flags, FL_CLIENT|FL_FAKECLIENT ); // mark it as fakeclient |
|
cl->connection_started = host.realtime; |
|
cl->state = cs_spawned; |
|
|
|
if( count == 1 || count == svs.maxclients ) |
|
NET_MasterClear(); |
|
|
|
return cl->edict; |
|
} |
|
|
|
/* |
|
================== |
|
SV_Kick_f |
|
|
|
Kick a user off of the server |
|
================== |
|
*/ |
|
void SV_KickPlayer( sv_client_t *cl, const char *fmt, ... ) |
|
{ |
|
const char *clientId; |
|
va_list va; |
|
char buf[MAX_VA_STRING]; |
|
|
|
if( NET_IsLocalAddress( cl->netchan.remote_address )) |
|
{ |
|
Con_Printf( "The local player cannot be kicked!\n" ); |
|
return; |
|
} |
|
|
|
clientId = SV_GetClientIDString( cl ); |
|
|
|
va_start( va, fmt ); |
|
Q_vsnprintf( buf, sizeof( buf ), fmt, va ); |
|
va_end( va ); |
|
|
|
if( buf[0] ) |
|
{ |
|
Log_Printf( "Kick: \"%s<%i><%s><>\" was kicked by \"Console\" (message \"%s\")\n", cl->name, cl->userid, clientId, buf ); |
|
SV_BroadcastPrintf( cl, "%s was kicked with message: \"%s\"\n", cl->name, buf ); |
|
SV_ClientPrintf( cl, "You were kicked from the game with message: \"%s\"\n", buf ); |
|
if( cl->useragent[0] ) |
|
Netchan_OutOfBandPrint( NS_SERVER, cl->netchan.remote_address, "errormsg\nKicked with message:\n%s\n", buf ); |
|
} |
|
else |
|
{ |
|
Log_Printf( "Kick: \"%s<%i><%s><>\" was kicked by \"Console\"\n", cl->name, cl->userid, clientId ); |
|
SV_BroadcastPrintf( cl, "%s was kicked\n", cl->name ); |
|
SV_ClientPrintf( cl, "You were kicked from the game\n" ); |
|
if( cl->useragent[0] ) |
|
Netchan_OutOfBandPrint( NS_SERVER, cl->netchan.remote_address, "errormsg\nYou were kicked from the game\n" ); |
|
} |
|
|
|
SV_DropClient( cl, false ); |
|
} |
|
|
|
/* |
|
===================== |
|
SV_DropClient |
|
|
|
Called when the player is totally leaving the server, either willingly |
|
or unwillingly. This is NOT called if the entire server is quiting |
|
or crashing. |
|
===================== |
|
*/ |
|
void SV_DropClient( sv_client_t *cl, qboolean crash ) |
|
{ |
|
int i; |
|
|
|
if( cl->state == cs_zombie ) |
|
return; // already dropped |
|
|
|
if( !crash ) |
|
{ |
|
// add the disconnect |
|
if( !FBitSet( cl->flags, FCL_FAKECLIENT ) ) |
|
{ |
|
MSG_BeginServerCmd( &cl->netchan.message, svc_disconnect ); |
|
} |
|
|
|
if( cl->edict && cl->state == cs_spawned ) |
|
{ |
|
svgame.dllFuncs.pfnClientDisconnect( cl->edict ); |
|
} |
|
|
|
if( !FBitSet( cl->flags, FCL_FAKECLIENT ) ) |
|
{ |
|
Netchan_TransmitBits( &cl->netchan, 0, NULL ); |
|
} |
|
} |
|
|
|
ClearBits( cl->flags, FCL_FAKECLIENT ); |
|
ClearBits( cl->flags, FCL_HLTV_PROXY ); |
|
cl->state = cs_zombie; // become free in a few seconds |
|
cl->name[0] = 0; |
|
|
|
if( cl->frames ) |
|
Mem_Free( cl->frames ); // release delta |
|
cl->frames = NULL; |
|
|
|
if( NET_CompareBaseAdr( cl->netchan.remote_address, host.rd.address )) |
|
SV_EndRedirect( &host.rd ); |
|
|
|
// throw away any residual garbage in the channel. |
|
Netchan_Clear( &cl->netchan ); |
|
|
|
// clean client data on disconnect |
|
memset( cl->userinfo, 0, MAX_INFO_STRING ); |
|
memset( cl->physinfo, 0, MAX_INFO_STRING ); |
|
COM_ClearCustomizationList( &cl->customdata, false ); |
|
|
|
// don't send to other clients |
|
cl->edict = NULL; |
|
|
|
// send notification to all other clients |
|
SV_FullClientUpdate( cl, &sv.reliable_datagram ); |
|
|
|
// if this was the last client on the server, send a heartbeat |
|
// to the master so it is known the server is empty |
|
// send a heartbeat now so the master will get up to date info |
|
// if there is already a slot for this ip, reuse it |
|
for( i = 0; i < svs.maxclients; i++ ) |
|
{ |
|
if( svs.clients[i].state >= cs_connected ) |
|
break; |
|
} |
|
|
|
if( i == svs.maxclients ) |
|
NET_MasterClear(); |
|
} |
|
|
|
/* |
|
============================================================================== |
|
|
|
SVC COMMAND REDIRECT |
|
|
|
============================================================================== |
|
*/ |
|
static void SV_BeginRedirect( host_redirect_t *rd, netadr_t adr, rdtype_t target, char *buffer, size_t buffersize, void (*flush)) |
|
{ |
|
rd->target = target; |
|
rd->buffer = buffer; |
|
rd->buffersize = buffersize; |
|
rd->flush = flush; |
|
rd->address = adr; |
|
rd->buffer[0] = 0; |
|
if( rd->lines == 0 ) |
|
rd->lines = -1; |
|
} |
|
|
|
static void SV_FlushRedirect( netadr_t adr, int dest, char *buf ) |
|
{ |
|
if( sv.current_client && FBitSet( sv.current_client->flags, FCL_FAKECLIENT )) |
|
return; |
|
|
|
switch( dest ) |
|
{ |
|
case RD_PACKET: |
|
Netchan_OutOfBandPrint( NS_SERVER, adr, "print\n%s", buf ); |
|
break; |
|
case RD_CLIENT: |
|
if( !sv.current_client ) return; // client not set |
|
MSG_BeginServerCmd( &sv.current_client->netchan.message, svc_print ); |
|
MSG_WriteString( &sv.current_client->netchan.message, buf ); |
|
break; |
|
case RD_NONE: |
|
Con_Printf( S_ERROR "SV_FlushRedirect: %s: invalid destination\n", NET_AdrToString( adr )); |
|
break; |
|
} |
|
} |
|
|
|
void SV_EndRedirect( host_redirect_t *rd ) |
|
{ |
|
if( rd->lines > 0 ) |
|
return; |
|
|
|
if( rd->flush ) |
|
rd->flush( rd->address, rd->target, rd->buffer ); |
|
|
|
rd->target = RD_NONE; |
|
rd->buffer = NULL; |
|
rd->buffersize = 0; |
|
rd->flush = NULL; |
|
} |
|
|
|
/* |
|
================ |
|
Rcon_Print |
|
|
|
Print message to rcon buffer and send to rcon redirect target |
|
================ |
|
*/ |
|
void Rcon_Print( host_redirect_t *rd, const char *pMsg ) |
|
{ |
|
size_t len; |
|
|
|
if( !rd->target || !rd->lines || !rd->flush || !rd->buffer ) |
|
return; |
|
|
|
len = Q_strncat( rd->buffer, pMsg, rd->buffersize ); |
|
|
|
if( len && rd->buffer[len - 1] == '\n' ) |
|
{ |
|
rd->flush( rd->address, rd->target, rd->buffer ); |
|
|
|
if( rd->lines > 0 ) |
|
rd->lines--; |
|
|
|
rd->buffer[0] = 0; |
|
|
|
if( !rd->lines ) |
|
Msg( "End of redirection!\n" ); |
|
} |
|
} |
|
|
|
/* |
|
=============== |
|
SV_GetClientIDString |
|
|
|
Returns a pointer to a static char for most likely only printing. |
|
=============== |
|
*/ |
|
const char *SV_GetClientIDString( sv_client_t *cl ) |
|
{ |
|
static char result[MAX_QPATH]; |
|
|
|
if( !cl ) return ""; |
|
|
|
if( FBitSet( cl->flags, FCL_FAKECLIENT )) |
|
{ |
|
Q_strncpy( result, "ID_BOT", sizeof( result )); |
|
} |
|
else if( NET_IsLocalAddress( cl->netchan.remote_address )) |
|
{ |
|
Q_strncpy( result, "ID_LOOPBACK", sizeof( result )); |
|
} |
|
else if( sv_lan.value ) |
|
{ |
|
Q_strncpy( result, "ID_LAN", sizeof( result )); |
|
} |
|
else |
|
{ |
|
Q_snprintf( result, sizeof( result ), "ID_%s", MD5_Print( (byte *)cl->hashedcdkey )); |
|
} |
|
|
|
return result; |
|
} |
|
|
|
sv_client_t *SV_ClientById( int id ) |
|
{ |
|
sv_client_t *cl; |
|
int i; |
|
|
|
ASSERT( id >= 0 ); |
|
|
|
for( i = 0, cl = svs.clients; i < svgame.globals->maxClients; i++, cl++ ) |
|
{ |
|
if( !cl->state ) |
|
continue; |
|
|
|
if( cl->userid == id ) |
|
return cl; |
|
} |
|
|
|
return NULL; |
|
} |
|
|
|
sv_client_t *SV_ClientByName( const char *name ) |
|
{ |
|
sv_client_t *cl; |
|
int i; |
|
|
|
if( !COM_CheckString( name )) |
|
return NULL; |
|
|
|
for( i = 0, cl = svs.clients; i < svgame.globals->maxClients; i++, cl++ ) |
|
{ |
|
if( !cl->state ) |
|
continue; |
|
|
|
if( !Q_strcmp( cl->name, name ) ) |
|
return cl; |
|
} |
|
|
|
return NULL; |
|
} |
|
|
|
/* |
|
================ |
|
SV_TestBandWidth |
|
|
|
================ |
|
*/ |
|
static void SV_TestBandWidth( netadr_t from ) |
|
{ |
|
const int version = Q_atoi( Cmd_Argv( 1 )); |
|
const int packetsize = Q_atoi( Cmd_Argv( 2 )); |
|
uint32_t crc; |
|
int ofs; |
|
|
|
// don't waste time of protocol mismatched |
|
if( version != PROTOCOL_VERSION ) |
|
{ |
|
SV_RejectConnection( from, "unsupported protocol (%i should be %i)\n", version, PROTOCOL_VERSION ); |
|
return; |
|
} |
|
|
|
// quickly reject invalid packets |
|
if( !svs.testpacket_buf || |
|
( packetsize <= FRAGMENT_MIN_SIZE ) || |
|
( packetsize > FRAGMENT_MAX_SIZE )) |
|
{ |
|
// skip the test and just get challenge |
|
SV_GetChallenge( from ); |
|
return; |
|
} |
|
|
|
// don't go out of bounds |
|
ofs = packetsize - svs.testpacket_filepos - 1; |
|
if(( ofs < 0 ) || ( ofs > svs.testpacket_filelen )) |
|
{ |
|
SV_GetChallenge( from ); |
|
return; |
|
} |
|
|
|
crc = svs.testpacket_crcs[ofs]; |
|
memcpy( svs.testpacket_crcpos, &crc, sizeof( crc )); |
|
|
|
// send the datagram |
|
NET_SendPacket( NS_SERVER, packetsize, MSG_GetData( &svs.testpacket ), from ); |
|
} |
|
|
|
/* |
|
================ |
|
SV_Ack |
|
|
|
================ |
|
*/ |
|
static void SV_Ack( netadr_t from ) |
|
{ |
|
Con_Printf( "ping %s\n", NET_AdrToString( from )); |
|
} |
|
|
|
/* |
|
================ |
|
SV_Info |
|
|
|
Responds with short info for broadcast scans |
|
The second parameter should be the current protocol version number. |
|
================ |
|
*/ |
|
static void SV_Info( netadr_t from, int protocolVersion ) |
|
{ |
|
char s[512]; |
|
|
|
// ignore in single player |
|
if( svs.maxclients == 1 || !svs.initialized ) |
|
return; |
|
|
|
s[0] = '\0'; |
|
|
|
if( protocolVersion != PROTOCOL_VERSION ) |
|
{ |
|
Q_snprintf( s, sizeof( s ), "%s: wrong version\n", hostname.string ); |
|
} |
|
else |
|
{ |
|
int count; |
|
int bots; |
|
int remaining; |
|
char temp[sizeof( s )]; |
|
qboolean have_password = COM_CheckStringEmpty( sv_password.string ); |
|
|
|
SV_GetPlayerCount( &count, &bots ); |
|
|
|
// a1ba: send protocol version to distinguish old engine and new |
|
Info_SetValueForKeyf( s, "p", sizeof( s ), "%i", PROTOCOL_VERSION ); |
|
Info_SetValueForKey( s, "map", sv.name, sizeof( s )); |
|
Info_SetValueForKey( s, "dm", svgame.globals->deathmatch ? "1" : "0", sizeof( s )); |
|
Info_SetValueForKey( s, "team", svgame.globals->teamplay ? "1" : "0", sizeof( s )); |
|
Info_SetValueForKey( s, "coop", svgame.globals->coop ? "1" : "0", sizeof( s )); |
|
Info_SetValueForKeyf( s, "numcl", sizeof( s ), "%i", count ); |
|
Info_SetValueForKeyf( s, "maxcl", sizeof( s ), "%i", svs.maxclients ); |
|
Info_SetValueForKey( s, "gamedir", GI->gamefolder, sizeof( s )); |
|
Info_SetValueForKey( s, "password", have_password ? "1" : "0", sizeof( s )); |
|
|
|
// write host last so we can try to cut off too long hostnames |
|
// TODO: value size limit for infostrings |
|
remaining = sizeof( s ) - Q_strlen( s ) - sizeof( "\\host\\" ) - 1; |
|
if( remaining < 0 ) |
|
{ |
|
// should never happen? |
|
Con_Printf( S_ERROR "SV_Info: infostring overflow!\n" ); |
|
return; |
|
} |
|
Q_strncpy( temp, hostname.string, remaining ); |
|
Info_SetValueForKey( s, "host", temp, sizeof( s )); |
|
} |
|
|
|
Netchan_OutOfBandPrint( NS_SERVER, from, "info\n%s", s ); |
|
} |
|
|
|
/* |
|
================ |
|
SV_BuildNetAnswer |
|
|
|
Responds with long info for local and broadcast requests |
|
================ |
|
*/ |
|
static void SV_BuildNetAnswer( netadr_t from ) |
|
{ |
|
char string[MAX_INFO_STRING]; |
|
int version, context, type; |
|
int i, count = 0; |
|
|
|
// ignore in single player |
|
if( svs.maxclients == 1 || !svs.initialized ) |
|
return; |
|
|
|
version = Q_atoi( Cmd_Argv( 1 )); |
|
context = Q_atoi( Cmd_Argv( 2 )); |
|
type = Q_atoi( Cmd_Argv( 3 )); |
|
|
|
if( version != PROTOCOL_VERSION ) |
|
{ |
|
// handle the unsupported protocol |
|
string[0] = '\0'; |
|
Info_SetValueForKey( string, "neterror", "protocol", MAX_INFO_STRING ); |
|
|
|
// send error unsupported protocol |
|
Netchan_OutOfBandPrint( NS_SERVER, from, "netinfo %i %i %s\n", context, type, string ); |
|
return; |
|
} |
|
|
|
if( type == NETAPI_REQUEST_PING ) |
|
{ |
|
Netchan_OutOfBandPrint( NS_SERVER, from, "netinfo %i %i %s\n", context, type, "" ); |
|
} |
|
else if( type == NETAPI_REQUEST_RULES ) |
|
{ |
|
// send serverinfo |
|
Netchan_OutOfBandPrint( NS_SERVER, from, "netinfo %i %i %s\n", context, type, svs.serverinfo ); |
|
} |
|
else if( type == NETAPI_REQUEST_PLAYERS ) |
|
{ |
|
size_t len = 0; |
|
|
|
string[0] = '\0'; |
|
|
|
for( i = 0; i < svs.maxclients; i++ ) |
|
{ |
|
if( svs.clients[i].state >= cs_connected ) |
|
{ |
|
int ret; |
|
edict_t *ed = svs.clients[i].edict; |
|
float time = host.realtime - svs.clients[i].connection_started; |
|
ret = Q_snprintf( &string[len], sizeof( string ) - len, "%c\\%s\\%i\\%f\\", count, svs.clients[i].name, (int)ed->v.frags, time ); |
|
|
|
if( ret == -1 ) |
|
{ |
|
Con_DPrintf( S_WARN "SV_BuildNetAnswer: NETAPI_REQUEST_PLAYERS: buffer overflow!\n" ); |
|
break; |
|
} |
|
|
|
len += ret; |
|
count++; |
|
} |
|
} |
|
|
|
// send playernames |
|
Netchan_OutOfBandPrint( NS_SERVER, from, "netinfo %i %i %s\n", context, type, string ); |
|
} |
|
else if( type == NETAPI_REQUEST_DETAILS ) |
|
{ |
|
for( i = 0; i < svs.maxclients; i++ ) |
|
if( svs.clients[i].state >= cs_connected ) |
|
count++; |
|
|
|
string[0] = '\0'; |
|
Info_SetValueForKey( string, "hostname", hostname.string, MAX_INFO_STRING ); |
|
Info_SetValueForKey( string, "gamedir", GI->gamefolder, MAX_INFO_STRING ); |
|
Info_SetValueForKeyf( string, "current", MAX_INFO_STRING, "%i", count ); |
|
Info_SetValueForKeyf( string, "max", MAX_INFO_STRING, "%i", svs.maxclients ); |
|
Info_SetValueForKey( string, "map", sv.name, MAX_INFO_STRING ); |
|
|
|
// send serverinfo |
|
Netchan_OutOfBandPrint( NS_SERVER, from, "netinfo %i %i %s\n", context, type, string ); |
|
} |
|
else |
|
{ |
|
string[0] = '\0'; |
|
Info_SetValueForKey( string, "neterror", "undefined", MAX_INFO_STRING ); |
|
|
|
// send error undefined request type |
|
Netchan_OutOfBandPrint( NS_SERVER, from, "netinfo %i %i %s\n", context, type, string ); |
|
} |
|
} |
|
|
|
/* |
|
================ |
|
SV_Ping |
|
|
|
Just responds with an acknowledgement |
|
================ |
|
*/ |
|
static void SV_Ping( netadr_t from ) |
|
{ |
|
Netchan_OutOfBandPrint( NS_SERVER, from, "ack" ); |
|
} |
|
|
|
/* |
|
================ |
|
Rcon_Validate |
|
================ |
|
*/ |
|
static qboolean Rcon_Validate( void ) |
|
{ |
|
if( !COM_CheckString( rcon_password.string )) |
|
return false; |
|
if( Q_strcmp( Cmd_Argv( 1 ), rcon_password.string )) |
|
return false; |
|
return true; |
|
} |
|
|
|
/* |
|
=============== |
|
SV_RemoteCommand |
|
|
|
A client issued an rcon command. |
|
Shift down the remaining args |
|
Redirect all printfs |
|
=============== |
|
*/ |
|
void SV_RemoteCommand( netadr_t from, sizebuf_t *msg ) |
|
{ |
|
static char outputbuf[2048]; |
|
const char *adr; |
|
char remaining[1024]; |
|
char *p = remaining; |
|
int i; |
|
|
|
if( !rcon_enable.value || !COM_CheckStringEmpty( rcon_password.string )) |
|
return; |
|
|
|
adr = NET_AdrToString( from ); |
|
|
|
Con_Printf( "Rcon from %s:\n%s\n", adr, MSG_GetData( msg ) + 4 ); |
|
Log_Printf( "Rcon: \"%s\" from \"%s\"\n", MSG_GetData( msg ) + 4, adr ); |
|
|
|
if( Rcon_Validate( )) |
|
{ |
|
SV_BeginRedirect( &host.rd, from, RD_PACKET, outputbuf, sizeof( outputbuf ) - 16, SV_FlushRedirect ); |
|
|
|
remaining[0] = 0; |
|
for( i = 2; i < Cmd_Argc(); i++ ) |
|
{ |
|
p += Q_strncpy( p, "\"", sizeof( remaining ) - ( p - remaining )); |
|
p += Q_strncpy( p, Cmd_Argv( i ), sizeof( remaining ) - ( p - remaining )); |
|
p += Q_strncpy( p, "\" ", sizeof( remaining ) - ( p - remaining )); |
|
} |
|
Cmd_ExecuteString( remaining ); |
|
|
|
SV_EndRedirect( &host.rd ); |
|
} |
|
else Con_Printf( S_ERROR "Bad rcon_password.\n" ); |
|
} |
|
|
|
/* |
|
=================== |
|
SV_CalcPing |
|
|
|
recalc ping on current client |
|
=================== |
|
*/ |
|
int SV_CalcPing( sv_client_t *cl ) |
|
{ |
|
float ping = 0; |
|
int i, count; |
|
int idx, back; |
|
client_frame_t *frame; |
|
|
|
// bots don't have a real ping |
|
if( FBitSet( cl->flags, FCL_FAKECLIENT ) || !cl->frames ) |
|
return 0; |
|
|
|
if( SV_UPDATE_BACKUP <= 31 ) |
|
{ |
|
back = SV_UPDATE_BACKUP / 2; |
|
if( back <= 0 ) return 0; |
|
} |
|
else back = 16; |
|
|
|
count = 0; |
|
|
|
for( i = 0; i < back; i++ ) |
|
{ |
|
idx = cl->netchan.incoming_acknowledged + ~i; |
|
frame = &cl->frames[idx & SV_UPDATE_MASK]; |
|
|
|
if( frame->ping_time > 0.0f ) |
|
{ |
|
ping += frame->ping_time; |
|
count++; |
|
} |
|
} |
|
|
|
if( count > 0 ) |
|
return (( ping / count ) * 1000.0f ); |
|
return 0; |
|
} |
|
|
|
/* |
|
=================== |
|
SV_EstablishTimeBase |
|
|
|
Finangles latency and the like. |
|
=================== |
|
*/ |
|
static void SV_EstablishTimeBase( sv_client_t *cl, usercmd_t *cmds, int dropped, int numbackup, int numcmds ) |
|
{ |
|
double runcmd_time = 0.0; |
|
int i, cmdnum = dropped; |
|
|
|
if( dropped < 24 ) |
|
{ |
|
while( dropped > numbackup ) |
|
{ |
|
runcmd_time = (double)cl->lastcmd.msec / 1000.0; |
|
dropped--; |
|
} |
|
|
|
while( dropped > 0 ) |
|
{ |
|
cmdnum = dropped + numcmds - 1; |
|
runcmd_time += (double)cmds[cmdnum].msec / 1000.0; |
|
dropped--; |
|
} |
|
} |
|
|
|
for( i = numcmds - 1; i >= 0; i-- ) |
|
runcmd_time += cmds[i].msec / 1000.0; |
|
|
|
cl->timebase = sv.time + sv.frametime - runcmd_time; |
|
} |
|
|
|
/* |
|
=================== |
|
SV_CalcClientTime |
|
|
|
compute latency for client |
|
=================== |
|
*/ |
|
static float SV_CalcClientTime( sv_client_t *cl ) |
|
{ |
|
float minping, maxping; |
|
float ping = 0.0f; |
|
int i, count = 0; |
|
int backtrack; |
|
|
|
backtrack = (int)sv_unlagsamples.value; |
|
if( backtrack < 1 ) backtrack = 1; |
|
|
|
if( backtrack >= (SV_UPDATE_BACKUP <= 16 ? SV_UPDATE_BACKUP : 16 )) |
|
backtrack = ( SV_UPDATE_BACKUP <= 16 ? SV_UPDATE_BACKUP : 16 ); |
|
|
|
if( backtrack <= 0 ) |
|
return 0.0f; |
|
|
|
for( i = 0; i < backtrack; i++ ) |
|
{ |
|
client_frame_t *frame = &cl->frames[SV_UPDATE_MASK & (cl->netchan.incoming_acknowledged - i)]; |
|
if( frame->ping_time <= 0.0f ) |
|
continue; |
|
|
|
ping += frame->ping_time; |
|
count++; |
|
} |
|
|
|
if( !count ) return 0.0f; |
|
|
|
minping = 9999.0f; |
|
maxping = -9999.0f; |
|
ping /= count; |
|
|
|
for( i = 0; i < ( SV_UPDATE_BACKUP <= 4 ? SV_UPDATE_BACKUP : 4 ); i++ ) |
|
{ |
|
client_frame_t *frame = &cl->frames[SV_UPDATE_MASK & (cl->netchan.incoming_acknowledged - i)]; |
|
if( frame->ping_time <= 0.0f ) |
|
continue; |
|
|
|
if( frame->ping_time < minping ) |
|
minping = frame->ping_time; |
|
|
|
if( frame->ping_time > maxping ) |
|
maxping = frame->ping_time; |
|
} |
|
|
|
if( maxping < minping || fabs( maxping - minping ) <= 0.2f ) |
|
return ping; |
|
|
|
return 0.0f; |
|
} |
|
|
|
/* |
|
=================== |
|
SV_FullClientUpdate |
|
|
|
Writes all update values to a bitbuf |
|
=================== |
|
*/ |
|
void SV_FullClientUpdate( sv_client_t *cl, sizebuf_t *msg ) |
|
{ |
|
char info[MAX_INFO_STRING]; |
|
char digest[16]; |
|
MD5Context_t ctx; |
|
int i; |
|
|
|
// process userinfo before updating |
|
SV_UserinfoChanged( cl ); |
|
|
|
i = cl - svs.clients; |
|
|
|
MSG_BeginServerCmd( msg, svc_updateuserinfo ); |
|
MSG_WriteUBitLong( msg, i, MAX_CLIENT_BITS ); |
|
MSG_WriteLong( msg, cl->userid ); |
|
|
|
if( cl->name[0] ) |
|
{ |
|
MSG_WriteOneBit( msg, 1 ); |
|
|
|
Q_strncpy( info, cl->userinfo, sizeof( info )); |
|
|
|
// remove server passwords, etc. |
|
Info_RemovePrefixedKeys( info, '_' ); |
|
MSG_WriteString( msg, info ); |
|
|
|
MD5Init( &ctx ); |
|
MD5Update( &ctx, (byte *)cl->hashedcdkey, sizeof( cl->hashedcdkey )); |
|
MD5Final( digest, &ctx ); |
|
|
|
MSG_WriteBytes( msg, digest, sizeof( digest )); |
|
} |
|
else MSG_WriteOneBit( msg, 0 ); |
|
} |
|
|
|
/* |
|
=================== |
|
SV_RefreshUserinfo |
|
|
|
=================== |
|
*/ |
|
void SV_RefreshUserinfo( void ) |
|
{ |
|
sv_client_t *cl; |
|
int i; |
|
|
|
for( i = 0, cl = svs.clients; i < svs.maxclients; i++, cl++ ) |
|
{ |
|
if( cl->state >= cs_connected ) |
|
SetBits( cl->flags, FCL_RESEND_USERINFO ); |
|
} |
|
} |
|
|
|
/* |
|
=================== |
|
SV_FullUpdateMovevars |
|
|
|
this is send all movevars values when client connected |
|
otherwise see code SV_UpdateMovevars() |
|
=================== |
|
*/ |
|
void SV_FullUpdateMovevars( sv_client_t *cl, sizebuf_t *msg ) |
|
{ |
|
movevars_t nullmovevars; |
|
|
|
memset( &nullmovevars, 0, sizeof( nullmovevars )); |
|
MSG_WriteDeltaMovevars( msg, &nullmovevars, &svgame.movevars ); |
|
} |
|
|
|
/* |
|
=================== |
|
SV_ShouldUpdatePing |
|
|
|
determine should we recalculate |
|
ping times now |
|
=================== |
|
*/ |
|
qboolean SV_ShouldUpdatePing( sv_client_t *cl ) |
|
{ |
|
if( FBitSet( cl->flags, FCL_HLTV_PROXY )) |
|
{ |
|
if( host.realtime < cl->next_checkpingtime ) |
|
return false; |
|
|
|
cl->next_checkpingtime = host.realtime + 2.0; |
|
return true; |
|
} |
|
|
|
// they are viewing the scoreboard. Send them pings. |
|
return FBitSet( cl->lastcmd.buttons, IN_SCORE ) ? true : false; |
|
} |
|
|
|
/* |
|
=================== |
|
SV_IsPlayerIndex |
|
|
|
=================== |
|
*/ |
|
qboolean SV_IsPlayerIndex( int idx ) |
|
{ |
|
if( idx > 0 && idx <= svs.maxclients ) |
|
return true; |
|
return false; |
|
} |
|
|
|
/* |
|
=================== |
|
SV_GetPlayerStats |
|
|
|
This function and its static vars track some of the networking |
|
conditions. I haven't bothered to trace it beyond that, because |
|
this fucntion sucks pretty badly. |
|
=================== |
|
*/ |
|
void SV_GetPlayerStats( sv_client_t *cl, int *ping, int *packet_loss ) |
|
{ |
|
static int last_ping[MAX_CLIENTS]; |
|
static int last_loss[MAX_CLIENTS]; |
|
int i; |
|
|
|
i = cl - svs.clients; |
|
|
|
if( host.realtime >= cl->next_checkpingtime ) |
|
{ |
|
cl->next_checkpingtime = host.realtime + 2.0; |
|
last_ping[i] = SV_CalcPing( cl ); |
|
last_loss[i] = cl->packet_loss; |
|
} |
|
|
|
if( ping ) *ping = last_ping[i]; |
|
if( packet_loss ) *packet_loss = last_loss[i]; |
|
} |
|
|
|
/* |
|
=========== |
|
PutClientInServer |
|
|
|
Called when a player connects to a server or respawns in |
|
a deathmatch. |
|
============ |
|
*/ |
|
static void SV_PutClientInServer( sv_client_t *cl ) |
|
{ |
|
static byte msg_buf[MAX_INIT_MSG + 0x200]; // MAX_INIT_MSG + some space |
|
edict_t *ent = cl->edict; |
|
sizebuf_t msg; |
|
|
|
MSG_Init( &msg, "Spawn", msg_buf, sizeof( msg_buf )); |
|
|
|
if( sv.loadgame ) |
|
{ |
|
// NOTE: we needs to setup angles on restore here |
|
if( ent->v.fixangle == 1 ) |
|
{ |
|
MSG_BeginServerCmd( &msg, svc_setangle ); |
|
MSG_WriteVec3Angles( &msg, ent->v.angles ); |
|
ent->v.fixangle = 0; |
|
} |
|
|
|
if( svgame.dllFuncs.pfnParmsChangeLevel ) |
|
{ |
|
SAVERESTOREDATA levelData; |
|
string name; |
|
int i; |
|
|
|
memset( &levelData, 0, sizeof( levelData )); |
|
svgame.globals->pSaveData = &levelData; |
|
svgame.dllFuncs.pfnParmsChangeLevel(); |
|
|
|
MSG_BeginServerCmd( &msg, svc_restore ); |
|
Q_snprintf( name, sizeof( name ), DEFAULT_SAVE_DIRECTORY "%s.HL2", sv.name ); |
|
COM_FixSlashes( name ); |
|
MSG_WriteString( &msg, name ); |
|
MSG_WriteByte( &msg, levelData.connectionCount ); |
|
|
|
for( i = 0; i < levelData.connectionCount; i++ ) |
|
MSG_WriteString( &msg, levelData.levelList[i].mapName ); |
|
|
|
svgame.globals->pSaveData = NULL; |
|
} |
|
|
|
// reset weaponanim |
|
MSG_BeginServerCmd( &msg, svc_weaponanim ); |
|
MSG_WriteByte( &msg, 0 ); |
|
MSG_WriteByte( &msg, 0 ); |
|
|
|
sv.loadgame = false; |
|
sv.paused = false; |
|
} |
|
else |
|
{ |
|
if( Q_atoi( Info_ValueForKey( cl->userinfo, "hltv" ))) |
|
SetBits( cl->flags, FCL_HLTV_PROXY ); |
|
|
|
// need to realloc private data for client |
|
SV_InitEdict( ent ); |
|
|
|
if( FBitSet( cl->flags, FCL_HLTV_PROXY )) |
|
SetBits( ent->v.flags, FL_PROXY ); |
|
else ent->v.flags = 0; |
|
|
|
ent->v.netname = MAKE_STRING( cl->name ); |
|
ent->v.colormap = NUM_FOR_EDICT( ent ); // ??? |
|
|
|
// fisrt entering |
|
svgame.globals->time = sv.time; |
|
svgame.dllFuncs.pfnClientPutInServer( ent ); |
|
|
|
if( sv.background ) // don't attack player in background mode |
|
SetBits( ent->v.flags, FL_GODMODE|FL_NOTARGET ); |
|
|
|
cl->pViewEntity = NULL; // reset pViewEntity |
|
} |
|
|
|
if( svgame.globals->cdAudioTrack ) |
|
{ |
|
MSG_BeginServerCmd( &msg, svc_stufftext ); |
|
MSG_WriteStringf( &msg, "cd loop %3d\n", svgame.globals->cdAudioTrack ); |
|
svgame.globals->cdAudioTrack = 0; |
|
} |
|
|
|
#ifdef HACKS_RELATED_HLMODS |
|
// enable dev-mode to prevent crash cheat-protecting from Invasion mod |
|
if( FBitSet( ent->v.flags, FL_GODMODE|FL_NOTARGET ) && !Q_stricmp( GI->gamefolder, "invasion" )) |
|
SV_ExecuteClientCommand( cl, "test\n" ); |
|
#endif |
|
// refresh the userinfo and movevars |
|
// NOTE: because movevars can be changed during the connection process |
|
SetBits( cl->flags, FCL_RESEND_USERINFO|FCL_RESEND_MOVEVARS ); |
|
|
|
// reset client times |
|
cl->connecttime = 0.0; |
|
cl->ignorecmdtime = 0.0; |
|
cl->cmdtime = 0.0; |
|
|
|
if( !FBitSet( cl->flags, FCL_FAKECLIENT )) |
|
{ |
|
int viewEnt; |
|
|
|
// NOTE: it's will be fragmented automatically in right ordering |
|
MSG_WriteBits( &msg, MSG_GetData( &sv.signon ), MSG_GetNumBitsWritten( &sv.signon )); |
|
|
|
if( cl->pViewEntity ) |
|
viewEnt = NUM_FOR_EDICT( cl->pViewEntity ); |
|
else viewEnt = NUM_FOR_EDICT( cl->edict ); |
|
|
|
MSG_BeginServerCmd( &msg, svc_setview ); |
|
MSG_WriteWord( &msg, viewEnt ); |
|
|
|
MSG_BeginServerCmd( &msg, svc_signonnum ); |
|
MSG_WriteByte( &msg, 1 ); |
|
|
|
if( MSG_CheckOverflow( &msg )) |
|
{ |
|
if( svs.maxclients == 1 ) |
|
Host_Error( "spawn player: overflowed\n" ); |
|
else SV_DropClient( cl, false ); |
|
} |
|
else |
|
{ |
|
// send initialization data |
|
Netchan_CreateFragments( &cl->netchan, &msg ); |
|
Netchan_FragSend( &cl->netchan ); |
|
} |
|
} |
|
} |
|
|
|
/* |
|
=========== |
|
SV_UpdateClientView |
|
|
|
Resend the client viewentity (used for demos) |
|
============ |
|
*/ |
|
static void SV_UpdateClientView( sv_client_t *cl ) |
|
{ |
|
int viewEnt; |
|
|
|
if( cl->pViewEntity ) |
|
viewEnt = NUM_FOR_EDICT( cl->pViewEntity ); |
|
else viewEnt = NUM_FOR_EDICT( cl->edict ); |
|
|
|
MSG_BeginServerCmd( &cl->netchan.message, svc_setview ); |
|
MSG_WriteWord( &cl->netchan.message, viewEnt ); |
|
} |
|
|
|
/* |
|
================== |
|
SV_TogglePause |
|
================== |
|
*/ |
|
void SV_TogglePause( const char *msg ) |
|
{ |
|
if( sv.background ) return; |
|
|
|
sv.paused ^= 1; |
|
|
|
if( COM_CheckString( msg )) |
|
SV_BroadcastPrintf( NULL, "%s", msg ); |
|
|
|
// send notification to all clients |
|
MSG_BeginServerCmd( &sv.reliable_datagram, svc_setpause ); |
|
MSG_WriteOneBit( &sv.reliable_datagram, sv.paused ); |
|
} |
|
|
|
/* |
|
================ |
|
SV_SendReconnect |
|
|
|
Tell all the clients that the server is changing levels |
|
================ |
|
*/ |
|
void SV_BuildReconnect( sizebuf_t *msg ) |
|
{ |
|
MSG_BeginServerCmd( msg, svc_stufftext ); |
|
MSG_WriteString( msg, "reconnect\n" ); |
|
} |
|
|
|
/* |
|
================ |
|
SV_SendServerdata |
|
|
|
Sends the first message from the server to a connected client. |
|
This will be sent on the initial connection and upon each server load. |
|
================ |
|
*/ |
|
void SV_SendServerdata( sizebuf_t *msg, sv_client_t *cl ) |
|
{ |
|
string message; |
|
int i; |
|
|
|
// Only send this message to developer console, or multiplayer clients. |
|
if(( host_developer.value ) || ( svs.maxclients > 1 )) |
|
{ |
|
MSG_BeginServerCmd( msg, svc_print ); |
|
Q_snprintf( message, sizeof( message ), "\n^3BUILD %d SERVER (%i CRC)\nServer #%i\n", Q_buildnum(), sv.progsCRC, svs.spawncount ); |
|
MSG_WriteString( msg, message ); |
|
} |
|
|
|
// send the serverdata |
|
MSG_BeginServerCmd( msg, svc_serverdata ); |
|
MSG_WriteLong( msg, PROTOCOL_VERSION ); |
|
MSG_WriteLong( msg, svs.spawncount ); |
|
MSG_WriteLong( msg, sv.worldmapCRC ); |
|
MSG_WriteByte( msg, cl - svs.clients ); |
|
MSG_WriteByte( msg, svs.maxclients ); |
|
MSG_WriteWord( msg, GI->max_edicts ); |
|
MSG_WriteWord( msg, MAX_MODELS ); |
|
MSG_WriteString( msg, sv.name ); |
|
MSG_WriteString( msg, STRING( svgame.edicts->v.message )); // Map Message |
|
MSG_WriteOneBit( msg, sv.background ); // tell client about background map |
|
MSG_WriteString( msg, GI->gamefolder ); |
|
MSG_WriteLong( msg, host.features ); |
|
|
|
// send the player hulls |
|
for( i = 0; i < MAX_MAP_HULLS * 3; i++ ) |
|
{ |
|
MSG_WriteChar( msg, host.player_mins[i/3][i%3] ); |
|
MSG_WriteChar( msg, host.player_maxs[i/3][i%3] ); |
|
} |
|
|
|
// send delta-encoding |
|
Delta_WriteDescriptionToClient( msg ); |
|
|
|
// now client know delta and can reading encoded messages |
|
SV_FullUpdateMovevars( cl, msg ); |
|
|
|
// send the user messages registration |
|
for( i = 1; i < MAX_USER_MESSAGES && svgame.msg[i].name[0]; i++ ) |
|
SV_SendUserReg( msg, &svgame.msg[i] ); |
|
|
|
for( i = 0; i < MAX_LIGHTSTYLES; i++ ) |
|
{ |
|
if( !sv.lightstyles[i].pattern[0] ) |
|
continue; // unused style |
|
|
|
MSG_BeginServerCmd( msg, svc_lightstyle ); |
|
MSG_WriteByte( msg, i ); // stylenum |
|
MSG_WriteString( msg, sv.lightstyles[i].pattern ); |
|
MSG_WriteFloat( msg, sv.lightstyles[i].time ); |
|
} |
|
} |
|
|
|
/* |
|
============================================================ |
|
|
|
CLIENT COMMAND EXECUTION |
|
|
|
============================================================ |
|
*/ |
|
/* |
|
================ |
|
SV_New_f |
|
|
|
Sends the first message from the server to a connected client. |
|
This will be sent on the initial connection and upon each server load. |
|
================ |
|
*/ |
|
static qboolean SV_New_f( sv_client_t *cl ) |
|
{ |
|
byte msg_buf[MAX_INIT_MSG]; |
|
char szRejectReason[128]; |
|
char szAddress[128]; |
|
char szName[32]; |
|
sv_client_t *cur; |
|
sizebuf_t msg; |
|
int i; |
|
|
|
memset( msg_buf, 0, sizeof( msg_buf )); |
|
MSG_Init( &msg, "New", msg_buf, sizeof( msg_buf )); |
|
|
|
if( cl->state != cs_connected ) |
|
return false; |
|
|
|
// send the serverdata |
|
SV_SendServerdata( &msg, cl ); |
|
|
|
// if the client was connected, tell the game .dll to disconnect him/her. |
|
if(( cl->state == cs_spawned ) && cl->edict ) |
|
svgame.dllFuncs.pfnClientDisconnect( cl->edict ); |
|
|
|
Q_strncpy( szName, cl->name, sizeof( szName ) ); |
|
Q_strncpy( szAddress, NET_AdrToString( cl->netchan.remote_address ), sizeof( szAddress ) ); |
|
Q_strncpy( szRejectReason, "Connection rejected by game\n", sizeof( szRejectReason ) ); |
|
|
|
// Allow the game dll to reject this client. |
|
if( !svgame.dllFuncs.pfnClientConnect( cl->edict, szName, szAddress, szRejectReason )) |
|
{ |
|
// reject the connection and drop the client. |
|
SV_RejectConnection( cl->netchan.remote_address, "%s\n", szRejectReason ); |
|
SV_DropClient( cl, false ); |
|
return true; |
|
} |
|
|
|
// server info string |
|
MSG_BeginServerCmd( &msg, svc_stufftext ); |
|
MSG_WriteStringf( &msg, "fullserverinfo \"%s\"\n", SV_Serverinfo( )); |
|
|
|
// collect the info about all the players and send to me |
|
for( i = 0, cur = svs.clients; i < svs.maxclients; i++, cur++ ) |
|
{ |
|
if( !cur->edict || cur->state != cs_spawned ) |
|
continue; // not in game yet |
|
SV_FullClientUpdate( cur, &msg ); |
|
} |
|
|
|
// g-cont. why this is there? |
|
memset( &cl->lastcmd, 0, sizeof( cl->lastcmd )); |
|
|
|
Netchan_CreateFragments( &cl->netchan, &msg ); |
|
Netchan_FragSend( &cl->netchan ); |
|
|
|
return true; |
|
} |
|
|
|
/* |
|
================= |
|
SV_Disconnect_f |
|
|
|
The client is going to disconnect, so remove the connection immediately |
|
================= |
|
*/ |
|
static qboolean SV_Disconnect_f( sv_client_t *cl ) |
|
{ |
|
SV_DropClient( cl, false ); |
|
return true; |
|
} |
|
|
|
/* |
|
================== |
|
SV_ShowServerinfo_f |
|
|
|
Dumps the serverinfo info string |
|
================== |
|
*/ |
|
static qboolean SV_ShowServerinfo_f( sv_client_t *cl ) |
|
{ |
|
Info_Print( svs.serverinfo ); |
|
return true; |
|
} |
|
|
|
/* |
|
================== |
|
SV_Pause_f |
|
================== |
|
*/ |
|
static qboolean SV_Pause_f( sv_client_t *cl ) |
|
{ |
|
string message; |
|
|
|
if( UI_CreditsActive( )) |
|
return true; |
|
|
|
if( !sv_pausable.value ) |
|
{ |
|
SV_ClientPrintf( cl, "Pause not allowed.\n" ); |
|
return true; |
|
} |
|
|
|
if( FBitSet( cl->flags, FCL_HLTV_PROXY )) |
|
{ |
|
SV_ClientPrintf( cl, "Spectators can not pause.\n" ); |
|
return true; |
|
} |
|
|
|
if( !sv.paused ) Q_snprintf( message, MAX_STRING, "^2%s^7 paused the game\n", cl->name ); |
|
else Q_snprintf( message, MAX_STRING, "^2%s^7 unpaused the game\n", cl->name ); |
|
|
|
SV_TogglePause( message ); |
|
|
|
return true; |
|
} |
|
|
|
static qboolean SV_ShouldUpdateUserinfo( sv_client_t *cl ) |
|
{ |
|
qboolean allow = true; // predict state |
|
|
|
if( !sv_userinfo_enable_penalty.value ) |
|
return allow; |
|
|
|
if( FBitSet( cl->flags, FCL_FAKECLIENT )) |
|
return allow; |
|
|
|
if( Host_IsLocalGame( )) |
|
return allow; |
|
|
|
// start from 1 second |
|
if( !cl->userinfo_penalty ) |
|
cl->userinfo_penalty = sv_userinfo_penalty_time.value; |
|
|
|
// player changes userinfo after limit time window, but before |
|
// next timewindow |
|
// he seems to be spammer, so just increase change attempts |
|
if( host.realtime < cl->userinfo_next_changetime + cl->userinfo_penalty * sv_userinfo_penalty_multiplier.value ) |
|
{ |
|
// player changes userinfo too quick! ignore! |
|
if( host.realtime < cl->userinfo_next_changetime ) |
|
{ |
|
Con_Reportf( "SV_ShouldUpdateUserinfo: ignore userinfo update for %s: penalty %f, attempts %i\n", |
|
cl->name, cl->userinfo_penalty, cl->userinfo_change_attempts ); |
|
allow = false; |
|
} |
|
|
|
cl->userinfo_change_attempts++; |
|
} |
|
|
|
// they spammed too fast, increase penalty |
|
if( cl->userinfo_change_attempts > sv_userinfo_penalty_attempts.value ) |
|
{ |
|
Con_Reportf( "SV_ShouldUpdateUserinfo: penalty set %f for %s\n", |
|
cl->userinfo_penalty, cl->name ); |
|
cl->userinfo_penalty *= sv_userinfo_penalty_multiplier.value; |
|
cl->userinfo_change_attempts = 0; |
|
} |
|
|
|
cl->userinfo_next_changetime = host.realtime + cl->userinfo_penalty; |
|
|
|
return allow; |
|
} |
|
|
|
/* |
|
================= |
|
SV_UserinfoChanged |
|
|
|
Pull specific info from a newly changed userinfo string |
|
into a more C freindly form. |
|
================= |
|
*/ |
|
static void SV_UserinfoChanged( sv_client_t *cl ) |
|
{ |
|
int i, dupc = 1; |
|
edict_t *ent = cl->edict; |
|
string name1, name2; |
|
sv_client_t *current; |
|
const char *val; |
|
|
|
if( !COM_CheckString( cl->userinfo )) |
|
return; |
|
|
|
if( !SV_ShouldUpdateUserinfo( cl )) |
|
return; |
|
|
|
if( !Info_IsValid( cl->userinfo )) |
|
return; |
|
|
|
val = Info_ValueForKey( cl->userinfo, "name" ); |
|
Q_strncpy( name2, val, sizeof( name2 )); |
|
COM_TrimSpace( name2, name1 ); |
|
|
|
if( !Q_stricmp( name1, "console" )) |
|
{ |
|
Info_SetValueForKey( cl->userinfo, "name", "unnamed", MAX_INFO_STRING ); |
|
val = Info_ValueForKey( cl->userinfo, "name" ); |
|
} |
|
else if( Q_strcmp( name1, val )) |
|
{ |
|
Info_SetValueForKey( cl->userinfo, "name", name1, MAX_INFO_STRING ); |
|
val = Info_ValueForKey( cl->userinfo, "name" ); |
|
} |
|
|
|
if( !COM_CheckStringEmpty( name1 ) ) |
|
{ |
|
Info_SetValueForKey( cl->userinfo, "name", "unnamed", MAX_INFO_STRING ); |
|
val = Info_ValueForKey( cl->userinfo, "name" ); |
|
Q_strncpy( name2, "unnamed", sizeof( name2 )); |
|
Q_strncpy( name1, "unnamed", sizeof( name1 )); |
|
} |
|
|
|
// check to see if another user by the same name exists |
|
while( 1 ) |
|
{ |
|
for( i = 0, current = svs.clients; i < svs.maxclients; i++, current++ ) |
|
{ |
|
if( current == cl || current->state != cs_spawned ) |
|
continue; |
|
|
|
if( !Q_stricmp( current->name, val )) |
|
break; |
|
} |
|
|
|
if( i != svs.maxclients ) |
|
{ |
|
// dup name |
|
Q_snprintf( name2, sizeof( name2 ), "%s (%u)", name1, dupc++ ); |
|
Info_SetValueForKey( cl->userinfo, "name", name2, MAX_INFO_STRING ); |
|
val = Info_ValueForKey( cl->userinfo, "name" ); |
|
Q_strncpy( cl->name, name2, sizeof( cl->name )); |
|
} |
|
else |
|
{ |
|
if( dupc == 1 ) // unchanged |
|
Q_strncpy( cl->name, name1, sizeof( cl->name )); |
|
break; |
|
} |
|
} |
|
|
|
// rate command |
|
val = Info_ValueForKey( cl->userinfo, "rate" ); |
|
if( COM_CheckString( val ) ) |
|
cl->netchan.rate = bound( sv_minrate.value, Q_atoi( val ), sv_maxrate.value ); |
|
else cl->netchan.rate = DEFAULT_RATE; |
|
|
|
// movement prediction |
|
if( Q_atoi( Info_ValueForKey( cl->userinfo, "cl_nopred" ))) |
|
ClearBits( cl->flags, FCL_PREDICT_MOVEMENT ); |
|
else SetBits( cl->flags, FCL_PREDICT_MOVEMENT ); |
|
|
|
// lag compensation |
|
if( Q_atoi( Info_ValueForKey( cl->userinfo, "cl_lc" ))) |
|
SetBits( cl->flags, FCL_LAG_COMPENSATION ); |
|
else ClearBits( cl->flags, FCL_LAG_COMPENSATION ); |
|
|
|
// weapon perdiction |
|
if( Q_atoi( Info_ValueForKey( cl->userinfo, "cl_lw" ))) |
|
SetBits( cl->flags, FCL_LOCAL_WEAPONS ); |
|
else ClearBits( cl->flags, FCL_LOCAL_WEAPONS ); |
|
|
|
val = Info_ValueForKey( cl->userinfo, "cl_updaterate" ); |
|
|
|
if( COM_CheckString( val ) ) |
|
{ |
|
if( Q_atoi( val ) != 0 ) |
|
{ |
|
cl->cl_updaterate = 1.0 / bound( sv_minupdaterate.value, Q_atoi( val ), sv_maxupdaterate.value ); |
|
} |
|
else cl->cl_updaterate = 0.0; |
|
} |
|
|
|
// call prog code to allow overrides |
|
svgame.dllFuncs.pfnClientUserInfoChanged( cl->edict, cl->userinfo ); |
|
|
|
val = Info_ValueForKey( cl->userinfo, "name" ); |
|
Q_strncpy( cl->name, val, sizeof( cl->name )); |
|
ent->v.netname = MAKE_STRING( cl->name ); |
|
} |
|
|
|
/* |
|
================== |
|
SV_UpdateUserinfo_f |
|
================== |
|
*/ |
|
static qboolean SV_UpdateUserinfo_f( sv_client_t *cl ) |
|
{ |
|
Q_strncpy( cl->userinfo, Cmd_Argv( 1 ), sizeof( cl->userinfo )); |
|
|
|
if( cl->state >= cs_connected ) |
|
SetBits( cl->flags, FCL_RESEND_USERINFO ); // needs for update client info |
|
return true; |
|
} |
|
|
|
/* |
|
================== |
|
SV_SetInfo_f |
|
================== |
|
*/ |
|
static qboolean SV_SetInfo_f( sv_client_t *cl ) |
|
{ |
|
Info_SetValueForKey( cl->userinfo, Cmd_Argv( 1 ), Cmd_Argv( 2 ), MAX_INFO_STRING ); |
|
|
|
if( cl->state >= cs_connected ) |
|
SetBits( cl->flags, FCL_RESEND_USERINFO ); // needs for update client info |
|
return true; |
|
} |
|
|
|
/* |
|
================== |
|
SV_Noclip_f |
|
================== |
|
*/ |
|
static qboolean SV_Noclip_f( sv_client_t *cl ) |
|
{ |
|
edict_t *pEntity = cl->edict; |
|
|
|
if( !Cvar_VariableInteger( "sv_cheats" ) || sv.background ) |
|
return true; |
|
|
|
if( pEntity->v.movetype != MOVETYPE_NOCLIP ) |
|
{ |
|
SV_ClientPrintf( cl, "noclip ON\n" ); |
|
pEntity->v.movetype = MOVETYPE_NOCLIP; |
|
} |
|
else |
|
{ |
|
SV_ClientPrintf( cl, "noclip OFF\n" ); |
|
pEntity->v.movetype = MOVETYPE_WALK; |
|
} |
|
|
|
return true; |
|
} |
|
|
|
/* |
|
================== |
|
SV_Godmode_f |
|
================== |
|
*/ |
|
static qboolean SV_Godmode_f( sv_client_t *cl ) |
|
{ |
|
edict_t *pEntity = cl->edict; |
|
|
|
if( !Cvar_VariableInteger( "sv_cheats" ) || sv.background ) |
|
return true; |
|
|
|
pEntity->v.flags = pEntity->v.flags ^ FL_GODMODE; |
|
|
|
if( !FBitSet( pEntity->v.flags, FL_GODMODE )) |
|
SV_ClientPrintf( cl, "godmode OFF\n" ); |
|
else SV_ClientPrintf( cl, "godmode ON\n" ); |
|
|
|
return true; |
|
} |
|
|
|
/* |
|
================== |
|
SV_Notarget_f |
|
================== |
|
*/ |
|
static qboolean SV_Notarget_f( sv_client_t *cl ) |
|
{ |
|
edict_t *pEntity = cl->edict; |
|
|
|
if( !Cvar_VariableInteger( "sv_cheats" ) || sv.background ) |
|
return true; |
|
|
|
pEntity->v.flags = pEntity->v.flags ^ FL_NOTARGET; |
|
|
|
if( !FBitSet( pEntity->v.flags, FL_NOTARGET )) |
|
SV_ClientPrintf( cl, "notarget OFF\n" ); |
|
else SV_ClientPrintf( cl, "notarget ON\n" ); |
|
|
|
return true; |
|
} |
|
|
|
/* |
|
================== |
|
SV_Kill_f |
|
================== |
|
*/ |
|
static qboolean SV_Kill_f( sv_client_t *cl ) |
|
{ |
|
if( !SV_IsValidEdict( cl->edict )) |
|
return true; |
|
|
|
if( cl->state != cs_spawned ) |
|
{ |
|
SV_ClientPrintf( cl, "Can't suicide - not connected!\n" ); |
|
return true; |
|
} |
|
|
|
if( cl->edict->v.health <= 0.0f ) |
|
{ |
|
SV_ClientPrintf( cl, "Can't suicide - already dead!\n"); |
|
return true; |
|
} |
|
|
|
svgame.dllFuncs.pfnClientKill( cl->edict ); |
|
|
|
return true; |
|
} |
|
|
|
/* |
|
================== |
|
SV_SendRes_f |
|
================== |
|
*/ |
|
static qboolean SV_SendRes_f( sv_client_t *cl ) |
|
{ |
|
byte buffer[MAX_INIT_MSG]; |
|
sizebuf_t msg; |
|
|
|
if( cl->state != cs_connected ) |
|
return false; |
|
|
|
memset( buffer, 0, sizeof( buffer )); |
|
MSG_Init( &msg, "SendResources", buffer, sizeof( buffer )); |
|
|
|
if( svs.maxclients > 1 && FBitSet( cl->flags, FCL_SEND_RESOURCES )) |
|
return true; |
|
|
|
SetBits( cl->flags, FCL_SEND_RESOURCES ); |
|
SV_SendResources( cl, &msg ); |
|
|
|
Netchan_CreateFragments( &cl->netchan, &msg ); |
|
Netchan_FragSend( &cl->netchan ); |
|
|
|
return true; |
|
} |
|
|
|
/* |
|
================== |
|
SV_DownloadFile_f |
|
================== |
|
*/ |
|
static qboolean SV_DownloadFile_f( sv_client_t *cl ) |
|
{ |
|
const char *name; |
|
|
|
if( Cmd_Argc() < 2 ) |
|
return true; |
|
|
|
name = Cmd_Argv( 1 ); |
|
|
|
if( !COM_CheckString( name )) |
|
return true; |
|
|
|
if( !COM_IsSafeFileToDownload( name ) || !sv_allow_download.value ) |
|
{ |
|
SV_FailDownload( cl, name ); |
|
return true; |
|
} |
|
|
|
// g-cont. now we supports hot precache |
|
if( name[0] != '!' ) |
|
{ |
|
if( sv_send_resources.value ) |
|
{ |
|
int i; |
|
|
|
// security: allow download only precached resources |
|
for( i = 0; i < sv.num_resources; i++ ) |
|
{ |
|
const char *cmpname = name; |
|
|
|
if( sv.resources[i].type == t_sound ) |
|
cmpname += sizeof( DEFAULT_SOUNDPATH ) - 1; // cut "sound/" off |
|
|
|
if( !Q_strncmp( sv.resources[i].szFileName, cmpname, 64 ) ) |
|
break; |
|
} |
|
|
|
if( i == sv.num_resources ) |
|
{ |
|
SV_FailDownload( cl, name ); |
|
return true; |
|
} |
|
|
|
// also check the model textures |
|
if( !Q_stricmp( COM_FileExtension( name ), "mdl" )) |
|
{ |
|
if( FS_FileExists( Mod_StudioTexName( name ), false ) > 0 ) |
|
Netchan_CreateFileFragments( &cl->netchan, Mod_StudioTexName( name )); |
|
} |
|
|
|
if( Netchan_CreateFileFragments( &cl->netchan, name )) |
|
{ |
|
Netchan_FragSend( &cl->netchan ); |
|
return true; |
|
} |
|
} |
|
|
|
SV_FailDownload( cl, name ); |
|
return true; |
|
} |
|
|
|
if( Q_strlen( name ) == 36 && !Q_strnicmp( name, "!MD5", 4 ) && sv_send_logos.value ) |
|
{ |
|
resource_t custResource; |
|
byte md5[32]; |
|
byte *pbuf; |
|
int size; |
|
|
|
memset( &custResource, 0, sizeof( custResource ) ); |
|
COM_HexConvert( name + 4, 32, md5 ); |
|
|
|
if( HPAK_ResourceForHash( CUSTOM_RES_PATH, md5, &custResource )) |
|
{ |
|
if( HPAK_GetDataPointer( CUSTOM_RES_PATH, &custResource, &pbuf, &size )) |
|
{ |
|
if( size ) |
|
{ |
|
Netchan_CreateFileFragmentsFromBuffer( &cl->netchan, name, pbuf, size ); |
|
Netchan_FragSend( &cl->netchan ); |
|
Mem_Free( pbuf ); |
|
} |
|
} |
|
} |
|
} |
|
else |
|
{ |
|
SV_FailDownload( cl, name ); |
|
} |
|
|
|
return true; |
|
} |
|
|
|
/* |
|
================== |
|
SV_Spawn_f |
|
================== |
|
*/ |
|
static qboolean SV_Spawn_f( sv_client_t *cl ) |
|
{ |
|
if( cl->state != cs_connected ) |
|
return false; |
|
|
|
// handle the case of a level changing while a client was connecting |
|
if( Q_atoi( Cmd_Argv( 1 )) != svs.spawncount ) |
|
{ |
|
SV_New_f( cl ); |
|
return true; |
|
} |
|
|
|
SV_PutClientInServer( cl ); |
|
|
|
// if we are paused, tell the clients |
|
if( sv.paused ) |
|
{ |
|
MSG_BeginServerCmd( &sv.reliable_datagram, svc_setpause ); |
|
MSG_WriteByte( &sv.reliable_datagram, sv.paused ); |
|
SV_ClientPrintf( cl, "Server is paused.\n" ); |
|
} |
|
return true; |
|
} |
|
|
|
/* |
|
================== |
|
SV_Begin_f |
|
================== |
|
*/ |
|
static qboolean SV_Begin_f( sv_client_t *cl ) |
|
{ |
|
if( cl->state != cs_connected ) |
|
return false; |
|
|
|
// now client is spawned |
|
cl->state = cs_spawned; |
|
cl->connecttime = host.realtime; |
|
|
|
return true; |
|
} |
|
|
|
/* |
|
================== |
|
SV_SendBuildInfo_f |
|
================== |
|
*/ |
|
static qboolean SV_SendBuildInfo_f( sv_client_t *cl ) |
|
{ |
|
SV_ClientPrintf( cl, "Server running " XASH_ENGINE_NAME " " XASH_VERSION " (build %i-%s, %s-%s)\n", |
|
Q_buildnum(), Q_buildcommit(), Q_buildos(), Q_buildarch() ); |
|
return true; |
|
} |
|
|
|
/* |
|
================== |
|
SV_GetCrossEnt |
|
================== |
|
*/ |
|
static edict_t *SV_GetCrossEnt( edict_t *player ) |
|
{ |
|
edict_t *ent = EDICT_NUM(1); |
|
edict_t *closest = NULL; |
|
float flMaxDot = 0.94; |
|
vec3_t forward; |
|
vec3_t viewPos; |
|
int i; |
|
float maxLen = 1000; |
|
|
|
AngleVectors( player->v.v_angle, forward, NULL, NULL ); |
|
VectorAdd( player->v.origin, player->v.view_ofs, viewPos ); |
|
|
|
// find bmodels by trace |
|
{ |
|
trace_t trace; |
|
vec3_t target; |
|
|
|
VectorMA( viewPos, 1000, forward, target ); |
|
trace = SV_Move( viewPos, vec3_origin, vec3_origin, target, 0, player, false ); |
|
closest = trace.ent; |
|
VectorSubtract( viewPos, trace.endpos, target ); |
|
maxLen = VectorLength(target) + 30; |
|
} |
|
|
|
// check untraceable entities |
|
for ( i = 1; i < svgame.numEntities; i++, ent++ ) |
|
{ |
|
vec3_t vecLOS; |
|
vec3_t vecOrigin; |
|
float flDot, traceLen; |
|
vec3_t boxSize; |
|
trace_t trace; |
|
vec3_t vecTrace; |
|
|
|
if( ent->free ) |
|
continue; |
|
|
|
if( ent->v.solid == SOLID_BSP || ent->v.movetype == MOVETYPE_PUSHSTEP ) |
|
continue; // bsp models will be found by trace later |
|
|
|
// do not touch following weapons |
|
if( ent->v.movetype == MOVETYPE_FOLLOW ) |
|
continue; |
|
|
|
if( ent == player ) |
|
continue; |
|
|
|
VectorAdd( ent->v.absmin, ent->v.absmax, vecOrigin ); |
|
VectorScale( vecOrigin, 0.5, vecOrigin ); |
|
|
|
VectorSubtract( vecOrigin, viewPos, vecLOS ); |
|
traceLen = VectorLength(vecLOS); |
|
|
|
if( traceLen > maxLen ) |
|
continue; |
|
|
|
VectorCopy( ent->v.size, boxSize); |
|
VectorScale( boxSize, 0.5, boxSize ); |
|
|
|
if ( vecLOS[0] > boxSize[0] ) |
|
vecLOS[0] -= boxSize[0]; |
|
else if ( vecLOS[0] < -boxSize[0] ) |
|
vecLOS[0] += boxSize[0]; |
|
else |
|
vecLOS[0] = 0; |
|
|
|
if ( vecLOS[1] > boxSize[1] ) |
|
vecLOS[1] -= boxSize[1]; |
|
else if ( vecLOS[1] < -boxSize[1] ) |
|
vecLOS[1] += boxSize[1]; |
|
else |
|
vecLOS[1] = 0; |
|
|
|
if ( vecLOS[2] > boxSize[2] ) |
|
vecLOS[2] -= boxSize[2]; |
|
else if ( vecLOS[2] < -boxSize[2] ) |
|
vecLOS[2] += boxSize[2]; |
|
else |
|
vecLOS[2] = 0; |
|
VectorNormalize( vecLOS ); |
|
|
|
flDot = DotProduct (vecLOS , forward); |
|
if ( flDot <= flMaxDot ) |
|
continue; |
|
|
|
trace = SV_Move( viewPos, vec3_origin, vec3_origin, vecOrigin, 0, player, false ); |
|
VectorSubtract( trace.endpos, viewPos, vecTrace ); |
|
if( VectorLength( vecTrace ) + 30 < traceLen ) |
|
continue; |
|
closest = ent, flMaxDot = flDot; |
|
} |
|
|
|
return closest; |
|
} |
|
|
|
/* |
|
================== |
|
SV_EntFindSingle |
|
================== |
|
*/ |
|
static edict_t *SV_EntFindSingle( sv_client_t *cl, const char *pattern ) |
|
{ |
|
edict_t *ent = NULL; |
|
int i = 0; |
|
|
|
if( Q_isdigit( pattern ) ) |
|
{ |
|
i = Q_atoi( pattern ); |
|
|
|
if( i >= svgame.numEntities ) |
|
return NULL; |
|
} |
|
else if( !Q_stricmp( pattern, "!cross" ) ) |
|
{ |
|
ent = SV_GetCrossEnt( cl->edict ); |
|
|
|
if( !SV_IsValidEdict( ent ) ) |
|
return NULL; |
|
|
|
i = NUM_FOR_EDICT( ent ); |
|
} |
|
else if( pattern[0] == '!' ) // check for correct instance with !(num)_(serial) |
|
{ |
|
const char *p = pattern + 1; |
|
i = Q_atoi( p ); |
|
|
|
while( Q_isdigit( p )) p++; |
|
|
|
if( *p++ != '_' ) |
|
return NULL; |
|
|
|
if( i >= svgame.numEntities ) |
|
return NULL; |
|
|
|
ent = EDICT_NUM( i ); |
|
|
|
if( ent->serialnumber != Q_atoi( p ) ) |
|
return NULL; |
|
} |
|
else |
|
{ |
|
for( i = svgame.globals->maxClients + 1; i < svgame.numEntities; i++ ) |
|
{ |
|
ent = EDICT_NUM( i ); |
|
|
|
if( !SV_IsValidEdict( ent ) ) |
|
continue; |
|
|
|
if( Q_stricmpext( pattern, STRING( ent->v.targetname ) ) ) |
|
break; |
|
} |
|
} |
|
|
|
ent = EDICT_NUM( i ); |
|
|
|
if( !SV_IsValidEdict( ent ) ) |
|
return NULL; |
|
|
|
return ent; |
|
} |
|
|
|
/* |
|
=============== |
|
SV_EntList_f |
|
|
|
Print list of entities to client |
|
=============== |
|
*/ |
|
static qboolean SV_EntList_f( sv_client_t *cl ) |
|
{ |
|
vec3_t borigin; |
|
edict_t *ent = NULL; |
|
int i; |
|
|
|
for( i = 0; i < svgame.numEntities; i++ ) |
|
{ |
|
ent = EDICT_NUM( i ); |
|
if( !SV_IsValidEdict( ent )) |
|
continue; |
|
|
|
// filter by string |
|
if( Cmd_Argc() > 1 ) |
|
{ |
|
if( !Q_stricmpext( Cmd_Argv( 1 ), STRING( ent->v.classname ) ) && !Q_stricmpext( Cmd_Argv( 1 ), STRING( ent->v.targetname ) ) ) |
|
continue; |
|
} |
|
|
|
VectorAdd( ent->v.absmin, ent->v.absmax, borigin ); |
|
VectorScale( borigin, 0.5, borigin ); |
|
|
|
SV_ClientPrintf( cl, "%5i origin: %.f %.f %.f", i, ent->v.origin[0], ent->v.origin[1], ent->v.origin[2] ); |
|
SV_ClientPrintf( cl, "%5i borigin: %.f %.f %.f", i, borigin[0], borigin[1], borigin[2] ); |
|
|
|
if( ent->v.classname ) |
|
SV_ClientPrintf( cl, ", class: %s", STRING( ent->v.classname )); |
|
|
|
if( ent->v.globalname ) |
|
SV_ClientPrintf( cl, ", global: %s", STRING( ent->v.globalname )); |
|
|
|
if( ent->v.targetname ) |
|
SV_ClientPrintf( cl, ", name: %s", STRING( ent->v.targetname )); |
|
|
|
if( ent->v.target ) |
|
SV_ClientPrintf( cl, ", target: %s", STRING( ent->v.target )); |
|
|
|
if( ent->v.model ) |
|
SV_ClientPrintf( cl, ", model: %s", STRING( ent->v.model )); |
|
|
|
SV_ClientPrintf( cl, "\n" ); |
|
} |
|
return true; |
|
} |
|
|
|
/* |
|
=============== |
|
SV_EntInfo_f |
|
|
|
Print specified entity information to client |
|
=============== |
|
*/ |
|
static qboolean SV_EntInfo_f( sv_client_t *cl ) |
|
{ |
|
edict_t *ent = NULL; |
|
vec3_t borigin; |
|
|
|
if( Cmd_Argc() != 2 ) |
|
{ |
|
SV_ClientPrintf( cl, "Use ent_info <index|name|inst>\n" ); |
|
return false; |
|
} |
|
|
|
ent = SV_EntFindSingle( cl, Cmd_Argv( 1 ) ); |
|
|
|
if( !SV_IsValidEdict( ent )) |
|
return false; |
|
|
|
VectorAdd( ent->v.absmin, ent->v.absmax, borigin ); |
|
VectorScale( borigin, 0.5, borigin ); |
|
|
|
SV_ClientPrintf( cl, "origin: %.f %.f %.f\n", ent->v.origin[0], ent->v.origin[1], ent->v.origin[2] ); |
|
SV_ClientPrintf( cl, "angles: %.f %.f %.f\n", ent->v.angles[0], ent->v.angles[1], ent->v.angles[2] ); |
|
SV_ClientPrintf( cl, "borigin: %.f %.f %.f\n", borigin[0], borigin[1], borigin[2] ); |
|
|
|
if( ent->v.classname ) |
|
SV_ClientPrintf( cl, "class: %s\n", STRING( ent->v.classname )); |
|
|
|
if( ent->v.globalname ) |
|
SV_ClientPrintf( cl, "global: %s\n", STRING( ent->v.globalname )); |
|
|
|
if( ent->v.targetname ) |
|
SV_ClientPrintf( cl, "name: %s\n", STRING( ent->v.targetname )); |
|
|
|
if( ent->v.target ) |
|
SV_ClientPrintf( cl, "target: %s\n", STRING( ent->v.target )); |
|
|
|
if( ent->v.model ) |
|
SV_ClientPrintf( cl, "model: %s\n", STRING( ent->v.model )); |
|
|
|
SV_ClientPrintf( cl, "health: %.f\n", ent->v.health ); |
|
|
|
if( ent->v.gravity != 1.0f ) |
|
SV_ClientPrintf( cl, "gravity: %.2f\n", ent->v.gravity ); |
|
|
|
SV_ClientPrintf( cl, "movetype: %d\n", ent->v.movetype ); |
|
SV_ClientPrintf( cl, "rendermode: %d\n", ent->v.rendermode ); |
|
SV_ClientPrintf( cl, "renderfx: %d\n", ent->v.renderfx ); |
|
SV_ClientPrintf( cl, "renderamt: %f\n", ent->v.renderamt ); |
|
SV_ClientPrintf( cl, "rendercolor: %f %f %f\n", ent->v.rendercolor[0], ent->v.rendercolor[1], ent->v.rendercolor[2] ); |
|
SV_ClientPrintf( cl, "maxspeed: %f\n", ent->v.maxspeed ); |
|
|
|
if( ent->v.solid ) |
|
SV_ClientPrintf( cl, "solid: %d\n", ent->v.solid ); |
|
|
|
SV_ClientPrintf( cl, "flags: 0x%x\n", ent->v.flags ); |
|
SV_ClientPrintf( cl, "spawnflags: 0x%x\n", ent->v.spawnflags ); |
|
return true; |
|
} |
|
|
|
/* |
|
=============== |
|
SV_EntFire_f |
|
|
|
Perform some actions |
|
=============== |
|
*/ |
|
static qboolean SV_EntFire_f( sv_client_t *cl ) |
|
{ |
|
edict_t *ent = NULL; |
|
int i = 1, count = 0; |
|
qboolean single; // true if user specified something that match single entity |
|
|
|
if( Cmd_Argc() < 3 ) |
|
{ |
|
SV_ClientPrintf( cl, "Use ent_fire <index||pattern> <command> [<values>]\n" |
|
"Use ent_fire 0 help to get command list\n" ); |
|
return false; |
|
} |
|
|
|
if( ( single = Q_isdigit( Cmd_Argv( 1 ) ) ) ) |
|
{ |
|
i = Q_atoi( Cmd_Argv( 1 ) ); |
|
|
|
if( i < 0 || i >= svgame.numEntities ) |
|
return false; |
|
|
|
ent = EDICT_NUM( i ); |
|
} |
|
else if( ( single = !Q_stricmp( Cmd_Argv( 1 ), "!cross" ) ) ) |
|
{ |
|
ent = SV_GetCrossEnt( cl->edict ); |
|
|
|
if (!SV_IsValidEdict(ent)) |
|
return false; |
|
|
|
i = NUM_FOR_EDICT( ent ); |
|
} |
|
else if( ( single = ( Cmd_Argv( 1 )[0] == '!') ) ) // check for correct instance with !(num)_(serial) |
|
{ |
|
const char *cmd = Cmd_Argv( 1 ) + 1; |
|
i = Q_atoi( cmd ); |
|
|
|
while( Q_isdigit( cmd )) cmd++; |
|
|
|
if( *cmd++ != '_' ) |
|
return false; |
|
|
|
if( i < 0 || i >= svgame.numEntities ) |
|
return false; |
|
|
|
ent = EDICT_NUM( i ); |
|
if( ent->serialnumber != Q_atoi( cmd ) ) |
|
return false; |
|
} |
|
else |
|
{ |
|
i = svgame.globals->maxClients + 1; |
|
} |
|
|
|
for( ; ( i < svgame.numEntities ) && ( count < sv_enttools_maxfire.value ); i++ ) |
|
{ |
|
ent = EDICT_NUM( i ); |
|
if( !SV_IsValidEdict( ent )) |
|
{ |
|
// SV_ClientPrintf( cl, PRINT_LOW, "Got invalid entity\n" ); |
|
if( single ) |
|
break; |
|
continue; |
|
} |
|
|
|
// if user specified not a number, try find such entity |
|
if( !single ) |
|
{ |
|
if( !Q_stricmpext( Cmd_Argv( 1 ), STRING( ent->v.targetname ) ) && !Q_stricmpext( Cmd_Argv( 1 ), STRING( ent->v.classname ) )) |
|
continue; |
|
} |
|
|
|
SV_ClientPrintf( cl, "entity %i\n", i ); |
|
|
|
count++; |
|
|
|
if( !Q_stricmp( Cmd_Argv( 2 ), "health" ) ) |
|
ent->v.health = Q_atoi( Cmd_Argv ( 3 ) ); |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "gravity" ) ) |
|
ent->v.gravity = Q_atof( Cmd_Argv ( 3 ) ); |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "movetype" ) ) |
|
ent->v.movetype = Q_atoi( Cmd_Argv ( 3 ) ); |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "solid" ) ) |
|
ent->v.solid = Q_atoi( Cmd_Argv ( 3 ) ); |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "rename" ) ) |
|
ent->v.targetname = ALLOC_STRING( Cmd_Argv ( 3 ) ); |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "settarget" ) ) |
|
ent->v.target = ALLOC_STRING( Cmd_Argv ( 3 ) ); |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "setmodel" ) ) |
|
SV_SetModel( ent, Cmd_Argv( 3 ) ); |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "set" ) ) |
|
{ |
|
string keyname; |
|
string value; |
|
KeyValueData pkvd; |
|
if( Cmd_Argc() != 5 ) |
|
return false; |
|
|
|
pkvd.szClassName = (char*)STRING( ent->v.classname ); |
|
Q_strncpy( keyname, Cmd_Argv( 3 ), sizeof( keyname )); |
|
Q_strncpy( value, Cmd_Argv( 4 ), sizeof( value )); |
|
pkvd.szKeyName = keyname; |
|
pkvd.szValue = value; |
|
pkvd.fHandled = false; |
|
svgame.dllFuncs.pfnKeyValue( ent, &pkvd ); |
|
|
|
if( pkvd.fHandled ) |
|
SV_ClientPrintf( cl, "value set successfully!\n" ); |
|
} |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "touch" ) ) |
|
{ |
|
if( Cmd_Argc() == 4 ) |
|
{ |
|
edict_t *other = SV_EntFindSingle( cl, Cmd_Argv( 3 ) ); |
|
if( other && other->pvPrivateData ) |
|
svgame.dllFuncs.pfnTouch( ent, other ); |
|
} |
|
else |
|
svgame.dllFuncs.pfnTouch( ent, cl->edict ); |
|
} |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "use" ) ) |
|
{ |
|
if( Cmd_Argc() == 4 ) |
|
{ |
|
edict_t *other = SV_EntFindSingle( cl, Cmd_Argv( 3 ) ); |
|
if( other && other->pvPrivateData ) |
|
svgame.dllFuncs.pfnUse( ent, other ); |
|
} |
|
else |
|
svgame.dllFuncs.pfnUse( ent, cl->edict ); |
|
} |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "movehere" ) ) |
|
{ |
|
ent->v.origin[2] = cl->edict->v.origin[2] + 25; |
|
ent->v.origin[1] = cl->edict->v.origin[1] + 100 * sin( DEG2RAD( cl->edict->v.angles[1] ) ); |
|
ent->v.origin[0] = cl->edict->v.origin[0] + 100 * cos( DEG2RAD( cl->edict->v.angles[1] ) ); |
|
SV_LinkEdict( ent, true ); |
|
} |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "drop2floor" ) ) |
|
{ |
|
pfnDropToFloor( ent ); |
|
} |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "moveup" ) ) |
|
{ |
|
float dist = 25; |
|
if( Cmd_Argc() >= 4 ) |
|
dist = Q_atof( Cmd_Argv( 3 ) ); |
|
ent->v.origin[2] += dist; |
|
if( Cmd_Argc() >= 5 ) |
|
{ |
|
dist = Q_atof( Cmd_Argv( 4 ) ); |
|
ent->v.origin[0] += dist * cos( DEG2RAD( cl->edict->v.angles[1] ) ); |
|
ent->v.origin[1] += dist * sin( DEG2RAD( cl->edict->v.angles[1] ) ); |
|
} |
|
|
|
} |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "becomeowner" ) ) |
|
{ |
|
if( Cmd_Argc() == 4 ) |
|
ent->v.owner = SV_EntFindSingle( cl, Cmd_Argv( 3 ) ); |
|
else |
|
ent->v.owner = cl->edict; |
|
} |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "becomeenemy" ) ) |
|
{ |
|
if( Cmd_Argc() == 4 ) |
|
ent->v.enemy = SV_EntFindSingle( cl, Cmd_Argv( 3 ) ); |
|
else |
|
ent->v.enemy = cl->edict; |
|
} |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "becomeaiment" ) ) |
|
{ |
|
if( Cmd_Argc() == 4 ) |
|
ent->v.aiment= SV_EntFindSingle( cl, Cmd_Argv( 3 ) ); |
|
else |
|
ent->v.aiment = cl->edict; |
|
} |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "hullmin" ) ) |
|
{ |
|
if( Cmd_Argc() != 6 ) |
|
return false; |
|
ent->v.mins[0] = Q_atof( Cmd_Argv( 3 ) ); |
|
ent->v.mins[1] = Q_atof( Cmd_Argv( 4 ) ); |
|
ent->v.mins[2] = Q_atof( Cmd_Argv( 5 ) ); |
|
} |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "hullmax" ) ) |
|
{ |
|
if( Cmd_Argc() != 6 ) |
|
return false; |
|
ent->v.maxs[0] = Q_atof( Cmd_Argv( 3 ) ); |
|
ent->v.maxs[1] = Q_atof( Cmd_Argv( 4 ) ); |
|
ent->v.maxs[2] = Q_atof( Cmd_Argv( 5 ) ); |
|
} |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "rendercolor" ) ) |
|
{ |
|
if( Cmd_Argc() != 6 ) |
|
return false; |
|
ent->v.rendercolor[0] = Q_atof( Cmd_Argv( 3 ) ); |
|
ent->v.rendercolor[1] = Q_atof( Cmd_Argv( 4 ) ); |
|
ent->v.rendercolor[2] = Q_atof( Cmd_Argv( 5 ) ); |
|
} |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "renderamt" ) ) |
|
{ |
|
ent->v.renderamt = Q_atof( Cmd_Argv( 3 ) ); |
|
} |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "renderfx" ) ) |
|
{ |
|
ent->v.renderfx = Q_atoi( Cmd_Argv( 3 ) ); |
|
} |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "rendermode" ) ) |
|
{ |
|
ent->v.rendermode = Q_atoi( Cmd_Argv( 3 ) ); |
|
} |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "angles" ) ) |
|
{ |
|
ent->v.angles[0] = Q_atof( Cmd_Argv( 3 ) ); |
|
ent->v.angles[1] = Q_atof( Cmd_Argv( 4 ) ); |
|
ent->v.angles[2] = Q_atof( Cmd_Argv( 5 ) ); |
|
} |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "setflag" ) ) |
|
{ |
|
ent->v.flags |= 1U << Q_atoi( Cmd_Argv ( 3 ) ); |
|
SV_ClientPrintf( cl, "flags set to 0x%x\n", ent->v.flags ); |
|
} |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "clearflag" ) ) |
|
{ |
|
ent->v.flags &= ~( 1U << Q_atoi( Cmd_Argv ( 3 ) ) ); |
|
SV_ClientPrintf( cl, "flags set to 0x%x\n", ent->v.flags ); |
|
} |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "setspawnflag" ) ) |
|
{ |
|
ent->v.spawnflags |= 1U << Q_atoi( Cmd_Argv ( 3 ) ); |
|
SV_ClientPrintf( cl, "spawnflags set to 0x%x\n", ent->v.spawnflags ); |
|
} |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "clearspawnflag" ) ) |
|
{ |
|
ent->v.spawnflags &= ~( 1U << Q_atoi( Cmd_Argv ( 3 ) ) ); |
|
SV_ClientPrintf( cl, "spawnflags set to 0x%x\n", ent->v.flags ); |
|
} |
|
else if( !Q_stricmp( Cmd_Argv( 2 ), "help" ) ) |
|
{ |
|
SV_ClientPrintf( cl, "Available commands:\n" |
|
"Set fields:\n" |
|
" (Only set entity field, does not call any functions)\n" |
|
" health\n" |
|
" gravity\n" |
|
" movetype\n" |
|
" solid\n" |
|
" rendermode\n" |
|
" rendercolor (vector)\n" |
|
" renderfx\n" |
|
" renderamt\n" |
|
" hullmin (vector)\n" |
|
" hullmax (vector)\n" |
|
"Actions\n" |
|
" rename: set entity targetname\n" |
|
" settarget: set entity target (only targetnames)\n" |
|
" setmodel: set entity model\n" |
|
" set: set <key> <value> by server library\n" |
|
" See game FGD to get list.\n" |
|
" command takes two arguments\n" |
|
" touch: touch entity by current player.\n" |
|
" use: use entity by current player.\n" |
|
" movehere: place entity in player fov.\n" |
|
" drop2floor: place entity to nearest floor surface\n" |
|
" moveup: move entity to 25 units up\n" |
|
"Flags:\n" |
|
" (Set/clear specified flag bit, arg is bit number)\n" |
|
" setflag\n" |
|
" clearflag\n" |
|
" setspawnflag\n" |
|
" clearspawnflag\n" |
|
); |
|
return true; |
|
} |
|
else |
|
{ |
|
SV_ClientPrintf( cl, "Unknown command %s!\nUse \"ent_fire 0 help\" to list commands.\n", Cmd_Argv( 2 ) ); |
|
return false; |
|
} |
|
if( single ) |
|
break; |
|
} |
|
return true; |
|
} |
|
|
|
/* |
|
=============== |
|
SV_EntSendVars |
|
=============== |
|
*/ |
|
static void SV_EntSendVars( sv_client_t *cl, edict_t *ent ) |
|
{ |
|
if( !ent ) |
|
return; |
|
|
|
MSG_WriteByte( &cl->netchan.message, svc_stufftext ); |
|
MSG_WriteStringf( &cl->netchan.message, "set ent_last_name \"%s\"\n", STRING( ent->v.targetname )); |
|
MSG_WriteByte( &cl->netchan.message, svc_stufftext ); |
|
MSG_WriteStringf( &cl->netchan.message, "set ent_last_num %i\n", NUM_FOR_EDICT( ent )); |
|
MSG_WriteByte( &cl->netchan.message, svc_stufftext ); |
|
MSG_WriteStringf( &cl->netchan.message, "set ent_last_inst !%i_%i\n", NUM_FOR_EDICT( ent ), ent->serialnumber ); |
|
MSG_WriteByte( &cl->netchan.message, svc_stufftext ); |
|
MSG_WriteStringf( &cl->netchan.message, "set ent_last_origin \"%f %f %f\"\n", ent->v.origin[0], ent->v.origin[1], ent->v.origin[2] ); |
|
MSG_WriteByte( &cl->netchan.message, svc_stufftext ); |
|
MSG_WriteStringf( &cl->netchan.message, "set ent_last_class \"%s\"\n", STRING( ent->v.classname )); |
|
MSG_WriteByte( &cl->netchan.message, svc_stufftext ); |
|
MSG_WriteString( &cl->netchan.message, "ent_getvars_cb\n" ); // why do we need this? |
|
} |
|
|
|
/* |
|
=============== |
|
SV_EntCreate_f |
|
|
|
Create new entity with specified name. |
|
=============== |
|
*/ |
|
static qboolean SV_EntCreate_f( sv_client_t *cl ) |
|
{ |
|
edict_t *ent = NULL; |
|
int i = 0; |
|
string_t classname; |
|
|
|
if( Cmd_Argc() < 2 ) |
|
{ |
|
SV_ClientPrintf( cl, "Use ent_create <classname> <key1> <value1> <key2> <value2> ...\n" ); |
|
return false; |
|
} |
|
|
|
classname = ALLOC_STRING( Cmd_Argv( 1 ) ); |
|
|
|
ent = SV_CreateNamedEntity( 0, classname ); |
|
|
|
// Xash3D extension |
|
if( !ent && svgame.physFuncs.SV_CreateEntity ) |
|
{ |
|
ent = SV_AllocEdict(); |
|
ent->v.classname = classname; |
|
if( svgame.physFuncs.SV_CreateEntity( ent, (char*)STRING( classname ) ) == -1 ) |
|
{ |
|
if( ent && !ent->free ) |
|
SV_FreeEdict( ent ); |
|
ent = NULL; |
|
} |
|
} |
|
|
|
// XashXT does not implement SV_CreateEntity, use saverestore export |
|
if( !ent && svgame.physFuncs.pfnCreateEntitiesInRestoreList ) |
|
{ |
|
SAVERESTOREDATA data = { 0 }; |
|
ENTITYTABLE table = { 0 }; |
|
data.tableCount = 1; |
|
data.pTable = &table; |
|
table.classname = classname; |
|
table.id = -1; |
|
table.size = 1; |
|
svgame.physFuncs.pfnCreateEntitiesInRestoreList( &data, 0, false ); |
|
ent = table.pent; |
|
} |
|
|
|
if( !ent ) |
|
{ |
|
SV_ClientPrintf( cl, "Invalid entity!\n" ); |
|
return false; |
|
} |
|
|
|
// choose default origin |
|
ent->v.origin[2] = cl->edict->v.origin[2] + 25; |
|
ent->v.origin[1] = cl->edict->v.origin[1] + 100 * sin( DEG2RAD( cl->edict->v.angles[1] ) ); |
|
ent->v.origin[0] = cl->edict->v.origin[0] + 100 * cos( DEG2RAD( cl->edict->v.angles[1] ) ); |
|
|
|
SV_LinkEdict( ent, false ); |
|
|
|
// apply keyvalues if supported |
|
if( svgame.dllFuncs.pfnKeyValue ) |
|
{ |
|
for( i = 2; i < Cmd_Argc() - 1; i++ ) |
|
{ |
|
string keyname; |
|
string value; |
|
KeyValueData pkvd; |
|
|
|
// allow split keyvalues to prespawn and postspawn |
|
if( !Q_strcmp( Cmd_Argv( i ), "|" ) ) |
|
break; |
|
|
|
Q_strncpy( keyname, Cmd_Argv( i++ ), sizeof( keyname )); |
|
Q_strncpy( value, Cmd_Argv( i ), sizeof( value )); |
|
pkvd.fHandled = false; |
|
pkvd.szClassName = (char*)STRING( ent->v.classname ); |
|
pkvd.szKeyName = keyname; |
|
pkvd.szValue = value; |
|
svgame.dllFuncs.pfnKeyValue( ent, &pkvd ); |
|
|
|
if( pkvd.fHandled ) |
|
SV_ClientPrintf( cl, "value \"%s\" set to \"%s\"!\n", pkvd.szKeyName, pkvd.szValue ); |
|
} |
|
} |
|
|
|
// set default targetname |
|
if( !ent->v.targetname ) |
|
{ |
|
string newname, clientname; |
|
int j; |
|
|
|
for( j = 0; j < sizeof( cl->name ); j++ ) |
|
{ |
|
char c = Q_tolower( cl->name[j] ); |
|
if( c < 'a' || c > 'z' ) |
|
c = '_'; |
|
if( !cl->name[j] ) |
|
{ |
|
clientname[j] = 0; |
|
break; |
|
} |
|
clientname[j] = c; |
|
} |
|
|
|
// generate name based on nick name and index |
|
Q_snprintf( newname, sizeof( newname ), "%s_%i_e%i", clientname, cl->userid, NUM_FOR_EDICT( ent )); |
|
|
|
// i know, it may break strict aliasing rules |
|
// but we will not lose anything in this case. |
|
Q_strnlwr( newname, newname, sizeof( newname )); |
|
ent->v.targetname = ALLOC_STRING( newname ); |
|
SV_EntSendVars( cl, ent ); |
|
} |
|
|
|
SV_ClientPrintf( cl, "Created %i: %s, targetname %s\n", NUM_FOR_EDICT( ent ), Cmd_Argv( 1 ), STRING( ent->v.targetname ) ); |
|
|
|
if( svgame.dllFuncs.pfnSpawn ) |
|
svgame.dllFuncs.pfnSpawn( ent ); |
|
|
|
// now drop entity to floor. |
|
pfnDropToFloor( ent ); |
|
|
|
// force think. Otherwise given weapon may crash server if player touch it before. |
|
svgame.dllFuncs.pfnThink( ent ); |
|
pfnDropToFloor( ent ); |
|
|
|
// apply postspawn keyvales if supported |
|
if( svgame.dllFuncs.pfnKeyValue ) |
|
{ |
|
for( i = i + 1; i < Cmd_Argc() - 1; i++ ) |
|
{ |
|
string keyname; |
|
string value; |
|
KeyValueData pkvd; |
|
|
|
Q_strncpy( keyname, Cmd_Argv( i++ ), sizeof( keyname )); |
|
Q_strncpy( value, Cmd_Argv( i ), sizeof( value )); |
|
pkvd.fHandled = false; |
|
pkvd.szClassName = (char*)STRING( ent->v.classname ); |
|
pkvd.szKeyName = keyname; |
|
pkvd.szValue = value; |
|
svgame.dllFuncs.pfnKeyValue( ent, &pkvd ); |
|
|
|
if( pkvd.fHandled ) |
|
SV_ClientPrintf( cl, "value \"%s\" set to \"%s\"!\n", pkvd.szKeyName, pkvd.szValue ); |
|
} |
|
} |
|
return true; |
|
} |
|
|
|
static qboolean SV_EntGetVars_f( sv_client_t *cl ) |
|
{ |
|
edict_t *ent = NULL; |
|
|
|
if( Cmd_Argc() != 2 ) |
|
{ |
|
SV_ClientPrintf( cl, "Use ent_getvars <index|name|inst>\n" ); |
|
return false; |
|
} |
|
|
|
ent = SV_EntFindSingle( cl, Cmd_Argv( 1 ) ); |
|
if( Cmd_Argc() ) |
|
{ |
|
if( !SV_IsValidEdict( ent )) |
|
return false; |
|
} |
|
|
|
SV_EntSendVars( cl, ent ); |
|
return true; |
|
} |
|
|
|
ucmd_t ucmds[] = |
|
{ |
|
{ "new", SV_New_f }, |
|
{ "god", SV_Godmode_f }, |
|
{ "kill", SV_Kill_f }, |
|
{ "begin", SV_Begin_f }, |
|
{ "spawn", SV_Spawn_f }, |
|
{ "pause", SV_Pause_f }, |
|
{ "noclip", SV_Noclip_f }, |
|
{ "setinfo", SV_SetInfo_f }, |
|
{ "sendres", SV_SendRes_f }, |
|
{ "notarget", SV_Notarget_f }, |
|
{ "info", SV_ShowServerinfo_f }, |
|
{ "dlfile", SV_DownloadFile_f }, |
|
{ "disconnect", SV_Disconnect_f }, |
|
{ "userinfo", SV_UpdateUserinfo_f }, |
|
{ "_sv_build_info", SV_SendBuildInfo_f }, |
|
{ NULL, NULL } |
|
}; |
|
|
|
ucmd_t enttoolscmds[] = |
|
{ |
|
{ "ent_list", SV_EntList_f }, |
|
{ "ent_info", SV_EntInfo_f }, |
|
{ "ent_fire", SV_EntFire_f }, |
|
{ "ent_create", SV_EntCreate_f }, |
|
{ "ent_getvars", SV_EntGetVars_f }, |
|
{ NULL, NULL } |
|
}; |
|
|
|
/* |
|
================== |
|
SV_ExecuteUserCommand |
|
================== |
|
*/ |
|
static void SV_ExecuteClientCommand( sv_client_t *cl, const char *s ) |
|
{ |
|
ucmd_t *u; |
|
|
|
Cmd_TokenizeString( s ); |
|
|
|
for( u = ucmds; u->name; u++ ) |
|
{ |
|
if( !Q_strcmp( Cmd_Argv( 0 ), u->name )) |
|
{ |
|
if( !u->func( cl )) |
|
Con_Printf( "'%s' is not valid from the console\n", u->name ); |
|
else Con_Reportf( "ucmd->%s()\n", u->name ); |
|
break; |
|
} |
|
} |
|
|
|
if( !u->name && sv_enttools_enable.value > 0.0f && !sv.background ) |
|
{ |
|
for( u = enttoolscmds; u->name; u++ ) |
|
{ |
|
if( !Q_strcmp( Cmd_Argv( 0 ), u->name )) |
|
{ |
|
Con_Reportf( "enttools->%s(): %s\n", u->name, s ); |
|
Log_Printf( "\"%s<%i><%s><>\" performed: %s\n", Info_ValueForKey( cl->userinfo, "name" ), |
|
cl->userid, SV_GetClientIDString( cl ), s ); |
|
|
|
if( u->func ) |
|
u->func( cl ); |
|
|
|
break; |
|
} |
|
} |
|
} |
|
|
|
if( !u->name && sv.state == ss_active ) |
|
{ |
|
qboolean fullupdate = !Q_strcmp( Cmd_Argv( 0 ), "fullupdate" ); |
|
|
|
if( fullupdate ) |
|
{ |
|
if( sv_fullupdate_penalty_time.value && host.realtime < cl->fullupdate_next_calltime ) |
|
return; |
|
} |
|
|
|
// custom client commands |
|
svgame.dllFuncs.pfnClientCommand( cl->edict ); |
|
|
|
if( fullupdate ) |
|
{ |
|
// resend the ambient sounds for demo recording |
|
SV_RestartAmbientSounds(); |
|
// resend all the decals for demo recording |
|
SV_RestartDecals(); |
|
// resend all the static ents for demo recording |
|
SV_RestartStaticEnts(); |
|
// resend the viewentity |
|
SV_UpdateClientView( cl ); |
|
|
|
if( sv_fullupdate_penalty_time.value ) |
|
cl->fullupdate_next_calltime = host.realtime + sv_fullupdate_penalty_time.value; |
|
} |
|
} |
|
} |
|
|
|
/* |
|
================= |
|
SV_ConnectionlessPacket |
|
|
|
A connectionless packet has four leading 0xff |
|
characters to distinguish it from a game channel. |
|
Clients that are in the game can still send |
|
connectionless packets. |
|
================= |
|
*/ |
|
void SV_ConnectionlessPacket( netadr_t from, sizebuf_t *msg ) |
|
{ |
|
char *args; |
|
const char *pcmd; |
|
char buf[MAX_SYSPATH]; |
|
int len = sizeof( buf ); |
|
|
|
// prevent flooding from banned address |
|
if( SV_CheckIP( &from ) ) |
|
return; |
|
|
|
MSG_Clear( msg ); |
|
MSG_ReadLong( msg );// skip the -1 marker |
|
|
|
args = MSG_ReadStringLine( msg ); |
|
Cmd_TokenizeString( args ); |
|
|
|
pcmd = Cmd_Argv( 0 ); |
|
|
|
if( sv_log_outofband.value ) |
|
Con_Reportf( "SV_ConnectionlessPacket: %s : %s\n", NET_AdrToString( from ), pcmd ); |
|
|
|
if( !Q_strcmp( pcmd, "ping" )) SV_Ping( from ); |
|
else if( !Q_strcmp( pcmd, "ack" )) SV_Ack( from ); |
|
else if( !Q_strcmp( pcmd, "info" )) SV_Info( from, Q_atoi( Cmd_Argv( 1 ))); |
|
else if( !Q_strcmp( pcmd, "bandwidth" )) SV_TestBandWidth( from ); |
|
else if( !Q_strcmp( pcmd, "getchallenge" )) SV_GetChallenge( from ); |
|
else if( !Q_strcmp( pcmd, "connect" )) SV_ConnectClient( from ); |
|
else if( !Q_strcmp( pcmd, "rcon" )) SV_RemoteCommand( from, msg ); |
|
else if( !Q_strcmp( pcmd, "netinfo" )) SV_BuildNetAnswer( from ); |
|
else if( !Q_strcmp( pcmd, "s" )) SV_AddToMaster( from, msg ); |
|
else if( !Q_strcmp( pcmd, "i" )) NET_SendPacket( NS_SERVER, 5, "\xFF\xFF\xFF\xFFj", from ); // A2A_PING |
|
else if( SV_SourceQuery_HandleConnnectionlessPacket( pcmd, from )) { } // function handles replies |
|
else if( !Q_strcmp( pcmd, "c" ) && sv_nat.value && NET_IsMasterAdr( from )) |
|
{ |
|
netadr_t to; |
|
if( NET_StringToAdr( Cmd_Argv( 1 ), &to ) && !NET_IsReservedAdr( to )) |
|
SV_Info( to, PROTOCOL_VERSION ); |
|
} |
|
else if( svgame.dllFuncs.pfnConnectionlessPacket( &from, args, buf, &len )) |
|
{ |
|
// user out of band message (must be handled in CL_ConnectionlessPacket) |
|
if( len > 0 ) Netchan_OutOfBand( NS_SERVER, from, len, (byte*)buf ); |
|
} |
|
else Con_DPrintf( S_ERROR "bad connectionless packet from %s:\n%s\n", NET_AdrToString( from ), args ); |
|
} |
|
|
|
/* |
|
================== |
|
SV_ParseClientMove |
|
|
|
The message usually contains all the movement commands |
|
that were in the last three packets, so that the information |
|
in dropped packets can be recovered. |
|
|
|
On very fast clients, there may be multiple usercmd packed into |
|
each of the backup packets. |
|
================== |
|
*/ |
|
static void SV_ParseClientMove( sv_client_t *cl, sizebuf_t *msg ) |
|
{ |
|
client_frame_t *frame; |
|
int key, size, checksum1, checksum2; |
|
int i, numbackup, totalcmds, numcmds; |
|
usercmd_t nullcmd, *to, *from; |
|
usercmd_t cmds[CMD_BACKUP]; |
|
float packet_loss; |
|
edict_t *player; |
|
model_t *model; |
|
|
|
player = cl->edict; |
|
|
|
frame = &cl->frames[cl->netchan.incoming_acknowledged & SV_UPDATE_MASK]; |
|
memset( &nullcmd, 0, sizeof( usercmd_t )); |
|
memset( cmds, 0, sizeof( cmds )); |
|
|
|
key = MSG_GetRealBytesRead( msg ); |
|
checksum1 = MSG_ReadByte( msg ); |
|
packet_loss = MSG_ReadByte( msg ); |
|
|
|
numbackup = MSG_ReadByte( msg ); |
|
numcmds = MSG_ReadByte( msg ); |
|
|
|
totalcmds = numcmds + numbackup; |
|
net_drop -= (numcmds - 1); |
|
|
|
if( totalcmds < 0 || totalcmds >= CMD_MASK ) |
|
{ |
|
Con_Reportf( S_ERROR "SV_ParseClientMove: %s sending too many commands %i\n", cl->name, totalcmds ); |
|
SV_DropClient( cl, false ); |
|
return; |
|
} |
|
|
|
from = &nullcmd; // first cmd are starting from null-compressed usercmd_t |
|
|
|
for( i = totalcmds - 1; i >= 0; i-- ) |
|
{ |
|
to = &cmds[i]; |
|
MSG_ReadDeltaUsercmd( msg, from, to ); |
|
from = to; // get new baseline |
|
} |
|
|
|
if( cl->state != cs_spawned ) |
|
return; |
|
|
|
// if the checksum fails, ignore the rest of the packet |
|
size = MSG_GetRealBytesRead( msg ) - key - 1; |
|
checksum2 = CRC32_BlockSequence( msg->pData + key + 1, size, cl->netchan.incoming_sequence ); |
|
|
|
if( checksum2 != checksum1 ) |
|
{ |
|
Con_Reportf( S_ERROR "SV_UserMove: failed command checksum for %s (%d != %d)\n", cl->name, checksum2, checksum1 ); |
|
return; |
|
} |
|
|
|
cl->packet_loss = packet_loss; |
|
|
|
// freeze player for some reasons if loadgame was executed |
|
if( GameState->loadGame ) |
|
return; |
|
|
|
// check for pause or frozen |
|
if( sv.paused || !CL_IsInGame() || SV_PlayerIsFrozen( player )) |
|
{ |
|
for( i = 0; i < numcmds; i++ ) |
|
{ |
|
cmds[i].msec = 0; |
|
cmds[i].forwardmove = 0; |
|
cmds[i].sidemove = 0; |
|
cmds[i].upmove = 0; |
|
cmds[i].buttons = 0; |
|
|
|
if( SV_PlayerIsFrozen( player )) |
|
cmds[i].impulse = 0; |
|
|
|
VectorCopy( cmds[i].viewangles, player->v.v_angle ); |
|
} |
|
net_drop = 0; |
|
} |
|
else |
|
{ |
|
if( !player->v.fixangle ) |
|
VectorCopy( cmds[0].viewangles, player->v.v_angle ); |
|
} |
|
|
|
SV_EstablishTimeBase( cl, cmds, net_drop, numbackup, numcmds ); |
|
|
|
if( net_drop < 24 ) |
|
{ |
|
while( net_drop > numbackup ) |
|
{ |
|
SV_RunCmd( cl, &cl->lastcmd, 0 ); |
|
net_drop--; |
|
} |
|
|
|
while( net_drop > 0 ) |
|
{ |
|
i = numcmds + net_drop - 1; |
|
SV_RunCmd( cl, &cmds[i], cl->netchan.incoming_sequence - i ); |
|
net_drop--; |
|
} |
|
} |
|
|
|
for( i = numcmds - 1; i >= 0; i-- ) |
|
{ |
|
SV_RunCmd( cl, &cmds[i], cl->netchan.incoming_sequence - i ); |
|
} |
|
|
|
cl->lastcmd = cmds[0]; |
|
|
|
// adjust latency time by 1/2 last client frame since |
|
// the message probably arrived 1/2 through client's frame loop |
|
frame->ping_time -= ( cl->lastcmd.msec * 0.5f ) / 1000.0f; |
|
frame->ping_time = Q_max( 0.0f, frame->ping_time ); |
|
model = SV_ModelHandle( player->v.modelindex ); |
|
|
|
if( model && model->type == mod_studio ) |
|
{ |
|
// g-cont. yes we using svgame.globals->time instead of sv.time |
|
if( player->v.animtime > svgame.globals->time + sv.frametime ) |
|
player->v.animtime = svgame.globals->time + sv.frametime; |
|
} |
|
} |
|
|
|
/* |
|
=================== |
|
SV_ParseResourceList |
|
|
|
Parse resource list |
|
=================== |
|
*/ |
|
static void SV_ParseResourceList( sv_client_t *cl, sizebuf_t *msg ) |
|
{ |
|
int totalsize; |
|
resource_t *resource; |
|
int i, total; |
|
resourceinfo_t ri; |
|
|
|
total = MSG_ReadShort( msg ); |
|
|
|
SV_ClearResourceList( &cl->resourcesneeded ); |
|
SV_ClearResourceList( &cl->resourcesonhand ); |
|
|
|
for( i = 0; i < total; i++ ) |
|
{ |
|
resource = Z_Calloc( sizeof( resource_t ) ); |
|
Q_strncpy( resource->szFileName, MSG_ReadString( msg ), sizeof( resource->szFileName )); |
|
resource->type = MSG_ReadByte( msg ); |
|
resource->nIndex = MSG_ReadShort( msg ); |
|
resource->nDownloadSize = MSG_ReadLong( msg ); |
|
resource->ucFlags = MSG_ReadByte( msg ); |
|
resource->pNext = NULL; |
|
resource->pPrev = NULL; |
|
ClearBits( resource->ucFlags, RES_WASMISSING ); |
|
|
|
if( FBitSet( resource->ucFlags, RES_CUSTOM )) |
|
MSG_ReadBytes( msg, resource->rgucMD5_hash, 16 ); |
|
|
|
if( resource->type > t_world || resource->nDownloadSize > 1024 * 1024 * 1024 ) |
|
{ |
|
SV_ClearResourceList( &cl->resourcesneeded ); |
|
SV_ClearResourceList( &cl->resourcesonhand ); |
|
return; |
|
} |
|
SV_AddToResourceList( resource, &cl->resourcesneeded ); |
|
} |
|
|
|
totalsize = COM_SizeofResourceList( &cl->resourcesneeded, &ri ); |
|
|
|
if( totalsize != 0 && sv_allow_upload.value ) |
|
{ |
|
Con_DPrintf( "Verifying and uploading resources...\n" ); |
|
|
|
if( totalsize != 0 ) |
|
{ |
|
Con_DPrintf( "Custom resources total %.2fK\n", totalsize / 1024.0 ); |
|
|
|
if ( ri.info[t_model].size != 0 ) |
|
Con_DPrintf( " Models: %.2fK\n", ri.info[t_model].size / 1024.0 ); |
|
|
|
if ( ri.info[t_sound].size != 0 ) |
|
Con_DPrintf( " Sounds: %.2fK\n", ri.info[t_sound].size / 1024.0 ); |
|
|
|
if ( ri.info[t_decal].size != 0 ) |
|
Con_DPrintf( " Decals: %.2fK\n", ri.info[t_decal].size / 1024.0 ); |
|
|
|
if ( ri.info[t_skin].size != 0 ) |
|
Con_DPrintf( " Skins : %.2fK\n", ri.info[t_skin].size / 1024.0 ); |
|
|
|
if ( ri.info[t_generic].size != 0 ) |
|
Con_DPrintf( " Generic : %.2fK\n", ri.info[t_generic].size / 1024.0 ); |
|
|
|
if ( ri.info[t_eventscript].size != 0 ) |
|
Con_DPrintf( " Events : %.2fK\n", ri.info[t_eventscript].size / 1024.0 ); |
|
|
|
Con_DPrintf( "----------------------\n" ); |
|
} |
|
|
|
totalsize = SV_EstimateNeededResources( cl ); |
|
|
|
if( totalsize > sv_uploadmax.value * 1024 * 1024 ) |
|
{ |
|
SV_ClearResourceList( &cl->resourcesneeded ); |
|
SV_ClearResourceList( &cl->resourcesonhand ); |
|
return; |
|
} |
|
Con_DPrintf( "resources to request: %s\n", Q_memprint( totalsize )); |
|
} |
|
|
|
cl->upstate = us_processing; |
|
SV_BatchUploadRequest( cl ); |
|
} |
|
|
|
/* |
|
=================== |
|
SV_ParseCvarValue |
|
|
|
Parse a requested value from client cvar |
|
=================== |
|
*/ |
|
static void SV_ParseCvarValue( sv_client_t *cl, sizebuf_t *msg ) |
|
{ |
|
const char *value = MSG_ReadString( msg ); |
|
|
|
if( svgame.dllFuncs2.pfnCvarValue != NULL ) |
|
svgame.dllFuncs2.pfnCvarValue( cl->edict, value ); |
|
Con_Reportf( "Cvar query response: name:%s, value:%s\n", cl->name, value ); |
|
} |
|
|
|
/* |
|
=================== |
|
SV_ParseCvarValue2 |
|
|
|
Parse a requested value from client cvar |
|
=================== |
|
*/ |
|
static void SV_ParseCvarValue2( sv_client_t *cl, sizebuf_t *msg ) |
|
{ |
|
string name, value; |
|
int requestID = MSG_ReadLong( msg ); |
|
|
|
Q_strncpy( name, MSG_ReadString( msg ), sizeof( name )); |
|
Q_strncpy( value, MSG_ReadString( msg ), sizeof( value )); |
|
|
|
if( svgame.dllFuncs2.pfnCvarValue2 != NULL ) |
|
svgame.dllFuncs2.pfnCvarValue2( cl->edict, requestID, name, value ); |
|
Con_Reportf( "Cvar query response: name:%s, request ID %d, cvar:%s, value:%s\n", cl->name, requestID, name, value ); |
|
} |
|
|
|
/* |
|
=================== |
|
SV_ParseVoiceData |
|
=================== |
|
*/ |
|
static void SV_ParseVoiceData( sv_client_t *cl, sizebuf_t *msg ) |
|
{ |
|
char received[4096]; |
|
sv_client_t *cur; |
|
int i, client; |
|
uint length, size, frames; |
|
|
|
cl->m_bLoopback = MSG_ReadByte( msg ); |
|
|
|
frames = MSG_ReadByte( msg ); |
|
|
|
size = MSG_ReadShort( msg ); |
|
client = cl - svs.clients; |
|
|
|
if( size > sizeof( received )) |
|
{ |
|
Con_DPrintf( "SV_ParseVoiceData: invalid incoming packet.\n" ); |
|
SV_DropClient( cl, false ); |
|
return; |
|
} |
|
|
|
MSG_ReadBytes( msg, received, size ); |
|
|
|
if( !sv_voiceenable.value ) |
|
return; |
|
|
|
for( i = 0, cur = svs.clients; i < svs.maxclients; i++, cur++ ) |
|
{ |
|
if( cl != cur ) |
|
{ |
|
if( cur->state < cs_connected ) |
|
continue; |
|
|
|
if( !FBitSet( cur->listeners, BIT( client ))) |
|
continue; |
|
} |
|
|
|
length = size; |
|
|
|
// 6 is a number of bytes for other parts of message |
|
if( MSG_GetNumBytesLeft( &cur->datagram ) < length + 6 ) |
|
continue; |
|
|
|
if( cl == cur && !cur->m_bLoopback ) |
|
length = 0; |
|
|
|
MSG_BeginServerCmd( &cur->datagram, svc_voicedata ); |
|
MSG_WriteByte( &cur->datagram, client ); |
|
MSG_WriteByte( &cur->datagram, frames ); |
|
MSG_WriteShort( &cur->datagram, length ); |
|
MSG_WriteBytes( &cur->datagram, received, length ); |
|
} |
|
} |
|
|
|
/* |
|
=================== |
|
SV_ExecuteClientMessage |
|
|
|
Parse a client packet |
|
=================== |
|
*/ |
|
void SV_ExecuteClientMessage( sv_client_t *cl, sizebuf_t *msg ) |
|
{ |
|
qboolean move_issued = false; |
|
client_frame_t *frame; |
|
int c; |
|
|
|
ASSERT( cl->frames != NULL ); |
|
|
|
// calc ping time |
|
frame = &cl->frames[cl->netchan.incoming_acknowledged & SV_UPDATE_MASK]; |
|
|
|
// ping time doesn't factor in message interval, either |
|
frame->ping_time = host.realtime - frame->senttime - cl->cl_updaterate; |
|
|
|
// on first frame ( no senttime ) don't skew ping |
|
if( frame->senttime == 0.0f ) frame->ping_time = 0.0f; |
|
|
|
// don't skew ping based on signon stuff either |
|
if(( host.realtime - cl->connection_started ) < 2.0f && ( frame->ping_time > 0.0f )) |
|
frame->ping_time = 0.0f; |
|
|
|
cl->latency = SV_CalcClientTime( cl ); |
|
cl->delta_sequence = -1; // no delta unless requested |
|
|
|
// read optional clientCommand strings |
|
while( cl->state != cs_zombie ) |
|
{ |
|
if( MSG_CheckOverflow( msg )) |
|
{ |
|
Con_DPrintf( S_ERROR "incoming overflow for %s\n", cl->name ); |
|
SV_DropClient( cl, false ); |
|
return; |
|
} |
|
|
|
// end of message |
|
if( MSG_GetNumBitsLeft( msg ) < 8 ) |
|
break; |
|
|
|
c = MSG_ReadClientCmd( msg ); |
|
|
|
switch( c ) |
|
{ |
|
case clc_nop: |
|
break; |
|
case clc_delta: |
|
cl->delta_sequence = MSG_ReadByte( msg ); |
|
break; |
|
case clc_move: |
|
if( move_issued ) return; // someone is trying to cheat... |
|
move_issued = true; |
|
SV_ParseClientMove( cl, msg ); |
|
break; |
|
case clc_stringcmd: |
|
SV_ExecuteClientCommand( cl, MSG_ReadString( msg )); |
|
if( cl->state == cs_zombie ) |
|
return; // disconnect command |
|
break; |
|
case clc_resourcelist: |
|
SV_ParseResourceList( cl, msg ); |
|
break; |
|
case clc_fileconsistency: |
|
SV_ParseConsistencyResponse( cl, msg ); |
|
break; |
|
case clc_voicedata: |
|
SV_ParseVoiceData( cl, msg ); |
|
break; |
|
case clc_requestcvarvalue: |
|
SV_ParseCvarValue( cl, msg ); |
|
break; |
|
case clc_requestcvarvalue2: |
|
SV_ParseCvarValue2( cl, msg ); |
|
break; |
|
default: |
|
Con_DPrintf( S_ERROR "%s: clc_bad\n", cl->name ); |
|
SV_DropClient( cl, false ); |
|
return; |
|
} |
|
} |
|
}
|
|
|