source-engine/engine/matchmakingshared.cpp

2365 lines
59 KiB
C++
Raw Permalink Normal View History

2020-04-22 16:56:21 +00:00
//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose: Handles joining clients together in a matchmaking session before a multiplayer
// game, tracking new players and dropped players during the game, and reporting
// game results and stats after the game is complete.
//
//=============================================================================//
#include "proto_oob.h"
#include "matchmaking.h"
#include "matchmakingqos.h"
#include "Session.h"
#include "vgui_baseui_interface.h"
#include "cdll_engine_int.h"
#include "convar.h"
#include "cmd.h"
#include "iclient.h"
#include "server.h"
#include "host.h"
#if defined( _X360 )
#include "xbox/xbox_win32stubs.h"
#include "audio/private/snd_dev_xaudio.h"
#include "audio_pch.h"
#endif
// memdbgon must be the last include file in a .cpp file!!!
#include "tier0/memdbgon.h"
static CMatchmaking s_Matchmaking;
CMatchmaking *g_pMatchmaking = &s_Matchmaking;
extern IVEngineClient *engineClient;
extern IXboxSystem *g_pXboxSystem;
// Expose an interface for GameUI
EXPOSE_SINGLE_INTERFACE_GLOBALVAR( CMatchmaking, IMatchmaking, VENGINE_MATCHMAKING_VERSION, s_Matchmaking );
bool Channel_LessFunc( const uint &a, const uint &b )
{
return a < b;
}
//-----------------------------------------------------------------------------
// Purpose: Constructor
//-----------------------------------------------------------------------------
CMatchmaking::CMatchmaking() : m_Channels( Channel_LessFunc )
{
m_bPreventFullServerStartup = false;
m_bCleanup = false;
m_bEnteredLobby = false;
m_nTotalTeams = 0;
m_pSearchResults = NULL;
m_pSessionKeys = new KeyValues( "SessionKeys" );
m_Session.SetParent( this );
m_CurrentState = MMSTATE_INITIAL;
m_InviteState = INVITE_NONE;
memset( &m_InviteWaitingInfo, 0, sizeof( m_InviteWaitingInfo ) );
}
CMatchmaking::~CMatchmaking()
{
Cleanup();
m_pSessionKeys->deleteThis();
}
//-----------------------------------------------------------------------------
// Purpose: Cleanup the matchmaking class to enable re-entry
//-----------------------------------------------------------------------------
void CMatchmaking::Cleanup()
{
m_bInitialized = false;
m_bCleanup = false;
m_bEnteredLobby = false;
m_Host.Clear();
#ifdef _X360
if ( Audio_GetXVoice() )
{
CClientInfo *pLocal = NULL;
if ( m_bCreatedLocalTalker )
{
pLocal = &m_Local;
}
Audio_GetXVoice()->RemoveAllTalkers( pLocal );
}
#endif
m_bCreatedLocalTalker = false;
SetPreventFullServerStartup( false, "Cleanup\n" );
m_Session.ResetSession();
// TODO: Check on overlapped operations and cancel them
// g_pXboxSystem->CancelAsyncOperations();
ClearSearchResults();
m_pSessionKeys->Clear();
m_pGameServer = NULL;
Q_memset( m_Mutelist, 0, sizeof( m_Mutelist ) );
for ( int i = 0; i < MAX_PLAYERS_PER_CLIENT; ++i )
{
m_MutedBy[i].Purge();
}
m_Channels.RemoveAll();
m_SessionContexts.Purge();
m_SessionProperties.Purge();
m_PlayerStats.Purge();
m_Remote.PurgeAndDeleteElements();
m_nGameSize = 0;
m_nPrivateSlots = 0;
m_nSendCount = 0;
m_fHeartbeatInterval = HEARTBEAT_INTERVAL_SHORT;
m_fNextHeartbeatTime = GetTime();
}
int CMatchmaking::FindOrCreateContext( const uint id )
{
int idx = m_SessionContexts.InvalidIndex();
for ( int i = 0; i < m_SessionContexts.Count(); ++i )
{
if ( m_SessionContexts[i].dwContextId == id )
{
idx = i;
}
}
if ( !m_SessionContexts.IsValidIndex( idx ) )
{
idx = m_SessionContexts.AddToTail();
}
return idx;
}
int CMatchmaking::FindOrCreateProperty( const uint id )
{
int idx = m_SessionProperties.InvalidIndex();
for ( int i = 0; i < m_SessionProperties.Count(); ++i )
{
if ( m_SessionProperties[i].dwPropertyId == id )
{
idx = i;
}
}
if ( !m_SessionProperties.IsValidIndex( idx ) )
{
idx = m_SessionProperties.AddToTail();
}
return idx;
}
//-----------------------------------------------------------------------------
// Purpose: Add an additional property to the current session
//-----------------------------------------------------------------------------
void CMatchmaking::AddSessionProperty( const uint nType, const char *pID, const char *pValue, const char *pValueType )
{
KeyValues *pProperty = m_pSessionKeys->FindKey( pID, true );
pProperty->SetName( pID );
pProperty->SetInt( "type", nType );
pProperty->SetString( "valuestring", pValue );
pProperty->SetString( "valuetype", pValueType );
AddSessionPropertyInternal( pProperty );
}
//-----------------------------------------------------------------------------
// Purpose: Set properties and contexts for the current session
//-----------------------------------------------------------------------------
void CMatchmaking::SetSessionProperties( KeyValues *pPropertyKeys )
{
m_SessionContexts.RemoveAll();
m_SessionProperties.RemoveAll();
m_pSessionKeys->Clear();
pPropertyKeys->CopySubkeys( m_pSessionKeys );
for ( KeyValues *pProperty = m_pSessionKeys->GetFirstSubKey(); pProperty != NULL; pProperty = pProperty->GetNextKey() )
{
AddSessionPropertyInternal( pProperty );
}
}
//-----------------------------------------------------------------------------
// Purpose:
//-----------------------------------------------------------------------------
void CMatchmaking::AddSessionPropertyInternal( KeyValues *pProperty )
{
const char *pID = pProperty->GetName();
const char *pValue = pProperty->GetString( "valuestring" );
switch( pProperty->GetInt( "type" ) )
{
case SESSION_CONTEXT:
{
Msg( "Adding Context: %s : %s\n", pID, pValue );
int id = g_ClientDLL->GetPresenceID( pID );
int val = g_ClientDLL->GetPresenceID( pValue );
int idx = FindOrCreateContext( id );
XUSER_CONTEXT &ctx = m_SessionContexts[idx];
ctx.dwContextId = id;
ctx.dwValue = val;
// Set the display string for gameUI
char szBuffer[MAX_PATH];
g_ClientDLL->GetPropertyDisplayString( ctx.dwContextId, ctx.dwValue, szBuffer, sizeof( szBuffer ) );
pProperty->SetString( "displaystring", szBuffer );
// X360TBD: Such game specifics as this shouldn't be hard-coded
if ( m_CurrentState != MMSTATE_INITIAL && !Q_stricmp( pID, "CONTEXT_SCENARIO" ) )
{
// Set the scenario in our host data structure
Q_strncpy( m_HostData.scenario, szBuffer, sizeof( m_HostData.scenario ) );
UpdateSessionReplyData( XNET_QOS_LISTEN_SET_DATA );
}
}
break;
case SESSION_PROPERTY:
{
Msg( "Adding Property: %s : %s\n", pID, pValue );
if ( !Q_stricmp( pID, "PROPERTY_PRIVATE_SLOTS" ) )
{
// "Private Slots" is not a search criteria
m_nPrivateSlots = atoi( pValue );
break;
}
int id = g_ClientDLL->GetPresenceID( pID );
int idx = FindOrCreateProperty( id );
XUSER_PROPERTY &prop = m_SessionProperties[idx];
prop.dwPropertyId = id;
if ( !Q_stricmp( pProperty->GetString( "valuetype" ), "int" ) )
{
prop.value.nData = atoi( pValue );
prop.value.type = XUSER_DATA_TYPE_INT32;
}
// Build out the property keyvalues for gameUI
char szBuffer[MAX_PATH];
g_ClientDLL->GetPropertyDisplayString( prop.dwPropertyId, prop.value.nData, szBuffer, sizeof( szBuffer ) );
pProperty->SetString( "displaystring", szBuffer );
// X360TBD: Such game specifics as these shouldn't be so hard-coded
if ( !Q_stricmp( pID, "PROPERTY_GAME_SIZE" ) )
{
m_nGameSize = atoi( pValue );
}
if ( !Q_stricmp( pID, "PROPERTY_NUMBER_OF_TEAMS" ) )
{
m_nTotalTeams = atoi( pValue );
}
if ( m_CurrentState != MMSTATE_INITIAL && !Q_stricmp( pID, "PROPERTY_MAX_GAME_TIME" ) )
{
// Set the game time in our host data structure
m_HostData.gameTime = prop.value.nData;
UpdateSessionReplyData( XNET_QOS_LISTEN_SET_DATA );
}
}
break;
case SESSION_FLAG:
m_Session.SetFlag( g_ClientDLL->GetPresenceID( pID ) );
break;
default:
Warning( "Session option type %d not recognized/n", pProperty->GetInt( "type" ) );
break;
}
}
KeyValues *CMatchmaking::GetSessionProperties()
{
return m_pSessionKeys;
}
double CMatchmaking::GetTime()
{
return Plat_FloatTime();
}
//-----------------------------------------------------------------------------
// Purpose: At netchannel connection, register the messages
//-----------------------------------------------------------------------------
void CMatchmaking::ConnectionStart( INetChannel *chan )
{
REGISTER_MM_MSG( JoinResponse );
REGISTER_MM_MSG( ClientInfo );
REGISTER_MM_MSG( RegisterResponse );
REGISTER_MM_MSG( Migrate );
REGISTER_MM_MSG( Mutelist );
REGISTER_MM_MSG( Checkpoint );
REGISTER_MM_MSG( Heartbeat );
REGISTER_CLC_MSG( VoiceData );
}
//-----------------------------------------------------------------------------
// Purpose: Process a networked voice packet
//-----------------------------------------------------------------------------
bool CMatchmaking::ProcessVoiceData( CLC_VoiceData *pVoice )
{
char chReceived[4096];
DWORD dwLength = pVoice->m_nLength;
pVoice->m_DataIn.ReadBits( chReceived, dwLength );
if ( m_Session.IsHost() )
{
char chCopyBuffer[4096];
// Forward this message on to everyone else
pVoice->m_DataOut.StartWriting( chCopyBuffer, sizeof ( chCopyBuffer ) );
Q_memcpy( chCopyBuffer, chReceived, sizeof( chCopyBuffer ) );
pVoice->m_DataOut.SeekToBit( dwLength );
SendToRemoteClients( pVoice, true, pVoice->m_xuid );
}
// Playback the voice data locally through xaudio
#if defined ( _X360 )
if ( pVoice->m_xuid != m_Local.m_xuids[0] )
{
Audio_GetXVoice()->PlayIncomingVoiceData( pVoice->m_xuid, (byte*)chReceived, dwLength );
}
#endif
return true;
}
//-----------------------------------------------------------------------------
// Purpose: Delete channels that have been marked for deletion
//-----------------------------------------------------------------------------
void CMatchmaking::CleanupMarkedChannels()
{
// Clean up net channels that need to be deleted
for ( int i = 0; i < m_ChannelsToRemove.Count(); ++i )
{
INetChannel *pNetChannel = FindChannel( m_ChannelsToRemove[i] );
if ( pNetChannel )
{
if ( !m_Channels.Remove( m_ChannelsToRemove[i] ) )
{
Warning( "CleanupMarkedChannels: Failed to remove a channel!\n" );
}
}
}
m_ChannelsToRemove.Purge();
}
//-----------------------------------------------------------------------------
// Purpose: Channels can be flagged for deletion during packet processing.
// Now that processing is finished, delete them.
//-----------------------------------------------------------------------------
void CMatchmaking::PacketEnd()
{
CleanupMarkedChannels();
}
//-----------------------------------------------------------------------------
// Purpose: Find a specific client from a netchannel
//-----------------------------------------------------------------------------
CClientInfo *CMatchmaking::FindClient( netadr_t *adr )
{
CClientInfo *pClient = NULL;
unsigned int ip = adr->GetIPNetworkByteOrder();
if ( ip == m_Host.m_adr.GetIPNetworkByteOrder() )
{
pClient = &m_Host;
}
else
{
for ( int i = 0; i < m_Remote.Count(); ++i )
{
if ( ip == m_Remote[i]->m_adr.GetIPNetworkByteOrder() )
{
pClient = m_Remote[i];
break;
}
}
}
return pClient;
}
//-----------------------------------------------------------------------------
// Purpose: Find a specific client by his XUID
//-----------------------------------------------------------------------------
CClientInfo *CMatchmaking::FindClientByXUID( XUID xuid )
{
CClientInfo *pClient = NULL;
if ( xuid == m_Host.m_xuids[0] )
{
pClient = &m_Host;
}
else
{
for ( int i = 0; i < m_Remote.Count(); ++i )
{
if ( xuid == m_Remote[i]->m_xuids[0] )
{
pClient = m_Remote[i];
break;
}
}
}
return pClient;
}
//-----------------------------------------------------------------------------
// Purpose: Find a specific client's netchannel
//-----------------------------------------------------------------------------
INetChannel *CMatchmaking::FindChannel( const unsigned int ip )
{
INetChannel *pChannel = NULL;
int idx = m_Channels.Find( ip );
if ( idx != m_Channels.InvalidIndex() )
{
pChannel = m_Channels.Element( idx );
}
return pChannel;
}
//-----------------------------------------------------------------------------
// Purpose: Create a new netchannel
//-----------------------------------------------------------------------------
INetChannel *CMatchmaking::CreateNetChannel( netadr_t *adr )
{
INetChannel *pNewChannel = FindChannel( adr->GetIPNetworkByteOrder() );
if ( !pNewChannel )
{
pNewChannel = NET_CreateNetChannel( NS_MATCHMAKING, adr, "MATCHMAKING", this );
}
if( pNewChannel )
{
// Set a rate limit and other relevant properties
pNewChannel->SetTimeout( HEARTBEAT_TIMEOUT );
}
return pNewChannel;
}
//-----------------------------------------------------------------------------
// Purpose: Add a netchannel for a session client
//-----------------------------------------------------------------------------
INetChannel *CMatchmaking::AddRemoteChannel( netadr_t *adr )
{
INetChannel *pNetChannel = CreateNetChannel( adr );
if ( pNetChannel )
{
// Save this new channel
m_Channels.Insert( adr->GetIPNetworkByteOrder(), pNetChannel );
}
return pNetChannel;
}
//-----------------------------------------------------------------------------
// Purpose: Remove a netchannel for a session client
//-----------------------------------------------------------------------------
void CMatchmaking::RemoveRemoteChannel( netadr_t *adr, const char *pReason )
{
INetChannel *pNetChannel = FindChannel( adr->GetIPNetworkByteOrder() );
if ( pNetChannel )
{
m_Channels.Remove( adr->GetIPNetworkByteOrder() );
}
}
//-----------------------------------------------------------------------------
// Purpose: Mark a net channel to be removed
//-----------------------------------------------------------------------------
void CMatchmaking::MarkChannelForRemoval( netadr_t *adr )
{
m_ChannelsToRemove.AddToTail( adr->GetIPNetworkByteOrder() );
}
//-----------------------------------------------------------------------------
// Purpose: Set the timeout for a net channel
//-----------------------------------------------------------------------------
void CMatchmaking::SetChannelTimeout( netadr_t *adr, int timeout )
{
INetChannel *pChannel = FindChannel( adr->GetIPNetworkByteOrder() );
if ( pChannel )
{
Msg( "Setting new timeout for ip %d: %d\n", adr->GetIPNetworkByteOrder(), timeout );
pChannel->SetTimeout( timeout );
}
}
//-----------------------------------------------------------------------------
// Purpose: Send a net message to a specific address
//-----------------------------------------------------------------------------
void CMatchmaking::SendMessage( INetMessage *msg, netadr_t *adr, bool bVoice )
{
// Find the matching net channel
INetChannel *pChannel = FindChannel( adr->GetIPNetworkByteOrder() );
if ( pChannel )
{
pChannel->SendNetMsg( *msg, false, bVoice );
if ( !pChannel->Transmit() )
{
Msg( "Transmit failed\n" );
}
}
}
//-----------------------------------------------------------------------------
// Purpose: Send a net message to a specific client
//-----------------------------------------------------------------------------
void CMatchmaking::SendMessage( INetMessage *msg, CClientInfo *pClient, bool bVoice )
{
// Find the matching net channel
INetChannel *pChannel = FindChannel( pClient->m_adr.GetIPNetworkByteOrder() );
if ( pChannel )
{
pChannel->SendNetMsg( *msg, false, bVoice );
if ( !pChannel->Transmit() )
{
Msg( "Transmit failed\n" );
}
}
}
//-----------------------------------------------------------------------------
// Purpose: Send a net message to all remote clients
//-----------------------------------------------------------------------------
void CMatchmaking::SendToRemoteClients( INetMessage *msg, bool bVoice, XUID excludeXUID )
{
for ( int i = 0; i < m_Remote.Count(); ++i )
{
CClientInfo *pInfo = m_Remote[i];
if ( excludeXUID != -1 && pInfo->m_xuids[0] == excludeXUID )
continue;
SendMessage( msg, m_Remote[i], true );
}
}
//-----------------------------------------------------------------------------
// Purpose: Send a heartbeat to a specific client
//-----------------------------------------------------------------------------
bool CMatchmaking::SendHeartbeat( CClientInfo *pClient )
{
if ( pClient->m_adr.GetIPNetworkByteOrder() == 0 )
return false;
// Check for timeout
INetChannel *pChannel = FindChannel( pClient->m_adr.GetIPNetworkByteOrder() );
if ( pChannel )
{
// Msg( "Sending HB\n" );
if ( pChannel->IsTimedOut() )
{
ClientDropped( pClient );
return false;
}
// Send a heartbeat to the client
MM_Heartbeat beat;
pChannel->SendNetMsg( beat );
pChannel->Transmit();
}
return true;
}
//-----------------------------------------------------------------------------
// Purpose: Transmit regular messages to keep the connection alive
//-----------------------------------------------------------------------------
void CMatchmaking::SendHeartbeat()
{
double time = GetTime();
if ( time < m_fNextHeartbeatTime )
return;
m_fNextHeartbeatTime = time + m_fHeartbeatInterval;
if ( m_Session.IsHost() )
{
for ( int i = 0; i < m_Remote.Count(); ++i )
{
SendHeartbeat( m_Remote[i] );
}
}
else
{
SendHeartbeat( &m_Host );
}
}
//-----------------------------------------------------------------------------
// Purpose: Look up a player by name
//-----------------------------------------------------------------------------
static uint64 FindPlayerByName( CClientInfo *pClient, const char *pName )
{
for ( int i = 0; i < XUSER_MAX_COUNT; ++i )
{
if ( pClient->m_xuids[i] && !Q_stricmp( pClient->m_szGamertags[i], pName ) )
{
return pClient->m_xuids[i];
}
}
return 0;
}
//-----------------------------------------------------------------------------
// Purpose: Get an xuid from a CBasePlayer id
//-----------------------------------------------------------------------------
uint64 CMatchmaking::PlayerIdToXuid( int playerId )
{
uint64 ret = 0;
player_info_t info;
if ( engineClient->GetPlayerInfo( playerId, &info ) )
{
// find the client with a matching name
for ( int i = 0; i < m_Remote.Count(); ++i )
{
ret = FindPlayerByName( m_Remote[i], info.name );
if ( ret )
{
break;
}
}
}
if ( !ret )
{
// Try ourselves
ret = FindPlayerByName( &m_Local, info.name );
}
if ( !ret )
{
// Try the host
ret = FindPlayerByName( &m_Host, info.name );
}
return ret;
}
bool CMatchmaking::GameIsActive()
{
return m_CurrentState > MMSTATE_GAME_ACTIVE;
}
bool CMatchmaking::GameIsLocked()
{
return ( m_Session.IsArbitrated() && m_CurrentState > MMSTATE_GAME_LOCKED );
}
bool CMatchmaking::ConnectedToServer()
{
return engineClient->IsConnected();
}
bool CMatchmaking::IsInMigration()
{
return ( m_CurrentState >= MMSTATE_HOSTMIGRATE_STARTINGMIGRATION &&
m_CurrentState <= MMSTATE_HOSTMIGRATE_WAITINGFORHOST );
}
bool CMatchmaking::IsAcceptingConnections()
{
if ( !m_Session.IsHost() ||
m_CurrentState < MMSTATE_ACCEPTING_CONNECTIONS ||
m_CurrentState == MMSTATE_PREGAME ||
m_CurrentState == MMSTATE_LOADING ||
GameIsLocked() )
{
return false;
}
return true;
}
bool CMatchmaking::IsServer()
{
// for now, the host is the server
return m_Session.IsHost();
}
//-----------------------------------------------------------------------------
// Purpose: Helpers to convert between CClientInfo and MM_ClientInfo
//-----------------------------------------------------------------------------
void CMatchmaking::ClientInfoToNetMessage( MM_ClientInfo *pInfo, const CClientInfo *pClient )
{
pInfo->m_id = pClient->m_id;
pInfo->m_xnaddr = pClient->m_xnaddr;
pInfo->m_cPlayers = pClient->m_cPlayers;
pInfo->m_bInvited = pClient->m_bInvited;
Q_memcpy( pInfo->m_xuids, pClient->m_xuids, sizeof( pInfo->m_xuids ) );
Q_memcpy( pInfo->m_cVoiceState, pClient->m_cVoiceState, sizeof( pInfo->m_cVoiceState ) );
Q_memcpy( pInfo->m_iTeam, pClient->m_iTeam, sizeof( pInfo->m_iTeam ) );
Q_memcpy( pInfo->m_iControllers, pClient->m_iControllers, sizeof( pInfo->m_iControllers ) );
Q_memcpy( pInfo->m_szGamertags, pClient->m_szGamertags, sizeof( pInfo->m_szGamertags ) );
}
void CMatchmaking::NetMessageToClientInfo( CClientInfo *pClient, const MM_ClientInfo *pInfo )
{
pClient->m_id = pInfo->m_id;
pClient->m_xnaddr = pInfo->m_xnaddr;
pClient->m_cPlayers = pInfo->m_cPlayers;
pClient->m_bInvited = pInfo->m_bInvited;
#if defined( _X360 )
IN_ADDR winaddr;
XNKID xid = m_Session.GetSessionId();
if ( XNetXnAddrToInAddr( &pClient->m_xnaddr, &xid, &winaddr ) != 0 )
{
Warning( "Error resolving client IP\n" );
}
pClient->m_adr.SetType( NA_IP );
pClient->m_adr.SetIPAndPort( winaddr.S_un.S_addr, PORT_MATCHMAKING );
#endif
Q_memcpy( pClient->m_xuids, pInfo->m_xuids, sizeof( pClient->m_xuids ) );
Q_memcpy( pClient->m_cVoiceState, pInfo->m_cVoiceState, sizeof( pClient->m_cVoiceState ) );
Q_memcpy( pClient->m_iTeam, pInfo->m_iTeam, sizeof( pClient->m_iTeam ) );
Q_memcpy( pClient->m_iControllers, pInfo->m_iControllers, sizeof( pClient->m_iControllers ) );
Q_memcpy( pClient->m_szGamertags, pInfo->m_szGamertags, sizeof( pClient->m_szGamertags ) );
}
//----------------------------------------
//
// Host/Client Shared
//
//----------------------------------------
//-----------------------------------------------------------------------------
// Purpose: Set up the properties of the local client machine
//-----------------------------------------------------------------------------
bool CMatchmaking::InitializeLocalClient( bool bIsHost )
{
Q_memset( &m_Local, 0, sizeof( m_Local ) );
m_Local.m_bInvited = bIsHost;
#if defined( _X360 )
while( XNetGetTitleXnAddr( &m_Local.m_xnaddr ) == XNET_GET_XNADDR_PENDING )
;
// machine id
if ( 0 != XNetXnAddrToMachineId( &m_Local.m_xnaddr, &m_Local.m_id ) )
{
// User isn't signed in to live, use their xuid instead
XUserGetXUID( XBX_GetPrimaryUserId(), &m_Local.m_id );
}
m_Local.m_cPlayers = 0;
// Set up the players
for ( uint i = 0; i < MAX_PLAYERS_PER_CLIENT; ++i )
{
// We currently only allow one player per console
if ( i != XBX_GetPrimaryUserId() )
{
continue;
}
// xuid
uint ret = XUserGetXUID( i, &m_Local.m_xuids[m_Local.m_cPlayers] );
if ( ret == ERROR_NO_SUCH_USER )
{
continue;
}
else if ( ret != ERROR_SUCCESS )
{
return false;
}
// gamertag
ret = XUserGetName( XBX_GetPrimaryUserId(), m_Local.m_szGamertags[m_Local.m_cPlayers], MAX_PLAYER_NAME_LENGTH );
if ( ret != ERROR_SUCCESS )
{
return false;
}
m_Local.m_szGamertags[m_Local.m_cPlayers][MAX_PLAYER_NAME_LENGTH - 1] = '\0';
// Set the player's name in the game
char szNameCmd[MAX_PLAYER_NAME_LENGTH + 16];
Q_snprintf( szNameCmd, sizeof( szNameCmd ), "name %s", m_Local.m_szGamertags[m_Local.m_cPlayers] );
engineClient->ClientCmd( szNameCmd );
m_Local.m_iControllers[m_Local.m_cPlayers] = i;
m_Local.m_iTeam[m_Local.m_cPlayers] = -1;
m_Local.m_cVoiceState[m_Local.m_cPlayers] = 0;
// number of players on this console
++m_Local.m_cPlayers;
}
// Source can only support one player per console.
if( m_Local.m_cPlayers > 1 )
{
Warning( "Too many players on this console\n" );
return false;
}
// Set up the host data that gets sent back to searching clients
// By default, the first player is considered the host
Q_strncpy( m_HostData.hostName, m_Local.m_szGamertags[0], sizeof( m_HostData.hostName ) );
m_HostData.gameState = GAMESTATE_INLOBBY;
m_HostData.xuid = m_Local.m_xuids[ XBX_GetPrimaryUserId() ];
#endif
m_bInitialized = true;
return true;
}
//-----------------------------------------------------------------------------
// Purpose: Connection to the game server has been established, so we can
// add the local players to the teams that were setup in the lobby.
//-----------------------------------------------------------------------------
void CMatchmaking::AddLocalPlayersToTeams()
{
if ( !m_bInitialized || XBX_GetPrimaryUserId() == INVALID_USER_ID )
return;
if ( m_Local.m_iTeam[0] == -1 )
return;
// Convert the team number into a team name
char szTeamName[32];
uint id = g_ClientDLL->GetPresenceID( "PROPERTY_TEAM" );
g_ClientDLL->GetPropertyDisplayString( id, m_Local.m_iTeam[0], szTeamName, sizeof( szTeamName ) );
Msg( "Joining team: %s\n", szTeamName );
char cmd[32];
Q_snprintf( cmd, sizeof( cmd ), "jointeam_nomenus %s", szTeamName );
engineClient->ClientCmd( cmd );
}
//-----------------------------------------------------------------------------
// Purpose: Map loading is completed - restore full communication
//-----------------------------------------------------------------------------
void CMatchmaking::OnLevelLoadingFinished()
{
// This functions gets called from some odd places
if ( m_CurrentState != MMSTATE_CONNECTED_TO_SERVER )
return;
// Test code to force a disconnect at end of map load
// if ( !IsServer() )
// {
// char cmd[MAX_PATH];
// Q_snprintf( cmd, sizeof( cmd ), "connect 127.0.0.1\n" );
// Cbuf_AddText( cmd );
// return;
// }
SwitchToState( MMSTATE_INGAME );
MM_Checkpoint msg;
msg.m_Checkpoint = MM_Checkpoint::CHECKPOINT_LOADING_COMPLETE;
if ( m_Session.IsHost() )
{
if ( !m_Session.IsArbitrated() )
{
// Re-enable response to probes
m_HostData.gameState = GAMESTATE_INPROGRESS;
UpdateSessionReplyData( XNET_QOS_LISTEN_ENABLE|XNET_QOS_LISTEN_SET_DATA );
}
// Reset netchannel timeouts for any clients that are also finished loading
for ( int i = 0; i < m_Remote.Count(); ++i )
{
CClientInfo *pClient = m_Remote[i];
if ( pClient->m_bLoaded )
{
// Send a reply and reset the netchannel timeout
SendMessage( &msg, pClient );
SetChannelTimeout( &pClient->m_adr, HEARTBEAT_TIMEOUT );
}
}
}
else
{
// Tell the host we're finished loading
SendMessage( &msg, &m_Host );
}
}
//-----------------------------------------------------------------------------
// Purpose: Process packet from another client
//-----------------------------------------------------------------------------
bool CMatchmaking::ProcessConnectionlessPacket( netpacket_t *pPacket )
{
Assert( pPacket );
bf_read &msg = pPacket->message;
int type = msg.ReadByte();
switch( type )
{
case PTH_SYSTEMLINK_SEARCH:
HandleSystemLinkSearch( pPacket );
break;
case HTP_SYSTEMLINK_REPLY:
HandleSystemLinkReply( pPacket );
break;
case PTH_CONNECT:
HandleJoinRequest( pPacket );
break;
default:
break;
}
return true;
}
//-----------------------------------------------------------------------------
// Purpose: Process an info update about another client
//-----------------------------------------------------------------------------
bool CMatchmaking::ProcessClientInfo( MM_ClientInfo *pInfo )
{
CClientInfo *pClient = NULL;
bool bHost = false;
if ( m_CurrentState == MMSTATE_INITIAL )
{
// Session has been reset, this is a stale message
Msg( "Received MM_ClientInfo with MMSTATE_INITIAL\n" );
return true;
}
if ( pInfo->m_id == m_Local.m_id )
{
if ( pInfo->m_cPlayers == 0 )
{
if ( m_CurrentState != MMSTATE_SESSION_DISCONNECTING )
{
// We've been kicked
KickPlayerFromSession( 0 );
SessionNotification( SESSION_NOTIFY_CLIENT_KICKED );
}
return true;
}
else
{
pClient = &m_Local;
}
}
// Check against our host id
if ( pInfo->m_id == m_Host.m_id )
{
pClient = &m_Host;
bHost = true;
}
else
{
// Look for the client in our remote list
for ( int i = 0; i < m_Remote.Count(); ++i )
{
if ( m_Remote[i]->m_id == pInfo->m_id )
{
pClient = m_Remote[i];
break;
}
}
}
// If we didn't find it, this must be a new client
if ( !pClient && pInfo->m_cPlayers != 0 )
{
Msg( "New client. %s\n", pInfo->ToString() );
pClient = new CClientInfo();
m_Remote.AddToTail( pClient );
// Copy the new client info
NetMessageToClientInfo( pClient, pInfo );
AddPlayersToSession( pClient );
SendPlayerInfoToLobby( pClient );
}
else
{
// We're updating an existing client
if ( pInfo->m_cPlayers )
{
// Cache off the old client info, as pClient gets updated through this function
CClientInfo tempClient = *pClient;
// Check for player changes
if ( Q_memcmp( &tempClient.m_xuids, pInfo->m_xuids, sizeof( tempClient.m_xuids ) ) )
{
// Remove the old players and add the new
RemovePlayersFromSession( pClient );
NetMessageToClientInfo( pClient, pInfo );
AddPlayersToSession( pClient );
}
// Check for team changes
for ( int i = 0; i < pInfo->m_cPlayers; ++i )
{
if ( pInfo->m_iTeam[i] != tempClient.m_iTeam[i] || pInfo->m_cVoiceState[i] != tempClient.m_cVoiceState[i] )
{
// X360TBD: send real "ready" setting, or remove entirely?
EngineVGui()->UpdatePlayerInfo( pInfo->m_xuids[i], pInfo->m_szGamertags[i], pInfo->m_iTeam[i], pInfo->m_cVoiceState[i], GetPlayersNeeded(), bHost );
}
}
// Store the new info
NetMessageToClientInfo( pClient, pInfo );
if ( m_Session.IsHost() )
{
SendToRemoteClients( pInfo );
}
}
else
{
// A client has been dropped
ClientDropped( pClient );
}
}
return true;
}
//-----------------------------------------------------------------------------
// Purpose: A connection was lost - respond accordingly
//-----------------------------------------------------------------------------
void CMatchmaking::ClientDropped( CClientInfo *pClient )
{
if ( !pClient )
{
Warning( "Null client pointer in ClientDropped!\n" );
return;
}
if ( m_CurrentState == MMSTATE_SESSION_CONNECTING )
{
// Not really dropped, we just failed to connect to the host
SessionNotification( SESSION_NOTIFY_CONNECT_NOTAVAILABLE );
return;
}
Warning( "Dropped player: %llu!", pClient->m_id );
// Do this first, before the team assignment gets cleared
RemovePlayersFromSession( pClient );
// Remove all players from the lobby
for ( int i = 0; i < pClient->m_cPlayers; ++i )
{
pClient->m_iTeam[i] = -1;
}
SendPlayerInfoToLobby( pClient );
MarkChannelForRemoval( &pClient->m_adr );
if ( pClient == &m_Host )
{
// The host was lost
if ( m_Session.IsSystemLink() )
{
// Can't migrate system link sessions
SessionNotification( SESSION_NOTIFY_LOST_HOST );
}
else
{
// X360TBD: Migration still doesn't work correctly
SessionNotification( SESSION_NOTIFY_LOST_HOST );
/*
// Start migrating
if ( !IsInMigration() )
{
m_PreMigrateState = m_CurrentState;
StartHostMigration();
}
*/
}
}
else
{
if ( m_Session.IsHost() )
{
// Send a disconnect ack back to the client
MM_Checkpoint msg;
msg.m_Checkpoint = MM_Checkpoint::CHECKPOINT_SESSION_DISCONNECT;
SendMessage( &msg, &pClient->m_adr );
// Tell everyone else this client is gone
MM_ClientInfo droppedPlayer;
ClientInfoToNetMessage( &droppedPlayer, pClient );
droppedPlayer.m_cPlayers = 0;
SendToRemoteClients( &droppedPlayer );
for ( int i = 0; i < sv.GetClientCount(); i++ )
{
IClient *pIClient = sv.GetClient(i);
bool bFound = false;
if ( pIClient )
{
for ( int j = 0; j < pClient->m_cPlayers; ++j )
{
if ( pClient->m_xuids[j] == 0 )
continue;
if ( Q_stricmp( pIClient->GetClientName(), pClient->m_szGamertags[j] ) == 0 )
{
bFound = true;
pIClient->Disconnect( "Timed Out" );
break;
}
}
}
if ( bFound == true )
break;
}
}
}
if ( m_Remote.FindAndRemove( pClient ) )
{
delete pClient;
}
}
//-----------------------------------------------------------------------------
// Purpose: A player is leaving the session
//-----------------------------------------------------------------------------
void CMatchmaking::PerformDisconnect()
{
if ( m_CurrentState == MMSTATE_SESSION_DISCONNECTING )
{
if ( ConnectedToServer() )
{
engineClient->ExecuteClientCmd( "disconnect" );
EngineVGui()->ActivateGameUI();
}
if ( m_bEnteredLobby )
{
EngineVGui()->SessionNotification( SESSION_NOTIFY_WELCOME );
}
SwitchToState( MMSTATE_INITIAL );
}
}
//-----------------------------------------------------------------------------
// Purpose: A player is leaving the session
//-----------------------------------------------------------------------------
void CMatchmaking::KickPlayerFromSession( uint64 id )
{
MM_ClientInfo droppedPlayer;
if ( m_Local.m_xuids[0] == id || id == 0 )
{
// We've been kicked, or voluntarily left the session
ClientInfoToNetMessage( &droppedPlayer, &m_Local );
droppedPlayer.m_cPlayers = 0;
if ( m_Session.IsHost() )
{
// Tell all the clients
SendToRemoteClients( &droppedPlayer );
}
else
{
// tell the host to drop us
SendMessage( &droppedPlayer, &m_Host );
}
// Prepare to close the session and reset
SwitchToState( MMSTATE_SESSION_DISCONNECTING );
}
else if ( m_Session.IsHost() )
{
// Host wants to kick a client
for ( int i = 0; i < m_Remote.Count(); ++i )
{
CClientInfo *pClient = m_Remote[i];
for ( int j = 0; j < pClient->m_cPlayers; ++j )
{
if ( pClient->m_xuids[j] == id )
{
ClientInfoToNetMessage( &droppedPlayer, pClient );
droppedPlayer.m_cPlayers = 0;
SendMessage( &droppedPlayer, pClient );
return;
}
}
}
}
}
//-----------------------------------------------------------------------------
// Purpose: Add players to the session
//-----------------------------------------------------------------------------
void CMatchmaking::AddPlayersToSession( CClientInfo *pClient )
{
bool bIsLocal = false;
if ( &m_Local == pClient )
{
m_Session.JoinLocal( pClient );
bIsLocal = true;
}
else
{
m_Session.JoinRemote( pClient );
}
#if defined ( _X360 )
if ( Audio_GetXVoice() )
{
Audio_GetXVoice()->AddPlayerToVoiceList( pClient, bIsLocal );
if ( bIsLocal )
{
m_bCreatedLocalTalker = true;
}
}
#endif
UpdateMuteList();
}
//-----------------------------------------------------------------------------
// Purpose: Remove players from the session
//-----------------------------------------------------------------------------
void CMatchmaking::RemovePlayersFromSession( CClientInfo *pClient )
{
if ( !pClient->m_cPlayers )
return;
bool bIsLocal = false;
if ( &m_Local == pClient )
{
m_Session.RemoveLocal( pClient );
bIsLocal = true;
}
else
{
m_Session.RemoveRemote( pClient );
}
#if defined ( _X360 )
if ( Audio_GetXVoice() )
{
Audio_GetXVoice()->RemovePlayerFromVoiceList( pClient, bIsLocal );
}
#endif
UpdateMuteList();
}
//-----------------------------------------------------------------------------
// Purpose: Check if a client is muted
//-----------------------------------------------------------------------------
bool CMatchmaking::IsPlayerMuted( int iUserId, XUID playerId )
{
for ( int i = 0; i < MAX_PLAYERS; ++i )
{
if ( m_Mutelist[iUserId][i] == playerId )
{
return true;
}
}
if ( m_MutedBy[iUserId].HasElement( playerId ) )
return true;
return false;
}
//-----------------------------------------------------------------------------
// Purpose: Determine which clients should be muted for the local client
//-----------------------------------------------------------------------------
void CMatchmaking::UpdateMuteList()
{
// Process our mute list
MM_Mutelist msg;
GenerateMutelist( &msg );
// Send that to everyone else
if ( !m_Session.IsHost() )
{
SendMessage( &msg, &m_Host );
}
else
{
ProcessMutelist( &msg );
}
}
void Con_PrintTalkers( const CCommand &args )
{
g_pMatchmaking->PrintVoiceStatus();
}
void CMatchmaking::PrintVoiceStatus( void )
{
#ifdef _X360
int iRemoteClient = 0;
int numRemoteTalkers;
XUID remoteTalkers[MAX_PLAYERS];
Audio_GetXVoice()->GetRemoteTalkers( &numRemoteTalkers, remoteTalkers );
CClientInfo *pClient = &m_Local;
if ( m_Session.IsHost() == false )
{
pClient = &m_Host;
}
Msg( "Num Remote Talkers: %d\n", numRemoteTalkers );
bool bFound = false;
while ( pClient )
{
if ( pClient != &m_Local )
{
for ( int iRemote = 0; iRemote < numRemoteTalkers; iRemote++ )
{
if ( pClient->m_xuids[0] == remoteTalkers[iRemote] )
{
bFound = true;
break;
}
}
if ( bFound == true )
{
Msg( "Found a Talker: %s\n", pClient->m_szGamertags[0] );
}
else
{
Msg( "ALERT!!! %s not in Talker list\n", pClient->m_szGamertags[0] );
}
}
if ( iRemoteClient < m_Remote.Count() )
{
pClient = m_Remote[iRemoteClient];
iRemoteClient++;
}
else
{
pClient = NULL;
}
}
#endif
}
static ConCommand voice_printtalkers( "voice_printtalkers", Con_PrintTalkers, "voice debug.", FCVAR_DONTRECORD );
//-----------------------------------------------------------------------------
// Purpose: Update a client's mute list
//-----------------------------------------------------------------------------
void CMatchmaking::GenerateMutelist( MM_Mutelist *pMsg )
{
#if defined( _X360 )
// Get our remote talker list
if ( !Audio_GetXVoice() )
return;
int numRemoteTalkers;
XUID remoteTalkers[MAX_PLAYERS];
Audio_GetXVoice()->GetRemoteTalkers( &numRemoteTalkers, remoteTalkers );
pMsg->m_cPlayers = 0;
// Loop through local players and update mutes
for ( int iLocal = 0; iLocal < m_Local.m_cPlayers; ++iLocal )
{
pMsg->m_cMuted[iLocal] = 0;
pMsg->m_xuid[pMsg->m_cPlayers] = m_Local.m_xuids[iLocal];
for ( int iRemote = 0; iRemote < numRemoteTalkers; ++iRemote )
{
BOOL bIsMuted = false;
DWORD ret = XUserMuteListQuery( m_Local.m_iControllers[iLocal], remoteTalkers[iRemote], &bIsMuted );
if( ERROR_SUCCESS != ret )
{
Warning( "Warning: XUserMuteListQuery() returned 0x%08x for user %d\n", ret, iLocal );
}
if( bIsMuted )
{
pMsg->m_Muted[pMsg->m_cPlayers].AddToTail( remoteTalkers[iRemote ] );
++pMsg->m_cMuted[pMsg->m_cPlayers];
}
else
{
bIsMuted = m_MutedBy[iLocal].HasElement( remoteTalkers[iRemote] );
}
if( bIsMuted )
{
Audio_GetXVoice()->SetPlaybackPriority( remoteTalkers[iRemote], m_Local.m_iControllers[iLocal], XHV_PLAYBACK_PRIORITY_NEVER );
}
else
{
Audio_GetXVoice()->SetPlaybackPriority( remoteTalkers[iRemote], m_Local.m_iControllers[iLocal], XHV_PLAYBACK_PRIORITY_MAX );
}
}
pMsg->m_cRemoteTalkers[pMsg->m_cPlayers] = m_MutedBy[pMsg->m_cPlayers].Count();
++pMsg->m_cPlayers;
}
#endif
}
//-----------------------------------------------------------------------------
// Purpose: Handle the mutelist from another client
//-----------------------------------------------------------------------------
bool CMatchmaking::ProcessMutelist( MM_Mutelist *pMsg )
{
#if defined( _X360 )
// local players
for( int i = 0; i < m_Local.m_cPlayers; ++i )
{
// remote players
for( int j = 0; j < pMsg->m_cPlayers; ++j )
{
m_MutedBy[i].FindAndRemove( pMsg->m_xuid[j] );
// players muted by remote player
for( int k = 0; k < pMsg->m_cMuted[j]; ++k )
{
if( m_Local.m_xuids[i] == pMsg->m_Muted[j][k] )
{
m_MutedBy[i].AddToTail( pMsg->m_xuid[j] );
}
}
BOOL bIsMuted = m_MutedBy[i].HasElement( pMsg->m_xuid[j] );
if ( !bIsMuted )
{
XUserMuteListQuery( m_Local.m_iControllers[i], pMsg->m_xuid[j], &bIsMuted );
}
if( bIsMuted )
{
Audio_GetXVoice()->SetPlaybackPriority( pMsg->m_xuid[j], m_Local.m_iControllers[i], XHV_PLAYBACK_PRIORITY_NEVER );
}
else
{
Audio_GetXVoice()->SetPlaybackPriority( pMsg->m_xuid[j], m_Local.m_iControllers[i], XHV_PLAYBACK_PRIORITY_MAX );
}
}
}
if ( m_Session.IsHost() )
{
// Pass this along to everyone else
SendToRemoteClients( pMsg );
}
#endif
return true;
}
//-----------------------------------------------------------------------------
// Purpose: A client is changing to another team
//-----------------------------------------------------------------------------
void CMatchmaking::ChangeTeam( const char *pTeamName )
{
if ( !pTeamName )
{
// Automatic switch to next team
if ( m_Session.IsHost() )
{
if ( m_CurrentState == MMSTATE_ACCEPTING_CONNECTIONS )
{
// Put ourselves on another team and
// tell the other players
SwitchToNextOpenTeam( &m_Local );
}
}
else
{
// Send a request to the host
MM_Checkpoint msg;
msg.m_Checkpoint = MM_Checkpoint::CHECKPOINT_CHANGETEAM;
SendMessage( &msg, &m_Host );
}
}
else
{
// Find a team name that matches, and tell everyone our new team number
char szTeamName[32];
uint id = g_ClientDLL->GetPresenceID( "PROPERTY_TEAM" );
for ( int iTeam = 0; iTeam < m_nTotalTeams; ++iTeam )
{
g_ClientDLL->GetPropertyDisplayString( id, iTeam, szTeamName, sizeof( szTeamName ) );
if ( !Q_stricmp( szTeamName, pTeamName ) )
{
bool bChanged = false;
MM_ClientInfo info;
ClientInfoToNetMessage( &info, &m_Local );
for ( int i = 0; i < m_Local.m_cPlayers; ++i )
{
if ( info.m_iTeam[i] != iTeam )
{
bChanged = true;
}
info.m_iTeam[i] = iTeam;
}
if ( !bChanged )
{
return;
}
if ( m_Session.IsHost() )
{
ProcessClientInfo( &info );
}
else
{
SendMessage( &info, &m_Host );
}
break;
}
}
}
}
//-----------------------------------------------------------------------------
// Purpose: Handle various matchmaking checkpoint messages
//-----------------------------------------------------------------------------
bool CMatchmaking::ProcessCheckpoint( MM_Checkpoint *pMsg )
{
switch( pMsg->m_Checkpoint )
{
case MM_Checkpoint::CHECKPOINT_CHANGETEAM:
if ( m_Session.IsHost() )
{
// if the countdown has started, deny
if ( m_CurrentState == MMSTATE_ACCEPTING_CONNECTIONS )
{
netadr_t adr = pMsg->GetNetChannel()->GetRemoteAddress();
CClientInfo *pClient = FindClient( &adr );
if ( pClient )
{
SwitchToNextOpenTeam( pClient );
}
}
}
break;
case MM_Checkpoint::CHECKPOINT_PREGAME:
if ( m_Session.IsArbitrated() && !m_Session.IsHost() )
{
m_Session.RegisterForArbitration();
}
// Start the countdown timer to map load
SwitchToState( MMSTATE_PREGAME );
break;
case MM_Checkpoint::CHECKPOINT_GAME_LOBBY:
// returning to game lobby, pregame canceled
// reset the countdown
SessionNotification( SESSION_NOTIFY_COUNTDOWN, -1 );
if ( m_Session.IsHost() )
{
SwitchToState( MMSTATE_ACCEPTING_CONNECTIONS );
}
else
{
SwitchToState( MMSTATE_SESSION_CONNECTED );
}
break;
case MM_Checkpoint::CHECKPOINT_LOADING_COMPLETE:
if ( m_Session.IsHost() )
{
// Mark this client as loaded
netadr_t adr = pMsg->GetNetChannel()->GetRemoteAddress();
CClientInfo *pClient = FindClient( &adr );
if ( pClient )
{
pClient->m_bLoaded = true;
if ( GameIsActive() )
{
// Send a reply and reset the netchannel timeout
SendMessage( pMsg, pClient );
SetChannelTimeout( &pClient->m_adr, HEARTBEAT_TIMEOUT );
}
}
}
else
{
// The host is also loaded, so reset the netchannel timeout
SetChannelTimeout( &m_Host.m_adr, HEARTBEAT_TIMEOUT );
}
break;
case MM_Checkpoint::CHECKPOINT_CONNECT:
// If we're already connected or in the game, don't connect again.
if ( m_CurrentState == MMSTATE_CONNECTED_TO_SERVER ||
m_CurrentState == MMSTATE_INGAME )
{
break;
}
if ( m_Session.IsHost() )
{
// Send the message to everyone
SendToRemoteClients( pMsg );
}
else
{
// The host is asking us to connect to the game server
if ( m_CurrentState != MMSTATE_LOADING )
{
// Set the longer timeout for loading
SetChannelTimeout( &m_Host.m_adr, HEARTBEAT_TIMEOUT_LOADING );
SwitchToState( MMSTATE_LOADING );
}
}
// Make sure we're not preventing full startup
SetPreventFullServerStartup( false, "CHECKPOINT_CONNECT\n" );
if ( !IsServer() )
{
char cmd[MAX_PATH];
Q_snprintf( cmd, sizeof( cmd ), "connect %d.%d.%d.%d", m_Host.m_adr.ip[0], m_Host.m_adr.ip[1], m_Host.m_adr.ip[2], m_Host.m_adr.ip[3] );
Cbuf_AddText( cmd );
SessionNotification( SESSION_NOTIFY_CONNECTED_TOSERVER );
}
break;
case MM_Checkpoint::CHECKPOINT_SESSION_DISCONNECT:
PerformDisconnect();
break;
case MM_Checkpoint::CHECKPOINT_REPORT_STATS:
// Start stats reporting
g_ClientDLL->StartStatsReporting( m_Session.GetSessionHandle(), m_Session.IsArbitrated() );
m_Local.m_bReportedStats = true;
m_fHeartbeatInterval = HEARTBEAT_INTERVAL_SHORT;
SwitchToState( MMSTATE_REPORTING_STATS );
// Host needs to wait for clients to finish reporting
if ( !m_Session.IsHost() )
{
EndStatsReporting();
}
break;
case MM_Checkpoint::CHECKPOINT_REPORTING_COMPLETE:
{
// Mark this client as finished reporting stats
bool bFinished = false;
netadr_t adr = pMsg->GetNetChannel()->GetRemoteAddress();
for ( int i = 0; i < m_Remote.Count(); ++i )
{
CClientInfo *pClient = m_Remote[i];
if ( pClient->m_adr.CompareAdr( adr ) )
{
pClient->m_bReportedStats = true;
}
if ( !pClient->m_bReportedStats )
{
bFinished = false;
}
}
if ( bFinished && m_Local.m_bReportedStats )
{
EndStatsReporting();
}
}
break;
case MM_Checkpoint::CHECKPOINT_POSTGAME:
g_pXboxSystem->SessionEnd( m_Session.GetSessionHandle(), false );
engineClient->ClientCmd( "disconnect" );
EngineVGui()->ActivateGameUI();
if ( m_Session.IsArbitrated() )
{
// Tell gameui to return to the main menu
SessionNotification( SESSION_NOTIFY_ENDGAME_RANKED );
SwitchToState( MMSTATE_INITIAL );
}
else
{
if ( m_Session.IsHost() )
{
// Make ourselves available to queries again
m_HostData.gameState = GAMESTATE_INLOBBY;
UpdateSessionReplyData( XNET_QOS_LISTEN_ENABLE|XNET_QOS_LISTEN_SET_DATA );
}
// Tell gameui to activate the lobby
SessionNotification( m_Session.IsHost() ? SESSION_NOTIFY_ENDGAME_HOST : SESSION_NOTIFY_ENDGAME_CLIENT );
// Fill the lobby with all of the clients
if ( !m_Session.IsHost() )
{
SendPlayerInfoToLobby( &m_Host, m_nHostOwnerId );
SendPlayerInfoToLobby( &m_Local );
}
else
{
SendPlayerInfoToLobby( &m_Local, 0 );
}
for ( int i = 0; i < m_Remote.Count(); ++i )
{
SendPlayerInfoToLobby( m_Remote[i] );
}
if ( m_Session.IsHost() )
{
SwitchToState( MMSTATE_ACCEPTING_CONNECTIONS );
}
else
{
SwitchToState( MMSTATE_SESSION_CONNECTED );
}
}
break;
}
return true;
}
//-----------------------------------------------------------------------------
// Purpose: Stop any asynchronous operation that is currently running
//-----------------------------------------------------------------------------
void CMatchmaking::CancelCurrentOperation()
{
switch( m_CurrentState )
{
case MMSTATE_CREATING:
m_Session.CancelCreateSession();
break;
case MMSTATE_SEARCHING:
CancelSearch();
break;
case MMSTATE_WAITING_QOS:
CancelQosLookup();
break;
case MMSTATE_SESSION_CONNECTING:
break;
default:
break;
}
}
//-----------------------------------------------------------------------------
// Purpose: Send player info to be displayed in the session lobby
//-----------------------------------------------------------------------------
void CMatchmaking::SendPlayerInfoToLobby( CClientInfo *pClient, int iHostIdx )
{
for ( int i = 0; i < pClient->m_cPlayers; ++i )
{
EngineVGui()->UpdatePlayerInfo( pClient->m_xuids[i], pClient->m_szGamertags[i], pClient->m_iTeam[i], pClient->m_cVoiceState[i], GetPlayersNeeded(), iHostIdx == i );
}
}
//-----------------------------------------------------------------------------
// Purpose: Update the start game countdown timer
//-----------------------------------------------------------------------------
void CMatchmaking::UpdatePregame()
{
int elapsedTime = GetTime() - m_fCountdownStartTime;
if ( elapsedTime > REGISTRATION_MAXWAITTIME )
{
// Check the registration timer.
if ( m_Session.IsHost() && m_Session.IsArbitrated() && !m_Local.m_bRegistered )
{
// Time's up, register ourselves
m_Local.m_bRegistered = true;
m_Session.RegisterForArbitration();
}
}
// Check the countdown timer. When it's zero, start the game.
if ( elapsedTime < STARTGAME_COUNTDOWN )
{
SessionNotification( SESSION_NOTIFY_COUNTDOWN, STARTGAME_COUNTDOWN - elapsedTime );
return;
}
// Send the zero count
SessionNotification( SESSION_NOTIFY_COUNTDOWN, 0 );
// Set a longer timeout for loading
if ( m_Session.IsHost() )
{
for ( int i = 0; i < m_Remote.Count(); ++i )
{
SetChannelTimeout( &m_Remote[i]->m_adr, HEARTBEAT_TIMEOUT_LOADING );
}
// Set commentary & cheats off
ConVarRef commentary( "commentary" );
commentary.SetValue( false );
ConVarRef sv_cheats( "sv_cheats" );
sv_cheats.SetValue( 0 );
}
else
{
SetChannelTimeout( &m_Host.m_adr, HEARTBEAT_TIMEOUT_LOADING );
}
g_pXboxSystem->SessionStart( m_Session.GetSessionHandle(), 0, false );
if ( !IsServer() )
{
SetPreventFullServerStartup( true, "SESSION_NOTIFY_COUNTDOWN == 0 and not the host\n" );
}
SwitchToState( MMSTATE_LOADING );
}
//-----------------------------------------------------------------------------
// Purpose: Receive notifications from the session
//-----------------------------------------------------------------------------
void CMatchmaking::SessionNotification( const SESSION_NOTIFY notification, const int param )
{
// Notify GameUI
EngineVGui()->SessionNotification( notification, param );
switch( notification )
{
case SESSION_NOTIFY_CREATED_HOST:
m_bEnteredLobby = true;
OnHostSessionCreated();
break;
case SESSION_NOTIFY_CREATED_CLIENT:
Msg( "Client: CreateSession successful\n" );
// Session has been created according to the advertised specifications.
// Now send a connection request to the session host.
SwitchToState( MMSTATE_SESSION_CONNECTING );
break;
case SESSION_NOFIFY_MODIFYING_SESSION:
SwitchToState( MMSTATE_MODIFYING );
break;
case SESSION_NOTIFY_MODIFYING_COMPLETED_HOST:
SwitchToState( MMSTATE_ACCEPTING_CONNECTIONS );
break;
case SESSION_NOTIFY_MODIFYING_COMPLETED_CLIENT:
SwitchToState( MMSTATE_SESSION_CONNECTED );
break;
case SESSION_NOTIFY_SEARCH_COMPLETED:
// Waiting for the player to choose a session
SwitchToState( MMSTATE_BROWSING );
break;
case SESSION_NOTIFY_CONNECTED_TOSESSION:
m_bEnteredLobby = true;
SwitchToState( MMSTATE_SESSION_CONNECTED );
break;
case SESSION_NOTIFY_CONNECTED_TOSERVER:
SwitchToState( MMSTATE_CONNECTED_TO_SERVER );
break;
case SESSION_NOTIFY_MIGRATION_COMPLETED:
if ( !m_Session.IsHost() )
{
// Finished
EndMigration();
}
else
{
// Get ready to send join requests too peers
for ( int i = 0; i < m_Remote.Count(); ++i )
{
m_Remote[i]->m_bMigrated = false;
AddRemoteChannel( &m_Remote[i]->m_adr );
}
m_nSendCount = 0;
m_fSendTimer = 0;
SwitchToState( MMSTATE_HOSTMIGRATE_WAITINGFORCLIENTS );
}
break;
case SESSION_NOTIFY_FAIL_MIGRATE:
// X360TBD: How to handle this error?
Warning( "Migrate Failed\n" );
break;
case SESSION_NOTIFY_REGISTER_COMPLETED:
if( !m_Session.IsHost() )
{
// Tell the host we're registered
MM_RegisterResponse msg;
SendMessage( &msg, &m_Host.m_adr );
}
else
{
// Process the results of registration
ProcessRegistrationResults();
}
break;
case SESSION_NOTIFY_ENDGAME_HOST:
case SESSION_NOTIFY_ENDGAME_CLIENT:
m_fHeartbeatInterval = HEARTBEAT_INTERVAL_SHORT;
break;
case SESSION_NOTIFY_LOST_HOST:
case SESSION_NOTIFY_LOST_SERVER:
SwitchToState( MMSTATE_SESSION_DISCONNECTING );
break;
case SESSION_NOTIFY_FAIL_REGISTER:
case SESSION_NOTIFY_FAIL_CREATE:
case SESSION_NOTIFY_FAIL_SEARCH:
case SESSION_NOTIFY_CONNECT_SESSIONFULL:
case SESSION_NOTIFY_CONNECT_NOTAVAILABLE:
case SESSION_NOTIFY_CONNECT_FAILED:
// Reset the session
SwitchToState( MMSTATE_INITIAL );
break;
default:
break;
}
}
//-----------------------------------------------------------------------------
// Purpose: Switch between matchmaking states
//-----------------------------------------------------------------------------
void CMatchmaking::SwitchToState( int newState )
{
// Clean up from the previous state
switch( m_CurrentState )
{
case MMSTATE_INITIAL:
break;
case MMSTATE_BROWSING:
// Clean up the search results
ClearSearchResults();
break;
default:
break;
}
// Initialize the next state
switch( newState )
{
case MMSTATE_INITIAL:
m_bCleanup = true;
break;
case MMSTATE_CREATING:
case MMSTATE_SEARCHING:
case MMSTATE_ACCEPTING_CONNECTIONS:
break;
case MMSTATE_MODIFYING:
m_fSendTimer = GetTime();
break;
case MMSTATE_WAITING_QOS:
m_fWaitTimer = GetTime();
break;
case MMSTATE_SESSION_CONNECTING:
ConnectToHost();
break;
case MMSTATE_PREGAME:
m_fCountdownStartTime = GetTime();
break;
case MMSTATE_LOADING:
m_fHeartbeatInterval = HEARTBEAT_INTERVAL_LONG;
break;
case MMSTATE_SESSION_DISCONNECTING:
m_fWaitTimer = GetTime();
break;
case MMSTATE_REPORTING_STATS:
m_fWaitTimer = GetTime();
break;
default:
break;
}
m_CurrentState = newState;
}
void CMatchmaking::UpdateVoiceStatus( void )
{
#if defined( _X360 )
if ( m_flVoiceBlinkTime > GetTime() )
return;
m_flVoiceBlinkTime = GetTime() + VOICE_ICON_BLINK_TIME;
CClientInfo *pClient = &m_Local;
bool bIsHost = m_Session.IsHost();
bool bShouldSendInfo = false;
if ( pClient )
{
for ( int i = 0; i < pClient->m_cPlayers; ++i )
{
if ( pClient->m_xuids[i] == 0 )
continue;
byte cOldVoiceState = pClient->m_cVoiceState[i];
if ( Audio_GetXVoice()->IsHeadsetPresent( pClient->m_iControllers[i] ) == false )
{
pClient->m_cVoiceState[i] = VOICE_STATUS_OFF;
}
else
{
if ( Audio_GetXVoice()->IsPlayerTalking( pClient->m_xuids[i], true ) )
{
pClient->m_cVoiceState[i] = VOICE_STATUS_TALKING;
}
else
{
pClient->m_cVoiceState[i] = VOICE_STATUS_IDLE;
}
}
if ( cOldVoiceState != pClient->m_cVoiceState[i] )
{
bShouldSendInfo = true;
}
if ( bShouldSendInfo == true )
{
EngineVGui()->UpdatePlayerInfo( pClient->m_xuids[i], pClient->m_szGamertags[i], pClient->m_iTeam[i], pClient->m_cVoiceState[i], GetPlayersNeeded(), bIsHost );
}
}
if ( bShouldSendInfo )
{
MM_ClientInfo info;
ClientInfoToNetMessage( &info, pClient );
if ( bIsHost == true )
{
// Tell all the clients
ProcessClientInfo( &info );
}
else
{
// Tell all the clients
SendMessage( &info, &m_Host );
}
}
}
#endif
}
//-----------------------------------------------------------------------------
// Update matchmaking and any active session
//-----------------------------------------------------------------------------
void CMatchmaking::RunFrame()
{
if ( !m_bInitialized )
{
RunFrameInvite();
return;
}
if ( NET_IsMultiplayer() )
{
NET_ProcessSocket( NS_MATCHMAKING, this );
if ( m_Session.IsSystemLink() )
{
NET_ProcessSocket( NS_SYSTEMLINK, this );
}
}
#if defined( _X360 )
if ( Audio_GetXVoice() != NULL )
{
if ( !GameIsActive() )
{
if ( Audio_GetXVoice()->VoiceUpdateData() == true )
{
CLC_VoiceData voice;
Audio_GetXVoice()->GetVoiceData( &voice );
if ( m_Session.IsHost() )
{
// Send this message on to everyone else
SendToRemoteClients( &voice, true );
}
else
{
// Send to the host
SendMessage( &voice, &m_Host );
}
Audio_GetXVoice()->VoiceResetLocalData();
}
UpdateVoiceStatus();
}
}
#endif
// Tell the session to run its update
m_Session.RunFrame();
// Check state:
switch( m_CurrentState )
{
case MMSTATE_CREATING:
// Waiting for success or failure from CreateSession()
// GameUI is displaying a "Creating Game" dialog.
break;
case MMSTATE_ACCEPTING_CONNECTIONS:
// Host is sitting in the Lobby waiting for connection requests. Once the game
// is full enough (player count >= min players) the host will be able to start the game.
UpdateAcceptingConnections();
break;
case MMSTATE_SEARCHING:
UpdateSearch();
break;
case MMSTATE_WAITING_QOS:
UpdateQosLookup();
break;
case MMSTATE_SESSION_CONNECTING:
UpdateConnecting();
break;
case MMSTATE_PREGAME:
UpdatePregame();
break;
case MMSTATE_MODIFYING:
UpdateSessionModify();
break;
case MMSTATE_HOSTMIGRATE_WAITINGFORCLIENTS:
if ( GetTime() - m_fSendTimer > HOSTMIGRATION_RETRYINTERVAL )
{
if ( m_nSendCount > HOSTMIGRATION_MAXRETRIES )
{
EndMigration();
}
else
{
TellClientsToMigrate();
}
}
break;
case MMSTATE_HOSTMIGRATE_WAITINGFORHOST:
if ( GetTime() - m_fWaitTimer > HOSTMIGRATION_MAXWAITTIME )
{
// Give up on that host and try the next one in the list
Msg( "Giving up on this host\n" );
ClientDropped( m_pNewHost );
StartHostMigration();
}
break;
case MMSTATE_SESSION_DISCONNECTING:
// Wait for the host reply, or timeout
if ( GetTime() - m_fWaitTimer > DISCONNECT_WAITTIME )
{
PerformDisconnect();
}
break;
case MMSTATE_REPORTING_STATS:
if ( GetTime() - m_fWaitTimer > REPORTSTATS_WAITTIME )
{
EndStatsReporting();
}
break;
}
CleanupMarkedChannels();
SendHeartbeat();
if ( m_bCleanup )
{
Cleanup();
}
RunFrameInvite();
}
//-----------------------------------------------------------------------------
// Purpose: Let the invite system determine a good moment to start connecting to inviter's host
//-----------------------------------------------------------------------------
void CMatchmaking::RunFrameInvite()
{
if ( m_InviteState != INVITE_NONE )
{
JoinInviteSession( &m_InviteSessionInfo );
}
}
//-----------------------------------------------------------------------------
// Purpose: Get Quality-of-Service with LIVE
//-----------------------------------------------------------------------------
MM_QOS_t CMatchmaking::GetQosWithLIVE()
{
return MM_GetQos();
}
//-----------------------------------------------------------------------------
// Debugging helpers
//-----------------------------------------------------------------------------
void CMatchmaking::ShowSessionInfo()
{
Msg( "[MM] Filled Slots:\n[MM] Public: %d of %d\n[MM] Private: %d of %d\n",
m_Session.GetSessionSlots( SLOTS_FILLEDPUBLIC ),
m_Session.GetSessionSlots( SLOTS_TOTALPUBLIC ),
m_Session.GetSessionSlots( SLOTS_FILLEDPRIVATE ),
m_Session.GetSessionSlots( SLOTS_TOTALPRIVATE ) );
Msg( "[MM] Current state: %d\n", m_CurrentState );
Msg( "[MM] Send timer: %f\n", GetTime() - m_fSendTimer );
Msg( "[MM] Wait timer: %f\n", GetTime() - m_fWaitTimer );
int total = 0;
for ( int i = 0; i < m_nTotalTeams; ++i )
{
total += CountPlayersOnTeam( i );
}
Msg( "[MM] Total players: %d\n", total );
EngineVGui()->SessionNotification( SESSION_NOTIFY_DUMPSTATS );
}
//-----------------------------------------------------------------------------
// Debugging helpers
//-----------------------------------------------------------------------------
void CMatchmaking::TestSendMessage()
{
for ( int i = 0; i < m_Remote.Count(); ++i )
{
AddRemoteChannel( &m_Remote[i]->m_adr );
}
NET_StringCmd msg;
SendToRemoteClients( &msg );
}
bool CMatchmaking::PreventFullServerStartup()
{
return m_bPreventFullServerStartup;
}
void CMatchmaking::SetPreventFullServerStartup( bool bState, char const *fmt, ... )
{
char desc[ 256 ];
va_list argptr;
va_start( argptr, fmt );
Q_vsnprintf( desc, sizeof( desc ), fmt, argptr );
va_end( argptr );
DevMsg( 1, "Setting state from prevent %s to prevent %s: %s",
m_bPreventFullServerStartup ? "yes" : "no",
bState ? "yes" : "no",
desc );
m_bPreventFullServerStartup = bState;
}
CON_COMMAND( mm_session_info, "Dump session information" )
{
if ( g_pMatchmaking )
{
g_pMatchmaking->ShowSessionInfo();
}
}
CON_COMMAND( mm_message, "Send a message to all remote clients" )
{
if ( g_pMatchmaking )
{
g_pMatchmaking->TestSendMessage();
}
}
CON_COMMAND( mm_stats, "" )
{
if ( g_pMatchmaking )
{
g_pMatchmaking->TestStats();
}
}