668 lines
20 KiB
C++
668 lines
20 KiB
C++
/*==========================================================================
|
|
*
|
|
* Copyright (C) 1998-2002 Microsoft Corporation. All Rights Reserved.
|
|
*
|
|
* File: Send.cpp
|
|
* Content: This file contains code which implements the front end of the
|
|
* SendData API. It also contains code to Get and Release Message
|
|
* Descriptors (MSD) with the FPM package.
|
|
*
|
|
* History:
|
|
* Date By Reason
|
|
* ==== == ======
|
|
* 11/06/98 ejs Created
|
|
* 07/01/2000 masonb Assumed Ownership
|
|
*
|
|
****************************************************************************/
|
|
|
|
#include "dnproti.h"
|
|
|
|
|
|
/*
|
|
** Direct Net Protocol -- Send Data
|
|
**
|
|
** Data is always address to a PlayerID, which is represented internally
|
|
** by an End Point Descriptor (EPD).
|
|
**
|
|
** Data can be sent reliably or unreliably using the same API with the appropriate
|
|
** class of service flag set.
|
|
**
|
|
** Sends are never delivered directly to the SP because there will always be
|
|
** a possibility that the thread might block. So to guarentee immediate return
|
|
** we will always queue the packet and submit it on our dedicated sending thread.
|
|
*/
|
|
|
|
|
|
#if (DN_SENDFLAGS_SET_USER_FLAG - PACKET_COMMAND_USER_1)
|
|
This will not compile. Flags must be equal
|
|
#endif
|
|
#if (DN_SENDFLAGS_SET_USER_FLAG_TWO - PACKET_COMMAND_USER_2)
|
|
This will not compile. Flags must be equal
|
|
#endif
|
|
|
|
// locals
|
|
|
|
VOID SendDatagram(PMSD, PEPD);
|
|
VOID SendReliable(PMSD, PEPD);
|
|
|
|
#undef DPF_MODNAME
|
|
#define DPF_MODNAME "PROTOCOL"
|
|
|
|
/*
|
|
** Send Data
|
|
**
|
|
** This routine will initiate a data transfer with the specified endpoint. It will
|
|
** normally start the operation and then return immediately, returning a handle used to
|
|
** indicate completion of the operation at a later time.
|
|
*/
|
|
|
|
#undef DPF_MODNAME
|
|
#define DPF_MODNAME "DNPSendData"
|
|
|
|
HRESULT
|
|
DNPSendData(HANDLE hProtocolData, HANDLE hDestination, UINT uiBufferCount, PBUFFERDESC pBufferDesc, UINT uiTimeout, ULONG ulFlags, VOID* pvContext, HANDLE* phSendHandle)
|
|
{
|
|
HRESULT hr;
|
|
ProtocolData* pPData;
|
|
PEPD pEPD;
|
|
PMSD pMSD;
|
|
PFMD pFMD;
|
|
UINT i;
|
|
UINT Length = 0;
|
|
PSPD pSPD;
|
|
ULONG ulFrameFlags;
|
|
BYTE bCommand;
|
|
// Following variables are used for mapping buffers to frames
|
|
PBUFFERDESC FromBuffer, ToBuffer;
|
|
UINT TotalRemain, FromRemain, ToRemain, size;
|
|
PCHAR FromPtr;
|
|
#ifdef DBG
|
|
INT FromBufferCount;
|
|
#endif // DBG
|
|
// End of variables for mapping frames
|
|
#ifndef DPNBUILD_NOMULTICAST
|
|
BOOL fMulticastSend;
|
|
#endif // !DPNBUILD_NOMULTICAST
|
|
|
|
DPFX(DPFPREP,DPF_CALLIN_LVL, "Parameters: hProtocolData[%p], hDestination[%x], uiBufferCount[%x], pBufferDesc[%p], uiTimeout[%x], ulFlags[%x], pvContext[%p], phSendHandle[%p]", hProtocolData, hDestination, uiBufferCount, pBufferDesc, uiTimeout, ulFlags, pvContext, phSendHandle);
|
|
|
|
hr = DPNERR_PENDING;
|
|
pPData = (ProtocolData*)hProtocolData;
|
|
ASSERT_PPD(pPData);
|
|
|
|
pEPD = (PEPD) hDestination;
|
|
ASSERT_EPD(pEPD);
|
|
|
|
// Unified Send Processing -- Do this for all classes of service
|
|
|
|
// We will do all of the work to build up the frames and create the send command before we check
|
|
// the state of the EPD, that way we don't have to have complicated code to handle an endpoint that
|
|
// goes away between the top and bottom of this function and we don't have to hold the EPDLock while
|
|
// we do all of the buffer manipulation.
|
|
|
|
// Hold a reference throughout the operation so we don't have to deal with the EPD going away.
|
|
LOCK_EPD(pEPD, "LOCK (SEND)");
|
|
|
|
// Count the bytes in all user buffers
|
|
for(i=0; i < uiBufferCount; i++)
|
|
{
|
|
Length += pBufferDesc[i].dwBufferSize;
|
|
}
|
|
if (Length == 0)
|
|
{
|
|
DPFX(DPFPREP,0, "Attempt to send zero length packet, returning DPNERR_GENERIC");
|
|
return DPNERR_GENERIC;
|
|
}
|
|
|
|
#ifndef DPNBUILD_NOMULTICAST
|
|
fMulticastSend = pEPD->ulEPFlags2 & (EPFLAGS2_MULTICAST_SEND|EPFLAGS2_MULTICAST_RECEIVE);
|
|
if (fMulticastSend && Length > pEPD->uiUserFrameLength)
|
|
{
|
|
DPFX(DPFPREP,0, "Multicast send too large to fit in one frame, returning DPNERR_SENDTOOLARGE");
|
|
Lock(&pEPD->EPLock);
|
|
RELEASE_EPD(pEPD, "UNLOCK (SEND)");
|
|
return DPNERR_SENDTOOLARGE;
|
|
}
|
|
#endif // !DPNBUILD_NOMULTICAST
|
|
|
|
// Allocate and fill out a Message Descriptor for this operation
|
|
if((pMSD = (PMSD)POOLALLOC(MEMID_SEND_MSD, &MSDPool)) == NULL)
|
|
{
|
|
DPFX(DPFPREP,0, "Failed to allocate MSD, returning DPNERR_OUTOFMEMORY");
|
|
Lock(&pEPD->EPLock);
|
|
RELEASE_EPD(pEPD, "UNLOCK (SEND)");
|
|
hr = DPNERR_OUTOFMEMORY;
|
|
goto Exit;
|
|
}
|
|
|
|
// Copy SendData parameters into the Message Descriptor
|
|
pMSD->ulSendFlags = ulFlags; // Store the actual flags passed into the API call
|
|
pMSD->Context = pvContext;
|
|
pMSD->iMsgLength = Length;
|
|
|
|
pMSD->uiFrameCount = (Length + pEPD->uiUserFrameLength - 1) / pEPD->uiUserFrameLength; // round up
|
|
DPFX(DPFPREP, DPF_FRAMECNT_LVL, "Initialize Frame count, pMSD[%p], framecount[%u]", pMSD, pMSD->uiFrameCount);
|
|
|
|
#ifndef DPNBUILD_NOMULTICAST
|
|
ASSERT(!fMulticastSend || pMSD->uiFrameCount == 1);
|
|
#endif // !DPNBUILD_NOMULTICAST
|
|
|
|
if(ulFlags & DN_SENDFLAGS_RELIABLE)
|
|
{
|
|
#ifndef DPNBUILD_NOMULTICAST
|
|
ASSERT(!fMulticastSend);
|
|
#endif // !DPNBUILD_NOMULTICAST
|
|
pMSD->CommandID = COMMAND_ID_SEND_RELIABLE;
|
|
ulFrameFlags = FFLAGS_RELIABLE;
|
|
bCommand = PACKET_COMMAND_DATA | PACKET_COMMAND_RELIABLE;
|
|
}
|
|
else
|
|
{
|
|
pMSD->CommandID = COMMAND_ID_SEND_DATAGRAM;
|
|
ulFrameFlags = 0;
|
|
bCommand = PACKET_COMMAND_DATA;
|
|
}
|
|
|
|
if(!(ulFlags & DN_SENDFLAGS_COALESCE))
|
|
{
|
|
#ifdef DPNBUILD_COALESCEALWAYS
|
|
DPFX(DPFPREP,7, "(%p) Attempting to coalesce send despite missing flag.", pEPD);
|
|
#else // ! DPNBUILD_COALESCEALWAYS
|
|
ulFrameFlags |= FFLAGS_DONT_COALESCE;
|
|
#endif // ! DPNBUILD_COALESCEALWAYS
|
|
}
|
|
|
|
if(!(ulFlags & DN_SENDFLAGS_NON_SEQUENTIAL))
|
|
{
|
|
#ifndef DPNBUILD_NOMULTICAST
|
|
ASSERT(!fMulticastSend);
|
|
#endif // !DPNBUILD_NOMULTICAST
|
|
bCommand |= PACKET_COMMAND_SEQUENTIAL;
|
|
}
|
|
|
|
bCommand |= (ulFlags & (DN_SENDFLAGS_SET_USER_FLAG | DN_SENDFLAGS_SET_USER_FLAG_TWO)); // preserve user flag values
|
|
|
|
// Map user buffers directly into frame's buffer descriptors
|
|
//
|
|
// We will loop through each required frame, filling out buffer descriptors
|
|
// from those provided as parameters. Frames may span user buffers or vice-versa...
|
|
|
|
TotalRemain = Length;
|
|
#ifdef DBG
|
|
FromBufferCount = uiBufferCount - 1; // sanity check
|
|
#endif // DBG
|
|
FromBuffer = pBufferDesc;
|
|
FromRemain = FromBuffer->dwBufferSize;
|
|
FromPtr = reinterpret_cast<PCHAR>( (FromBuffer++)->pBufferData ); // note post-increment to next descriptor
|
|
|
|
for(i=0; i<pMSD->uiFrameCount; i++)
|
|
{
|
|
ASSERT(TotalRemain > 0);
|
|
|
|
// Grab a new frame
|
|
if((pFMD = (PFMD)POOLALLOC(MEMID_SEND_FMD, &FMDPool)) == NULL)
|
|
{
|
|
// MSD_Release will clean up any previous frames if this isn't the first.
|
|
// Release MSD before EPD since final EPD will call out to SP and we don't want any locks held
|
|
Lock(&pMSD->CommandLock);
|
|
pMSD->uiFrameCount = 0; // reset to prevent assert in pool release function
|
|
RELEASE_MSD(pMSD, "Base Ref"); // MSD Release operation will also free frames
|
|
Lock(&pEPD->EPLock);
|
|
RELEASE_EPD(pEPD, "UNLOCK (SEND)");
|
|
DPFX(DPFPREP,0, "Failed to allocate FMD, returning DPNERR_OUTOFMEMORY");
|
|
hr = DPNERR_OUTOFMEMORY;
|
|
goto Exit;
|
|
}
|
|
|
|
pFMD->pMSD = pMSD; // Link frame back to message
|
|
pFMD->pEPD = pEPD;
|
|
pFMD->CommandID = pMSD->CommandID;
|
|
pFMD->bPacketFlags = bCommand; // save packet flags for each frame
|
|
pFMD->blMSDLinkage.InsertBefore( &pMSD->blFrameList);
|
|
ToRemain = pEPD->uiUserFrameLength;
|
|
ToBuffer = pFMD->rgBufferList; // Address first user buffer desc
|
|
|
|
pFMD->uiFrameLength = pEPD->uiUserFrameLength; // Assume we fill frame- only need to change size of last one
|
|
pFMD->ulFFlags = ulFrameFlags; // Set control flags for frame (Sequential, Reliable)
|
|
|
|
// Until this frame is full
|
|
while((ToRemain != 0) && (TotalRemain != 0) && (pFMD->SendDataBlock.dwBufferCount <= MAX_USER_BUFFERS_IN_FRAME))
|
|
{
|
|
size = _MIN(FromRemain, ToRemain); // choose smaller of framesize or buffersize
|
|
FromRemain -= size;
|
|
ToRemain -= size;
|
|
TotalRemain -= size;
|
|
|
|
ToBuffer->dwBufferSize = size; // Fill in the next frame descriptor
|
|
(ToBuffer++)->pBufferData = reinterpret_cast<BYTE*>( FromPtr ); // note post-increment
|
|
ASSERT(pFMD->SendDataBlock.dwBufferCount <= MAX_USER_BUFFERS_IN_FRAME); // remember we already have 1 immediate data buffer
|
|
pFMD->SendDataBlock.dwBufferCount++; // Count buffers as we add them
|
|
|
|
// Get next user buffer
|
|
if((FromRemain == 0) && (TotalRemain != 0))
|
|
{
|
|
FromRemain = FromBuffer->dwBufferSize;
|
|
FromPtr = reinterpret_cast<PCHAR>( (FromBuffer++)->pBufferData ); // note post-increment to next descriptor
|
|
#ifdef DBG
|
|
FromBufferCount--; // Keep this code honest...
|
|
ASSERT(FromBufferCount >= 0);
|
|
#endif // DBG
|
|
}
|
|
else
|
|
{ // Either filled this frame, or have mapped the whole send
|
|
FromPtr += size; // advance ptr to start next frame (if any)
|
|
pFMD->uiFrameLength = pEPD->uiUserFrameLength - ToRemain; // wont be full at end of message
|
|
}
|
|
} // While (frame not full)
|
|
} // For (each frame in message)
|
|
|
|
pFMD->ulFFlags |= FFLAGS_END_OF_MESSAGE; // Mark last frame with EOM
|
|
pFMD->bPacketFlags |= PACKET_COMMAND_END_MSG; // Set EOM in frame
|
|
|
|
#ifdef DBG
|
|
ASSERT(FromBufferCount == 0);
|
|
ASSERT(TotalRemain == 0);
|
|
#endif // DBG
|
|
|
|
Lock(&pMSD->CommandLock);
|
|
Lock(&pEPD->EPLock);
|
|
|
|
// Don't allow sends if we are not connected or if a disconnect has been initiated
|
|
if( ((pEPD->ulEPFlags & (EPFLAGS_END_POINT_IN_USE | EPFLAGS_STATE_CONNECTED)) !=
|
|
(EPFLAGS_END_POINT_IN_USE | EPFLAGS_STATE_CONNECTED))
|
|
|| (pEPD->ulEPFlags & (EPFLAGS_SENT_DISCONNECT | EPFLAGS_HARD_DISCONNECT_SOURCE)))
|
|
{
|
|
// Release MSD before EPD since final EPD will call out to SP and we don't want any locks held
|
|
pMSD->uiFrameCount = 0;
|
|
RELEASE_MSD(pMSD, "Base Ref"); // MSD Release operation will also free frames, releases CommandLock
|
|
RELEASE_EPD(pEPD, "UNLOCK (SEND)"); // Releases EPLock
|
|
|
|
DPFX(DPFPREP,0, "(%p) Rejecting Send on invalid EPD, returning DPNERR_INVALIDENDPOINT", pEPD);
|
|
hr = DPNERR_INVALIDENDPOINT;
|
|
goto Exit;
|
|
}
|
|
|
|
pSPD = pEPD->pSPD;
|
|
ASSERT_SPD(pSPD);
|
|
|
|
pMSD->pSPD = pSPD;
|
|
pMSD->pEPD = pEPD;
|
|
|
|
// hang the message off a global command queue
|
|
|
|
#ifdef DBG
|
|
Lock(&pSPD->SPLock);
|
|
pMSD->blSPLinkage.InsertBefore( &pSPD->blMessageList);
|
|
pMSD->ulMsgFlags1 |= MFLAGS_ONE_ON_GLOBAL_LIST;
|
|
Unlock(&pSPD->SPLock);
|
|
#endif // DBG
|
|
|
|
*phSendHandle = pMSD; // We will use the MSD as our handle.
|
|
|
|
// Enqueue the message before setting the timeout
|
|
EnqueueMessage(pMSD, pEPD);
|
|
Unlock(&pEPD->EPLock);
|
|
|
|
if(uiTimeout != 0)
|
|
{
|
|
LOCK_MSD(pMSD, "Send Timeout Timer"); // Add reference for timer
|
|
DPFX(DPFPREP,7, "(%p) Setting Timeout Send Timer", pEPD);
|
|
ScheduleProtocolTimer(pSPD, uiTimeout, 100, TimeoutSend, pMSD, &pMSD->TimeoutTimer, &pMSD->TimeoutTimerUnique);
|
|
}
|
|
|
|
Unlock(&pMSD->CommandLock);
|
|
|
|
Exit:
|
|
AssertNoCriticalSectionsFromGroupTakenByThisThread(&g_blProtocolCritSecsHeld);
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
** Enqueue Message
|
|
**
|
|
** Add complete MSD to the appropriate send queue, and kick start sending process if necessary.
|
|
**
|
|
** ** This routine is called and returns with EPD->EPLOCK held **
|
|
*/
|
|
|
|
#undef DPF_MODNAME
|
|
#define DPF_MODNAME "EnqueueMessage"
|
|
|
|
VOID
|
|
EnqueueMessage(PMSD pMSD, PEPD pEPD)
|
|
{
|
|
PSPD pSPD = pEPD->pSPD;
|
|
|
|
// Place Message in appriopriate priority queue. Datagrams get enqueued twice (!). They get put in the Master
|
|
// queue where they are processed FIFO with all messages of the same priority. Datagrams also get placed in a priority
|
|
// specific queue of only datagrams which is drawn from when the reliable stream is blocked.
|
|
|
|
AssertCriticalSectionIsTakenByThisThread(&pEPD->EPLock, TRUE);
|
|
|
|
pEPD->uiQueuedMessageCount++;
|
|
if(pMSD->ulSendFlags & DN_SENDFLAGS_HIGH_PRIORITY)
|
|
{
|
|
DPFX(DPFPREP,7, "(%p) Placing message on High Priority Q (total queued = %u)", pEPD, pEPD->uiQueuedMessageCount);
|
|
pMSD->blQLinkage.InsertBefore( &pEPD->blHighPriSendQ);
|
|
pEPD->uiMsgSentHigh++;
|
|
}
|
|
else if (pMSD->ulSendFlags & DN_SENDFLAGS_LOW_PRIORITY)
|
|
{
|
|
DPFX(DPFPREP,7, "(%p) Placing message on Low Priority Q (total queued = %u)", pEPD, pEPD->uiQueuedMessageCount);
|
|
pMSD->blQLinkage.InsertBefore( &pEPD->blLowPriSendQ);
|
|
pEPD->uiMsgSentLow++;
|
|
}
|
|
else
|
|
{
|
|
DPFX(DPFPREP,7, "(%p) Placing message on Normal Priority Q (total queued = %u)", pEPD, pEPD->uiQueuedMessageCount);
|
|
pMSD->blQLinkage.InsertBefore( &pEPD->blNormPriSendQ);
|
|
pEPD->uiMsgSentNorm++;
|
|
}
|
|
|
|
#ifdef DBG
|
|
pMSD->ulMsgFlags2 |= MFLAGS_TWO_ENQUEUED;
|
|
#endif // DBG
|
|
|
|
pEPD->ulEPFlags |= EPFLAGS_SDATA_READY; // Note that there is *something* in one or more queues
|
|
|
|
// If the session is not currently in the send pipeline then we will want to insert it here as long as the
|
|
// the stream is not blocked.
|
|
|
|
if(((pEPD->ulEPFlags & EPFLAGS_IN_PIPELINE)==0) && (pEPD->ulEPFlags & EPFLAGS_STREAM_UNBLOCKED))
|
|
{
|
|
ASSERT(pEPD->SendTimer == NULL);
|
|
DPFX(DPFPREP,7, "(%p) Send On Idle Link -- Returning to pipeline", pEPD);
|
|
|
|
pEPD->ulEPFlags |= EPFLAGS_IN_PIPELINE;
|
|
LOCK_EPD(pEPD, "LOCK (pipeline)"); // Add Ref for pipeline Q
|
|
|
|
// We dont call send on users thread, but we dont have a dedicated send thread either. Use a thread
|
|
// from the timer-worker pool to submit the sends to SP
|
|
|
|
DPFX(DPFPREP,7, "(%p) Scheduling Send Thread", pEPD);
|
|
ScheduleProtocolWork(pSPD, ScheduledSend, pEPD);
|
|
}
|
|
else if ((pEPD->ulEPFlags & EPFLAGS_IN_PIPELINE)==0)
|
|
{
|
|
DPFX(DPFPREP,7, "(%p) Declining to re-enter pipeline on blocked stream", pEPD);
|
|
}
|
|
else
|
|
{
|
|
DPFX(DPFPREP,7, "(%p) Already in pipeline", pEPD);
|
|
}
|
|
}
|
|
|
|
|
|
#undef DPF_MODNAME
|
|
#define DPF_MODNAME "TimeoutSend"
|
|
|
|
VOID CALLBACK
|
|
TimeoutSend(void * const pvUser, void * const uID, const UINT uMsg)
|
|
{
|
|
PMSD pMSD = (PMSD) pvUser;
|
|
PEPD pEPD = pMSD->pEPD;
|
|
|
|
DPFX(DPFPREP,7, "(%p) Timeout Send pMSD=%p, RefCnt=%d", pEPD, pMSD, pMSD->lRefCnt);
|
|
|
|
Lock(&pMSD->CommandLock);
|
|
|
|
if((pMSD->TimeoutTimer != uID)||(pMSD->TimeoutTimerUnique != uMsg))
|
|
{
|
|
DPFX(DPFPREP,7, "(%p) Ignoring late send timeout timer, pMSD[%p]", pEPD, pMSD);
|
|
RELEASE_MSD(pMSD, "Timeout Timer"); // releases EPLock
|
|
return;
|
|
}
|
|
|
|
pMSD->TimeoutTimer = NULL;
|
|
|
|
if(pMSD->ulMsgFlags1 & (MFLAGS_ONE_CANCELLED | MFLAGS_ONE_TIMEDOUT))
|
|
{
|
|
DPFX(DPFPREP,7, "(%p) Timed out send has completed already pMSD=%p", pEPD, pMSD);
|
|
RELEASE_MSD(pMSD, "Send Timout Timer"); // Releases CommandLock
|
|
return;
|
|
}
|
|
|
|
pMSD->ulMsgFlags1 |= MFLAGS_ONE_TIMEDOUT;
|
|
|
|
DPFX(DPFPREP,7, "(%p) Calling DoCancel to cancel pMSD=%p", pEPD, pMSD);
|
|
|
|
if(DoCancel(pMSD, DPNERR_TIMEDOUT) == DPN_OK) // Releases CommandLock
|
|
{
|
|
ASSERT_EPD(pEPD);
|
|
|
|
if(pMSD->ulSendFlags & DN_SENDFLAGS_HIGH_PRIORITY)
|
|
{
|
|
pEPD->uiMsgTOHigh++;
|
|
}
|
|
else if(pMSD->ulSendFlags & DN_SENDFLAGS_LOW_PRIORITY)
|
|
{
|
|
pEPD->uiMsgTOLow++;
|
|
}
|
|
else
|
|
{
|
|
pEPD->uiMsgTONorm++;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
DPFX(DPFPREP,7, "(%p) DoCancel did not succeed pMSD=%p", pEPD, pMSD);
|
|
}
|
|
|
|
Lock(&pMSD->CommandLock);
|
|
RELEASE_MSD(pMSD, "Send Timout Timer"); // Release Ref for timer
|
|
}
|
|
|
|
|
|
/***********************
|
|
========SPACER==========
|
|
************************/
|
|
|
|
/*
|
|
** MSD Pool support routines
|
|
**
|
|
** These are the functions called by Fixed Pool Manager as it handles MSDs.
|
|
*/
|
|
|
|
#define pELEMENT ((PMSD) pElement)
|
|
|
|
#undef DPF_MODNAME
|
|
#define DPF_MODNAME "MSD_Allocate"
|
|
|
|
BOOL MSD_Allocate(PVOID pElement, PVOID pvContext)
|
|
{
|
|
DPFX(DPFPREP,7, "(%p) Allocating new MSD", pELEMENT);
|
|
|
|
ZeroMemory(pELEMENT, sizeof(messagedesc));
|
|
|
|
if (DNInitializeCriticalSection(&pELEMENT->CommandLock) == FALSE)
|
|
{
|
|
DPFX(DPFPREP,0, "Failed to initialize MSD CS");
|
|
return FALSE;
|
|
}
|
|
DebugSetCriticalSectionRecursionCount(&pELEMENT->CommandLock,0);
|
|
DebugSetCriticalSectionGroup(&pELEMENT->CommandLock, &g_blProtocolCritSecsHeld);
|
|
|
|
pELEMENT->blFrameList.Initialize();
|
|
pELEMENT->blQLinkage.Initialize();
|
|
pELEMENT->blSPLinkage.Initialize();
|
|
pELEMENT->Sign = MSD_SIGN;
|
|
pELEMENT->lRefCnt = -1;
|
|
|
|
// NOTE: pELEMENT->pEPD NULL'd by ZeroMemory above
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
// Get is called each time an MSD is used
|
|
|
|
|
|
#undef DPF_MODNAME
|
|
#define DPF_MODNAME "MSD_Get"
|
|
|
|
VOID MSD_Get(PVOID pElement, PVOID pvContext)
|
|
{
|
|
DPFX(DPFPREP,DPF_REFCNT_FINAL_LVL, "CREATING MSD %p", pELEMENT);
|
|
|
|
// NOTE: First sizeof(PVOID) bytes will have been overwritten by the pool code,
|
|
// we must set them to acceptable values.
|
|
|
|
pELEMENT->CommandID = COMMAND_ID_NONE;
|
|
pELEMENT->ulMsgFlags1 = MFLAGS_ONE_IN_USE; // Dont need InUse flag since we have RefCnt
|
|
pELEMENT->lRefCnt = 0; // One initial reference
|
|
pELEMENT->hCommand = 0;
|
|
|
|
ASSERT_MSD(pELEMENT);
|
|
}
|
|
|
|
/*
|
|
** MSD Release
|
|
**
|
|
** This is called with the CommandLock held. The Lock should not be
|
|
** freed until the INUSE flag is cleared. This is to synchronize with
|
|
** last minute Cancel threads waiting on lock.
|
|
**
|
|
** When freeing a message desc we will free all frame descriptors
|
|
** attached to it first.
|
|
*/
|
|
|
|
#undef DPF_MODNAME
|
|
#define DPF_MODNAME "MSD_Release"
|
|
|
|
VOID MSD_Release(PVOID pElement)
|
|
{
|
|
CBilink *pLink;
|
|
PFMD pFMD;
|
|
|
|
#ifdef DBG
|
|
ASSERT_MSD(pELEMENT);
|
|
|
|
AssertCriticalSectionIsTakenByThisThread(&pELEMENT->CommandLock, TRUE);
|
|
|
|
DPFX(DPFPREP,DPF_REFCNT_FINAL_LVL, "RELEASING MSD %p", pELEMENT);
|
|
|
|
ASSERT(pELEMENT->ulMsgFlags1 & MFLAGS_ONE_IN_USE);
|
|
ASSERT(pELEMENT->lRefCnt == -1);
|
|
ASSERT((pELEMENT->ulMsgFlags1 & MFLAGS_ONE_ON_GLOBAL_LIST)==0);
|
|
#endif // DBG
|
|
|
|
while( (pLink = pELEMENT->blFrameList.GetNext()) != &pELEMENT->blFrameList)
|
|
{
|
|
pLink->RemoveFromList(); // remove from bilink
|
|
|
|
pFMD = CONTAINING_OBJECT(pLink, FMD, blMSDLinkage);
|
|
ASSERT_FMD(pFMD);
|
|
RELEASE_FMD(pFMD, "MSD Frame List"); // If this is still submitted it will be referenced and wont be released here.
|
|
}
|
|
|
|
ASSERT(pELEMENT->blFrameList.IsEmpty());
|
|
ASSERT(pELEMENT->blQLinkage.IsEmpty());
|
|
ASSERT(pELEMENT->blSPLinkage.IsEmpty());
|
|
|
|
ASSERT(pELEMENT->uiFrameCount == 0);
|
|
|
|
pELEMENT->ulMsgFlags1 = 0;
|
|
pELEMENT->ulMsgFlags2 = 0;
|
|
|
|
ASSERT(pELEMENT->pEPD == NULL); // This should have gotten cleaned up before here.
|
|
|
|
Unlock(&pELEMENT->CommandLock);
|
|
}
|
|
|
|
#undef DPF_MODNAME
|
|
#define DPF_MODNAME "MSD_Free"
|
|
|
|
VOID MSD_Free(PVOID pElement)
|
|
{
|
|
DNDeleteCriticalSection(&pELEMENT->CommandLock);
|
|
}
|
|
|
|
#undef pELEMENT
|
|
|
|
/*
|
|
** FMD Pool support routines
|
|
*/
|
|
|
|
#define pELEMENT ((PFMD) pElement)
|
|
|
|
#undef DPF_MODNAME
|
|
#define DPF_MODNAME "FMD_Allocate"
|
|
|
|
BOOL FMD_Allocate(PVOID pElement, PVOID pvContext)
|
|
{
|
|
DPFX(DPFPREP,7, "(%p) Allocating new FMD", pELEMENT);
|
|
|
|
pELEMENT->Sign = FMD_SIGN;
|
|
pELEMENT->ulFFlags = 0;
|
|
pELEMENT->lRefCnt = 0;
|
|
|
|
pELEMENT->blMSDLinkage.Initialize();
|
|
pELEMENT->blQLinkage.Initialize();
|
|
pELEMENT->blWindowLinkage.Initialize();
|
|
pELEMENT->blCoalesceLinkage.Initialize();
|
|
pELEMENT->pCSD = NULL;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
// Get is called each time an MSD is used
|
|
//
|
|
// Probably dont need to do this everytime, but some random SP might
|
|
// munch the parameters someday and that could be bad if I dont...
|
|
|
|
#undef DPF_MODNAME
|
|
#define DPF_MODNAME "FMD_Get"
|
|
|
|
VOID FMD_Get(PVOID pElement, PVOID pvContext)
|
|
{
|
|
DPFX(DPFPREP,DPF_REFCNT_FINAL_LVL, "CREATING FMD %p", pELEMENT);
|
|
|
|
pELEMENT->CommandID = COMMAND_ID_NONE;
|
|
pELEMENT->lpImmediatePointer = (LPVOID) pELEMENT->ImmediateData;
|
|
pELEMENT->SendDataBlock.pBuffers = (PBUFFERDESC) &pELEMENT->uiImmediateLength;
|
|
pELEMENT->SendDataBlock.dwBufferCount = 1; // always count one buffer for immediate data
|
|
pELEMENT->SendDataBlock.dwFlags = 0;
|
|
pELEMENT->SendDataBlock.pvContext = pElement;
|
|
pELEMENT->SendDataBlock.hCommand = 0;
|
|
pELEMENT->ulFFlags = 0;
|
|
pELEMENT->bSubmitted = FALSE;
|
|
pELEMENT->bPacketFlags = 0;
|
|
pELEMENT->tAcked = -1;
|
|
|
|
pELEMENT->lRefCnt = 1; // Assign first reference
|
|
|
|
ASSERT_FMD(pELEMENT);
|
|
}
|
|
|
|
#undef DPF_MODNAME
|
|
#define DPF_MODNAME "FMD_Release"
|
|
|
|
VOID FMD_Release(PVOID pElement)
|
|
{
|
|
DPFX(DPFPREP,DPF_REFCNT_FINAL_LVL, "RELEASING FMD %p", pELEMENT);
|
|
|
|
ASSERT_FMD(pELEMENT);
|
|
ASSERT(pELEMENT->lRefCnt == 0);
|
|
ASSERT(pELEMENT->bSubmitted == FALSE);
|
|
pELEMENT->pMSD = NULL;
|
|
|
|
ASSERT(pELEMENT->blMSDLinkage.IsEmpty());
|
|
ASSERT(pELEMENT->blQLinkage.IsEmpty());
|
|
ASSERT(pELEMENT->blWindowLinkage.IsEmpty());
|
|
ASSERT(pELEMENT->blCoalesceLinkage.IsEmpty());
|
|
ASSERT(pELEMENT->pCSD == NULL);
|
|
}
|
|
|
|
#undef DPF_MODNAME
|
|
#define DPF_MODNAME "FMD_Free"
|
|
|
|
VOID FMD_Free(PVOID pElement)
|
|
{
|
|
}
|
|
|
|
#undef pELEMENT
|
|
|
|
|