2024-08-29 19:18:30 -04:00
//===== Copyright © 1996-2005, Valve Corporation, All rights reserved. ======//
//
// Purpose: encapsulates and implements all the accessing of the game dll from external
// sources (only the engine at the time of writing)
// This files ONLY contains functions and data necessary to build an interface
// to external modules
//===========================================================================//
# include "cbase.h"
# include "gamestringpool.h"
# include "mapentities_shared.h"
# include "game.h"
# include "entityapi.h"
# include "client.h"
# include "saverestore.h"
# include "entitylist.h"
# include "gamerules.h"
# include "soundent.h"
# include "player.h"
# include "server_class.h"
# include "ai_node.h"
# include "ai_link.h"
# include "ai_saverestore.h"
# include "ai_networkmanager.h"
# include "ndebugoverlay.h"
# include "ivoiceserver.h"
# include <stdarg.h>
# include "movehelper_server.h"
# include "networkstringtable_gamedll.h"
# include "filesystem.h"
# include "func_areaportalwindow.h"
# include "igamesystem.h"
# include "init_factory.h"
# include "vstdlib/random.h"
# include "env_wind_shared.h"
# include "engine/IEngineSound.h"
# include "ispatialpartition.h"
# include "textstatsmgr.h"
# include "bitbuf.h"
# include "saverestoretypes.h"
# include "physics_saverestore.h"
# include "achievement_saverestore.h"
# include "tier0/vprof.h"
# include "effect_dispatch_data.h"
# include "engine/IStaticPropMgr.h"
# include "TemplateEntities.h"
# include "ai_speech.h"
# include "soundenvelope.h"
# include "usermessages.h"
# include "physics.h"
# include "igameevents.h"
# include "EventLog.h"
# include "datacache/idatacache.h"
# include "engine/ivdebugoverlay.h"
# include "shareddefs.h"
# include "props.h"
# include "timedeventmgr.h"
# include "gameinterface.h"
# include "eventqueue.h"
# include "hltvdirector.h"
# if defined( REPLAY_ENABLED )
# include "replaydirector.h"
# endif
# include "SoundEmitterSystem/isoundemittersystembase.h"
# include "nav_mesh.h"
# include "AI_ResponseSystem.h"
# include "saverestore_stringtable.h"
# include "util.h"
# include "tier0/icommandline.h"
# include "datacache/imdlcache.h"
# include "engine/iserverplugin.h"
# include "env_debughistory.h"
# include "util_shared.h"
# include "player_voice_listener.h"
# ifdef _WIN32
# include "ienginevgui.h"
# include "vgui_gamedll_int.h"
# include "vgui_controls/AnimationController.h"
# endif
# include "ragdoll_shared.h"
# include "toolframework/iserverenginetools.h"
# include "sceneentity.h"
# include "appframework/IAppSystemGroup.h"
# include "scenefilecache/ISceneFileCache.h"
# include "tier2/tier2.h"
# include "particles/particles.h"
# include "GameStats.h"
# include "ixboxsystem.h"
# include "matchmaking/imatchframework.h"
# include "querycache.h"
# include "particle_parse.h"
# include "steam/steam_gameserver.h"
# include "tier3/tier3.h"
# include "serverbenchmark_base.h"
# include "vscript/ivscript.h"
# include "foundryhelpers_server.h"
# include "entity_tools_server.h"
# include "foundry/iserverfoundry.h"
# include "point_template.h"
# include "../../engine/iblackbox.h"
# include "vstdlib/jobthread.h"
# include "vscript_server.h"
# include "tier2/tier2_logging.h"
# include "fmtstr.h"
# ifdef INFESTED_DLL
# include "missionchooser/iasw_mission_chooser.h"
# include "missionchooser/iasw_mission_chooser_source.h"
# include "matchmaking/swarm/imatchext_swarm.h"
# include "asw_gamerules.h"
# endif
# ifdef _WIN32
# include "IGameUIFuncs.h"
# endif
extern IToolFrameworkServer * g_pToolFrameworkServer ;
extern IParticleSystemQuery * g_pParticleSystemQuery ;
extern ConVar commentary ;
// this context is not available on dedicated servers
// WARNING! always check if interfaces are available before using
# if !defined(NO_STEAM)
static CSteamAPIContext s_SteamAPIContext ;
CSteamAPIContext * steamapicontext = & s_SteamAPIContext ;
// this context is not available on a pure client connected to a remote server.
// WARNING! always check if interfaces are available before using
static CSteamGameServerAPIContext s_SteamGameServerAPIContext ;
CSteamGameServerAPIContext * steamgameserverapicontext = & s_SteamGameServerAPIContext ;
# endif
IUploadGameStats * gamestatsuploader = NULL ;
// memdbgon must be the last include file in a .cpp file!!!
# include "tier0/memdbgon.h"
CTimedEventMgr g_NetworkPropertyEventMgr ;
ISaveRestoreBlockHandler * GetEventQueueSaveRestoreBlockHandler ( ) ;
ISaveRestoreBlockHandler * GetCommentarySaveRestoreBlockHandler ( ) ;
CUtlLinkedList < CMapEntityRef , unsigned short > g_MapEntityRefs ;
// Engine interfaces.
IVEngineServer * engine = NULL ;
IVoiceServer * g_pVoiceServer = NULL ;
# if !defined(_STATIC_LINKED)
IFileSystem * filesystem = NULL ;
# else
extern IFileSystem * filesystem ;
# endif
INetworkStringTableContainer * networkstringtable = NULL ;
IStaticPropMgrServer * staticpropmgr = NULL ;
IUniformRandomStream * random = NULL ;
IEngineSound * enginesound = NULL ;
ISpatialPartition * partition = NULL ;
IVModelInfo * modelinfo = NULL ;
IEngineTrace * enginetrace = NULL ;
IFileLoggingListener * filelogginglistener = NULL ;
IGameEventManager2 * gameeventmanager = NULL ;
IDataCache * datacache = NULL ;
IVDebugOverlay * debugoverlay = NULL ;
ISoundEmitterSystemBase * soundemitterbase = NULL ;
IServerPluginHelpers * serverpluginhelpers = NULL ;
# ifdef SERVER_USES_VGUI
IEngineVGui * enginevgui = NULL ;
# endif // SERVER_USES_VGUI
IServerEngineTools * serverenginetools = NULL ;
IServerFoundry * serverfoundry = NULL ;
ISceneFileCache * scenefilecache = NULL ;
# ifdef SERVER_USES_VGUI
IGameUIFuncs * gameuifuncs = NULL ;
# endif // SERVER_USES_VGUI
IXboxSystem * xboxsystem = NULL ; // Xbox 360 only
IScriptManager * scriptmanager = NULL ;
IBlackBox * blackboxrecorder = NULL ;
# ifdef INFESTED_DLL
IASW_Mission_Chooser * missionchooser = NULL ;
IMatchExtSwarm * g_pMatchExtSwarm = NULL ;
# endif
IGameSystem * SoundEmitterSystem ( ) ;
void SoundSystemPreloadSounds ( void ) ;
bool ModelSoundsCacheInit ( ) ;
void ModelSoundsCacheShutdown ( ) ;
void SceneManager_ClientActive ( CBasePlayer * player ) ;
class IMaterialSystem ;
class IStudioRender ;
# ifdef _DEBUG
static ConVar s_UseNetworkVars ( " UseNetworkVars " , " 1 " , FCVAR_CHEAT , " For profiling, toggle network vars. " ) ;
# endif
extern ConVar sv_noclipduringpause ;
ConVar sv_massreport ( " sv_massreport " , " 0 " ) ;
ConVar sv_force_transmit_ents ( " sv_force_transmit_ents " , " 0 " , FCVAR_CHEAT | FCVAR_DEVELOPMENTONLY , " Will transmit all entities to client, regardless of PVS conditions (will still skip based on transmit flags, however) . " ) ;
ConVar sv_autosave ( " sv_autosave " , " 1 " , 0 , " Set to 1 to autosave game on level transition. Does not affect autosave triggers. " ) ;
ConVar * sv_maxreplay = NULL ;
static ConVar * g_pcv_commentary = NULL ;
static ConVar * g_pcv_ThreadMode = NULL ;
# if !defined(NO_STEAM)
//-----------------------------------------------------------------------------
// Purpose: singleton accessor
//-----------------------------------------------------------------------------
static CSteam3Server s_Steam3Server ;
CSteam3Server & Steam3Server ( )
{
return s_Steam3Server ;
}
//-----------------------------------------------------------------------------
// Purpose: Constructor
//-----------------------------------------------------------------------------
CSteam3Server : : CSteam3Server ( )
{
m_bInitialized = false ;
}
# endif
// String tables
INetworkStringTable * g_pStringTableParticleEffectNames = NULL ;
INetworkStringTable * g_pStringTableEffectDispatch = NULL ;
INetworkStringTable * g_pStringTableVguiScreen = NULL ;
INetworkStringTable * g_pStringTableMaterials = NULL ;
INetworkStringTable * g_pStringTableInfoPanel = NULL ;
INetworkStringTable * g_pStringTableClientSideChoreoScenes = NULL ;
INetworkStringTable * g_pStringTableExtraParticleFiles = NULL ;
CStringTableSaveRestoreOps g_VguiScreenStringOps ;
// Holds global variables shared between engine and game.
CGlobalVars * gpGlobals ;
static int g_nCommandClientIndex = 0 ;
// The chapter number of the current
static int g_nCurrentChapterIndex = - 1 ;
static ConVar sv_showhitboxes ( " sv_showhitboxes " , " -1 " , FCVAR_CHEAT , " Send server-side hitboxes for specified entity to client (NOTE: this uses lots of bandwidth, use on listen server only) . " ) ;
static ClientPutInServerOverrideFn g_pClientPutInServerOverride = NULL ;
static void UpdateChapterRestrictions ( const char * mapname ) ;
CSharedEdictChangeInfo * g_pSharedChangeInfo = NULL ;
IChangeInfoAccessor * CBaseEdict : : GetChangeAccessor ( )
{
return engine - > GetChangeAccessor ( ( const edict_t * ) this ) ;
}
const IChangeInfoAccessor * CBaseEdict : : GetChangeAccessor ( ) const
{
return engine - > GetChangeAccessor ( ( const edict_t * ) this ) ;
}
const char * GetHintTypeDescription ( CAI_Hint * pHint ) ;
void ClientPutInServerOverride ( ClientPutInServerOverrideFn fn )
{
g_pClientPutInServerOverride = fn ;
}
ConVar ai_post_frame_navigation ( " ai_post_frame_navigation " , " 0 " ) ;
class CPostFrameNavigationHook ;
extern CPostFrameNavigationHook * PostFrameNavigationSystem ( void ) ;
static bool g_bHeadTrackingEnabled = false ;
bool IsHeadTrackingEnabled ( )
{
# if defined( HL2_DLL )
return g_bHeadTrackingEnabled ;
# else
return false ;
# endif
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : int
//-----------------------------------------------------------------------------
int UTIL_GetCommandClientIndex ( void )
{
// -1 == unknown,dedicated server console
// 0 == player 1
// Convert to 1 based offset
return ( g_nCommandClientIndex + 1 ) ;
}
//-----------------------------------------------------------------------------
// Purpose:
// Output : CBasePlayer
//-----------------------------------------------------------------------------
CBasePlayer * UTIL_GetCommandClient ( void )
{
int idx = UTIL_GetCommandClientIndex ( ) ;
if ( idx > 0 )
{
return UTIL_PlayerByIndex ( idx ) ;
}
// HLDS console issued command
return NULL ;
}
extern void InitializeCvars ( void ) ;
float GetFloorZ ( const Vector & origin ) ;
void UpdateAllClientData ( void ) ;
void DrawMessageEntities ( ) ;
# include "ai_network.h"
// For now just using one big AI network
extern ConVar think_limit ;
#if 0
//-----------------------------------------------------------------------------
// Purpose: Draw output overlays for any measure sections
// Input :
//-----------------------------------------------------------------------------
void DrawMeasuredSections ( void )
{
int row = 1 ;
float rowheight = 0.025 ;
CMeasureSection * p = CMeasureSection : : GetList ( ) ;
while ( p )
{
char str [ 256 ] ;
Q_snprintf ( str , sizeof ( str ) , " %s " , p - > GetName ( ) ) ;
NDebugOverlay : : ScreenText ( 0.01 , 0.51 + ( row * rowheight ) , str , 255 , 255 , 255 , 255 , 0.0 ) ;
Q_snprintf ( str , sizeof ( str ) , " %5.2f \n " , p - > GetTime ( ) . GetMillisecondsF ( ) ) ;
//Q_snprintf(str,sizeof(str),"%3.3f\n",p->GetTime().GetSeconds() * 100.0 / Plat_FloatTime());
NDebugOverlay : : ScreenText ( 0.28 , 0.51 + ( row * rowheight ) , str , 255 , 255 , 255 , 255 , 0.0 ) ;
Q_snprintf ( str , sizeof ( str ) , " %5.2f \n " , p - > GetMaxTime ( ) . GetMillisecondsF ( ) ) ;
//Q_snprintf(str,sizeof(str),"%3.3f\n",p->GetTime().GetSeconds() * 100.0 / Plat_FloatTime());
NDebugOverlay : : ScreenText ( 0.34 , 0.51 + ( row * rowheight ) , str , 255 , 255 , 255 , 255 , 0.0 ) ;
row + + ;
p = p - > GetNext ( ) ;
}
bool sort_reset = false ;
// Time to redo sort?
if ( measure_resort . GetFloat ( ) > 0.0 & &
Plat_FloatTime ( ) > = CMeasureSection : : m_dNextResort )
{
// Redo it
CMeasureSection : : SortSections ( ) ;
// Set next time
CMeasureSection : : m_dNextResort = Plat_FloatTime ( ) + measure_resort . GetFloat ( ) ;
// Flag to reset sort accumulator, too
sort_reset = true ;
}
// Iterate through the sections now
p = CMeasureSection : : GetList ( ) ;
while ( p )
{
// Update max
p - > UpdateMax ( ) ;
// Reset regular accum.
p - > Reset ( ) ;
// Reset sort accum less often
if ( sort_reset )
{
p - > SortReset ( ) ;
}
p = p - > GetNext ( ) ;
}
}
# endif
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void DrawAllDebugOverlays ( void )
{
NDebugOverlay : : PurgeTextOverlays ( ) ;
// If in debug select mode print the selection entities name or classname
if ( CBaseEntity : : m_bInDebugSelect )
{
CBasePlayer * pPlayer = UTIL_PlayerByIndex ( CBaseEntity : : m_nDebugPlayer ) ;
if ( pPlayer )
{
// First try to trace a hull to an entity
CBaseEntity * pEntity = pPlayer - > FindPickerEntity ( ) ;
if ( pEntity )
{
pEntity - > DrawDebugTextOverlays ( ) ;
pEntity - > DrawBBoxOverlay ( ) ;
pEntity - > SendDebugPivotOverlay ( ) ;
}
}
}
// --------------------------------------------------------
// Draw debug overlay lines
// --------------------------------------------------------
UTIL_DrawOverlayLines ( ) ;
// ------------------------------------------------------------------------
// If in wc_edit mode draw a box to highlight which node I'm looking at
// ------------------------------------------------------------------------
if ( engine - > IsInEditMode ( ) )
{
CBasePlayer * pPlayer = UTIL_PlayerByIndex ( CBaseEntity : : m_nDebugPlayer ) ;
if ( pPlayer )
{
if ( g_pAINetworkManager - > GetEditOps ( ) - > m_bLinkEditMode )
{
CAI_Link * pAILink = pPlayer - > FindPickerAILink ( ) ;
if ( pAILink )
{
// For now just using one big AI network
Vector startPos = g_pBigAINet - > GetNode ( pAILink - > m_iSrcID ) - > GetPosition ( g_pAINetworkManager - > GetEditOps ( ) - > m_iHullDrawNum ) ;
Vector endPos = g_pBigAINet - > GetNode ( pAILink - > m_iDestID ) - > GetPosition ( g_pAINetworkManager - > GetEditOps ( ) - > m_iHullDrawNum ) ;
Vector linkDir = startPos - endPos ;
float linkLen = VectorNormalize ( linkDir ) ;
// Draw in green if link that's been turned off
if ( pAILink - > m_LinkInfo & bits_LINK_OFF )
{
NDebugOverlay : : BoxDirection ( startPos , Vector ( - 4 , - 4 , - 4 ) , Vector ( - linkLen , 4 , 4 ) , linkDir , 0 , 255 , 0 , 40 , 0 ) ;
}
// Draw in a pukey yellowish green if link that's "bashable", which is off to everyone but a special door basher
else if ( pAILink - > m_LinkInfo & bits_LINK_ASW_BASHABLE )
{
NDebugOverlay : : BoxDirection ( startPos , Vector ( - 4 , - 4 , - 4 ) , Vector ( - linkLen , 4 , 4 ) , linkDir , 40 , 255 , 0 , 40 , 0 ) ;
}
else
{
NDebugOverlay : : BoxDirection ( startPos , Vector ( - 4 , - 4 , - 4 ) , Vector ( - linkLen , 4 , 4 ) , linkDir , 255 , 0 , 0 , 40 , 0 ) ;
}
}
}
else
{
CAI_Node * pAINode ;
if ( g_pAINetworkManager - > GetEditOps ( ) - > m_bAirEditMode )
{
pAINode = pPlayer - > FindPickerAINode ( NODE_AIR ) ;
}
else
{
pAINode = pPlayer - > FindPickerAINode ( NODE_GROUND ) ;
}
if ( pAINode )
{
Vector vecPos = pAINode - > GetPosition ( g_pAINetworkManager - > GetEditOps ( ) - > m_iHullDrawNum ) ;
NDebugOverlay : : Box ( vecPos , Vector ( - 8 , - 8 , - 8 ) , Vector ( 8 , 8 , 8 ) , 255 , 0 , 0 , 40 , 0 ) ;
if ( pAINode - > GetHint ( ) )
{
CBaseEntity * pEnt = ( CBaseEntity * ) pAINode - > GetHint ( ) ;
if ( pEnt - > GetEntityName ( ) ! = NULL_STRING )
{
NDebugOverlay : : Text ( vecPos + Vector ( 0 , 0 , 6 ) , STRING ( pEnt - > GetEntityName ( ) ) , false , 0 ) ;
}
NDebugOverlay : : Text ( vecPos , GetHintTypeDescription ( pAINode - > GetHint ( ) ) , false , 0 ) ;
}
}
}
// ------------------------------------
// If in air edit mode draw guide line
// ------------------------------------
if ( g_pAINetworkManager - > GetEditOps ( ) - > m_bAirEditMode )
{
UTIL_DrawPositioningOverlay ( g_pAINetworkManager - > GetEditOps ( ) - > m_flAirEditDistance ) ;
}
else
{
NDebugOverlay : : DrawGroundCrossHairOverlay ( ) ;
}
}
}
// For not just using one big AI Network
if ( g_pAINetworkManager )
{
g_pAINetworkManager - > GetEditOps ( ) - > DrawAINetworkOverlay ( ) ;
}
// PERFORMANCE: only do this in developer mode
if ( g_pDeveloper - > GetInt ( ) )
{
// iterate through all objects for debug overlays
const CEntInfo * pInfo = gEntList . FirstEntInfo ( ) ;
for ( ; pInfo ; pInfo = pInfo - > m_pNext )
{
CBaseEntity * ent = ( CBaseEntity * ) pInfo - > m_pEntity ;
// HACKHACK: to flag off these calls
if ( ent - > m_debugOverlays | | ent - > m_pTimedOverlay )
{
MDLCACHE_CRITICAL_SECTION ( ) ;
ent - > DrawDebugGeometryOverlays ( ) ;
}
}
}
if ( sv_massreport . GetInt ( ) )
{
// iterate through all objects for debug overlays
const CEntInfo * pInfo = gEntList . FirstEntInfo ( ) ;
for ( ; pInfo ; pInfo = pInfo - > m_pNext )
{
CBaseEntity * ent = ( CBaseEntity * ) pInfo - > m_pEntity ;
if ( ! ent - > VPhysicsGetObject ( ) )
continue ;
char tempstr [ 512 ] ;
Q_snprintf ( tempstr , sizeof ( tempstr ) , " %s: Mass: %.2f kg / %.2f lb (%s) " ,
STRING ( ent - > GetModelName ( ) ) , ent - > VPhysicsGetObject ( ) - > GetMass ( ) ,
kg2lbs ( ent - > VPhysicsGetObject ( ) - > GetMass ( ) ) ,
GetMassEquivalent ( ent - > VPhysicsGetObject ( ) - > GetMass ( ) ) ) ;
ent - > EntityText ( 0 , tempstr , 0 ) ;
}
}
// A hack to draw point_message entities w/o developer required
DrawMessageEntities ( ) ;
}
// enable threading of init functions on x360
static ConVar sv_threaded_init ( " sv_threaded_init " , IsX360 ( ) ? " 1 " : " 0 " ) ;
static bool InitGameSystems ( CreateInterfaceFn appSystemFactory )
{
// The string system must init first + shutdown last
IGameSystem : : Add ( GameStringSystem ( ) ) ;
// Physics must occur before the sound envelope manager
IGameSystem : : Add ( PhysicsGameSystem ( ) ) ;
// Precache system must be next (requires physics game system)
IGameSystem : : Add ( g_pPrecacheRegister ) ;
// Used to service deferred navigation queries for NPCs
IGameSystem : : Add ( ( IGameSystem * ) PostFrameNavigationSystem ( ) ) ;
// Add game log system
IGameSystem : : Add ( GameLogSystem ( ) ) ;
// Add HLTV director
IGameSystem : : Add ( HLTVDirectorSystem ( ) ) ;
# if defined( REPLAY_ENABLED )
// Add Replay director
IGameSystem : : Add ( ReplayDirectorSystem ( ) ) ;
# endif
// Add sound emitter
IGameSystem : : Add ( SoundEmitterSystem ( ) ) ;
# ifdef SERVER_USES_VGUI
// Startup vgui
if ( enginevgui )
{
if ( ! VGui_Startup ( appSystemFactory ) )
return false ;
}
# endif // SERVER_USES_VGUI
// load Mod specific game events ( MUST be before InitAllSystems() so it can pickup the mod specific events)
gameeventmanager - > LoadEventsFromFile ( " resource/ModEvents.res " ) ;
if ( ! IGameSystem : : InitAllSystems ( ) )
return false ;
// Due to dependencies, these are not autogamesystems
if ( ! ModelSoundsCacheInit ( ) )
{
return false ;
}
// Parse the particle manifest file & register the effects within it
// ParseParticleEffects( false );
InvalidateQueryCache ( ) ;
// create the Navigation Mesh interface
TheNavMesh = NavMeshFactory ( ) ;
// init the gamestatsupload connection
gamestatsuploader - > InitConnection ( ) ;
return true ;
}
CServerGameDLL g_ServerGameDLL ;
EXPOSE_SINGLE_INTERFACE_GLOBALVAR ( CServerGameDLL , IServerGameDLL , INTERFACEVERSION_SERVERGAMEDLL , g_ServerGameDLL ) ;
bool CServerGameDLL : : DLLInit ( CreateInterfaceFn appSystemFactory ,
CreateInterfaceFn physicsFactory , CreateInterfaceFn fileSystemFactory ,
CGlobalVars * pGlobals )
{
COM_TimestampedLog ( " ConnectTier1/2/3Libraries - Start " ) ;
ConnectTier1Libraries ( & appSystemFactory , 1 ) ;
ConnectTier2Libraries ( & appSystemFactory , 1 ) ;
ConnectTier3Libraries ( & appSystemFactory , 1 ) ;
COM_TimestampedLog ( " ConnectTier1/2/3Libraries - Finish " ) ;
// Connected in ConnectTier1Libraries
if ( cvar = = NULL )
return false ;
# if !defined( SWDS ) && !defined(NO_STEAM)
SteamAPI_InitSafe ( ) ;
s_SteamAPIContext . Init ( ) ;
# endif
# if !defined(NO_STEAM)
s_SteamGameServerAPIContext . Init ( ) ;
# endif
COM_TimestampedLog ( " Factories - Start " ) ;
// init each (seperated for ease of debugging)
if ( ( engine = ( IVEngineServer * ) appSystemFactory ( INTERFACEVERSION_VENGINESERVER , NULL ) ) = = NULL )
return false ;
if ( ( g_pVoiceServer = ( IVoiceServer * ) appSystemFactory ( INTERFACEVERSION_VOICESERVER , NULL ) ) = = NULL )
return false ;
if ( ( networkstringtable = ( INetworkStringTableContainer * ) appSystemFactory ( INTERFACENAME_NETWORKSTRINGTABLESERVER , NULL ) ) = = NULL )
return false ;
if ( ( staticpropmgr = ( IStaticPropMgrServer * ) appSystemFactory ( INTERFACEVERSION_STATICPROPMGR_SERVER , NULL ) ) = = NULL )
return false ;
if ( ( random = ( IUniformRandomStream * ) appSystemFactory ( VENGINE_SERVER_RANDOM_INTERFACE_VERSION , NULL ) ) = = NULL )
return false ;
if ( ( enginesound = ( IEngineSound * ) appSystemFactory ( IENGINESOUND_SERVER_INTERFACE_VERSION , NULL ) ) = = NULL )
return false ;
if ( ( partition = ( ISpatialPartition * ) appSystemFactory ( INTERFACEVERSION_SPATIALPARTITION , NULL ) ) = = NULL )
return false ;
if ( ( modelinfo = ( IVModelInfo * ) appSystemFactory ( VMODELINFO_SERVER_INTERFACE_VERSION , NULL ) ) = = NULL )
return false ;
if ( ( enginetrace = ( IEngineTrace * ) appSystemFactory ( INTERFACEVERSION_ENGINETRACE_SERVER , NULL ) ) = = NULL )
return false ;
if ( ( filelogginglistener = ( IFileLoggingListener * ) appSystemFactory ( FILELOGGINGLISTENER_INTERFACE_VERSION , NULL ) ) = = NULL )
return false ;
if ( ( filesystem = ( IFileSystem * ) fileSystemFactory ( FILESYSTEM_INTERFACE_VERSION , NULL ) ) = = NULL )
return false ;
if ( ( gameeventmanager = ( IGameEventManager2 * ) appSystemFactory ( INTERFACEVERSION_GAMEEVENTSMANAGER2 , NULL ) ) = = NULL )
return false ;
if ( ( datacache = ( IDataCache * ) appSystemFactory ( DATACACHE_INTERFACE_VERSION , NULL ) ) = = NULL )
return false ;
if ( ( soundemitterbase = ( ISoundEmitterSystemBase * ) appSystemFactory ( SOUNDEMITTERSYSTEM_INTERFACE_VERSION , NULL ) ) = = NULL )
return false ;
if ( ( gamestatsuploader = ( IUploadGameStats * ) appSystemFactory ( INTERFACEVERSION_UPLOADGAMESTATS , NULL ) ) = = NULL )
return false ;
if ( ! mdlcache )
return false ;
if ( ( serverpluginhelpers = ( IServerPluginHelpers * ) appSystemFactory ( INTERFACEVERSION_ISERVERPLUGINHELPERS , NULL ) ) = = NULL )
return false ;
if ( ( scenefilecache = ( ISceneFileCache * ) appSystemFactory ( SCENE_FILE_CACHE_INTERFACE_VERSION , NULL ) ) = = NULL )
return false ;
if ( ( blackboxrecorder = ( IBlackBox * ) appSystemFactory ( BLACKBOX_INTERFACE_VERSION , NULL ) ) = = NULL )
return false ;
if ( ( xboxsystem = ( IXboxSystem * ) appSystemFactory ( XBOXSYSTEM_INTERFACE_VERSION , NULL ) ) = = NULL )
return false ;
if ( ! CommandLine ( ) - > CheckParm ( " -noscripting " ) )
{
scriptmanager = ( IScriptManager * ) appSystemFactory ( VSCRIPT_INTERFACE_VERSION , NULL ) ;
}
# ifdef SERVER_USES_VGUI
// If not running dedicated, grab the engine vgui interface
if ( ! engine - > IsDedicatedServer ( ) )
{
# ifdef _WIN32
if ( ( enginevgui = ( IEngineVGui * ) appSystemFactory ( VENGINE_VGUI_VERSION , NULL ) ) = = NULL )
return false ;
// This interface is optional, and is only valid when running with -tools
serverenginetools = ( IServerEngineTools * ) appSystemFactory ( VSERVERENGINETOOLS_INTERFACE_VERSION , NULL ) ;
gameuifuncs = ( IGameUIFuncs * ) appSystemFactory ( VENGINE_GAMEUIFUNCS_VERSION , NULL ) ;
# endif
}
# endif // SERVER_USES_VGUI
# ifdef INFESTED_DLL
if ( ( missionchooser = ( IASW_Mission_Chooser * ) appSystemFactory ( ASW_MISSION_CHOOSER_VERSION , NULL ) ) = = NULL )
return false ;
if ( ( g_pMatchExtSwarm = ( IMatchExtSwarm * ) appSystemFactory ( IMATCHEXT_SWARM_INTERFACE , NULL ) ) = = NULL )
return false ;
# endif
if ( ! g_pMatchFramework )
return false ;
if ( IMatchExtensions * pIMatchExtensions = g_pMatchFramework - > GetMatchExtensions ( ) )
pIMatchExtensions - > RegisterExtensionInterface (
INTERFACEVERSION_SERVERGAMEDLL , static_cast < IServerGameDLL * > ( this ) ) ;
COM_TimestampedLog ( " Factories - Finish " ) ;
COM_TimestampedLog ( " soundemitterbase->Connect " ) ;
// Yes, both the client and game .dlls will try to Connect, the soundemittersystem.dll will handle this gracefully
if ( ! soundemitterbase - > Connect ( appSystemFactory ) )
return false ;
if ( CommandLine ( ) - > FindParm ( " -headtracking " ) )
g_bHeadTrackingEnabled = true ;
// cache the globals
gpGlobals = pGlobals ;
g_pSharedChangeInfo = engine - > GetSharedEdictChangeInfo ( ) ;
COM_TimestampedLog ( " MathLib_Init " ) ;
MathLib_Init ( 2.2f , 2.2f , 0.0f , 2.0f ) ;
// save these in case other system inits need them
factorylist_t factories ;
factories . engineFactory = appSystemFactory ;
factories . fileSystemFactory = fileSystemFactory ;
factories . physicsFactory = physicsFactory ;
FactoryList_Store ( factories ) ;
COM_TimestampedLog ( " gameeventmanager->LoadEventsFromFile " ) ;
// load used game events
gameeventmanager - > LoadEventsFromFile ( " resource/gameevents.res " ) ;
COM_TimestampedLog ( " InitializeCvars " ) ;
// init the cvar list first in case inits want to reference them
InitializeCvars ( ) ;
COM_TimestampedLog ( " g_pParticleSystemMgr->Init " ) ;
// Initialize the particle system
bool bPrecacheParticles = IsPC ( ) & & ! engine - > IsCreatingXboxReslist ( ) ;
if ( ! g_pParticleSystemMgr - > Init ( g_pParticleSystemQuery , bPrecacheParticles ) )
{
return false ;
}
sv_cheats = g_pCVar - > FindVar ( " sv_cheats " ) ;
if ( ! sv_cheats )
return false ;
g_pcv_commentary = g_pCVar - > FindVar ( " commentary " ) ;
g_pcv_ThreadMode = g_pCVar - > FindVar ( " host_thread_mode " ) ;
sv_maxreplay = g_pCVar - > FindVar ( " sv_maxreplay " ) ;
COM_TimestampedLog ( " g_pGameSaveRestoreBlockSet " ) ;
g_pGameSaveRestoreBlockSet - > AddBlockHandler ( GetEntitySaveRestoreBlockHandler ( ) ) ;
g_pGameSaveRestoreBlockSet - > AddBlockHandler ( GetPhysSaveRestoreBlockHandler ( ) ) ;
g_pGameSaveRestoreBlockSet - > AddBlockHandler ( GetAISaveRestoreBlockHandler ( ) ) ;
g_pGameSaveRestoreBlockSet - > AddBlockHandler ( GetTemplateSaveRestoreBlockHandler ( ) ) ;
g_pGameSaveRestoreBlockSet - > AddBlockHandler ( GetDefaultResponseSystemSaveRestoreBlockHandler ( ) ) ;
g_pGameSaveRestoreBlockSet - > AddBlockHandler ( GetCommentarySaveRestoreBlockHandler ( ) ) ;
g_pGameSaveRestoreBlockSet - > AddBlockHandler ( GetEventQueueSaveRestoreBlockHandler ( ) ) ;
g_pGameSaveRestoreBlockSet - > AddBlockHandler ( GetAchievementSaveRestoreBlockHandler ( ) ) ;
g_pGameSaveRestoreBlockSet - > AddBlockHandler ( GetVScriptSaveRestoreBlockHandler ( ) ) ;
bool bInitSuccess = false ;
if ( sv_threaded_init . GetBool ( ) )
{
CFunctorJob * pGameJob = new CFunctorJob ( CreateFunctor ( ParseParticleEffects , false ) ) ;
g_pThreadPool - > AddJob ( pGameJob ) ;
bInitSuccess = InitGameSystems ( appSystemFactory ) ;
// FIXME: This method is a bit of a hack.
// Try to update the screen every .06 seconds while waiting.
float flLastUpdateTime = - 1.0f ;
while ( ! pGameJob - > IsFinished ( ) )
{
float flTime = Plat_FloatTime ( ) ;
if ( flTime - flLastUpdateTime > .06f )
{
flLastUpdateTime = flTime ;
engine - > RefreshScreenIfNecessary ( ) ;
}
ThreadSleep ( 0 ) ;
}
pGameJob - > Release ( ) ;
}
else
{
COM_TimestampedLog ( " ParseParticleEffects " ) ;
ParseParticleEffects ( false ) ;
COM_TimestampedLog ( " InitGameSystems - Start " ) ;
bInitSuccess = InitGameSystems ( appSystemFactory ) ;
COM_TimestampedLog ( " InitGameSystems - Finish " ) ;
}
// try to get debug overlay, may be NULL if on HLDS
debugoverlay = ( IVDebugOverlay * ) appSystemFactory ( VDEBUG_OVERLAY_INTERFACE_VERSION , NULL ) ;
// init the gamestatsupload connection
gamestatsuploader - > InitConnection ( ) ;
return true ;
}
void CServerGameDLL : : PostInit ( )
{
IGameSystem : : PostInitAllSystems ( ) ;
# ifdef SERVER_USES_VGUI
if ( ! engine - > IsDedicatedServer ( ) & & enginevgui )
{
if ( VGui_PostInit ( ) )
{
// all good
}
}
# endif // SERVER_USES_VGUI
}
void CServerGameDLL : : PostToolsInit ( )
{
if ( serverenginetools )
{
serverfoundry = ( IServerFoundry * ) serverenginetools - > QueryInterface ( VSERVERFOUNDRY_INTERFACE_VERSION ) ;
}
}
void CServerGameDLL : : DLLShutdown ( void )
{
// Due to dependencies, these are not autogamesystems
ModelSoundsCacheShutdown ( ) ;
g_pGameSaveRestoreBlockSet - > RemoveBlockHandler ( GetVScriptSaveRestoreBlockHandler ( ) ) ;
g_pGameSaveRestoreBlockSet - > RemoveBlockHandler ( GetAchievementSaveRestoreBlockHandler ( ) ) ;
g_pGameSaveRestoreBlockSet - > RemoveBlockHandler ( GetCommentarySaveRestoreBlockHandler ( ) ) ;
g_pGameSaveRestoreBlockSet - > RemoveBlockHandler ( GetEventQueueSaveRestoreBlockHandler ( ) ) ;
g_pGameSaveRestoreBlockSet - > RemoveBlockHandler ( GetDefaultResponseSystemSaveRestoreBlockHandler ( ) ) ;
g_pGameSaveRestoreBlockSet - > RemoveBlockHandler ( GetTemplateSaveRestoreBlockHandler ( ) ) ;
g_pGameSaveRestoreBlockSet - > RemoveBlockHandler ( GetAISaveRestoreBlockHandler ( ) ) ;
g_pGameSaveRestoreBlockSet - > RemoveBlockHandler ( GetPhysSaveRestoreBlockHandler ( ) ) ;
g_pGameSaveRestoreBlockSet - > RemoveBlockHandler ( GetEntitySaveRestoreBlockHandler ( ) ) ;
char * pFilename = g_TextStatsMgr . GetStatsFilename ( ) ;
if ( ! pFilename | | ! pFilename [ 0 ] )
{
g_TextStatsMgr . SetStatsFilename ( " stats.txt " ) ;
}
g_TextStatsMgr . WriteFile ( filesystem ) ;
IGameSystem : : ShutdownAllSystems ( ) ;
# ifdef SERVER_USES_VGUI
if ( enginevgui )
{
VGui_Shutdown ( ) ;
}
# endif // SERVER_USES_VGUI
// destroy the Navigation Mesh interface
if ( TheNavMesh )
{
delete TheNavMesh ;
TheNavMesh = NULL ;
}
# if !defined(NO_STEAM)
s_SteamAPIContext . Clear ( ) ; // Steam API context shutdown
s_SteamGameServerAPIContext . Clear ( ) ;
// SteamAPI_Shutdown(); << Steam shutdown is controlled by engine
# endif
DisconnectTier3Libraries ( ) ;
DisconnectTier2Libraries ( ) ;
ConVar_Unregister ( ) ;
DisconnectTier1Libraries ( ) ;
}
//-----------------------------------------------------------------------------
// Purpose: See shareddefs.h for redefining this. Don't even think about it, though, for HL2. Or you will pay. ywb 9/22/03
// Output : float
//-----------------------------------------------------------------------------
float CServerGameDLL : : GetTickInterval ( void ) const
{
float tickinterval = DEFAULT_TICK_INTERVAL ;
// override if tick rate specified in command line
if ( CommandLine ( ) - > CheckParm ( " -tickrate " ) )
{
float tickrate = CommandLine ( ) - > ParmValue ( " -tickrate " , 0 ) ;
if ( tickrate > 10 )
tickinterval = 1.0f / tickrate ;
}
return tickinterval ;
}
// This is called when a new game is started. (restart, map)
bool CServerGameDLL : : GameInit ( void )
{
ResetGlobalState ( ) ;
engine - > ServerCommand ( " exec game.cfg \n " ) ;
engine - > ServerExecute ( ) ;
CBaseEntity : : sm_bAccurateTriggerBboxChecks = true ;
IGameEvent * event = gameeventmanager - > CreateEvent ( " game_init " ) ;
if ( event )
{
gameeventmanager - > FireEvent ( event ) ;
}
return true ;
}
// This is called when a game ends (server disconnect, death, restart, load)
// NOT on level transitions within a game
void CServerGameDLL : : GameShutdown ( void )
{
ResetGlobalState ( ) ;
}
static bool g_OneWayTransition = false ;
void Game_SetOneWayTransition ( void )
{
g_OneWayTransition = true ;
}
static CUtlVector < EHANDLE > g_RestoredEntities ;
// just for debugging, assert that this is the only time this function is called
static bool g_InRestore = false ;
void AddRestoredEntity ( CBaseEntity * pEntity )
{
Assert ( g_InRestore ) ;
if ( ! pEntity )
return ;
g_RestoredEntities . AddToTail ( EHANDLE ( pEntity ) ) ;
}
void EndRestoreEntities ( )
{
if ( ! g_InRestore )
return ;
// The entire hierarchy is restored, so we can call GetAbsOrigin again.
//CBaseEntity::SetAbsQueriesValid( true );
// Call all entities' OnRestore handlers
for ( int i = g_RestoredEntities . Count ( ) - 1 ; i > = 0 ; - - i )
{
CBaseEntity * pEntity = g_RestoredEntities [ i ] . Get ( ) ;
if ( pEntity & & ! pEntity - > IsDormant ( ) )
{
MDLCACHE_CRITICAL_SECTION ( ) ;
pEntity - > OnRestore ( ) ;
}
}
g_RestoredEntities . Purge ( ) ;
IGameSystem : : OnRestoreAllSystems ( ) ;
g_InRestore = false ;
gEntList . CleanupDeleteList ( ) ;
// HACKHACK: UNDONE: We need to redesign the main loop with respect to save/load/server activate
g_ServerGameDLL . ServerActivate ( NULL , 0 , 0 ) ;
CBaseEntity : : SetAllowPrecache ( false ) ;
}
void BeginRestoreEntities ( )
{
if ( g_InRestore )
{
DevMsg ( " BeginRestoreEntities without previous EndRestoreEntities. \n " ) ;
gEntList . CleanupDeleteList ( ) ;
}
g_RestoredEntities . Purge ( ) ;
g_InRestore = true ;
CBaseEntity : : SetAllowPrecache ( true ) ;
// No calls to GetAbsOrigin until the entire hierarchy is restored!
//CBaseEntity::SetAbsQueriesValid( false );
}
//-----------------------------------------------------------------------------
// Purpose: This prevents sv.tickcount/gpGlobals->tickcount from advancing during restore which
// would cause a lot of the NPCs to fast forward their think times to the same
// tick due to some ticks being elapsed during restore where there was no simulation going on
//-----------------------------------------------------------------------------
bool CServerGameDLL : : IsRestoring ( )
{
return g_InRestore ;
}
bool CServerGameDLL : : SupportsSaveRestore ( )
{
# ifdef INFESTED_DLL
return false ;
# endif
return true ;
}
// Called any time a new level is started (after GameInit() also on level transitions within a game)
bool CServerGameDLL : : LevelInit ( const char * pMapName , char const * pMapEntities , char const * pOldLevel , char const * pLandmarkName , bool loadGame , bool background )
{
VPROF ( " CServerGameDLL::LevelInit " ) ;
ResetWindspeed ( ) ;
UpdateChapterRestrictions ( pMapName ) ;
// IGameSystem::LevelInitPreEntityAllSystems() is called when the world is precached
// That happens either in LoadGameState() or in MapEntity_ParseAllEntities()
if ( loadGame )
{
if ( pOldLevel )
{
gpGlobals - > eLoadType = MapLoad_Transition ;
}
else
{
gpGlobals - > eLoadType = MapLoad_LoadGame ;
}
BeginRestoreEntities ( ) ;
if ( ! engine - > LoadGameState ( pMapName , 1 ) )
{
if ( pOldLevel )
{
MapEntity_ParseAllEntities ( pMapEntities ) ;
}
else
{
// Regular save load case
return false ;
}
}
if ( pOldLevel )
{
engine - > LoadAdjacentEnts ( pOldLevel , pLandmarkName ) ;
}
if ( g_OneWayTransition )
{
engine - > ClearSaveDirAfterClientLoad ( ) ;
}
if ( pOldLevel & & sv_autosave . GetBool ( ) = = true & & gpGlobals - > maxClients = = 1 )
{
// This is a single-player style level transition.
// Queue up an autosave one second into the level
CBaseEntity * pAutosave = CBaseEntity : : Create ( " logic_autosave " , vec3_origin , vec3_angle , NULL ) ;
if ( pAutosave )
{
g_EventQueue . AddEvent ( pAutosave , " Save " , 1.0 , NULL , NULL ) ;
g_EventQueue . AddEvent ( pAutosave , " Kill " , 1.1 , NULL , NULL ) ;
}
}
}
else
{
if ( background )
{
gpGlobals - > eLoadType = MapLoad_Background ;
}
else
{
gpGlobals - > eLoadType = MapLoad_NewGame ;
}
// Clear out entity references, and parse the entities into it.
g_MapEntityRefs . Purge ( ) ;
CMapLoadEntityFilter filter ;
MapEntity_ParseAllEntities ( pMapEntities , & filter ) ;
g_pServerBenchmark - > StartBenchmark ( ) ;
// Now call the mod specific parse
LevelInit_ParseAllEntities ( pMapEntities ) ;
}
// Check low violence settings for this map
g_RagdollLVManager . SetLowViolence ( pMapName ) ;
// Now that all of the active entities have been loaded in, precache any entities who need point_template parameters
// to be parsed (the above code has loaded all point_template entities)
PrecachePointTemplates ( ) ;
// load MOTD from file into stringtable
LoadMessageOfTheDay ( ) ;
// Sometimes an ent will Remove() itself during its precache, so RemoveImmediate won't happen.
// This makes sure those ents get cleaned up.
gEntList . CleanupDeleteList ( ) ;
g_AIFriendliesTalkSemaphore . Release ( ) ;
g_AIFoesTalkSemaphore . Release ( ) ;
g_OneWayTransition = false ;
// clear any pending autosavedangerous
m_fAutoSaveDangerousTime = 0.0f ;
m_fAutoSaveDangerousMinHealthToCommit = 0.0f ;
// ask for the latest game rules
GameRules ( ) - > UpdateGameplayStatsFromSteam ( ) ;
return true ;
}
//-----------------------------------------------------------------------------
// Purpose: called after every level change and load game, iterates through all the
// active entities and gives them a chance to fix up their state
//-----------------------------------------------------------------------------
# ifdef DEBUG
bool g_bReceivedChainedActivate ;
bool g_bCheckForChainedActivate ;
# define BeginCheckChainedActivate() if (0) ; else { g_bCheckForChainedActivate = true; g_bReceivedChainedActivate = false; }
# define EndCheckChainedActivate( bCheck ) \
if ( 0 ) ; else \
{ \
if ( bCheck ) \
{ \
char msg [ 1024 ] ; \
Q_snprintf ( msg , sizeof ( msg ) , " Entity (%i/%s/%s) failed to call base class Activate() \n " , pClass - > entindex ( ) , pClass - > GetClassname ( ) , STRING ( pClass - > GetEntityName ( ) ) ) ; \
AssertMsg ( g_bReceivedChainedActivate = = true , msg ) ; \
} \
g_bCheckForChainedActivate = false ; \
}
# else
# define BeginCheckChainedActivate() ((void)0)
# define EndCheckChainedActivate( bCheck ) ((void)0)
# endif
void CServerGameDLL : : ServerActivate ( edict_t * pEdictList , int edictCount , int clientMax )
{
// HACKHACK: UNDONE: We need to redesign the main loop with respect to save/load/server activate
if ( g_InRestore )
return ;
if ( gEntList . ResetDeleteList ( ) ! = 0 )
{
Msg ( " ERROR: Entity delete queue not empty on level start! \n " ) ;
}
for ( CBaseEntity * pClass = gEntList . FirstEnt ( ) ; pClass ! = NULL ; pClass = gEntList . NextEnt ( pClass ) )
{
if ( pClass & & ! pClass - > IsDormant ( ) )
{
MDLCACHE_CRITICAL_SECTION ( ) ;
BeginCheckChainedActivate ( ) ;
pClass - > Activate ( ) ;
// We don't care if it finished activating if it decided to remove itself.
EndCheckChainedActivate ( ! ( pClass - > GetEFlags ( ) & EFL_KILLME ) ) ;
}
}
IGameSystem : : LevelInitPostEntityAllSystems ( ) ;
// No more precaching after PostEntityAllSystems!!!
CBaseEntity : : SetAllowPrecache ( false ) ;
// only display the think limit when the game is run with "developer" mode set
if ( ! g_pDeveloper - > GetInt ( ) )
{
think_limit . SetValue ( 0 ) ;
}
# ifndef _XBOX
// load the Navigation Mesh for this map
TheNavMesh - > Load ( ) ;
# endif
}
//-----------------------------------------------------------------------------
// Purpose: Called after the steam API has been activated post-level startup
//-----------------------------------------------------------------------------
void CServerGameDLL : : GameServerSteamAPIActivated ( void )
{
# if !defined( NO_STEAM )
steamgameserverapicontext - > Init ( ) ;
# endif
}
//-----------------------------------------------------------------------------
// Purpose: Called at the start of every game frame
//-----------------------------------------------------------------------------
ConVar trace_report ( " trace_report " , " 0 " ) ;
void CServerGameDLL : : GameFrame ( bool simulating )
{
VPROF ( " CServerGameDLL::GameFrame " ) ;
// Don't run frames until fully restored
if ( g_InRestore )
return ;
# ifndef NO_STEAM
// All the calls to us from the engine prior to gameframe (like LevelInit & ServerActivate)
// are done before the engine has got the Steam API connected, so we have to wait until now to connect ourselves.
if ( Steam3Server ( ) . CheckInitialized ( ) )
{
GameRules ( ) - > UpdateGameplayStatsFromSteam ( ) ;
}
# endif
g_bIsLogging = engine - > IsLogEnabled ( ) ;
if ( CBaseEntity : : IsSimulatingOnAlternateTicks ( ) )
{
// only run simulation on even numbered ticks
if ( gpGlobals - > tickcount & 1 )
{
UpdateAllClientData ( ) ;
return ;
}
// If we're skipping frames, then the frametime is 2x the normal tick
gpGlobals - > frametime * = 2.0f ;
}
float oldframetime = gpGlobals - > frametime ;
# ifdef _DEBUG
// For profiling.. let them enable/disable the networkvar manual mode stuff.
g_bUseNetworkVars = s_UseNetworkVars . GetBool ( ) ;
# endif
extern void GameStartFrame ( void ) ;
extern void ServiceEventQueue ( void ) ;
extern void Physics_RunThinkFunctions ( bool simulating ) ;
// Delete anything that was marked for deletion
// outside of server frameloop (e.g., in response to concommand)
gEntList . CleanupDeleteList ( ) ;
HandleFoundryEntitySpawnRecords ( ) ;
IGameSystem : : FrameUpdatePreEntityThinkAllSystems ( ) ;
GameStartFrame ( ) ;
TheNavMesh - > Update ( ) ;
{
VPROF ( " gamestatsuploader->UpdateConnection " ) ;
gamestatsuploader - > UpdateConnection ( ) ;
}
{
VPROF ( " UpdateQueryCache " ) ;
UpdateQueryCache ( ) ;
}
{
VPROF ( " g_pServerBenchmark->UpdateBenchmark " ) ;
g_pServerBenchmark - > UpdateBenchmark ( ) ;
}
Physics_RunThinkFunctions ( simulating ) ;
IGameSystem : : FrameUpdatePostEntityThinkAllSystems ( ) ;
// UNDONE: Make these systems IGameSystems and move these calls into FrameUpdatePostEntityThink()
// service event queue, firing off any actions whos time has come
ServiceEventQueue ( ) ;
// free all ents marked in think functions
gEntList . CleanupDeleteList ( ) ;
// FIXME: Should this only occur on the final tick?
UpdateAllClientData ( ) ;
if ( g_pGameRules )
{
VPROF ( " g_pGameRules->EndGameFrame " ) ;
g_pGameRules - > EndGameFrame ( ) ;
}
if ( trace_report . GetBool ( ) )
{
int total = 0 , totals [ 3 ] ;
for ( int i = 0 ; i < 3 ; i + + )
{
totals [ i ] = enginetrace - > GetStatByIndex ( i , true ) ;
if ( totals [ i ] > 0 )
{
total + = totals [ i ] ;
}
}
if ( total )
{
Msg ( " Trace: %d, contents %d, enumerate %d \n " , totals [ 0 ] , totals [ 1 ] , totals [ 2 ] ) ;
}
}
// Any entities that detect network state changes on a timer do it here.
g_NetworkPropertyEventMgr . FireEvents ( ) ;
gpGlobals - > frametime = oldframetime ;
}
//-----------------------------------------------------------------------------
// Purpose: Called every frame even if not ticking
// Input : simulating -
//-----------------------------------------------------------------------------
void CServerGameDLL : : PreClientUpdate ( bool simulating )
{
if ( ! simulating )
return ;
/*
if ( game_speeds . GetInt ( ) )
{
DrawMeasuredSections ( ) ;
}
*/
//#ifdef _DEBUG - allow this in release for now
DrawAllDebugOverlays ( ) ;
//#endif
IGameSystem : : PreClientUpdateAllSystems ( ) ;
if ( sv_showhitboxes . GetInt ( ) = = - 1 )
return ;
if ( sv_showhitboxes . GetInt ( ) = = 0 )
{
// assume it's text
CBaseEntity * pEntity = NULL ;
while ( 1 )
{
pEntity = gEntList . FindEntityByName ( pEntity , sv_showhitboxes . GetString ( ) ) ;
if ( ! pEntity )
break ;
CBaseAnimating * anim = dynamic_cast < CBaseAnimating * > ( pEntity ) ;
if ( anim )
{
anim - > DrawServerHitboxes ( ) ;
}
}
return ;
}
CBaseAnimating * anim = dynamic_cast < CBaseAnimating * > ( CBaseEntity : : Instance ( INDEXENT ( sv_showhitboxes . GetInt ( ) ) ) ) ;
if ( ! anim )
return ;
anim - > DrawServerHitboxes ( ) ;
}
void CServerGameDLL : : Think ( bool finalTick )
{
if ( m_fAutoSaveDangerousTime ! = 0.0f & & m_fAutoSaveDangerousTime < gpGlobals - > curtime )
{
// The safety timer for a dangerous auto save has expired
CBasePlayer * pPlayer = UTIL_PlayerByIndex ( 1 ) ;
if ( pPlayer & & ( pPlayer - > GetDeathTime ( ) = = 0.0f | | pPlayer - > GetDeathTime ( ) > gpGlobals - > curtime )
& & ! pPlayer - > IsSinglePlayerGameEnding ( )
)
{
if ( pPlayer - > GetHealth ( ) > = m_fAutoSaveDangerousMinHealthToCommit )
{
// The player isn't dead, so make the dangerous auto save safe
engine - > ServerCommand ( " autosavedangerousissafe \n " ) ;
}
}
m_fAutoSaveDangerousTime = 0.0f ;
m_fAutoSaveDangerousMinHealthToCommit = 0.0f ;
}
}
void CServerGameDLL : : OnQueryCvarValueFinished ( QueryCvarCookie_t iCookie , edict_t * pPlayerEntity , EQueryCvarValueStatus eStatus , const char * pCvarName , const char * pCvarValue )
{
}
// Called when a level is shutdown (including changing levels)
void CServerGameDLL : : LevelShutdown ( void )
{
# if !defined( NO_STEAM )
steamgameserverapicontext - > Clear ( ) ;
# endif
MDLCACHE_CRITICAL_SECTION ( ) ;
IGameSystem : : LevelShutdownPreEntityAllSystems ( ) ;
// YWB:
// This entity pointer is going away now and is corrupting memory on level transitions/restarts
CSoundEnt : : ShutdownSoundEnt ( ) ;
ClearDebugHistory ( ) ;
gEntList . Clear ( ) ;
InvalidateQueryCache ( ) ;
IGameSystem : : LevelShutdownPostEntityAllSystems ( ) ;
// In case we quit out during initial load
CBaseEntity : : SetAllowPrecache ( false ) ;
TheNavMesh - > Reset ( ) ;
g_nCurrentChapterIndex = - 1 ;
CStudioHdr : : CActivityToSequenceMapping : : ResetMappings ( ) ;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input :
// Output : ServerClass*
//-----------------------------------------------------------------------------
ServerClass * CServerGameDLL : : GetAllServerClasses ( )
{
return g_pServerClassHead ;
}
const char * CServerGameDLL : : GetGameDescription ( void )
{
return : : GetGameDescription ( ) ;
}
void CServerGameDLL : : CreateNetworkStringTables ( void )
{
// Create any shared string tables here (and only here!)
// E.g.: xxx = networkstringtable->CreateStringTable( "SceneStrings", 512 );
g_pStringTableExtraParticleFiles = networkstringtable - > CreateStringTable ( " ExtraParticleFilesTable " , MAX_PARTICLESYSTEMS_STRINGS , 0 , 0 , NSF_DICTIONARY_ENABLED ) ;
g_pStringTableParticleEffectNames = networkstringtable - > CreateStringTable ( " ParticleEffectNames " , MAX_PARTICLESYSTEMS_STRINGS , 0 , 0 , NSF_DICTIONARY_ENABLED ) ;
g_pStringTableEffectDispatch = networkstringtable - > CreateStringTable ( " EffectDispatch " , MAX_EFFECT_DISPATCH_STRINGS ) ;
g_pStringTableVguiScreen = networkstringtable - > CreateStringTable ( " VguiScreen " , MAX_VGUI_SCREEN_STRINGS ) ;
g_pStringTableMaterials = networkstringtable - > CreateStringTable ( " Materials " , MAX_MATERIAL_STRINGS , 0 , 0 , NSF_DICTIONARY_ENABLED ) ;
g_pStringTableInfoPanel = networkstringtable - > CreateStringTable ( " InfoPanel " , MAX_INFOPANEL_STRINGS ) ;
g_pStringTableClientSideChoreoScenes = networkstringtable - > CreateStringTable ( " Scenes " , MAX_CHOREO_SCENES_STRINGS , 0 , 0 , NSF_DICTIONARY_ENABLED ) ;
Assert ( g_pStringTableParticleEffectNames & &
g_pStringTableEffectDispatch & &
g_pStringTableVguiScreen & &
g_pStringTableMaterials & &
g_pStringTableInfoPanel & &
g_pStringTableClientSideChoreoScenes & &
g_pStringTableExtraParticleFiles ) ;
// Need this so we have the error material always handy
PrecacheMaterial ( " debug/debugempty " ) ;
Assert ( GetMaterialIndex ( " debug/debugempty " ) = = 0 ) ;
PrecacheParticleSystem ( " error " ) ; // ensure error particle system is handy
Assert ( GetParticleSystemIndex ( " error " ) = = 0 ) ;
PrecacheEffect ( " error " ) ; // ensure error effect is handy
Assert ( GetEffectIndex ( " error " ) = = 0 ) ;
CreateNetworkStringTables_GameRules ( ) ;
// Set up save/load utilities for string tables
g_VguiScreenStringOps . Init ( g_pStringTableVguiScreen ) ;
}
CSaveRestoreData * CServerGameDLL : : SaveInit ( int size )
{
return : : SaveInit ( size ) ;
}
//-----------------------------------------------------------------------------
// Purpose: Saves data from a struct into a saverestore object, to be saved to disk
// Input : *pSaveData - the saverestore object
// char *pname - the name of the data to write
// *pBaseData - the struct into which the data is to be read
// *pFields - pointer to an array of data field descriptions
// fieldCount - the size of the array (number of field descriptions)
//-----------------------------------------------------------------------------
void CServerGameDLL : : SaveWriteFields ( CSaveRestoreData * pSaveData , const char * pname , void * pBaseData , datamap_t * pMap , typedescription_t * pFields , int fieldCount )
{
CSave saveHelper ( pSaveData ) ;
saveHelper . WriteFields ( pname , pBaseData , pMap , pFields , fieldCount ) ;
}
//-----------------------------------------------------------------------------
// Purpose: Reads data from a save/restore block into a structure
// Input : *pSaveData - the saverestore object
// char *pname - the name of the data to extract from
// *pBaseData - the struct into which the data is to be restored
// *pFields - pointer to an array of data field descriptions
// fieldCount - the size of the array (number of field descriptions)
//-----------------------------------------------------------------------------
//-----------------------------------------------------------------------------
void CServerGameDLL : : SaveReadFields ( CSaveRestoreData * pSaveData , const char * pname , void * pBaseData , datamap_t * pMap , typedescription_t * pFields , int fieldCount )
{
CRestore restoreHelper ( pSaveData ) ;
restoreHelper . ReadFields ( pname , pBaseData , pMap , pFields , fieldCount ) ;
}
//-----------------------------------------------------------------------------
void CServerGameDLL : : SaveGlobalState ( CSaveRestoreData * s )
{
: : SaveGlobalState ( s ) ;
}
void CServerGameDLL : : RestoreGlobalState ( CSaveRestoreData * s )
{
: : RestoreGlobalState ( s ) ;
}
void CServerGameDLL : : Save ( CSaveRestoreData * s )
{
CSave saveHelper ( s ) ;
g_pGameSaveRestoreBlockSet - > Save ( & saveHelper ) ;
}
void CServerGameDLL : : Restore ( CSaveRestoreData * s , bool b )
{
if ( engine - > IsOverrideLoadGameEntsOn ( ) )
FoundryHelpers_ClearEntityHighlightEffects ( ) ;
CRestore restore ( s ) ;
g_pGameSaveRestoreBlockSet - > Restore ( & restore , b ) ;
g_pGameSaveRestoreBlockSet - > PostRestore ( ) ;
if ( serverfoundry & & engine - > IsOverrideLoadGameEntsOn ( ) )
serverfoundry - > OnFinishedRestoreSavegame ( ) ;
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : msg_type -
// *name -
// size -
// Output : Returns true on success, false on failure.
//-----------------------------------------------------------------------------
bool CServerGameDLL : : GetUserMessageInfo ( int msg_type , char * name , int maxnamelength , int & size )
{
if ( ! usermessages - > IsValidIndex ( msg_type ) )
return false ;
Q_strncpy ( name , usermessages - > GetUserMessageName ( msg_type ) , maxnamelength ) ;
size = usermessages - > GetUserMessageSize ( msg_type ) ;
return true ;
}
CStandardSendProxies * CServerGameDLL : : GetStandardSendProxies ( )
{
return & g_StandardSendProxies ;
}
int CServerGameDLL : : CreateEntityTransitionList ( CSaveRestoreData * s , int a )
{
CRestore restoreHelper ( s ) ;
// save off file base
int base = restoreHelper . GetReadPos ( ) ;
int movedCount = : : CreateEntityTransitionList ( s , a ) ;
if ( movedCount )
{
g_pGameSaveRestoreBlockSet - > CallBlockHandlerRestore ( GetPhysSaveRestoreBlockHandler ( ) , base , & restoreHelper , false ) ;
g_pGameSaveRestoreBlockSet - > CallBlockHandlerRestore ( GetAISaveRestoreBlockHandler ( ) , base , & restoreHelper , false ) ;
}
GetPhysSaveRestoreBlockHandler ( ) - > PostRestore ( ) ;
GetAISaveRestoreBlockHandler ( ) - > PostRestore ( ) ;
return movedCount ;
}
void CServerGameDLL : : PreSave ( CSaveRestoreData * s )
{
g_pGameSaveRestoreBlockSet - > PreSave ( s ) ;
}
# include "client_textmessage.h"
// This little hack lets me marry BSP names to messages in titles.txt
typedef struct
{
char * pBSPName ;
char * pTitleName ;
} TITLECOMMENT ;
// this list gets searched for the first partial match, so some are out of order
static TITLECOMMENT gTitleComments [ ] =
{
{ " intro " , " #HL2_Chapter1_Title " } ,
{ " d1_trainstation_05 " , " #HL2_Chapter2_Title " } ,
{ " d1_trainstation_06 " , " #HL2_Chapter2_Title " } ,
{ " d1_trainstation_ " , " #HL2_Chapter1_Title " } ,
{ " d1_canals_06 " , " #HL2_Chapter4_Title " } ,
{ " d1_canals_07 " , " #HL2_Chapter4_Title " } ,
{ " d1_canals_08 " , " #HL2_Chapter4_Title " } ,
{ " d1_canals_09 " , " #HL2_Chapter4_Title " } ,
{ " d1_canals_1 " , " #HL2_Chapter4_Title " } ,
{ " d1_canals_0 " , " #HL2_Chapter3_Title " } ,
{ " d1_eli_ " , " #HL2_Chapter5_Title " } ,
{ " d1_town_ " , " #HL2_Chapter6_Title " } ,
{ " d2_coast_09 " , " #HL2_Chapter8_Title " } ,
{ " d2_coast_1 " , " #HL2_Chapter8_Title " } ,
{ " d2_prison_01 " , " #HL2_Chapter8_Title " } ,
{ " d2_coast_ " , " #HL2_Chapter7_Title " } ,
{ " d2_prison_06 " , " #HL2_Chapter9a_Title " } ,
{ " d2_prison_07 " , " #HL2_Chapter9a_Title " } ,
{ " d2_prison_08 " , " #HL2_Chapter9a_Title " } ,
{ " d2_prison_ " , " #HL2_Chapter9_Title " } ,
{ " d3_c17_01 " , " #HL2_Chapter9a_Title " } ,
{ " d3_c17_09 " , " #HL2_Chapter11_Title " } ,
{ " d3_c17_1 " , " #HL2_Chapter11_Title " } ,
{ " d3_c17_ " , " #HL2_Chapter10_Title " } ,
{ " d3_citadel_ " , " #HL2_Chapter12_Title " } ,
{ " d3_breen_ " , " #HL2_Chapter13_Title " } ,
{ " credits " , " #HL2_Chapter14_Title " } ,
{ " ep1_citadel_00 " , " #episodic_Chapter1_Title " } ,
{ " ep1_citadel_01 " , " #episodic_Chapter1_Title " } ,
{ " ep1_citadel_02b " , " #episodic_Chapter1_Title " } ,
{ " ep1_citadel_02 " , " #episodic_Chapter1_Title " } ,
{ " ep1_citadel_03 " , " #episodic_Chapter2_Title " } ,
{ " ep1_citadel_04 " , " #episodic_Chapter2_Title " } ,
{ " ep1_c17_00a " , " #episodic_Chapter3_Title " } ,
{ " ep1_c17_00 " , " #episodic_Chapter3_Title " } ,
{ " ep1_c17_01 " , " #episodic_Chapter4_Title " } ,
{ " ep1_c17_02b " , " #episodic_Chapter4_Title " } ,
{ " ep1_c17_02 " , " #episodic_Chapter4_Title " } ,
{ " ep1_c17_05 " , " #episodic_Chapter5_Title " } ,
{ " ep1_c17_06 " , " #episodic_Chapter5_Title " } ,
{ " ep2_outland_01a " , " #ep2_Chapter1_Title " } ,
{ " ep2_outland_01 " , " #ep2_Chapter1_Title " } ,
{ " ep2_outland_02 " , " #ep2_Chapter2_Title " } ,
{ " ep2_outland_03 " , " #ep2_Chapter2_Title " } ,
{ " ep2_outland_04 " , " #ep2_Chapter2_Title " } ,
{ " ep2_outland_05 " , " #ep2_Chapter3_Title " } ,
{ " ep2_outland_06a " , " #ep2_Chapter4_Title " } ,
{ " ep2_outland_06 " , " #ep2_Chapter3_Title " } ,
{ " ep2_outland_07 " , " #ep2_Chapter4_Title " } ,
{ " ep2_outland_08 " , " #ep2_Chapter4_Title " } ,
{ " ep2_outland_09 " , " #ep2_Chapter5_Title " } ,
{ " ep2_outland_10a " , " #ep2_Chapter5_Title " } ,
{ " ep2_outland_10 " , " #ep2_Chapter5_Title " } ,
{ " ep2_outland_11a " , " #ep2_Chapter6_Title " } ,
{ " ep2_outland_11 " , " #ep2_Chapter6_Title " } ,
{ " ep2_outland_12a " , " #ep2_Chapter7_Title " } ,
{ " ep2_outland_12 " , " #ep2_Chapter6_Title " } ,
} ;
void CServerGameDLL : : GetSaveComment ( char * text , int maxlength , float flMinutes , float flSeconds , bool bNoTime )
{
char comment [ 64 ] ;
const char * pName ;
int i ;
char const * mapname = STRING ( gpGlobals - > mapname ) ;
pName = NULL ;
// Try to find a matching title comment for this mapname
for ( i = 0 ; i < ARRAYSIZE ( gTitleComments ) & & ! pName ; i + + )
{
if ( ! Q_strnicmp ( mapname , gTitleComments [ i ] . pBSPName , strlen ( gTitleComments [ i ] . pBSPName ) ) )
{
// found one
int j ;
// Got a message, post-process it to be save name friendly
Q_strncpy ( comment , gTitleComments [ i ] . pTitleName , sizeof ( comment ) ) ;
pName = comment ;
j = 0 ;
// Strip out CRs
while ( j < 64 & & comment [ j ] )
{
if ( comment [ j ] = = ' \n ' | | comment [ j ] = = ' \r ' )
comment [ j ] = 0 ;
else
j + + ;
}
break ;
}
}
// If we didn't get one, use the designer's map name, or the BSP name itself
if ( ! pName )
{
pName = mapname ;
}
if ( bNoTime )
{
Q_snprintf ( text , maxlength , " %-64.64s " , pName ) ;
}
else
{
int minutes = flMinutes ;
int seconds = flSeconds ;
// Wow, this guy/gal must suck...!
if ( minutes > = 1000 )
{
minutes = 999 ;
seconds = 59 ;
}
int minutesAdd = ( seconds / 60 ) ;
seconds % = 60 ;
// add the elapsed time at the end of the comment, for the ui to parse out
Q_snprintf ( text , maxlength , " %-64.64s %03d:%02d " , pName , ( minutes + minutesAdd ) , seconds ) ;
}
}
void CServerGameDLL : : WriteSaveHeaders ( CSaveRestoreData * s )
{
CSave saveHelper ( s ) ;
g_pGameSaveRestoreBlockSet - > WriteSaveHeaders ( & saveHelper ) ;
g_pGameSaveRestoreBlockSet - > PostSave ( ) ;
}
void CServerGameDLL : : ReadRestoreHeaders ( CSaveRestoreData * s )
{
CRestore restoreHelper ( s ) ;
g_pGameSaveRestoreBlockSet - > PreRestore ( ) ;
g_pGameSaveRestoreBlockSet - > ReadRestoreHeaders ( & restoreHelper ) ;
}
void CServerGameDLL : : PreSaveGameLoaded ( char const * pSaveName , bool bInGame )
{
gamestats - > Event_PreSaveGameLoaded ( pSaveName , bInGame ) ;
}
//-----------------------------------------------------------------------------
// Purpose: Returns true if the game DLL wants the server not to be made public.
// Used by commentary system to hide multiplayer commentary servers from the master.
//-----------------------------------------------------------------------------
bool CServerGameDLL : : ShouldHideServer ( void )
{
if ( g_pcv_commentary & & g_pcv_commentary - > GetBool ( ) )
return true ;
if ( gpGlobals & & gpGlobals - > eLoadType = = MapLoad_Background )
return true ;
return false ;
}
//-----------------------------------------------------------------------------
//
//-----------------------------------------------------------------------------
void CServerGameDLL : : InvalidateMdlCache ( )
{
CBaseAnimating * pAnimating ;
for ( CBaseEntity * pEntity = gEntList . FirstEnt ( ) ; pEntity ! = NULL ; pEntity = gEntList . NextEnt ( pEntity ) )
{
pAnimating = dynamic_cast < CBaseAnimating * > ( pEntity ) ;
if ( pAnimating )
{
pAnimating - > InvalidateMdlCache ( ) ;
}
}
CStudioHdr : : CActivityToSequenceMapping : : ResetMappings ( ) ;
}
KeyValues * CServerGameDLL : : FindLaunchOptionByValue ( KeyValues * pLaunchOptions , char const * szLaunchOption )
{
if ( ! pLaunchOptions | | ! szLaunchOption | | ! * szLaunchOption )
return NULL ;
for ( KeyValues * val = pLaunchOptions - > GetFirstSubKey ( ) ; val ; val = val - > GetNextKey ( ) )
{
char const * szValue = val - > GetString ( ) ;
if ( szValue & & * szValue & & ! Q_stricmp ( szValue , szLaunchOption ) )
return val ;
}
return NULL ;
}
bool CServerGameDLL : : ShouldPreferSteamAuth ( )
{
return true ;
}
bool CServerGameDLL : : SupportsRandomMaps ( )
{
# ifdef INFESTED_DLL
return true ;
# else
return false ;
# endif
}
// return true to disconnect client due to timeout (used to do stricter timeouts when the game is sure the client isn't loading a map)
bool CServerGameDLL : : ShouldTimeoutClient ( int nUserID , float flTimeSinceLastReceived )
{
if ( ! g_pGameRules )
return false ;
return g_pGameRules - > ShouldTimeoutClient ( nUserID , flTimeSinceLastReceived ) ;
}
void CServerGameDLL : : GetMatchmakingTags ( char * buf , size_t bufSize )
{
# ifdef TERROR
// Additional "tags" that L4D matchmaking wants to know about
// static ConVarRef mp_gamemode( "mp_gamemode" );
extern ConVar mp_gamemode ;
if ( g_pMatchExtL4D )
{
char const * szBaseGameMode = g_pMatchExtL4D - > GetGameModeInfo ( mp_gamemode . GetString ( ) ) - > GetString ( " base " , mp_gamemode . GetString ( ) ) ;
if ( szBaseGameMode & & szBaseGameMode [ 0 ] )
{
Q_strncpy ( buf , szBaseGameMode , bufSize ) ;
return ;
}
}
Q_strncpy ( buf , mp_gamemode . GetString ( ) , bufSize ) ;
# endif
2024-08-29 19:58:34 -04:00
# ifdef INFESTED_DLL
extern ConVar asw_marine_ff_absorption ;
extern ConVar asw_sentry_friendly_fire_scale ;
extern ConVar asw_skill ;
char * const bufBase = buf ;
int len = 0 ;
// hardcore friendly fire
if ( asw_marine_ff_absorption . GetInt ( ) ! = 1 | | asw_sentry_friendly_fire_scale . GetFloat ( ) ! = 0.0f )
{
Q_strncpy ( buf , " HardcoreFF, " , bufSize ) ;
len = strlen ( buf ) ;
buf + = len ;
bufSize - = len ;
}
// difficulty level
const char * szSkill = " Normal, " ;
switch ( asw_skill . GetInt ( ) )
{
case 1 : szSkill = " Easy, " ; break ;
case 3 : szSkill = " Hard, " ; break ;
case 4 : szSkill = " Insane, " ; break ;
}
Q_strncpy ( buf , szSkill , bufSize ) ;
len = strlen ( buf ) ;
buf + = len ;
bufSize - = len ;
if ( ASWGameRules ( ) & & ASWGameRules ( ) - > GetGameState ( ) = = ASW_GS_BRIEFING )
{
Q_strncpy ( buf , " Briefing, " , bufSize ) ;
len = strlen ( buf ) ;
buf + = len ;
bufSize - = len ;
}
// Trim the last comma if anything was written
if ( buf > bufBase )
buf [ - 1 ] = 0 ;
# endif
2024-08-29 19:18:30 -04:00
}
void CServerGameDLL : : GetMatchmakingGameData ( char * buf , size_t bufSize )
{
char * const bufBase = buf ;
# ifdef TERROR
int len = 0 ;
// Put the game key
Q_snprintf ( buf , bufSize , " g:l4d2, " ) ;
len = strlen ( buf ) ;
buf + = len ;
bufSize - = len ;
// Supported l4d2 game types
static ConVarRef sv_gametypes ( " sv_gametypes " ) ;
if ( sv_gametypes . IsValid ( ) & & sv_gametypes . GetString ( ) [ 0 ] )
{
Q_snprintf ( buf , bufSize , " %s, " , sv_gametypes . GetString ( ) ) ;
len = strlen ( buf ) ;
buf + = len ;
bufSize - = len ;
}
// Additional "game data" that L4D matchmaking wants to know about
KeyValues * pAllMissions = g_pMatchExtL4D - > GetAllMissions ( ) ;
// Build a list of missions
int numMissions = 0 ;
for ( KeyValues * pMission = pAllMissions ? pAllMissions - > GetFirstTrueSubKey ( ) : NULL ;
pMission ;
pMission = pMission - > GetNextTrueSubKey ( ) )
{
if ( pMission - > GetInt ( " BuiltIn " ) )
continue ;
char const * szMission = pMission - > GetString ( " CfgTag " ) ;
int len = strlen ( szMission ) ;
if ( bufSize < = len + 1 )
{
Warning ( " GameData: Too many missions installed, not advertising for mission \" %s \" \n " , pMission - > GetString ( " Name " ) ) ;
continue ;
}
Q_strncpy ( buf , szMission , len + 1 ) ;
buf + = len ;
* ( buf + + ) = ' , ' ;
bufSize - = len + 1 ;
+ + numMissions ;
}
# endif
// Trim the last comma if anything was written
if ( buf > bufBase )
buf [ - 1 ] = 0 ;
}
void CServerGameDLL : : ServerHibernationUpdate ( bool bHibernating )
{
m_bIsHibernating = bHibernating ;
# ifdef INFESTED_DLL
if ( engine & & engine - > IsDedicatedServer ( ) & & m_bIsHibernating & & ASWGameRules ( ) )
{
ASWGameRules ( ) - > OnServerHibernating ( ) ;
}
# endif
}
//-----------------------------------------------------------------------------
// Purpose: Called during a transition, to build a map adjacency list
//-----------------------------------------------------------------------------
void CServerGameDLL : : BuildAdjacentMapList ( void )
{
// retrieve the pointer to the save data
CSaveRestoreData * pSaveData = gpGlobals - > pSaveData ;
if ( pSaveData )
pSaveData - > levelInfo . connectionCount = BuildChangeList ( pSaveData - > levelInfo . levelList , MAX_LEVEL_CONNECTIONS ) ;
}
//-----------------------------------------------------------------------------
// Purpose: Sanity-check to verify that a path is a relative path inside the game dir
// Taken From: engine/cmd.cpp
//-----------------------------------------------------------------------------
static bool IsValidPath ( const char * pszFilename )
{
if ( ! pszFilename )
{
return false ;
}
if ( Q_strlen ( pszFilename ) < = 0 | |
Q_IsAbsolutePath ( pszFilename ) | | // to protect absolute paths
Q_strstr ( pszFilename , " .. " ) ) // to protect relative paths
{
return false ;
}
return true ;
}
static void ValidateMOTDFilename ( IConVar * pConVar , const char * oldValue , float flOldValue )
{
ConVarRef var ( pConVar ) ;
if ( ! IsValidPath ( var . GetString ( ) ) )
{
var . SetValue ( var . GetDefault ( ) ) ;
}
}
static ConVar motdfile ( " motdfile " , " motd.txt " , FCVAR_RELEASE , " The MOTD file to load. " , ValidateMOTDFilename ) ;
static ConVar hostfile ( " hostfile " , " host.txt " , FCVAR_RELEASE , " The HOST file to load. " , ValidateMOTDFilename ) ;
void LoadMOTDFile ( const char * stringname , ConVar * pConvarFilename )
{
char data [ 2048 ] ;
int length = filesystem - > Size ( pConvarFilename - > GetString ( ) , " GAME " ) ;
if ( length < = 0 | | length > = ( sizeof ( data ) - 1 ) )
{
DevMsg ( " Invalid file size for %s \n " , pConvarFilename - > GetString ( ) ) ;
return ;
}
FileHandle_t hFile = filesystem - > Open ( pConvarFilename - > GetString ( ) , " rb " , " GAME " ) ;
if ( hFile = = FILESYSTEM_INVALID_HANDLE )
return ;
filesystem - > Read ( data , length , hFile ) ;
filesystem - > Close ( hFile ) ;
data [ length ] = 0 ;
g_pStringTableInfoPanel - > AddString ( CBaseEntity : : IsServer ( ) , stringname , length + 1 , data ) ;
}
void CServerGameDLL : : LoadMessageOfTheDay ( )
{
LoadMOTDFile ( " motd " , & motdfile ) ;
LoadMOTDFile ( " hostfile " , & hostfile ) ;
}
// keeps track of which chapters the user has unlocked
ConVar sv_unlockedchapters ( " sv_unlockedchapters " , " 1 " , FCVAR_ARCHIVE | FCVAR_ARCHIVE_XBOX ) ;
//-----------------------------------------------------------------------------
// Purpose: Updates which chapters are unlocked
//-----------------------------------------------------------------------------
void UpdateChapterRestrictions ( const char * mapname )
{
// look at the chapter for this map
char chapterTitle [ 64 ] ;
chapterTitle [ 0 ] = 0 ;
for ( int i = 0 ; i < ARRAYSIZE ( gTitleComments ) ; i + + )
{
if ( ! Q_strnicmp ( mapname , gTitleComments [ i ] . pBSPName , strlen ( gTitleComments [ i ] . pBSPName ) ) )
{
// found
Q_strncpy ( chapterTitle , gTitleComments [ i ] . pTitleName , sizeof ( chapterTitle ) ) ;
int j = 0 ;
while ( j < 64 & & chapterTitle [ j ] )
{
if ( chapterTitle [ j ] = = ' \n ' | | chapterTitle [ j ] = = ' \r ' )
chapterTitle [ j ] = 0 ;
else
j + + ;
}
break ;
}
}
if ( ! chapterTitle [ 0 ] )
return ;
// make sure the specified chapter title is unlocked
strlwr ( chapterTitle ) ;
// Get our active mod directory name
char modDir [ MAX_PATH ] ;
if ( UTIL_GetModDir ( modDir , sizeof ( modDir ) ) = = false )
return ;
char chapterNumberPrefix [ 64 ] ;
Q_snprintf ( chapterNumberPrefix , sizeof ( chapterNumberPrefix ) , " #%s_chapter " , modDir ) ;
const char * newChapterNumber = strstr ( chapterTitle , chapterNumberPrefix ) ;
if ( newChapterNumber )
{
// cut off the front
newChapterNumber + = strlen ( chapterNumberPrefix ) ;
char newChapter [ 32 ] ;
Q_strncpy ( newChapter , newChapterNumber , sizeof ( newChapter ) ) ;
// cut off the end
char * end = strstr ( newChapter , " _title " ) ;
if ( end )
{
* end = 0 ;
}
int nNewChapter = atoi ( newChapter ) ;
// HACK: HL2 added a zany chapter "9a" which wreaks
// havoc in this stupid atoi-based chapter code.
if ( ! Q_stricmp ( modDir , " hl2 " ) )
{
if ( ! Q_stricmp ( newChapter , " 9a " ) )
{
nNewChapter = 10 ;
}
else if ( nNewChapter > 9 )
{
nNewChapter + + ;
}
}
// ok we have the string, see if it's newer
const char * unlockedChapter = sv_unlockedchapters . GetString ( ) ;
int nUnlockedChapter = atoi ( unlockedChapter ) ;
if ( nUnlockedChapter < nNewChapter )
{
// ok we're at a higher chapter, unlock
sv_unlockedchapters . SetValue ( nNewChapter ) ;
// HACK: Call up through a better function than this? 7/23/07 - jdw
if ( IsX360 ( ) )
{
engine - > ServerCommand ( " host_writeconfig \n " ) ;
}
}
g_nCurrentChapterIndex = nNewChapter ;
}
}
//-----------------------------------------------------------------------------
// Precaches a vgui screen overlay material
//-----------------------------------------------------------------------------
void PrecacheMaterial ( const char * pMaterialName )
{
Assert ( pMaterialName & & pMaterialName [ 0 ] ) ;
g_pStringTableMaterials - > AddString ( CBaseEntity : : IsServer ( ) , pMaterialName ) ;
}
//-----------------------------------------------------------------------------
// Converts a previously precached material into an index
//-----------------------------------------------------------------------------
int GetMaterialIndex ( const char * pMaterialName )
{
if ( pMaterialName )
{
int nIndex = g_pStringTableMaterials - > FindStringIndex ( pMaterialName ) ;
if ( nIndex ! = INVALID_STRING_INDEX )
{
return nIndex ;
}
else
{
DevMsg ( " Warning! GetMaterialIndex: couldn't find material %s \n " , pMaterialName ) ;
return 0 ;
}
}
// This is the invalid string index
return 0 ;
}
//-----------------------------------------------------------------------------
// Converts a previously precached material index into a string
//-----------------------------------------------------------------------------
const char * GetMaterialNameFromIndex ( int nMaterialIndex )
{
return g_pStringTableMaterials - > GetString ( nMaterialIndex ) ;
}
//-----------------------------------------------------------------------------
// Precaches a vgui screen overlay material
//-----------------------------------------------------------------------------
int PrecacheParticleSystem ( const char * pParticleSystemName )
{
Assert ( pParticleSystemName & & pParticleSystemName [ 0 ] ) ;
return g_pStringTableParticleEffectNames - > AddString ( CBaseEntity : : IsServer ( ) , pParticleSystemName ) ;
}
void PrecacheParticleFileAndSystems ( const char * pParticleSystemFile )
{
g_pParticleSystemMgr - > ShouldLoadSheets ( true ) ;
g_pParticleSystemMgr - > ReadParticleConfigFile ( pParticleSystemFile , true , false ) ;
g_pParticleSystemMgr - > DecommitTempMemory ( ) ;
Assert ( pParticleSystemFile & & pParticleSystemFile [ 0 ] ) ;
g_pStringTableExtraParticleFiles - > AddString ( CBaseEntity : : IsServer ( ) , pParticleSystemFile ) ;
CUtlVector < CUtlString > systems ;
g_pParticleSystemMgr - > GetParticleSystemsInFile ( pParticleSystemFile , & systems ) ;
int nCount = systems . Count ( ) ;
for ( int i = 0 ; i < nCount ; + + i )
{
PrecacheParticleSystem ( systems [ i ] ) ;
}
}
void PrecacheGameSoundsFile ( const char * pSoundFile )
{
soundemitterbase - > AddSoundsFromFile ( pSoundFile , true ) ;
SoundSystemPreloadSounds ( ) ;
}
//-----------------------------------------------------------------------------
// Converts a previously precached material into an index
//-----------------------------------------------------------------------------
int GetParticleSystemIndex ( const char * pParticleSystemName )
{
if ( pParticleSystemName )
{
int nIndex = g_pStringTableParticleEffectNames - > FindStringIndex ( pParticleSystemName ) ;
if ( nIndex ! = INVALID_STRING_INDEX )
return nIndex ;
DevWarning ( " Server: Missing precache for particle system \" %s \" ! \n " , pParticleSystemName ) ;
}
// This is the invalid string index
return 0 ;
}
//-----------------------------------------------------------------------------
// Converts a previously precached material index into a string
//-----------------------------------------------------------------------------
const char * GetParticleSystemNameFromIndex ( int nMaterialIndex )
{
if ( nMaterialIndex < g_pStringTableParticleEffectNames - > GetMaxStrings ( ) )
return g_pStringTableParticleEffectNames - > GetString ( nMaterialIndex ) ;
return " error " ;
}
//-----------------------------------------------------------------------------
// Precaches an effect (used by DispatchEffect)
//-----------------------------------------------------------------------------
void PrecacheEffect ( const char * pEffectName )
{
Assert ( pEffectName & & pEffectName [ 0 ] ) ;
g_pStringTableEffectDispatch - > AddString ( CBaseEntity : : IsServer ( ) , pEffectName ) ;
}
//-----------------------------------------------------------------------------
// Converts a previously precached effect into an index
//-----------------------------------------------------------------------------
int GetEffectIndex ( const char * pEffectName )
{
if ( pEffectName )
{
int nIndex = g_pStringTableEffectDispatch - > FindStringIndex ( pEffectName ) ;
if ( nIndex ! = INVALID_STRING_INDEX )
return nIndex ;
DevWarning ( " Server: Missing precache for effect \" %s \" ! \n " , pEffectName ) ;
}
// This is the invalid string index
return 0 ;
}
//-----------------------------------------------------------------------------
// Converts a previously precached effect index into a string
//-----------------------------------------------------------------------------
const char * GetEffectNameFromIndex ( int nEffectIndex )
{
if ( nEffectIndex < g_pStringTableEffectDispatch - > GetMaxStrings ( ) )
return g_pStringTableEffectDispatch - > GetString ( nEffectIndex ) ;
return " error " ;
}
//-----------------------------------------------------------------------------
// Returns true if host_thread_mode is set to non-zero (and engine is running in threaded mode)
//-----------------------------------------------------------------------------
bool IsEngineThreaded ( )
{
if ( g_pcv_ThreadMode )
{
return g_pcv_ThreadMode - > GetBool ( ) ;
}
return false ;
}
class CServerGameEnts : public IServerGameEnts
{
public :
virtual void MarkEntitiesAsTouching ( edict_t * e1 , edict_t * e2 ) ;
virtual void FreeContainingEntity ( edict_t * ) ;
virtual edict_t * BaseEntityToEdict ( CBaseEntity * pEnt ) ;
virtual CBaseEntity * EdictToBaseEntity ( edict_t * pEdict ) ;
virtual void CheckTransmit ( CCheckTransmitInfo * pInfo , const unsigned short * pEdictIndices , int nEdicts ) ;
virtual void PrepareForFullUpdate ( edict_t * pEdict ) ;
} ;
EXPOSE_SINGLE_INTERFACE ( CServerGameEnts , IServerGameEnts , INTERFACEVERSION_SERVERGAMEENTS ) ;
//-----------------------------------------------------------------------------
// Purpose: Marks entities as touching
// Input : *e1 -
// *e2 -
//-----------------------------------------------------------------------------
void CServerGameEnts : : MarkEntitiesAsTouching ( edict_t * e1 , edict_t * e2 )
{
CBaseEntity * entity = GetContainingEntity ( e1 ) ;
CBaseEntity * entityTouched = GetContainingEntity ( e2 ) ;
if ( entity & & entityTouched )
{
// HACKHACK: UNDONE: Pass in the trace here??!?!?
trace_t tr ;
UTIL_ClearTrace ( tr ) ;
tr . endpos = ( entity - > GetAbsOrigin ( ) + entityTouched - > GetAbsOrigin ( ) ) * 0.5 ;
entity - > PhysicsMarkEntitiesAsTouching ( entityTouched , tr ) ;
}
}
void CServerGameEnts : : FreeContainingEntity ( edict_t * e )
{
: : FreeContainingEntity ( e ) ;
}
edict_t * CServerGameEnts : : BaseEntityToEdict ( CBaseEntity * pEnt )
{
if ( pEnt )
return pEnt - > edict ( ) ;
else
return NULL ;
}
CBaseEntity * CServerGameEnts : : EdictToBaseEntity ( edict_t * pEdict )
{
if ( pEdict )
return CBaseEntity : : Instance ( pEdict ) ;
else
return NULL ;
}
/* Yuck.. ideally this would be in CServerNetworkProperty's header, but it requires CBaseEntity and
// inlining it gives a nice speedup.
inline void CServerNetworkProperty : : CheckTransmit ( CCheckTransmitInfo * pInfo )
{
// If we have a transmit proxy, let it hook our ShouldTransmit return value.
if ( m_pTransmitProxy )
{
nShouldTransmit = m_pTransmitProxy - > ShouldTransmit ( pInfo , nShouldTransmit ) ;
}
if ( m_pOuter - > ShouldTransmit ( pInfo ) )
{
m_pOuter - > SetTransmit ( pInfo ) ;
}
} */
void CServerGameEnts : : CheckTransmit ( CCheckTransmitInfo * pInfo , const unsigned short * pEdictIndices , int nEdicts )
{
// NOTE: for speed's sake, this assumes that all networkables are CBaseEntities and that the edict list
// is consecutive in memory. If either of these things change, then this routine needs to change, but
// ideally we won't be calling any virtual from this routine. This speedy routine was added as an
// optimization which would be nice to keep.
edict_t * pBaseEdict = gpGlobals - > pEdicts ;
CBaseEntity * pRecipientEntity = CBaseEntity : : Instance ( pInfo - > m_pClientEnt ) ;
Assert ( pRecipientEntity & & pRecipientEntity - > IsPlayer ( ) ) ;
if ( ! pRecipientEntity )
return ;
MDLCACHE_CRITICAL_SECTION ( ) ;
CBasePlayer * pRecipientPlayer = static_cast < CBasePlayer * > ( pRecipientEntity ) ;
const int skyBoxArea = pRecipientPlayer - > m_Local . m_skybox3d . area ;
# ifndef _X360
const bool bIsHLTV = pRecipientPlayer - > IsHLTV ( ) ;
# if defined( REPLAY_ENABLED )
const bool bIsReplay = pRecipientPlayer - > IsReplay ( ) ;
# else
const bool bIsReplay = false ;
# endif
// m_pTransmitAlways must be set if HLTV client
Assert ( bIsHLTV = = ( pInfo - > m_pTransmitAlways ! = NULL ) | |
bIsReplay = = ( pInfo - > m_pTransmitAlways ! = NULL ) ) ;
# endif
for ( int i = 0 ; i < nEdicts ; i + + )
{
int iEdict = pEdictIndices [ i ] ;
# if _X360
if ( i < nEdicts - 1 )
{
PREFETCH360 ( & pBaseEdict [ pEdictIndices [ i + 1 ] ] , 0 ) ;
}
# endif
edict_t * pEdict = & pBaseEdict [ iEdict ] ;
int nFlags = pEdict - > m_fStateFlags & ( FL_EDICT_DONTSEND | FL_EDICT_ALWAYS | FL_EDICT_PVSCHECK | FL_EDICT_FULLCHECK ) ;
// entity needs no transmit
if ( nFlags & FL_EDICT_DONTSEND )
continue ;
PREFETCH360 ( pEdict - > GetUnknown ( ) , 0 ) ;
// entity is already marked for sending
if ( pInfo - > m_pTransmitEdict - > Get ( iEdict ) )
continue ;
if ( nFlags & FL_EDICT_ALWAYS )
{
// FIXME: Hey! Shouldn't this be using SetTransmit so as
// to also force network down dependent entities?
while ( true )
{
// mark entity for sending
pInfo - > m_pTransmitEdict - > Set ( iEdict ) ;
# ifndef _X360
if ( bIsHLTV | | bIsReplay )
{
pInfo - > m_pTransmitAlways - > Set ( iEdict ) ;
}
# endif
CServerNetworkProperty * pEnt = static_cast < CServerNetworkProperty * > ( pEdict - > GetNetworkable ( ) ) ;
if ( ! pEnt )
break ;
CServerNetworkProperty * pParent = pEnt - > GetNetworkParent ( ) ;
if ( ! pParent )
break ;
pEdict = pParent - > edict ( ) ;
iEdict = pParent - > entindex ( ) ;
}
continue ;
}
// FIXME: Would like to remove all dependencies
CBaseEntity * pEnt = ( CBaseEntity * ) pEdict - > GetUnknown ( ) ;
Assert ( dynamic_cast < CBaseEntity * > ( pEdict - > GetUnknown ( ) ) = = pEnt ) ;
if ( nFlags = = FL_EDICT_FULLCHECK )
{
// do a full ShouldTransmit() check, may return FL_EDICT_CHECKPVS
nFlags = pEnt - > ShouldTransmit ( pInfo ) ;
Assert ( ! ( nFlags & FL_EDICT_FULLCHECK ) ) ;
if ( nFlags & FL_EDICT_ALWAYS )
{
pEnt - > SetTransmit ( pInfo , true ) ;
continue ;
}
}
// don't send this entity
if ( ! ( nFlags & FL_EDICT_PVSCHECK ) )
continue ;
CServerNetworkProperty * netProp = static_cast < CServerNetworkProperty * > ( pEdict - > GetNetworkable ( ) ) ;
# ifndef _X360
if ( bIsHLTV | | bIsReplay )
{
// for the HLTV/Replay we don't cull against PVS
if ( netProp - > AreaNum ( ) = = skyBoxArea )
{
pEnt - > SetTransmit ( pInfo , true ) ;
}
else
{
pEnt - > SetTransmit ( pInfo , false ) ;
}
continue ;
}
# endif
// Always send entities in the player's 3d skybox.
// Sidenote: call of AreaNum() ensures that PVS data is up to date for this entity
bool bSameAreaAsSky = netProp - > AreaNum ( ) = = skyBoxArea ;
if ( bSameAreaAsSky )
{
pEnt - > SetTransmit ( pInfo , true ) ;
continue ;
}
bool bInPVS = netProp - > IsInPVS ( pInfo ) ;
if ( bInPVS | | sv_force_transmit_ents . GetBool ( ) )
{
// only send if entity is in PVS
pEnt - > SetTransmit ( pInfo , false ) ;
continue ;
}
// If the entity is marked "check PVS" but it's in hierarchy, walk up the hierarchy looking for the
// for any parent which is also in the PVS. If none are found, then we don't need to worry about sending ourself
CBaseEntity * orig = pEnt ;
CServerNetworkProperty * check = netProp - > GetNetworkParent ( ) ;
// BUG BUG: I think it might be better to build up a list of edict indices which "depend" on other answers and then
// resolve them in a second pass. Not sure what happens if an entity has two parents who both request PVS check?
while ( check )
{
int checkIndex = check - > entindex ( ) ;
// Parent already being sent
if ( pInfo - > m_pTransmitEdict - > Get ( checkIndex ) )
{
orig - > SetTransmit ( pInfo , true ) ;
break ;
}
edict_t * checkEdict = check - > edict ( ) ;
int checkFlags = checkEdict - > m_fStateFlags & ( FL_EDICT_DONTSEND | FL_EDICT_ALWAYS | FL_EDICT_PVSCHECK | FL_EDICT_FULLCHECK ) ;
if ( checkFlags & FL_EDICT_DONTSEND )
break ;
if ( checkFlags & FL_EDICT_ALWAYS )
{
orig - > SetTransmit ( pInfo , true ) ;
break ;
}
if ( checkFlags = = FL_EDICT_FULLCHECK )
{
// do a full ShouldTransmit() check, may return FL_EDICT_CHECKPVS
CBaseEntity * pCheckEntity = check - > GetBaseEntity ( ) ;
nFlags = pCheckEntity - > ShouldTransmit ( pInfo ) ;
Assert ( ! ( nFlags & FL_EDICT_FULLCHECK ) ) ;
if ( nFlags & FL_EDICT_ALWAYS )
{
pCheckEntity - > SetTransmit ( pInfo , true ) ;
orig - > SetTransmit ( pInfo , true ) ;
}
break ;
}
if ( checkFlags & FL_EDICT_PVSCHECK )
{
// Check pvs
check - > RecomputePVSInformation ( ) ;
bool bMoveParentInPVS = check - > IsInPVS ( pInfo ) ;
if ( bMoveParentInPVS )
{
orig - > SetTransmit ( pInfo , true ) ;
break ;
}
}
// Continue up chain just in case the parent itself has a parent that's in the PVS...
check = check - > GetNetworkParent ( ) ;
}
}
// Msg("A:%i, N:%i, F: %i, P: %i\n", always, dontSend, fullCheck, PVS );
}
//-----------------------------------------------------------------------------
// Purpose: called before a full update, so the server can flush any custom PVS info, etc
//-----------------------------------------------------------------------------
void CServerGameEnts : : PrepareForFullUpdate ( edict_t * pEdict )
{
CBaseEntity * pEntity = CBaseEntity : : Instance ( pEdict ) ;
Assert ( pEntity & & pEntity - > IsPlayer ( ) ) ;
if ( ! pEntity )
return ;
CBasePlayer * pPlayer = static_cast < CBasePlayer * > ( pEntity ) ;
pPlayer - > PrepareForFullUpdate ( ) ;
}
CServerGameClients g_ServerGameClients ;
EXPOSE_SINGLE_INTERFACE_GLOBALVAR ( CServerGameClients , IServerGameClients , INTERFACEVERSION_SERVERGAMECLIENTS , g_ServerGameClients ) ;
//-----------------------------------------------------------------------------
// Purpose: called when a player tries to connect to the server
// Input : *pEdict - the new player
// char *pszName - the players name
// char *pszAddress - the IP address of the player
// reject - output - fill in with the reason why
// maxrejectlen -- sizeof output buffer
// the player was not allowed to connect.
// Output : Returns TRUE if player is allowed to join, FALSE if connection is denied.
//-----------------------------------------------------------------------------
bool CServerGameClients : : ClientConnect ( edict_t * pEdict , const char * pszName , const char * pszAddress , char * reject , int maxrejectlen )
{
return g_pGameRules - > ClientConnected ( pEdict , pszName , pszAddress , reject , maxrejectlen ) ;
}
//-----------------------------------------------------------------------------
// Purpose: Called when a player is fully active (i.e. ready to receive messages)
// Input : *pEntity - the player
//-----------------------------------------------------------------------------
void CServerGameClients : : ClientActive ( edict_t * pEdict , bool bLoadGame )
{
MDLCACHE_CRITICAL_SECTION ( ) ;
: : ClientActive ( pEdict , bLoadGame ) ;
// If we just loaded from a save file, call OnRestore on valid entities
EndRestoreEntities ( ) ;
if ( gpGlobals - > eLoadType ! = MapLoad_LoadGame )
{
// notify all entities that the player is now in the game
for ( CBaseEntity * pEntity = gEntList . FirstEnt ( ) ; pEntity ! = NULL ; pEntity = gEntList . NextEnt ( pEntity ) )
{
pEntity - > PostClientActive ( ) ;
}
}
// Tell the sound controller to check looping sounds
CBasePlayer * pPlayer = ( CBasePlayer * ) CBaseEntity : : Instance ( pEdict ) ;
CSoundEnvelopeController : : GetController ( ) . CheckLoopingSoundsForPlayer ( pPlayer ) ;
SceneManager_ClientActive ( pPlayer ) ;
}
//-----------------------------------------------------------------------------
// Purpose: Called when a player is fully connect ( initial baseline entities have been received )
// Input : *pEntity - the player
//-----------------------------------------------------------------------------
void CServerGameClients : : ClientFullyConnect ( edict_t * pEdict )
{
: : ClientFullyConnect ( pEdict ) ;
}
//-----------------------------------------------------------------------------
// Purpose: called when a player disconnects from a server
// Input : *pEdict - the player
//-----------------------------------------------------------------------------
void CServerGameClients : : ClientDisconnect ( edict_t * pEdict )
{
extern bool g_fGameOver ;
CBasePlayer * player = ( CBasePlayer * ) CBaseEntity : : Instance ( pEdict ) ;
if ( player )
{
if ( ! g_fGameOver )
{
player - > SetMaxSpeed ( 0.0f ) ;
CSound * pSound ;
pSound = CSoundEnt : : SoundPointerForIndex ( CSoundEnt : : ClientSoundIndex ( pEdict ) ) ;
{
// since this client isn't around to think anymore, reset their sound.
if ( pSound )
{
pSound - > Reset ( ) ;
}
}
// since the edict doesn't get deleted, fix it so it doesn't interfere.
player - > RemoveFlag ( FL_AIMTARGET ) ; // don't attract autoaim
player - > AddFlag ( FL_DONTTOUCH ) ; // stop it touching anything
player - > AddFlag ( FL_NOTARGET ) ; // stop NPCs noticing it
player - > AddSolidFlags ( FSOLID_NOT_SOLID ) ; // nonsolid
if ( g_pGameRules )
{
g_pGameRules - > ClientDisconnected ( pEdict ) ;
gamestats - > Event_PlayerDisconnected ( player ) ;
}
}
// Make sure all Untouch()'s are called for this client leaving
CBaseEntity : : PhysicsRemoveTouchedList ( player ) ;
CBaseEntity : : PhysicsRemoveGroundList ( player ) ;
# if !defined( NO_ENTITY_PREDICTION )
// Make sure anything we "own" is simulated by the server from now on
player - > ClearPlayerSimulationList ( ) ;
# endif
}
}
void CServerGameClients : : ClientPutInServer ( edict_t * pEntity , const char * playername )
{
if ( g_pClientPutInServerOverride )
g_pClientPutInServerOverride ( pEntity , playername ) ;
else
: : ClientPutInServer ( pEntity , playername ) ;
CBasePlayer * pPlayer = ToBasePlayer ( GetContainingEntity ( pEntity ) ) ;
if ( pPlayer )
{
bool bIsSplitScreenPlayer = engine - > IsSplitScreenPlayer ( pPlayer - > entindex ( ) ) ;
CBasePlayer * pAttachedTo = NULL ;
if ( bIsSplitScreenPlayer )
{
pAttachedTo = ( CBasePlayer * ) : : GetContainingEntity ( engine - > GetSplitScreenPlayerAttachToEdict ( pPlayer - > entindex ( ) ) ) ;
}
pPlayer - > SetSplitScreenPlayer ( bIsSplitScreenPlayer , pAttachedTo ) ;
}
}
void CServerGameClients : : ClientCommand ( edict_t * pEntity , const CCommand & args )
{
CBasePlayer * pPlayer = ToBasePlayer ( GetContainingEntity ( pEntity ) ) ;
: : ClientCommand ( pPlayer , args ) ;
}
//-----------------------------------------------------------------------------
// Purpose: called after the player changes userinfo - gives dll a chance to modify
// it before it gets sent into the rest of the engine->
// Input : *pEdict - the player
// *infobuffer - their infobuffer
//-----------------------------------------------------------------------------
void CServerGameClients : : ClientSettingsChanged ( edict_t * pEdict )
{
// Is the client spawned yet?
if ( ! pEdict - > GetUnknown ( ) )
return ;
CBasePlayer * player = ( CBasePlayer * ) CBaseEntity : : Instance ( pEdict ) ;
if ( ! player )
return ;
# define QUICKGETCVARVALUE(v) (engine->GetClientConVarValue( player->entindex(), v ))
// get network setting for prediction & lag compensation
// Unfortunately, we have to duplicate the code in cdll_bounded_cvars.cpp here because the client
// doesn't send the virtualized value up (because it has no way to know when the virtualized value
// changes). Possible todo: put the responsibility on the bounded cvar to notify the engine when
// its virtualized value has changed.
player - > m_nUpdateRate = Q_atoi ( QUICKGETCVARVALUE ( " cl_updaterate " ) ) ;
static const ConVar * pMinUpdateRate = g_pCVar - > FindVar ( " sv_minupdaterate " ) ;
static const ConVar * pMaxUpdateRate = g_pCVar - > FindVar ( " sv_maxupdaterate " ) ;
if ( pMinUpdateRate & & pMaxUpdateRate )
player - > m_nUpdateRate = ( int ) clamp ( player - > m_nUpdateRate , pMinUpdateRate - > GetFloat ( ) , pMaxUpdateRate - > GetFloat ( ) ) ;
bool useInterpolation = Q_atoi ( QUICKGETCVARVALUE ( " cl_interpolate " ) ) ! = 0 ;
if ( useInterpolation )
{
float flLerpRatio = Q_atof ( QUICKGETCVARVALUE ( " cl_interp_ratio " ) ) ;
if ( flLerpRatio = = 0 )
flLerpRatio = 1.0f ;
float flLerpAmount = Q_atof ( QUICKGETCVARVALUE ( " cl_interp " ) ) ;
static const ConVar * pMin = g_pCVar - > FindVar ( " sv_client_min_interp_ratio " ) ;
static const ConVar * pMax = g_pCVar - > FindVar ( " sv_client_max_interp_ratio " ) ;
if ( pMin & & pMax & & pMin - > GetFloat ( ) ! = - 1 )
{
flLerpRatio = clamp ( flLerpRatio , pMin - > GetFloat ( ) , pMax - > GetFloat ( ) ) ;
}
else
{
if ( flLerpRatio = = 0 )
flLerpRatio = 1.0f ;
}
// #define FIXME_INTERP_RATIO
player - > m_fLerpTime = MAX ( flLerpAmount , flLerpRatio / player - > m_nUpdateRate ) ;
}
else
{
player - > m_fLerpTime = 0.0f ;
}
# if !defined( NO_ENTITY_PREDICTION )
bool usePrediction = Q_atoi ( QUICKGETCVARVALUE ( " cl_predict " ) ) ! = 0 ;
if ( usePrediction )
{
player - > m_bPredictWeapons = Q_atoi ( QUICKGETCVARVALUE ( " cl_predictweapons " ) ) ! = 0 ;
player - > m_bLagCompensation = Q_atoi ( QUICKGETCVARVALUE ( " cl_lagcompensation " ) ) ! = 0 ;
}
else
# endif
{
player - > m_bPredictWeapons = false ;
player - > m_bLagCompensation = false ;
}
# undef QUICKGETCVARVALUE
g_pGameRules - > ClientSettingsChanged ( player ) ;
}
//-----------------------------------------------------------------------------
// Purpose: A client can have a separate "view entity" indicating that his/her view should depend on the origin of that
// view entity. If that's the case, then pViewEntity will be non-NULL and will be used. Otherwise, the current
// entity's origin is used. Either is offset by the m_vecViewOffset to get the eye position.
// From the eye position, we set up the PAS and PVS to use for filtering network messages to the client. At this point, we could
// override the actual PAS or PVS values, or use a different origin.
// NOTE: Do not cache the values of pas and pvs, as they depend on reusable memory in the engine, they are only good for this one frame
// Input : *pViewEntity -
// *pClient -
// **pvs -
// **pas -
//-----------------------------------------------------------------------------
void CServerGameClients : : ClientSetupVisibility ( edict_t * pViewEntity , edict_t * pClient , unsigned char * pvs , int pvssize )
{
Vector org ;
// Reset the PVS!!!
engine - > ResetPVS ( pvs , pvssize ) ;
g_pToolFrameworkServer - > PreSetupVisibility ( ) ;
// Find the client's PVS
CBaseEntity * pVE = NULL ;
if ( pViewEntity )
{
pVE = GetContainingEntity ( pViewEntity ) ;
// If we have a viewentity, it overrides the player's origin
if ( pVE )
{
org = pVE - > EyePosition ( ) ;
engine - > AddOriginToPVS ( org ) ;
}
}
float fovDistanceAdjustFactor = 1 ;
CUtlVector < Vector > areaPortalOrigins ;
CBasePlayer * pPlayer = ( CBasePlayer * ) GetContainingEntity ( pClient ) ;
if ( pPlayer )
{
if ( ! pVE )
{
org = pPlayer - > EyePosition ( ) ;
}
pPlayer - > SetupVisibility ( pVE , pvs , pvssize ) ;
UTIL_SetClientVisibilityPVS ( pClient , pvs , pvssize ) ;
fovDistanceAdjustFactor = pPlayer - > GetFOVDistanceAdjustFactorForNetworking ( ) ;
areaPortalOrigins . AddToTail ( org ) ;
// Merge in areaportal "window" states from from split screen players by passing in the extra PVS origins!!!
for ( int i = 1 ; i < MAX_SPLITSCREEN_PLAYERS ; + + i )
{
CBasePlayer * pl = ( CBasePlayer * ) ToBasePlayer ( GetContainingEntity ( engine - > GetSplitScreenPlayerForEdict ( pPlayer - > entindex ( ) , i ) ) ) ;
if ( ! pl )
continue ;
org = pl - > EyePosition ( ) ;
areaPortalOrigins . AddToTail ( org ) ;
}
}
else
{
Warning ( " ClientSetupVisibility: No entity for edict! \n " ) ;
areaPortalOrigins . AddToTail ( org ) ;
}
unsigned char portalBits [ MAX_AREA_PORTAL_STATE_BYTES ] ;
memset ( portalBits , 0 , sizeof ( portalBits ) ) ;
int portalNums [ 512 ] ;
int isOpen [ 512 ] ;
int iOutPortal = 0 ;
for ( unsigned short i = g_AreaPortals . Head ( ) ; i ! = g_AreaPortals . InvalidIndex ( ) ; i = g_AreaPortals . Next ( i ) )
{
CFuncAreaPortalBase * pCur = g_AreaPortals [ i ] ;
bool bIsOpenOnClient = true ;
// Update our array of which portals are open and flush it if necessary.
portalNums [ iOutPortal ] = pCur - > m_portalNumber ;
isOpen [ iOutPortal ] = pCur - > UpdateVisibility ( areaPortalOrigins , fovDistanceAdjustFactor , bIsOpenOnClient ) ;
+ + iOutPortal ;
if ( iOutPortal > = ARRAYSIZE ( portalNums ) )
{
engine - > SetAreaPortalStates ( portalNums , isOpen , iOutPortal ) ;
iOutPortal = 0 ;
}
// Version 0 portals (ie: shipping Half-Life 2 era) are always treated as open
// for purposes of the m_chAreaPortalBits array on the client.
if ( pCur - > m_iPortalVersion = = 0 )
bIsOpenOnClient = true ;
if ( bIsOpenOnClient )
{
if ( pCur - > m_portalNumber < 0 )
continue ;
else if ( pCur - > m_portalNumber > = sizeof ( portalBits ) * 8 )
Error ( " ClientSetupVisibility: portal number (%d) too large " , pCur - > m_portalNumber ) ;
else
portalBits [ pCur - > m_portalNumber > > 3 ] | = ( 1 < < ( pCur - > m_portalNumber & 7 ) ) ;
}
}
// Flush the remaining areaportal states.
engine - > SetAreaPortalStates ( portalNums , isOpen , iOutPortal ) ;
// Update the area bits that get sent to the client.
Assert ( pPlayer ) ;
if ( pPlayer )
{
pPlayer - > m_Local . UpdateAreaBits ( pPlayer , portalBits ) ;
}
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *player -
// *buf -
// numcmds -
// totalcmds -
// dropped_packets -
// ignore -
// paused -
// Output : float
//-----------------------------------------------------------------------------
# define CMD_MAXBACKUP 64
float CServerGameClients : : ProcessUsercmds ( edict_t * player , bf_read * buf , int numcmds , int totalcmds ,
int dropped_packets , bool ignore , bool paused )
{
int i ;
CUserCmd * from , * to ;
// We track last three command in case we drop some
// packets but get them back.
CUserCmd cmds [ CMD_MAXBACKUP ] ;
CUserCmd cmdNull ; // For delta compression
Assert ( numcmds > = 0 ) ;
Assert ( ( totalcmds - numcmds ) > = 0 ) ;
CBasePlayer * pPlayer = NULL ;
CBaseEntity * pEnt = CBaseEntity : : Instance ( player ) ;
if ( pEnt & & pEnt - > IsPlayer ( ) )
{
pPlayer = static_cast < CBasePlayer * > ( pEnt ) ;
}
// Too many commands?
if ( totalcmds < 0 | | totalcmds > = ( CMD_MAXBACKUP - 1 ) )
{
const char * name = " unknown " ;
if ( pPlayer )
{
name = pPlayer - > GetPlayerName ( ) ;
}
Msg ( " CBasePlayer::ProcessUsercmds: too many cmds %i sent for player %s \n " , totalcmds , name ) ;
// FIXME: Need a way to drop the client from here
//SV_DropClient ( host_client, false, "CMD_MAXBACKUP hit" );
buf - > SetOverflowFlag ( ) ;
return 0.0f ;
}
// Initialize for reading delta compressed usercmds
cmdNull . Reset ( ) ;
from = & cmdNull ;
for ( i = totalcmds - 1 ; i > = 0 ; i - - )
{
to = & cmds [ i ] ;
ReadUsercmd ( buf , to , from ) ;
from = to ;
}
// Client not fully connected or server has gone inactive or is paused, just ignore
if ( ignore | | ! pPlayer )
{
return 0.0f ;
}
MDLCACHE_CRITICAL_SECTION ( ) ;
pPlayer - > ProcessUsercmds ( cmds , numcmds , totalcmds , dropped_packets , paused ) ;
return TICK_INTERVAL ;
}
void CServerGameClients : : PostClientMessagesSent ( void )
{
VPROF ( " CServerGameClients::PostClient " ) ;
gEntList . PostClientMessagesSent ( ) ;
}
// Sets the client index for the client who typed the command into his/her console
void CServerGameClients : : SetCommandClient ( int index )
{
g_nCommandClientIndex = index ;
}
int CServerGameClients : : GetReplayDelay ( edict_t * pEdict , int & entity )
{
CBasePlayer * pPlayer = ( CBasePlayer * ) CBaseEntity : : Instance ( pEdict ) ;
if ( ! pPlayer )
return 0 ;
entity = pPlayer - > GetReplayEntity ( ) ;
return pPlayer - > GetDelayTicks ( ) ;
}
//-----------------------------------------------------------------------------
// The client's userinfo data lump has changed
//-----------------------------------------------------------------------------
void CServerGameClients : : ClientEarPosition ( edict_t * pEdict , Vector * pEarOrigin )
{
CBasePlayer * pPlayer = ( CBasePlayer * ) CBaseEntity : : Instance ( pEdict ) ;
if ( pPlayer )
{
* pEarOrigin = pPlayer - > EarPosition ( ) ;
}
else
{
// Shouldn't happen
Assert ( 0 ) ;
* pEarOrigin = vec3_origin ;
}
}
//-----------------------------------------------------------------------------
// Purpose:
// Input : *player -
// Output : CPlayerState
//-----------------------------------------------------------------------------
CPlayerState * CServerGameClients : : GetPlayerState ( edict_t * player )
{
// Is the client spawned yet?
if ( ! player | | ! player - > GetUnknown ( ) )
return NULL ;
CBasePlayer * pBasePlayer = ( CBasePlayer * ) CBaseEntity : : Instance ( player ) ;
if ( ! pBasePlayer )
return NULL ;
return & pBasePlayer - > pl ;
}
//-----------------------------------------------------------------------------
// Purpose: Anything this game .dll wants to add to the bug reporter text (e.g., the entity/model under the picker crosshair)
// can be added here
// Input : *buf -
// buflen -
//-----------------------------------------------------------------------------
void CServerGameClients : : GetBugReportInfo ( char * buf , int buflen )
{
recentNPCSpeech_t speech [ SPEECH_LIST_MAX_SOUNDS ] ;
int num ;
int i ;
buf [ 0 ] = 0 ;
if ( gpGlobals - > maxClients = = 1 )
{
CBaseEntity * ent = UTIL_PlayerByIndex ( 1 ) ? UTIL_PlayerByIndex ( 1 ) - > FindPickerEntity ( ) : NULL ;
if ( ent )
{
Q_snprintf ( buf , buflen , " Picker %i/%s - ent %s model %s \n " ,
ent - > entindex ( ) ,
ent - > GetClassname ( ) ,
STRING ( ent - > GetEntityName ( ) ) ,
STRING ( ent - > GetModelName ( ) ) ) ;
}
// get any sounds that were spoken by NPCs recently
num = GetRecentNPCSpeech ( speech ) ;
if ( num > 0 )
{
Q_snprintf ( buf , buflen , " %sRecent NPC speech: \n " , buf ) ;
for ( i = 0 ; i < num ; i + + )
{
Q_snprintf ( buf , buflen , " %s time: %6.3f sound name: %s scene: %s \n " , buf , speech [ i ] . time , speech [ i ] . name , speech [ i ] . sceneName ) ;
}
Q_snprintf ( buf , buflen , " %sCurrent time: %6.3f \n " , buf , gpGlobals - > curtime ) ;
}
}
}
//-----------------------------------------------------------------------------
// Purpose: A player sent a voice packet
//-----------------------------------------------------------------------------
void CServerGameClients : : ClientVoice ( edict_t * pEdict )
{
CBasePlayer * pPlayer = ( CBasePlayer * ) CBaseEntity : : Instance ( pEdict ) ;
if ( pPlayer )
{
pPlayer - > OnVoiceTransmit ( ) ;
// Notify the voice listener that we've spoken
PlayerVoiceListener ( ) . AddPlayerSpeakTime ( pPlayer ) ;
}
}
//-----------------------------------------------------------------------------
// Purpose: A user has had their network id setup and validated
//-----------------------------------------------------------------------------
void CServerGameClients : : NetworkIDValidated ( const char * pszUserName , const char * pszNetworkID )
{
}
int CServerGameClients : : GetMaxSplitscreenPlayers ( )
{
return MAX_SPLITSCREEN_PLAYERS ;
}
int CServerGameClients : : GetMaxHumanPlayers ( )
{
if ( g_pGameRules )
{
return g_pGameRules - > GetMaxHumanPlayers ( ) ;
}
return - 1 ;
}
// The client has submitted a keyvalues command
void CServerGameClients : : ClientCommandKeyValues ( edict_t * pEntity , KeyValues * pKeyValues )
{
if ( ! pKeyValues )
return ;
char const * szCommand = pKeyValues - > GetName ( ) ;
if ( FStrEq ( szCommand , " avatarinfo " ) )
{
// Player is communicating team and avatar setting
//TheDirector->PlayerAvatarSet( pEntity, pKeyValues );
}
g_pGameRules - > ClientCommandKeyValues ( pEntity , pKeyValues ) ;
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
static bf_write * g_pMsgBuffer = NULL ;
void EntityMessageBegin ( CBaseEntity * entity , bool reliable /*= false*/ )
{
Assert ( ! g_pMsgBuffer ) ;
Assert ( entity ) ;
g_pMsgBuffer = engine - > EntityMessageBegin ( entity - > entindex ( ) , entity - > GetServerClass ( ) , reliable ) ;
}
void UserMessageBegin ( IRecipientFilter & filter , const char * messagename )
{
Assert ( ! g_pMsgBuffer ) ;
Assert ( messagename ) ;
int msg_type = usermessages - > LookupUserMessage ( messagename ) ;
if ( msg_type = = - 1 )
{
Error ( " UserMessageBegin: Unregistered message '%s' \n " , messagename ) ;
}
g_pMsgBuffer = engine - > UserMessageBegin ( & filter , msg_type , messagename ) ;
}
void MessageEnd ( void )
{
Assert ( g_pMsgBuffer ) ;
engine - > MessageEnd ( ) ;
g_pMsgBuffer = NULL ;
}
void MessageWriteByte ( int iValue )
{
if ( ! g_pMsgBuffer )
Error ( " WRITE_BYTE called with no active message \n " ) ;
g_pMsgBuffer - > WriteByte ( iValue ) ;
}
void MessageWriteChar ( int iValue )
{
if ( ! g_pMsgBuffer )
Error ( " WRITE_CHAR called with no active message \n " ) ;
g_pMsgBuffer - > WriteChar ( iValue ) ;
}
void MessageWriteShort ( int iValue )
{
if ( ! g_pMsgBuffer )
Error ( " WRITE_SHORT called with no active message \n " ) ;
g_pMsgBuffer - > WriteShort ( iValue ) ;
}
void MessageWriteWord ( int iValue )
{
if ( ! g_pMsgBuffer )
Error ( " WRITE_WORD called with no active message \n " ) ;
g_pMsgBuffer - > WriteWord ( iValue ) ;
}
void MessageWriteLong ( int iValue )
{
if ( ! g_pMsgBuffer )
Error ( " WriteLong called with no active message \n " ) ;
g_pMsgBuffer - > WriteLong ( iValue ) ;
}
void MessageWriteFloat ( float flValue )
{
if ( ! g_pMsgBuffer )
Error ( " WriteFloat called with no active message \n " ) ;
g_pMsgBuffer - > WriteFloat ( flValue ) ;
}
void MessageWriteAngle ( float flValue )
{
if ( ! g_pMsgBuffer )
Error ( " WriteAngle called with no active message \n " ) ;
g_pMsgBuffer - > WriteBitAngle ( flValue , 8 ) ;
}
void MessageWriteCoord ( float flValue )
{
if ( ! g_pMsgBuffer )
Error ( " WriteCoord called with no active message \n " ) ;
g_pMsgBuffer - > WriteBitCoord ( flValue ) ;
}
void MessageWriteVec3Coord ( const Vector & rgflValue )
{
if ( ! g_pMsgBuffer )
Error ( " WriteVec3Coord called with no active message \n " ) ;
g_pMsgBuffer - > WriteBitVec3Coord ( rgflValue ) ;
}
void MessageWriteVec3Normal ( const Vector & rgflValue )
{
if ( ! g_pMsgBuffer )
Error ( " WriteVec3Normal called with no active message \n " ) ;
g_pMsgBuffer - > WriteBitVec3Normal ( rgflValue ) ;
}
void MessageWriteBitVecIntegral ( const Vector & vecValue )
{
if ( ! g_pMsgBuffer )
Error ( " MessageWriteBitVecIntegral called with no active message \n " ) ;
for ( int i = 0 ; i < 3 ; + + i )
{
g_pMsgBuffer - > WriteBitCoordMP ( vecValue [ i ] , kCW_Integral ) ;
}
}
void MessageWriteAngles ( const QAngle & rgflValue )
{
if ( ! g_pMsgBuffer )
Error ( " WriteVec3Normal called with no active message \n " ) ;
g_pMsgBuffer - > WriteBitAngles ( rgflValue ) ;
}
void MessageWriteString ( const char * sz )
{
if ( ! g_pMsgBuffer )
Error ( " WriteString called with no active message \n " ) ;
g_pMsgBuffer - > WriteString ( sz ) ;
}
void MessageWriteEntity ( int iValue )
{
if ( ! g_pMsgBuffer )
Error ( " WriteEntity called with no active message \n " ) ;
g_pMsgBuffer - > WriteShort ( iValue ) ;
}
void MessageWriteEHandle ( CBaseEntity * pEntity )
{
if ( ! g_pMsgBuffer )
Error ( " WriteEHandle called with no active message \n " ) ;
long iEncodedEHandle ;
if ( pEntity )
{
EHANDLE hEnt = pEntity ;
int iSerialNum = hEnt . GetSerialNumber ( ) & ( 1 < < NUM_NETWORKED_EHANDLE_SERIAL_NUMBER_BITS ) - 1 ;
iEncodedEHandle = hEnt . GetEntryIndex ( ) | ( iSerialNum < < MAX_EDICT_BITS ) ;
}
else
{
iEncodedEHandle = INVALID_NETWORKED_EHANDLE_VALUE ;
}
g_pMsgBuffer - > WriteLong ( iEncodedEHandle ) ;
}
// bitwise
void MessageWriteBool ( bool bValue )
{
if ( ! g_pMsgBuffer )
Error ( " WriteBool called with no active message \n " ) ;
g_pMsgBuffer - > WriteOneBit ( bValue ? 1 : 0 ) ;
}
void MessageWriteUBitLong ( unsigned int data , int numbits )
{
if ( ! g_pMsgBuffer )
Error ( " WriteUBitLong called with no active message \n " ) ;
g_pMsgBuffer - > WriteUBitLong ( data , numbits ) ;
}
void MessageWriteSBitLong ( int data , int numbits )
{
if ( ! g_pMsgBuffer )
Error ( " WriteSBitLong called with no active message \n " ) ;
g_pMsgBuffer - > WriteSBitLong ( data , numbits ) ;
}
void MessageWriteBits ( const void * pIn , int nBits )
{
if ( ! g_pMsgBuffer )
Error ( " WriteBits called with no active message \n " ) ;
g_pMsgBuffer - > WriteBits ( pIn , nBits ) ;
}
class CServerDLLSharedAppSystems : public IServerDLLSharedAppSystems
{
public :
CServerDLLSharedAppSystems ( )
{
AddAppSystem ( " soundemittersystem " , SOUNDEMITTERSYSTEM_INTERFACE_VERSION ) ;
AddAppSystem ( " scenefilecache " , SCENE_FILE_CACHE_INTERFACE_VERSION ) ;
# ifdef INFESTED_DLL
AddAppSystem ( " missionchooser " , ASW_MISSION_CHOOSER_VERSION ) ;
# endif
}
virtual int Count ( )
{
return m_Systems . Count ( ) ;
}
virtual char const * GetDllName ( int idx )
{
return m_Systems [ idx ] . m_pModuleName ;
}
virtual char const * GetInterfaceName ( int idx )
{
return m_Systems [ idx ] . m_pInterfaceName ;
}
private :
void AddAppSystem ( char const * moduleName , char const * interfaceName )
{
AppSystemInfo_t sys ;
sys . m_pModuleName = moduleName ;
sys . m_pInterfaceName = interfaceName ;
m_Systems . AddToTail ( sys ) ;
}
CUtlVector < AppSystemInfo_t > m_Systems ;
} ;
EXPOSE_SINGLE_INTERFACE ( CServerDLLSharedAppSystems , IServerDLLSharedAppSystems , SERVER_DLL_SHARED_APPSYSTEMS ) ;
//-----------------------------------------------------------------------------
//
//-----------------------------------------------------------------------------
void CServerGameTags : : GetTaggedConVarList ( KeyValues * pCvarTagList )
{
if ( pCvarTagList & & g_pGameRules )
{
g_pGameRules - > GetTaggedConVarList ( pCvarTagList ) ;
}
}