- Added global data mutual exclusion

- Removed hack for creating processes suspended now that it's implemented
- Fixed ordinal numbering and added spec stubs
- Fixed EnumConnections callback to fill in all parameters with valid data
- Made direct play allocation/deallocation follow the same pattern as
  direct play lobby
This commit is contained in:
Peter Hunnisett 1999-11-25 22:04:53 +00:00 committed by Alexandre Julliard
parent 6c8a67c3af
commit 88a2954a97
4 changed files with 713 additions and 245 deletions

View File

@ -15,6 +15,12 @@
#include "debugtools.h" #include "debugtools.h"
#include "dpinit.h" #include "dpinit.h"
#include "dplayx_global.h"
/* FIXME: This stuff shouldn't really be here. It indicates a poor architectural coupling */
#include "dplobby.h"
extern HRESULT DPL_CreateCompoundAddress ( LPCDPCOMPOUNDADDRESSELEMENT lpElements, DWORD dwElementCount,
LPVOID lpAddress, LPDWORD lpdwAddressSize, BOOL bAnsiInterface );
DEFAULT_DEBUG_CHANNEL(dplay) DEFAULT_DEBUG_CHANNEL(dplay)
@ -22,12 +28,71 @@ DEFAULT_DEBUG_CHANNEL(dplay)
/***************************************************************************** /*****************************************************************************
* Predeclare the interface implementation structures * Predeclare the interface implementation structures
*/ */
typedef struct IDirectPlayImpl IDirectPlay2AImpl; typedef struct IDirectPlay2Impl IDirectPlay2AImpl;
typedef struct IDirectPlayImpl IDirectPlay2Impl; typedef struct IDirectPlay2Impl IDirectPlay2Impl;
typedef struct IDirectPlayImpl IDirectPlay3AImpl; typedef struct IDirectPlay3Impl IDirectPlay3AImpl;
typedef struct IDirectPlayImpl IDirectPlay3Impl; typedef struct IDirectPlay3Impl IDirectPlay3Impl;
typedef struct IDirectPlayImpl IDirectPlay4AImpl; typedef struct IDirectPlay4Impl IDirectPlay4AImpl;
typedef struct IDirectPlayImpl IDirectPlay4Impl; typedef struct IDirectPlay4Impl IDirectPlay4Impl;
/*****************************************************************************
* IDirectPlay implementation structure
*
* The philosophy behind this extra pointer derefernce is that I wanted to
* have the same structure for all types of objects without having to do
* alot of casting. I also only wanted to implement an interface in the
* object it was "released" with IUnknown interface being implemented in the 1 version.
* Of course, with these new interfaces comes the data required to keep the state required
* by these interfaces. So, basically, the pointers contain the data associated with
* a release. If you use the data associated with release 3 in a release 2 object, you'll
* get a run time trap, as that won't have any data.
*
*/
typedef struct tagDirectPlayIUnknownData
{
DWORD ref;
CRITICAL_SECTION DP_lock;
} DirectPlayIUnknownData;
/* Contains all dp1 and dp2 data members */
typedef struct tagDirectPlay2Data
{
BOOL dummy;
} DirectPlay2Data;
typedef struct tagDirectPlay3Data
{
BOOL connectionInitialized;
} DirectPlay3Data;
typedef struct tagDirectPlay4Data
{
BOOL dummy;
} DirectPlay4Data;
#define DP_IMPL_FIELDS \
DirectPlayIUnknownData* unk; \
DirectPlay2Data* dp2; \
DirectPlay3Data* dp3; \
DirectPlay4Data* dp4;
struct IDirectPlay2Impl
{
ICOM_VFIELD(IDirectPlay2);
DP_IMPL_FIELDS
};
struct IDirectPlay3Impl
{
ICOM_VFIELD(IDirectPlay3);
DP_IMPL_FIELDS
};
struct IDirectPlay4Impl
{
ICOM_VFIELD(IDirectPlay4);
DP_IMPL_FIELDS
};
/* Forward declarations of virtual tables */ /* Forward declarations of virtual tables */
static ICOM_VTABLE(IDirectPlay2) directPlay2AVT; static ICOM_VTABLE(IDirectPlay2) directPlay2AVT;
@ -38,21 +103,107 @@ static ICOM_VTABLE(IDirectPlay2) directPlay2WVT;
static ICOM_VTABLE(IDirectPlay3) directPlay3WVT; static ICOM_VTABLE(IDirectPlay3) directPlay3WVT;
static ICOM_VTABLE(IDirectPlay4) directPlay4WVT; static ICOM_VTABLE(IDirectPlay4) directPlay4WVT;
/***************************************************************************** BOOL DP_CreateIUnknown( LPVOID lpDP )
* IDirectPlay implementation structure
*/
struct IDirectPlayImpl
{ {
/* IUnknown fields */ ICOM_THIS(IDirectPlay2AImpl,lpDP);
ICOM_VFIELD(IDirectPlay4);
DWORD ref;
CRITICAL_SECTION DP_lock;
/* IDirectPlay3Impl fields */
/* none so far */
/* IDirectPlay4Impl fields */
/* none so far */
};
This->unk = (DirectPlayIUnknownData*)HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY,
sizeof( *(This->unk) ) );
if ( This->unk == NULL )
{
return FALSE;
}
InitializeCriticalSection( &This->unk->DP_lock );
IDirectPlay_AddRef( (LPDIRECTPLAY2A)lpDP );
return TRUE;
}
BOOL DP_DestroyIUnknown( LPVOID lpDP )
{
ICOM_THIS(IDirectPlay2AImpl,lpDP);
DeleteCriticalSection( &This->unk->DP_lock );
HeapFree( GetProcessHeap(), 0, This->unk );
return TRUE;
}
BOOL DP_CreateDirectPlay2( LPVOID lpDP )
{
ICOM_THIS(IDirectPlay2AImpl,lpDP);
This->dp2 = (DirectPlay2Data*)HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY,
sizeof( *(This->dp2) ) );
if ( This->dp2 == NULL )
{
return FALSE;
}
return TRUE;
}
BOOL DP_DestroyDirectPlay2( LPVOID lpDP )
{
ICOM_THIS(IDirectPlay2AImpl,lpDP);
/* Delete the contents */
HeapFree( GetProcessHeap(), 0, This->dp2 );
return TRUE;
}
BOOL DP_CreateDirectPlay3( LPVOID lpDP )
{
ICOM_THIS(IDirectPlay3AImpl,lpDP);
This->dp3 = (DirectPlay3Data*)HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY,
sizeof( *(This->dp3) ) );
if ( This->dp3 == NULL )
{
return FALSE;
}
This->dp3->connectionInitialized = FALSE;
return TRUE;
}
BOOL DP_DestroyDirectPlay3( LPVOID lpDP )
{
ICOM_THIS(IDirectPlay3AImpl,lpDP);
/* Delete the contents */
HeapFree( GetProcessHeap(), 0, This->dp3 );
return TRUE;
}
BOOL DP_CreateDirectPlay4( LPVOID lpDP )
{
ICOM_THIS(IDirectPlay4AImpl,lpDP);
This->dp4 = (DirectPlay4Data*)HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY,
sizeof( *(This->dp4) ) );
if ( This->dp4 == NULL )
{
return FALSE;
}
return TRUE;
}
BOOL DP_DestroyDirectPlay4( LPVOID lpDP )
{
ICOM_THIS(IDirectPlay3AImpl,lpDP);
/* Delete the contents */
HeapFree( GetProcessHeap(), 0, This->dp4 );
return TRUE;
}
/* Get a new interface. To be used by QueryInterface. */ /* Get a new interface. To be used by QueryInterface. */
@ -63,134 +214,188 @@ HRESULT directPlay_QueryInterface
if( IsEqualGUID( &IID_IDirectPlay2, riid ) ) if( IsEqualGUID( &IID_IDirectPlay2, riid ) )
{ {
IDirectPlay2Impl* lpDP; *ppvObj = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY,
sizeof( IDirectPlay2Impl ) );
lpDP = (IDirectPlay2Impl*)HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, if( *ppvObj == NULL )
sizeof( *lpDP ) );
if( !lpDP )
{ {
return DPERR_OUTOFMEMORY; return DPERR_OUTOFMEMORY;
} }
ICOM_VTBL(lpDP) = &directPlay2WVT; /* new scope for variable declaration */
{
ICOM_THIS(IDirectPlay2Impl,*ppvObj);
InitializeCriticalSection( &lpDP->DP_lock ); ICOM_VTBL(This) = &directPlay2WVT;
IDirectPlayX_AddRef( (LPDIRECTPLAY2A)lpDP );
*ppvObj = lpDP; if ( DP_CreateIUnknown( (LPVOID)This ) &&
DP_CreateDirectPlay2( (LPVOID)This )
)
{
return S_OK;
}
return S_OK; }
goto error;
} }
else if( IsEqualGUID( &IID_IDirectPlay2A, riid ) ) else if( IsEqualGUID( &IID_IDirectPlay2A, riid ) )
{ {
IDirectPlay2AImpl* lpDP; *ppvObj = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY,
sizeof( IDirectPlay2AImpl ) );
lpDP = (IDirectPlay2AImpl*)HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, if( *ppvObj == NULL )
sizeof( *lpDP ) );
if( !lpDP )
{ {
return DPERR_OUTOFMEMORY; return DPERR_OUTOFMEMORY;
} }
ICOM_VTBL(lpDP) = &directPlay2AVT; /* new scope for variable declaration */
{
ICOM_THIS(IDirectPlay2AImpl,*ppvObj);
InitializeCriticalSection( &lpDP->DP_lock ); ICOM_VTBL(This) = &directPlay2AVT;
IDirectPlayX_AddRef( (LPDIRECTPLAY2A)lpDP );
*ppvObj = lpDP; if ( DP_CreateIUnknown( (LPVOID)This ) &&
DP_CreateDirectPlay2( (LPVOID)This )
)
{
return S_OK;
}
return S_OK; }
goto error;
} }
else if( IsEqualGUID( &IID_IDirectPlay3, riid ) ) else if( IsEqualGUID( &IID_IDirectPlay3, riid ) )
{ {
IDirectPlay3Impl* lpDP; *ppvObj = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY,
sizeof( IDirectPlay3Impl ) );
lpDP = (IDirectPlay3Impl*)HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, if( *ppvObj == NULL )
sizeof( *lpDP ) );
if( !lpDP )
{ {
return DPERR_OUTOFMEMORY; return DPERR_OUTOFMEMORY;
} }
ICOM_VTBL(lpDP) = &directPlay3WVT; /* new scope for variable declaration */
{
ICOM_THIS(IDirectPlay3Impl,*ppvObj);
InitializeCriticalSection( &lpDP->DP_lock ); ICOM_VTBL(This) = &directPlay3WVT;
IDirectPlayX_AddRef( (LPDIRECTPLAY2A)lpDP );
*ppvObj = lpDP; if ( DP_CreateIUnknown( (LPVOID)This ) &&
DP_CreateDirectPlay2( (LPVOID)This ) &&
DP_CreateDirectPlay3( (LPVOID)This )
)
{
return S_OK;
}
return S_OK; }
goto error;
} }
else if( IsEqualGUID( &IID_IDirectPlay3A, riid ) ) else if( IsEqualGUID( &IID_IDirectPlay3A, riid ) )
{ {
IDirectPlay3AImpl* lpDP; *ppvObj = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY,
sizeof( IDirectPlay3AImpl ) );
lpDP = (IDirectPlay3AImpl*)HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, if( *ppvObj == NULL )
sizeof( *lpDP ) );
if( !lpDP )
{ {
return DPERR_OUTOFMEMORY; return DPERR_OUTOFMEMORY;
} }
ICOM_VTBL(lpDP) = &directPlay3AVT; /* new scope for variable declaration */
{
ICOM_THIS(IDirectPlay3AImpl,*ppvObj);
InitializeCriticalSection( &lpDP->DP_lock ); ICOM_VTBL(This) = &directPlay3AVT;
IDirectPlayX_AddRef( (LPDIRECTPLAY2A)lpDP );
*ppvObj = lpDP; if ( DP_CreateIUnknown( (LPVOID)This ) &&
DP_CreateDirectPlay2( (LPVOID)This ) &&
DP_CreateDirectPlay3( (LPVOID)This )
)
{
return S_OK;
}
return S_OK; }
goto error;
} }
else if( IsEqualGUID( &IID_IDirectPlay4, riid ) ) else if( IsEqualGUID( &IID_IDirectPlay4, riid ) )
{ {
IDirectPlay4Impl* lpDP; *ppvObj = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY,
sizeof( IDirectPlay4Impl ) );
lpDP = (IDirectPlay4Impl*)HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, if( *ppvObj == NULL )
sizeof( *lpDP ) );
if( !lpDP )
{ {
return DPERR_OUTOFMEMORY; return DPERR_OUTOFMEMORY;
} }
ICOM_VTBL(lpDP) = &directPlay4WVT; /* new scope for variable declaration */
{
ICOM_THIS(IDirectPlay4Impl,*ppvObj);
InitializeCriticalSection( &lpDP->DP_lock ); ICOM_VTBL(This) = &directPlay4WVT;
IDirectPlayX_AddRef( (LPDIRECTPLAY2A)lpDP );
*ppvObj = lpDP; if ( DP_CreateIUnknown( (LPVOID)This ) &&
DP_CreateDirectPlay2( (LPVOID)This ) &&
DP_CreateDirectPlay3( (LPVOID)This ) &&
DP_CreateDirectPlay4( (LPVOID)This )
)
{
return S_OK;
}
return S_OK; }
goto error;
} }
else if( IsEqualGUID( &IID_IDirectPlay4A, riid ) ) else if( IsEqualGUID( &IID_IDirectPlay4A, riid ) )
{ {
IDirectPlay4AImpl* lpDP; *ppvObj = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY,
sizeof( IDirectPlay4AImpl ) );
lpDP = (IDirectPlay4AImpl*)HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, if( *ppvObj == NULL )
sizeof( *lpDP ) );
if( !lpDP )
{ {
return DPERR_OUTOFMEMORY; return DPERR_OUTOFMEMORY;
} }
ICOM_VTBL(lpDP) = &directPlay4AVT; /* new scope for variable declaration */
{
ICOM_THIS(IDirectPlay4AImpl,*ppvObj);
InitializeCriticalSection( &lpDP->DP_lock ); ICOM_VTBL(This) = &directPlay4AVT;
IDirectPlayX_AddRef( (LPDIRECTPLAY2A)lpDP );
*ppvObj = lpDP; if ( DP_CreateIUnknown( (LPVOID)This ) &&
DP_CreateDirectPlay2( (LPVOID)This ) &&
DP_CreateDirectPlay3( (LPVOID)This ) &&
DP_CreateDirectPlay4( (LPVOID)This )
)
{
return S_OK;
}
return S_OK; }
goto error;
} }
/* Unsupported interface */ /* Unsupported interface */
*ppvObj = NULL; *ppvObj = NULL;
return E_NOINTERFACE; return E_NOINTERFACE;
error:
DP_DestroyDirectPlay4( *ppvObj );
DP_DestroyDirectPlay3( *ppvObj );
DP_DestroyDirectPlay2( *ppvObj );
DP_DestroyIUnknown( *ppvObj );
HeapFree( GetProcessHeap(), 0, *ppvObj );
*ppvObj = NULL;
return DPERR_NOMEMORY;
} }
@ -311,15 +516,15 @@ static ULONG WINAPI DirectPlay2AImpl_AddRef
ULONG refCount; ULONG refCount;
ICOM_THIS(IDirectPlay3Impl,iface); ICOM_THIS(IDirectPlay3Impl,iface);
EnterCriticalSection( &This->DP_lock ); EnterCriticalSection( &This->unk->DP_lock );
{ {
refCount = ++(This->ref); refCount = ++(This->unk->ref);
} }
LeaveCriticalSection( &This->DP_lock ); LeaveCriticalSection( &This->unk->DP_lock );
TRACE("ref count incremented to %lu for %p\n", refCount, This ); TRACE("ref count incremented to %lu for %p\n", refCount, This );
return (This->ref); return refCount;
} }
static ULONG WINAPI DirectPlay2AImpl_Release static ULONG WINAPI DirectPlay2AImpl_Release
@ -329,21 +534,22 @@ static ULONG WINAPI DirectPlay2AImpl_Release
ICOM_THIS(IDirectPlay3Impl,iface); ICOM_THIS(IDirectPlay3Impl,iface);
EnterCriticalSection( &This->DP_lock ); EnterCriticalSection( &This->unk->DP_lock );
{ {
refCount = --(This->ref); refCount = --(This->unk->ref);
} }
LeaveCriticalSection( &This->DP_lock ); LeaveCriticalSection( &This->unk->DP_lock );
TRACE("ref count decremeneted to %lu for %p\n", refCount, This ); TRACE("ref count decremeneted to %lu for %p\n", refCount, This );
/* Deallocate if this is the last reference to the object */ /* Deallocate if this is the last reference to the object */
if( refCount == 0 ) if( refCount == 0 )
{ {
FIXME("memory leak\n" ); DP_DestroyDirectPlay4( This );
/* Implement memory deallocation */ DP_DestroyDirectPlay3( This );
DP_DestroyDirectPlay2( This );
HeapFree( GetProcessHeap(), 0, This ); DP_DestroyIUnknown( This );
HeapFree( GetProcessHeap(), 0, This );
} }
return refCount; return refCount;
@ -924,6 +1130,11 @@ static HRESULT WINAPI DirectPlay3AImpl_EnumConnections
char returnBuffer[51]; char returnBuffer[51];
LPWSTR lpWGUIDString; LPWSTR lpWGUIDString;
DPNAME dpName; DPNAME dpName;
HRESULT hr;
DPCOMPOUNDADDRESSELEMENT dpCompoundAddress;
LPVOID lpAddressBuffer = NULL;
DWORD dwAddressBufferSize = 0;
TRACE(" this time through: %s\n", subKeyName ); TRACE(" this time through: %s\n", subKeyName );
@ -949,16 +1160,46 @@ static HRESULT WINAPI DirectPlay3AImpl_EnumConnections
HeapFree( GetProcessHeap(), 0, lpWGUIDString ); HeapFree( GetProcessHeap(), 0, lpWGUIDString );
/* FIXME: Have I got a memory leak on the serviceProviderGUID? */ /* FIXME: Have I got a memory leak on the serviceProviderGUID? */
/* Fill in the DPNAME struct for the service provider */
dpName.dwSize = sizeof( dpName ); dpName.dwSize = sizeof( dpName );
dpName.dwFlags = 0; dpName.dwFlags = 0;
dpName.psn.lpszShortNameA = subKeyName; /* FIXME: Is this right? */ dpName.psn.lpszShortNameA = subKeyName;
dpName.pln.lpszLongNameA = subKeyName; /* FIXME: Is this right? */ dpName.pln.lpszLongNameA = NULL;
/* Create the compound address for the service provider.
NOTE: This is a gruesome architectural scar right now. DP uses DPL and DPL uses DP
nast stuff. This may be why the native dll just gets around this little bit by
allocating an 80 byte buffer which isn't even a filled with a valid compound
address. Oh well. Creating a proper compound address is the way to go anyways
despite this method taking slightly more heap space and realtime :) */
dpCompoundAddress.guidDataType = DPAID_ServiceProvider;
dpCompoundAddress.dwDataSize = sizeof( GUID );
dpCompoundAddress.lpData = &serviceProviderGUID;
if( ( hr = DPL_CreateCompoundAddress( &dpCompoundAddress, 1, lpAddressBuffer,
&dwAddressBufferSize, TRUE ) ) != DPERR_BUFFERTOOSMALL )
{
ERR( "can't get buffer size: %s\n", DPLAYX_HresultToString( hr ) );
return hr;
}
/* Now allocate the buffer */
lpAddressBuffer = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, dwAddressBufferSize );
if( ( hr = DPL_CreateCompoundAddress( &dpCompoundAddress, 1, lpAddressBuffer,
&dwAddressBufferSize, TRUE ) ) != DP_OK )
{
ERR( "can't create address: %s\n", DPLAYX_HresultToString( hr ) );
return hr;
}
/* The enumeration will return FALSE if we are not to continue */ /* The enumeration will return FALSE if we are not to continue */
if( !lpEnumCallback( &serviceProviderGUID, NULL,0, &dpName, 0, lpContext ) ) if( !lpEnumCallback( &serviceProviderGUID, lpAddressBuffer, dwAddressBufferSize,
&dpName, DPCONNECTION_DIRECTPLAY, lpContext ) )
{ {
WARN("lpEnumCallback returning FALSE\n" ); WARN("lpEnumCallback returning FALSE\n" );
break;
return DP_OK;
} }
} }
} }
@ -1015,7 +1256,13 @@ static HRESULT WINAPI DirectPlay3WImpl_GetGroupConnectionSettings
static HRESULT WINAPI DirectPlay3AImpl_InitializeConnection static HRESULT WINAPI DirectPlay3AImpl_InitializeConnection
( LPDIRECTPLAY3A iface, LPVOID lpConnection, DWORD dwFlags ) ( LPDIRECTPLAY3A iface, LPVOID lpConnection, DWORD dwFlags )
{ {
HMODULE hServiceProvider;
typedef DWORD (WINAPI *SP_SPInit)(LPVOID lpCompoundAddress, ...); /* FIXME: How many arguments? */
SP_SPInit SPInit;
DWORD dwReturnValue = 0;
ICOM_THIS(IDirectPlay3Impl,iface); ICOM_THIS(IDirectPlay3Impl,iface);
FIXME("(%p)->(%p,0x%08lx): stub\n", This, lpConnection, dwFlags ); FIXME("(%p)->(%p,0x%08lx): stub\n", This, lpConnection, dwFlags );
if( dwFlags != 0 ) if( dwFlags != 0 )
@ -1023,6 +1270,42 @@ static HRESULT WINAPI DirectPlay3AImpl_InitializeConnection
return DPERR_INVALIDFLAGS; return DPERR_INVALIDFLAGS;
} }
if( This->dp3->connectionInitialized == TRUE )
{
return DPERR_ALREADYINITIALIZED;
}
/* Parse lpConnection as a compound address for the service provider */
/* Take service provider GUID and find the path to it */
/* FIXME: Hard coded to only load the tcp/ip service provider for now... */
hServiceProvider = LoadLibraryA( "dpwsockx.dll" );
if( hServiceProvider == 0 )
{
ERR( "Unable to load service provider\n" );
return DPERR_UNAVAILABLE;
}
/* Initialize the service provider by calling SPInit */
SPInit = (SP_SPInit)GetProcAddress( hServiceProvider, "SPInit" );
if( SPInit == NULL )
{
ERR( "Service provider doesn't provide SPInit interface?\n" );
}
#if 0
/* NOTE: This will crash until I know what parameters/interface this has */
/* FIXME: Take a guess that we just pass the compound address to the SP */
/* Hmmm...how to say which parameters need to be gotten from the SP. They must
come from the compound address, but how do we communicate what's required? */
dwReturnValue = (*SPInit)( lpConnection );
#endif
/* This interface is now initialized */
This->dp3->connectionInitialized = TRUE;
return DP_OK; return DP_OK;
} }

