Windows2003-3790/multimedia/directx/dplay/dplay8/lobby/dplcommon.cpp
2020-09-30 16:53:55 +02:00

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;
}