2021-07-24 21:11:47 -07:00

565 lines
11 KiB
C++

//========= Copyright © 1996-2005, Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
// $NoKeywords: $
//
//=============================================================================//
// netadr.h
#ifndef NS_ADDRESS_H
#define NS_ADDRESS_H
#pragma once
#include "tier0/platform.h"
#include "tier0/dbg.h"
#include "tier1/netadr.h"
#include "steam/steamclientpublic.h" // for CSteamID
#include "tier1/strtools.h" // V_memset
#if defined( NO_STEAM )
typedef CSteamID uint64;
#endif
enum PeerToPeerAddressType_t
{
P2P_STEAMID,
};
// Build int subchannel types
// FIXME - these really are game specific. I wish we could standardize!
enum SteamPeerToPeerChannel_t
{
STEAM_P2P_GAME_CLIENT = 0,
STEAM_P2P_GAME_SERVER = 1,
STEAM_P2P_LOBBY = 2,
STEAM_P2P_HLTV = 3,
STEAM_P2P_HLTV1 = 4,
};
class CPeerToPeerAddress
{
public:
CPeerToPeerAddress ( void )
: m_AddrType( P2P_STEAMID )
, m_steamChannel( STEAM_P2P_GAME_CLIENT )
{}
void Clear ( void )
{
m_AddrType = P2P_STEAMID;
m_steamID.Clear();
m_steamChannel = STEAM_P2P_GAME_CLIENT;
}
void SetSteamChannel( int nChannel )
{
m_steamChannel = nChannel;
}
int GetSteamChannel() const
{
return m_steamChannel;
}
const CSteamID &GetSteamID() const
{
return m_steamID;
}
CPeerToPeerAddress ( const CSteamID &steamID, int nSteamChannel )
: m_AddrType ( P2P_STEAMID )
, m_steamID ( steamID )
, m_steamChannel( nSteamChannel )
{
}
private:
// disallow since we can't deal with steamchannel
CPeerToPeerAddress &operator=(const CSteamID &steamID);
public:
// Like operator =
CPeerToPeerAddress &SetFromSteamID( const CSteamID &steamID, int nSteamChannel )
{
m_AddrType = P2P_STEAMID;
m_steamID = steamID;
m_steamChannel = nSteamChannel;
return *this;
}
bool IsValid ( void ) const
{
switch ( m_AddrType )
{
case P2P_STEAMID:
return m_steamID.IsValid ( );
};
return false;
}
const char *ToString ( void ) const
{
switch ( m_AddrType )
{
case P2P_STEAMID:
return m_steamID.Render ( );
};
return "";
}
bool CompareAdr ( const CPeerToPeerAddress &other, bool onlyBase = false ) const
{
if ( m_AddrType != -other.m_AddrType )
return false;
switch ( m_AddrType )
{
case P2P_STEAMID:
return ((m_steamID == other.m_steamID) && (m_steamChannel == other.m_steamChannel));
};
return false;
}
bool operator==(const CPeerToPeerAddress &rhs) const
{
return CompareAdr( rhs, false );
}
PeerToPeerAddressType_t GetAddressType ( void ) const
{
return m_AddrType;
}
template <typename T> bool IsType ( void ) const
{
return false;
}
template <typename T> T *Get ( void ) { return NULL; }
template <typename T> const T *Get ( void ) const { return NULL; }
template <typename T> T &AsType ( void ) { static T dummy; return dummy; }
template <typename T> const T &AsType ( void ) const { static T dummy; return dummy; }
private:
CSteamID m_steamID;
int m_steamChannel; // SteamID channel (like a port number to disambiguate multiple connections)
PeerToPeerAddressType_t m_AddrType;
};
template <>
inline bool CPeerToPeerAddress::IsType<CSteamID> ( void ) const
{
return (m_AddrType == P2P_STEAMID);
}
template <>
inline CSteamID * CPeerToPeerAddress::Get<CSteamID> ( void )
{
return IsType<CSteamID> ( ) ? &m_steamID : NULL;
}
template <>
inline const CSteamID * CPeerToPeerAddress::Get<CSteamID> ( void ) const
{
return IsType<CSteamID> ( ) ? &m_steamID : NULL;
}
template <>
inline CSteamID &CPeerToPeerAddress::AsType<CSteamID> ( void )
{
Assert ( IsType<CSteamID> ( ) );
return m_steamID;
}
template <>
inline const CSteamID &CPeerToPeerAddress::AsType<CSteamID> ( void ) const
{
Assert ( IsType<CSteamID> ( ) );
return m_steamID;
}
enum NetworkSystemAddressType_t
{
NSAT_NETADR,
NSAT_P2P,
NSAT_PROXIED_GAMESERVER, // Client proxied through Steam Datagram Transport
NSAT_PROXIED_CLIENT, // Client proxied through Steam Datagram Transport
};
// Used to represent a remote address which can be a network address or SteamID
// Basically if m_steamID.IsValid() then we send to that otherwise we use the m_adr
struct ns_address
{
netadr_t m_adr; // ip:port and network type (NULL/IP/BROADCAST/etc).
CPeerToPeerAddress m_steamID; // SteamID destination
NetworkSystemAddressType_t m_AddrType;
ns_address() : m_AddrType( NSAT_NETADR ){}
~ns_address()
{
Clear();
}
void Clear ( )
{
m_AddrType = NSAT_NETADR;
m_adr.Clear ( );
m_steamID.Clear ( );
}
ns_address ( const netadr_t &other )
: m_AddrType ( NSAT_NETADR )
, m_adr ( other )
{
m_steamID.Clear();
}
ns_address& operator=(const netadr_t &other)
{
Clear ( );
m_AddrType = NSAT_NETADR;
m_adr = other;
return *this;
}
ns_address ( const CPeerToPeerAddress &steamID )
: m_AddrType ( NSAT_P2P )
, m_steamID ( steamID )
{
m_adr.Clear ( );
}
ns_address& operator=(const CPeerToPeerAddress &steamID)
{
Clear ( );
m_AddrType = NSAT_P2P;
m_steamID = steamID;
return *this;
}
ns_address &SetFromSteamID ( const CSteamID &steamID, int nSteamChannel )
{
Clear ( );
m_AddrType = NSAT_P2P;
m_steamID.SetFromSteamID( steamID, nSteamChannel );
return *this;
}
bool IsLoopback ( void ) const
{
return (m_AddrType == NSAT_NETADR) && m_adr.IsLoopback ( );
}
bool IsLocalhost ( void ) const
{
return (m_AddrType == NSAT_NETADR) && m_adr.IsLocalhost ( );
}
bool IsBroadcast( void ) const
{
return (m_AddrType == NSAT_NETADR) && m_adr.GetType() == NA_BROADCAST;
}
bool IsReservedAdr( void ) const
{
return (m_AddrType == NSAT_NETADR) && m_adr.IsReservedAdr();
}
bool IsNull ( void ) const
{
switch ( m_AddrType )
{
case NSAT_NETADR:
return m_adr.GetType ( ) == NA_NULL;
case NSAT_P2P:
case NSAT_PROXIED_GAMESERVER:
case NSAT_PROXIED_CLIENT:
return !m_steamID.IsValid ( );
};
return true;
}
bool IsValid ( ) const
{
switch ( m_AddrType )
{
case NSAT_NETADR:
return m_adr.IsValid ( );
case NSAT_P2P:
case NSAT_PROXIED_GAMESERVER:
case NSAT_PROXIED_CLIENT:
return m_steamID.IsValid ( );
};
return false;
}
bool CompareAdr ( const ns_address &other, bool onlyBase = false ) const
{
if ( m_AddrType != other.m_AddrType )
return false;
switch ( m_AddrType )
{
case NSAT_NETADR:
return m_adr.CompareAdr ( other.m_adr, onlyBase );
case NSAT_P2P:
case NSAT_PROXIED_GAMESERVER:
case NSAT_PROXIED_CLIENT:
return m_steamID.CompareAdr( other.m_steamID, onlyBase );
};
return false;
}
bool operator==( const ns_address &rhs ) const
{
return CompareAdr( rhs, false );
}
bool operator!=( const ns_address &rhs ) const
{
return !CompareAdr( rhs, false );
}
NetworkSystemAddressType_t GetAddressType ( void ) const
{
return m_AddrType;
}
void SetAddrType( NetworkSystemAddressType_t type )
{
Clear();
m_AddrType = type;
}
bool SetFromSockadr ( const struct sockaddr * s )
{
m_AddrType = NSAT_NETADR;
m_steamID.Clear ( );
return m_adr.SetFromSockadr ( s );
}
unsigned int GetIP ( void ) const
{
return (m_AddrType == NSAT_NETADR) ? m_adr.GetIPHostByteOrder( ) : 0;
}
unsigned short GetPort ( void ) const
{
return (m_AddrType == NSAT_NETADR) ? m_adr.GetPort ( ) : 0;
}
template <typename T> bool IsType ( void ) const
{
//defer to subtype
switch ( m_AddrType )
{
case NSAT_P2P:
return m_steamID.IsType<T> ( );
};
return false;
}
template <typename T> T *Get ( void )
{
//defer to subtype
switch ( m_AddrType )
{
case NSAT_P2P:
return m_steamID.Get<T> ( );
};
return NULL;
}
template <typename T> const T *Get ( void ) const
{
//defer to subtype
switch ( m_AddrType )
{
case NSAT_P2P:
return m_steamID.Get<T> ( );
};
return NULL;
}
template <typename T> T &AsType ( void )
{
//defer to subtype
switch ( m_AddrType )
{
case NSAT_P2P:
return m_steamID.AsType<T> ( );
};
Assert ( false );
static T dummy;
return dummy;
}
template <typename T> const T &AsType ( void ) const
{
//defer to subtype
switch ( m_AddrType )
{
case NSAT_P2P:
return m_steamID.AsType<T> ( );
};
Assert ( false );
static T dummy;
return dummy;
}
bool SetFromString( const char *s )
{
Clear();
if ( !s )
return false;
bool bProxied = false;
if ( *s == '=' )
{
++s;
bProxied = true;
}
if ( *s == '[' )
{
CSteamID tempSteamID( s );
if ( !tempSteamID.IsValid() )
return false;
if ( bProxied && tempSteamID.BGameServerAccount() )
{
m_AddrType = NSAT_PROXIED_GAMESERVER;
m_steamID.SetFromSteamID( tempSteamID, STEAM_P2P_GAME_SERVER );
}
else
{
m_AddrType = bProxied ? NSAT_PROXIED_CLIENT : NSAT_P2P;
m_steamID.SetFromSteamID( tempSteamID, STEAM_P2P_GAME_CLIENT );
}
s = strchr( s, ']' );
int nChannel = -1;
if ( s && s[1] == ':' && sscanf( s+2, "%d", &nChannel ) == 1 && nChannel >= 0 )
{
m_steamID.SetSteamChannel( nChannel );
}
return true;
}
if ( !bProxied && m_adr.SetFromString( s, true ) )
{
m_AddrType = NSAT_NETADR;
return true;
}
return false;
}
};
template <>
inline bool ns_address::IsType<netadr_t> ( void ) const
{
return (m_AddrType == NSAT_NETADR);
}
template <>
inline bool ns_address::IsType<CPeerToPeerAddress> ( void ) const
{
return (m_AddrType == NSAT_P2P);
}
template <>
inline netadr_t *ns_address::Get<netadr_t> ( void )
{
return IsType<netadr_t> ( ) ? &m_adr : NULL;
}
template <>
inline const netadr_t *ns_address::Get<netadr_t> ( void ) const
{
return IsType<netadr_t> ( ) ? &m_adr : NULL;
}
template <>
inline CPeerToPeerAddress *ns_address::Get<CPeerToPeerAddress> ( void )
{
return IsType<CPeerToPeerAddress> ( ) ? &m_steamID : NULL;
}
template <>
inline const CPeerToPeerAddress *ns_address::Get<CPeerToPeerAddress> ( void ) const
{
return IsType<CPeerToPeerAddress> ( ) ? &m_steamID : NULL;
}
template <>
inline netadr_t &ns_address::AsType<netadr_t> ( void )
{
Assert ( IsType<netadr_t> ( ) );
return m_adr;
}
template <>
inline const netadr_t &ns_address::AsType<netadr_t> ( void ) const
{
Assert ( IsType<netadr_t> ( ) );
return m_adr;
}
template <>
inline CPeerToPeerAddress &ns_address::AsType<CPeerToPeerAddress> ( void )
{
Assert ( IsType<CPeerToPeerAddress> ( ) );
return m_steamID;
}
template <>
inline const CPeerToPeerAddress &ns_address::AsType<CPeerToPeerAddress> ( void ) const
{
Assert ( IsType<CPeerToPeerAddress> ( ) );
return m_steamID;
}
/// Utility class used to render an address
class ns_address_render
{
char m_buf[64];
public:
ns_address_render( const ns_address &a )
{
switch ( a.m_AddrType )
{
case NSAT_NETADR:
a.m_adr.ToString( m_buf, sizeof( m_buf ) );
break;
case NSAT_P2P:
V_sprintf_safe( m_buf, "%s:%d", a.m_steamID.ToString(), a.m_steamID.GetSteamChannel() );
break;
case NSAT_PROXIED_CLIENT:
case NSAT_PROXIED_GAMESERVER:
V_sprintf_safe( m_buf, "=%s:%d", a.m_steamID.ToString(), a.m_steamID.GetSteamChannel() );
break;
default:
m_buf[0] = '\0';
};
}
const char *String() const { return m_buf; }
};
#endif // NS_ADDRESS_H