View File

@ -20,16 +20,18 @@
DEFAULT_DEBUG_CHANNEL(dplay) DEFAULT_DEBUG_CHANNEL(dplay)
/* FIXME: Need to do all that fun other dll referencing type of stuff */ /* FIXME: Need to do all that fun other dll referencing type of stuff */
/* FIXME: Need to allocate a giant static area - or a global data type thing for Get/Set */ /* FIXME: Need to allocate a giant static area */
/* Static data for all processes */ /* Static data for all processes */
static LPSTR lpszDplayxSemaName = "DPLAYX_SM"; static LPSTR lpszDplayxSemaName = "WINE_DPLAYX_SM";
static HANDLE hDplayxSema; static HANDLE hDplayxSema;
#define DPLAYX_AquireSemaphore() 0L /* WaitForSingleObject( hDplayxSema, INFINITE? ) */ #define DPLAYX_AquireSemaphore() TRACE( "Waiting for DPLAYX sema\n" ); \
#define DPLAYX_ReleaseSemaphore() 0L /* ReleaseSemaphore( hDplayxSema, ..., ... ) */ WaitForSingleObject( hDplayxSema, INFINITE ); TRACE( "Through wait\n" )
#define DPLAYX_ReleaseSemaphore() ReleaseSemaphore( hDplayxSema, 1, NULL ); \
TRACE( "DPLAYX Sema released\n" ); /* FIXME: Is this correct? */
/* HACK for simple global data right now */ /* HACK for simple global data right now */
@ -44,12 +46,15 @@ typedef struct tagDirectPlayLobbyData
DWORD dwAddressSize; DWORD dwAddressSize;
DWORD dwAppID; DWORD dwAppID;
HANDLE hReceiveEvent; HANDLE hReceiveEvent;
DWORD dwAppLaunchedFromID;
} DirectPlayLobbyData, *lpDirectPlayLobbyData; } DirectPlayLobbyData, *lpDirectPlayLobbyData;
static DirectPlayLobbyData lobbyData[ numSupportedLobbies ]; static DirectPlayLobbyData lobbyData[ numSupportedLobbies ];
/* Function prototypes */ /* Function prototypes */
BOOL DPLAYX_AppIdLobbied( DWORD dwAppId, lpDirectPlayLobbyData* dplData ); BOOL DPLAYX_IsAppIdLobbied( DWORD dwAppId, lpDirectPlayLobbyData* dplData );
void DPLAYX_InitializeLobbyDataEntry( lpDirectPlayLobbyData lpData );
@ -64,8 +69,10 @@ void DPLAYX_ConstructData(void)
TRACE( "DPLAYX dll loaded - construct called\n" ); TRACE( "DPLAYX dll loaded - construct called\n" );
/* FIXME: This needs to be allocated shared */ /* Create a semahopre to block access to DPLAYX global data structs
hDplayxSema = CreateSemaphoreA( NULL, 0, 1, lpszDplayxSemaName ); It starts unblocked, and allows up to 65000 users blocked on it. Seems reasonable
for the time being */
hDplayxSema = CreateSemaphoreA( NULL, 1, 65000, lpszDplayxSemaName );
if( !hDplayxSema ) if( !hDplayxSema )
{ {
@ -76,7 +83,7 @@ void DPLAYX_ConstructData(void)
/* Set all lobbies to be "empty" */ /* Set all lobbies to be "empty" */
for( i=0; i < numSupportedLobbies; i++ ) for( i=0; i < numSupportedLobbies; i++ )
{ {
lobbyData[ i ].dwAppID = 0; DPLAYX_InitializeLobbyDataEntry( &lobbyData[ i ] );
} }
} }
@ -87,21 +94,35 @@ void DPLAYX_ConstructData(void)
***************************************************************************/ ***************************************************************************/
void DPLAYX_DestructData(void) void DPLAYX_DestructData(void)
{ {
/* Hmmm...what to call to delete the semaphore? Auto delete? */
TRACE( "DPLAYX dll unloaded - destruct called\n" ); TRACE( "DPLAYX dll unloaded - destruct called\n" );
/* delete the semaphore */
CloseHandle( hDplayxSema );
} }
void DPLAYX_InitializeLobbyDataEntry( lpDirectPlayLobbyData lpData )
{
ZeroMemory( lpData, sizeof( *lpData ) );
/* Set the handle to a better invalid value */
lpData->hReceiveEvent = INVALID_HANDLE_VALUE;
}
/* NOTE: This must be called with the semaphore aquired. /* NOTE: This must be called with the semaphore aquired.
* TRUE/FALSE with a pointer to it's data returned * TRUE/FALSE with a pointer to it's data returned. Pointer data is
* is only valid if TRUE is returned.
*/ */
BOOL DPLAYX_AppIdLobbied( DWORD dwAppID, lpDirectPlayLobbyData* lplpDplData ) BOOL DPLAYX_IsAppIdLobbied( DWORD dwAppID, lpDirectPlayLobbyData* lplpDplData )
{ {
INT i; INT i;
*lplpDplData = NULL;
if( dwAppID == 0 ) if( dwAppID == 0 )
{ {
dwAppID = GetCurrentProcessId(); dwAppID = GetCurrentProcessId();
TRACE( "Translated dwAppID == 0 into 0x%08lx\n", dwAppID );
} }
for( i=0; i < numSupportedLobbies; i++ ) for( i=0; i < numSupportedLobbies; i++ )
@ -117,40 +138,28 @@ BOOL DPLAYX_AppIdLobbied( DWORD dwAppID, lpDirectPlayLobbyData* lplpDplData )
return FALSE; return FALSE;
} }
#if !defined( WORKING_PROCESS_SUSPEND )
/* These two functions should not exist. We would normally create a process initially
suspended when we RunApplication. This gives us time to actually store some data
before the new process might try to read it. However, process suspension doesn't
work yet and I'm too stupid to get it going. So, we'll just hack in fine fashion */
DWORD DPLAYX_AquireSemaphoreHack( void )
{
return DPLAYX_AquireSemaphore();
}
DWORD DPLAYX_ReleaseSemaphoreHack( void )
{
return DPLAYX_ReleaseSemaphore();
}
#endif
/* Reserve a spot for the new appliction. TRUE means success and FALSE failure. */ /* Reserve a spot for the new appliction. TRUE means success and FALSE failure. */
BOOL DPLAYX_CreateLobbyApplication( DWORD dwAppID, HANDLE hReceiveEvent ) BOOL DPLAYX_CreateLobbyApplication( DWORD dwAppID, HANDLE hReceiveEvent )
{ {
INT i; UINT i;
/* 0 is the marker for unused application data slots */
if( dwAppID == 0 )
{
return FALSE;
}
DPLAYX_AquireSemaphore(); DPLAYX_AquireSemaphore();
/* Find an empty space in the list and insert the data */
for( i=0; i < numSupportedLobbies; i++ ) for( i=0; i < numSupportedLobbies; i++ )
{ {
if( lobbyData[ i ].dwAppID == 0 ) if( lobbyData[ i ].dwAppID == 0 )
{ {
/* This process is now lobbied */ /* This process is now lobbied */
lobbyData[ i ].dwAppID = dwAppID; lobbyData[ i ].dwAppID = dwAppID;
lobbyData[ i ].hReceiveEvent = hReceiveEvent; lobbyData[ i ].hReceiveEvent = hReceiveEvent;
lobbyData[ i ].dwAppLaunchedFromID = GetCurrentProcessId();
DPLAYX_ReleaseSemaphore(); DPLAYX_ReleaseSemaphore();
return TRUE; return TRUE;
@ -161,6 +170,31 @@ BOOL DPLAYX_CreateLobbyApplication( DWORD dwAppID, HANDLE hReceiveEvent )
return FALSE; return FALSE;
} }
/* I'm not sure when I'm going to need this, but here it is */
BOOL DPLAYX_DestroyLobbyApplication( DWORD dwAppID )
{
UINT i;
DPLAYX_AquireSemaphore();
/* Find an empty space in the list and insert the data */
for( i=0; i < numSupportedLobbies; i++ )
{
if( lobbyData[ i ].dwAppID == dwAppID )
{
/* Mark this entry unused */
DPLAYX_InitializeLobbyDataEntry( &lobbyData[ i ] );
DPLAYX_ReleaseSemaphore();
return TRUE;
}
}
DPLAYX_ReleaseSemaphore();
ERR( "Unable to find global entry for application\n" );
return FALSE;
}
HRESULT DPLAYX_GetConnectionSettingsA HRESULT DPLAYX_GetConnectionSettingsA
( DWORD dwAppID, ( DWORD dwAppID,
LPVOID lpData, LPVOID lpData,
@ -169,26 +203,24 @@ HRESULT DPLAYX_GetConnectionSettingsA
lpDirectPlayLobbyData lpDplData; lpDirectPlayLobbyData lpDplData;
LPDPLCONNECTION lpDplConnection; LPDPLCONNECTION lpDplConnection;
DPLAYX_AquireSemaphore();
if ( !DPLAYX_AppIdLobbied( dwAppID, &lpDplData ) )
{
DPLAYX_ReleaseSemaphore();
return DPERR_NOTLOBBIED;
}
/* Verify buffer size */ /* Verify buffer size */
if ( ( lpData == NULL ) || if ( ( lpData == NULL ) ||
( *lpdwDataSize < sizeof( DPLCONNECTION ) ) ( *lpdwDataSize < sizeof( DPLCONNECTION ) )
) )
{ {
DPLAYX_ReleaseSemaphore();
*lpdwDataSize = sizeof( DPLCONNECTION ); *lpdwDataSize = sizeof( DPLCONNECTION );
return DPERR_BUFFERTOOSMALL; return DPERR_BUFFERTOOSMALL;
} }
DPLAYX_AquireSemaphore();
if ( ! DPLAYX_IsAppIdLobbied( dwAppID, &lpDplData ) )
{
DPLAYX_ReleaseSemaphore();
return DPERR_NOTLOBBIED;
}
/* Copy the information */ /* Copy the information */
lpDplConnection = (LPDPLCONNECTION)lpData; lpDplConnection = (LPDPLCONNECTION)lpData;
@ -239,7 +271,7 @@ HRESULT DPLAYX_GetConnectionSettingsA
lpDplConnection->dwAddressSize = lpDplData->dwAddressSize; lpDplConnection->dwAddressSize = lpDplData->dwAddressSize;
/* Send a message - or only the first time? */ /* FIXME: Send a message - or only the first time? */
DPLAYX_ReleaseSemaphore(); DPLAYX_ReleaseSemaphore();
@ -254,26 +286,24 @@ HRESULT DPLAYX_GetConnectionSettingsW
lpDirectPlayLobbyData lpDplData; lpDirectPlayLobbyData lpDplData;
LPDPLCONNECTION lpDplConnection; LPDPLCONNECTION lpDplConnection;
DPLAYX_AquireSemaphore();
if ( !DPLAYX_AppIdLobbied( dwAppID, &lpDplData ) )
{
DPLAYX_ReleaseSemaphore();
return DPERR_NOTLOBBIED;
}
/* Verify buffer size */ /* Verify buffer size */
if ( ( lpData == NULL ) || if ( ( lpData == NULL ) ||
( *lpdwDataSize < sizeof( DPLCONNECTION ) ) ( *lpdwDataSize < sizeof( DPLCONNECTION ) )
) )
{ {
DPLAYX_ReleaseSemaphore();
*lpdwDataSize = sizeof( DPLCONNECTION ); *lpdwDataSize = sizeof( DPLCONNECTION );
return DPERR_BUFFERTOOSMALL; return DPERR_BUFFERTOOSMALL;
} }
DPLAYX_AquireSemaphore();
if ( ! DPLAYX_IsAppIdLobbied( dwAppID, &lpDplData ) )
{
DPLAYX_ReleaseSemaphore();
return DPERR_NOTLOBBIED;
}
/* Copy the information */ /* Copy the information */
lpDplConnection = (LPDPLCONNECTION)lpData; lpDplConnection = (LPDPLCONNECTION)lpData;
@ -324,7 +354,7 @@ HRESULT DPLAYX_GetConnectionSettingsW
lpDplConnection->dwAddressSize = lpDplData->dwAddressSize; lpDplConnection->dwAddressSize = lpDplData->dwAddressSize;
/* Send a message - or only the first time? */ /* FIXME: Send a message - or only the first time? */
DPLAYX_ReleaseSemaphore(); DPLAYX_ReleaseSemaphore();
@ -339,21 +369,9 @@ HRESULT DPLAYX_SetConnectionSettingsA
{ {
lpDirectPlayLobbyData lpDplData; lpDirectPlayLobbyData lpDplData;
DPLAYX_AquireSemaphore();
if ( !DPLAYX_AppIdLobbied( dwAppID, &lpDplData ) )
{
/* FIXME: Create a new entry for this dwAppID? */
DPLAYX_ReleaseSemaphore();
return DPERR_GENERIC;
}
/* Paramater check */ /* Paramater check */
if( dwFlags || !lpConn ) if( dwFlags || !lpConn )
{ {
DPLAYX_ReleaseSemaphore();
ERR("invalid parameters.\n"); ERR("invalid parameters.\n");
return DPERR_INVALIDPARAMS; return DPERR_INVALIDPARAMS;
} }
@ -361,9 +379,33 @@ HRESULT DPLAYX_SetConnectionSettingsA
/* Store information */ /* Store information */
if( lpConn->dwSize != sizeof(DPLCONNECTION) ) if( lpConn->dwSize != sizeof(DPLCONNECTION) )
{ {
ERR(": old/new DPLCONNECTION type? Size=%08lx vs. expected=%ul bytes\n",
lpConn->dwSize, sizeof( DPLCONNECTION ) );
return DPERR_INVALIDPARAMS;
}
if( dwAppID == 0 )
{
dwAppID = GetCurrentProcessId();
}
DPLAYX_AquireSemaphore();
if ( ! DPLAYX_IsAppIdLobbied( dwAppID, &lpDplData ) )
{
/* FIXME: Create a new entry for this dwAppID? */
DPLAYX_ReleaseSemaphore(); DPLAYX_ReleaseSemaphore();
ERR(": old/new DPLCONNECTION type? Size=%08lx vs. expected=%ul bytes\n", return DPERR_GENERIC;
}
/* Store information */
if( lpConn->dwSize != sizeof(DPLCONNECTION) )
{
DPLAYX_ReleaseSemaphore();
ERR(": old/new DPLCONNECTION type? Size=%lu vs. expected=%u bytes\n",
lpConn->dwSize, sizeof( DPLCONNECTION ) ); lpConn->dwSize, sizeof( DPLCONNECTION ) );
return DPERR_INVALIDPARAMS; return DPERR_INVALIDPARAMS;
@ -378,7 +420,7 @@ HRESULT DPLAYX_SetConnectionSettingsA
{ {
DPLAYX_ReleaseSemaphore(); DPLAYX_ReleaseSemaphore();
ERR("DPSESSIONDESC passed in? Size=%08lx vs. expected=%ul bytes\n", ERR("DPSESSIONDESC passed in? Size=%lu vs. expected=%u bytes\n",
lpConn->lpSessionDesc->dwSize, sizeof( DPSESSIONDESC2 ) ); lpConn->lpSessionDesc->dwSize, sizeof( DPSESSIONDESC2 ) );
return DPERR_INVALIDPARAMS; return DPERR_INVALIDPARAMS;
@ -426,7 +468,7 @@ HRESULT DPLAYX_SetConnectionSettingsA
lpDplData->dwAddressSize = lpConn->dwAddressSize; lpDplData->dwAddressSize = lpConn->dwAddressSize;
/* Send a message - I think */ /* FIXME: Send a message - I think */
DPLAYX_ReleaseSemaphore(); DPLAYX_ReleaseSemaphore();
@ -440,18 +482,9 @@ HRESULT DPLAYX_SetConnectionSettingsW
{ {
lpDirectPlayLobbyData lpDplData; lpDirectPlayLobbyData lpDplData;
DPLAYX_AquireSemaphore();
if ( !DPLAYX_AppIdLobbied( dwAppID, &lpDplData ) )
{
DPLAYX_ReleaseSemaphore();
return DPERR_NOTLOBBIED;
}
/* Paramater check */ /* Paramater check */
if( dwFlags || !lpConn ) if( dwFlags || !lpConn )
{ {
DPLAYX_ReleaseSemaphore();
ERR("invalid parameters.\n"); ERR("invalid parameters.\n");
return DPERR_INVALIDPARAMS; return DPERR_INVALIDPARAMS;
} }
@ -459,14 +492,25 @@ HRESULT DPLAYX_SetConnectionSettingsW
/* Store information */ /* Store information */
if( lpConn->dwSize != sizeof(DPLCONNECTION) ) if( lpConn->dwSize != sizeof(DPLCONNECTION) )
{ {
DPLAYX_ReleaseSemaphore(); ERR(": old/new DPLCONNECTION type? Size=%lu vs. expected=%u bytes\n",
ERR(": old/new DPLCONNECTION type? Size=%08lx vs. expected=%ul bytes\n",
lpConn->dwSize, sizeof( DPLCONNECTION ) ); lpConn->dwSize, sizeof( DPLCONNECTION ) );
return DPERR_INVALIDPARAMS; return DPERR_INVALIDPARAMS;
} }
if( dwAppID == 0 )
{
dwAppID = GetCurrentProcessId();
}
DPLAYX_AquireSemaphore();
if ( ! DPLAYX_IsAppIdLobbied( dwAppID, &lpDplData ) )
{
DPLAYX_ReleaseSemaphore();
return DPERR_NOTLOBBIED;
}
/* Need to investigate the lpConn->lpSessionDesc to figure out /* Need to investigate the lpConn->lpSessionDesc to figure out
* what type of session we need to join/create. * what type of session we need to join/create.
*/ */
@ -476,7 +520,7 @@ HRESULT DPLAYX_SetConnectionSettingsW
{ {
DPLAYX_ReleaseSemaphore(); DPLAYX_ReleaseSemaphore();
ERR("DPSESSIONDESC passed in? Size=%08lx vs. expected=%ul bytes\n", ERR("DPSESSIONDESC passed in? Size=%lu vs. expected=%u bytes\n",
lpConn->lpSessionDesc->dwSize, sizeof( DPSESSIONDESC2 ) ); lpConn->lpSessionDesc->dwSize, sizeof( DPSESSIONDESC2 ) );
return DPERR_INVALIDPARAMS; return DPERR_INVALIDPARAMS;
@ -524,9 +568,161 @@ HRESULT DPLAYX_SetConnectionSettingsW
lpDplData->dwAddressSize = lpConn->dwAddressSize; lpDplData->dwAddressSize = lpConn->dwAddressSize;
/* Send a message - I think */ /* FIXME: Send a message - I think */
DPLAYX_ReleaseSemaphore(); DPLAYX_ReleaseSemaphore();
return DP_OK; return DP_OK;
} }
/* NOTE: This is potentially not thread safe. You are not guaranteed to end up
with the correct string printed in the case where the HRESULT is not
known. You'll just get the last hr passed in printed. This can change
over time if this method is used alot :) */
LPCSTR DPLAYX_HresultToString(HRESULT hr)
{
static char szTempStr[12];
switch (hr)
{
case DP_OK:
return "DP_OK";
case DPERR_ALREADYINITIALIZED:
return "DPERR_ALREADYINITIALIZED";
case DPERR_ACCESSDENIED:
return "DPERR_ACCESSDENIED";
case DPERR_ACTIVEPLAYERS:
return "DPERR_ACTIVEPLAYERS";
case DPERR_BUFFERTOOSMALL:
return "DPERR_BUFFERTOOSMALL";
case DPERR_CANTADDPLAYER:
return "DPERR_CANTADDPLAYER";
case DPERR_CANTCREATEGROUP:
return "DPERR_CANTCREATEGROUP";
case DPERR_CANTCREATEPLAYER:
return "DPERR_CANTCREATEPLAYER";
case DPERR_CANTCREATESESSION:
return "DPERR_CANTCREATESESSION";
case DPERR_CAPSNOTAVAILABLEYET:
return "DPERR_CAPSNOTAVAILABLEYET";
case DPERR_EXCEPTION:
return "DPERR_EXCEPTION";
case DPERR_GENERIC:
return "DPERR_GENERIC";
case DPERR_INVALIDFLAGS:
return "DPERR_INVALIDFLAGS";
case DPERR_INVALIDOBJECT:
return "DPERR_INVALIDOBJECT";
case DPERR_INVALIDPARAMS:
return "DPERR_INVALIDPARAMS";
case DPERR_INVALIDPLAYER:
return "DPERR_INVALIDPLAYER";
case DPERR_INVALIDGROUP:
return "DPERR_INVALIDGROUP";
case DPERR_NOCAPS:
return "DPERR_NOCAPS";
case DPERR_NOCONNECTION:
return "DPERR_NOCONNECTION";
case DPERR_OUTOFMEMORY:
return "DPERR_OUTOFMEMORY";
case DPERR_NOMESSAGES:
return "DPERR_NOMESSAGES";
case DPERR_NONAMESERVERFOUND:
return "DPERR_NONAMESERVERFOUND";
case DPERR_NOPLAYERS:
return "DPERR_NOPLAYERS";
case DPERR_NOSESSIONS:
return "DPERR_NOSESSIONS";
/* This one isn't defined yet in WINE sources. I don't know the value
case DPERR_PENDING:
return "DPERR_PENDING";
*/
case DPERR_SENDTOOBIG:
return "DPERR_SENDTOOBIG";
case DPERR_TIMEOUT:
return "DPERR_TIMEOUT";
case DPERR_UNAVAILABLE:
return "DPERR_UNAVAILABLE";
case DPERR_UNSUPPORTED:
return "DPERR_UNSUPPORTED";
case DPERR_BUSY:
return "DPERR_BUSY";
case DPERR_USERCANCEL:
return "DPERR_USERCANCEL";
case DPERR_NOINTERFACE:
return "DPERR_NOINTERFACE";
case DPERR_CANNOTCREATESERVER:
return "DPERR_CANNOTCREATESERVER";
case DPERR_PLAYERLOST:
return "DPERR_PLAYERLOST";
case DPERR_SESSIONLOST:
return "DPERR_SESSIONLOST";
case DPERR_UNINITIALIZED:
return "DPERR_UNINITIALIZED";
case DPERR_NONEWPLAYERS:
return "DPERR_NONEWPLAYERS";
case DPERR_INVALIDPASSWORD:
return "DPERR_INVALIDPASSWORD";
case DPERR_CONNECTING:
return "DPERR_CONNECTING";
case DPERR_CONNECTIONLOST:
return "DPERR_CONNECTIONLOST";
case DPERR_UNKNOWNMESSAGE:
return "DPERR_UNKNOWNMESSAGE";
case DPERR_CANCELFAILED:
return "DPERR_CANCELFAILED";
case DPERR_INVALIDPRIORITY:
return "DPERR_INVALIDPRIORITY";
case DPERR_NOTHANDLED:
return "DPERR_NOTHANDLED";
case DPERR_CANCELLED:
return "DPERR_CANCELLED";
case DPERR_ABORTED:
return "DPERR_ABORTED";
case DPERR_BUFFERTOOLARGE:
return "DPERR_BUFFERTOOLARGE";
case DPERR_CANTCREATEPROCESS:
return "DPERR_CANTCREATEPROCESS";
case DPERR_APPNOTSTARTED:
return "DPERR_APPNOTSTARTED";
case DPERR_INVALIDINTERFACE:
return "DPERR_INVALIDINTERFACE";
case DPERR_NOSERVICEPROVIDER:
return "DPERR_NOSERVICEPROVIDER";
case DPERR_UNKNOWNAPPLICATION:
return "DPERR_UNKNOWNAPPLICATION";
case DPERR_NOTLOBBIED:
return "DPERR_NOTLOBBIED";
case DPERR_SERVICEPROVIDERLOADED:
return "DPERR_SERVICEPROVIDERLOADED";
case DPERR_ALREADYREGISTERED:
return "DPERR_ALREADYREGISTERED";
case DPERR_NOTREGISTERED:
return "DPERR_NOTREGISTERED";
case DPERR_AUTHENTICATIONFAILED:
return "DPERR_AUTHENTICATIONFAILED";
case DPERR_CANTLOADSSPI:
return "DPERR_CANTLOADSSPI";
case DPERR_ENCRYPTIONFAILED:
return "DPERR_ENCRYPTIONFAILED";
case DPERR_SIGNFAILED:
return "DPERR_SIGNFAILED";
case DPERR_CANTLOADSECURITYPACKAGE:
return "DPERR_CANTLOADSECURITYPACKAGE";
case DPERR_ENCRYPTIONNOTSUPPORTED:
return "DPERR_ENCRYPTIONNOTSUPPORTED";
case DPERR_CANTLOADCAPI:
return "DPERR_CANTLOADCAPI";
case DPERR_NOTLOGGEDIN:
return "DPERR_NOTLOGGEDIN";
case DPERR_LOGONDENIED:
return "DPERR_LOGONDENIED";
default:
/* For errors not in the list, return HRESULT as a string
This part is not thread safe */
WARN( "Unknown error 0x%08lx\n", hr );
sprintf( szTempStr, "0x%08lx", hr );
return szTempStr;
}
}

View File

@ -14,14 +14,9 @@ HRESULT DPLAYX_SetConnectionSettingsA ( DWORD dwFlags, DWORD dwAppID, LPDPLCONNE
HRESULT DPLAYX_SetConnectionSettingsW ( DWORD dwFlags, DWORD dwAppID, LPDPLCONNECTION lpConn ); HRESULT DPLAYX_SetConnectionSettingsW ( DWORD dwFlags, DWORD dwAppID, LPDPLCONNECTION lpConn );
BOOL DPLAYX_CreateLobbyApplication( DWORD dwAppID, HANDLE hReceiveEvent ); BOOL DPLAYX_CreateLobbyApplication( DWORD dwAppID, HANDLE hReceiveEvent );
BOOL DPLAYX_DestroyLobbyApplication( DWORD dwAppID );
/* This is a hack to ensure synchronization during application spawn */ /* Convert a DP or DPL HRESULT code into a string for human consumption */
#if !defined( WORKING_PROCESS_SUSPEND ) LPCSTR DPLAYX_HresultToString( HRESULT hr );
DWORD DPLAYX_AquireSemaphoreHack( void );
DWORD DPLAYX_ReleaseSemaphoreHack( void );
#endif
#endif /* __WINE_DPLAYX_GLOBAL */ #endif /* __WINE_DPLAYX_GLOBAL */

View File

@ -20,12 +20,14 @@
DEFAULT_DEBUG_CHANNEL(dplay) DEFAULT_DEBUG_CHANNEL(dplay)
/* FIXME: All the data structures are presently not needed except for unk */
/* FIXME: Move lock from unk to dpl */
/* Forward declarations for this module helper methods */ /* Forward declarations for this module helper methods */
static HRESULT DPL_CreateCompoundAddress ( LPCDPCOMPOUNDADDRESSELEMENT lpElements, DWORD dwElementCount, HRESULT DPL_CreateCompoundAddress ( LPCDPCOMPOUNDADDRESSELEMENT lpElements, DWORD dwElementCount,
LPVOID lpAddress, LPDWORD lpdwAddressSize, BOOL bAnsiInterface ); LPVOID lpAddress, LPDWORD lpdwAddressSize, BOOL bAnsiInterface );
HRESULT DPL_CreateAddress( REFGUID guidSP, REFGUID guidDataType, LPCVOID lpData, DWORD dwDataSize,
LPVOID lpAddress, LPDWORD lpdwAddressSize, BOOL bAnsiInterface );
static HRESULT DPL_EnumAddress( LPDPENUMADDRESSCALLBACK lpEnumAddressCallback, LPCVOID lpAddress, static HRESULT DPL_EnumAddress( LPDPENUMADDRESSCALLBACK lpEnumAddressCallback, LPCVOID lpAddress,
DWORD dwAddressSize, LPVOID lpContext ); DWORD dwAddressSize, LPVOID lpContext );
@ -76,27 +78,28 @@ typedef struct tagDirectPlayLobby3Data
BOOL dummy; BOOL dummy;
} DirectPlayLobby3Data; } DirectPlayLobby3Data;
#define LOBBY_IMPL_FIELDS DirectPlayLobbyIUnknownData* unk; \ #define DPL_IMPL_FIELDS \
DirectPlayLobbyData* dpl; \ DirectPlayLobbyIUnknownData* unk; \
DirectPlayLobby2Data* dpl2; \ DirectPlayLobbyData* dpl; \
DirectPlayLobby3Data* dpl3; DirectPlayLobby2Data* dpl2; \
DirectPlayLobby3Data* dpl3;
struct IDirectPlayLobbyImpl struct IDirectPlayLobbyImpl
{ {
ICOM_VFIELD(IDirectPlayLobby); ICOM_VFIELD(IDirectPlayLobby);
LOBBY_IMPL_FIELDS DPL_IMPL_FIELDS
}; };
struct IDirectPlayLobby2Impl struct IDirectPlayLobby2Impl
{ {
ICOM_VFIELD(IDirectPlayLobby2); ICOM_VFIELD(IDirectPlayLobby2);
LOBBY_IMPL_FIELDS DPL_IMPL_FIELDS
}; };
struct IDirectPlayLobby3Impl struct IDirectPlayLobby3Impl
{ {
ICOM_VFIELD(IDirectPlayLobby3); ICOM_VFIELD(IDirectPlayLobby3);
LOBBY_IMPL_FIELDS DPL_IMPL_FIELDS
}; };
@ -146,16 +149,16 @@ BOOL DPL_CreateIUnknown( LPVOID lpDPL )
This->unk = (DirectPlayLobbyIUnknownData*)HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, This->unk = (DirectPlayLobbyIUnknownData*)HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY,
sizeof( *(This->unk) ) ); sizeof( *(This->unk) ) );
if ( This->unk != NULL ) if ( This->unk == NULL )
{ {
InitializeCriticalSection( &This->unk->DPL_lock ); return FALSE;
IDirectPlayLobby_AddRef( (LPDIRECTPLAYLOBBYA)lpDPL );
return TRUE;
} }
return FALSE; InitializeCriticalSection( &This->unk->DPL_lock );
IDirectPlayLobby_AddRef( (LPDIRECTPLAYLOBBYA)lpDPL );
return TRUE;
} }
BOOL DPL_DestroyIUnknown( LPVOID lpDPL ) BOOL DPL_DestroyIUnknown( LPVOID lpDPL )
@ -710,27 +713,8 @@ static HRESULT WINAPI IDirectPlayLobbyAImpl_CreateAddress
LPVOID lpAddress, LPVOID lpAddress,
LPDWORD lpdwAddressSize ) LPDWORD lpdwAddressSize )
{ {
ICOM_THIS(IDirectPlayLobbyAImpl,iface); return DPL_CreateAddress( guidSP, guidDataType, lpData, dwDataSize,
lpAddress, lpdwAddressSize, TRUE );
const DWORD dwNumAddElements = 2; /* Service Provide & address data type */
DPCOMPOUNDADDRESSELEMENT addressElements[ dwNumAddElements ];
TRACE( "(%p)->(%p,%p,%p,0x%08lx,%p,%p)\n", This, guidSP, guidDataType, lpData,
dwDataSize, lpAddress, lpdwAddressSize );
addressElements[ 0 ].guidDataType = DPAID_ServiceProvider;
addressElements[ 0 ].dwDataSize = sizeof( GUID );
addressElements[ 0 ].lpData = (LPVOID)guidSP;
addressElements[ 1 ].guidDataType = *guidDataType;
addressElements[ 1 ].dwDataSize = dwDataSize;
addressElements[ 1 ].lpData = (LPVOID)lpData;
/* Call CreateCompoundAddress to cut down on code.
NOTE: We can do this because we don't support DPL 1 interfaces! */
return IDirectPlayLobby_CreateCompoundAddress( (LPDIRECTPLAYLOBBY2A)iface,
addressElements, dwNumAddElements,
lpAddress, lpdwAddressSize );
} }
static HRESULT WINAPI IDirectPlayLobbyWImpl_CreateAddress static HRESULT WINAPI IDirectPlayLobbyWImpl_CreateAddress
@ -742,12 +726,24 @@ static HRESULT WINAPI IDirectPlayLobbyWImpl_CreateAddress
LPVOID lpAddress, LPVOID lpAddress,
LPDWORD lpdwAddressSize ) LPDWORD lpdwAddressSize )
{ {
ICOM_THIS(IDirectPlayLobbyWImpl,iface); return DPL_CreateAddress( guidSP, guidDataType, lpData, dwDataSize,
lpAddress, lpdwAddressSize, FALSE );
}
HRESULT DPL_CreateAddress(
REFGUID guidSP,
REFGUID guidDataType,
LPCVOID lpData,
DWORD dwDataSize,
LPVOID lpAddress,
LPDWORD lpdwAddressSize,
BOOL bAnsiInterface )
{
const DWORD dwNumAddElements = 2; /* Service Provide & address data type */ const DWORD dwNumAddElements = 2; /* Service Provide & address data type */
DPCOMPOUNDADDRESSELEMENT addressElements[ dwNumAddElements ]; DPCOMPOUNDADDRESSELEMENT addressElements[ dwNumAddElements ];
TRACE( "(%p)->(%p,%p,%p,0x%08lx,%p,%p)\n", This, guidSP, guidDataType, lpData, TRACE( "(%p)->(%p,%p,0x%08lx,%p,%p,%d)\n", guidSP, guidDataType, lpData, dwDataSize,
dwDataSize, lpAddress, lpdwAddressSize ); lpAddress, lpdwAddressSize, bAnsiInterface );
addressElements[ 0 ].guidDataType = DPAID_ServiceProvider; addressElements[ 0 ].guidDataType = DPAID_ServiceProvider;
addressElements[ 0 ].dwDataSize = sizeof( GUID ); addressElements[ 0 ].dwDataSize = sizeof( GUID );
@ -759,12 +755,12 @@ static HRESULT WINAPI IDirectPlayLobbyWImpl_CreateAddress
/* Call CreateCompoundAddress to cut down on code. /* Call CreateCompoundAddress to cut down on code.
NOTE: We can do this because we don't support DPL 1 interfaces! */ NOTE: We can do this because we don't support DPL 1 interfaces! */
return IDirectPlayLobby_CreateCompoundAddress( (LPDIRECTPLAYLOBBY2)iface, return DPL_CreateCompoundAddress( addressElements, dwNumAddElements,
addressElements, dwNumAddElements, lpAddress, lpdwAddressSize, bAnsiInterface );
lpAddress, lpdwAddressSize );
} }
/******************************************************************** /********************************************************************
* *
* Parses out chunks from the DirectPlay Address buffer by calling the * Parses out chunks from the DirectPlay Address buffer by calling the
@ -1274,7 +1270,7 @@ static HRESULT WINAPI IDirectPlayLobbyAImpl_RunApplication
PROCESS_INFORMATION newProcessInfo; PROCESS_INFORMATION newProcessInfo;
LPSTR appName; LPSTR appName;
FIXME( "(%p)->(0x%08lx,%p,%p,%p):semi stub\n", This, dwFlags, lpdwAppID, lpConn, hReceiveEvent ); TRACE( "(%p)->(0x%08lx,%p,%p,%p)\n", This, dwFlags, lpdwAppID, lpConn, hReceiveEvent );
if( dwFlags != 0 ) if( dwFlags != 0 )
{ {
@ -1312,10 +1308,6 @@ static HRESULT WINAPI IDirectPlayLobbyAImpl_RunApplication
ZeroMemory( &newProcessInfo, sizeof( newProcessInfo ) ); ZeroMemory( &newProcessInfo, sizeof( newProcessInfo ) );
#if !defined( WORKING_PROCESS_SUSPEND )
DPLAYX_AquireSemaphoreHack();
#endif
if( !CreateProcessA( appName, if( !CreateProcessA( appName,
enumData.lpszCommandLine, enumData.lpszCommandLine,
NULL, NULL,
@ -1330,6 +1322,12 @@ static HRESULT WINAPI IDirectPlayLobbyAImpl_RunApplication
) )
{ {
FIXME( "Failed to create process for app %s\n", appName ); FIXME( "Failed to create process for app %s\n", appName );
HeapFree( GetProcessHeap(), 0, appName );
HeapFree( GetProcessHeap(), 0, enumData.lpszCommandLine );
HeapFree( GetProcessHeap(), 0, enumData.lpszCurrentDirectory );
return DPERR_CANTCREATEPROCESS;
} }
HeapFree( GetProcessHeap(), 0, appName ); HeapFree( GetProcessHeap(), 0, appName );
@ -1346,19 +1344,15 @@ static HRESULT WINAPI IDirectPlayLobbyAImpl_RunApplication
if( hr != DP_OK ) if( hr != DP_OK )
{ {
FIXME( "SetConnectionSettings failure 0x%08lx\n", hr ); FIXME( "SetConnectionSettings failure %s\n", DPLAYX_HresultToString( hr ) );
return hr; return hr;
} }
/* Everything seems to have been set correctly, update the dwAppID */ /* Everything seems to have been set correctly, update the dwAppID */
*lpdwAppID = newProcessInfo.dwProcessId; *lpdwAppID = newProcessInfo.dwProcessId;
#if !defined( WORKING_PROCESS_SUSPEND ) /* Unsuspend the process */
FIXME( ": It would be at this point that we would allow the process to resume\n" );
DPLAYX_ReleaseSemaphoreHack();
#else
ResumeThread( newProcessInfo.dwThreadId ); ResumeThread( newProcessInfo.dwThreadId );
#endif
LeaveCriticalSection( &This->unk->DPL_lock ); LeaveCriticalSection( &This->unk->DPL_lock );
@ -1504,7 +1498,7 @@ static HRESULT WINAPI IDirectPlayLobby2AImpl_CreateCompoundAddress
return DPL_CreateCompoundAddress( lpElements, dwElementCount, lpAddress, lpdwAddressSize, TRUE ); return DPL_CreateCompoundAddress( lpElements, dwElementCount, lpAddress, lpdwAddressSize, TRUE );
} }
static HRESULT DPL_CreateCompoundAddress HRESULT DPL_CreateCompoundAddress
( LPCDPCOMPOUNDADDRESSELEMENT lpElements, ( LPCDPCOMPOUNDADDRESSELEMENT lpElements,
DWORD dwElementCount, DWORD dwElementCount,
LPVOID lpAddress, LPVOID lpAddress,