1553 lines
45 KiB
C++
Raw Normal View History

2001-01-01 00:00:00 +01:00
/*==========================================================================
*
* Copyright (C) 2001-2002 Microsoft Corporation. All Rights Reserved.
*
* File: mcast.cpp
* Content: DirectPlay8 Mcast interface routines
*@@BEGIN_MSINTERNAL
* History:
* Date By Reason
* ==== == ======
* 10/08/01 vanceo Created
*@@END_MSINTERNAL
*
***************************************************************************/
#include "dncorei.h"
#ifndef DPNBUILD_NOMULTICAST
//**********************************************************************
// Constant definitions
//**********************************************************************
//**********************************************************************
// Macro definitions
//**********************************************************************
//**********************************************************************
// Structure definitions
//**********************************************************************
//**********************************************************************
// Variable definitions
//**********************************************************************
typedef STDMETHODIMP McastQueryInterface(IDirectPlay8Multicast *pInterface, DP8REFIID riid, LPVOID *ppvObj);
typedef STDMETHODIMP_(ULONG) McastAddRef(IDirectPlay8Multicast *pInterface);
typedef STDMETHODIMP_(ULONG) McastRelease(IDirectPlay8Multicast *pInterface);
typedef STDMETHODIMP McastInitialize(IDirectPlay8Multicast *pInterface, PVOID const pvUserContext, const PFNDPNMESSAGEHANDLER pfn, const DWORD dwFlags);
typedef STDMETHODIMP McastJoin(IDirectPlay8Multicast *pInterface, IDirectPlay8Address *const pGroupAddr, IUnknown *const pDeviceInfo, const DPN_SECURITY_DESC *const pdnSecurity, const DPN_SECURITY_CREDENTIALS *const pdnCredentials, void *const pvAsyncContext, DPNHANDLE *const phAsyncHandle, const DWORD dwFlags);
typedef STDMETHODIMP McastClose(IDirectPlay8Multicast *pInterface, const DWORD dwFlags);
typedef STDMETHODIMP McastCreateSenderContext(IDirectPlay8Multicast *pInterface, IDirectPlay8Address *const pSenderAddress, void *const pvSenderContext, const DWORD dwFlags);
typedef STDMETHODIMP McastDestroySenderContext(IDirectPlay8Multicast *pInterface, IDirectPlay8Address *const pSenderAddress, const DWORD dwFlags);
typedef STDMETHODIMP McastSend(IDirectPlay8Multicast *pInterface, const DPN_BUFFER_DESC *const prgBufferDesc,const DWORD cBufferDesc,const DWORD dwTimeOut, void *const pvAsyncContext, DPNHANDLE *const phAsyncHandle, const DWORD dwFlags);
typedef STDMETHODIMP McastGetGroupAddress(IDirectPlay8Multicast *pInterface, IDirectPlay8Address **const ppAddress, const DWORD dwFlags);
typedef STDMETHODIMP McastGetSendQueueInfo( IDirectPlay8Multicast *pInterface, DWORD *const lpdwNumMsgs, DWORD *const lpdwNumBytes, const DWORD dwFlags);
typedef STDMETHODIMP McastCancelAsyncOperation( IDirectPlay8Multicast *pInterface, const DPNHANDLE hAsyncHandle, const DWORD dwFlags);
typedef STDMETHODIMP McastReturnBuffer( IDirectPlay8Multicast *pInterface, const DPNHANDLE hBufferHandle,const DWORD dwFlags);
typedef STDMETHODIMP McastEnumServiceProviders( IDirectPlay8Multicast *pInterface, const GUID *const pguidServiceProvider, const GUID *const pguidApplication, DPN_SERVICE_PROVIDER_INFO *const pSPInfoBuffer, DWORD *const pcbEnumData, DWORD *const pcReturned, const DWORD dwFlags);
typedef STDMETHODIMP McastEnumMulticastScopes( IDirectPlay8Multicast *pInterface, const GUID *const pguidServiceProvider, const GUID *const pguidDevice, const GUID *const pguidApplication, DPN_MULTICAST_SCOPE_INFO *const pScopeInfoBuffer, PDWORD const pcbEnumData, PDWORD const pcReturned, const DWORD dwFlags);
typedef STDMETHODIMP McastGetSPCaps(IDirectPlay8Multicast *pInterface, const GUID * const pguidSP, DPN_SP_CAPS *const pdpspCaps,const DWORD dwFlags);
typedef STDMETHODIMP McastSetSPCaps(IDirectPlay8Multicast *pInterface, const GUID * const pguidSP, const DPN_SP_CAPS *const pdpspCaps, const DWORD dwFlags);
IDirectPlay8MulticastVtbl DNMcast_Vtbl =
{
(McastQueryInterface*) DN_QueryInterface,
(McastAddRef*) DN_AddRef,
(McastRelease*) DN_Release,
(McastInitialize*) DN_Initialize,
(McastJoin*) DN_Join,
(McastClose*) DN_Close,
(McastCreateSenderContext*) DN_CreateSenderContext,
(McastDestroySenderContext*) DN_DestroySenderContext,
(McastSend*) DN_Send,
(McastGetGroupAddress*) DN_GetGroupAddress,
(McastGetSendQueueInfo*) DN_GetHostSendQueueInfo,
(McastCancelAsyncOperation*) DN_CancelAsyncOperation,
(McastReturnBuffer*) DN_ReturnBuffer,
(McastEnumServiceProviders*) DN_EnumServiceProviders,
(McastEnumMulticastScopes*) DN_EnumMulticastScopes,
(McastGetSPCaps*) DN_GetSPCaps,
(McastSetSPCaps*) DN_SetSPCaps
};
//**********************************************************************
// Function prototypes
//**********************************************************************
//**********************************************************************
// Function definitions
//**********************************************************************
//
// Completion for join connect parent
//
#undef DPF_MODNAME
#define DPF_MODNAME "DNCompleteJoinOperation"
void DNCompleteJoinOperation(DIRECTNETOBJECT *const pdnObject,
CAsyncOp *const pAsyncOp)
{
CServiceProvider *pSP;
IDirectPlay8Address *pIDevice;
pSP = NULL;
pIDevice = NULL;
//
// Save the result code on the parent (if it exists - it will be the CONNECT handle)
//
if (pAsyncOp->GetParent())
{
pAsyncOp->GetParent()->Lock();
pAsyncOp->GetParent()->SetResult( pAsyncOp->GetResult() );
pAsyncOp->GetParent()->Unlock();
if (SUCCEEDED(pdnObject->HandleTable.Destroy( pAsyncOp->GetParent()->GetHandle(), NULL )))
{
// Release the HandleTable reference
pAsyncOp->GetParent()->Release();
}
}
//
// Clear CONNECTING flag, and DISCONNECTING flag in case this was aborted.
// If the connect succeeded, set the CONNECTED flag.
//
DPFX(DPFPREP, 8,"Clearing CONNECTING and DISCONNECTING flags");
DNEnterCriticalSection(&pdnObject->csDirectNetObject);
pdnObject->dwFlags &= (~DN_OBJECT_FLAG_CONNECTING | DN_OBJECT_FLAG_DISCONNECTING);
if (pAsyncOp->GetResult() == DPN_OK)
{
pdnObject->dwFlags |= DN_OBJECT_FLAG_CONNECTED;
}
else
{
//
// Clean up DirectNet object
//
pSP = pdnObject->pConnectSP;
pdnObject->pConnectSP = NULL;
pIDevice = pdnObject->pIDP8ADevice;
pdnObject->pIDP8ADevice = NULL;
}
DNLeaveCriticalSection(&pdnObject->csDirectNetObject);
if (pSP)
{
pSP->Release();
pSP = NULL;
}
if (pIDevice)
{
IDirectPlay8Address_Release(pIDevice);
pIDevice = NULL;
}
DNASSERT( pSP == NULL );
DNASSERT( pIDevice == NULL );
}
//
// Completion for join handle parent
//
#undef DPF_MODNAME
#define DPF_MODNAME "DNCompleteUserJoin"
void DNCompleteUserJoin(DIRECTNETOBJECT *const pdnObject,
CAsyncOp *const pAsyncOp)
{
DNUserJoinComplete( pdnObject,
pAsyncOp->GetHandle(),
pAsyncOp->GetContext(),
pAsyncOp->GetResult());
}
#undef DPF_MODNAME
#define DPF_MODNAME "DN_Join"
STDMETHODIMP DN_Join( IDirectPlay8Multicast *pInterface,
IDirectPlay8Address *const pGroupAddr,
IUnknown *const pDeviceInfo,
const DPN_SECURITY_DESC *const pdnSecurity,
const DPN_SECURITY_CREDENTIALS *const pdnCredentials,
void *const pvAsyncContext,
DPNHANDLE *const phAsyncHandle,
const DWORD dwFlags)
{
HRESULT hResultCode;
HRESULT volatile hrOperation;
DIRECTNETOBJECT *pdnObject;
IDirectPlay8Address *pIGroup;
IDirectPlay8Address *pIDevice;
#ifndef DPNBUILD_ONLYONESP
GUID guidSP;
#endif // ! DPNBUILD_ONLYONESP
#ifndef DPNBUILD_ONLYONEADAPTER
GUID guidAdapter;
#endif // ! DPNBUILD_ONLYONEADAPTER
PVOID pvNewInterface;
DWORD dwListenFlags;
CAsyncOp *pListenParent;
CAsyncOp *pParent;
CAsyncOp *pAsyncOp;
CAsyncOp *pConnectParent;
CAsyncOp *pHandleParent;
CServiceProvider *pSP;
CSyncEvent *pSyncEvent;
HANDLE hEndpoint;
SPGETADDRESSINFODATA spInfoData;
// Variables used when device interface is a DIRECTNETOBJECT
DIRECTNETOBJECT *pdnDeviceObject;
CConnection *pExistingConnection;
CNameTableEntry *pNTEntry;
CServiceProvider *pShareSP;
IDP8ServiceProvider *pShareDP8ServiceProvider;
HANDLE hEndPt;
CCallbackThread CallbackThread;
CAsyncOp *pShareListenParent;
CBilink *pBilink;
CAsyncOp *pShareParent;
IDP8ServiceProvider *pLocalDP8ServiceProvider;
SPSHAREENDPOINTINFODATA spShareData;
DPFX(DPFPREP, 2,"Parameters: pInterface [0x%p], pGroupAddr [0x%p], pDeviceInfo [0x%p], pdnSecurity [0x%p], pdnCredentials [0x%p], pvAsyncContext [0x%p], phAsyncHandle [0x%p], dwFlags [0x%lx]",
pInterface,pGroupAddr,pDeviceInfo,pdnSecurity,pdnCredentials,pvAsyncContext,phAsyncHandle,dwFlags);
pdnObject = (DIRECTNETOBJECT*) GET_OBJECT_FROM_INTERFACE(pInterface);
DNASSERT(pdnObject != NULL);
#ifndef DPNBUILD_NOPARAMVAL
if( pdnObject->dwFlags & DN_OBJECT_FLAG_PARAMVALIDATION )
{
if( FAILED( hResultCode = DN_ValidateJoin( pInterface, pGroupAddr, pDeviceInfo,
pdnSecurity, pdnCredentials,
pvAsyncContext,phAsyncHandle,dwFlags ) ) )
{
DPFERR( "Error validating join params" );
DPF_RETURN( hResultCode );
}
}
#endif // !DPNBUILD_NOPARAMVAL
// Check to ensure message handler registered
if (!(pdnObject->dwFlags & DN_OBJECT_FLAG_INITIALIZED))
{
DPFERR( "Object is not initialized" );
DPF_RETURN(DPNERR_UNINITIALIZED);
}
// Check to ensure not already connected/connecting
if (pdnObject->dwFlags & DN_OBJECT_FLAG_CONNECTING)
{
DPFERR( "Object is already connecting" );
DPF_RETURN(DPNERR_CONNECTING);
}
if (pdnObject->dwFlags & DN_OBJECT_FLAG_CONNECTED)
{
DPFERR( "Object is already connected" );
DPF_RETURN(DPNERR_ALREADYCONNECTED);
}
if (pdnObject->dwFlags & (DN_OBJECT_FLAG_CLOSING | DN_OBJECT_FLAG_DISCONNECTING))
{
DPFERR( "Object is closing or disconnecting" );
DPF_RETURN(DPNERR_ALREADYCLOSING);
}
pIGroup = NULL;
pIDevice = NULL;
pvNewInterface = NULL;
pListenParent = NULL;
pParent = NULL;
pConnectParent = NULL;
pHandleParent = NULL;
pSP = NULL;
pSyncEvent = NULL;
pdnDeviceObject = NULL;
pExistingConnection = NULL;
pNTEntry = NULL;
pShareSP = NULL;
pShareDP8ServiceProvider = NULL;
CallbackThread.Deinitialize();
pShareListenParent = NULL;
pShareParent = NULL;
pLocalDP8ServiceProvider = NULL;
//
// Flag as CONNECTING to prevent other operations here
//
DNEnterCriticalSection(&pdnObject->csDirectNetObject);
if (pdnObject->dwFlags & (DN_OBJECT_FLAG_CONNECTING | DN_OBJECT_FLAG_CONNECTED))
{
DNLeaveCriticalSection(&pdnObject->csDirectNetObject);
hResultCode = DPNERR_ALREADYCONNECTED;
goto Failure;
}
pdnObject->dwFlags |= DN_OBJECT_FLAG_CONNECTING;
// Adding local host flag
//pdnObject->dwFlags |= DN_OBJECT_FLAG_LOCALHOST;
DNLeaveCriticalSection(&pdnObject->csDirectNetObject);
//
// Duplicate specified Group Address, or create a blank one if NULL
//
if (pGroupAddr != NULL)
{
if ((hResultCode = IDirectPlay8Address_Duplicate(pGroupAddr,&pIGroup)) != DPN_OK)
{
DPFERR("Could not duplicate group address");
DisplayDNError(0,hResultCode);
DNASSERT(FALSE);
goto Failure;
}
}
else
{
#ifdef DPNBUILD_LIBINTERFACE
hResultCode = DP8ACF_CreateInstance(IID_IDirectPlay8Address,
reinterpret_cast<void**>(&pIGroup));
#else // ! DPNBUILD_LIBINTERFACE
hResultCode = COM_CoCreateInstance(CLSID_DirectPlay8Address,
NULL,
CLSCTX_INPROC_SERVER,
IID_IDirectPlay8Address,
reinterpret_cast<void**>(&pIGroup),
FALSE);
#endif // ! DPNBUILD_LIBINTERFACE
if (hResultCode != S_OK)
{
DPFERR("Could not create Group Address");
DisplayDNError(0,hResultCode);
DNASSERT(FALSE);
goto Failure;
}
}
//
// Generate a Device Address from the DPlay interface, or duplicate
// the user's. Start by finding out which type of interface it is.
//
if (((IDirectPlay8Peer_QueryInterface((IDirectPlay8Peer*) pDeviceInfo, IID_IDirectPlay8Peer, &pvNewInterface)) == S_OK) ||
#ifndef DPNBUILD_NOSERVER
((IDirectPlay8Server_QueryInterface((IDirectPlay8Server*) pDeviceInfo, IID_IDirectPlay8Server, &pvNewInterface)) == S_OK) ||
#endif // ! DPNBUILD_NOSERVER
((IDirectPlay8Client_QueryInterface((IDirectPlay8Client*) pDeviceInfo, IID_IDirectPlay8Client, &pvNewInterface)) == S_OK) ||
((IDirectPlay8Multicast_QueryInterface((IDirectPlay8Multicast*) pDeviceInfo, IID_IDirectPlay8Multicast, &pvNewInterface)) == S_OK))
{
//
// It's a DIRECTNETOBJECT.
//
pdnDeviceObject = (DIRECTNETOBJECT*) GET_OBJECT_FROM_INTERFACE(pvNewInterface);
DNEnterCriticalSection(&pdnDeviceObject->csDirectNetObject);
if (! (pdnDeviceObject->dwFlags & DN_OBJECT_FLAG_CONNECTED))
{
DPFERR("DirectPlay device object is not connected or hosting" );
DNLeaveCriticalSection(&pdnDeviceObject->csDirectNetObject);
hResultCode = DPNERR_INVALIDDEVICEADDRESS;
goto Failure;
}
DNLeaveCriticalSection(&pdnDeviceObject->csDirectNetObject);
//
// Get the host player's connection (if not a multicast object).
//
if (! (pdnDeviceObject->dwFlags & DN_OBJECT_FLAG_MULTICAST))
{
if ((hResultCode = pdnDeviceObject->NameTable.GetHostPlayerRef(&pNTEntry)) != DPN_OK)
{
DPFERR("Could not find device object's Host player");
DisplayDNError(0,hResultCode);
hResultCode = DPNERR_INVALIDDEVICEADDRESS;
goto Failure;
}
//
// If we're the host, then the name table entry won't have a valid
// connection object.
//
if (! pNTEntry->IsLocal())
{
hResultCode = pNTEntry->GetConnectionRef(&pExistingConnection);
if (hResultCode != DPN_OK)
{
DPFX(DPFPREP, 0, "Could not find device object's Host player (err = 0x%lx)! Ignoring.",
pdnDeviceObject, pExistingConnection);
DNASSERT(pExistingConnection == NULL);
}
}
pNTEntry->Release();
pNTEntry = NULL;
}
//
// If we got a connection, simply extract its device address.
// If we couldn't get the non-local host player's connection, we
// have to select the device a different way.
//
if (pExistingConnection != NULL)
{
if ((hResultCode = pExistingConnection->GetEndPt(&hEndPt,&CallbackThread)) != DPN_OK)
{
DPFERR( "Couldn't retrieve host player's endpoint" );
DisplayDNError(0, hResultCode);
goto Failure;
}
spInfoData.Flags = SP_GET_ADDRESS_INFO_LOCAL_ADAPTER;
//
// Note that we are calling our Protocol object with the other
// interface's endpoint.
//
hResultCode = DNPCrackEndPointDescriptor(pdnObject->pdnProtocolData, hEndPt,&spInfoData);
if (hResultCode != DPN_OK)
{
DPFERR("Unknown error from DNPCrackEndPointDescriptor");
DisplayDNError(0,hResultCode);
DNASSERT(FALSE);
pExistingConnection->ReleaseEndPt(&CallbackThread);
goto Failure;
}
pIDevice = spInfoData.pAddress;
spInfoData.pAddress = NULL;
pExistingConnection->ReleaseEndPt(&CallbackThread);
pShareSP = pExistingConnection->GetSP();
if (pShareSP == NULL)
{
DPFERR("Could not get host player connection's SP!");
DNASSERT(FALSE);
hResultCode = DPNERR_INVALIDDEVICEADDRESS;
goto Failure;
}
pExistingConnection->Release();
pExistingConnection = NULL;
}
else
{
//
// Get the listen operation.
//
DNEnterCriticalSection(&pdnDeviceObject->csDirectNetObject);
if (pdnDeviceObject->pListenParent == NULL)
{
DNLeaveCriticalSection(&pdnObject->csDirectNetObject);
DPFERR("Could not find device object's listen parent operation!");
DNASSERT(FALSE);
hResultCode = DPNERR_INVALIDDEVICEADDRESS;
goto Failure;
}
pdnDeviceObject->pListenParent->AddRef();
pShareListenParent = pdnDeviceObject->pListenParent;
DNLeaveCriticalSection(&pdnDeviceObject->csDirectNetObject);
//
// Get the first SP listen.
//
pShareListenParent->Lock();
pBilink = pShareListenParent->m_bilinkParent.GetNext();
if (pBilink == &pShareListenParent->m_bilinkParent)
{
DPFERR("Could not find device object's first listen operation!");
DNASSERT(FALSE);
hResultCode = DPNERR_INVALIDDEVICEADDRESS;
goto Failure;
}
pShareParent = CONTAINING_OBJECT(pBilink,CAsyncOp,m_bilinkChildren);
pShareSP = pShareParent->GetSP();
if (pShareSP == NULL)
{
DPFERR("Could not get first listen operation's SP!");
DNASSERT(FALSE);
hResultCode = DPNERR_INVALIDDEVICEADDRESS;
goto Failure;
}
pShareSP->AddRef();
pShareListenParent->Unlock();
pShareListenParent->Release();
pShareListenParent = NULL;
#ifndef DPNBUILD_ONLYONESP
//
// Get the share SP's GUID.
//
pShareSP->GetGUID(&guidSP);
#endif // ! DPNBUILD_ONLYONESP
//
// Build a device address object.
//
#ifdef DPNBUILD_LIBINTERFACE
hResultCode = DP8ACF_CreateInstance(IID_IDirectPlay8Address,
reinterpret_cast<void**>(&pIDevice));
#else // ! DPNBUILD_LIBINTERFACE
hResultCode = COM_CoCreateInstance(CLSID_DirectPlay8Address,
NULL,
CLSCTX_INPROC_SERVER,
IID_IDirectPlay8Address,
reinterpret_cast<void**>(&pIDevice),
FALSE);
#endif // ! DPNBUILD_LIBINTERFACE
if (hResultCode != S_OK)
{
DPFERR("Could not create Device Address");
DisplayDNError(0,hResultCode);
DNASSERT(FALSE);
goto Failure;
}
#ifndef DPNBUILD_ONLYONESP
if ((hResultCode = IDirectPlay8Address_SetSP(pIDevice,&guidSP)) != DPN_OK)
{
DPFERR("Could not set SP on Device Address");
DisplayDNError(0,hResultCode);
goto Failure;
}
#endif // ! DPNBUILD_ONLYONESP
#ifndef DPNBUILD_ONLYONEADAPTER
//
// For multicast objects we can reuse the same device GUID, too,
// since we know there will be only one.
//
if (pdnDeviceObject->dwFlags & DN_OBJECT_FLAG_MULTICAST)
{
pShareParent->Lock();
DNASSERT(! pShareParent->IsCancelled());
DNASSERT(! pShareParent->m_bilinkParent.IsEmpty());
pAsyncOp = CONTAINING_OBJECT(pParent->m_bilinkParent.GetNext(),CAsyncOp,m_bilinkChildren);
pAsyncOp->Lock();
DNASSERT((! pAsyncOp->IsCancelled()) && (! pAsyncOp->IsComplete()));
DNASSERT(pAsyncOp->m_bilinkChildren.GetNext() == &pShareParent->m_bilinkParent);
hEndpoint = pAsyncOp->GetProtocolHandle();
pAsyncOp->Unlock();
pShareParent->Unlock();
spInfoData.hEndpoint = hEndpoint;
spInfoData.Flags = SP_GET_ADDRESS_INFO_LOCAL_ADAPTER;
//
// Note that we are calling our Protocol object with the other
// interface's endpoint.
//
if ((hResultCode = DNPGetListenAddressInfo(pdnObject->pdnProtocolData, spInfoData.hEndpoint,&spInfoData)) != DPN_OK)
{
DPFERR("Could not get LISTEN device address!");
DisplayDNError(0,hResultCode);
DNASSERT(FALSE);
goto Failure;
}
//
// Retrieve the device GUID from the listen address.
//
if ((hResultCode = IDirectPlay8Address_GetDevice(spInfoData.pAddress, &guidAdapter)) != DPN_OK)
{
DPFERR("Could not get adapter GUID!");
DisplayDNError(0,hResultCode);
DNASSERT(FALSE);
IDirectPlay8Address_Release(spInfoData.pAddress);
spInfoData.pAddress = NULL;
goto Failure;
}
IDirectPlay8Address_Release(spInfoData.pAddress);
spInfoData.pAddress = NULL;
//
// Store the device GUID on our device address.
//
if ((hResultCode = IDirectPlay8Address_SetDevice(pIDevice, &guidAdapter)) != DPN_OK)
{
DPFERR("Could not set adapter GUID!");
DisplayDNError(0,hResultCode);
DNASSERT(FALSE);
goto Failure;
}
}
#endif // ! DPNBUILD_ONLYONEADAPTER
}
//
// Get the share SP's COM interface.
//
if ((hResultCode = pShareSP->GetInterfaceRef(&pShareDP8ServiceProvider)) != DPN_OK)
{
DPFERR("Could not get shared SP interface");
DisplayDNError(0,hResultCode);
DNASSERT(FALSE);
goto Failure;
}
pShareSP->Release();
pShareSP = NULL;
//
// Load our own local instance of that SP.
//
hResultCode = DN_SPEnsureLoaded(pdnObject,
#ifndef DPNBUILD_ONLYONESP
&guidSP,
#endif // ! DPNBUILD_ONLYONESP
#ifndef DPNBUILD_LIBINTERFACE
NULL,
#endif // ! DPNBUILD_LIBINTERFACE
&pSP);
if (hResultCode != DPN_OK)
{
DPFERR("Could not find or load SP");
DisplayDNError(0,hResultCode);
goto Failure;
}
//
// Get the local SP's COM interface.
//
if ((hResultCode = pSP->GetInterfaceRef(&pLocalDP8ServiceProvider)) != DPN_OK)
{
DPFERR("Could not get local SP interface");
DisplayDNError(0,hResultCode);
goto Failure;
}
//
// Tell our SP interface to get its endpoint information from the
// other interface's SP.
//
spShareData.pDP8ServiceProvider = pShareDP8ServiceProvider;
spShareData.dwFlags = 0;
hResultCode = IDP8ServiceProvider_ShareEndpointInfo(pLocalDP8ServiceProvider,
&spShareData);
if (hResultCode != DPN_OK)
{
DPFERR("Could not have SPs share endpoint info");
DisplayDNError(0,hResultCode);
goto Failure;
}
IDP8ServiceProvider_Release(pLocalDP8ServiceProvider);
pLocalDP8ServiceProvider = NULL;
IDP8ServiceProvider_Release(pShareDP8ServiceProvider);
pShareDP8ServiceProvider = NULL;
//
// Release the ref we held on the device object.
//
IDirectPlay8Peer_Release((IDirectPlay8Peer*) pvNewInterface); // all core objects have Release in same location in Vtbl
pvNewInterface = NULL;
}
else if ((IDirectPlay8Address_QueryInterface((IDirectPlay8Address*) pDeviceInfo, IID_IDirectPlay8Address, &pvNewInterface)) == S_OK)
{
//
// It's an address.
//
IDirectPlay8Address_Release((IDirectPlay8Address*) pvNewInterface);
pvNewInterface = NULL;
if ((hResultCode = IDirectPlay8Address_Duplicate(reinterpret_cast<IDirectPlay8Address*>(pDeviceInfo),&pIDevice)) != DPN_OK)
{
DPFERR("Could not duplicate device info");
DisplayDNError(0,hResultCode);
DNASSERT(FALSE);
goto Failure;
}
}
else
{
DPFERR( "Invalid device address, it must be an IDirectPlay8Peer, IDirectPlay8Server, IDirectPlay8Client, IDirectPlay8Multicast, or IDirectPlay8Address object" );
return( DPNERR_INVALIDDEVICEADDRESS );
}
#ifndef DPNBUILD_ONLYONESP
//
// If there is no SP on the group address, then steal it from the device address
//
if ((hResultCode = IDirectPlay8Address_GetSP(pIGroup,&guidSP)) != DPN_OK)
{
if ((hResultCode = IDirectPlay8Address_GetSP(pIDevice,&guidSP)) != DPN_OK)
{
DPFERR("Could not retrieve SP from Device Address");
DisplayDNError(0,hResultCode);
goto Failure;
}
if ((hResultCode = IDirectPlay8Address_SetSP(pIGroup,&guidSP)) != DPN_OK)
{
DPFERR("Could not set SP on Group Address");
DisplayDNError(0,hResultCode);
goto Failure;
}
}
#endif // ! DPNBUILD_ONLYONESP
//
// Ensure SP is loaded, if we haven't already.
//
if (pSP == NULL)
{
hResultCode = DN_SPEnsureLoaded(pdnObject,
#ifndef DPNBUILD_ONLYONESP
&guidSP,
#endif // ! DPNBUILD_ONLYONESP
#ifndef DPNBUILD_LIBINTERFACE
NULL,
#endif // ! DPNBUILD_LIBINTERFACE
&pSP);
if (hResultCode != DPN_OK)
{
DPFERR("Could not find or load SP");
DisplayDNError(0,hResultCode);
goto Failure;
}
}
#ifndef DPNBUILD_ONLYONEADAPTER
//
// Multicast listen device addresses are formed by taking the adapter GUID from
// the user's device address, combined with the entire user specified group address.
// If there is no adapter, then we will pick the best one as reported by the service
// provider (that part occurs inside DNPerformSPListen).
//
if ((hResultCode = IDirectPlay8Address_GetDevice(pIDevice,&guidAdapter)) == DPN_OK)
{
if ((hResultCode = IDirectPlay8Address_SetDevice(pIGroup,&guidAdapter)) != DPN_OK)
{
DPFERR("Could not set SP on Group Address");
DisplayDNError(0,hResultCode);
goto Failure;
}
}
#endif // ! DPNBUILD_ONLYONEADAPTER
dwListenFlags = DN_LISTENFLAGS_MULTICAST;
if (dwFlags & DPNJOIN_ALLOWUNKNOWNSENDERS)
{
dwListenFlags |= DN_LISTENFLAGS_ALLOWUNKNOWNSENDERS;
}
//
// Start multicast listen
//
if ((hResultCode = AsyncOpNew(pdnObject,&pListenParent)) != DPN_OK)
{
DPFERR("Could not create AsyncOp");
DisplayDNError(0,hResultCode);
goto Failure;
}
pListenParent->SetOpType( ASYNC_OP_LISTEN_MULTICAST );
pListenParent->SetOpFlags( dwListenFlags );
pListenParent->MakeParent();
pListenParent->SetCompletion( DNCompleteListen );
// See note above about why it's pIGroup instead of pIDevice.
if ((hResultCode = DNPerformSPListen(pdnObject,pIGroup,pListenParent,&pParent)) != DPN_OK)
{
DPFERR("Could not start LISTEN");
DisplayDNError(0,hResultCode);
goto Failure;
}
pListenParent->AddRef();
DNEnterCriticalSection(&pdnObject->csDirectNetObject);
pdnObject->pListenParent = pListenParent;
DNLeaveCriticalSection(&pdnObject->csDirectNetObject);
pListenParent->Release();
pListenParent = NULL;
//
// Multicast send endpoints are created using the user specified device address for the
// device, and what the SP reports it's using for the multicast listen as the group
// address.
//
pParent->Lock();
DNASSERT(! pParent->IsCancelled());
DNASSERT(! pParent->m_bilinkParent.IsEmpty());
pAsyncOp = CONTAINING_OBJECT(pParent->m_bilinkParent.GetNext(),CAsyncOp,m_bilinkChildren);
pAsyncOp->Lock();
DNASSERT((! pAsyncOp->IsCancelled()) && (! pAsyncOp->IsComplete()));
DNASSERT(pAsyncOp->m_bilinkChildren.GetNext() == &pParent->m_bilinkParent);
hEndpoint = pAsyncOp->GetProtocolHandle();
pAsyncOp->Unlock();
pParent->Unlock();
spInfoData.hEndpoint = hEndpoint;
spInfoData.Flags = SP_GET_ADDRESS_INFO_MULTICAST_GROUP;
if ((hResultCode = DNPGetListenAddressInfo(pdnObject->pdnProtocolData, spInfoData.hEndpoint,&spInfoData)) != DPN_OK)
{
DPFERR("Could not get LISTEN multicast group address!");
DisplayDNError(0,hResultCode);
DNASSERT(FALSE);
goto Failure;
}
// Release the old group address and save this new one.
IDirectPlay8Address_Release(pIGroup);
pIGroup = spInfoData.pAddress;
spInfoData.pAddress = NULL;
//
// If the user did not specify an adapter GUID, we selected one and we need to
// copy that selection over to the connect operation's device address. If the user
// did specify an adapter GUID, then this DNPGetListenAddressInfo will just be
// echoing it back, so there's no harm in copying it in either case.
//
spInfoData.hEndpoint = hEndpoint;
spInfoData.Flags = SP_GET_ADDRESS_INFO_LOCAL_ADAPTER;
if ((hResultCode = DNPGetListenAddressInfo(pdnObject->pdnProtocolData, spInfoData.hEndpoint,&spInfoData)) != DPN_OK)
{
DPFERR("Could not get LISTEN device address!");
DisplayDNError(0,hResultCode);
DNASSERT(FALSE);
goto Failure;
}
#ifndef DPNBUILD_ONLYONEADAPTER
if ((hResultCode = IDirectPlay8Address_GetDevice(spInfoData.pAddress,&guidAdapter)) != DPN_OK)
{
DPFERR("Could not get LISTEN device GUID!");
DisplayDNError(0,hResultCode);
DNASSERT(FALSE);
IDirectPlay8Address_Release(spInfoData.pAddress);
spInfoData.pAddress = NULL;
goto Failure;
}
if ((hResultCode = IDirectPlay8Address_SetDevice(pIDevice,&guidAdapter)) != DPN_OK)
{
DPFERR("Could not set CONNECT device GUID!");
DisplayDNError(0,hResultCode);
DNASSERT(FALSE);
IDirectPlay8Address_Release(spInfoData.pAddress);
spInfoData.pAddress = NULL;
goto Failure;
}
#endif // ! DPNBUILD_ONLYONEADAPTER
//
// Keep device address and connect SP on DirectNet object
//
pSP->AddRef();
DNEnterCriticalSection(&pdnObject->csDirectNetObject);
pdnObject->pIDP8ADevice = spInfoData.pAddress; // Transfering reference
spInfoData.pAddress = NULL;
pdnObject->pConnectSP = pSP;
DNLeaveCriticalSection(&pdnObject->csDirectNetObject);
pParent->Release();
pParent = NULL;
//
// Create parent async op, which will be released when the ENTIRE connection is finished
//
if ((hResultCode = AsyncOpNew(pdnObject,&pConnectParent)) != DPN_OK)
{
DPFERR("Could not create AsyncOp");
DisplayDNError(0,hResultCode);
DNASSERT(FALSE);
goto Failure;
}
pConnectParent->SetOpType( ASYNC_OP_CONNECT_MULTICAST_SEND );
pConnectParent->MakeParent();
pConnectParent->SetResult( DPNERR_NOCONNECTION );
pConnectParent->SetCompletion( DNCompleteJoinOperation );
// pConnectParent->SetReserved(1);
if (dwFlags & DPNCONNECT_SYNC)
{
DPFX(DPFPREP, 5,"Sync operation - create sync event");
if ((hResultCode = SyncEventNew(pdnObject,&pSyncEvent)) != DPN_OK)
{
DPFERR("Could not create synchronization event");
DisplayDNError(0,hResultCode);
DNASSERT(FALSE);
goto Failure;
}
pConnectParent->SetSyncEvent( pSyncEvent );
pConnectParent->SetResultPointer( &hrOperation );
}
else
{
DPFX(DPFPREP, 5,"Async operation - create handle parent");
if ((hResultCode = DNCreateUserHandle(pdnObject,&pHandleParent)) != DPN_OK)
{
DPFERR("Could not create handle parent");
DisplayDNError(0,hResultCode);
DNASSERT(FALSE);
goto Failure;
}
pHandleParent->SetContext( pvAsyncContext );
pHandleParent->Lock();
if (pHandleParent->IsCancelled())
{
pHandleParent->Unlock();
pConnectParent->SetResult( DPNERR_USERCANCEL );
hResultCode = DPNERR_USERCANCEL;
goto Failure;
}
pConnectParent->MakeChild( pHandleParent );
pHandleParent->Unlock();
}
hResultCode = DNPerformConnect(pdnObject,
0,
pIDevice,
pIGroup,
pSP,
DN_CONNECTFLAGS_MULTICAST_SEND,
pConnectParent);
if (hResultCode != DPN_OK)
{
DPFERR("Could not start CONNECT");
goto Failure;
}
pConnectParent->Release();
pConnectParent = NULL;
pSP->Release();
pSP = NULL;
if (pIGroup)
{
IDirectPlay8Address_Release(pIGroup);
pIGroup = NULL;
}
if (pIDevice)
{
IDirectPlay8Address_Release(pIDevice);
pIDevice = NULL;
}
if (dwFlags & DPNCONNECT_SYNC)
{
if ((hResultCode = pSyncEvent->WaitForEvent()) != DPN_OK)
{
DPFERR("DNSyncEventWait() terminated bizarrely");
DNASSERT(FALSE);
}
else
{
hResultCode = hrOperation;
}
pSyncEvent->ReturnSelfToPool();
pSyncEvent = NULL;
}
else
{
pHandleParent->SetCompletion( DNCompleteUserJoin );
if (phAsyncHandle)
{
*phAsyncHandle = pHandleParent->GetHandle();
}
pHandleParent->Release();
pHandleParent = NULL;
hResultCode = DPNERR_PENDING;
}
Exit:
CallbackThread.Deinitialize();
DPFX(DPFPREP, 2,"Returning: [0x%lx]",hResultCode);
return(hResultCode);
Failure:
if (pIGroup)
{
IDirectPlay8Address_Release(pIGroup);
pIGroup = NULL;
}
if (pIDevice)
{
IDirectPlay8Address_Release(pIDevice);
pIDevice = NULL;
}
if (pExistingConnection != NULL)
{
pExistingConnection->Release();
pExistingConnection = NULL;
}
if (pShareListenParent != NULL)
{
pShareListenParent->Unlock();
pShareListenParent->Release();
pShareListenParent = NULL;
}
if (pShareSP != NULL)
{
pShareSP->Release();
pShareSP = NULL;
}
if (pShareDP8ServiceProvider != NULL)
{
IDP8ServiceProvider_Release(pShareDP8ServiceProvider);
pShareDP8ServiceProvider = NULL;
}
if (pLocalDP8ServiceProvider != NULL)
{
IDP8ServiceProvider_Release(pLocalDP8ServiceProvider);
pLocalDP8ServiceProvider = NULL;
}
if (pvNewInterface != NULL)
{
// Even if it's not an address, the Vtbl should be the same.
IDirectPlay8Address_Release((IDirectPlay8Address*) pvNewInterface);
pvNewInterface = NULL;
}
if (pSP)
{
pSP->Release();
pSP = NULL;
}
if (pListenParent)
{
pListenParent->Release();
pListenParent = NULL;
}
if (pListenParent)
{
pParent->Release();
pParent = NULL;
}
if (pConnectParent)
{
if (SUCCEEDED(pdnObject->HandleTable.Destroy( pConnectParent->GetHandle(), NULL )))
{
// Release the HandleTable reference
pConnectParent->Release();
}
pConnectParent->Release();
pConnectParent = NULL;
}
if (pHandleParent)
{
pHandleParent->Release();
pHandleParent = NULL;
}
if (pSyncEvent)
{
pSyncEvent->ReturnSelfToPool();
pSyncEvent = NULL;
}
DNEnterCriticalSection(&pdnObject->csDirectNetObject);
pListenParent = pdnObject->pListenParent;
pdnObject->pListenParent = NULL;
pSP = pdnObject->pConnectSP;
pdnObject->pConnectSP = NULL;
pIDevice = pdnObject->pIDP8ADevice;
pdnObject->pIDP8ADevice = NULL;
DNLeaveCriticalSection(&pdnObject->csDirectNetObject);
if (pListenParent)
{
DNCancelChildren(pdnObject,pListenParent);
pListenParent->Release();
pListenParent = NULL;
}
if (pSP)
{
pSP->Release();
pSP = NULL;
}
if (pIDevice)
{
IDirectPlay8Address_Release(pIDevice);
pIDevice = NULL;
}
DNEnterCriticalSection(&pdnObject->csDirectNetObject);
pdnObject->dwFlags &= ~(DN_OBJECT_FLAG_CONNECTING|DN_OBJECT_FLAG_CONNECTED|DN_OBJECT_FLAG_LOCALHOST);
DNLeaveCriticalSection(&pdnObject->csDirectNetObject);
goto Exit;
} // DN_Join
//
// Completion for create sender context
//
#undef DPF_MODNAME
#define DPF_MODNAME "DNCompleteCreateSenderContext"
void DNCompleteCreateSenderContext(DIRECTNETOBJECT *const pdnObject,
CAsyncOp *const pAsyncOp)
{
}
//
// Associate a context value with a given multicast sender's address
//
#undef DPF_MODNAME
#define DPF_MODNAME "DN_CreateSenderContext"
STDMETHODIMP DN_CreateSenderContext( IDirectPlay8Multicast *pInterface,
IDirectPlay8Address *const pSenderAddress,
void *const pvSenderContext,
const DWORD dwFlags )
{
HRESULT hResultCode;
PDIRECTNETOBJECT pdnObject;
IDirectPlay8Address *pIDevice;
CServiceProvider *pSP;
CAsyncOp *pConnectParent;
CSyncEvent *pSyncEvent;
HRESULT hrConnect;
DPFX(DPFPREP, 2,"Parameters: pInterface [0x%p], pSenderAddress [0x%p], pvSenderContext [0x%p], dwFlags [0x%lx]",
pInterface,pSenderAddress,pvSenderContext,dwFlags);
pdnObject = (DIRECTNETOBJECT*) GET_OBJECT_FROM_INTERFACE(pInterface);
DNASSERT(pdnObject != NULL);
#ifndef DPNBUILD_NOPARAMVAL
if( pdnObject->dwFlags & DN_OBJECT_FLAG_PARAMVALIDATION )
{
if( FAILED( hResultCode = DN_ValidateCreateSenderContext( pInterface, pSenderAddress,
pvSenderContext, dwFlags ) ) )
{
DPFERR( "Error validating params" );
DPF_RETURN(hResultCode);
}
}
#endif // DPNBUILD_NOPARAMVAL
if( !(pdnObject->dwFlags & DN_OBJECT_FLAG_INITIALIZED) )
{
DPFERR( "Object is not initialized" );
DPF_RETURN( DPNERR_UNINITIALIZED );
}
pIDevice = NULL;
pSP = NULL;
pConnectParent = NULL;
pSyncEvent = NULL;
#pragma BUGBUG( minara, "Need to ensure not closing" )
//
// Extract device and SP from DirectNet object
//
DNEnterCriticalSection(&pdnObject->csDirectNetObject);
DNASSERT( pdnObject->pConnectSP != NULL );
DNASSERT( pdnObject->pIDP8ADevice != NULL );
if (pdnObject->pConnectSP)
{
pdnObject->pConnectSP->AddRef();
pSP = pdnObject->pConnectSP;
}
if (pdnObject->pIDP8ADevice)
{
IDirectPlay8Address_AddRef(pdnObject->pIDP8ADevice);
pIDevice = pdnObject->pIDP8ADevice;
}
DNLeaveCriticalSection(&pdnObject->csDirectNetObject);
if (pSP == NULL || pIDevice == NULL)
{
DPFERR("Invalid connect SP or device address");
hResultCode = DPNERR_GENERIC;
goto Failure;
}
//
// Create SyncEvent
//
if ((hResultCode = SyncEventNew(pdnObject,&pSyncEvent)) != DPN_OK)
{
DPFERR("Could not create SyncEvent");
goto Failure;
}
//
// Create AsyncOp parent
//
if ((hResultCode = AsyncOpNew(pdnObject,&pConnectParent)) != DPN_OK)
{
DPFERR("Could not create AsyncOp");
goto Failure;
}
pConnectParent->SetOpType( ASYNC_OP_CONNECT_MULTICAST_RECEIVE );
pConnectParent->SetSyncEvent( pSyncEvent );
pConnectParent->SetResultPointer( &hrConnect );
pConnectParent->SetContext( pvSenderContext );
//
// We will call connect on the Protocol to associate the sender's endpoint with a context
// This call to connect must be turned into a synchronous operation since this API call is synchronous
//
if ((hResultCode = DNPerformConnect(pdnObject,
0,
pIDevice,
pSenderAddress,
pSP,
DN_CONNECTFLAGS_MULTICAST_RECEIVE,
pConnectParent)) != DPN_OK)
{
DPFERR("Failed to connect");
DisplayDNError(0,hResultCode);
goto Failure;
}
//
// Release references and wait for completion
//
pConnectParent->SetCompletion( DNCompleteCreateSenderContext );
pConnectParent->Release();
pConnectParent = NULL;
pSP->Release();
pSP = NULL;
IDirectPlay8Address_Release(pIDevice);
pIDevice = NULL;
pSyncEvent->WaitForEvent();
pSyncEvent->ReturnSelfToPool();
pSyncEvent = NULL;
if (hrConnect == DPN_OK)
{
DNUserCreateSenderContext(pdnObject,pvSenderContext);
}
hResultCode = hrConnect;
Exit:
DNASSERT( pIDevice == NULL );
DNASSERT( pSP == NULL );
DNASSERT( pConnectParent == NULL );
DNASSERT( pSyncEvent == NULL );
DPFX(DPFPREP, 2,"Returning: [0x%lx]",hResultCode);
return(hResultCode);
Failure:
if (pIDevice)
{
IDirectPlay8Address_Release(pIDevice);
pIDevice = NULL;
}
if (pSP)
{
pSP->Release();
pSP = NULL;
}
if (pConnectParent)
{
pConnectParent->Release();
pConnectParent = NULL;
}
if (pSyncEvent)
{
pSyncEvent->ReturnSelfToPool();
pSyncEvent = NULL;
}
goto Exit;
} // DN_CreateSenderContext
//
// Removes a previously associated context value from a given multicast sender's address
//
#undef DPF_MODNAME
#define DPF_MODNAME "DN_DestroySenderContext"
STDMETHODIMP DN_DestroySenderContext( IDirectPlay8Multicast *pInterface,
IDirectPlay8Address *const pSenderAddress,
const DWORD dwFlags )
{
HRESULT hResultCode;
DIRECTNETOBJECT *pdnObject;
CServiceProvider *pSP;
CConnection *pConnection;
IDirectPlay8Address *pDevice;
DPFX(DPFPREP, 2,"Parameters: pInterface [0x%p], pSenderAddress [0x%p], dwFlags [0x%lx]",
pInterface,pSenderAddress,dwFlags);
pdnObject = (DIRECTNETOBJECT*) GET_OBJECT_FROM_INTERFACE(pInterface);
DNASSERT(pdnObject != NULL);
#ifndef DPNBUILD_NOPARAMVAL
if( pdnObject->dwFlags & DN_OBJECT_FLAG_PARAMVALIDATION )
{
if( FAILED( hResultCode = DN_ValidateDestroySenderContext( pInterface, pSenderAddress, dwFlags ) ) )
{
DPFERR( "Error validating params" );
DPF_RETURN(hResultCode);
}
}
#endif // DPNBUILD_NOPARAMVAL
if( !(pdnObject->dwFlags & DN_OBJECT_FLAG_INITIALIZED) )
{
DPFERR( "Object is not initialized" );
DPF_RETURN( DPNERR_UNINITIALIZED );
}
pSP = NULL;
pConnection = NULL;
DNEnterCriticalSection(&pdnObject->csDirectNetObject);
if (pdnObject->pIDP8ADevice)
{
IDirectPlay8Address_AddRef(pdnObject->pIDP8ADevice);
pDevice = pdnObject->pIDP8ADevice;
}
if (pdnObject->pConnectSP)
{
pdnObject->pConnectSP->AddRef();
pSP = pdnObject->pConnectSP;
}
DNLeaveCriticalSection(&pdnObject->csDirectNetObject);
//
// Get endpoint from address
//
if ((hResultCode = DNPGetEndPointContextFromAddress(pdnObject->pdnProtocolData, pSP->GetHandle(),pSenderAddress,pDevice,reinterpret_cast<void**>(&pConnection))) == DPN_OK)
{
pConnection->AddRef();
pConnection->Disconnect();
pConnection->Release();
pConnection = NULL;
}
//
// Clean up
//
IDirectPlay8Address_Release(pDevice);
pDevice = NULL;
pSP->Release();
pSP = NULL;
hResultCode = DPN_OK;
DNASSERT( pSP == NULL );
DNASSERT( pConnection == NULL );
DPFX(DPFPREP, 2,"Returning: [0x%lx]",hResultCode);
return(hResultCode);
} // DN_DestroySenderContext
//
// Retrieves the current multicast group address
//
#undef DPF_MODNAME
#define DPF_MODNAME "DN_GetGroupAddress"
STDMETHODIMP DN_GetGroupAddress(IDirectPlay8Multicast *pInterface,
IDirectPlay8Address **const ppAddress,
const DWORD dwFlags)
{
HRESULT hResultCode;
DIRECTNETOBJECT *pdnObject;
CConnection *pConnection;
HANDLE hEndPt;
SPGETADDRESSINFODATA spInfoData;
CCallbackThread CallbackThread;
DPFX(DPFPREP, 2,"Parameters : pInterface [0x%p], ppAddress [0x%p], dwFlags [0x%lx]",
pInterface,ppAddress,dwFlags);
pdnObject = (DIRECTNETOBJECT*) GET_OBJECT_FROM_INTERFACE(pInterface);
DNASSERT(pdnObject != NULL);
#ifndef DPNBUILD_NOPARAMVAL
if( pdnObject->dwFlags & DN_OBJECT_FLAG_PARAMVALIDATION )
{
if( FAILED( hResultCode = DN_ValidateGetGroupAddress( pInterface, ppAddress, dwFlags ) ) )
{
DPFX(DPFPREP, 0, "Error validating get group address info hr=[0x%lx]", hResultCode );
DPF_RETURN( hResultCode );
}
}
#endif // !DPNBUILD_NOPARAMVAL
// Check to ensure message handler registered
if (!(pdnObject->dwFlags & DN_OBJECT_FLAG_INITIALIZED))
{
DPFERR( "Object is not initialized" );
DPF_RETURN(DPNERR_UNINITIALIZED);
}
if( pdnObject->dwFlags & DN_OBJECT_FLAG_CONNECTING )
{
DPFERR("Object is connecting / starting to host" );
DPF_RETURN(DPNERR_CONNECTING);
}
if ( !(pdnObject->dwFlags & (DN_OBJECT_FLAG_CONNECTED | DN_OBJECT_FLAG_CLOSING | DN_OBJECT_FLAG_DISCONNECTING) ) )
{
DPFERR("You must be connected / disconnecting to use this function" );
DPF_RETURN(DPNERR_NOCONNECTION);
}
CallbackThread.Initialize();
DNEnterCriticalSection(&pdnObject->csDirectNetObject);
if (pdnObject->pMulticastSend != NULL)
{
pdnObject->pMulticastSend->AddRef();
pConnection = pdnObject->pMulticastSend;
}
else
{
pConnection = NULL;
}
DNLeaveCriticalSection(&pdnObject->csDirectNetObject);
if (pConnection == NULL)
{
DPFERR( "Couldn't retrieve multicast send connection" );
hResultCode = DPNERR_INVALIDGROUP;
goto Failure;
}
//
// Get the remote multicast address address
//
if ((hResultCode = pConnection->GetEndPt(&hEndPt,&CallbackThread)) != DPN_OK)
{
DPFERR( "Couldn't retrieve multicast send endpoint" );
DisplayDNError(0, hResultCode);
goto Failure;
}
spInfoData.Flags = SP_GET_ADDRESS_INFO_MULTICAST_GROUP;
hResultCode = DNPCrackEndPointDescriptor(pdnObject->pdnProtocolData, hEndPt,&spInfoData);
if (hResultCode != DPN_OK)
{
DPFERR("Unknown error from DNPCrackEndPointDescriptor");
DisplayDNError(0,hResultCode);
DNASSERT(FALSE);
// Drop through...
}
pConnection->ReleaseEndPt(&CallbackThread);
pConnection->Release();
pConnection = NULL;
*ppAddress = spInfoData.pAddress;
spInfoData.pAddress = NULL;
Exit:
CallbackThread.Deinitialize();
DPF_RETURN(hResultCode);
Failure:
if (pConnection)
{
pConnection->Release();
pConnection = NULL;
}
goto Exit;
} // DN_GetGroupAddress
//
// Enumerate multicast scopes reported by SP
//
#undef DPF_MODNAME
#define DPF_MODNAME "DN_EnumMulticastScopes"
STDMETHODIMP DN_EnumMulticastScopes( IDirectPlay8Multicast *pInterface,
const GUID *const pguidServiceProvider,
const GUID *const pguidDevice,
const GUID *const pguidApplication,
DPN_MULTICAST_SCOPE_INFO *const pScopeInfoBuffer,
DWORD *const pcbEnumData,
DWORD *const pcReturned,
const DWORD dwFlags )
{
HRESULT hResultCode;
PDIRECTNETOBJECT pdnObject;
DPFX(DPFPREP, 2,"Parameters: pInterface [0x%p], pguidServiceProvider [0x%p], pguidDevice [0x%p], pguidApplication [0x%p], pScopeInfoBuffer [0x%p], pcbEnumData [0x%p], pcReturned [0x%p], dwFlags [0x%lx]",
pInterface,pguidServiceProvider,pguidDevice,pguidApplication,pScopeInfoBuffer,pcbEnumData,pcReturned,dwFlags);
pdnObject = (DIRECTNETOBJECT*) GET_OBJECT_FROM_INTERFACE(pInterface);
DNASSERT(pdnObject != NULL);
#ifndef DPNBUILD_NOPARAMVAL
if( pdnObject->dwFlags & DN_OBJECT_FLAG_PARAMVALIDATION )
{
if( FAILED( hResultCode = DN_ValidateEnumMulticastScopes( pInterface, pguidServiceProvider, pguidDevice,
pguidApplication, pScopeInfoBuffer,
pcbEnumData, pcReturned, dwFlags ) ) )
{
DPFERR( "Error validating params" );
DPF_RETURN(hResultCode);
}
}
#endif // DPNBUILD_NOPARAMVAL
if( !(pdnObject->dwFlags & DN_OBJECT_FLAG_INITIALIZED) )
{
DPFERR( "Object is not initialized" );
DPF_RETURN( DPNERR_UNINITIALIZED );
}
hResultCode = DN_EnumMulticastScopes(pdnObject,
dwFlags,
#ifndef DPNBUILD_ONLYONESP
pguidServiceProvider,
#endif // ! DPNBUILD_ONLYONESP
#ifndef DPNBUILD_ONLYONEADAPTER
pguidDevice,
#endif // ! DPNBUILD_ONLYONEADAPTER
#ifndef DPNBUILD_LIBINTERFACE
pguidApplication,
#endif // ! DPNBUILD_LIBINTERFACE
pScopeInfoBuffer,
pcbEnumData,
pcReturned);
DPFX(DPFPREP, 3,"Set: *pcbEnumData [%ld], *pcReturned [%ld]",*pcbEnumData,*pcReturned);
DPFX(DPFPREP, 2,"Returning: [0x%lx]",hResultCode);
return(hResultCode);
} // DN_EnumMulticastScopes
#endif // ! DPNBUILD_NOMULTICAST