2020-09-30 16:53:55 +02:00

822 lines
20 KiB
C++

/*==========================================================================
*
* Copyright (C) 1995 Microsoft Corporation. All Rights Reserved.
*
* File: NTEntry.cpp
* Content: NameTable Entry Objects
*@@BEGIN_MSINTERNAL
* History:
* Date By Reason
* ==== == ======
* 03/10/00 mjn Created
* 04/06/00 mjn Added AvailableEvent to block pre-ADD_PLAYER-notification sends
* 05/05/00 mjn Added GetConnectionRef()
* 05/16/00 mjn Better locking during User notifications
* 06/27/00 rmt Added COM abstraction
* 07/22/00 mjn Pack/Unpack DNET version in DN_NAMETABLE_ENTRY_INFO
* 07/26/00 mjn Fix PackInfo() to handle NULL names and data
* 08/03/00 rmt Bug #41386 - Getting player info when no name and/or user data returns garbage in
* name / data field.
* 09/06/00 mjn Changed SetAddress() to return void instead of HRESULT
* 09/13/00 mjn Added PerformQueuedOperations()
* 09/17/00 mjn Added NotifyAddRef() and NotifyRelease()
* 09/28/00 mjn Flag AutoDestruct groups in PackInfo()
* 10/11/00 mjn Don't take locks in PackInfo()
* 01/25/01 mjn Fixed 64-bit alignment problem when unpacking entries
* 04/19/01 mjn Lock entry when packing in PackEntryInfo()
* 07/24/01 mjn Added DPNBUILD_NOSERVER compile flag
*@@END_MSINTERNAL
*
***************************************************************************/
#include "dncorei.h"
//**********************************************************************
// Constant definitions
//**********************************************************************
//**********************************************************************
// Macro definitions
//**********************************************************************
//**********************************************************************
// Structure definitions
//**********************************************************************
//**********************************************************************
// Variable definitions
//**********************************************************************
//**********************************************************************
// Function prototypes
//**********************************************************************
//**********************************************************************
// Function definitions
//**********************************************************************
void CNameTableEntry::ReturnSelfToPool( void )
{
g_NameTableEntryPool.Release( this );
};
#undef DPF_MODNAME
#define DPF_MODNAME "CNameTableEntry::Release"
void CNameTableEntry::Release(void)
{
LONG lRefCount;
DNASSERT(m_lRefCount > 0);
lRefCount = DNInterlockedDecrement(const_cast<LONG*>(&m_lRefCount));
DPFX(DPFPREP, 3,"NameTableEntry::Release [0x%p] RefCount [0x%lx]",this,lRefCount);
if (lRefCount == 0)
{
DNASSERT(!(m_dwFlags & NAMETABLE_ENTRY_FLAG_AVAILABLE));
DNASSERT(m_bilinkDeleted.IsEmpty());
DNASSERT(m_bilinkMembership.IsEmpty());
DNASSERT(m_bilinkConnections.IsEmpty());
DNASSERT(m_bilinkQueuedMsgs.IsEmpty());
if (m_pAddress)
{
IDirectPlay8Address_Release(m_pAddress);
m_pAddress = NULL;
}
if (m_pConnection)
{
m_pConnection->Release();
m_pConnection = NULL;
}
if (m_pwszName)
{
DNFree(m_pwszName);
m_pwszName = NULL;
}
if (m_pvData)
{
DNFree(m_pvData);
m_pvData = NULL;
m_dwDataSize = 0;
}
m_dwFlags = 0;
m_lRefCount = 0;
ReturnSelfToPool();
}
}
#undef DPF_MODNAME
#define DPF_MODNAME "CNameTableEntry::NotifyAddRef"
void CNameTableEntry::NotifyAddRef( void )
{
LONG lRefCount;
lRefCount = DNInterlockedIncrement( const_cast<LONG*>(&m_lNotifyRefCount) );
DNASSERT( lRefCount >= 0 );
}
#undef DPF_MODNAME
#define DPF_MODNAME "CNameTableEntry::NotifyRelease"
void CNameTableEntry::NotifyRelease( void )
{
LONG lRefCount;
lRefCount = DNInterlockedDecrement( const_cast<LONG*>(&m_lNotifyRefCount) );
DNASSERT( lRefCount >= 0 );
if (lRefCount == 0)
{
Lock();
// DNASSERT(IsDisconnecting());
if (IsNeedToDestroy())
{
Unlock();
//
// Generate notifications
//
if (IsGroup())
{
if (!IsAllPlayersGroup())
{
DNUserDestroyGroup(m_pdnObject,this);
}
}
else
{
if (IsIndicated() && !IsCreated())
{
DNUserIndicatedConnectAborted(m_pdnObject,m_pvContext);
}
else
{
DNASSERT(IsCreated());
DNUserDestroyPlayer(m_pdnObject,this);
}
}
m_pdnObject->NameTable.WriteLock();
Lock();
m_bilinkDeleted.RemoveFromList();
m_pdnObject->NameTable.Unlock();
ClearNeedToDestroy();
ClearCreated();
}
Unlock();
}
}
#undef DPF_MODNAME
#define DPF_MODNAME "CNameTableEntry::UpdateEntryInfo"
HRESULT CNameTableEntry::UpdateEntryInfo(UNALIGNED WCHAR *const pwszName,
const DWORD dwNameSize,
void *const pvData,
const DWORD dwDataSize,
const DWORD dwInfoFlags,
BOOL fNotify)
{
PWSTR pwszTempName;
DWORD dwTempNameSize;
void *pvTempData;
DWORD dwTempDataSize;
Lock();
if (dwInfoFlags & DPNINFO_NAME)
{
if (pwszName && dwNameSize)
{
if ((pwszTempName = static_cast<WCHAR*>(DNMalloc(dwNameSize))) == NULL)
{
return(DPNERR_OUTOFMEMORY);
}
memcpy(pwszTempName,pwszName,dwNameSize);
dwTempNameSize = dwNameSize;
}
else
{
pwszTempName = NULL;
dwTempNameSize = 0;
}
if (m_pwszName)
{
DNFree(m_pwszName);
}
m_pwszName = pwszTempName;
m_dwNameSize = dwTempNameSize;
}
if (dwInfoFlags & DPNINFO_DATA)
{
if (pvData && dwDataSize)
{
if ((pvTempData = DNMalloc(dwDataSize)) == NULL)
{
return(DPNERR_OUTOFMEMORY);
}
memcpy(pvTempData,pvData,dwDataSize);
dwTempDataSize = dwDataSize;
}
else
{
pvTempData = NULL;
dwTempDataSize = 0;
}
if (m_pvData)
{
DNFree(m_pvData);
}
m_pvData = pvTempData;
m_dwDataSize = dwTempDataSize;
}
// Generate notifications
if (m_dwFlags & NAMETABLE_ENTRY_FLAG_AVAILABLE && fNotify)
{
DPNID dpnid = m_dpnid;
PVOID pvContext = m_pvContext;
DIRECTNETOBJECT* pdnObject = m_pdnObject;
if (m_dwFlags & NAMETABLE_ENTRY_FLAG_GROUP)
{
Unlock();
DNUserUpdateGroupInfo(pdnObject,dpnid,pvContext);
}
else
{
if (m_dwFlags & NAMETABLE_ENTRY_FLAG_PEER)
{
Unlock();
DNUserUpdatePeerInfo(pdnObject,dpnid,pvContext);
}
#ifndef DPNBUILD_NOSERVER
else if (m_dwFlags & NAMETABLE_ENTRY_FLAG_CLIENT && pdnObject->dwFlags & DN_OBJECT_FLAG_SERVER)
{
Unlock();
DNUserUpdateClientInfo(pdnObject,dpnid,pvContext);
}
#endif // DPNBUILD_NOSERVER
else if (m_dwFlags & NAMETABLE_ENTRY_FLAG_SERVER && pdnObject->dwFlags & DN_OBJECT_FLAG_CLIENT)
{
Unlock();
// Clients do not get to see server's DPNID or context
DNUserUpdateServerInfo(pdnObject,0,0);
}
else
{
Unlock();
DNASSERT(FALSE);
}
}
}
else
{
Unlock();
}
return(DPN_OK);
}
#undef DPF_MODNAME
#define DPF_MODNAME "CNameTableEntry::SetAddress"
void CNameTableEntry::SetAddress( IDirectPlay8Address *const pAddress )
{
if (pAddress)
{
IDirectPlay8Address_AddRef(pAddress);
}
if (m_pAddress)
{
IDirectPlay8Address_Release(m_pAddress);
m_pAddress = NULL;
}
m_pAddress = pAddress;
}
#undef DPF_MODNAME
#define DPF_MODNAME "CNameTableEntry::SetConnection"
void CNameTableEntry::SetConnection( CConnection *const pConnection )
{
if (pConnection)
{
pConnection->AddRef();
}
m_pConnection = pConnection;
}
#undef DPF_MODNAME
#define DPF_MODNAME "CNameTableEntry::GetConnectionRef"
HRESULT CNameTableEntry::GetConnectionRef( CConnection **const ppConnection )
{
HRESULT hResultCode;
DNASSERT( ppConnection != NULL);
Lock();
if ( m_pConnection && !m_pConnection->IsInvalid())
{
m_pConnection->AddRef();
*ppConnection = m_pConnection;
hResultCode = DPN_OK;
}
else
{
hResultCode = DPNERR_NOCONNECTION;
}
Unlock();
return( hResultCode );
}
#undef DPF_MODNAME
#define DPF_MODNAME "CNameTableEntry::PackInfo"
HRESULT CNameTableEntry::PackInfo(CPackedBuffer *const pPackedBuffer)
{
HRESULT hResultCode;
DPN_PLAYER_INFO *pPlayerInfo;
DPN_GROUP_INFO *pGroupInfo;
DNASSERT(pPackedBuffer != NULL);
// Lock();
if (m_dwFlags & NAMETABLE_ENTRY_FLAG_GROUP)
{
pGroupInfo = static_cast<DPN_GROUP_INFO*>(pPackedBuffer->GetHeadAddress());
hResultCode = pPackedBuffer->AddToFront(NULL,sizeof(DPN_GROUP_INFO));
//
// Add data
//
if ((m_pvData) && (m_dwDataSize != 0))
{
if ((hResultCode = pPackedBuffer->AddToBack(m_pvData,m_dwDataSize)) == DPN_OK)
{
pGroupInfo->pvData = pPackedBuffer->GetTailAddress();
pGroupInfo->dwDataSize = m_dwDataSize;
}
}
else
{
if (pGroupInfo)
{
pGroupInfo->pvData = NULL;
pGroupInfo->dwDataSize = 0;
}
}
//
// Add name
//
if ((m_pwszName) && (m_dwNameSize != 0))
{
if ((hResultCode = pPackedBuffer->AddToBack(m_pwszName,m_dwNameSize)) == DPN_OK)
{
pGroupInfo->pwszName = static_cast<WCHAR*>(pPackedBuffer->GetTailAddress());
}
}
else
{
if (pGroupInfo)
{
pGroupInfo->pwszName = NULL;
}
}
//
// Update flags
//
if (hResultCode == DPN_OK)
{
if (pGroupInfo)
{
pGroupInfo->dwSize = sizeof(DPN_GROUP_INFO);
pGroupInfo->dwInfoFlags = DPNINFO_NAME | DPNINFO_DATA;
pGroupInfo->dwGroupFlags = 0;
if (IsAutoDestructGroup())
{
pGroupInfo->dwGroupFlags |= DPNGROUP_AUTODESTRUCT;
}
}
}
}
else
{
pPlayerInfo = static_cast<DPN_PLAYER_INFO*>(pPackedBuffer->GetHeadAddress());
hResultCode = pPackedBuffer->AddToFront(NULL,sizeof(DPN_PLAYER_INFO));
if( !m_dwDataSize )
{
if( pPlayerInfo )
{
pPlayerInfo->pvData = NULL;
pPlayerInfo->dwDataSize = 0;
}
}
else
{
if ((hResultCode = pPackedBuffer->AddToBack(m_pvData,m_dwDataSize)) == DPN_OK)
{
pPlayerInfo->pvData = pPackedBuffer->GetTailAddress();
pPlayerInfo->dwDataSize = m_dwDataSize;
}
}
if( !m_pwszName )
{
if( pPlayerInfo )
{
pPlayerInfo->pwszName = NULL;
}
}
else
{
if ((hResultCode = pPackedBuffer->AddToBack(m_pwszName,m_dwNameSize)) == DPN_OK)
{
pPlayerInfo->pwszName = static_cast<WCHAR*>(pPackedBuffer->GetTailAddress());
}
}
if (hResultCode == DPN_OK)
{
pPlayerInfo->dwSize = sizeof(DPN_PLAYER_INFO);
pPlayerInfo->dwInfoFlags = DPNINFO_NAME | DPNINFO_DATA;
pPlayerInfo->dwPlayerFlags = 0;
if (m_dwFlags & NAMETABLE_ENTRY_FLAG_HOST)
{
pPlayerInfo->dwPlayerFlags |= DPNPLAYER_HOST;
}
if (m_dwFlags & NAMETABLE_ENTRY_FLAG_LOCAL)
{
pPlayerInfo->dwPlayerFlags |= DPNPLAYER_LOCAL;
}
}
}
// Unlock();
return(hResultCode);
}
#undef DPF_MODNAME
#define DPF_MODNAME "CNameTableEntry::PackEntryInfo"
HRESULT CNameTableEntry::PackEntryInfo(CPackedBuffer *const pPackedBuffer)
{
DWORD dwURLSize;
HRESULT hResultCode;
DN_NAMETABLE_ENTRY_INFO dnEntryInfo;
DPFX(DPFPREP, 6,"Attempting to pack [0x%lx]",m_dpnid);
DNASSERT(pPackedBuffer != NULL);
Lock();
dnEntryInfo.dpnid = m_dpnid;
dnEntryInfo.dpnidOwner = m_dpnidOwner;
dnEntryInfo.dwFlags = m_dwFlags & ( NAMETABLE_ENTRY_FLAG_HOST
| NAMETABLE_ENTRY_FLAG_ALL_PLAYERS_GROUP
| NAMETABLE_ENTRY_FLAG_GROUP
| NAMETABLE_ENTRY_FLAG_GROUP_AUTODESTRUCT
| NAMETABLE_ENTRY_FLAG_PEER
| NAMETABLE_ENTRY_FLAG_CLIENT
| NAMETABLE_ENTRY_FLAG_SERVER );
dnEntryInfo.dwVersion = m_dwVersion;
dnEntryInfo.dwVersionNotUsed = m_dwVersionNotUsed;
dnEntryInfo.dwDNETVersion = m_dwDNETVersion;
// Entry name
if (m_pwszName != NULL)
{
if ((hResultCode = pPackedBuffer->AddToBack(m_pwszName,m_dwNameSize)) == DPN_OK)
{
dnEntryInfo.dwNameOffset = pPackedBuffer->GetTailOffset();
dnEntryInfo.dwNameSize = m_dwNameSize;
}
}
else
{
dnEntryInfo.dwNameOffset = 0;
dnEntryInfo.dwNameSize = 0;
}
// Entry data
if (m_pvData != NULL && m_dwDataSize != 0)
{
if ((hResultCode = pPackedBuffer->AddToBack(m_pvData,m_dwDataSize)) == DPN_OK)
{
dnEntryInfo.dwDataOffset = pPackedBuffer->GetTailOffset();
dnEntryInfo.dwDataSize = m_dwDataSize;
}
}
else
{
dnEntryInfo.dwDataOffset = 0;
dnEntryInfo.dwDataSize = 0;
}
// Entry address (URL)
if ((m_pdnObject->dwFlags & DN_OBJECT_FLAG_PEER) && (m_pAddress != NULL))
{
dwURLSize = 0;
hResultCode = IDirectPlay8Address_GetURLA(m_pAddress,NULL,&dwURLSize);
if (hResultCode != DPN_OK && hResultCode != DPNERR_BUFFERTOOSMALL)
{
DPFERR("Could not determine URL size");
DisplayDNError(0,hResultCode);
Unlock();
goto EXIT_PackEntry;
}
if (dwURLSize != 0)
{
if ((hResultCode = pPackedBuffer->AddToBack(NULL,dwURLSize)) == DPN_OK)
{
if ((hResultCode = IDirectPlay8Address_GetURLA(m_pAddress,
static_cast<char*>(pPackedBuffer->GetTailAddress()),&dwURLSize)) == DPN_OK)
{
dnEntryInfo.dwURLOffset = pPackedBuffer->GetTailOffset();
dnEntryInfo.dwURLSize = dwURLSize;
}
else
{
DPFERR("Could not extract URL from DirectPlayAddress");
DisplayDNError(0,hResultCode);
Unlock();
goto EXIT_PackEntry;
}
}
}
else
{
dnEntryInfo.dwURLOffset = 0;
dnEntryInfo.dwURLSize = 0;
}
}
else
{
dnEntryInfo.dwURLOffset = 0;
dnEntryInfo.dwURLSize = 0;
}
hResultCode = pPackedBuffer->AddToFront(&dnEntryInfo,sizeof(DN_NAMETABLE_ENTRY_INFO));
Unlock();
EXIT_PackEntry:
DPFX(DPFPREP, 6,"Returning: [0x%lx]",hResultCode);
return(hResultCode);
}
#undef DPF_MODNAME
#define DPF_MODNAME "CNameTableEntry::UnpackEntryInfo"
HRESULT CNameTableEntry::UnpackEntryInfo(UNALIGNED const DN_NAMETABLE_ENTRY_INFO *const pdnEntryInfo,
BYTE *const pBufferStart)
{
HRESULT hResultCode;
PWSTR pwszName;
DWORD dwNameSize;
void *pvData;
DWORD dwDataSize;
IDirectPlay8Address *pAddress;
DNASSERT(m_pwszName == NULL);
DNASSERT(m_pvData == NULL);
DNASSERT(m_pAddress == NULL);
if (pdnEntryInfo->dwNameOffset && pdnEntryInfo->dwNameSize)
{
pwszName = reinterpret_cast<WCHAR*>(pBufferStart + pdnEntryInfo->dwNameOffset);
dwNameSize = pdnEntryInfo->dwNameSize;
}
else
{
pwszName = NULL;
dwNameSize = 0;
}
if (pdnEntryInfo->dwDataOffset && pdnEntryInfo->dwDataSize)
{
pvData = static_cast<void*>(pBufferStart + pdnEntryInfo->dwDataOffset);
dwDataSize = pdnEntryInfo->dwDataSize;
}
else
{
pvData = NULL;
dwDataSize = 0;
}
// This function takes the lock internally
UpdateEntryInfo(pwszName,dwNameSize,pvData,dwDataSize,DPNINFO_NAME|DPNINFO_DATA, FALSE);
pAddress = NULL;
if (pdnEntryInfo->dwURLOffset)
{
#ifdef DPNBUILD_LIBINTERFACE
hResultCode = DP8ACF_CreateInstance(IID_IDirectPlay8Address,
reinterpret_cast<void**>(&pAddress));
#else // ! DPNBUILD_LIBINTERFACE
hResultCode = COM_CoCreateInstance(CLSID_DirectPlay8Address,
NULL,
CLSCTX_INPROC_SERVER,
IID_IDirectPlay8Address,
reinterpret_cast<void**>(&pAddress),
FALSE);
#endif // ! DPNBUILD_LIBINTERFACE
if (hResultCode != S_OK)
{
DPFERR("Could not create empty DirectPlayAddress");
DisplayDNError(0,hResultCode);
return(DPNERR_OUTOFMEMORY);
}
hResultCode = IDirectPlay8Address_BuildFromURLA(pAddress,reinterpret_cast<char*>(pBufferStart + pdnEntryInfo->dwURLOffset));
if (hResultCode != DPN_OK)
{
DPFERR("Could not build URL");
DisplayDNError(0,hResultCode);
DNASSERT(FALSE);
IDirectPlay8Address_Release(pAddress);
pAddress = NULL;
return(hResultCode);
}
SetAddress(pAddress);
IDirectPlay8Address_Release(pAddress);
pAddress = NULL;
}
m_dpnid = pdnEntryInfo->dpnid;
m_dpnidOwner = pdnEntryInfo->dpnidOwner;
m_dwFlags = pdnEntryInfo->dwFlags;
m_dwDNETVersion = pdnEntryInfo->dwDNETVersion;
m_dwVersion = pdnEntryInfo->dwVersion;
m_dwVersionNotUsed = pdnEntryInfo->dwVersionNotUsed;
return(DPN_OK);
}
#undef DPF_MODNAME
#define DPF_MODNAME "CNameTableEntry::PerformQueuedOperations"
void CNameTableEntry::PerformQueuedOperations( void )
{
HRESULT hResultCode;
CQueuedMsg *pQueuedMsg;
BOOL fDestroy;
DPFX(DPFPREP, 6,"Parameters: (none)");
fDestroy = FALSE;
Lock();
fDestroy = IsNeedToDestroy();
//
// This assumes that the InUse flag is set. We will clear it before returning.
//
#ifdef DBG
DNASSERT( IsInUse() );
if (!m_bilinkQueuedMsgs.IsEmpty())
{
DPFX(DPFPREP, 7, "Nametable entry 0x%p has %i queued messages.", this, m_lNumQueuedMsgs);
}
#endif // DBG
while (!m_bilinkQueuedMsgs.IsEmpty())
{
pQueuedMsg = CONTAINING_OBJECT(m_bilinkQueuedMsgs.GetNext(),CQueuedMsg,m_bilinkQueuedMsgs);
pQueuedMsg->m_bilinkQueuedMsgs.RemoveFromList();
DEBUG_ONLY(m_lNumQueuedMsgs--);
Unlock();
switch (pQueuedMsg->GetOpType())
{
case RECEIVE:
{
HRESULT hrProcess;
DNASSERT(pQueuedMsg->GetAsyncOp() != NULL);
DNASSERT(pQueuedMsg->GetAsyncOp()->GetHandle() != 0);
#ifndef DPNBUILD_NOVOICE
if (pQueuedMsg->IsVoiceMessage())
{
hrProcess = Voice_Receive( m_pdnObject,
GetDPNID(),
0,
pQueuedMsg->GetBuffer(),
pQueuedMsg->GetBufferSize());
NotifyRelease();
}
else
#endif // DPNBUILD_NOVOICE
{
hrProcess = DNUserReceive( m_pdnObject,
this,
pQueuedMsg->GetBuffer(),
pQueuedMsg->GetBufferSize(),
pQueuedMsg->GetAsyncOp()->GetHandle());
if (pQueuedMsg->GetCompletionOp() != 0)
{
//
// Send completion message
//
CConnection *pConnection;
pConnection = NULL;
if ((hResultCode = GetConnectionRef( &pConnection )) == DPN_OK)
{
hResultCode = DNSendUserProcessCompletion( m_pdnObject,
pConnection,
pQueuedMsg->GetCompletionOp());
}
pConnection->Release();
pConnection = NULL;
}
}
//
// See if we can return this buffer now
//
if (hrProcess == DPNERR_PENDING)
{
pQueuedMsg->GetAsyncOp()->Release();
pQueuedMsg->SetAsyncOp( NULL );
}
else
{
DNEnterCriticalSection(&m_pdnObject->csActiveList);
pQueuedMsg->GetAsyncOp()->m_bilinkActiveList.RemoveFromList();
DNLeaveCriticalSection(&m_pdnObject->csActiveList);
pQueuedMsg->GetAsyncOp()->Lock();
if (!pQueuedMsg->GetAsyncOp()->IsCancelled() && !pQueuedMsg->GetAsyncOp()->IsComplete())
{
pQueuedMsg->GetAsyncOp()->SetComplete();
pQueuedMsg->GetAsyncOp()->Unlock();
if (SUCCEEDED(m_pdnObject->HandleTable.Destroy( pQueuedMsg->GetAsyncOp()->GetHandle(), NULL )))
{
// Release the HandleTable reference
pQueuedMsg->GetAsyncOp()->Release();
}
}
else
{
pQueuedMsg->GetAsyncOp()->Unlock();
}
pQueuedMsg->GetAsyncOp()->Release();
pQueuedMsg->SetAsyncOp( NULL );
}
break;
}
default:
{
DPFERR("Invalid Queued Operation");
DNASSERT(FALSE);
break;
}
}
//
// Return this queued message
//
pQueuedMsg->ReturnSelfToPool();
pQueuedMsg = NULL;
Lock();
fDestroy = IsNeedToDestroy();
}
//
// No longer processing
//
ClearInUse();
Unlock();
DPFX(DPFPREP, 6,"Returning");
}