490 lines
15 KiB
C++
490 lines
15 KiB
C++
/*==========================================================================
|
|
*
|
|
* Copyright (C) 2000 Microsoft Corporation. All Rights Reserved.
|
|
*
|
|
* File: Disconnect.cpp
|
|
* Content: DNET disconnection routines
|
|
*@@BEGIN_MSINTERNAL
|
|
* History:
|
|
* Date By Reason
|
|
* ==== == ======
|
|
* 09/15/99 mjn Created
|
|
* 12/23/99 mjn Hand all NameTable update sends from Host to worker thread
|
|
* 12/23/99 mjn Fixed PlayerDisconnect to prevent user notification of
|
|
* deletion from ALL_PLAYERS group
|
|
* 12/23/99 mjn Added basic host migration functionality
|
|
* 12/28/99 mjn Complete outstanding operations in DNPlayerDisconnectNew
|
|
* 12/28/99 mjn Moved Async Op stuff to Async.h
|
|
* 12/29/99 mjn Reformed DN_ASYNC_OP to use hParentOp instead of lpvUserContext
|
|
* 01/03/00 mjn Added DNPrepareToDeletePlayer
|
|
* 01/04/00 mjn Added code to allow outstanding ops to complete at host migration
|
|
* 01/06/99 mjn Moved NameTable stuff to NameTable.h
|
|
* 01/09/00 mjn Keep number of players in Application Description
|
|
* 01/10/00 mjn Check AppDesc for host migration
|
|
* 01/11/00 mjn Use CPackedBuffers instead of DN_ENUM_BUFFER_INFOs
|
|
* 01/15/00 mjn Replaced DN_COUNT_BUFFER with CRefCountBuffer
|
|
* 01/16/00 mjn Moved User callback stuff to User.h
|
|
* 01/18/00 mjn Added DNAutoDestructGroups
|
|
* 01/19/00 mjn Replaced DN_SYNC_EVENT with CSyncEvent
|
|
* 01/20/00 mjn Fixed CBilink usage problem in DNLocalDisconnect
|
|
* 01/22/00 mjn Added DNProcessHostDestroyPlayer
|
|
* 01/23/00 mjn Update NameTable version for instructed disconnects
|
|
* 01/24/00 mjn Use DNNTUpdateVersion to update NameTable version
|
|
* 01/25/00 mjn Changed Host Migration to multi-step affair
|
|
* 02/01/00 mjn Implement Player/Group context values
|
|
* 04/05/00 mjn Updated DNProcessHostDestroyPlayer()
|
|
* 04/12/00 mjn Removed DNAutoDestructGroups - covered in NameTable.DeletePlayer()
|
|
* mjn Don't set DN_OBJECT_FLAG_DISCONNECTING in DNPlayerDisconnect()
|
|
* 04/18/00 mjn Fixed player count problem
|
|
* 04/19/00 mjn Update NameTable operations to use DN_WORKER_JOB_SEND_NAMETABLE_OPERATION
|
|
* 05/16/00 mjn Do not take locks when clearing NameTable short-cut pointers
|
|
* 06/06/00 mjn Fixed DNPlayerDisconnect to always check for host migration in peer-peer mode w/ host migration flag
|
|
* 07/07/00 mjn Clear host migration status if new host disconnects during migration process
|
|
* 07/20/00 mjn Use ClearHostWithDPNID() to clear HostPlayer in DNPlayerDisconnectNew()
|
|
* 07/29/00 mjn Fix calls to DNUserConnectionTerminated()
|
|
* 07/30/00 mjn Use DNUserTerminateSession() rather than DNUserConnectionTerminated()
|
|
* 07/31/00 mjn Added hrReason to DNTerminateSession()
|
|
* mjn Added dwDestroyReason to DNHostDisconnect()
|
|
* mjn Removed DNProcessHostDestroyPlayer()
|
|
* 07/31/00 mjn Change DN_MSG_INTERNAL_DELETE_PLAYER to DN_MSG_INTERNAL_DESTROY_PLAYER
|
|
* 08/05/00 RichGr IA64: Use %p format specifier in DPFs for 32/64-bit pointers and handles.
|
|
* 08/05/00 mjn Prevent DN_MSG_INTERNAL_DESTROY_PLAYER from being sent out in client/server
|
|
* 08/06/00 mjn Added CWorkerJob
|
|
* 08/07/00 mjn Added code to request peer-peer integrity checks and clean up afterwards
|
|
* 09/04/00 mjn Added CApplicationDesc
|
|
* 09/26/00 mjn Removed locking from CNameTable::SetVersion() and CNameTable::GetNewVersion()
|
|
* 01/25/01 mjn Fixed 64-bit alignment problem in received messages
|
|
* 02/12/01 mjn Fixed CConnection::GetEndPt() to track calling thread
|
|
* 04/13/01 mjn Remove request for integrity check from request list in DNInstructedDisconnect()
|
|
* 07/22/01 mjn Added DPNBUILD_NOHOSTMIGRATE compile flag
|
|
*@@END_MSINTERNAL
|
|
*
|
|
***************************************************************************/
|
|
|
|
#include "dncorei.h"
|
|
|
|
|
|
// DNPlayerDisconnectNew
|
|
//
|
|
// Another player has issued a disconnect with the local player.
|
|
// - If the disconnecting player is still in the nametable
|
|
// - prepare to delete player
|
|
// - Save one refcount to be released by DELETE_PLAYER from host or Close
|
|
// - check host migration
|
|
|
|
#undef DPF_MODNAME
|
|
#define DPF_MODNAME "DNPlayerDisconnectNew"
|
|
|
|
HRESULT DNPlayerDisconnectNew(DIRECTNETOBJECT *const pdnObject,
|
|
const DPNID dpnid)
|
|
{
|
|
CNameTableEntry *pNTEntry;
|
|
CNameTableEntry *pLocalPlayer;
|
|
HRESULT hResultCode;
|
|
#ifndef DPNBUILD_NOHOSTMIGRATE
|
|
DPNID dpnidNewHost;
|
|
#endif // !DPNBUILD_NOHOSTMIGRATE
|
|
BOOL fWasHost;
|
|
BOOL fRequestIntegrityCheck;
|
|
|
|
DPFX(DPFPREP, 4,"Parameters: dpnid [0x%lx]",dpnid);
|
|
|
|
DNASSERT(pdnObject != NULL);
|
|
|
|
pNTEntry = NULL;
|
|
pLocalPlayer = NULL;
|
|
|
|
if (pdnObject->dwFlags & DN_OBJECT_FLAG_CLIENT)
|
|
{
|
|
//
|
|
// The Server has disconnected
|
|
// We will indicate the connection terminated and shut down
|
|
//
|
|
DPFX(DPFPREP, 5,"Server has disconnected from this client");
|
|
DNUserTerminateSession(pdnObject,DPNERR_CONNECTIONLOST,NULL,0);
|
|
DNTerminateSession(pdnObject,DPNERR_CONNECTIONLOST);
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Another peer has disconnected from this peer
|
|
// We will delete this player from the NameTable
|
|
// We may have to ask the host to perform an integrity check
|
|
//
|
|
DPFX(DPFPREP, 5,"Peer has disconnected from this peer");
|
|
if ((hResultCode = pdnObject->NameTable.FindEntry(dpnid,&pNTEntry)) != DPN_OK)
|
|
{
|
|
DPFERR("Could not find disconnecting player in NameTable");
|
|
DisplayDNError(0,hResultCode);
|
|
goto Failure;
|
|
}
|
|
fRequestIntegrityCheck = FALSE;
|
|
pNTEntry->Lock();
|
|
if (pNTEntry->IsAvailable())
|
|
{
|
|
fRequestIntegrityCheck = TRUE;
|
|
}
|
|
pNTEntry->Unlock();
|
|
if (fRequestIntegrityCheck)
|
|
{
|
|
DNRequestIntegrityCheck(pdnObject,dpnid);
|
|
}
|
|
pdnObject->NameTable.DeletePlayer(dpnid,0);
|
|
|
|
//
|
|
// If this was the Host, clear the short-cut pointer
|
|
//
|
|
fWasHost = pdnObject->NameTable.ClearHostWithDPNID( dpnid );
|
|
|
|
//
|
|
// May need to clear HOST_MIGRATING flag
|
|
//
|
|
DNEnterCriticalSection(&pdnObject->csDirectNetObject);
|
|
if ((pdnObject->dwFlags & DN_OBJECT_FLAG_HOST_MIGRATING) && (pdnObject->pNewHost == pNTEntry))
|
|
{
|
|
pdnObject->dwFlags &= ~(DN_OBJECT_FLAG_HOST_MIGRATING);
|
|
pdnObject->pNewHost->Release();
|
|
pdnObject->pNewHost = NULL;
|
|
}
|
|
DNLeaveCriticalSection(&pdnObject->csDirectNetObject);
|
|
|
|
#ifndef DPNBUILD_NOHOSTMIGRATE
|
|
//
|
|
// If HostMigration flag is set, check to see if we are the new Host.
|
|
// Otherwise, if the disconnecting player was the Host, the session is lost.
|
|
//
|
|
if (pdnObject->ApplicationDesc.AllowHostMigrate())
|
|
{
|
|
DPFX(DPFPREP, 5,"Host-Migration was set - check for new Host");
|
|
dpnidNewHost = 0;
|
|
if ((hResultCode = DNFindNewHost(pdnObject,&dpnidNewHost)) == DPN_OK)
|
|
{
|
|
DPFX(DPFPREP, 5,"New Host [0x%lx]",dpnidNewHost);
|
|
if ((hResultCode = pdnObject->NameTable.GetLocalPlayerRef(&pLocalPlayer)) == DPN_OK)
|
|
{
|
|
if (pLocalPlayer->GetDPNID() == dpnidNewHost)
|
|
{
|
|
DPFX(DPFPREP, 5,"Local player is new Host");
|
|
hResultCode = DNPerformHostMigration1(pdnObject,dpnid);
|
|
}
|
|
|
|
pLocalPlayer->Release();
|
|
pLocalPlayer = NULL;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
#endif // DPNBUILD_NOHOSTMIGRATE
|
|
{
|
|
if (fWasHost)
|
|
{
|
|
DPFX(DPFPREP, 5,"Host-Migration was not set - terminating session");
|
|
DNUserTerminateSession(pdnObject,DPNERR_CONNECTIONLOST,NULL,0);
|
|
DNTerminateSession(pdnObject,DPNERR_CONNECTIONLOST);
|
|
}
|
|
}
|
|
|
|
pNTEntry->Release();
|
|
pNTEntry = NULL;
|
|
}
|
|
|
|
hResultCode = DPN_OK;
|
|
|
|
Exit:
|
|
DPFX(DPFPREP, 4,"Returning: [0x%lx]",hResultCode);
|
|
return(hResultCode);
|
|
|
|
Failure:
|
|
if (pNTEntry)
|
|
{
|
|
pNTEntry->Release();
|
|
pNTEntry = NULL;
|
|
}
|
|
if (pLocalPlayer)
|
|
{
|
|
pLocalPlayer->Release();
|
|
pLocalPlayer = NULL;
|
|
}
|
|
goto Exit;
|
|
}
|
|
|
|
|
|
// DNHostDisconnect
|
|
//
|
|
// A player has initiated a disconnect with the host.
|
|
// - Remove player from the name table
|
|
// - Propegate DELETE_PLAYER messages to each player
|
|
#pragma TODO(minara,"Use pConnection instead of dpnidDisconnecting ?")
|
|
|
|
#undef DPF_MODNAME
|
|
#define DPF_MODNAME "DNHostDisconnect"
|
|
|
|
HRESULT DNHostDisconnect(DIRECTNETOBJECT *const pdnObject,
|
|
const DPNID dpnidDisconnecting,
|
|
const DWORD dwDestroyReason)
|
|
{
|
|
HRESULT hResultCode;
|
|
CRefCountBuffer *pRefCountBuffer;
|
|
CPendingDeletion *pPending;
|
|
CWorkerJob *pWorkerJob;
|
|
DN_INTERNAL_MESSAGE_DESTROY_PLAYER *pMsg;
|
|
|
|
DPFX(DPFPREP, 4,"Parameters: pdnObject [0x%p], dpnidDisconnecting [0x%lx], dwDestroyReason [0x%lx]",
|
|
pdnObject,dpnidDisconnecting,dwDestroyReason);
|
|
|
|
DNASSERT(pdnObject != NULL);
|
|
DNASSERT(dpnidDisconnecting != 0);
|
|
|
|
pRefCountBuffer = NULL;
|
|
pPending = NULL;
|
|
pWorkerJob = NULL;
|
|
|
|
// Remove entry from NameTable and inform other players, only if Host is NOT migrating
|
|
// Otherwise, clean-up first and wait for migration to complete before informing others
|
|
if (!(pdnObject->dwFlags & DN_OBJECT_FLAG_HOST_MIGRATING))
|
|
{
|
|
DWORD dwVersion;
|
|
CNameTableEntry *pNTEntry;
|
|
|
|
dwVersion = 0;
|
|
pNTEntry = NULL;
|
|
|
|
if ((hResultCode = pdnObject->NameTable.FindEntry(dpnidDisconnecting,&pNTEntry)) == DPN_OK)
|
|
{
|
|
pNTEntry->Lock();
|
|
if (pNTEntry->GetDestroyReason() == 0)
|
|
{
|
|
pNTEntry->SetDestroyReason( dwDestroyReason );
|
|
}
|
|
pNTEntry->Unlock();
|
|
pNTEntry->Release();
|
|
pNTEntry = NULL;
|
|
}
|
|
hResultCode = pdnObject->NameTable.DeletePlayer(dpnidDisconnecting,&dwVersion);
|
|
|
|
if (pdnObject->dwFlags & DN_OBJECT_FLAG_PEER)
|
|
{
|
|
//
|
|
// Prepare internal message
|
|
//
|
|
hResultCode = RefCountBufferNew(pdnObject,
|
|
sizeof(DN_INTERNAL_MESSAGE_DESTROY_PLAYER),
|
|
MemoryBlockAlloc,
|
|
MemoryBlockFree,
|
|
&pRefCountBuffer);
|
|
if (hResultCode != DPN_OK)
|
|
{
|
|
DPFERR("Could not allocate message buffer");
|
|
DisplayDNError(0,hResultCode);
|
|
goto Failure;
|
|
}
|
|
pMsg = reinterpret_cast<DN_INTERNAL_MESSAGE_DESTROY_PLAYER*>(pRefCountBuffer->GetBufferAddress());
|
|
pMsg->dpnidLeaving = dpnidDisconnecting;
|
|
pMsg->dwVersion = dwVersion;
|
|
pMsg->dwVersionNotUsed = 0;
|
|
pMsg->dwDestroyReason = dwDestroyReason;
|
|
|
|
if ((hResultCode = WorkerJobNew(pdnObject,&pWorkerJob)) != DPN_OK)
|
|
{
|
|
DPFERR("Could not create worker job");
|
|
DisplayDNError(0,hResultCode);
|
|
goto Failure;
|
|
}
|
|
pWorkerJob->SetJobType( WORKER_JOB_SEND_NAMETABLE_OPERATION );
|
|
pWorkerJob->SetSendNameTableOperationMsgId( DN_MSG_INTERNAL_DESTROY_PLAYER );
|
|
pWorkerJob->SetSendNameTableOperationVersion( dwVersion );
|
|
pWorkerJob->SetSendNameTableOperationDPNIDExclude( dpnidDisconnecting );
|
|
pWorkerJob->SetRefCountBuffer( pRefCountBuffer );
|
|
|
|
DNQueueWorkerJob(pdnObject,pWorkerJob);
|
|
pWorkerJob = NULL;
|
|
|
|
pRefCountBuffer->Release();
|
|
pRefCountBuffer = NULL;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Put this on the Outstanding operation list
|
|
//
|
|
if ((hResultCode = PendingDeletionNew(pdnObject,&pPending)) == DPN_OK)
|
|
{
|
|
pPending->SetDPNID( dpnidDisconnecting );
|
|
|
|
DNEnterCriticalSection(&pdnObject->csNameTableOpList);
|
|
pPending->m_bilinkPendingDeletions.InsertBefore(&pdnObject->m_bilinkPendingDeletions);
|
|
DNLeaveCriticalSection(&pdnObject->csNameTableOpList);
|
|
|
|
pPending = NULL;
|
|
}
|
|
|
|
#ifndef DPNBUILD_NOHOSTMIGRATE
|
|
// See if we can continue with Host migration
|
|
DNCheckReceivedAllVersions(pdnObject);
|
|
#endif // DPNBUILD_NOHOSTMIGRATE
|
|
}
|
|
|
|
hResultCode = DPN_OK;
|
|
|
|
Exit:
|
|
DPFX(DPFPREP, 4,"Returning: [0x%lx]", hResultCode);
|
|
return(hResultCode);
|
|
|
|
Failure:
|
|
if (pRefCountBuffer)
|
|
{
|
|
pRefCountBuffer->Release();
|
|
pRefCountBuffer = NULL;
|
|
}
|
|
if (pPending)
|
|
{
|
|
pPending->ReturnSelfToPool();
|
|
pPending = NULL;
|
|
}
|
|
goto Exit;
|
|
}
|
|
|
|
|
|
// DNInstructedDisconnect
|
|
//
|
|
// The host has instructed the local player to delete another player from the nametable
|
|
// - If already closing
|
|
// - ignore this message and return
|
|
// - Prepare to delete player
|
|
// - Release refcount of player
|
|
|
|
#undef DPF_MODNAME
|
|
#define DPF_MODNAME "DNInstructedDisconnect"
|
|
|
|
HRESULT DNInstructedDisconnect(DIRECTNETOBJECT *const pdnObject,
|
|
PVOID pv)
|
|
{
|
|
HRESULT hResultCode;
|
|
DWORD dwVersion;
|
|
CNameTableEntry *pNTEntry;
|
|
UNALIGNED DN_INTERNAL_MESSAGE_DESTROY_PLAYER *pInfo;
|
|
|
|
DPFX(DPFPREP, 4,"Parameters: pv [0x%p]",pv);
|
|
|
|
DNASSERT(pdnObject != NULL);
|
|
DNASSERT(pv != NULL);
|
|
|
|
pNTEntry = NULL;
|
|
|
|
pInfo = static_cast<DN_INTERNAL_MESSAGE_DESTROY_PLAYER*>(pv);
|
|
|
|
DNASSERT(pInfo != NULL);
|
|
DNASSERT(pInfo->dpnidLeaving != NULL);
|
|
DNASSERT(pInfo->dwVersion != 0);
|
|
|
|
DPFX(DPFPREP, 5,"Deleting player [0x%lx]",pInfo->dpnidLeaving);
|
|
|
|
//
|
|
// If the player is still in the NameTable, we will preset the destroy reason.
|
|
// We will also use this "hint" to initiate a disconnect just in case the protocol
|
|
//
|
|
if ((hResultCode = pdnObject->NameTable.FindEntry(pInfo->dpnidLeaving,&pNTEntry)) == DPN_OK)
|
|
{
|
|
CConnection *pConnection;
|
|
CCallbackThread CallbackThread;
|
|
HANDLE hEndPt;
|
|
|
|
pConnection = NULL;
|
|
CallbackThread.Initialize();
|
|
hEndPt = NULL;
|
|
|
|
pNTEntry->Lock();
|
|
if (pNTEntry->GetDestroyReason() == 0)
|
|
{
|
|
pNTEntry->SetDestroyReason( pInfo->dwDestroyReason );
|
|
}
|
|
pNTEntry->Unlock();
|
|
|
|
//
|
|
// Attempt a disconnect
|
|
//
|
|
if ((hResultCode = pNTEntry->GetConnectionRef( &pConnection )) == DPN_OK)
|
|
{
|
|
if ((hResultCode = pConnection->GetEndPt(&hEndPt,&CallbackThread)) == DPN_OK)
|
|
{
|
|
DNPerformDisconnect(pdnObject,pConnection,hEndPt,FALSE);
|
|
|
|
pConnection->ReleaseEndPt(&CallbackThread);
|
|
}
|
|
pConnection->Release();
|
|
pConnection = NULL;
|
|
}
|
|
pNTEntry->Release();
|
|
pNTEntry = NULL;
|
|
CallbackThread.Deinitialize();
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Scan oustanding op list for integrity check request for this player.
|
|
// If found, remove it from the request list and the handle table
|
|
//
|
|
CBilink *pBilink;
|
|
CAsyncOp *pAsyncOp;
|
|
DN_SEND_OP_DATA *pSendOpData;
|
|
|
|
pAsyncOp = NULL;
|
|
DNEnterCriticalSection(&pdnObject->csActiveList);
|
|
pBilink = pdnObject->m_bilinkRequestList.GetNext();
|
|
while (pBilink != &pdnObject->m_bilinkRequestList)
|
|
{
|
|
pAsyncOp = CONTAINING_OBJECT(pBilink,CAsyncOp,m_bilinkActiveList);
|
|
DNASSERT(pAsyncOp->GetOpType() == ASYNC_OP_REQUEST);
|
|
pSendOpData = pAsyncOp->GetLocalSendOpData();
|
|
if (pSendOpData->dwMsgId == DN_MSG_INTERNAL_REQ_INTEGRITY_CHECK)
|
|
{
|
|
UNALIGNED DN_INTERNAL_MESSAGE_REQ_INTEGRITY_CHECK *pMsg;
|
|
|
|
pMsg = reinterpret_cast<DN_INTERNAL_MESSAGE_REQ_INTEGRITY_CHECK*>
|
|
(reinterpret_cast<UNALIGNED DN_INTERNAL_MESSAGE_REQ_PROCESS_COMPLETION*>(pSendOpData->BufferDesc[1].pBufferData) + 1);
|
|
if (pMsg->dpnidTarget == pInfo->dpnidLeaving)
|
|
{
|
|
pAsyncOp->m_bilinkActiveList.RemoveFromList();
|
|
pAsyncOp->AddRef();
|
|
break;
|
|
}
|
|
}
|
|
pSendOpData = NULL;
|
|
pAsyncOp = NULL;
|
|
pBilink = pBilink->GetNext();
|
|
}
|
|
DNLeaveCriticalSection(&pdnObject->csActiveList);
|
|
|
|
if (pAsyncOp != NULL)
|
|
{
|
|
DNASSERT(pAsyncOp->GetHandle() != 0);
|
|
if (SUCCEEDED(pdnObject->HandleTable.Destroy( pAsyncOp->GetHandle(), NULL )))
|
|
{
|
|
// Release the HandleTable reference
|
|
pAsyncOp->Release();
|
|
}
|
|
pAsyncOp->Release();
|
|
pAsyncOp = NULL;
|
|
}
|
|
DNASSERT(pAsyncOp == NULL);
|
|
}
|
|
|
|
dwVersion = pInfo->dwVersion;
|
|
pdnObject->NameTable.DeletePlayer(pInfo->dpnidLeaving,&dwVersion);
|
|
|
|
//
|
|
// Update NameTable version
|
|
//
|
|
pdnObject->NameTable.WriteLock();
|
|
pdnObject->NameTable.SetVersion(pInfo->dwVersion);
|
|
pdnObject->NameTable.Unlock();
|
|
|
|
hResultCode = DPN_OK;
|
|
|
|
DPFX(DPFPREP, 4,"Returning: [0x%lx]",hResultCode);
|
|
return(hResultCode);
|
|
}
|
|
|
|
|