1223 lines
36 KiB
C++
1223 lines
36 KiB
C++
|
/*==========================================================================
|
||
|
*
|
||
|
* Copyright (C) 2000 Microsoft Corporation. All Rights Reserved.
|
||
|
*
|
||
|
* File: DPLCommon.cpp
|
||
|
* Content: DirectPlay Lobby Common Functions
|
||
|
*@@BEGIN_MSINTERNAL
|
||
|
* History:
|
||
|
* Date By Reason
|
||
|
* ==== == ======
|
||
|
* 02/21/00 mjn Created
|
||
|
* 04/13/00 rmt First pass param validation
|
||
|
* 04/26/00 mjn Removed dwTimeOut from Send() API call
|
||
|
* 05/01/00 rmt Bug #33108 -- Initialize returns DPNERR_NORESPONSE when not lobbied
|
||
|
* 05/03/00 rmt Updated initialize so if lobby launched automatically establishes a
|
||
|
* connection and makes self unavailable. (Also waits for connection).
|
||
|
* 05/16/00 rmt Bug #34734 -- Init Client, Init App, Close App hangs --
|
||
|
* both client and app were using 'C' prefix, should have been 'C' for
|
||
|
* client and 'A' for app.
|
||
|
* 06/14/00 rmt Fixed build break with new compiler (added ')''s).
|
||
|
* 06/15/00 rmt Bug #33617 - Must provide method for providing automatic launch of DirectPlay instances
|
||
|
* 06/28/00 rmt Prefix Bug #38082
|
||
|
* 07/06/00 rmt Bug #38717 ASSERTION when sending data
|
||
|
* 07/08/2000 rmt Bug #38725 - Need to provide method to detect if app was lobby launched
|
||
|
* rmt Bug #38757 - Callback messages for connections may return AFTER WaitForConnection returns
|
||
|
* rmt Bug #38755 - No way to specify player name in Connection Settings
|
||
|
* rmt Bug #38758 - DPLOBBY8.H has incorrect comments
|
||
|
* rmt Bug #38783 - pvUserApplicationContext is only partially implemented
|
||
|
* rmt Added DPLHANDLE_ALLCONNECTIONS and dwFlags (reserved field to couple of funcs).
|
||
|
* 07/13/2000 rmt Fixed memory leak
|
||
|
* 07/14/2000 rmt Bug #39257 - LobbyClient::ReleaseApp returns E_OUTOFMEMORY when called when no one connected
|
||
|
* 07/21/2000 rmt Removed assert which wasn't needed
|
||
|
* 08/03/2000 rmt Removed assert which wasn't needed
|
||
|
* 08/05/2000 RichGr IA64: Use %p format specifier in DPFs for 32/64-bit pointers and handles.
|
||
|
* 08/18/2000 rmt Bug #42751 - DPLOBBY8: Prohibit more than one lobby client or lobby app per process
|
||
|
* 08/24/2000 rmt Bug #43317 - DP8LOBBY: Occasionally when closing Lobby App right after releasing handles, causes assertion.
|
||
|
* 02/06/2001 rodtoll WINBUG #293871: DPLOBBY8: [IA64] Lobby launching a 64-bit
|
||
|
* app from 64-bit lobby launcher crashes with unaligned memory error.
|
||
|
*
|
||
|
*@@END_MSINTERNAL
|
||
|
*
|
||
|
***************************************************************************/
|
||
|
|
||
|
#include "dnlobbyi.h"
|
||
|
|
||
|
|
||
|
//**********************************************************************
|
||
|
// Constant definitions
|
||
|
//**********************************************************************
|
||
|
|
||
|
//**********************************************************************
|
||
|
// Macro definitions
|
||
|
//**********************************************************************
|
||
|
|
||
|
//**********************************************************************
|
||
|
// Structure definitions
|
||
|
//**********************************************************************
|
||
|
|
||
|
//**********************************************************************
|
||
|
// Variable definitions
|
||
|
//**********************************************************************
|
||
|
|
||
|
LONG volatile g_lLobbyAppCount = 0;
|
||
|
LONG volatile g_lLobbyClientCount = 0;
|
||
|
|
||
|
//**********************************************************************
|
||
|
// Function prototypes
|
||
|
//**********************************************************************
|
||
|
|
||
|
//**********************************************************************
|
||
|
// Function definitions
|
||
|
//**********************************************************************
|
||
|
|
||
|
// DPL_GetConnectionSettings
|
||
|
//
|
||
|
// Retrieves the pdplSessionInfo (if any) associated with the specified connection. This method
|
||
|
// is shared between the client and app interfaces.
|
||
|
//
|
||
|
#undef DPF_MODNAME
|
||
|
#define DPF_MODNAME "DPL_GetConnectionSettings"
|
||
|
STDMETHODIMP DPL_GetConnectionSettings(LPVOID lpv,const DPNHANDLE hLobbyClient, DPL_CONNECTION_SETTINGS * const pdplSessionInfo, DWORD *pdwInfoSize, const DWORD dwFlags )
|
||
|
{
|
||
|
HRESULT hResultCode;
|
||
|
DIRECTPLAYLOBBYOBJECT *pdpLobbyObject;
|
||
|
|
||
|
DPFX(DPFPREP, 3,"Parameters: hTarget [0x%lx], pdplSessionInfo [0x%p], pdwInfoSize [%p], dwFlags [0x%lx]",
|
||
|
hLobbyClient,pdplSessionInfo,pdwInfoSize,dwFlags);
|
||
|
|
||
|
#ifndef DPNBUILD_NOPARAMVAL
|
||
|
TRY
|
||
|
{
|
||
|
#endif // !DPNBUILD_NOPARAMVAL
|
||
|
pdpLobbyObject = static_cast<DIRECTPLAYLOBBYOBJECT*>(GET_OBJECT_FROM_INTERFACE(lpv));
|
||
|
|
||
|
#ifndef DPNBUILD_NOPARAMVAL
|
||
|
if( pdpLobbyObject->dwFlags & DPL_OBJECT_FLAG_PARAMVALIDATION )
|
||
|
{
|
||
|
if( FAILED( hResultCode = DPL_ValidateGetConnectionSettings( lpv, hLobbyClient, pdplSessionInfo, pdwInfoSize, dwFlags ) ) )
|
||
|
{
|
||
|
DPFX(DPFPREP, 0, "Error validating getconnectsettings params hr=[0x%lx]", hResultCode );
|
||
|
DPF_RETURN( hResultCode );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Ensure we've been initialized
|
||
|
if (pdpLobbyObject->pReceiveQueue == NULL)
|
||
|
{
|
||
|
DPFERR("Not initialized");
|
||
|
DPF_RETURN(DPNERR_UNINITIALIZED);
|
||
|
}
|
||
|
}
|
||
|
EXCEPT(EXCEPTION_EXECUTE_HANDLER)
|
||
|
{
|
||
|
DPFERR("Invalid object" );
|
||
|
DPF_RETURN(DPNERR_INVALIDOBJECT);
|
||
|
}
|
||
|
#endif // !DPNBUILD_NOPARAMVAL
|
||
|
|
||
|
// Attempt to retrieve connection settings.
|
||
|
hResultCode = DPLConnectionGetConnectSettings( pdpLobbyObject, hLobbyClient, pdplSessionInfo, pdwInfoSize );
|
||
|
|
||
|
DPF_RETURN( hResultCode );
|
||
|
}
|
||
|
|
||
|
// DPL_SetConnectionSettings
|
||
|
//
|
||
|
// Sets the pdplSessionInfo structure associated with the specified connection. This method
|
||
|
// is shared between the client and app interfaces.
|
||
|
//
|
||
|
// This function will generate a DPL_MSGID_CONNECTION_SETTINGS message to be sent to the specified
|
||
|
// connection.
|
||
|
//
|
||
|
#undef DPF_MODNAME
|
||
|
#define DPF_MODNAME "DPL_SetConnectionSettings"
|
||
|
STDMETHODIMP DPL_SetConnectionSettings(LPVOID lpv,const DPNHANDLE hLobbyClient, const DPL_CONNECTION_SETTINGS * const pdplSessionInfo, const DWORD dwFlags )
|
||
|
{
|
||
|
HRESULT hResultCode;
|
||
|
DPL_CONNECTION *pdplConnection;
|
||
|
DIRECTPLAYLOBBYOBJECT *pdpLobbyObject;
|
||
|
DPNHANDLE *hTargets = NULL;
|
||
|
DWORD dwNumTargets = 0;
|
||
|
DWORD dwTargetIndex = 0;
|
||
|
CConnectionSettings *pConnectionSettings = NULL;
|
||
|
|
||
|
DPFX(DPFPREP, 3,"Parameters: hLobbyClient [0x%lx], pBuffer [0x%p], dwFlags [0x%lx]",
|
||
|
hLobbyClient,pdplSessionInfo,dwFlags);
|
||
|
|
||
|
#ifndef DPNBUILD_NOPARAMVAL
|
||
|
TRY
|
||
|
{
|
||
|
#endif // !DPNBUILD_NOPARAMVAL
|
||
|
pdpLobbyObject = static_cast<DIRECTPLAYLOBBYOBJECT*>(GET_OBJECT_FROM_INTERFACE(lpv));
|
||
|
|
||
|
#ifndef DPNBUILD_NOPARAMVAL
|
||
|
if( pdpLobbyObject->dwFlags & DPL_OBJECT_FLAG_PARAMVALIDATION )
|
||
|
{
|
||
|
if( FAILED( hResultCode = DPL_ValidateSetConnectionSettings( lpv, hLobbyClient, pdplSessionInfo, dwFlags ) ) )
|
||
|
{
|
||
|
DPFX(DPFPREP, 0, "Error validating setconnectsettings params hr=[0x%lx]", hResultCode );
|
||
|
DPF_RETURN( hResultCode );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Ensure we've been initialized
|
||
|
if (pdpLobbyObject->pReceiveQueue == NULL)
|
||
|
{
|
||
|
DPFERR("Not initialized");
|
||
|
DPF_RETURN(DPNERR_UNINITIALIZED);
|
||
|
}
|
||
|
}
|
||
|
EXCEPT(EXCEPTION_EXECUTE_HANDLER)
|
||
|
{
|
||
|
DPFERR("Invalid object" );
|
||
|
DPF_RETURN(DPNERR_INVALIDOBJECT);
|
||
|
}
|
||
|
#endif // !DPNBUILD_NOPARAMVAL
|
||
|
|
||
|
if( hLobbyClient == DPLHANDLE_ALLCONNECTIONS )
|
||
|
{
|
||
|
dwNumTargets = 0;
|
||
|
|
||
|
// We need loop so if someone adds a connection during our run
|
||
|
// it gets added to our list
|
||
|
//
|
||
|
while( 1 )
|
||
|
{
|
||
|
hResultCode = DPLConnectionEnum( pdpLobbyObject, hTargets, &dwNumTargets );
|
||
|
|
||
|
if( hResultCode == DPNERR_BUFFERTOOSMALL )
|
||
|
{
|
||
|
if( hTargets )
|
||
|
{
|
||
|
delete [] hTargets;
|
||
|
}
|
||
|
|
||
|
hTargets = new DPNHANDLE[dwNumTargets];
|
||
|
|
||
|
if( hTargets == NULL )
|
||
|
{
|
||
|
DPFERR("Error allocating memory" );
|
||
|
dwNumTargets = 0;
|
||
|
hResultCode = DPNERR_OUTOFMEMORY;
|
||
|
goto SETCONNECT_EXIT;
|
||
|
}
|
||
|
|
||
|
|
||
|
continue;
|
||
|
}
|
||
|
else if( FAILED( hResultCode ) )
|
||
|
{
|
||
|
DPFX(DPFPREP, 0, "Error getting list of connections hr=0x%x", hResultCode );
|
||
|
break;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Failed getting connection information
|
||
|
if( FAILED( hResultCode ) )
|
||
|
{
|
||
|
if( hTargets )
|
||
|
{
|
||
|
delete [] hTargets;
|
||
|
hTargets = NULL;
|
||
|
}
|
||
|
dwNumTargets = 0;
|
||
|
goto SETCONNECT_EXIT;
|
||
|
}
|
||
|
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
hTargets = new DPNHANDLE[1]; // We use array delete below so we need array new
|
||
|
|
||
|
if( hTargets == NULL )
|
||
|
{
|
||
|
DPFERR("Error allocating memory" );
|
||
|
dwNumTargets = 0;
|
||
|
hResultCode = DPNERR_OUTOFMEMORY;
|
||
|
goto SETCONNECT_EXIT;
|
||
|
}
|
||
|
|
||
|
dwNumTargets = 1;
|
||
|
hTargets[0] = hLobbyClient;
|
||
|
}
|
||
|
|
||
|
for( dwTargetIndex = 0; dwTargetIndex < dwNumTargets; dwTargetIndex++ )
|
||
|
{
|
||
|
if ((hResultCode = DPLConnectionFind(pdpLobbyObject,hTargets[dwTargetIndex],&pdplConnection,TRUE)) != DPN_OK)
|
||
|
{
|
||
|
DPFERR("Invalid send target");
|
||
|
DisplayDNError(0,hResultCode);
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
if( pdplSessionInfo )
|
||
|
{
|
||
|
pConnectionSettings = new CConnectionSettings();
|
||
|
|
||
|
if( !pConnectionSettings )
|
||
|
{
|
||
|
DPFERR("Error allocating memory" );
|
||
|
hResultCode = DPNERR_OUTOFMEMORY;
|
||
|
goto SETCONNECT_EXIT;
|
||
|
}
|
||
|
|
||
|
hResultCode = pConnectionSettings->InitializeAndCopy( pdplSessionInfo );
|
||
|
|
||
|
if( FAILED( hResultCode ) )
|
||
|
{
|
||
|
DPFX( DPFPREP, 0, "Error setting up connection settings hr [0x%x]", hResultCode );
|
||
|
goto SETCONNECT_EXIT;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Attempt to set connection settings.
|
||
|
hResultCode = DPLConnectionSetConnectSettings( pdpLobbyObject, hTargets[dwTargetIndex], pConnectionSettings );
|
||
|
|
||
|
if( FAILED( hResultCode ) )
|
||
|
{
|
||
|
DPFX(DPFPREP, 0, "Error setting connct settings for 0x%x hr=0x%x", hTargets[dwTargetIndex], hResultCode );
|
||
|
delete pConnectionSettings;
|
||
|
}
|
||
|
|
||
|
hResultCode = DPLSendConnectionSettings( pdpLobbyObject, hTargets[dwTargetIndex] );
|
||
|
|
||
|
if( FAILED( hResultCode ) )
|
||
|
{
|
||
|
DPFX(DPFPREP, 0, "Error sending connection settings to client 0x%x hr=0x%x", hTargets[dwTargetIndex], hResultCode );
|
||
|
}
|
||
|
|
||
|
pConnectionSettings = NULL;
|
||
|
}
|
||
|
|
||
|
SETCONNECT_EXIT:
|
||
|
|
||
|
for( dwTargetIndex = 0; dwTargetIndex < dwNumTargets; dwTargetIndex++ )
|
||
|
{
|
||
|
if( hTargets[dwTargetIndex] )
|
||
|
DPLConnectionRelease(pdpLobbyObject,hTargets[dwTargetIndex]);
|
||
|
}
|
||
|
|
||
|
if( hTargets )
|
||
|
delete [] hTargets;
|
||
|
|
||
|
DPF_RETURN(hResultCode);
|
||
|
}
|
||
|
|
||
|
|
||
|
#undef DPF_MODNAME
|
||
|
#define DPF_MODNAME "DPL_RegisterMessageHandlerClient"
|
||
|
STDMETHODIMP DPL_RegisterMessageHandlerClient(PVOID pv,
|
||
|
const PVOID pvUserContext,
|
||
|
const PFNDPNMESSAGEHANDLER pfn,
|
||
|
const DWORD dwFlags)
|
||
|
{
|
||
|
return DPL_RegisterMessageHandler( pv, pvUserContext, pfn, NULL, dwFlags );
|
||
|
}
|
||
|
|
||
|
// HRESULT DPL_RegisterMessageHandler
|
||
|
// PVOID pv Interface pointer
|
||
|
// PVOID pvUserContext User context
|
||
|
// PFNDPNMESSAGEHANDLER pfn User supplied message handler
|
||
|
// DWORD dwFlags Not Used
|
||
|
//
|
||
|
// Returns
|
||
|
// DPN_OK If the message handler was registered without incident
|
||
|
// DPNERR_INVALIDPARAM If there was an invalid parameter
|
||
|
// DPNERR_GENERIC If there were any problems
|
||
|
//
|
||
|
// Notes
|
||
|
// This function registers a user supplied message handler function. This function should
|
||
|
// only be called once, even in cases where a game is being re-connected (i.e. after ending)
|
||
|
//
|
||
|
// This will set up the required message queues, handshake the lobby client's PID (if supplied on the
|
||
|
// command line) and spawn the application's receive message queue thread.
|
||
|
|
||
|
#undef DPF_MODNAME
|
||
|
#define DPF_MODNAME "DPL_RegisterMessageHandler"
|
||
|
|
||
|
STDMETHODIMP DPL_RegisterMessageHandler(PVOID pv,
|
||
|
const PVOID pvUserContext,
|
||
|
const PFNDPNMESSAGEHANDLER pfn,
|
||
|
DPNHANDLE * const pdpnhConnection,
|
||
|
const DWORD dwFlags)
|
||
|
{
|
||
|
HRESULT hResultCode = DPN_OK;
|
||
|
DWORD dwCurrentPid;
|
||
|
DWORD dwThreadId;
|
||
|
PDIRECTPLAYLOBBYOBJECT pdpLobbyObject;
|
||
|
char cSuffix;
|
||
|
|
||
|
DPFX(DPFPREP, 3,"Parameters: pv [0x%p], pfn [0x%p], dwFlags [%lx]",pv,pfn,dwFlags);
|
||
|
|
||
|
#ifndef DPNBUILD_NOPARAMVAL
|
||
|
TRY
|
||
|
{
|
||
|
#endif // !DPNBUILD_NOPARAMVAL
|
||
|
pdpLobbyObject = static_cast<DIRECTPLAYLOBBYOBJECT*>(GET_OBJECT_FROM_INTERFACE(pv));
|
||
|
|
||
|
#ifndef DPNBUILD_NOPARAMVAL
|
||
|
// TODO: MASONB: Why no paramval flag wrapping this?
|
||
|
if( FAILED( hResultCode = DPL_ValidateRegisterMessageHandler( pv, pvUserContext, pfn, pdpnhConnection, dwFlags ) ) )
|
||
|
{
|
||
|
DPFX(DPFPREP, 0, "Error validating register message handler params hr=[0x%lx]", hResultCode );
|
||
|
DPF_RETURN( hResultCode );
|
||
|
}
|
||
|
|
||
|
// Ensure we've been initialized
|
||
|
if (pdpLobbyObject->pReceiveQueue != NULL)
|
||
|
{
|
||
|
DPFERR("Already initialized");
|
||
|
DPF_RETURN(DPNERR_ALREADYINITIALIZED);
|
||
|
}
|
||
|
}
|
||
|
EXCEPT(EXCEPTION_EXECUTE_HANDLER)
|
||
|
{
|
||
|
DPFERR("Invalid object" );
|
||
|
DPF_RETURN(DPNERR_INVALIDOBJECT);
|
||
|
}
|
||
|
#endif // !DPNBUILD_NOPARAMVAL
|
||
|
|
||
|
if (pdpLobbyObject->dwFlags & DPL_OBJECT_FLAG_LOBBIEDAPPLICATION )
|
||
|
{
|
||
|
// If we weren't at zero complain, otherwise put us at 1
|
||
|
if( DNInterlockedCompareExchange((LONG*)&g_lLobbyAppCount, 1, 0) != 0 )
|
||
|
{
|
||
|
DPFERR( "You can only start one lobbied application per process!" );
|
||
|
DPF_RETURN( DPNERR_NOTALLOWED );
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
DNASSERT(pdpLobbyObject->dwFlags & DPL_OBJECT_FLAG_LOBBYCLIENT);
|
||
|
// If we weren't at zero complain, otherwise put us at 1
|
||
|
if( DNInterlockedCompareExchange((LONG*)&g_lLobbyClientCount, 1, 0) != 0 )
|
||
|
{
|
||
|
DPFERR( "You can only start one lobby client per process!" );
|
||
|
DPF_RETURN( DPNERR_NOTALLOWED );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#ifndef DPNBUILD_NOPARAMVAL
|
||
|
// Disable parameter validation flag if DPNINITIALIZE_DISABLEPARAMVAL
|
||
|
// is specified
|
||
|
if( dwFlags & DPLINITIALIZE_DISABLEPARAMVAL )
|
||
|
{
|
||
|
pdpLobbyObject->dwFlags &= ~(DPL_OBJECT_FLAG_PARAMVALIDATION);
|
||
|
}
|
||
|
#endif // !DPNBUILD_NOPARAMVAL
|
||
|
|
||
|
pdpLobbyObject->pfnMessageHandler = pfn;
|
||
|
pdpLobbyObject->pvUserContext = pvUserContext;
|
||
|
|
||
|
pdpLobbyObject->pReceiveQueue = new CMessageQueue;
|
||
|
|
||
|
|
||
|
if( pdpLobbyObject->pReceiveQueue == NULL )
|
||
|
{
|
||
|
DPFX(DPFPREP, 0, "Error allocating receive queue" );
|
||
|
hResultCode = DPNERR_OUTOFMEMORY;
|
||
|
goto ERROR_DPL_RegisterMessageHandler;
|
||
|
}
|
||
|
|
||
|
pdpLobbyObject->pReceiveQueue->SetMessageHandler(static_cast<PVOID>(pdpLobbyObject),DPLMessageHandler);
|
||
|
|
||
|
if (pdpLobbyObject->dwFlags & DPL_OBJECT_FLAG_LOBBIEDAPPLICATION)
|
||
|
{
|
||
|
cSuffix = DPL_MSGQ_OBJECT_SUFFIX_APPLICATION;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
cSuffix = DPL_MSGQ_OBJECT_SUFFIX_CLIENT;
|
||
|
}
|
||
|
|
||
|
// Open application receive message queue
|
||
|
dwCurrentPid = GetCurrentProcessId();
|
||
|
if ((hResultCode = pdpLobbyObject->pReceiveQueue->Open(dwCurrentPid,
|
||
|
cSuffix,DPL_MSGQ_SIZE,DPL_MSGQ_TIMEOUT_IDLE,0)) != DPN_OK)
|
||
|
{
|
||
|
DPFERR("Could not open App Rec Q");
|
||
|
goto ERROR_DPL_RegisterMessageHandler;
|
||
|
}
|
||
|
|
||
|
if ((pdpLobbyObject->hReceiveThread =
|
||
|
DNCreateThread(NULL,(DWORD)NULL,(LPTHREAD_START_ROUTINE)DPLProcessMessageQueue,
|
||
|
static_cast<void*>(pdpLobbyObject->pReceiveQueue),(DWORD)NULL,&dwThreadId)) == NULL)
|
||
|
{
|
||
|
DPFERR("CreateThread() failed");
|
||
|
hResultCode = DPNERR_GENERIC;
|
||
|
pdpLobbyObject->pReceiveQueue->Close();
|
||
|
goto ERROR_DPL_RegisterMessageHandler;
|
||
|
}
|
||
|
|
||
|
pdpLobbyObject->pReceiveQueue->WaitForReceiveThread(INFINITE);
|
||
|
|
||
|
if (pdpLobbyObject->dwFlags & DPL_OBJECT_FLAG_LOBBIEDAPPLICATION)
|
||
|
{
|
||
|
DPFX(DPFPREP, 5,"Attempt lobby connection");
|
||
|
|
||
|
hResultCode = DPLAttemptLobbyConnection(pdpLobbyObject);
|
||
|
|
||
|
if ( hResultCode == DPN_OK)
|
||
|
{
|
||
|
if( pdpnhConnection )
|
||
|
*pdpnhConnection = pdpLobbyObject->dpnhLaunchedConnection;
|
||
|
|
||
|
DPFX(DPFPREP, 5,"Application was lobby launched");
|
||
|
DPFX(DPFPREP, 5,"Waiting for true connect notification" );
|
||
|
|
||
|
DWORD dwReturnValue = DNWaitForSingleObject( pdpLobbyObject->hConnectEvent, DPL_LOBBYLAUNCHED_CONNECT_TIMEOUT );
|
||
|
|
||
|
DNASSERT( dwReturnValue == WAIT_OBJECT_0 );
|
||
|
}
|
||
|
else if( hResultCode != DPNERR_TIMEDOUT )
|
||
|
{
|
||
|
DPFX(DPFPREP, 5,"Application was not lobby launched");
|
||
|
|
||
|
if( pdpnhConnection )
|
||
|
*pdpnhConnection = NULL;
|
||
|
|
||
|
// Need to reset return code to OK.. this is not an error
|
||
|
hResultCode = DPN_OK;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
DPFERR( "App was lobby launched but timed out establishing a connection" );
|
||
|
if( pdpnhConnection )
|
||
|
*pdpnhConnection = NULL;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
EXIT_DPL_RegisterMessageHandler:
|
||
|
|
||
|
DPF_RETURN(hResultCode);
|
||
|
|
||
|
ERROR_DPL_RegisterMessageHandler:
|
||
|
|
||
|
if (pdpLobbyObject->dwFlags & DPL_OBJECT_FLAG_LOBBIEDAPPLICATION )
|
||
|
{
|
||
|
DNInterlockedExchange((LONG*)&g_lLobbyAppCount, 0);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
DNASSERT(pdpLobbyObject->dwFlags & DPL_OBJECT_FLAG_LOBBYCLIENT);
|
||
|
DNInterlockedExchange((LONG*)&g_lLobbyClientCount, 0);
|
||
|
}
|
||
|
|
||
|
goto EXIT_DPL_RegisterMessageHandler;
|
||
|
}
|
||
|
|
||
|
#undef DPF_MODNAME
|
||
|
#define DPF_MODNAME "DPL_Close"
|
||
|
|
||
|
STDMETHODIMP DPL_Close(PVOID pv, const DWORD dwFlags )
|
||
|
{
|
||
|
HRESULT hResultCode;
|
||
|
DWORD dwNumHandles;
|
||
|
DPNHANDLE *prgHandles;
|
||
|
DWORD dw;
|
||
|
DIRECTPLAYLOBBYOBJECT *pdpLobbyObject;
|
||
|
DPL_CONNECTION *pConnection;
|
||
|
|
||
|
DPFX(DPFPREP, 3,"Parameters: (none)");
|
||
|
|
||
|
#ifndef DPNBUILD_NOPARAMVAL
|
||
|
TRY
|
||
|
{
|
||
|
#endif // !DPNBUILD_NOPARAMVAL
|
||
|
pdpLobbyObject = static_cast<DIRECTPLAYLOBBYOBJECT*>(GET_OBJECT_FROM_INTERFACE(pv));
|
||
|
|
||
|
#ifndef DPNBUILD_NOPARAMVAL
|
||
|
// TODO: MASONB: Why no paramval flag wrapping this?
|
||
|
if( FAILED( hResultCode = DPL_ValidateClose( pv, dwFlags ) ) )
|
||
|
{
|
||
|
DPFX(DPFPREP, 0, "Error validating close params hr=[0x%lx]", hResultCode );
|
||
|
return hResultCode;
|
||
|
}
|
||
|
|
||
|
// Ensure we've been initialized
|
||
|
if (pdpLobbyObject->pReceiveQueue == NULL)
|
||
|
{
|
||
|
DPFERR("Already closed");
|
||
|
return DPNERR_UNINITIALIZED;
|
||
|
}
|
||
|
}
|
||
|
EXCEPT(EXCEPTION_EXECUTE_HANDLER)
|
||
|
{
|
||
|
DPFERR("Invalid object" );
|
||
|
return DPNERR_INVALIDOBJECT;
|
||
|
}
|
||
|
#endif // !DPNBUILD_NOPARAMVAL
|
||
|
|
||
|
// Shutdown the queue first to ensure that we don't end up shutting down a connection
|
||
|
// twice! (E.g. disconnect comes in as we are disconnecting it).
|
||
|
if (pdpLobbyObject->pReceiveQueue)
|
||
|
{
|
||
|
if (pdpLobbyObject->pReceiveQueue->IsOpen())
|
||
|
{
|
||
|
|
||
|
// Ask receive thread to terminate
|
||
|
DPFX(DPFPREP, 5,"Terminate Receive Msg Thread");
|
||
|
pdpLobbyObject->pReceiveQueue->Terminate();
|
||
|
|
||
|
// Wait for termination to occur
|
||
|
if (DNWaitForSingleObject(pdpLobbyObject->hReceiveThread,INFINITE) != WAIT_OBJECT_0)
|
||
|
{
|
||
|
hResultCode = DPNERR_GENERIC;
|
||
|
DPFERR("WaitForSingleObject failed");
|
||
|
}
|
||
|
pdpLobbyObject->pReceiveQueue->Close();
|
||
|
|
||
|
if (pdpLobbyObject->pReceiveQueue)
|
||
|
{
|
||
|
delete pdpLobbyObject->pReceiveQueue;
|
||
|
pdpLobbyObject->pReceiveQueue = NULL;
|
||
|
}
|
||
|
|
||
|
if (pdpLobbyObject->hReceiveThread)
|
||
|
{
|
||
|
DNCloseHandle(pdpLobbyObject->hReceiveThread);
|
||
|
pdpLobbyObject->hReceiveThread = NULL;
|
||
|
}
|
||
|
|
||
|
if (pdpLobbyObject->hConnectEvent)
|
||
|
{
|
||
|
DNCloseHandle(pdpLobbyObject->hConnectEvent);
|
||
|
pdpLobbyObject->hConnectEvent = NULL;
|
||
|
}
|
||
|
|
||
|
if (pdpLobbyObject->hLobbyLaunchConnectEvent)
|
||
|
{
|
||
|
DNCloseHandle(pdpLobbyObject->hLobbyLaunchConnectEvent);
|
||
|
pdpLobbyObject->hLobbyLaunchConnectEvent = NULL;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Enumerate handles outstanding
|
||
|
dwNumHandles = 0;
|
||
|
prgHandles = NULL;
|
||
|
hResultCode = DPLConnectionEnum(pdpLobbyObject,prgHandles,&dwNumHandles);
|
||
|
while (hResultCode == DPNERR_BUFFERTOOSMALL)
|
||
|
{
|
||
|
if (prgHandles)
|
||
|
DNFree(prgHandles);
|
||
|
|
||
|
if ((prgHandles = static_cast<DPNHANDLE*>(DNMalloc(dwNumHandles*sizeof(DPNHANDLE)))) != NULL)
|
||
|
{
|
||
|
hResultCode = DPLConnectionEnum(pdpLobbyObject,prgHandles,&dwNumHandles);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
DPFERR("Could not allocate space for handle array");
|
||
|
hResultCode = DPNERR_OUTOFMEMORY;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Send DISCONNECTs to all attached msg queues, for which there are handles
|
||
|
if (hResultCode == DPN_OK)
|
||
|
{
|
||
|
for (dw = 0 ; dw < dwNumHandles ; dw++)
|
||
|
{
|
||
|
hResultCode = DPLConnectionFind(pdpLobbyObject,prgHandles[dw],&pConnection,TRUE );
|
||
|
|
||
|
if( SUCCEEDED( hResultCode ) )
|
||
|
{
|
||
|
|
||
|
hResultCode = DPLConnectionDisconnect(pdpLobbyObject,prgHandles[dw]);
|
||
|
|
||
|
if( FAILED( hResultCode ) )
|
||
|
{
|
||
|
DPFX(DPFPREP, 0, "Error disconnecting connection 0x%x", hResultCode );
|
||
|
}
|
||
|
|
||
|
DPLConnectionRelease( pdpLobbyObject,prgHandles[dw]);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Errors above are irrelevant, it's quite possible after building the list of outstanding
|
||
|
// connections that before we attempt to close the list one has gone away.
|
||
|
//
|
||
|
hResultCode = DPN_OK;
|
||
|
}
|
||
|
|
||
|
if (prgHandles)
|
||
|
{
|
||
|
DNFree(prgHandles);
|
||
|
prgHandles = NULL;
|
||
|
}
|
||
|
|
||
|
if (pdpLobbyObject->dwFlags & DPL_OBJECT_FLAG_LOBBIEDAPPLICATION )
|
||
|
{
|
||
|
DNInterlockedExchange((LONG*)&g_lLobbyAppCount, 0);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
DNASSERT(pdpLobbyObject->dwFlags & DPL_OBJECT_FLAG_LOBBYCLIENT);
|
||
|
DNInterlockedExchange((LONG*)&g_lLobbyClientCount, 0);
|
||
|
}
|
||
|
|
||
|
DPF_RETURN( hResultCode );
|
||
|
}
|
||
|
|
||
|
|
||
|
#undef DPF_MODNAME
|
||
|
#define DPF_MODNAME "DPL_Send"
|
||
|
|
||
|
STDMETHODIMP DPL_Send(PVOID pv,
|
||
|
const DPNHANDLE hTarget,
|
||
|
BYTE *const pBuffer,
|
||
|
const DWORD dwBufferSize,
|
||
|
const DWORD dwFlags)
|
||
|
{
|
||
|
HRESULT hResultCode;
|
||
|
DPL_CONNECTION *pdplConnection;
|
||
|
DIRECTPLAYLOBBYOBJECT *pdpLobbyObject;
|
||
|
DPNHANDLE *hTargets = NULL;
|
||
|
DWORD dwNumTargets = 0;
|
||
|
DWORD dwTargetIndex = 0;
|
||
|
|
||
|
DPFX(DPFPREP, 3,"Parameters: hTarget [0x%lx], pBuffer [0x%p], dwBufferSize [%ld], dwFlags [0x%lx]",
|
||
|
hTarget,pBuffer,dwBufferSize,dwFlags);
|
||
|
|
||
|
#ifndef DPNBUILD_NOPARAMVAL
|
||
|
TRY
|
||
|
{
|
||
|
#endif // !DPNBUILD_NOPARAMVAL
|
||
|
pdpLobbyObject = static_cast<DIRECTPLAYLOBBYOBJECT*>(GET_OBJECT_FROM_INTERFACE(pv));
|
||
|
|
||
|
#ifndef DPNBUILD_NOPARAMVAL
|
||
|
if( pdpLobbyObject->dwFlags & DPL_OBJECT_FLAG_PARAMVALIDATION )
|
||
|
{
|
||
|
if( FAILED( hResultCode = DPL_ValidateSend( pv, hTarget, pBuffer, dwBufferSize, dwFlags ) ) )
|
||
|
{
|
||
|
DPFX(DPFPREP, 0, "Error validating send params hr=[0x%lx]", hResultCode );
|
||
|
DPF_RETURN( hResultCode );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Ensure we've been initialized
|
||
|
if (pdpLobbyObject->pReceiveQueue == NULL)
|
||
|
{
|
||
|
DPFERR("Not initialized");
|
||
|
DPF_RETURN(DPNERR_UNINITIALIZED);
|
||
|
}
|
||
|
}
|
||
|
EXCEPT(EXCEPTION_EXECUTE_HANDLER)
|
||
|
{
|
||
|
DPFERR("Invalid object" );
|
||
|
DPF_RETURN(DPNERR_INVALIDOBJECT);
|
||
|
}
|
||
|
#endif // !DPNBUILD_NOPARAMVAL
|
||
|
|
||
|
if( hTarget == DPLHANDLE_ALLCONNECTIONS )
|
||
|
{
|
||
|
dwNumTargets = 0;
|
||
|
|
||
|
// We need loop so if someone adds a connection during our run
|
||
|
// it gets added to our list
|
||
|
//
|
||
|
while( 1 )
|
||
|
{
|
||
|
hResultCode = DPLConnectionEnum( pdpLobbyObject, hTargets, &dwNumTargets );
|
||
|
|
||
|
if( hResultCode == DPNERR_BUFFERTOOSMALL )
|
||
|
{
|
||
|
if( hTargets )
|
||
|
{
|
||
|
delete [] hTargets;
|
||
|
}
|
||
|
|
||
|
hTargets = new DPNHANDLE[dwNumTargets];
|
||
|
|
||
|
if( hTargets == NULL )
|
||
|
{
|
||
|
DPFERR("Error allocating memory" );
|
||
|
dwNumTargets = 0;
|
||
|
hResultCode = DPNERR_OUTOFMEMORY;
|
||
|
goto EXIT_AND_CLEANUP;
|
||
|
}
|
||
|
|
||
|
memset( hTargets, 0x00, sizeof(DPNHANDLE)*dwNumTargets);
|
||
|
|
||
|
continue;
|
||
|
}
|
||
|
else if( FAILED( hResultCode ) )
|
||
|
{
|
||
|
DPFX(DPFPREP, 0, "Error getting list of connections hr=0x%x", hResultCode );
|
||
|
break;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Failed getting connection information
|
||
|
if( FAILED( hResultCode ) )
|
||
|
{
|
||
|
if( hTargets )
|
||
|
{
|
||
|
delete [] hTargets;
|
||
|
hTargets = NULL;
|
||
|
}
|
||
|
dwNumTargets = 0;
|
||
|
goto EXIT_AND_CLEANUP;
|
||
|
}
|
||
|
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
hTargets = new DPNHANDLE[1]; // We use array delete below so we need array new
|
||
|
|
||
|
if( hTargets == NULL )
|
||
|
{
|
||
|
DPFERR("Error allocating memory" );
|
||
|
dwNumTargets = 0;
|
||
|
hResultCode = DPNERR_OUTOFMEMORY;
|
||
|
goto EXIT_AND_CLEANUP;
|
||
|
}
|
||
|
|
||
|
dwNumTargets = 1;
|
||
|
hTargets[0] = hTarget;
|
||
|
}
|
||
|
|
||
|
for( dwTargetIndex = 0; dwTargetIndex < dwNumTargets; dwTargetIndex++ )
|
||
|
{
|
||
|
if ((hResultCode = DPLConnectionFind(pdpLobbyObject,hTargets[dwTargetIndex],&pdplConnection,TRUE)) != DPN_OK)
|
||
|
{
|
||
|
DPFERR("Invalid send target");
|
||
|
DisplayDNError(0,hResultCode);
|
||
|
hResultCode = DPNERR_INVALIDHANDLE;
|
||
|
goto EXIT_AND_CLEANUP;
|
||
|
}
|
||
|
|
||
|
DNASSERT(pdplConnection->pSendQueue != NULL);
|
||
|
|
||
|
if (!pdplConnection->pSendQueue->IsReceiving())
|
||
|
{
|
||
|
DPFERR("Other side is not listening");
|
||
|
DPLConnectionRelease(pdpLobbyObject,hTarget);
|
||
|
hResultCode = DPNERR_INVALIDHANDLE;
|
||
|
goto EXIT_AND_CLEANUP;
|
||
|
}
|
||
|
|
||
|
hResultCode = pdplConnection->pSendQueue->Send(pBuffer,dwBufferSize,INFINITE,DPL_MSGQ_MSGFLAGS_USER2,0);
|
||
|
|
||
|
if( FAILED( hResultCode ) )
|
||
|
{
|
||
|
DPFX(DPFPREP, 0, "Error sending to connection 0x%x hr=0x%x", hTargets[dwTargetIndex], hResultCode );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
EXIT_AND_CLEANUP:
|
||
|
|
||
|
for( dwTargetIndex = 0; dwTargetIndex < dwNumTargets; dwTargetIndex++ )
|
||
|
{
|
||
|
if( hTargets[dwTargetIndex] )
|
||
|
DPLConnectionRelease(pdpLobbyObject,hTargets[dwTargetIndex]);
|
||
|
}
|
||
|
|
||
|
if( hTargets )
|
||
|
delete [] hTargets;
|
||
|
|
||
|
DPF_RETURN(hResultCode);
|
||
|
|
||
|
}
|
||
|
|
||
|
#undef DPF_MODNAME
|
||
|
#define DPF_MODNAME "DPLReceiveIdleTimeout"
|
||
|
HRESULT DPLReceiveIdleTimeout(DIRECTPLAYLOBBYOBJECT *const pdpLobbyObject,
|
||
|
const DPNHANDLE hSender)
|
||
|
{
|
||
|
DPL_CONNECTION *pConnection;
|
||
|
CBilink* pblTemp;
|
||
|
CBilink blRemove;
|
||
|
|
||
|
blRemove.Initialize();
|
||
|
|
||
|
DPFX(DPFPREP, 6, "(%p) Enumerating processes, checking for exit", pdpLobbyObject );
|
||
|
|
||
|
DNEnterCriticalSection(&pdpLobbyObject->m_cs);
|
||
|
|
||
|
// Go through the list of all connections and build up a list of the ones that
|
||
|
// need to be removed.
|
||
|
pblTemp = pdpLobbyObject->m_blConnections.GetNext();
|
||
|
while (pblTemp != &pdpLobbyObject->m_blConnections)
|
||
|
{
|
||
|
pConnection = CONTAINING_OBJECT(pblTemp, DPL_CONNECTION, m_blLobbyObjectLinkage);
|
||
|
|
||
|
pblTemp = pblTemp->GetNext();
|
||
|
|
||
|
DWORD dwExitCode=0;
|
||
|
if (DNGetExitCodeProcess(pConnection->hTargetProcess, &dwExitCode)==FALSE || dwExitCode!=STILL_ACTIVE)
|
||
|
{
|
||
|
DPFX(DPFPREP, 5, "(%p) Process exit detected hTargetProcess %u dwExitCode %u GetLastError %u",
|
||
|
pdpLobbyObject, HANDLE_FROM_DNHANDLE(pConnection->hTargetProcess), dwExitCode, GetLastError());
|
||
|
// Take the connection off of the Lobby Object's list and on a temporary list
|
||
|
pConnection->m_blLobbyObjectLinkage.RemoveFromList();
|
||
|
pdpLobbyObject->m_dwConnectionCount--;
|
||
|
pConnection->m_blLobbyObjectLinkage.InsertBefore(&blRemove);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
DNLeaveCriticalSection(&pdpLobbyObject->m_cs);
|
||
|
|
||
|
// Go through the list of removed connections and signal the user
|
||
|
pblTemp = blRemove.GetNext();
|
||
|
while (pblTemp != &blRemove)
|
||
|
{
|
||
|
pConnection = CONTAINING_OBJECT(pblTemp, DPL_CONNECTION, m_blLobbyObjectLinkage);
|
||
|
|
||
|
pblTemp = pblTemp->GetNext();
|
||
|
|
||
|
pConnection->m_blLobbyObjectLinkage.RemoveFromList();
|
||
|
|
||
|
// Process has exited..
|
||
|
DPFX(DPFPREP, 6, "(%p) Process has exited PID %u hProcess", pdpLobbyObject,
|
||
|
pConnection->dwTargetProcessIdentity, HANDLE_FROM_DNHANDLE(pConnection->hTargetProcess ));
|
||
|
DPLConnectionReceiveDisconnect( pdpLobbyObject, pConnection->hConnect, NULL, DPNERR_CONNECTIONLOST );
|
||
|
}
|
||
|
|
||
|
return DPN_OK;
|
||
|
}
|
||
|
|
||
|
#undef DPF_MODNAME
|
||
|
#define DPF_MODNAME "DPLReceiveUserMessage"
|
||
|
|
||
|
HRESULT DPLReceiveUserMessage(DIRECTPLAYLOBBYOBJECT *const pdpLobbyObject,
|
||
|
const DPNHANDLE hSender,
|
||
|
BYTE *const pBuffer,
|
||
|
const DWORD dwBufferSize)
|
||
|
{
|
||
|
HRESULT hResultCode;
|
||
|
DPL_MESSAGE_RECEIVE Msg;
|
||
|
|
||
|
Msg.dwSize = sizeof(DPL_MESSAGE_RECEIVE);
|
||
|
Msg.pBuffer = pBuffer;
|
||
|
Msg.dwBufferSize = dwBufferSize;
|
||
|
Msg.hSender = hSender;
|
||
|
|
||
|
hResultCode = DPLConnectionGetContext( pdpLobbyObject, hSender, &Msg.pvConnectionContext );
|
||
|
|
||
|
// Failed to get the connection's context -- strange, but we're going to indicate anyhow.
|
||
|
//
|
||
|
if( FAILED( hResultCode ) )
|
||
|
{
|
||
|
DPFX(DPFPREP, 0, "Failed getting connection context hResultCode = 0x%x", hResultCode );
|
||
|
}
|
||
|
|
||
|
hResultCode = (pdpLobbyObject->pfnMessageHandler)(pdpLobbyObject->pvUserContext,
|
||
|
DPL_MSGID_RECEIVE,
|
||
|
reinterpret_cast<BYTE*>(&Msg));
|
||
|
|
||
|
DPFX(DPFPREP, 3,"Returning: [0x%lx]",hResultCode);
|
||
|
return(hResultCode);
|
||
|
}
|
||
|
|
||
|
|
||
|
#undef DPF_MODNAME
|
||
|
#define DPF_MODNAME "DPLMessageHandler"
|
||
|
|
||
|
HRESULT DPLMessageHandler(PVOID pvContext,
|
||
|
const DPNHANDLE hSender,
|
||
|
DWORD dwMessageFlags,
|
||
|
BYTE *const pBuffer,
|
||
|
const DWORD dwBufferSize)
|
||
|
{
|
||
|
DIRECTPLAYLOBBYOBJECT *pdpLobbyObject;
|
||
|
HRESULT hResultCode;
|
||
|
DWORD *pdwMsgId;
|
||
|
|
||
|
DPFX(DPFPREP, 3,"Parameters: hSender [0x%x], pBuffer [0x%p], dwBufferSize [%ld]",
|
||
|
hSender,pBuffer,dwBufferSize);
|
||
|
|
||
|
DNASSERT(pBuffer != NULL);
|
||
|
|
||
|
/*if (dwBufferSize < sizeof(DWORD))
|
||
|
{
|
||
|
DPFERR("Invalid message");
|
||
|
return(DPNERR_GENERIC);
|
||
|
}*/
|
||
|
|
||
|
pdpLobbyObject = static_cast<DIRECTPLAYLOBBYOBJECT*>(pvContext);
|
||
|
pdwMsgId = reinterpret_cast<DWORD*>(pBuffer);
|
||
|
|
||
|
if( dwMessageFlags & DPL_MSGQ_MSGFLAGS_USER1 )
|
||
|
{
|
||
|
DPFX(DPFPREP, 5,"Received INTERNAL message");
|
||
|
switch(*pdwMsgId)
|
||
|
{
|
||
|
case DPL_MSGID_INTERNAL_IDLE_TIMEOUT:
|
||
|
{
|
||
|
DPFX(DPFPREP, 5,"Received: DPL_MSGID_INTERNAL_IDLE_TIMEOUT" );
|
||
|
DPLReceiveIdleTimeout(pdpLobbyObject,hSender);
|
||
|
break;
|
||
|
}
|
||
|
case DPL_MSGID_INTERNAL_DISCONNECT:
|
||
|
{
|
||
|
DPFX(DPFPREP, 5,"Received: DPL_MSGID_INTERNAL_DISCONNECT");
|
||
|
DPLConnectionReceiveDisconnect(pdpLobbyObject,hSender,pBuffer,DPN_OK);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
case DPL_MSGID_INTERNAL_CONNECT_REQ:
|
||
|
{
|
||
|
DPFX(DPFPREP, 5,"Received: DPL_MSGID_INTERNAL_CONNECT_REQ");
|
||
|
DPLConnectionReceiveREQ(pdpLobbyObject,pBuffer);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
case DPL_MSGID_INTERNAL_CONNECT_ACK:
|
||
|
{
|
||
|
DPFX(DPFPREP, 5,"Received: DPL_MSGID_INTERNAL_CONNECT_ACK");
|
||
|
DPLConnectionReceiveACK(pdpLobbyObject,hSender,pBuffer);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
case DPL_MSGID_INTERNAL_UPDATE_STATUS:
|
||
|
{
|
||
|
DPFX(DPFPREP, 5,"Received: DPL_MSGID_INTERNAL_UPDATE_STATUS");
|
||
|
DPLUpdateAppStatus(pdpLobbyObject,hSender,pBuffer);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
case DPL_MSGID_INTERNAL_CONNECTION_SETTINGS:
|
||
|
{
|
||
|
DPFX(DPFPREP, 5,"Received: DPL_MSGID_INTERNAL_CONNECTION_SETTINGS");
|
||
|
DPLUpdateConnectionSettings(pdpLobbyObject,hSender,pBuffer);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
default:
|
||
|
{
|
||
|
DPFX(DPFPREP, 5,"Received: Unknown message [0x%lx]",*pdwMsgId);
|
||
|
DNASSERT(FALSE);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else if( dwMessageFlags & DPL_MSGQ_MSGFLAGS_USER2 )
|
||
|
{
|
||
|
DNASSERT( !(dwMessageFlags & DPL_MSGQ_MSGFLAGS_QUEUESYSTEM) );
|
||
|
DPFX(DPFPREP, 5,"Received USER message");
|
||
|
DPLReceiveUserMessage(pdpLobbyObject,hSender,pBuffer,dwBufferSize);
|
||
|
}
|
||
|
|
||
|
hResultCode = DPN_OK;
|
||
|
|
||
|
DPFX(DPFPREP, 3,"Returning: [0x%lx]",hResultCode);
|
||
|
return(hResultCode);
|
||
|
}
|
||
|
|
||
|
// DPLSendConnectionSettings
|
||
|
//
|
||
|
// This function is used to send a connection settings update message
|
||
|
#undef DPF_MODNAME
|
||
|
#define DPF_MODNAME "DPLSendConnectionSettings"
|
||
|
HRESULT DPLSendConnectionSettings( DIRECTPLAYLOBBYOBJECT * const pdpLobbyObject,
|
||
|
const DPNHANDLE hConnection )
|
||
|
{
|
||
|
BYTE *pbTransmitBuffer = NULL;
|
||
|
DWORD dwTransmitBufferSize = 0;
|
||
|
DPL_INTERNAL_CONNECTION_SETTINGS_UPDATE *pdplMsgSettings = NULL;
|
||
|
DPL_CONNECTION *pdplConnection = NULL;
|
||
|
CPackedBuffer PackBuffer;
|
||
|
|
||
|
HRESULT hResultCode = DPN_OK;
|
||
|
|
||
|
hResultCode = DPLConnectionFind(pdpLobbyObject, hConnection, &pdplConnection, TRUE );
|
||
|
|
||
|
if( FAILED( hResultCode ) )
|
||
|
{
|
||
|
DPFERR( "Unable to find specified connection" );
|
||
|
return hResultCode;
|
||
|
}
|
||
|
|
||
|
// Grab lock to prevent other people from interfering
|
||
|
DNEnterCriticalSection( &pdplConnection->csLock );
|
||
|
|
||
|
PackBuffer.Initialize( NULL, 0 );
|
||
|
|
||
|
PackBuffer.AddToFront( NULL, sizeof( DPL_INTERNAL_CONNECTION_SETTINGS_UPDATE_HEADER ) );
|
||
|
|
||
|
if( pdplConnection->pConnectionSettings )
|
||
|
{
|
||
|
pdplConnection->pConnectionSettings->BuildWireStruct( &PackBuffer );
|
||
|
}
|
||
|
|
||
|
dwTransmitBufferSize = PackBuffer.GetSizeRequired();
|
||
|
|
||
|
pbTransmitBuffer = new BYTE[ dwTransmitBufferSize ];
|
||
|
|
||
|
if( !pbTransmitBuffer )
|
||
|
{
|
||
|
DPFX( DPFPREP, 0, "Error allocating memory" );
|
||
|
hResultCode = DPNERR_OUTOFMEMORY;
|
||
|
goto DPLSENDCONNECTSETTINGS_DONE;
|
||
|
}
|
||
|
|
||
|
pdplMsgSettings = (DPL_INTERNAL_CONNECTION_SETTINGS_UPDATE *) pbTransmitBuffer;
|
||
|
|
||
|
PackBuffer.Initialize( pbTransmitBuffer, dwTransmitBufferSize );
|
||
|
|
||
|
DNASSERT( pdplMsgSettings );
|
||
|
|
||
|
hResultCode = PackBuffer.AddToFront( NULL, sizeof( DPL_INTERNAL_CONNECTION_SETTINGS_UPDATE_HEADER ) );
|
||
|
|
||
|
if( FAILED( hResultCode ) )
|
||
|
{
|
||
|
DPFX( DPFPREP, 0, "Error adding main struct hr [0x%x]", hResultCode );
|
||
|
goto DPLSENDCONNECTSETTINGS_DONE;
|
||
|
}
|
||
|
|
||
|
if( pdplConnection->pConnectionSettings )
|
||
|
{
|
||
|
hResultCode = pdplConnection->pConnectionSettings->BuildWireStruct( &PackBuffer );
|
||
|
|
||
|
if( FAILED( hResultCode ) )
|
||
|
{
|
||
|
DPFX( DPFPREP, 0, "Error adding connect struct hr [0x%x]", hResultCode );
|
||
|
goto DPLSENDCONNECTSETTINGS_DONE;
|
||
|
}
|
||
|
|
||
|
pdplMsgSettings->dwConnectionSettingsSize = 1;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
pdplMsgSettings->dwConnectionSettingsSize = 0;
|
||
|
}
|
||
|
|
||
|
pdplMsgSettings->dwMsgId = DPL_MSGID_INTERNAL_CONNECTION_SETTINGS;
|
||
|
|
||
|
if (!pdplConnection->pSendQueue->IsReceiving())
|
||
|
{
|
||
|
DPFERR("Other side is not receiving");
|
||
|
goto DPLSENDCONNECTSETTINGS_DONE;
|
||
|
}
|
||
|
|
||
|
hResultCode = pdplConnection->pSendQueue->Send(reinterpret_cast<BYTE*>(pdplMsgSettings),
|
||
|
PackBuffer.GetSizeRequired(),
|
||
|
INFINITE,
|
||
|
DPL_MSGQ_MSGFLAGS_USER1,
|
||
|
0);
|
||
|
if ( FAILED( hResultCode ) )
|
||
|
{
|
||
|
DPFX(DPFPREP, 0, "Could not send connect settings hr [0x%x]", hResultCode );
|
||
|
goto DPLSENDCONNECTSETTINGS_DONE;
|
||
|
}
|
||
|
|
||
|
hResultCode = DPN_OK;
|
||
|
|
||
|
DPLSENDCONNECTSETTINGS_DONE:
|
||
|
|
||
|
if( pbTransmitBuffer )
|
||
|
delete [] pbTransmitBuffer;
|
||
|
|
||
|
DNLeaveCriticalSection( &pdplConnection->csLock );
|
||
|
|
||
|
DPLConnectionRelease(pdpLobbyObject,hConnection);
|
||
|
|
||
|
return hResultCode;
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
// DPLUpdateConnectionSettings
|
||
|
//
|
||
|
// This function is called when a connection settings update message has been received.
|
||
|
//
|
||
|
#undef DPF_MODNAME
|
||
|
#define DPF_MODNAME "DPLUpdateConnectionSettings"
|
||
|
HRESULT DPLUpdateConnectionSettings(DIRECTPLAYLOBBYOBJECT *const pdpLobbyObject,
|
||
|
const DPNHANDLE hSender,
|
||
|
BYTE *const pBuffer )
|
||
|
{
|
||
|
HRESULT hr;
|
||
|
DPL_MESSAGE_CONNECTION_SETTINGS MsgConnectionSettings;
|
||
|
DWORD dwSettingsBufferSize = 0;
|
||
|
BOOL fAddressReferences = FALSE;
|
||
|
CConnectionSettings *pConnectionSettings = NULL;
|
||
|
DPL_INTERNAL_CONNECTION_SETTINGS_UPDATE *pConnectionSettingsMsg = NULL;
|
||
|
|
||
|
DPFX(DPFPREP, 3,"Parameters: pBuffer [0x%p]",pBuffer);
|
||
|
|
||
|
DNASSERT(pdpLobbyObject != NULL);
|
||
|
DNASSERT(pBuffer != NULL);
|
||
|
|
||
|
pConnectionSettingsMsg = (DPL_INTERNAL_CONNECTION_SETTINGS_UPDATE *) pBuffer;
|
||
|
|
||
|
if( pConnectionSettingsMsg->dwConnectionSettingsSize )
|
||
|
{
|
||
|
pConnectionSettings = new CConnectionSettings();
|
||
|
|
||
|
if( !pConnectionSettings )
|
||
|
{
|
||
|
DPFX( DPFPREP, 0, "Error allocating connection settings" );
|
||
|
hr = DPNERR_OUTOFMEMORY;
|
||
|
goto UPDATESETTINGS_FAILURE;
|
||
|
}
|
||
|
|
||
|
hr = pConnectionSettings->Initialize( &pConnectionSettingsMsg->dplConnectionSettings, (UNALIGNED BYTE *) pConnectionSettingsMsg );
|
||
|
|
||
|
if( FAILED( hr ) )
|
||
|
{
|
||
|
DPFX( DPFPREP, 0, "Error building structure from wire struct hr [0x%x]", hr );
|
||
|
goto UPDATESETTINGS_FAILURE;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Set the connection settings on the object
|
||
|
hr = DPLConnectionSetConnectSettings( pdpLobbyObject, hSender, pConnectionSettings );
|
||
|
|
||
|
if( FAILED( hr ) )
|
||
|
{
|
||
|
DPFX(DPFPREP, 0, "Error setting connection settings hr = 0x%x", hr );
|
||
|
goto UPDATESETTINGS_FAILURE;
|
||
|
}
|
||
|
|
||
|
// Setup message to indicate to user
|
||
|
MsgConnectionSettings.dwSize = sizeof(DPL_MESSAGE_CONNECTION_SETTINGS);
|
||
|
MsgConnectionSettings.hSender = hSender;
|
||
|
|
||
|
if( pConnectionSettings )
|
||
|
MsgConnectionSettings.pdplConnectionSettings = pConnectionSettings->GetConnectionSettings();
|
||
|
else
|
||
|
MsgConnectionSettings.pdplConnectionSettings = NULL;
|
||
|
|
||
|
hr = DPLConnectionGetContext( pdpLobbyObject, hSender, &MsgConnectionSettings.pvConnectionContext );
|
||
|
|
||
|
if( FAILED( hr ) )
|
||
|
{
|
||
|
DPFX(DPFPREP, 0, "Error getting connection's context value" );
|
||
|
goto UPDATESETTINGS_FAILURE;
|
||
|
}
|
||
|
|
||
|
hr = (pdpLobbyObject->pfnMessageHandler)(pdpLobbyObject->pvUserContext,
|
||
|
DPL_MSGID_CONNECTION_SETTINGS,
|
||
|
reinterpret_cast<BYTE*>(&MsgConnectionSettings));
|
||
|
|
||
|
if( FAILED( hr ) )
|
||
|
{
|
||
|
DPFX(DPFPREP, 1, "Error returned from user callback -- ignored hr [0x%x]", hr );
|
||
|
}
|
||
|
|
||
|
|
||
|
return DPN_OK;
|
||
|
|
||
|
UPDATESETTINGS_FAILURE:
|
||
|
|
||
|
if( pConnectionSettings )
|
||
|
delete pConnectionSettings;
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|