windows-nt/Source/XPSP1/NT/net/sockets/winsock2/wsp/afdsys/buffer.c
2020-09-26 16:20:57 +08:00

1310 lines
39 KiB
C
Raw Permalink Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*++
Copyright (c) 1989 Microsoft Corporation
Module Name:
buffer.c
Abstract:
This module contains routines for handling non-bufferring TDI
providers. The AFD interface assumes that bufferring will be done
below AFD; if the TDI provider doesn't buffer, then AFD must.
Author:
David Treadwell (davidtr) 21-Feb-1992
Revision History:
--*/
#include "afdp.h"
PAFD_BUFFER
AfdInitializeBuffer (
IN PVOID MemBlock,
IN ULONG BufferDataSize,
IN ULONG AddressSize
);
#if DBG
VOID
AfdFreeBufferReal (
PVOID AfdBuffer
);
#endif
#ifdef ALLOC_PRAGMA
#pragma alloc_text( PAGEAFD, AfdAllocateBuffer )
#pragma alloc_text( PAGEAFD, AfdFreeBuffer )
#if DBG
#pragma alloc_text( PAGEAFD, AfdFreeBufferReal )
#endif
#pragma alloc_text( PAGEAFD, AfdCalculateBufferSize )
#pragma alloc_text( PAGEAFD, AfdInitializeBuffer )
#pragma alloc_text( PAGEAFD, AfdGetBuffer )
#pragma alloc_text( PAGEAFD, AfdReturnBuffer )
#pragma alloc_text( PAGEAFD, AfdAllocateBufferTag )
#pragma alloc_text( PAGEAFD, AfdFreeBufferTag )
#pragma alloc_text( PAGEAFD, AfdAllocateRemoteAddress )
#pragma alloc_text( PAGEAFD, AfdFreeRemoteAddress )
#pragma alloc_text( PAGEAFD, AfdInitializeBufferTag )
#pragma alloc_text( PAGEAFD, AfdGetBufferTag )
#pragma alloc_text( INIT, AfdInitializeBufferManager)
#endif
PVOID
AfdAllocateBuffer (
IN POOL_TYPE PoolType,
IN SIZE_T NumberOfBytes,
IN ULONG Tag
)
/*++
Routine Description:
Used by the lookaside list allocation function to allocate a new
AFD buffer structure. The returned structure will be fully
initialized.
Arguments:
PoolType - passed to ExAllocatePoolWithTag.
NumberOfBytes - the number of bytes required for the data buffer
portion of the AFD buffer.
Tag - passed to ExAllocatePoolWithTag.
Return Value:
PVOID - a fully initialized PAFD_BUFFER, or NULL if the allocation
attempt fails.
--*/
{
ULONG dataLength;
PVOID memBlock;
//
// Get nonpaged pool for the buffer.
//
memBlock = AFD_ALLOCATE_POOL( PoolType, NumberOfBytes, Tag );
if ( memBlock == NULL ) {
return NULL;
}
if (NumberOfBytes==AfdLookasideLists->SmallBufferList.L.Size) {
dataLength = AfdSmallBufferSize;
}
else if (NumberOfBytes==AfdLookasideLists->MediumBufferList.L.Size) {
dataLength = AfdMediumBufferSize;
}
else if (NumberOfBytes==AfdLookasideLists->LargeBufferList.L.Size) {
dataLength = AfdLargeBufferSize;
}
else {
ASSERT (FALSE);
}
//
// Initialize the buffer and return a pointer to it.
//
#if DBG
{
PAFD_BUFFER afdBuffer = AfdInitializeBuffer( memBlock, dataLength, AfdStandardAddressLength );
ASSERT ((PCHAR)afdBuffer+sizeof (AFD_BUFFER)<=(PCHAR)memBlock+NumberOfBytes &&
(PCHAR)afdBuffer->Buffer+dataLength<=(PCHAR)memBlock+NumberOfBytes &&
(PCHAR)afdBuffer->Irp+IoSizeOfIrp(AfdIrpStackSize-1)<=(PCHAR)memBlock+NumberOfBytes &&
(PCHAR)afdBuffer->Mdl+MmSizeOfMdl(afdBuffer->Buffer, dataLength)<=(PCHAR)memBlock+NumberOfBytes &&
(PCHAR)afdBuffer->TdiInfo.RemoteAddress+AfdStandardAddressLength<=(PCHAR)memBlock+NumberOfBytes);
return afdBuffer;
}
#else
return AfdInitializeBuffer( memBlock, dataLength, AfdStandardAddressLength );
#endif
} // AfdAllocateBuffer
VOID
NTAPI
AfdFreeBuffer (
PVOID AfdBuffer
)
{
ASSERT( ((PAFD_BUFFER)AfdBuffer)->BufferLength == AfdSmallBufferSize ||
((PAFD_BUFFER)AfdBuffer)->BufferLength == AfdMediumBufferSize ||
((PAFD_BUFFER)AfdBuffer)->BufferLength == AfdLargeBufferSize );
#if DBG
AfdFreeBufferReal (AfdBuffer);
}
VOID
NTAPI
AfdFreeBufferReal (
PVOID AfdBuffer
)
{
#endif
{
PAFD_BUFFER hdr = AfdBuffer;
switch (hdr->Placement) {
case AFD_PLACEMENT_BUFFER:
AfdBuffer = hdr->Buffer;
break;
case AFD_PLACEMENT_HDR:
AfdBuffer = hdr;
break;
case AFD_PLACEMENT_MDL:
AfdBuffer = hdr->Mdl;
break;
case AFD_PLACEMENT_IRP:
AfdBuffer = hdr->Irp;
break;
default:
ASSERT (!"Unknown placement!");
__assume (0);
}
if (hdr->AlignmentAdjusted) {
//
// The original memory block was adjusted to meet alignment
// requirement of AFD buffers.
// The amount of adjustment should be stored in the space
// used for adjustment (right below the first piece).
//
ASSERT ((*(((PSIZE_T)AfdBuffer)-1))>0 &&
(*(((PSIZE_T)AfdBuffer)-1))<AfdBufferAlignment);
AfdBuffer = (PUCHAR)AfdBuffer - (*(((PSIZE_T)AfdBuffer)-1));
}
AFD_FREE_POOL (AfdBuffer, AFD_DATA_BUFFER_POOL_TAG);
}
}
PVOID
AfdAllocateBufferTag (
IN POOL_TYPE PoolType,
IN SIZE_T NumberOfBytes,
IN ULONG Tag
)
/*++
Routine Description:
Used by the lookaside list allocation function to allocate a new
AFD buffer tag structure. The returned structure will be fully
initialized.
Arguments:
PoolType - passed to ExAllocatePoolWithTag.
NumberOfBytes - the number of bytes required for the data buffer
portion of the AFD buffer tag (0).
Tag - passed to ExAllocatePoolWithTag.
Return Value:
PVOID - a fully initialized PAFD_BUFFER_TAG, or NULL if the allocation
attempt fails.
--*/
{
PAFD_BUFFER_TAG afdBufferTag;
//
// The requested length must be the same as buffer tag size
//
ASSERT(NumberOfBytes == sizeof (AFD_BUFFER_TAG) );
//
// Get nonpaged pool for the buffer tag.
//
afdBufferTag = AFD_ALLOCATE_POOL( PoolType, sizeof (AFD_BUFFER_TAG), Tag );
if ( afdBufferTag == NULL ) {
return NULL;
}
//
// Initialize the buffer tag and return a pointer to it.
//
AfdInitializeBufferTag( afdBufferTag, 0 );
return afdBufferTag;
} // AfdAllocateBufferTag
VOID
NTAPI
AfdFreeBufferTag (
PVOID AfdBufferTag
)
{
AFD_FREE_POOL (AfdBufferTag, AFD_DATA_BUFFER_POOL_TAG);
}
PVOID
AfdAllocateRemoteAddress (
IN POOL_TYPE PoolType,
IN SIZE_T NumberOfBytes,
IN ULONG Tag
)
/*++
Routine Description:
Used by the lookaside list allocation function to allocate a new
remote address structure. The returned structure will be fully
initialized.
Arguments:
PoolType - passed to ExAllocatePoolWithTag.
NumberOfBytes - the number of bytes required for the data buffer
portion of the AFD buffer tag (0).
Tag - passed to ExAllocatePoolWithTag.
Return Value:
PVOID - a fully initialized remote address, or NULL if the allocation
attempt fails.
--*/
{
//
// The requested length must be the same as standard address size
//
ASSERT(NumberOfBytes == AfdStandardAddressLength );
//
// Get nonpaged pool for the remote address.
//
return AFD_ALLOCATE_POOL( PoolType, AfdStandardAddressLength, Tag );
} // AfdAllocateRemoteAddress
VOID
NTAPI
AfdFreeRemoteAddress (
PVOID AfdBufferTag
)
{
AFD_FREE_POOL (AfdBufferTag, AFD_REMOTE_ADDRESS_POOL_TAG);
}
ULONG
AfdCalculateBufferSize (
IN ULONG BufferDataSize,
IN ULONG AddressSize
)
/*++
Routine Description:
Determines the size of an AFD buffer structure given the amount of
data that the buffer contains.
Arguments:
BufferDataSize - data length of the buffer.
AddressSize - length of address structure for the buffer.
Return Value:
Number of bytes needed for an AFD_BUFFER structure for data of
this size.
--*/
{
ULONG irpSize;
ULONG mdlSize;
ULONG hdrSize;
ULONG size;
//
// Determine the sizes of the various components of an AFD_BUFFER
// structure.
//
hdrSize = sizeof (AFD_BUFFER);
irpSize = IoSizeOfIrp( AfdIrpStackSize-1 );
//
// For mdl size calculation we rely on ex guarantee that buffer will be
// aligned on the page boundary (for allocations >= PAGE_SIZE)
// or will not spawn pages (for allocations < PAGE_SIZE).
//
mdlSize = (CLONG)MmSizeOfMdl( NULL, BufferDataSize );
size = ALIGN_UP_A(hdrSize,AFD_MINIMUM_BUFFER_ALIGNMENT) +
ALIGN_UP_A(irpSize,AFD_MINIMUM_BUFFER_ALIGNMENT) +
ALIGN_UP_A(mdlSize,AFD_MINIMUM_BUFFER_ALIGNMENT) +
ALIGN_UP_A(BufferDataSize,AFD_MINIMUM_BUFFER_ALIGNMENT) +
AddressSize;
if (size>=PAGE_SIZE)
return size;
else {
size += AfdAlignmentOverhead;
if (size>=PAGE_SIZE) {
return PAGE_SIZE;
}
else
return size;
}
} // AfdCalculateBufferSize
PAFD_BUFFER
AfdGetBuffer (
IN ULONG BufferDataSize,
IN ULONG AddressSize,
IN PEPROCESS Process
)
/*++
Routine Description:
Obtains a buffer of the appropriate size for the caller. Uses
the preallocated buffers if possible, or else allocates a new buffer
structure if required.
Arguments:
BufferDataSize - the size of the data buffer that goes along with the
buffer structure.
AddressSize - size of the address field required for the buffer.
Return Value:
PAFD_BUFFER - a pointer to an AFD_BUFFER structure, or NULL if one
was not available or could not be allocated.
--*/
{
ULONG bufferSize;
PNPAGED_LOOKASIDE_LIST lookasideList;
NTSTATUS status;
//
// If possible, allocate the buffer from one of the lookaside lists.
//
if ( AddressSize <= AfdStandardAddressLength &&
BufferDataSize <= AfdLargeBufferSize ) {
PAFD_BUFFER afdBuffer;
if ( BufferDataSize <= AfdSmallBufferSize ) {
lookasideList = &AfdLookasideLists->SmallBufferList;
} else if ( BufferDataSize <= AfdMediumBufferSize ) {
lookasideList = &AfdLookasideLists->MediumBufferList;
} else {
lookasideList = &AfdLookasideLists->LargeBufferList;
}
afdBuffer = ExAllocateFromNPagedLookasideList( lookasideList );
if ( afdBuffer != NULL) {
if (!afdBuffer->Lookaside) {
status = PsChargeProcessPoolQuota (
(PEPROCESS)((ULONG_PTR)Process & (~AFDB_RAISE_ON_FAILURE)),
NonPagedPool,
lookasideList->L.Size);
if (!NT_SUCCESS (status)) {
AfdFreeBuffer (afdBuffer);
goto ExitQuotaFailure;
}
AfdRecordQuotaHistory(
process,
(LONG)lookasideList->L.Size,
"BuffAlloc ",
afdBuffer
);
AfdRecordPoolQuotaCharged(lookasideList->L.Size);
}
#if DBG
RtlGetCallersAddress(
&afdBuffer->Caller,
&afdBuffer->CallersCaller
);
#endif
return afdBuffer;
}
}
else if (AddressSize<=0xFFFF) {
PVOID memBlock;
LONG sz;
//
// Couldn't find an appropriate buffer that was preallocated.
// Allocate one manually. If the buffer size requested was
// zero bytes, give them four bytes. This is because some of
// the routines like MmSizeOfMdl() cannot handle getting passed
// in a length of zero.
//
// !!! It would be good to ROUND_TO_PAGES for this allocation
// if appropriate, then use entire buffer size.
//
if ( BufferDataSize == 0 ) {
BufferDataSize = sizeof(ULONG);
}
bufferSize = AfdCalculateBufferSize( BufferDataSize, AddressSize );
//
// Check for overflow.
//
if (bufferSize>=BufferDataSize && bufferSize>=AddressSize) {
memBlock = AFD_ALLOCATE_POOL(
NonPagedPool,
bufferSize,
AFD_DATA_BUFFER_POOL_TAG
);
if ( memBlock != NULL) {
status = PsChargeProcessPoolQuota (
(PEPROCESS)((ULONG_PTR)Process & (~AFDB_RAISE_ON_FAILURE)),
NonPagedPool,
sz = BufferDataSize
+AfdBufferOverhead
+AddressSize
-AfdStandardAddressLength
+BufferDataSize<PAGE_SIZE
? min (AfdAlignmentOverhead, PAGE_SIZE-BufferDataSize)
: 0);
if (NT_SUCCESS (status)) {
PAFD_BUFFER afdBuffer;
//
// Initialize the AFD buffer structure and return it.
//
afdBuffer = AfdInitializeBuffer( memBlock, BufferDataSize, AddressSize );
ASSERT ((PCHAR)afdBuffer+sizeof (AFD_BUFFER)<=(PCHAR)memBlock+bufferSize &&
(PCHAR)afdBuffer->Buffer+BufferDataSize<=(PCHAR)memBlock+bufferSize &&
(PCHAR)afdBuffer->Irp+IoSizeOfIrp(AfdIrpStackSize-1)<=(PCHAR)memBlock+bufferSize &&
(PCHAR)afdBuffer->Mdl+MmSizeOfMdl(afdBuffer->Buffer, BufferDataSize)<=(PCHAR)memBlock+bufferSize &&
(PCHAR)afdBuffer->TdiInfo.RemoteAddress+AddressSize<=(PCHAR)memBlock+bufferSize);
AfdRecordPoolQuotaCharged(sz);
AfdRecordQuotaHistory(
process,
sz,
"BuffAlloc ",
afdBuffer
);
#if DBG
RtlGetCallersAddress(
&afdBuffer->Caller,
&afdBuffer->CallersCaller
);
#endif
return afdBuffer;
}
else {
AFD_FREE_POOL (memBlock, AFD_DATA_BUFFER_POOL_TAG);
goto ExitQuotaFailure;
}
} // memblock==NULL
} // overflow
}
else {
// TDI does not support addresses > USHORT
ASSERT (FALSE);
}
//
// This is default status code.
// Quota failures jump directly to the
// label below to raise the status returned by
// the quota charging code if requested by the caller..
//
status = STATUS_INSUFFICIENT_RESOURCES;
ExitQuotaFailure:
if ((ULONG_PTR)Process & AFDB_RAISE_ON_FAILURE) {
ExRaiseStatus (status);
}
return NULL;
} // AfdGetBuffer
PAFD_BUFFER_TAG
AfdGetBufferTag (
IN ULONG AddressSize,
IN PEPROCESS Process
)
/*++
Routine Description:
Obtains a buffer for tagging TDSU received via chained indication. Uses
the preallocated buffers if possible, or else allocates a new buffer
structure if required.
Arguments:
AddressSize - size of the address field required for the buffer.
Return Value:
PAFD_BUFFER_TAG - a pointer to an AFD_BUFFER_TAG structure, or NULL if one
was not available or could not be allocated.
--*/
{
PAFD_BUFFER_TAG afdBufferTag;
ULONG bufferSize;
NTSTATUS status;
if ( AddressSize <= AfdStandardAddressLength) {
if (AddressSize>0)
AddressSize = AfdStandardAddressLength;
afdBufferTag = ExAllocateFromNPagedLookasideList(
&AfdLookasideLists->BufferTagList );
if ( afdBufferTag != NULL &&
( AddressSize==0 ||
(afdBufferTag->TdiInfo.RemoteAddress =
ExAllocateFromNPagedLookasideList(
&AfdLookasideLists->RemoteAddrList ))!=NULL ) ) {
afdBufferTag->AllocatedAddressLength = (USHORT)AddressSize;
if (!afdBufferTag->Lookaside) {
status = PsChargeProcessPoolQuota (
(PEPROCESS)((ULONG_PTR)Process & (~AFDB_RAISE_ON_FAILURE)),
NonPagedPool,
sizeof (AFD_BUFFER_TAG)+AddressSize);
if (!NT_SUCCESS (status)) {
if ((afdBufferTag->TdiInfo.RemoteAddress!=NULL) &&
(afdBufferTag->TdiInfo.RemoteAddress != (PVOID)(afdBufferTag+1))) {
ExFreeToNPagedLookasideList( &AfdLookasideLists->RemoteAddrList,
afdBufferTag->TdiInfo.RemoteAddress );
}
AFD_FREE_POOL (afdBufferTag, AFD_DATA_BUFFER_POOL_TAG);
goto ExitQuotaFailure;
}
AfdRecordQuotaHistory(
process,
(LONG)(sizeof (AFD_BUFFER_TAG)+AddressSize),
"BuffAlloc ",
afdBufferTag
);
AfdRecordPoolQuotaCharged(sizeof (AFD_BUFFER_TAG)+AddressSize);
}
#if DBG
RtlGetCallersAddress(
&afdBufferTag->Caller,
&afdBufferTag->CallersCaller
);
#endif
return afdBufferTag;
} // afdBufferTag==NULL || RemoteAddress==NULL
}
else if (AddressSize<=0xFFFF) {
bufferSize = sizeof (AFD_BUFFER_TAG) + AddressSize;
afdBufferTag = AFD_ALLOCATE_POOL(
NonPagedPool,
bufferSize,
AFD_DATA_BUFFER_POOL_TAG
);
if (afdBufferTag!=NULL) {
status = PsChargeProcessPoolQuota (
(PEPROCESS)((ULONG_PTR)Process & (~AFDB_RAISE_ON_FAILURE)),
NonPagedPool,
bufferSize);
if (NT_SUCCESS (status)) {
//
// Initialize the AFD buffer structure and return it.
//
AfdInitializeBufferTag (afdBufferTag, AddressSize);
AfdRecordQuotaHistory(
process,
(LONG)bufferSize,
"BuffAlloc ",
afdBufferTag
);
AfdRecordPoolQuotaCharged(bufferSize);
#if DBG
RtlGetCallersAddress(
&afdBufferTag->Caller,
&afdBufferTag->CallersCaller
);
#endif
return afdBufferTag;
}
else {
AFD_FREE_POOL (afdBufferTag, AFD_DATA_BUFFER_POOL_TAG);
goto ExitQuotaFailure;
}
}
}
else {
// TDI does not support addresses > USHORT
ASSERT (FALSE);
}
//
// This is default status code.
// Quota failures jump directly to the
// label below to raise the status returned by
// the quota charging code if requested by the caller..
//
status = STATUS_INSUFFICIENT_RESOURCES;
ExitQuotaFailure:
if ((ULONG_PTR)Process & AFDB_RAISE_ON_FAILURE) {
ExRaiseStatus (status);
}
return NULL;
}
VOID
AfdReturnBuffer (
IN PAFD_BUFFER_HEADER AfdBufferHeader,
IN PEPROCESS Process
)
/*++
Routine Description:
Returns an AFD buffer to the appropriate global list, or frees
it if necessary.
Arguments:
AfdBufferHeader - points to the AFD_BUFFER_HEADER structure to return or free.
Return Value:
None.
--*/
{
if (AfdBufferHeader->BufferLength!=AfdBufferTagSize) {
PNPAGED_LOOKASIDE_LIST lookasideList;
PAFD_BUFFER AfdBuffer = CONTAINING_RECORD (AfdBufferHeader, AFD_BUFFER, Header);
ASSERT (IS_VALID_AFD_BUFFER (AfdBuffer));
//
// Most of the AFD buffer must be zeroed when returning the buffer.
//
ASSERT( !AfdBuffer->ExpeditedData );
ASSERT( AfdBuffer->Mdl->ByteCount == AfdBuffer->BufferLength );
ASSERT( AfdBuffer->Mdl->Next == NULL );
//
// If appropriate, return the buffer to one of the AFD buffer
// lookaside lists.
//
if (AfdBuffer->AllocatedAddressLength == AfdStandardAddressLength &&
AfdBuffer->BufferLength <= AfdLargeBufferSize) {
if (AfdBuffer->BufferLength==AfdSmallBufferSize) {
lookasideList = &AfdLookasideLists->SmallBufferList;
} else if (AfdBuffer->BufferLength == AfdMediumBufferSize) {
lookasideList = &AfdLookasideLists->MediumBufferList;
} else {
ASSERT (AfdBuffer->BufferLength==AfdLargeBufferSize);
lookasideList = &AfdLookasideLists->LargeBufferList;
}
if (!AfdBuffer->Lookaside) {
PsReturnPoolQuota (Process, NonPagedPool, lookasideList->L.Size);
AfdRecordQuotaHistory(
Process,
-(LONG)lookasideList->L.Size,
"BuffDealloc ",
AfdBuffer
);
AfdRecordPoolQuotaReturned(
lookasideList->L.Size
);
AfdBuffer->Lookaside = TRUE;
}
ExFreeToNPagedLookasideList( lookasideList, AfdBuffer );
return;
}
// The buffer was not from a lookaside list allocation, so just free
// the pool we used for it.
//
ASSERT (AfdBuffer->Lookaside==FALSE);
{
LONG sz;
PsReturnPoolQuota (Process,
NonPagedPool,
sz=AfdBuffer->BufferLength
+AfdBufferOverhead
+AfdBuffer->AllocatedAddressLength
-AfdStandardAddressLength
+AfdBuffer->BufferLength<PAGE_SIZE
? min (AfdAlignmentOverhead, PAGE_SIZE-AfdBuffer->BufferLength)
: 0);
AfdRecordQuotaHistory(
Process,
-(LONG)sz,
"BuffDealloc ",
AfdBuffer
);
AfdRecordPoolQuotaReturned(
sz
);
}
#if DBG
AfdFreeBufferReal (AfdBuffer);
#else
AfdFreeBuffer (AfdBuffer);
#endif
return;
}
else {
PAFD_BUFFER_TAG AfdBufferTag = CONTAINING_RECORD (AfdBufferHeader, AFD_BUFFER_TAG, Header);
ASSERT( !AfdBufferTag->ExpeditedData );
if (AfdBufferTag->NdisPacket) {
AfdBufferTag->NdisPacket = FALSE;
TdiReturnChainedReceives (&AfdBufferTag->Context, 1);
}
if (AfdBufferTag->TdiInfo.RemoteAddress != (PVOID)(AfdBufferTag+1)) {
if (AfdBufferTag->TdiInfo.RemoteAddress!=NULL) {
ASSERT (AfdBufferTag->AllocatedAddressLength==AfdStandardAddressLength);
ExFreeToNPagedLookasideList( &AfdLookasideLists->RemoteAddrList,
AfdBufferTag->TdiInfo.RemoteAddress );
AfdBufferTag->TdiInfo.RemoteAddress = NULL;
}
else {
ASSERT (AfdBufferTag->AllocatedAddressLength==0);
}
if (!AfdBufferTag->Lookaside) {
LONG sz;
PsReturnPoolQuota (
Process,
NonPagedPool,
sz=sizeof (AFD_BUFFER_TAG)
+ AfdBufferTag->AllocatedAddressLength);
AfdRecordQuotaHistory(
Process,
-(LONG)sz,
"BuffDealloc ",
AfdBufferTag
);
AfdRecordPoolQuotaReturned(
sz
);
AfdBufferTag->Lookaside = TRUE;
}
ExFreeToNPagedLookasideList( &AfdLookasideLists->BufferTagList, AfdBufferTag );
}
else {
LONG sz;
ASSERT (AfdBufferTag->AllocatedAddressLength>AfdStandardAddressLength);
ASSERT (AfdBufferTag->Lookaside == FALSE);
PsReturnPoolQuota (
Process,
NonPagedPool,
sz = sizeof (AFD_BUFFER_TAG)
+ AfdBufferTag->AllocatedAddressLength);
AfdRecordQuotaHistory(
Process,
-(LONG)sz,
"BuffDealloc ",
AfdBufferTag
);
AfdRecordPoolQuotaReturned(
sz
);
AFD_FREE_POOL(
AfdBufferTag,
AFD_DATA_BUFFER_POOL_TAG
);
}
}
} // AfdReturnBuffer
PAFD_BUFFER
AfdInitializeBuffer (
IN PVOID MemoryBlock,
IN ULONG BufferDataSize,
IN ULONG AddressSize
)
/*++
Routine Description:
Initializes an AFD buffer. Sets up fields in the actual AFD_BUFFER
structure and initializes the IRP and MDL associated with the
buffer. This routine assumes that the caller has properly allocated
sufficient space for all this.
Arguments:
AfdBuffer - points to the AFD_BUFFER structure to initialize.
BufferDataSize - the size of the data buffer that goes along with the
buffer structure.
AddressSize - the size of data allocated for the address buffer.
ListHead - the global list this buffer belongs to, or NULL if it
doesn't belong on any list. This routine does NOT place the
buffer structure on the list.
Return Value:
None.
--*/
{
USHORT irpSize;
SIZE_T mdlSize;
SIZE_T hdrSize;
PAFD_BUFFER hdr;
PMDL mdl;
PIRP irp;
PVOID buf;
PVOID addr;
UCHAR placement;
SIZE_T alignment;
#ifdef AFD_CHECK_ALIGNMENT
PLONG alignmentCounters = (PLONG)&AfdAlignmentTable[AfdAlignmentTableSize];
#endif
irpSize = IoSizeOfIrp( AfdIrpStackSize-1 );
mdlSize = (ULONG)MmSizeOfMdl( NULL, BufferDataSize );
hdrSize = sizeof (AFD_BUFFER);
//
// Compute the index into (mis)alignment table to determine
// what placement of the buffer block elements (e.g. hdr, IRP, MDL,
// and data buffer itself) we need to choose to compensate and
// align data buffer on AfdBufferAlignment boundary.
//
ASSERT ((PtrToUlong(MemoryBlock)%AFD_MINIMUM_BUFFER_ALIGNMENT)==0);
if (PAGE_ALIGN (MemoryBlock)==MemoryBlock) {
//
// For page-aligned blocks (which are >= page size),
// we always place the buffer first.
//
placement = AFD_PLACEMENT_BUFFER;
}
else {
placement = AfdAlignmentTable[
(PtrToUlong(MemoryBlock)&(AfdBufferAlignment-1))/AFD_MINIMUM_BUFFER_ALIGNMENT];
}
#ifdef AFD_CHECK_ALIGNMENT
InterlockedIncrement (&alignmentCounters[
(PtrToUlong(MemoryBlock)&(AfdBufferAlignment-1))/AFD_MINIMUM_BUFFER_ALIGNMENT]);
#endif
switch (placement) {
case AFD_PLACEMENT_BUFFER:
//
// Perfect case: the memory is aready aligned as we need it.
//
buf = ALIGN_UP_A_POINTER(MemoryBlock, AfdBufferAlignment);
alignment = (PUCHAR)buf-(PUCHAR)MemoryBlock;
ASSERT (alignment<=AfdAlignmentOverhead);
hdr = ALIGN_UP_TO_TYPE_POINTER((PCHAR)buf+BufferDataSize, AFD_BUFFER);
irp = ALIGN_UP_TO_TYPE_POINTER((PCHAR)hdr+hdrSize, IRP);
mdl = ALIGN_UP_TO_TYPE_POINTER((PCHAR)irp+irpSize, MDL);
addr = (PCHAR)mdl+mdlSize;
break;
//
// Other cases, we use hdr, mdl, and IRP to try to compensate
// and have the data buffer aligned at the AfdBufferAlignment
// boundary.
//
case AFD_PLACEMENT_HDR:
hdr = ALIGN_UP_TO_TYPE_POINTER(MemoryBlock, AFD_BUFFER);
alignment = (PUCHAR)hdr-(PUCHAR)MemoryBlock;
ASSERT (alignment<=AfdAlignmentOverhead);
buf = ALIGN_UP_A_POINTER((PCHAR)hdr+hdrSize, AfdBufferAlignment);
irp = ALIGN_UP_TO_TYPE_POINTER((PCHAR)buf+BufferDataSize, IRP);
mdl = ALIGN_UP_TO_TYPE_POINTER((PCHAR)irp+irpSize, MDL);
addr = (PCHAR)mdl+mdlSize;
break;
case AFD_PLACEMENT_MDL:
mdl = ALIGN_UP_TO_TYPE_POINTER(MemoryBlock, MDL);
alignment = (PUCHAR)mdl-(PUCHAR)MemoryBlock;
ASSERT (alignment<=AfdAlignmentOverhead);
buf = ALIGN_UP_A_POINTER((PCHAR)mdl+mdlSize, AfdBufferAlignment);
hdr = ALIGN_UP_TO_TYPE_POINTER((PCHAR)buf+BufferDataSize, AFD_BUFFER);
irp = ALIGN_UP_TO_TYPE_POINTER((PCHAR)hdr+hdrSize, IRP);
addr = (PCHAR)irp+irpSize;
break;
case AFD_PLACEMENT_IRP:
irp = ALIGN_UP_TO_TYPE_POINTER(MemoryBlock, IRP);
alignment = (PUCHAR)irp-(PUCHAR)MemoryBlock;
ASSERT (alignment<=AfdAlignmentOverhead);
buf = ALIGN_UP_A_POINTER((PCHAR)irp+irpSize, AfdBufferAlignment);
hdr = ALIGN_UP_TO_TYPE_POINTER((PCHAR)buf+BufferDataSize, AFD_BUFFER);
mdl = ALIGN_UP_TO_TYPE_POINTER((PCHAR)hdr+hdrSize, MDL);
addr = (PCHAR)mdl+mdlSize;
break;
case AFD_PLACEMENT_HDR_IRP:
hdr = ALIGN_UP_TO_TYPE_POINTER(MemoryBlock, AFD_BUFFER);
alignment = (PUCHAR)hdr-(PUCHAR)MemoryBlock;
ASSERT (alignment<=AfdAlignmentOverhead);
irp = ALIGN_UP_TO_TYPE_POINTER((PCHAR)hdr+hdrSize, IRP);
buf = ALIGN_UP_A_POINTER((PCHAR)irp+irpSize, AfdBufferAlignment);
mdl = ALIGN_UP_TO_TYPE_POINTER((PCHAR)buf+BufferDataSize, MDL);
addr = (PCHAR)mdl+mdlSize;
break;
case AFD_PLACEMENT_HDR_MDL:
hdr = ALIGN_UP_TO_TYPE_POINTER(MemoryBlock, AFD_BUFFER);
alignment = (PUCHAR)hdr-(PUCHAR)MemoryBlock;
ASSERT (alignment<=AfdAlignmentOverhead);
mdl = ALIGN_UP_TO_TYPE_POINTER((PCHAR)hdr+hdrSize, MDL);
buf = ALIGN_UP_A_POINTER((PCHAR)mdl+mdlSize, AfdBufferAlignment);
irp = ALIGN_UP_TO_TYPE_POINTER((PCHAR)buf+BufferDataSize, IRP);
addr = (PCHAR)irp+irpSize;
break;
case AFD_PLACEMENT_IRP_MDL:
irp = ALIGN_UP_TO_TYPE_POINTER(MemoryBlock, IRP);
alignment = (PUCHAR)irp-(PUCHAR)MemoryBlock;
ASSERT (alignment<=AfdAlignmentOverhead);
mdl = ALIGN_UP_TO_TYPE_POINTER((PCHAR)irp+irpSize, MDL);
buf = ALIGN_UP_A_POINTER((PCHAR)mdl+mdlSize, AfdBufferAlignment);
hdr = ALIGN_UP_TO_TYPE_POINTER((PCHAR)buf+BufferDataSize, AFD_BUFFER);
addr = (PCHAR)hdr+hdrSize;
break;
case AFD_PLACEMENT_HDR_IRP_MDL:
hdr = ALIGN_UP_TO_TYPE_POINTER(MemoryBlock, AFD_BUFFER);
alignment = (PUCHAR)hdr-(PUCHAR)MemoryBlock;
ASSERT (alignment<=AfdAlignmentOverhead);
irp = ALIGN_UP_TO_TYPE_POINTER((PCHAR)hdr+hdrSize, IRP);
mdl = ALIGN_UP_TO_TYPE_POINTER((PCHAR)irp+irpSize, MDL);
buf = ALIGN_UP_A_POINTER((PCHAR)mdl+mdlSize, AfdBufferAlignment);
addr = (PCHAR)buf+BufferDataSize;
break;
default:
ASSERT (!"Unknown placement!");
__assume (0);
}
//
// Initialize the AfdBuffer - most fields need to be 0.
//
RtlZeroMemory( hdr, sizeof(*hdr) );
//
// Setup buffer
//
hdr->Buffer = buf;
hdr->BufferLength = BufferDataSize;
//
// We just need to store first two bits of placement
// so we know which part comes first to free it properly.
//
hdr->Placement = placement & 3;
//
// If we have to align the memory block to meet the requirement
// store this information right below the first piece.
//
if (alignment!=0) {
C_ASSERT (AFD_MINIMUM_BUFFER_ALIGNMENT>=sizeof (SIZE_T));
C_ASSERT ((AFD_MINIMUM_BUFFER_ALIGNMENT & (sizeof(SIZE_T)-1))==0);
ASSERT (alignment>=sizeof (SIZE_T));
hdr->AlignmentAdjusted = TRUE;
*(((PSIZE_T)(((PUCHAR)MemoryBlock)+alignment))-1) = alignment;
}
//
// Initialize the IRP pointer.
//
hdr->Irp = irp;
IoInitializeIrp( hdr->Irp, irpSize, (CCHAR)(AfdIrpStackSize-1) );
hdr->Irp->MdlAddress = mdl;
//
// Set up the MDL pointer.
//
hdr->Mdl = mdl;
MmInitializeMdl( hdr->Mdl, buf, BufferDataSize );
MmBuildMdlForNonPagedPool( hdr->Mdl );
//
// Set up the address buffer pointer.
//
if (AddressSize>0) {
hdr->TdiInfo.RemoteAddress = ALIGN_UP_TO_TYPE_POINTER(addr, TRANSPORT_ADDRESS);;
hdr->AllocatedAddressLength = (USHORT)AddressSize;
}
#if DBG
hdr->BufferListEntry.Flink = UIntToPtr( 0xE0E1E2E3 );
hdr->BufferListEntry.Blink = UIntToPtr( 0xE4E5E6E7 );
#endif
return hdr;
} // AfdInitializeBuffer
VOID
AfdInitializeBufferTag (
IN PAFD_BUFFER_TAG AfdBufferTag,
IN CLONG AddressSize
)
/*++
Routine Description:
Initializes an AFD buffer. Sets up fields in the actual AFD_BUFFER
structure and initializes the IRP and MDL associated with the
buffer. This routine assumes that the caller has properly allocated
sufficient space for all this.
Arguments:
AfdBuffer - points to the AFD_BUFFER structure to initialize.
BufferDataSize - the size of the data buffer that goes along with the
buffer structure.
AddressSize - the size of data allocated for the address buffer.
ListHead - the global list this buffer belongs to, or NULL if it
doesn't belong on any list. This routine does NOT place the
buffer structure on the list.
Return Value:
None.
--*/
{
AfdBufferTag->Mdl = NULL;
AfdBufferTag->BufferLength = AfdBufferTagSize;
AfdBufferTag->TdiInfo.RemoteAddress = AddressSize ? AfdBufferTag+1 : NULL;
AfdBufferTag->AllocatedAddressLength = (USHORT)AddressSize;
AfdBufferTag->Flags = 0;
#if DBG
AfdBufferTag->BufferListEntry.Flink = UIntToPtr( 0xE0E1E2E3 );
AfdBufferTag->BufferListEntry.Blink = UIntToPtr( 0xE4E5E6E7 );
AfdBufferTag->Caller = NULL;
AfdBufferTag->CallersCaller = NULL;
#endif
}
VOID
AfdInitializeBufferManager (
VOID
)
{
SIZE_T irpSize = ALIGN_UP_A(IoSizeOfIrp (AfdIrpStackSize-1), AFD_MINIMUM_BUFFER_ALIGNMENT);
SIZE_T hdrSize = ALIGN_UP_A(sizeof (AFD_BUFFER), AFD_MINIMUM_BUFFER_ALIGNMENT);
SIZE_T mdlSize = ALIGN_UP_A(MmSizeOfMdl (NULL, PAGE_SIZE),AFD_MINIMUM_BUFFER_ALIGNMENT);
UCHAR placement;
ULONG i;
ULONG currentOverhead;
//
// Initialize the alignment table.
// This table is used to determine what kind of element
// placement to use in AFD_BUFFER depending on the alignment
// of the memory block returned by the executive pool manager.
// The goal is to align the data buffer on the cache line
// boundary. However, since executive only guarantees alignment of
// it's blocks at the CPU alignment requirements, we need to
// adjust and potentially waste up to CACHE_LIST_SIZE-CPU_ALIGNMENT_SIZE.
// With some machines having cache line alignment at 128 such memory
// waste is prohibitive (small buffers with default size of 128 will double
// in size).
// The table below allows us to rearrange pieces in AFD_BUFFER structure,
// namely, the header, IRP, MDL, and data buffer, so that pieces with
// lower alignment requirement can be used to consume the space needed
// to adjust the memory block to the cache line boundary.
//
// AfdAlignmentTable has an entry for each possible case of memory block
// misaligned against cache line size. For example, in typical X86 system
// case executive pool manager always returns blocks aligned on 8 byte bounday,
// while cache lines are typically 64 bytes long, so memory manager can
// theoretically return blocks misaligned against cache by:
// 8, 16, 24, 32, 40, 48, 56.
// For each of these cases we will try to adjust the alignment by using
// any possible combination of header, IRP, and MDL. There will be some
// cases that cannot be adjusted exactly, and we will have to pad.
//
//
// First initialize the table assuming the data buffer is placed first.
//
RtlFillMemory (AfdAlignmentTable, AfdAlignmentTableSize, AFD_PLACEMENT_BUFFER);
#ifdef AFD_CHECK_ALIGNMENT
RtlZeroMemory (&AfdAlignmentTable[AfdAlignmentTableSize],
AfdAlignmentTableSize*sizeof(LONG));
#endif
//
// Now identify the entries that can be padded with some combination of
// header, IRP, and MDL:
// extract the bits that can be used for padding
// reverse to get corresponding memory block alignments
// divide by the step of the alignment table
// make sure we won't go past table size (last entry => 0 entry).
//
#define AfdInitAlignmentTableRow(_size,_plcmnt) \
AfdAlignmentTable[ \
((AfdBufferAlignment-(_size&(AfdBufferAlignment-1))) \
/AFD_MINIMUM_BUFFER_ALIGNMENT) \
&(AfdAlignmentTableSize-1)] = _plcmnt
//
// We let placements beginning with header override others,
// since it is more natural and easier to debug (header has references
// to other pieces).
//
AfdInitAlignmentTableRow(mdlSize,AFD_PLACEMENT_MDL);
AfdInitAlignmentTableRow(irpSize,AFD_PLACEMENT_IRP);
AfdInitAlignmentTableRow((irpSize+mdlSize),AFD_PLACEMENT_IRP_MDL);
AfdInitAlignmentTableRow((hdrSize+mdlSize),AFD_PLACEMENT_HDR_MDL);
AfdInitAlignmentTableRow((hdrSize+irpSize),AFD_PLACEMENT_HDR_IRP);
AfdInitAlignmentTableRow((hdrSize+irpSize+mdlSize),AFD_PLACEMENT_HDR_IRP_MDL);
AfdInitAlignmentTableRow(hdrSize,AFD_PLACEMENT_HDR);
//
// Now scan the table from top to bottom and fill entries that do not have
// exact match using the above combinations. Use the closest entry above and
// in the process compute how much do we need to pad in addition to padding
// achieved via placement.
//
AfdAlignmentOverhead = 0;
currentOverhead = 0;
//
// By default use the placement of aligned block.
//
placement = AfdAlignmentTable[0];
for (i=AfdAlignmentTableSize-1; i>0; i--) {
if (AfdAlignmentTable[i]==AFD_PLACEMENT_BUFFER) {
AfdAlignmentTable[i] = placement;
currentOverhead += AFD_MINIMUM_BUFFER_ALIGNMENT;
}
else {
placement = AfdAlignmentTable[i];
if (AfdAlignmentOverhead<currentOverhead) {
AfdAlignmentOverhead = currentOverhead;
}
currentOverhead = 0;
}
}
if (AfdAlignmentOverhead<currentOverhead) {
AfdAlignmentOverhead = currentOverhead;
}
KdPrintEx(( DPFLTR_WSOCKTRANSPORT_ID, DPFLTR_INFO_LEVEL,
"AfdInitializeBufferManager: Alignment requirements: MM-%d, cache-%d, overhead-%d\n",
AFD_MINIMUM_BUFFER_ALIGNMENT,
AfdBufferAlignment,
AfdAlignmentOverhead));
{
CLONG oldBufferLengthForOnePage = AfdBufferLengthForOnePage;
AfdBufferOverhead = AfdCalculateBufferSize( PAGE_SIZE, AfdStandardAddressLength) - PAGE_SIZE;
AfdBufferLengthForOnePage = ALIGN_DOWN_A(
PAGE_SIZE-AfdBufferOverhead,
AFD_MINIMUM_BUFFER_ALIGNMENT);
if (AfdLargeBufferSize==oldBufferLengthForOnePage) {
AfdLargeBufferSize = AfdBufferLengthForOnePage;
}
}
}