windows-nt/Source/XPSP1/NT/inetsrv/iis/iisrearc/ul/drv/httpconn.cxx
2020-09-26 16:20:57 +08:00

2287 lines
54 KiB
C++

/*++
Copyright (c) 1998-2001 Microsoft Corporation
Module Name:
httpconn.cxx
Abstract:
This module implements the HTTP_CONNECTION object.
Author:
Keith Moore (keithmo) 08-Jul-1998
Revision History:
--*/
#include "precomp.h"
#include "httpconnp.h"
//
// Private globals.
//
//
// Global connection Limits stuff
//
ULONG g_MaxConnections = HTTP_LIMIT_INFINITE;
ULONG g_CurrentConnections = 0;
BOOLEAN g_InitGlobalConnections = FALSE;
#ifdef ALLOC_PRAGMA
#endif // ALLOC_PRAGMA
#if 0
NOT PAGEABLE -- UlBindConnectionToProcess
NOT PAGEABLE -- UlQueryProcessBinding
NOT PAGEABLE -- UlpCreateProcBinding
NOT PAGEABLE -- UlpFreeProcBinding
NOT PAGEABLE -- UlpFindProcBinding
NOT PAGEABLE -- UlCreateHttpConnection
NOT PAGEABLE -- UlReferenceHttpConnection
NOT PAGEABLE -- UlDereferenceHttpConnection
NOT PAGEABLE -- UlReferenceHttpRequest
NOT PAGEABLE -- UlDereferenceHttpRequest
NOT PAGEABLE -- UlpCreateHttpRequest
NOT PAGEABLE -- UlpFreeHttpRequest
#endif
//
// Public functions.
//
/***************************************************************************++
Routine Description:
Creates a new HTTP_CONNECTION object.
Arguments:
ppHttpConnection - Receives a pointer to the new HTTP_CONNECTION
object if successful.
pConnection - Supplies a pointer to the UL_CONNECTION to associate
with the newly created HTTP_CONNECTION.
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlCreateHttpConnection(
OUT PUL_HTTP_CONNECTION *ppHttpConnection,
IN PUL_CONNECTION pConnection
)
{
PUL_HTTP_CONNECTION pHttpConnection;
NTSTATUS Status;
ASSERT(KeGetCurrentIrql() <= DISPATCH_LEVEL);
pHttpConnection = &pConnection->HttpConnection;
if (pHttpConnection != NULL)
{
pHttpConnection->Signature = UL_HTTP_CONNECTION_POOL_TAG;
pHttpConnection->RefCount = 1;
pHttpConnection->pConnection = pConnection;
REFERENCE_CONNECTION( pConnection );
pHttpConnection->SecureConnection = pConnection->FilterInfo.SecureConnection;
HTTP_SET_NULL_ID(&pHttpConnection->ConnectionId);
pHttpConnection->SendBufferedBytes = 0;
pHttpConnection->NextRecvNumber = 0;
pHttpConnection->NextBufferNumber = 0;
pHttpConnection->NextBufferToParse = 0;
pHttpConnection->pRequest = NULL;
pHttpConnection->pCurrentBuffer = NULL;
pHttpConnection->pConnectionCountEntry = NULL;
pHttpConnection->pPrevSiteCounters = NULL;
//
// Init connection timeout info block; implicitly starts
// ConnectionIdle timer.
//
UlInitializeConnectionTimerInfo( &pHttpConnection->TimeoutInfo );
//
// Initialize the disconnect management fields.
//
pHttpConnection->NeedMoreData = 0;
pHttpConnection->UlconnDestroyed = 0;
pHttpConnection->WaitingForResponse = 0;
pHttpConnection->DisconnectFlag = FALSE;
UlInitializeNotifyHead(
&pHttpConnection->WaitForDisconnectHead,
NULL
);
//
// Init our buffer list
//
InitializeListHead(&(pHttpConnection->BufferHead));
//
// Init QoS parameters on the connection
//
pHttpConnection->BandwidthThrottlingEnabled = 0;
pHttpConnection->pFlow = NULL;
pHttpConnection->pFilter = NULL;
//
// init app pool process binding list
//
InitializeListHead(&(pHttpConnection->BindingHead));
UlInitializeSpinLock(
&pHttpConnection->BindingSpinLock,
"BindingSpinLock"
);
Status = UlInitializeResource(
&(pHttpConnection->Resource),
"UL_HTTP_CONNECTION[%p].Resource",
pHttpConnection,
UL_HTTP_CONNECTION_RESOURCE_TAG
);
if (NT_SUCCESS(Status) == FALSE)
{
RETURN(Status);
}
//
// Initialize BufferingInfo structure.
//
UlInitializeSpinLock(
&pHttpConnection->BufferingInfo.BufferingSpinLock,
"BufferingSpinLock"
);
pHttpConnection->BufferingInfo.BytesBuffered = 0;
pHttpConnection->BufferingInfo.TransportBytesNotTaken = 0;
pHttpConnection->BufferingInfo.DrainAfterDisconnect = FALSE;
pHttpConnection->BufferingInfo.ReadIrpPending = FALSE;
pHttpConnection->pRequestIdContext = NULL;
UlInitializeSpinLock(
&pHttpConnection->RequestIdSpinLock,
"RequestIdSpinLock"
);
UlTrace(
REFCOUNT, (
"ul!UlCreateHttpConnection conn=%p refcount=%d\n",
pHttpConnection,
pHttpConnection->RefCount)
);
*ppHttpConnection = pHttpConnection;
RETURN(STATUS_SUCCESS);
}
RETURN(STATUS_INSUFFICIENT_RESOURCES);
} // UlCreateHttpConnection
NTSTATUS
UlCreateHttpConnectionId(
IN PUL_HTTP_CONNECTION pHttpConnection
)
{
NTSTATUS Status;
//
// Sanity check.
//
PAGED_CODE();
//
// Create an opaque id for the connection
//
Status = UlAllocateOpaqueId(
&pHttpConnection->ConnectionId, // pOpaqueId
UlOpaqueIdTypeHttpConnection, // OpaqueIdType
pHttpConnection // pContext
);
if (NT_SUCCESS(Status))
{
UL_REFERENCE_HTTP_CONNECTION(pHttpConnection);
TRACE_TIME(
pHttpConnection->ConnectionId,
0,
TIME_ACTION_CREATE_CONNECTION
);
}
RETURN(Status);
} // UlCreateHttpConnectionId
VOID
UlConnectionDestroyedWorker(
IN PUL_WORK_ITEM pWorkItem
)
{
PUL_HTTP_CONNECTION pHttpConnection;
//
// Sanity check.
//
PAGED_CODE();
pHttpConnection = CONTAINING_RECORD(
pWorkItem,
UL_HTTP_CONNECTION,
WorkItem
);
ASSERT(UL_IS_VALID_HTTP_CONNECTION(pHttpConnection));
UlTrace(HTTP_IO, (
"ul!UlConnectionDestroyedWorker: httpconn=%p\n",
pHttpConnection
));
UlAcquireResourceExclusive( &pHttpConnection->Resource, TRUE );
//
// kill the request if there is one
//
if (pHttpConnection->pRequest) {
ASSERT(UL_IS_VALID_INTERNAL_REQUEST(pHttpConnection->pRequest));
UlUnlinkHttpRequest(pHttpConnection->pRequest);
pHttpConnection->pRequest = NULL;
}
//
// make sure no one adds a new request now that we're done
//
pHttpConnection->UlconnDestroyed = 1;
//
// Decrement the global connection limit. Its safe to decrement for
// global case because http object doesn't even allocated for global
// rejection which happens as tcp refusal early when acepting the
// connection request.
//
UlDecrementConnections( &g_CurrentConnections );
//
// Decrement the site connections and let our ref go away. If the
// pConnectionCountEntry is not null, we have been accepted.
//
if (pHttpConnection->pConnectionCountEntry)
{
UlDecrementConnections(
&pHttpConnection->pConnectionCountEntry->CurConnections );
DEREFERENCE_CONNECTION_COUNT_ENTRY(pHttpConnection->pConnectionCountEntry);
pHttpConnection->pConnectionCountEntry = NULL;
}
//
// and now remove the connection's id
//
if (HTTP_IS_NULL_ID(&pHttpConnection->ConnectionId) == FALSE)
{
UlFreeOpaqueId(
pHttpConnection->ConnectionId,
UlOpaqueIdTypeHttpConnection
);
HTTP_SET_NULL_ID(&pHttpConnection->ConnectionId);
UL_DEREFERENCE_HTTP_CONNECTION(pHttpConnection);
}
UlReleaseResource( &pHttpConnection->Resource );
//
// Complete all WaitForDisconnect IRPs
//
UlCompleteAllWaitForDisconnect(pHttpConnection);
//
// and remove the ULTDI reference from the HTTP_CONNECTION
//
UL_DEREFERENCE_HTTP_CONNECTION(pHttpConnection);
} // UlConnectionDestroyedWorker
/***************************************************************************++
Routine Description:
Tries to establish a binding between a connection and an app pool
process. You can query these bindings with UlQueryProcessBinding.
Arguments:
pHttpConnection - the connection to bind
pAppPool - the app pool that owns the process (key for lookups)
pProcess - the process to bind to
--***************************************************************************/
NTSTATUS
UlBindConnectionToProcess(
IN PUL_HTTP_CONNECTION pHttpConnection,
IN PUL_APP_POOL_OBJECT pAppPool,
IN PUL_APP_POOL_PROCESS pProcess
)
{
NTSTATUS Status = STATUS_SUCCESS;
PUL_APOOL_PROC_BINDING pBinding = NULL;
KIRQL OldIrql;
//
// Sanity check
//
ASSERT( UL_IS_VALID_HTTP_CONNECTION( pHttpConnection ) );
ASSERT( pAppPool->NumberActiveProcesses > 1 || pProcess == NULL );
UlAcquireSpinLock(&pHttpConnection->BindingSpinLock, &OldIrql);
//
// see if there's already a binding object
//
pBinding = UlpFindProcBinding(pHttpConnection, pAppPool);
if (pBinding) {
if (pProcess) {
//
// modify the binding
//
pBinding->pProcess = pProcess;
} else {
//
// we're supposed to kill this binding
//
RemoveEntryList(&pBinding->BindingEntry);
UlpFreeProcBinding(pBinding);
}
} else {
if (pProcess) {
//
// create a new entry
//
pBinding = UlpCreateProcBinding(pAppPool, pProcess);
if (pBinding) {
InsertHeadList(
&pHttpConnection->BindingHead,
&pBinding->BindingEntry
);
} else {
Status = STATUS_INSUFFICIENT_RESOURCES;
}
}
}
UlReleaseSpinLock(&pHttpConnection->BindingSpinLock, OldIrql);
UlTrace(ROUTING, (
"ul!UlBindConnectionToProcess(\n"
" pHttpConnection = %p (%I64x)\n"
" pAppPool = %p\n"
" pProcess = %p )\n"
" Status = 0x%x\n",
pHttpConnection,
pHttpConnection->ConnectionId,
pAppPool,
pProcess,
Status
));
return Status;
}
/***************************************************************************++
Routine Description:
Removes an HTTP request from all lists and cleans up it's opaque id.
Arguments:
pRequest - the request to be unlinked
--***************************************************************************/
VOID
UlCleanupHttpConnection(
IN PUL_HTTP_CONNECTION pHttpConnection
)
{
PLIST_ENTRY pEntry;
ASSERT( UL_IS_VALID_INTERNAL_REQUEST( pHttpConnection->pRequest ) );
ASSERT( pHttpConnection->WaitingForResponse );
UlUnlinkHttpRequest( pHttpConnection->pRequest );
pHttpConnection->pRequest = NULL;
pHttpConnection->WaitingForResponse = 0;
UlTrace( PARSER, (
"***1 pConnection %p->WaitingForResponse = 0\n",
pHttpConnection
));
//
// Free the old request's buffers
//
ASSERT( UL_IS_VALID_REQUEST_BUFFER( pHttpConnection->pCurrentBuffer ) );
pEntry = pHttpConnection->BufferHead.Flink;
while (pEntry != &pHttpConnection->BufferHead)
{
PUL_REQUEST_BUFFER pBuffer;
//
// Get the object
//
pBuffer = CONTAINING_RECORD(
pEntry,
UL_REQUEST_BUFFER,
ListEntry
);
//
// did we reach the end?
//
if (pBuffer == pHttpConnection->pCurrentBuffer) {
break;
}
//
// remember the next one
//
pEntry = pEntry->Flink;
//
// unlink it
//
UlFreeRequestBuffer(pBuffer);
}
}
VOID
UlReferenceHttpConnection(
IN PVOID pObject
REFERENCE_DEBUG_FORMAL_PARAMS
)
{
LONG refCount;
PUL_HTTP_CONNECTION pHttpConnection = (PUL_HTTP_CONNECTION) pObject;
refCount = InterlockedIncrement( &pHttpConnection->RefCount );
WRITE_REF_TRACE_LOG2(
g_pHttpConnectionTraceLog,
pHttpConnection->pTraceLog,
REF_ACTION_REFERENCE_HTTP_CONNECTION,
refCount,
pHttpConnection,
pFileName,
LineNumber
);
UlTrace(
REFCOUNT, (
"ul!UlReferenceHttpConnection conn=%p refcount=%d\n",
pHttpConnection,
refCount)
);
} // UlReferenceHttpConnection
VOID
UlDereferenceHttpConnection(
IN PUL_HTTP_CONNECTION pHttpConnection
REFERENCE_DEBUG_FORMAL_PARAMS
)
{
LONG refCount;
ASSERT( pHttpConnection );
ASSERT( pHttpConnection->RefCount > 0 );
WRITE_REF_TRACE_LOG2(
g_pHttpConnectionTraceLog,
pHttpConnection->pTraceLog,
REF_ACTION_DEREFERENCE_HTTP_CONNECTION,
pHttpConnection->RefCount - 1,
pHttpConnection,
pFileName,
LineNumber
);
refCount = InterlockedDecrement( &pHttpConnection->RefCount );
UlTrace(
REFCOUNT, (
"ul!UlDereferenceHttpConnection conn=%p refcount=%d\n",
pHttpConnection,
refCount)
);
if (refCount == 0)
{
//
// now it's time to free the connection, we need to do
// this at lower'd irql, let's check it out
//
UL_CALL_PASSIVE(
&(pHttpConnection->WorkItem),
&UlDeleteHttpConnection
);
}
} // UlDereferenceHttpConnection
/***************************************************************************++
Routine Description:
Frees all resources associated with the specified HTTP_CONNECTION.
Arguments:
pWorkItem - Supplies the HTTP_CONNECTION to free.
--***************************************************************************/
VOID
UlDeleteHttpConnection(
IN PUL_WORK_ITEM pWorkItem
)
{
NTSTATUS Status;
PLIST_ENTRY pEntry;
PUL_HTTP_CONNECTION pHttpConnection;
KIRQL OldIrql;
//
// Sanity check.
//
PAGED_CODE();
pHttpConnection = CONTAINING_RECORD(
pWorkItem,
UL_HTTP_CONNECTION,
WorkItem
);
ASSERT( pHttpConnection != NULL &&
pHttpConnection->Signature == UL_HTTP_CONNECTION_POOL_TAG );
ASSERT(HTTP_IS_NULL_ID(&pHttpConnection->ConnectionId));
ASSERT(pHttpConnection->SendBufferedBytes == 0);
WRITE_REF_TRACE_LOG2(
g_pHttpConnectionTraceLog,
pHttpConnection->pTraceLog,
REF_ACTION_DESTROY_HTTP_CONNECTION,
0,
pHttpConnection,
__FILE__,
__LINE__
);
//
// The request is gone by now
//
ASSERT(pHttpConnection->pRequest == NULL);
//
// remove from Timeout Timer Wheel(c)
//
UlTimeoutRemoveTimerWheelEntry(
&pHttpConnection->TimeoutInfo
);
//
// delete the buffer list
//
pEntry = pHttpConnection->BufferHead.Flink;
while (pEntry != &pHttpConnection->BufferHead)
{
PUL_REQUEST_BUFFER pBuffer;
//
// Get the object
//
pBuffer = CONTAINING_RECORD(
pEntry,
UL_REQUEST_BUFFER,
ListEntry
);
//
// remember the next one
//
pEntry = pEntry->Flink;
//
// unlink it
//
UlFreeRequestBuffer(pBuffer);
}
ASSERT(IsListEmpty(&pHttpConnection->BufferHead));
//
// get rid of any bindings we're keeping
//
while (!IsListEmpty(&pHttpConnection->BindingHead))
{
PUL_APOOL_PROC_BINDING pBinding;
//
// Get the object
//
pEntry = RemoveHeadList(&pHttpConnection->BindingHead);
pBinding = CONTAINING_RECORD(
pEntry,
UL_APOOL_PROC_BINDING,
BindingEntry
);
ASSERT( IS_VALID_PROC_BINDING(pBinding) );
//
// free it
//
UlpFreeProcBinding(pBinding);
}
//
// get rid of our resource
//
Status = UlDeleteResource(&(pHttpConnection->Resource));
ASSERT(NT_SUCCESS(Status));
//
// Attempt to remove any QoS filter on this connection,
// if BWT is enabled.
//
if (pHttpConnection->BandwidthThrottlingEnabled)
{
UlTcDeleteFilter(pHttpConnection);
}
//
// Remove final (previous) site counter entry reference
// (matches reference in UlSendCachedResponse/UlDeliverHttpRequest)
//
if (pHttpConnection->pPrevSiteCounters)
{
UlDecSiteCounter(
pHttpConnection->pPrevSiteCounters,
HttpSiteCounterCurrentConns
);
DEREFERENCE_SITE_COUNTER_ENTRY(pHttpConnection->pPrevSiteCounters);
pHttpConnection->pPrevSiteCounters = NULL;
}
//
// free the memory
//
pHttpConnection->Signature = MAKE_FREE_SIGNATURE(
UL_HTTP_CONNECTION_POOL_TAG
);
//
// let go the UL_CONNECTION
//
DEREFERENCE_CONNECTION( pHttpConnection->pConnection );
} // UlDeleteHttpConnection
/***************************************************************************++
Routine Description:
Arguments:
Return Value:
NTSTATUS - Completion status.
--***************************************************************************/
NTSTATUS
UlpCreateHttpRequest(
IN PUL_HTTP_CONNECTION pHttpConnection,
OUT PUL_INTERNAL_REQUEST *ppRequest
)
{
PUL_INTERNAL_REQUEST pRequest = NULL;
//
// Sanity check.
//
PAGED_CODE();
pRequest = UlPplAllocateInternalRequest();
if (pRequest == NULL)
{
return STATUS_NO_MEMORY;
}
ASSERT( pRequest->Signature == MAKE_FREE_TAG(UL_INTERNAL_REQUEST_POOL_TAG) );
pRequest->Signature = UL_INTERNAL_REQUEST_POOL_TAG;
pRequest->RefCount = 1;
//
// the verb comes first
//
pRequest->ParseState = ParseVerbState;
//
// keep a reference to the connection.
//
UL_REFERENCE_HTTP_CONNECTION( pHttpConnection );
pRequest->pHttpConn = pHttpConnection;
pRequest->ConnectionId = pHttpConnection->ConnectionId;
pRequest->Secure = pHttpConnection->SecureConnection;
//
// init our LIST_ENTRY's
//
InitializeListHead( &pRequest->UnknownHeaderList );
InitializeListHead( &pRequest->IrpHead );
//
// We start with no header buffers appended nor Irps pending.
//
pRequest->HeadersAppended = FALSE;
pRequest->IrpsPending = FALSE;
//
// Reset the index to our default unknown header table to 0 as well as
// the unknown header count.
//
pRequest->NextUnknownHeaderIndex = 0;
pRequest->UnknownHeaderCount = 0;
//
// Create an opaque id for the request.
//
// We also make a copy of the id in case we need
// to see what the id was after the real one
// is freed and set to null.
//
HTTP_SET_NULL_ID(&pRequest->RequestId);
pRequest->RequestIdCopy = pRequest->RequestId;
//
// Grab the raw connection id off the UL_CONNECTION.
//
pRequest->RawConnectionId = pHttpConnection->pConnection->FilterInfo.ConnectionId;
//
// Initialize the header index table.
//
pRequest->HeaderIndex[0] = HttpHeaderRequestMaximum;
//
// Initialize the referenced buffers arrary.
//
pRequest->AllocRefBuffers = 1;
pRequest->UsedRefBuffers = 0;
pRequest->pRefBuffers = pRequest->pInlineRefBuffers;
//
// Initialize the UL_URL_CONFIG_GROUP_INFO.
//
UlpInitializeUrlInfo(&pRequest->ConfigInfo);
//
// Zero the remaining fields.
//
RtlZeroMemory(
(PUCHAR)pRequest + FIELD_OFFSET(UL_INTERNAL_REQUEST, HeaderValid),
sizeof(*pRequest) - FIELD_OFFSET(UL_INTERNAL_REQUEST, HeaderValid)
);
CREATE_REF_TRACE_LOG( pRequest->pTraceLog, 32 - REF_TRACE_OVERHEAD, 0 );
//
// return it
//
*ppRequest = pRequest;
UlTrace(REFCOUNT, (
"ul!UlpCreateHttpRequest req=%p refcount=%d\n",
pRequest,
pRequest->RefCount
));
return STATUS_SUCCESS;
} // UlpCreateHttpRequest
VOID
UlReferenceHttpRequest(
IN PVOID pObject
REFERENCE_DEBUG_FORMAL_PARAMS
)
{
LONG refCount;
PUL_INTERNAL_REQUEST pRequest = (PUL_INTERNAL_REQUEST) pObject;
refCount = InterlockedIncrement( &pRequest->RefCount );
WRITE_REF_TRACE_LOG2(
g_pHttpRequestTraceLog,
pRequest->pTraceLog,
REF_ACTION_REFERENCE_HTTP_REQUEST,
refCount,
pRequest,
pFileName,
LineNumber
);
UlTrace(
REFCOUNT, (
"ul!UlReferenceHttpRequest req=%p refcount=%d\n",
pRequest,
refCount)
);
} // UlReferenceHttpRequest
VOID
UlDereferenceHttpRequest(
IN PUL_INTERNAL_REQUEST pRequest
REFERENCE_DEBUG_FORMAL_PARAMS
)
{
LONG refCount;
WRITE_REF_TRACE_LOG2(
g_pHttpRequestTraceLog,
pRequest->pTraceLog,
REF_ACTION_DEREFERENCE_HTTP_REQUEST,
pRequest->RefCount - 1,
pRequest,
pFileName,
LineNumber
);
refCount = InterlockedDecrement( &pRequest->RefCount );
UlTrace(
REFCOUNT, (
"ul!UlDereferenceHttpRequest req=%p refcount=%d\n",
pRequest,
refCount)
);
if (refCount == 0)
{
UL_CALL_PASSIVE(
&pRequest->WorkItem,
&UlpFreeHttpRequest
);
}
} // UlDereferenceHttpRequest
/***************************************************************************++
Routine Description:
Cancels all pending http io.
Arguments:
pRequest - Supplies the HTTP_REQUEST.
--***************************************************************************/
VOID
UlCancelRequestIo(
IN PUL_INTERNAL_REQUEST pRequest
)
{
//
// Sanity check.
//
PAGED_CODE();
ASSERT(UL_IS_VALID_INTERNAL_REQUEST(pRequest));
//
// Mark the request as InCleanup, so additional IRPs
// on the request will not be queued.
//
pRequest->InCleanup = 1;
//
// tank all pending io on this request
//
while (IsListEmpty(&pRequest->IrpHead) == FALSE)
{
PLIST_ENTRY pEntry;
PIRP pIrp;
//
// Pop it off the list.
//
pEntry = RemoveHeadList(&pRequest->IrpHead);
pEntry->Blink = pEntry->Flink = NULL;
pIrp = CONTAINING_RECORD(pEntry, IRP, Tail.Overlay.ListEntry);
ASSERT(IS_VALID_IRP(pIrp));
//
// pop the cancel routine
//
if (IoSetCancelRoutine(pIrp, NULL) == NULL)
{
//
// IoCancelIrp pop'd it first
//
// ok to just ignore this irp, it's been pop'd off the queue
// and will be completed in the cancel routine.
//
// keep looping
//
pIrp = NULL;
}
else
{
PUL_INTERNAL_REQUEST pIrpRequest;
//
// cancel it. even if pIrp->Cancel == TRUE we are supposed to
// complete it, our cancel routine will never run.
//
pIrpRequest = (PUL_INTERNAL_REQUEST)(
IoGetCurrentIrpStackLocation(pIrp)->
Parameters.DeviceIoControl.Type3InputBuffer
);
ASSERT(pIrpRequest == pRequest);
UL_DEREFERENCE_INTERNAL_REQUEST(pIrpRequest);
IoGetCurrentIrpStackLocation(pIrp)->
Parameters.DeviceIoControl.Type3InputBuffer = NULL;
pIrp->IoStatus.Status = STATUS_CANCELLED;
pIrp->IoStatus.Information = 0;
UlCompleteRequest(pIrp, g_UlPriorityBoost);
pIrp = NULL;
}
} // while (IsListEmpty(&pRequest->IrpHead) == FALSE)
} // UlCancelRequestIo
/***************************************************************************++
Routine Description:
Frees all resources associated with the specified UL_INTERNAL_REQUEST.
Arguments:
pRequest - Supplies the UL_INTERNAL_REQUEST to free.
--***************************************************************************/
VOID
UlpFreeHttpRequest(
IN PUL_WORK_ITEM pWorkItem
)
{
PUL_INTERNAL_REQUEST pRequest;
PLIST_ENTRY pEntry;
ULONG i;
NTSTATUS Status;
//
// Sanity check.
//
PAGED_CODE();
pRequest = CONTAINING_RECORD(
pWorkItem,
UL_INTERNAL_REQUEST,
WorkItem
);
//
// our opaque id should already be free'd (UlDeleteOpaqueIds)
//
ASSERT(HTTP_IS_NULL_ID(&pRequest->RequestId));
//
// free any known header buffers allocated
//
if (pRequest->HeadersAppended)
{
for (i = 0; i < HttpHeaderRequestMaximum; i++)
{
HTTP_HEADER_ID HeaderId = (HTTP_HEADER_ID)pRequest->HeaderIndex[i];
if (HeaderId == HttpHeaderRequestMaximum)
{
break;
}
ASSERT( pRequest->HeaderValid[HeaderId] );
if (pRequest->Headers[HeaderId].OurBuffer == 1)
{
UL_FREE_POOL(
pRequest->Headers[HeaderId].pHeader,
HEADER_VALUE_POOL_TAG
);
}
}
//
// and any unknown header buffers allocated
//
while (IsListEmpty(&pRequest->UnknownHeaderList) == FALSE)
{
PUL_HTTP_UNKNOWN_HEADER pUnknownHeader;
pEntry = RemoveHeadList(&pRequest->UnknownHeaderList);
pUnknownHeader = CONTAINING_RECORD(
pEntry,
UL_HTTP_UNKNOWN_HEADER,
List
);
if (pUnknownHeader->HeaderValue.OurBuffer == 1)
{
UL_FREE_POOL(
pUnknownHeader->HeaderValue.pHeader,
HEADER_VALUE_POOL_TAG
);
}
//
// Free the header structure
//
if (pUnknownHeader->HeaderValue.ExternalAllocated == 1)
{
UL_FREE_POOL(
pUnknownHeader,
UL_HTTP_UNKNOWN_HEADER_POOL_TAG
);
}
}
}
//
// there better not be any pending io, it would have been cancelled either
// in UlDeleteHttpConnection or in UlDetachProcessFromAppPool .
//
ASSERT(IsListEmpty(&pRequest->IrpHead));
//
// dereferenc request buffers.
//
for (i = 0; i < pRequest->UsedRefBuffers; i++)
{
ASSERT( UL_IS_VALID_REQUEST_BUFFER(pRequest->pRefBuffers[i]) );
UL_DEREFERENCE_REQUEST_BUFFER(pRequest->pRefBuffers[i]);
}
if (pRequest->AllocRefBuffers > 1)
{
UL_FREE_POOL(
pRequest->pRefBuffers,
UL_REF_REQUEST_BUFFER_POOL_TAG
);
}
//
// free any url that was allocated
//
if (pRequest->CookedUrl.pUrl != NULL)
{
if (pRequest->CookedUrl.pUrl != pRequest->pUrlBuffer)
{
UL_FREE_POOL(pRequest->CookedUrl.pUrl, URL_POOL_TAG);
}
}
//
// free any config group info
//
ASSERT( IS_VALID_URL_CONFIG_GROUP_INFO(&pRequest->ConfigInfo) );
ASSERT( pRequest->pHttpConn );
//
// Perf counters
// NOTE: Assumes cache & non-cache paths both go through here
// NOTE: If connection is refused the pConnectionCountEntry will be NULL
//
if (pRequest->ConfigInfo.pSiteCounters &&
pRequest->pHttpConn->pConnectionCountEntry)
{
PUL_SITE_COUNTER_ENTRY pCtr = pRequest->ConfigInfo.pSiteCounters;
UlAddSiteCounter64(
pCtr,
HttpSiteCounterBytesSent,
pRequest->BytesSent
);
UlAddSiteCounter64(
pCtr,
HttpSiteCounterBytesReceived,
pRequest->BytesReceived
);
UlAddSiteCounter64(
pCtr,
HttpSiteCounterBytesTransfered,
(pRequest->BytesSent + pRequest->BytesReceived)
);
}
//
// Release all the references from the UL_URL_CONFIG_GROUP_INFO.
//
UlpConfigGroupInfoRelease(&pRequest->ConfigInfo);
//
// release our reference to the connection
//
UL_DEREFERENCE_HTTP_CONNECTION( pRequest->pHttpConn );
//
// Free the object buffer
//
ASSERT(pRequest->Signature == UL_INTERNAL_REQUEST_POOL_TAG);
pRequest->Signature = MAKE_FREE_TAG(UL_INTERNAL_REQUEST_POOL_TAG);
DESTROY_REF_TRACE_LOG( pRequest->pTraceLog );
UlPplFreeInternalRequest( pRequest );
}
/***************************************************************************++
Routine Description:
Allocates and initializes a new UL_REQUEST_BUFFER.
Arguments:
BufferSize - size of the new buffer in bytes
--***************************************************************************/
PUL_REQUEST_BUFFER
UlCreateRequestBuffer(
ULONG BufferSize,
ULONG BufferNumber
)
{
PUL_REQUEST_BUFFER pBuffer;
BufferSize = MAX(BufferSize, DEFAULT_MAX_REQUEST_BUFFER_SIZE);
if (BufferSize > DEFAULT_MAX_REQUEST_BUFFER_SIZE)
{
BufferSize = (ULONG) MAX(
BufferSize,
PAGE_SIZE - ALIGN_UP(sizeof(UL_REQUEST_BUFFER),PVOID)
);
pBuffer = UL_ALLOCATE_STRUCT_WITH_SPACE(
NonPagedPool,
UL_REQUEST_BUFFER,
BufferSize,
UL_REQUEST_BUFFER_POOL_TAG
);
}
else
{
pBuffer = UlPplAllocateRequestBuffer();
}
if (pBuffer == NULL)
{
return NULL;
}
RtlZeroMemory(
(PCHAR)pBuffer + sizeof(SINGLE_LIST_ENTRY),
sizeof(UL_REQUEST_BUFFER) - sizeof(SINGLE_LIST_ENTRY)
);
pBuffer->Signature = UL_REQUEST_BUFFER_POOL_TAG;
UlTrace(HTTP_IO, (
"ul!UlCreateRequestBuffer buff=%p num=%d size=%d\n",
pBuffer,
BufferNumber,
BufferSize
));
pBuffer->RefCount = 1;
pBuffer->AllocBytes = BufferSize;
pBuffer->BufferNumber = BufferNumber;
return pBuffer;
}
/***************************************************************************++
Routine Description:
Removes a request buffer from the buffer list and destroys it.
Arguments:
pBuffer - the buffer to be deleted
--***************************************************************************/
VOID
UlFreeRequestBuffer(
PUL_REQUEST_BUFFER pBuffer
)
{
ASSERT( UL_IS_VALID_REQUEST_BUFFER( pBuffer ) );
UlTrace(HTTP_IO, (
"ul!UlFreeRequestBuffer(pBuffer = %p, Num = %d)\n",
pBuffer,
pBuffer->BufferNumber
));
if (pBuffer->ListEntry.Flink != NULL) {
RemoveEntryList(&pBuffer->ListEntry);
pBuffer->ListEntry.Blink = pBuffer->ListEntry.Flink = NULL;
}
UL_DEREFERENCE_REQUEST_BUFFER(pBuffer);
}
//
// Private functions.
//
/***************************************************************************++
Routine Description:
Retrieves a binding set with UlBindConnectionToProcess.
Arguments:
pHttpConnection - the connection to query
pAppPool - the key to use for the lookup
Return Values:
A pointer to the bound process if one was found. NULL otherwise.
--***************************************************************************/
PUL_APP_POOL_PROCESS
UlQueryProcessBinding(
IN PUL_HTTP_CONNECTION pHttpConnection,
IN PUL_APP_POOL_OBJECT pAppPool
)
{
PUL_APOOL_PROC_BINDING pBinding;
PUL_APP_POOL_PROCESS pProcess = NULL;
KIRQL OldIrql;
//
// Sanity check
//
ASSERT( UL_IS_VALID_HTTP_CONNECTION( pHttpConnection ) );
UlAcquireSpinLock(&pHttpConnection->BindingSpinLock, &OldIrql);
pBinding = UlpFindProcBinding(pHttpConnection, pAppPool);
if (pBinding) {
pProcess = pBinding->pProcess;
}
UlReleaseSpinLock(&pHttpConnection->BindingSpinLock, OldIrql);
return pProcess;
}
/***************************************************************************++
Routine Description:
Allocates and builds a UL_APOOL_PROC_BINDING object.
Arguments:
pAppPool - the lookup key
pProcess - the binding
Return Values:
a pointer to the allocated object, or NULL on failure
--***************************************************************************/
PUL_APOOL_PROC_BINDING
UlpCreateProcBinding(
IN PUL_APP_POOL_OBJECT pAppPool,
IN PUL_APP_POOL_PROCESS pProcess
)
{
PUL_APOOL_PROC_BINDING pBinding;
ASSERT( pAppPool->NumberActiveProcesses > 1 );
//
// CODEWORK: lookaside
//
pBinding = UL_ALLOCATE_STRUCT(
NonPagedPool,
UL_APOOL_PROC_BINDING,
UL_APOOL_PROC_BINDING_POOL_TAG
);
if (pBinding) {
pBinding->Signature = UL_APOOL_PROC_BINDING_POOL_TAG;
pBinding->pAppPool = pAppPool;
pBinding->pProcess = pProcess;
UlTrace(ROUTING, (
"ul!UlpCreateProcBinding(\n"
" pAppPool = %p\n"
" pProcess = %p )\n"
" pBinding = %p\n",
pAppPool,
pProcess,
pBinding
));
}
return pBinding;
}
/***************************************************************************++
Routine Description:
Gets rid of a proc binding
Arguments:
pBinding - the binding to get rid of
--***************************************************************************/
VOID
UlpFreeProcBinding(
IN PUL_APOOL_PROC_BINDING pBinding
)
{
UL_FREE_POOL_WITH_SIG(pBinding, UL_APOOL_PROC_BINDING_POOL_TAG);
}
/***************************************************************************++
Routine Description:
Searches a connection's list of bindings for one that has the right
app pool key
Arguments:
pHttpConnection - the connection to search
pAppPool - the key
Return Values:
The binding if found or NULL if not found.
--***************************************************************************/
PUL_APOOL_PROC_BINDING
UlpFindProcBinding(
IN PUL_HTTP_CONNECTION pHttpConnection,
IN PUL_APP_POOL_OBJECT pAppPool
)
{
PLIST_ENTRY pEntry;
PUL_APOOL_PROC_BINDING pBinding = NULL;
//
// Sanity check
//
ASSERT( UL_IS_VALID_HTTP_CONNECTION(pHttpConnection) );
pEntry = pHttpConnection->BindingHead.Flink;
while (pEntry != &pHttpConnection->BindingHead) {
pBinding = CONTAINING_RECORD(
pEntry,
UL_APOOL_PROC_BINDING,
BindingEntry
);
ASSERT( IS_VALID_PROC_BINDING(pBinding) );
if (pBinding->pAppPool == pAppPool) {
//
// got it!
//
break;
}
pEntry = pEntry->Flink;
}
return pBinding;
}
/***************************************************************************++
Routine Description:
Removes an HTTP request from all lists and cleans up it's opaque id.
Arguments:
pRequest - the request to be unlinked
--***************************************************************************/
VOID
UlUnlinkHttpRequest(
IN PUL_INTERNAL_REQUEST pRequest
)
{
//
// if we got far enough to deliver the request then
// unlink it from the app pool. this needs to happen here because the
// queue'd IRPs keep references to the UL_INTERNAL_REQUEST objects.
// this kicks their references off and allows them to delete.
//
if (pRequest->ConfigInfo.pAppPool)
{
UlUnlinkRequestFromProcess(pRequest->ConfigInfo.pAppPool, pRequest);
}
//
// cancel i/o
//
if (pRequest->IrpsPending)
{
UlCancelRequestIo(pRequest);
}
else
{
//
// Mark the request as InCleanup, so additional IRPs
// on the request will not be queued.
//
// Normally UlCancelRequestIo sets this flag, but we
// have to do it here since we're optimizing that
// call away.
//
pRequest->InCleanup = 1;
}
//
// delete its opaque id
//
if (HTTP_IS_NULL_ID(&pRequest->RequestId) == FALSE)
{
UlFreeRequestId(pRequest);
HTTP_SET_NULL_ID(&pRequest->RequestId);
//
// it is still referenced by this connection
//
ASSERT(pRequest->RefCount > 1);
UL_DEREFERENCE_INTERNAL_REQUEST(pRequest);
}
//
// deref it
//
UL_DEREFERENCE_INTERNAL_REQUEST(pRequest);
}
//
// Following code has been implemented for Global & Site specific connection
// limits feature. If enforced, incoming connections get refused when they
// exceed the existing limit. Control channel & config group (re)sets this
// values whereas httprcv and sendresponse updates the limits for incoming
// requests & connections. A seperate connection_count_entry structure keeps
// track of the limits per site. Global limits are tracked by the global
// variables g_MaxConnections & g_CurrentConnections same API has been used
// for both purposes.
//
/***************************************************************************++
Routine Description:
Allocates a connection count entry which will hold the site specific
connection count info. Get called by cgroup when Config group is
attempting to allocate a connection count entry.
Arguments:
pConfigGroup - To receive the newly allocated count entry
MaxConnections - The maximum allowed connections
--***************************************************************************/
NTSTATUS
UlCreateConnectionCountEntry(
IN OUT PUL_CONFIG_GROUP_OBJECT pConfigGroup,
IN ULONG MaxConnections
)
{
PUL_CONNECTION_COUNT_ENTRY pEntry;
// Sanity check.
PAGED_CODE();
ASSERT(IS_VALID_CONFIG_GROUP(pConfigGroup));
// Alloc new struct from Paged Pool
pEntry = UL_ALLOCATE_STRUCT(
PagedPool,
UL_CONNECTION_COUNT_ENTRY,
UL_CONNECTION_COUNT_ENTRY_POOL_TAG
);
if (!pEntry)
{
UlTrace(LIMITS,
("Http!UlCreateConnectionCountEntry: OutOfMemory pConfigGroup %p\n",
pConfigGroup
));
return STATUS_NO_MEMORY;
}
pEntry->Signature = UL_CONNECTION_COUNT_ENTRY_POOL_TAG;
pEntry->RefCount = 1;
pEntry->MaxConnections = MaxConnections;
pEntry->CurConnections = 0;
// Update cgroup pointer
ASSERT( pConfigGroup->pConnectionCountEntry == NULL );
pConfigGroup->pConnectionCountEntry = pEntry;
UlTrace(LIMITS,
("Http!UlCreateConnectionCountEntry: pNewEntry %p, pConfigGroup %p, Max %d\n",
pEntry,
pConfigGroup,
MaxConnections
));
return STATUS_SUCCESS;
} // UlCreateConnectionCountEntry
/***************************************************************************++
Routine Description:
increments the refcount for ConnectionCountEntry.
Arguments:
pEntry - the object to increment.
--***************************************************************************/
VOID
UlReferenceConnectionCountEntry(
IN PUL_CONNECTION_COUNT_ENTRY pEntry
REFERENCE_DEBUG_FORMAL_PARAMS
)
{
LONG refCount;
//
// Sanity check.
//
PAGED_CODE();
ASSERT( IS_VALID_CONNECTION_COUNT_ENTRY(pEntry) );
refCount = InterlockedIncrement( &pEntry->RefCount );
//
// Tracing.
//
WRITE_REF_TRACE_LOG(
g_pConnectionCountTraceLog,
REF_ACTION_REFERENCE_CONNECTION_COUNT_ENTRY,
refCount,
pEntry,
pFileName,
LineNumber
);
UlTrace(
REFCOUNT,
("Http!UlReferenceConnectionCountEntry pEntry=%p refcount=%d\n",
pEntry,
refCount
));
} // UlReferenceConnectionCountEntry
/***************************************************************************++
Routine Description:
decrements the refcount. if it hits 0, destruct's ConnectionCountEntry
Arguments:
pEntry - the object to decrement.
--***************************************************************************/
VOID
UlDereferenceConnectionCountEntry(
IN PUL_CONNECTION_COUNT_ENTRY pEntry
REFERENCE_DEBUG_FORMAL_PARAMS
)
{
LONG refCount;
NTSTATUS Status;
//
// Sanity check.
//
PAGED_CODE();
ASSERT( IS_VALID_CONNECTION_COUNT_ENTRY(pEntry) );
refCount = InterlockedDecrement( &pEntry->RefCount );
//
// Tracing.
//
WRITE_REF_TRACE_LOG(
g_pConnectionCountTraceLog,
REF_ACTION_DEREFERENCE_CONNECTION_COUNT_ENTRY,
refCount,
pEntry,
pFileName,
LineNumber
);
UlTrace(
REFCOUNT,
("Http!UlDereferenceConnectionCountEntry pEntry=%p refcount=%d\n",
pEntry,
refCount
));
//
// Cleanup the memory and do few checks !
//
if ( refCount == 0 )
{
// Make sure no connection on the site
ASSERT( 0 == pEntry->CurConnections );
UlTrace(
LIMITS,
("Http!UlDereferenceConnectionCountEntry: Removing pEntry=%p\n",
pEntry
));
// Release memory
UL_FREE_POOL_WITH_SIG(pEntry,UL_CONNECTION_COUNT_ENTRY_POOL_TAG);
}
} // UlDereferenceConnectionCountEntry
/***************************************************************************++
Routine Description:
This
function set the maximum limit. Maximum allowed number of connections
at any time by the active control channel.
Arguments:
MaxConnections - Maximum allowed number of connections
Return Value:
Old Max Connection count
--***************************************************************************/
ULONG
UlSetMaxConnections(
IN OUT PULONG pCurMaxConnection,
IN ULONG NewMaxConnection
)
{
ULONG OldMaxConnection;
UlTrace(LIMITS,
("Http!UlSetMaxConnections pCurMaxConnection=%p NewMaxConnection=%d\n",
pCurMaxConnection,
NewMaxConnection
));
//
// By setting this we are not forcing the existing connections to
// termination but this number will be effective for all newcoming
// connections, as soon as the atomic operation completed.
//
OldMaxConnection = (ULONG) InterlockedExchange((LONG *) pCurMaxConnection,
(LONG) NewMaxConnection
);
return OldMaxConnection;
} // UlSetMaxConnections
/***************************************************************************++
Routine Description:
Control channel uses this function to set or reset the global connection
limits.
--***************************************************************************/
VOID
UlSetGlobalConnectionLimit(
IN ULONG Limit
)
{
UlSetMaxConnections( &g_MaxConnections, Limit );
UlTrace(LIMITS,("Http!UlSetGlobalConnectionLimit: To %d\n", Limit));
} // ULSetGlobalConnectionLimit
/***************************************************************************++
Routine Description:
Control channel uses this function to initialize the global connection
limits. Assuming the existince of one and only one active control channel
this globals get set only once during init. But could be set again later
because of a reconfig.
--***************************************************************************/
NTSTATUS
UlInitGlobalConnectionLimits(
VOID
)
{
ASSERT(!g_InitGlobalConnections);
if (!g_InitGlobalConnections)
{
g_MaxConnections = HTTP_LIMIT_INFINITE;
g_CurrentConnections = 0;
UlTrace(LIMITS,
("Http!UlInitGlobalConnectionLimits: Init g_MaxConnections %d,"
"g_CurrentConnections %d\n",
g_MaxConnections,
g_CurrentConnections
));
g_InitGlobalConnections = TRUE;
}
return STATUS_SUCCESS;
} // UlInitGlobalConnectionLimits
/***************************************************************************++
Routine Description:
Wrapper around the Accept Connection for global connections
--***************************************************************************/
BOOLEAN
UlAcceptGlobalConnection(
VOID
)
{
return UlAcceptConnection( &g_MaxConnections, &g_CurrentConnections );
} // UlAcceptGlobalConnection
/***************************************************************************++
Routine Description:
This function checks if we are allowed to accept the incoming connection
based on the number enforced by the control channel.
Return value:
Decision for the newcoming connection either ACCEPT or REJECT as boolean
--***************************************************************************/
BOOLEAN
UlAcceptConnection(
IN PULONG pMaxConnection,
IN OUT PULONG pCurConnection
)
{
ULONG LocalCur;
ULONG LocalCurPlusOne;
ULONG LocalMax;
do
{
//
// Capture the Max & Cur. Do the comparison. If in the limit
// attempt to increment the connection count by ensuring nobody
// else did it before us.
//
LocalMax = *((volatile ULONG *) pMaxConnection);
LocalCur = *((volatile ULONG *) pCurConnection);
//
// Its greater than or equal because Max may get updated to
// a smaller number on-the-fly and we end up having current
// connections greater than the maximum allowed.
// NOTE: HTTP_LIMIT_INFINITE has been picked as (ULONG)-1 so
// following comparison won't reject for the infinite case.
//
if ( LocalCur >= LocalMax )
{
//
// We are already at the limit refuse it.
//
UlTrace(LIMITS,
("Http!UlAcceptConnection REFUSE pCurConnection=%p[%d]"
"pMaxConnection=%p[%d]\n",
pCurConnection,LocalCur,
pMaxConnection,LocalMax
));
return FALSE;
}
//
// Either the limit was infinite or conn count was not exceeding
// the limit. Lets attempt to increment the count and accept the
// connection in the while statement.
//
LocalCurPlusOne = LocalCur + 1;
}
while ( LocalCur != (ULONG) InterlockedCompareExchange(
(LONG *) pCurConnection,
(LONG) LocalCurPlusOne,
(LONG) LocalCur
) );
//
// Successfully incremented the counter. Let it go with success.
//
UlTrace(LIMITS,
("Http!UlAcceptConnection ACCEPT pCurConnection=%p[%d]"
"pMaxConnection=%p[%d]\n",
pCurConnection,LocalCur,
pMaxConnection,LocalMax
));
return TRUE;
} // UlAcceptConnection
/***************************************************************************++
Routine Description:
Everytime a disconnection happens we will decrement the count here.
Return Value:
The newly decremented value
--***************************************************************************/
LONG
UlDecrementConnections(
IN OUT PULONG pCurConnection
)
{
LONG NewConnectionCount;
NewConnectionCount = InterlockedDecrement( (LONG *) pCurConnection );
ASSERT( NewConnectionCount >= 0 );
return NewConnectionCount;
} // UlDecrementConnections
/***************************************************************************++
Routine Description:
For cache & non-cache hits this function get called. Connection resource
has assumed to be acquired at this time. The function decide to accept or
reject the request by looking at the corresponding count entries.
Arguments:
pConnection - For getting the previous site's connection count entry
pConfigInfo - Holds a pointer to newly received request's site's
connection count entry.
Return Value:
Shows reject or accept
--***************************************************************************/
BOOLEAN
UlCheckSiteConnectionLimit(
IN OUT PUL_HTTP_CONNECTION pConnection,
IN OUT PUL_URL_CONFIG_GROUP_INFO pConfigInfo
)
{
BOOLEAN Accept;
PUL_CONNECTION_COUNT_ENTRY pConEntry;
PUL_CONNECTION_COUNT_ENTRY pCIEntry;
if (pConfigInfo->pMaxConnections == NULL || pConfigInfo->pConnectionCountEntry == NULL)
{
//
// No connection count entry for the new request perhaps WAS never
// set this before otherwise its a problem.
//
UlTrace(LIMITS,
("Http!UlCheckSiteConnectionLimit: NO LIMIT pConnection=%p pConfigInfo=%p\n",
pConnection,
pConfigInfo
));
return TRUE;
}
ASSERT(IS_VALID_CONNECTION_COUNT_ENTRY(pConfigInfo->pConnectionCountEntry));
pCIEntry = pConfigInfo->pConnectionCountEntry;
pConEntry = pConnection->pConnectionCountEntry;
Accept = FALSE;
//
// Make a check on the connection limit of the site. Refuse the request
// if the limit is exceded.
//
if (pConEntry)
{
ASSERT(IS_VALID_CONNECTION_COUNT_ENTRY(pConEntry));
//
// For consecutive requests we decrement the previous site's connection count
// before proceeding to the decision on the newcoming request,if the two sides
// are not same.That means we assume this connection on site x until (if ever)
// a request changes this to site y. Naturally until the first request arrives
// and successfully parsed, the connection does not count to any specific site
//
if (pConEntry != pCIEntry)
{
UlDecrementConnections(&pConEntry->CurConnections);
DEREFERENCE_CONNECTION_COUNT_ENTRY(pConEntry);
//
// We do not increment the connection here yet, since the AcceptConnection
// will decide and do that.
//
REFERENCE_CONNECTION_COUNT_ENTRY(pCIEntry);
pConnection->pConnectionCountEntry = pCIEntry;
}
else
{
//
// There was an old entry, that means this connection already got through.
// And the entry hasn't been changed with this new request.
// No need to check again, our design is not forcing existing connections
// to disconnect.
//
return TRUE;
}
}
else
{
REFERENCE_CONNECTION_COUNT_ENTRY(pCIEntry);
pConnection->pConnectionCountEntry = pCIEntry;
}
Accept = UlAcceptConnection(
&pConnection->pConnectionCountEntry->MaxConnections,
&pConnection->pConnectionCountEntry->CurConnections
);
if (Accept == FALSE)
{
// We are going to refuse. Let our ref & refcount go away
// on the connection etry to prevent the possible incorrect
// decrement in the UlConnectionDestroyedWorker. If refused
// current connection hasn't been incremented in the accept
// connection. Perf counters also depends on the fact that
// pConnectionCountEntry will be Null when con got refused
DEREFERENCE_CONNECTION_COUNT_ENTRY(pConnection->pConnectionCountEntry);
pConnection->pConnectionCountEntry = NULL;
}
return Accept;
} // UlCheckSiteConnectionLimit
#ifdef REUSE_CONNECTION_OPAQUE_ID
/***************************************************************************++
Routine Description:
Allocate a request opaque ID.
Return Value:
NT_SUCCESS
--***************************************************************************/
NTSTATUS
UlAllocateRequestId(
IN PUL_INTERNAL_REQUEST pRequest
)
{
PUL_HTTP_CONNECTION pConnection;
KIRQL OldIrql;
PAGED_CODE();
pConnection = pRequest->pHttpConn;
UlAcquireSpinLock(&pConnection->RequestIdSpinLock, &OldIrql);
pConnection->pRequestIdContext = pRequest;
UlReleaseSpinLock(&pConnection->RequestIdSpinLock, OldIrql);
pRequest->RequestId = pConnection->ConnectionId;
return STATUS_SUCCESS;
}
/***************************************************************************++
Routine Description:
Free a request opaque ID.
Return Value:
VOID
--***************************************************************************/
VOID
UlFreeRequestId(
IN PUL_INTERNAL_REQUEST pRequest
)
{
PUL_HTTP_CONNECTION pConnection;
KIRQL OldIrql;
PAGED_CODE();
pConnection = pRequest->pHttpConn;
UlAcquireSpinLock(&pConnection->RequestIdSpinLock, &OldIrql);
pConnection->pRequestIdContext = NULL;
UlReleaseSpinLock(&pConnection->RequestIdSpinLock, OldIrql);
return;
}
/***************************************************************************++
Routine Description:
Get a request from the connection opaque ID.
Return Value:
PUL_INTERNAL_REQUEST
--***************************************************************************/
PUL_INTERNAL_REQUEST
UlGetRequestFromId(
IN HTTP_REQUEST_ID RequestId
)
{
PUL_HTTP_CONNECTION pConnection;
PUL_INTERNAL_REQUEST pRequest;
KIRQL OldIrql;
PAGED_CODE();
pConnection = UlGetConnectionFromId(RequestId);
if (pConnection != NULL)
{
UlAcquireSpinLock(&pConnection->RequestIdSpinLock, &OldIrql);
pRequest = pConnection->pRequestIdContext;
if (pRequest != NULL)
{
UL_REFERENCE_INTERNAL_REQUEST(pRequest);
}
UlReleaseSpinLock(&pConnection->RequestIdSpinLock, OldIrql);
UL_DEREFERENCE_HTTP_CONNECTION(pConnection);
return pRequest;
}
return NULL;
}
#endif