source-engine/engine/baseclientstate.h

257 lines
8.9 KiB
C
Raw Permalink Normal View History

2020-04-22 12:56:21 -04:00
//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
//=============================================================================//
#ifndef BASECLIENTSTATE_H
#define BASECLIENTSTATE_H
#ifdef _WIN32
#pragma once
#endif
#include <inetmsghandler.h>
#include <protocol.h>
#include <client_class.h>
#include <cdll_int.h>
#include <netadr.h>
#include "common.h"
#include "clockdriftmgr.h"
#include "convar.h"
#include "cl_bounded_cvars.h"
// Only send this many requests before timing out.
#define CL_CONNECTION_RETRIES 4
// Mininum time gap (in seconds) before a subsequent connection request is sent.
#define CL_MIN_RESEND_TIME 1.5f
// Max time. The cvar cl_resend is bounded by these.
#define CL_MAX_RESEND_TIME 20.0f
// In release, send commands at least this many times per second
#define MIN_CMD_RATE 10.0f
#define MAX_CMD_RATE 100.0f
extern ConVar cl_name;
// This represents a server's
class C_ServerClassInfo
{
public:
C_ServerClassInfo();
~C_ServerClassInfo();
public:
ClientClass *m_pClientClass;
char *m_ClassName;
char *m_DatatableName;
// This is an index into the network string table (cl.GetInstanceBaselineTable()).
int m_InstanceBaselineIndex; // INVALID_STRING_INDEX if not initialized yet.
};
#define EndGameAssertMsg( assertion, msg ) \
if ( !(assertion) )\
Host_EndGame msg
class CNetworkStringTableContainer;
class PackedEntity;
class INetworkStringTable;
class CEntityReadInfo;
abstract_class CBaseClientState : public INetChannelHandler, public IConnectionlessPacketHandler, public IServerMessageHandler
{
public:
CBaseClientState();
virtual ~CBaseClientState();
public: // IConnectionlessPacketHandler interface:
virtual bool ProcessConnectionlessPacket(struct netpacket_s *packet);
public: // INetMsgHandler interface:
virtual void ConnectionStart(INetChannel *chan);
virtual void ConnectionClosing( const char *reason );
virtual void ConnectionCrashed(const char *reason);
virtual void PacketStart(int incoming_sequence, int outgoing_acknowledged) {};
virtual void PacketEnd( void ) {};
virtual void FileReceived( const char *fileName, unsigned int transferID );
virtual void FileRequested( const char *fileName, unsigned int transferID );
virtual void FileDenied( const char *fileName, unsigned int transferID );
virtual void FileSent( const char *fileName, unsigned int transferID );
public: // IServerMessageHandlers
PROCESS_NET_MESSAGE( Tick );
PROCESS_NET_MESSAGE( StringCmd );
PROCESS_NET_MESSAGE( SetConVar );
PROCESS_NET_MESSAGE( SignonState );
PROCESS_SVC_MESSAGE( Print );
PROCESS_SVC_MESSAGE( ServerInfo );
PROCESS_SVC_MESSAGE( SendTable );
PROCESS_SVC_MESSAGE( ClassInfo );
PROCESS_SVC_MESSAGE( SetPause );
PROCESS_SVC_MESSAGE( CreateStringTable );
PROCESS_SVC_MESSAGE( UpdateStringTable );
PROCESS_SVC_MESSAGE( SetView );
PROCESS_SVC_MESSAGE( PacketEntities );
PROCESS_SVC_MESSAGE( Menu );
PROCESS_SVC_MESSAGE( GameEventList );
PROCESS_SVC_MESSAGE( GetCvarValue );
PROCESS_SVC_MESSAGE( CmdKeyValues );
PROCESS_SVC_MESSAGE( SetPauseTimed );
// Returns dem file protocol version, or, if not playing a demo, just returns PROTOCOL_VERSION
virtual int GetDemoProtocolVersion() const;
public:
inline bool IsActive( void ) const { return m_nSignonState == SIGNONSTATE_FULL; };
inline bool IsConnected( void ) const { return m_nSignonState >= SIGNONSTATE_CONNECTED; };
virtual void Clear( void );
virtual void FullConnect( netadr_t &adr ); // a connection was established
virtual void Connect(const char* adr, const char *pszSourceTag); // start a connection challenge
virtual bool SetSignonState ( int state, int count );
virtual void Disconnect( const char *pszReason, bool bShowMainMenu );
virtual void SendConnectPacket (int challengeNr, int authProtocol, uint64 unGSSteamID, bool bGSSecure );
virtual const char *GetCDKeyHash() { return "123"; }
virtual void RunFrame ( void );
virtual void CheckForResend ( void );
virtual void InstallStringTableCallback( char const *tableName ) { }
virtual bool HookClientStringTable( char const *tableName ) { return false; }
virtual bool LinkClasses( void );
virtual int GetConnectionRetryNumber() const { return CL_CONNECTION_RETRIES; }
virtual const char *GetClientName() { return cl_name.GetString(); }
static ClientClass* FindClientClass(const char *pClassName);
CClockDriftMgr& GetClockDriftMgr();
int GetClientTickCount() const; // Get the client tick count.
void SetClientTickCount( int tick );
int GetServerTickCount() const;
void SetServerTickCount( int tick );
void SetClientAndServerTickCount( int tick );
INetworkStringTable *GetStringTable( const char * name ) const;
PackedEntity *GetEntityBaseline( int iBaseline, int nEntityIndex );
void SetEntityBaseline(int iBaseline, ClientClass *pClientClass, int index, char *packedData, int length);
void CopyEntityBaseline( int iFrom, int iTo );
void FreeEntityBaselines();
bool GetClassBaseline( int iClass, void const **pData, int *pDatalen );
ClientClass *GetClientClass( int i );
void ForceFullUpdate( void );
void SendStringCmd(const char * command);
void ReadPacketEntities( CEntityReadInfo &u );
virtual void ReadEnterPVS( CEntityReadInfo &u ) = 0;
virtual void ReadLeavePVS( CEntityReadInfo &u ) = 0;
virtual void ReadDeltaEnt( CEntityReadInfo &u ) = 0;
virtual void ReadPreserveEnt( CEntityReadInfo &u ) = 0;
virtual void ReadDeletions( CEntityReadInfo &u ) = 0;
bool IsClientConnectionViaMatchMaking( void );
static bool ConnectMethodAllowsRedirects( void );
private:
bool PrepareSteamConnectResponse( uint64 unGSSteamID, bool bGSSecure, const netadr_t &adr, bf_write &msg );
public:
// Connection to server.
int m_Socket; // network socket
INetChannel *m_NetChannel; // Our sequenced channel to the remote server.
unsigned int m_nChallengeNr; // connection challenge number
double m_flConnectTime; // If gap of connect_time to net_time > 3000, then resend connect packet
int m_nRetryNumber; // number of retry connection attemps
char m_szRetryAddress[ MAX_OSPATH ];
CUtlString m_sRetrySourceTag; // string that describes why we decided to connect to this server (empty for command line, "serverbrowser", "quickplay", etc)
int m_retryChallenge; // challenge we sent to the server
int m_nSignonState; // see SIGNONSTATE_* definitions
double m_flNextCmdTime; // When can we send the next command packet?
int m_nServerCount; // server identification for prespawns, must match the svs.spawncount which
// is incremented on server spawning. This supercedes svs.spawn_issued, in that
// we can now spend a fair amount of time sitting connected to the server
// but downloading models, sounds, etc. So much time that it is possible that the
// server might change levels again and, if so, we need to know that.
uint64 m_ulGameServerSteamID; // Steam ID of the game server we are trying to connect to, or are connected to. Zero if unknown
int m_nCurrentSequence; // this is the sequence number of the current incoming packet
CClockDriftMgr m_ClockDriftMgr;
int m_nDeltaTick; // last valid received snapshot (server) tick
bool m_bPaused; // send over by server
float m_flPausedExpireTime;
int m_nViewEntity; // cl_entitites[cl.viewentity] == player point of view
int m_nPlayerSlot; // own player entity index-1. skips world. Add 1 to get cl_entitites index;
char m_szLevelFileName[ 128 ]; // for display on solo scoreboard
char m_szLevelBaseName[ 128 ]; // removes maps/ and .bsp extension
int m_nMaxClients; // max clients on server
PackedEntity *m_pEntityBaselines[2][MAX_EDICTS]; // storing entity baselines
// This stuff manages the receiving of data tables and instantiating of client versions
// of server-side classes.
C_ServerClassInfo *m_pServerClasses;
int m_nServerClasses;
int m_nServerClassBits;
char m_szEncrytionKey[STEAM_KEYSIZE];
unsigned int m_iEncryptionKeySize;
CNetworkStringTableContainer *m_StringTableContainer;
bool m_bRestrictServerCommands; // If true, then the server is only allowed to execute commands marked with FCVAR_SERVER_CAN_EXECUTE on the client.
bool m_bRestrictClientCommands; // If true, then IVEngineClient::ClientCmd is only allowed to execute commands marked with FCVAR_CLIENTCMD_CAN_EXECUTE on the client.
};
inline CClockDriftMgr& CBaseClientState::GetClockDriftMgr()
{
return m_ClockDriftMgr;
}
inline void CBaseClientState::SetClientTickCount( int tick )
{
m_ClockDriftMgr.m_nClientTick = tick;
}
inline int CBaseClientState::GetClientTickCount() const
{
return m_ClockDriftMgr.m_nClientTick;
}
inline int CBaseClientState::GetServerTickCount() const
{
return m_ClockDriftMgr.m_nServerTick;
}
inline void CBaseClientState::SetServerTickCount( int tick )
{
m_ClockDriftMgr.m_nServerTick = tick;
}
inline void CBaseClientState::SetClientAndServerTickCount( int tick )
{
m_ClockDriftMgr.m_nServerTick = m_ClockDriftMgr.m_nClientTick = tick;
}
#endif // BASECLIENTSTATE_H