/*========================================================================== * * 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(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(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 (reinterpret_cast(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); }