mirror of https://github.com/tongzx/nt5src
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1520 lines
43 KiB
1520 lines
43 KiB
/*++
|
|
|
|
Copyright (c) 1992-1999 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
blkendp.c
|
|
|
|
Abstract:
|
|
|
|
This module contains allocate, free, close, reference, and dereference
|
|
routines for AFD endpoints.
|
|
|
|
Author:
|
|
|
|
David Treadwell (davidtr) 10-Mar-1992
|
|
|
|
Revision History:
|
|
Vadim Eydelman (vadime) 1999 - Don't attach to system proces, use system handles instead
|
|
Delayed acceptance endpoints.
|
|
|
|
--*/
|
|
|
|
#include "afdp.h"
|
|
|
|
VOID
|
|
AfdFreeEndpointResources (
|
|
PAFD_ENDPOINT endpoint
|
|
);
|
|
|
|
VOID
|
|
AfdFreeEndpoint (
|
|
IN PVOID Context
|
|
);
|
|
|
|
PAFD_ENDPOINT
|
|
AfdReuseEndpoint (
|
|
VOID
|
|
);
|
|
|
|
#ifdef ALLOC_PRAGMA
|
|
#pragma alloc_text( PAGE, AfdAllocateEndpoint )
|
|
#pragma alloc_text( PAGE, AfdFreeEndpointResources )
|
|
#pragma alloc_text( PAGE, AfdFreeEndpoint )
|
|
#pragma alloc_text( PAGE, AfdReuseEndpoint )
|
|
#pragma alloc_text( PAGE, AfdGetTransportInfo )
|
|
#pragma alloc_text( PAGEAFD, AfdRefreshEndpoint )
|
|
#pragma alloc_text( PAGEAFD, AfdDereferenceEndpoint )
|
|
#if REFERENCE_DEBUG
|
|
#pragma alloc_text( PAGEAFD, AfdReferenceEndpoint )
|
|
#endif
|
|
#pragma alloc_text( PAGEAFD, AfdFreeQueuedConnections )
|
|
#endif
|
|
|
|
|
|
NTSTATUS
|
|
AfdAllocateEndpoint (
|
|
OUT PAFD_ENDPOINT * NewEndpoint,
|
|
IN PUNICODE_STRING TransportDeviceName,
|
|
IN LONG GroupID
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Allocates and initializes a new AFD endpoint structure.
|
|
|
|
Arguments:
|
|
|
|
NewEndpoint - Receives a pointer to the new endpoint structure if
|
|
successful.
|
|
|
|
TransportDeviceName - the name of the TDI transport provider
|
|
corresponding to the endpoint structure.
|
|
|
|
GroupID - Identifies the group ID for the new endpoint.
|
|
|
|
Return Value:
|
|
|
|
NTSTATUS - The completion status.
|
|
|
|
--*/
|
|
|
|
{
|
|
PAFD_ENDPOINT endpoint;
|
|
PAFD_TRANSPORT_INFO transportInfo = NULL;
|
|
NTSTATUS status;
|
|
AFD_GROUP_TYPE groupType;
|
|
|
|
PAGED_CODE( );
|
|
|
|
DEBUG *NewEndpoint = NULL;
|
|
|
|
if ( TransportDeviceName != NULL ) {
|
|
//
|
|
// First, make sure that the transport device name is stored globally
|
|
// for AFD. Since there will typically only be a small number of
|
|
// transport device names, we store the name strings once globally
|
|
// for access by all endpoints.
|
|
//
|
|
|
|
status = AfdGetTransportInfo( TransportDeviceName, &transportInfo );
|
|
|
|
//
|
|
// If transport device is not activated, we'll try again during bind
|
|
//
|
|
if ( !NT_SUCCESS (status) &&
|
|
(status!=STATUS_OBJECT_NAME_NOT_FOUND) &&
|
|
(status!=STATUS_OBJECT_PATH_NOT_FOUND) &&
|
|
(status!=STATUS_NO_SUCH_DEVICE) ) {
|
|
//
|
|
// Dereference transport info structure if one was created for us.
|
|
// (Should not happen in current implementation).
|
|
//
|
|
ASSERT (transportInfo==NULL);
|
|
return status;
|
|
}
|
|
|
|
ASSERT (transportInfo!=NULL);
|
|
}
|
|
|
|
|
|
//
|
|
// Validate the incoming group ID, allocate a new one if necessary.
|
|
//
|
|
|
|
if( AfdGetGroup( &GroupID, &groupType ) ) {
|
|
PEPROCESS process = IoGetCurrentProcess ();
|
|
|
|
|
|
status = PsChargeProcessPoolQuota(
|
|
process,
|
|
NonPagedPool,
|
|
sizeof (AFD_ENDPOINT)
|
|
);
|
|
|
|
if (!NT_SUCCESS (status)) {
|
|
|
|
KdPrintEx(( DPFLTR_WSOCKTRANSPORT_ID, DPFLTR_INFO_LEVEL,
|
|
"AfdAllocateEndpoint: PsChargeProcessPoolQuota failed.\n" ));
|
|
|
|
goto CleanupTransportInfo;
|
|
}
|
|
|
|
// See if we have too many endpoins waiting to be freed and reuse one of them
|
|
if ((AfdEndpointsFreeing<AFD_ENDPOINTS_FREEING_MAX)
|
|
|| ((endpoint = AfdReuseEndpoint ())==NULL)) {
|
|
//
|
|
// Allocate a buffer to hold the endpoint structure.
|
|
// We use the priority version of this routine because
|
|
// we are going to charge the process for this allocation
|
|
// before it can make any use of it.
|
|
//
|
|
|
|
endpoint = AFD_ALLOCATE_POOL_PRIORITY(
|
|
NonPagedPool,
|
|
sizeof(AFD_ENDPOINT),
|
|
AFD_ENDPOINT_POOL_TAG,
|
|
NormalPoolPriority
|
|
);
|
|
}
|
|
|
|
if ( endpoint != NULL ) {
|
|
|
|
AfdRecordQuotaHistory(
|
|
process,
|
|
(LONG)sizeof (AFD_ENDPOINT),
|
|
"CreateEndp ",
|
|
endpoint
|
|
);
|
|
|
|
AfdRecordPoolQuotaCharged(sizeof (AFD_ENDPOINT));
|
|
|
|
RtlZeroMemory( endpoint, sizeof(AFD_ENDPOINT) );
|
|
|
|
//
|
|
// Initialize the reference count to 2--one for the caller's
|
|
// reference, one for the active reference.
|
|
//
|
|
|
|
endpoint->ReferenceCount = 2;
|
|
|
|
//
|
|
// Initialize the endpoint structure.
|
|
//
|
|
|
|
if ( TransportDeviceName == NULL ) {
|
|
endpoint->Type = AfdBlockTypeHelper;
|
|
endpoint->State = AfdEndpointStateInvalid;
|
|
} else {
|
|
endpoint->Type = AfdBlockTypeEndpoint;
|
|
endpoint->State = AfdEndpointStateOpen;
|
|
endpoint->TransportInfo = transportInfo;
|
|
//
|
|
// Cache service flags for quick determination of provider characteristics
|
|
// such as bufferring and messaging
|
|
//
|
|
if (transportInfo->InfoValid) {
|
|
endpoint->TdiServiceFlags = endpoint->TransportInfo->ProviderInfo.ServiceFlags;
|
|
}
|
|
}
|
|
|
|
endpoint->GroupID = GroupID;
|
|
endpoint->GroupType = groupType;
|
|
|
|
|
|
AfdInitializeSpinLock( &endpoint->SpinLock );
|
|
InitializeListHead (&endpoint->RoutingNotifications);
|
|
InitializeListHead (&endpoint->RequestList);
|
|
|
|
#if REFERENCE_DEBUG
|
|
endpoint->CurrentReferenceSlot = -1;
|
|
#endif
|
|
|
|
#if DBG
|
|
InitializeListHead( &endpoint->OutstandingIrpListHead );
|
|
#endif
|
|
|
|
//
|
|
// Remember the process which opened the endpoint. We'll use this to
|
|
// charge quota to the process as necessary. Reference the process
|
|
// so that it does not go away until we have returned all charged
|
|
// quota to the process.
|
|
//
|
|
|
|
endpoint->OwningProcess = process;
|
|
|
|
ObReferenceObject(endpoint->OwningProcess);
|
|
|
|
//
|
|
// Insert the endpoint on the global list.
|
|
//
|
|
|
|
AfdInsertNewEndpointInList( endpoint );
|
|
|
|
//
|
|
// Return a pointer to the new endpoint to the caller.
|
|
//
|
|
|
|
IF_DEBUG(ENDPOINT) {
|
|
KdPrintEx(( DPFLTR_WSOCKTRANSPORT_ID, DPFLTR_TRACE_LEVEL,
|
|
"AfdAllocateEndpoint: new endpoint at %p\n",
|
|
endpoint ));
|
|
}
|
|
|
|
*NewEndpoint = endpoint;
|
|
return STATUS_SUCCESS;
|
|
}
|
|
else {
|
|
PsReturnPoolQuota(
|
|
process,
|
|
NonPagedPool,
|
|
sizeof (AFD_ENDPOINT)
|
|
);
|
|
status= STATUS_INSUFFICIENT_RESOURCES;
|
|
}
|
|
|
|
if( GroupID != 0 ) {
|
|
AfdDereferenceGroup( GroupID );
|
|
}
|
|
|
|
}
|
|
else {
|
|
status = STATUS_INVALID_PARAMETER;
|
|
}
|
|
|
|
CleanupTransportInfo:
|
|
|
|
if (transportInfo!=NULL) {
|
|
if (InterlockedDecrement ((PLONG)&transportInfo->ReferenceCount)==0) {
|
|
//
|
|
// Reference count has gone to 0, we need to remove the structure
|
|
// from the global list and free it.
|
|
// Note that no code increments reference count if doesn't
|
|
// know for fact that its current reference count is above 0).
|
|
//
|
|
//
|
|
// Make sure the thread in which we execute cannot get
|
|
// suspeneded in APC while we own the global resource.
|
|
//
|
|
KeEnterCriticalRegion ();
|
|
ExAcquireResourceExclusiveLite( AfdResource, TRUE );
|
|
ASSERT (transportInfo->ReferenceCount==0);
|
|
ASSERT (transportInfo->InfoValid==FALSE);
|
|
RemoveEntryList (&transportInfo->TransportInfoListEntry);
|
|
ExReleaseResourceLite( AfdResource );
|
|
KeLeaveCriticalRegion ();
|
|
AFD_FREE_POOL (transportInfo, AFD_TRANSPORT_INFO_POOL_TAG);
|
|
}
|
|
}
|
|
|
|
return status;
|
|
} // AfdAllocateEndpoint
|
|
|
|
|
|
VOID
|
|
AfdFreeQueuedConnections (
|
|
IN PAFD_ENDPOINT Endpoint
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Frees queued connection objects on a listening AFD endpoint.
|
|
|
|
Arguments:
|
|
|
|
Endpoint - a pointer to the AFD endpoint structure.
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
|
|
{
|
|
AFD_LOCK_QUEUE_HANDLE lockHandle;
|
|
KIRQL oldIrql;
|
|
PAFD_CONNECTION connection;
|
|
PIRP irp;
|
|
|
|
ASSERT( Endpoint->Type == AfdBlockTypeVcListening ||
|
|
Endpoint->Type == AfdBlockTypeVcBoth );
|
|
|
|
//
|
|
// Free the unaccepted connections.
|
|
//
|
|
// We must hold endpoint spinLock to call AfdGetUnacceptedConnection,
|
|
// but we may not hold it when calling AfdDereferenceConnection.
|
|
//
|
|
|
|
KeRaiseIrql (DISPATCH_LEVEL, &oldIrql);
|
|
|
|
AfdAcquireSpinLockAtDpcLevel( &Endpoint->SpinLock, &lockHandle );
|
|
|
|
while ( (connection = AfdGetUnacceptedConnection( Endpoint )) != NULL ) {
|
|
ASSERT( connection->Endpoint == Endpoint );
|
|
|
|
AfdReleaseSpinLockFromDpcLevel( &Endpoint->SpinLock, &lockHandle );
|
|
if (connection->SanConnection) {
|
|
AfdSanAbortConnection ( connection );
|
|
}
|
|
else {
|
|
AfdAbortConnection( connection );
|
|
}
|
|
AfdAcquireSpinLockAtDpcLevel( &Endpoint->SpinLock, &lockHandle );
|
|
|
|
}
|
|
|
|
|
|
//
|
|
// Free the returned connections.
|
|
//
|
|
|
|
while ( (connection = AfdGetReturnedConnection( Endpoint, 0 )) != NULL ) {
|
|
|
|
ASSERT( connection->Endpoint == Endpoint );
|
|
|
|
AfdReleaseSpinLockFromDpcLevel( &Endpoint->SpinLock, &lockHandle );
|
|
if (connection->SanConnection) {
|
|
AfdSanAbortConnection ( connection );
|
|
}
|
|
else {
|
|
AfdAbortConnection( connection );
|
|
}
|
|
|
|
AfdAcquireSpinLockAtDpcLevel( &Endpoint->SpinLock, &lockHandle );
|
|
|
|
}
|
|
|
|
|
|
|
|
if (IS_DELAYED_ACCEPTANCE_ENDPOINT (Endpoint)) {
|
|
while (!IsListEmpty (&Endpoint->Common.VcListening.ListenConnectionListHead)) {
|
|
PIRP listenIrp;
|
|
connection = CONTAINING_RECORD (
|
|
Endpoint->Common.VcListening.ListenConnectionListHead.Flink,
|
|
AFD_CONNECTION,
|
|
ListEntry
|
|
);
|
|
RemoveEntryList (&connection->ListEntry);
|
|
listenIrp = InterlockedExchangePointer ((PVOID *)&connection->ListenIrp, NULL);
|
|
if (listenIrp!=NULL) {
|
|
IoAcquireCancelSpinLock (&listenIrp->CancelIrql);
|
|
ASSERT (listenIrp->CancelIrql==DISPATCH_LEVEL);
|
|
AfdReleaseSpinLockFromDpcLevel (&Endpoint->SpinLock, &lockHandle);
|
|
|
|
AfdCancelIrp (listenIrp);
|
|
|
|
AfdAcquireSpinLockAtDpcLevel( &Endpoint->SpinLock, &lockHandle );
|
|
}
|
|
}
|
|
AfdReleaseSpinLockFromDpcLevel( &Endpoint->SpinLock, &lockHandle );
|
|
}
|
|
else {
|
|
AfdReleaseSpinLockFromDpcLevel( &Endpoint->SpinLock, &lockHandle );
|
|
//
|
|
// And finally, purge the free connection queue.
|
|
//
|
|
|
|
while ( (connection = AfdGetFreeConnection( Endpoint, &irp )) != NULL ) {
|
|
ASSERT( connection->Type == AfdBlockTypeConnection );
|
|
if (irp!=NULL) {
|
|
AfdCleanupSuperAccept (irp, STATUS_CANCELLED);
|
|
if (irp->Cancel) {
|
|
KIRQL cancelIrql;
|
|
//
|
|
// Need to sycn with cancel routine which may
|
|
// have been called from AfdCleanup for accepting
|
|
// endpoint
|
|
//
|
|
IoAcquireCancelSpinLock (&cancelIrql);
|
|
IoReleaseCancelSpinLock (cancelIrql);
|
|
}
|
|
IoCompleteRequest (irp, AfdPriorityBoost);
|
|
}
|
|
DEREFERENCE_CONNECTION( connection );
|
|
}
|
|
}
|
|
KeLowerIrql (oldIrql);
|
|
|
|
return;
|
|
|
|
} // AfdFreeQueuedConnections
|
|
|
|
|
|
|
|
VOID
|
|
AfdFreeEndpointResources (
|
|
PAFD_ENDPOINT endpoint
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
Does the actual work to deallocate an AFD endpoint structure and
|
|
associated structures. Note that all other references to the
|
|
endpoint structure must be gone before this routine is called, since
|
|
it frees the endpoint and assumes that nobody else will be looking
|
|
at the endpoint.
|
|
|
|
Arguments:
|
|
Endpoint to be cleaned up
|
|
|
|
Return Value:
|
|
|
|
None
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
PLIST_ENTRY listEntry;
|
|
|
|
PAGED_CODE ();
|
|
|
|
ASSERT( IS_AFD_ENDPOINT_TYPE( endpoint ) );
|
|
ASSERT( endpoint->ReferenceCount == 0 );
|
|
ASSERT( endpoint->State == AfdEndpointStateClosing );
|
|
ASSERT( endpoint->ObReferenceBias == 0 );
|
|
ASSERT( KeGetCurrentIrql( ) == 0 );
|
|
|
|
//
|
|
// If this is a listening endpoint, then purge the endpoint of all
|
|
// queued connections.
|
|
//
|
|
|
|
if ( (endpoint->Type & AfdBlockTypeVcListening) == AfdBlockTypeVcListening ) {
|
|
|
|
AfdFreeQueuedConnections( endpoint );
|
|
|
|
}
|
|
|
|
//
|
|
// Dereference any group ID associated with this endpoint.
|
|
//
|
|
|
|
if( endpoint->GroupID != 0 ) {
|
|
|
|
AfdDereferenceGroup( endpoint->GroupID );
|
|
|
|
}
|
|
|
|
//
|
|
// If this is a bufferring datagram endpoint, remove all the
|
|
// bufferred datagrams from the endpoint's list and free them.
|
|
//
|
|
|
|
if ( IS_DGRAM_ENDPOINT(endpoint) &&
|
|
endpoint->ReceiveDatagramBufferListHead.Flink != NULL ) {
|
|
|
|
while ( !IsListEmpty( &endpoint->ReceiveDatagramBufferListHead ) ) {
|
|
|
|
PAFD_BUFFER_HEADER afdBuffer;
|
|
|
|
listEntry = RemoveHeadList( &endpoint->ReceiveDatagramBufferListHead );
|
|
afdBuffer = CONTAINING_RECORD( listEntry, AFD_BUFFER_HEADER, BufferListEntry );
|
|
AfdReturnBuffer( afdBuffer, endpoint->OwningProcess );
|
|
}
|
|
}
|
|
|
|
//
|
|
// Close and dereference the TDI address object on the endpoint, if
|
|
// any.
|
|
//
|
|
|
|
if ( endpoint->AddressFileObject != NULL ) {
|
|
//
|
|
// Little extra precaution. It is possible that there exists
|
|
// a duplicated handle in user process, so transport can in
|
|
// theory call event handler with bogus endpoint pointer that
|
|
// we are about to free. The event handlers for datagram
|
|
// endpoints are reset in AfdCleanup.
|
|
// Connection-oriented accepted endpoints cannot have address handles
|
|
// in their structures because they share them with listening
|
|
// endpoint (it would be a grave mistake if we tried to close
|
|
// address handle owned by listening endpoint while closing connection
|
|
// accepted on it).
|
|
//
|
|
if ( endpoint->AddressHandle != NULL &&
|
|
IS_VC_ENDPOINT (endpoint)) {
|
|
|
|
ASSERT (((endpoint->Type&AfdBlockTypeVcConnecting)!=AfdBlockTypeVcConnecting)
|
|
|| (endpoint->Common.VcConnecting.ListenEndpoint==NULL));
|
|
|
|
|
|
status = AfdSetEventHandler(
|
|
endpoint->AddressFileObject,
|
|
TDI_EVENT_ERROR,
|
|
NULL,
|
|
NULL
|
|
);
|
|
//ASSERT( NT_SUCCESS(status) );
|
|
|
|
status = AfdSetEventHandler(
|
|
endpoint->AddressFileObject,
|
|
TDI_EVENT_DISCONNECT,
|
|
NULL,
|
|
NULL
|
|
);
|
|
//ASSERT( NT_SUCCESS(status) );
|
|
|
|
status = AfdSetEventHandler(
|
|
endpoint->AddressFileObject,
|
|
TDI_EVENT_RECEIVE,
|
|
NULL,
|
|
NULL
|
|
);
|
|
|
|
//ASSERT( NT_SUCCESS(status) );
|
|
|
|
if (IS_TDI_EXPEDITED (endpoint)) {
|
|
status = AfdSetEventHandler(
|
|
endpoint->AddressFileObject,
|
|
TDI_EVENT_RECEIVE_EXPEDITED,
|
|
NULL,
|
|
NULL
|
|
);
|
|
//ASSERT( NT_SUCCESS(status) );
|
|
}
|
|
|
|
if ( IS_TDI_BUFFERRING(endpoint) ) {
|
|
status = AfdSetEventHandler(
|
|
endpoint->AddressFileObject,
|
|
TDI_EVENT_SEND_POSSIBLE,
|
|
NULL,
|
|
NULL
|
|
);
|
|
//ASSERT( NT_SUCCESS(status) );
|
|
}
|
|
else {
|
|
status = AfdSetEventHandler(
|
|
endpoint->AddressFileObject,
|
|
TDI_EVENT_CHAINED_RECEIVE,
|
|
NULL,
|
|
NULL
|
|
);
|
|
//ASSERT( NT_SUCCESS(status) );
|
|
}
|
|
}
|
|
ObDereferenceObject( endpoint->AddressFileObject );
|
|
endpoint->AddressFileObject = NULL;
|
|
AfdRecordAddrDeref();
|
|
}
|
|
|
|
if ( endpoint->AddressHandle != NULL ) {
|
|
#if DBG
|
|
{
|
|
NTSTATUS status1;
|
|
OBJECT_HANDLE_FLAG_INFORMATION handleInfo;
|
|
handleInfo.Inherit = FALSE;
|
|
handleInfo.ProtectFromClose = FALSE;
|
|
status1 = ZwSetInformationObject (
|
|
endpoint->AddressHandle,
|
|
ObjectHandleFlagInformation,
|
|
&handleInfo,
|
|
sizeof (handleInfo)
|
|
);
|
|
ASSERT (NT_SUCCESS (status1));
|
|
}
|
|
#endif
|
|
status = ZwClose( endpoint->AddressHandle );
|
|
ASSERT( NT_SUCCESS(status) );
|
|
endpoint->AddressHandle = NULL;
|
|
AfdRecordAddrClosed();
|
|
}
|
|
|
|
//
|
|
// Remove the endpoint from the global list. Do this before any
|
|
// deallocations to prevent someone else from seeing an endpoint in
|
|
// an invalid state.
|
|
//
|
|
|
|
AfdRemoveEndpointFromList( endpoint );
|
|
|
|
//
|
|
// Return the quota we charged to this process when we allocated
|
|
// the endpoint object.
|
|
//
|
|
|
|
PsReturnPoolQuota(
|
|
endpoint->OwningProcess,
|
|
NonPagedPool,
|
|
sizeof (AFD_ENDPOINT)
|
|
);
|
|
AfdRecordQuotaHistory(
|
|
endpoint->OwningProcess,
|
|
-(LONG)sizeof (AFD_ENDPOINT),
|
|
"EndpDealloc ",
|
|
endpoint
|
|
);
|
|
AfdRecordPoolQuotaReturned(
|
|
sizeof (AFD_ENDPOINT)
|
|
);
|
|
|
|
//
|
|
// If we set up an owning process for the endpoint, dereference the
|
|
// process.
|
|
//
|
|
|
|
if ( endpoint->OwningProcess != NULL ) {
|
|
ObDereferenceObject( endpoint->OwningProcess );
|
|
endpoint->OwningProcess = NULL;
|
|
}
|
|
|
|
|
|
//
|
|
// Dereference the listening or c-root endpoint on the endpoint, if
|
|
// any.
|
|
//
|
|
|
|
if ( endpoint->Type == AfdBlockTypeVcConnecting &&
|
|
endpoint->Common.VcConnecting.ListenEndpoint != NULL ) {
|
|
PAFD_ENDPOINT listenEndpoint = endpoint->Common.VcConnecting.ListenEndpoint;
|
|
ASSERT (((listenEndpoint->Type&AfdBlockTypeVcListening)==AfdBlockTypeVcListening) ||
|
|
IS_CROOT_ENDPOINT (listenEndpoint));
|
|
ASSERT (endpoint->LocalAddress==listenEndpoint->LocalAddress);
|
|
DEREFERENCE_ENDPOINT( listenEndpoint );
|
|
|
|
endpoint->Common.VcConnecting.ListenEndpoint = NULL;
|
|
//
|
|
// We used the local address from the listening endpoint,
|
|
// simply reset it, it will be freed when listening endpoint
|
|
// is freed.
|
|
//
|
|
endpoint->LocalAddress = NULL;
|
|
endpoint->LocalAddressLength = 0;
|
|
}
|
|
|
|
if (IS_SAN_ENDPOINT (endpoint)) {
|
|
AfdSanCleanupEndpoint (endpoint);
|
|
|
|
}
|
|
else if (IS_SAN_HELPER (endpoint)) {
|
|
AfdSanCleanupHelper (endpoint);
|
|
}
|
|
//
|
|
// Free local and remote address buffers.
|
|
//
|
|
|
|
if ( endpoint->LocalAddress != NULL ) {
|
|
AFD_FREE_POOL(
|
|
endpoint->LocalAddress,
|
|
AFD_LOCAL_ADDRESS_POOL_TAG
|
|
);
|
|
endpoint->LocalAddress = NULL;
|
|
}
|
|
|
|
if ( IS_DGRAM_ENDPOINT(endpoint) &&
|
|
endpoint->Common.Datagram.RemoteAddress != NULL ) {
|
|
AFD_RETURN_REMOTE_ADDRESS(
|
|
endpoint->Common.Datagram.RemoteAddress,
|
|
endpoint->Common.Datagram.RemoteAddressLength
|
|
);
|
|
endpoint->Common.Datagram.RemoteAddress = NULL;
|
|
}
|
|
|
|
//
|
|
// Free context and connect data buffers.
|
|
//
|
|
|
|
if ( endpoint->Context != NULL ) {
|
|
|
|
AFD_FREE_POOL(
|
|
endpoint->Context,
|
|
AFD_CONTEXT_POOL_TAG
|
|
);
|
|
endpoint->Context = NULL;
|
|
|
|
}
|
|
|
|
if ( IS_VC_ENDPOINT (endpoint) &&
|
|
endpoint->Common.VirtualCircuit.ConnectDataBuffers != NULL ) {
|
|
AfdFreeConnectDataBuffers( endpoint->Common.VirtualCircuit.ConnectDataBuffers );
|
|
}
|
|
|
|
//
|
|
// If there's an active EventSelect() on this endpoint, dereference
|
|
// the associated event object.
|
|
//
|
|
|
|
if( endpoint->EventObject != NULL ) {
|
|
ObDereferenceObject( endpoint->EventObject );
|
|
endpoint->EventObject = NULL;
|
|
}
|
|
|
|
ASSERT ( endpoint->Irp == NULL );
|
|
|
|
|
|
if (endpoint->TransportInfo!=NULL) {
|
|
if (InterlockedDecrement ((PLONG)&endpoint->TransportInfo->ReferenceCount)==0) {
|
|
//
|
|
// Reference count has gone to 0, we need to remove the structure
|
|
// from the global list and free it.
|
|
// Note that no code increments reference count if doesn't
|
|
// know for fact that its current reference count is above 0).
|
|
//
|
|
//
|
|
// Make sure the thread in which we execute cannot get
|
|
// suspeneded in APC while we own the global resource.
|
|
//
|
|
KeEnterCriticalRegion ();
|
|
ExAcquireResourceExclusiveLite( AfdResource, TRUE );
|
|
ASSERT (endpoint->TransportInfo->ReferenceCount==0);
|
|
ASSERT (endpoint->TransportInfo->InfoValid==FALSE);
|
|
RemoveEntryList (&endpoint->TransportInfo->TransportInfoListEntry);
|
|
ExReleaseResourceLite( AfdResource );
|
|
KeLeaveCriticalRegion ();
|
|
AFD_FREE_POOL (endpoint->TransportInfo, AFD_TRANSPORT_INFO_POOL_TAG);
|
|
}
|
|
endpoint->TransportInfo = NULL;
|
|
}
|
|
|
|
ASSERT (endpoint->OutstandingIrpCount==0);
|
|
|
|
IF_DEBUG(ENDPOINT) {
|
|
KdPrintEx(( DPFLTR_WSOCKTRANSPORT_ID, DPFLTR_TRACE_LEVEL,
|
|
"AfdFreeEndpoint: freeing endpoint at %p\n",
|
|
endpoint ));
|
|
}
|
|
|
|
endpoint->Type = AfdBlockTypeInvalidEndpoint;
|
|
|
|
}
|
|
|
|
|
|
VOID
|
|
AfdFreeEndpoint (
|
|
IN PVOID Context
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
Calls AfdFreeEndpointResources to cleanup endpoint and frees the
|
|
endpoint structure itself
|
|
|
|
Arguments:
|
|
|
|
Context - Actually points to the endpoint's embedded AFD_WORK_ITEM
|
|
structure. From this we can determine the endpoint to free.
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
|
|
{
|
|
PAFD_ENDPOINT endpoint;
|
|
PAGED_CODE( );
|
|
|
|
|
|
ASSERT( Context != NULL );
|
|
|
|
InterlockedDecrement(&AfdEndpointsFreeing);
|
|
|
|
endpoint = CONTAINING_RECORD(
|
|
Context,
|
|
AFD_ENDPOINT,
|
|
WorkItem
|
|
);
|
|
|
|
|
|
AfdFreeEndpointResources (endpoint);
|
|
//
|
|
// Free the pool used for the endpoint itself.
|
|
//
|
|
|
|
AFD_FREE_POOL(
|
|
endpoint,
|
|
AFD_ENDPOINT_POOL_TAG
|
|
);
|
|
|
|
} // AfdFreeEndpoint
|
|
|
|
|
|
PAFD_ENDPOINT
|
|
AfdReuseEndpoint (
|
|
VOID
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
Finds a AfdFreeEndpoint work item in the list and calls
|
|
AfdFreeEndpointResources to cleanup endpoint
|
|
|
|
Arguments:
|
|
None
|
|
|
|
Return Value:
|
|
|
|
Reinitialized endpoint.
|
|
|
|
--*/
|
|
|
|
{
|
|
PAFD_ENDPOINT endpoint;
|
|
PVOID Context;
|
|
|
|
PAGED_CODE( );
|
|
|
|
Context = AfdGetWorkerByRoutine (AfdFreeEndpoint);
|
|
if (Context==NULL)
|
|
return NULL;
|
|
|
|
endpoint = CONTAINING_RECORD(
|
|
Context,
|
|
AFD_ENDPOINT,
|
|
WorkItem
|
|
);
|
|
|
|
|
|
AfdFreeEndpointResources (endpoint);
|
|
return endpoint;
|
|
} // AfdReuseEndpoint
|
|
|
|
|
|
#if REFERENCE_DEBUG
|
|
VOID
|
|
AfdDereferenceEndpoint (
|
|
IN PAFD_ENDPOINT Endpoint,
|
|
IN LONG LocationId,
|
|
IN ULONG Param
|
|
)
|
|
#else
|
|
VOID
|
|
AfdDereferenceEndpoint (
|
|
IN PAFD_ENDPOINT Endpoint
|
|
)
|
|
#endif
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Dereferences an AFD endpoint and calls the routine to free it if
|
|
appropriate.
|
|
|
|
Arguments:
|
|
|
|
Endpoint - a pointer to the AFD endpoint structure.
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
|
|
{
|
|
LONG result;
|
|
|
|
#if REFERENCE_DEBUG
|
|
IF_DEBUG(ENDPOINT) {
|
|
KdPrintEx(( DPFLTR_WSOCKTRANSPORT_ID, DPFLTR_TRACE_LEVEL,
|
|
"AfdDereferenceEndpoint: endpoint at %p, new refcnt %ld\n",
|
|
Endpoint, Endpoint->ReferenceCount-1 ));
|
|
}
|
|
|
|
ASSERT( IS_AFD_ENDPOINT_TYPE( Endpoint ) );
|
|
ASSERT( Endpoint->ReferenceCount > 0 );
|
|
ASSERT( Endpoint->ReferenceCount != 0xDAADF00D );
|
|
|
|
AFD_UPDATE_REFERENCE_DEBUG(Endpoint, Endpoint->ReferenceCount-1, LocationId, Param);
|
|
|
|
|
|
|
|
//
|
|
// We must hold AfdSpinLock while doing the dereference and check
|
|
// for free. This is because some code makes the assumption that
|
|
// the connection structure will not go away while AfdSpinLock is
|
|
// held, and that code references the endpoint before releasing
|
|
// AfdSpinLock. If we did the InterlockedDecrement() without the
|
|
// lock held, our count may go to zero, that code may reference the
|
|
// connection, and then a double free might occur.
|
|
//
|
|
// It is still valuable to use the interlocked routines for
|
|
// increment and decrement of structures because it allows us to
|
|
// avoid having to hold the spin lock for a reference.
|
|
//
|
|
// In NT40+ we use InterlockedCompareExchange and make sure that
|
|
// we do not increment reference count if it is 0, so holding
|
|
// a spinlock is no longer necessary
|
|
|
|
//
|
|
// Decrement the reference count; if it is 0, we may need to
|
|
// free the endpoint.
|
|
//
|
|
|
|
#endif
|
|
result = InterlockedDecrement( (PLONG)&Endpoint->ReferenceCount );
|
|
|
|
if ( result == 0 ) {
|
|
|
|
ASSERT( Endpoint->State == AfdEndpointStateClosing );
|
|
|
|
if ((Endpoint->Type==AfdBlockTypeVcConnecting) &&
|
|
(Endpoint->Common.VcConnecting.ListenEndpoint != NULL) &&
|
|
(KeGetCurrentIrql()==PASSIVE_LEVEL)) {
|
|
|
|
ASSERT (Endpoint->AddressHandle==NULL);
|
|
//
|
|
// If this is a connecting endpoint assoicated with the
|
|
// listening endpoint and we already at passive level,
|
|
// free the endpoint here. We can do this because in such
|
|
// a case we know that reference to the transport object
|
|
// is not the last one - at least one more is still in the
|
|
// listening endpoint and we remove transport object reference
|
|
// before dereferencing listening endpoint.
|
|
//
|
|
//
|
|
|
|
AfdFreeEndpointResources (Endpoint);
|
|
|
|
//
|
|
// Free the pool used for the endpoint itself.
|
|
//
|
|
|
|
AFD_FREE_POOL(
|
|
Endpoint,
|
|
AFD_ENDPOINT_POOL_TAG
|
|
);
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// We're going to do this by queueing a request to an executive
|
|
// worker thread. We do this for several reasons: to ensure
|
|
// that we're at IRQL 0 so we can free pageable memory, and to
|
|
// ensure that we're in a legitimate context for a close
|
|
// operation and not in conntext of event indication from
|
|
// the transport driver
|
|
//
|
|
|
|
InterlockedIncrement(&AfdEndpointsFreeing);
|
|
|
|
AfdQueueWorkItem(
|
|
AfdFreeEndpoint,
|
|
&Endpoint->WorkItem
|
|
);
|
|
}
|
|
}
|
|
|
|
} // AfdDereferenceEndpoint
|
|
|
|
#if REFERENCE_DEBUG
|
|
|
|
VOID
|
|
AfdReferenceEndpoint (
|
|
IN PAFD_ENDPOINT Endpoint,
|
|
IN LONG LocationId,
|
|
IN ULONG Param
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
References an AFD endpoint.
|
|
|
|
Arguments:
|
|
|
|
Endpoint - a pointer to the AFD endpoint structure.
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
|
|
{
|
|
|
|
LONG result;
|
|
|
|
ASSERT( Endpoint->ReferenceCount > 0 );
|
|
|
|
ASSERT( Endpoint->ReferenceCount < 0xFFFF ||
|
|
((Endpoint->Listening ||
|
|
Endpoint->afdC_Root) && Endpoint->ReferenceCount<0xFFFFFFF));
|
|
|
|
result = InterlockedIncrement( (PLONG)&Endpoint->ReferenceCount );
|
|
AFD_UPDATE_REFERENCE_DEBUG(Endpoint, result, LocationId, Param);
|
|
|
|
IF_DEBUG(ENDPOINT) {
|
|
KdPrintEx(( DPFLTR_WSOCKTRANSPORT_ID, DPFLTR_TRACE_LEVEL,
|
|
"AfdReferenceEndpoint: endpoint at %p, new refcnt %ld\n",
|
|
Endpoint, result ));
|
|
}
|
|
|
|
} // AfdReferenceEndpoint
|
|
|
|
VOID
|
|
AfdUpdateEndpoint (
|
|
IN PAFD_ENDPOINT Endpoint,
|
|
IN LONG LocationId,
|
|
IN ULONG Param
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Update an AFD endpoint reference debug information.
|
|
|
|
Arguments:
|
|
|
|
Endpoint - a pointer to the AFD endpoint structure.
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
|
|
{
|
|
ASSERT( Endpoint->ReferenceCount > 0 );
|
|
|
|
ASSERT( Endpoint->ReferenceCount < 0xFFFF ||
|
|
((Endpoint->Listening ||
|
|
Endpoint->afdC_Root) && Endpoint->ReferenceCount<0xFFFFFFF));
|
|
|
|
AFD_UPDATE_REFERENCE_DEBUG(Endpoint, Endpoint->ReferenceCount, LocationId, Param);
|
|
|
|
|
|
} // AfdUpdateEndpoint
|
|
#endif
|
|
|
|
|
|
#if REFERENCE_DEBUG
|
|
BOOLEAN
|
|
AfdCheckAndReferenceEndpoint (
|
|
IN PAFD_ENDPOINT Endpoint,
|
|
IN LONG LocationId,
|
|
IN ULONG Param
|
|
)
|
|
#else
|
|
BOOLEAN
|
|
AfdCheckAndReferenceEndpoint (
|
|
IN PAFD_ENDPOINT Endpoint
|
|
)
|
|
#endif
|
|
{
|
|
LONG result;
|
|
|
|
do {
|
|
result = Endpoint->ReferenceCount;
|
|
if (result<=0)
|
|
break;
|
|
}
|
|
while (InterlockedCompareExchange ((PLONG)&Endpoint->ReferenceCount,
|
|
(result+1),
|
|
result)!=result);
|
|
|
|
|
|
|
|
if (result>0) {
|
|
|
|
#if REFERENCE_DEBUG
|
|
AFD_UPDATE_REFERENCE_DEBUG(Endpoint, result+1, LocationId, Param);
|
|
|
|
IF_DEBUG(ENDPOINT) {
|
|
KdPrintEx(( DPFLTR_WSOCKTRANSPORT_ID, DPFLTR_TRACE_LEVEL,
|
|
"AfdReferenceEndpoint: endpoint at %p, new refcnt %ld\n",
|
|
Endpoint, result+1 ));
|
|
}
|
|
|
|
ASSERT( Endpoint->ReferenceCount < 0xFFFF ||
|
|
((Endpoint->Listening ||
|
|
Endpoint->afdC_Root) && Endpoint->ReferenceCount<0xFFFFFFF));
|
|
#endif
|
|
return TRUE;
|
|
}
|
|
else {
|
|
KdPrintEx(( DPFLTR_WSOCKTRANSPORT_ID, DPFLTR_ERROR_LEVEL,
|
|
"AfdCheckAndReferenceEndpoint: Endpoint %p is gone (refcount: %ld)!\n",
|
|
Endpoint, result));
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
|
|
VOID
|
|
AfdRefreshEndpoint (
|
|
IN PAFD_ENDPOINT Endpoint
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Prepares an AFD endpoint structure to be reused. All other
|
|
references to the endpoint must be freed before this routine is
|
|
called, since this routine assumes that nobody will access the old
|
|
information in the endpoint structure.
|
|
This fact is ensured by the state change primitive.
|
|
|
|
Arguments:
|
|
|
|
Endpoint - a pointer to the AFD endpoint structure.
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
|
|
{
|
|
|
|
ASSERT( Endpoint->Type == AfdBlockTypeVcConnecting );
|
|
ASSERT( Endpoint->Common.VcConnecting.Connection == NULL );
|
|
ASSERT( Endpoint->StateChangeInProgress!=0);
|
|
ASSERT( Endpoint->State == AfdEndpointStateTransmitClosing );
|
|
|
|
if ( Endpoint->Common.VcConnecting.ListenEndpoint != NULL ) {
|
|
//
|
|
// TransmitFile after SuperAccept, cleanup back to open state.
|
|
//
|
|
|
|
//
|
|
// Dereference the listening endpoint and its address object.
|
|
//
|
|
|
|
PAFD_ENDPOINT listenEndpoint = Endpoint->Common.VcConnecting.ListenEndpoint;
|
|
ASSERT (((listenEndpoint->Type&AfdBlockTypeVcListening)==AfdBlockTypeVcListening) ||
|
|
IS_CROOT_ENDPOINT (listenEndpoint));
|
|
ASSERT (Endpoint->LocalAddress==listenEndpoint->LocalAddress);
|
|
ASSERT (Endpoint->AddressFileObject==listenEndpoint->AddressFileObject);
|
|
|
|
DEREFERENCE_ENDPOINT( listenEndpoint );
|
|
Endpoint->Common.VcConnecting.ListenEndpoint = NULL;
|
|
|
|
//
|
|
// Close and dereference the TDI address object on the endpoint, if
|
|
// any.
|
|
//
|
|
|
|
|
|
ObDereferenceObject( Endpoint->AddressFileObject );
|
|
Endpoint->AddressFileObject = NULL;
|
|
AfdRecordAddrDeref();
|
|
|
|
//
|
|
// We used the local address from the listening endpoint,
|
|
// simply reset it, it will be freed when listening endpoint
|
|
// is freed.
|
|
//
|
|
Endpoint->LocalAddress = NULL;
|
|
Endpoint->LocalAddressLength = 0;
|
|
ASSERT (Endpoint->AddressHandle == NULL);
|
|
|
|
//
|
|
// Reinitialize the endpoint structure.
|
|
//
|
|
|
|
Endpoint->Type = AfdBlockTypeEndpoint;
|
|
Endpoint->State = AfdEndpointStateOpen;
|
|
}
|
|
else {
|
|
//
|
|
// TransmitFile after SuperConnect, cleanup back to bound state.
|
|
//
|
|
Endpoint->Type = AfdBlockTypeEndpoint;
|
|
ASSERT (Endpoint->AddressHandle!=NULL);
|
|
ASSERT (Endpoint->AddressFileObject!=NULL);
|
|
Endpoint->State = AfdEndpointStateBound;
|
|
}
|
|
|
|
Endpoint->DisconnectMode = 0;
|
|
Endpoint->EndpointStateFlags = 0;
|
|
Endpoint->EventsActive = 0;
|
|
AfdRecordEndpointsReused ();
|
|
return;
|
|
|
|
} // AfdRefreshEndpoint
|
|
|
|
|
|
NTSTATUS
|
|
AfdGetTransportInfo (
|
|
IN PUNICODE_STRING TransportDeviceName,
|
|
IN OUT PAFD_TRANSPORT_INFO *TransportInfo
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Returns a transport information structure corresponding to the
|
|
specified TDI transport provider. Each unique transport string gets
|
|
a single provider structure, so that multiple endpoints for the same
|
|
transport share the same transport information structure.
|
|
|
|
Arguments:
|
|
|
|
TransportDeviceName - the name of the TDI transport provider.
|
|
TransportInfo - place to return referenced pointer to transport info
|
|
|
|
Return Value:
|
|
|
|
STATUS_SUCCESS - returned transport info is valid.
|
|
STATUS_INSUFFICIENT_RESOURCES - not enough memory to allocate
|
|
transport info structure
|
|
STATUS_OBJECT_NAME_NOT_FOUND - transport's device is not available yet
|
|
|
|
--*/
|
|
|
|
{
|
|
PLIST_ENTRY listEntry;
|
|
PAFD_TRANSPORT_INFO transportInfo;
|
|
ULONG structureLength;
|
|
NTSTATUS status;
|
|
TDI_PROVIDER_INFO localProviderInfo;
|
|
BOOLEAN resourceShared = TRUE;
|
|
|
|
PAGED_CODE( );
|
|
|
|
//
|
|
// Make sure the thread in which we execute cannot get
|
|
// suspeneded in APC while we own the global resource.
|
|
//
|
|
KeEnterCriticalRegion ();
|
|
ExAcquireResourceSharedLite( AfdResource, TRUE );
|
|
|
|
//
|
|
// If this is the first endpoint, we may be paged out
|
|
// entirely.
|
|
//
|
|
if (!AfdLoaded) {
|
|
//
|
|
// Take the exclusive lock and page the locked sections in
|
|
// if necessary
|
|
//
|
|
|
|
//
|
|
// There should be no endpoints in the list.
|
|
//
|
|
ASSERT (IsListEmpty (&AfdEndpointListHead));
|
|
|
|
ExReleaseResourceLite ( AfdResource);
|
|
|
|
ExAcquireResourceExclusiveLite( AfdResource, TRUE );
|
|
resourceShared = FALSE;
|
|
if (!AfdLoaded) {
|
|
//
|
|
// There should be no endpoints in the list.
|
|
//
|
|
ASSERT (IsListEmpty (&AfdEndpointListHead));
|
|
MmResetDriverPaging (DriverEntry);
|
|
AfdLoaded = (PKEVENT)1;
|
|
}
|
|
}
|
|
ASSERT (AfdLoaded==(PKEVENT)1);
|
|
|
|
if (*TransportInfo==NULL) {
|
|
|
|
ScanTransportList:
|
|
//
|
|
// If caller did not have transport info allocated, walk the list
|
|
// of transport device names looking for an identical name.
|
|
//
|
|
|
|
|
|
for ( listEntry = AfdTransportInfoListHead.Flink;
|
|
listEntry != &AfdTransportInfoListHead;
|
|
listEntry = listEntry->Flink )
|
|
{
|
|
|
|
transportInfo = CONTAINING_RECORD(
|
|
listEntry,
|
|
AFD_TRANSPORT_INFO,
|
|
TransportInfoListEntry
|
|
);
|
|
|
|
if ( RtlCompareUnicodeString(
|
|
&transportInfo->TransportDeviceName,
|
|
TransportDeviceName,
|
|
TRUE ) == 0 ) {
|
|
|
|
//
|
|
// We found an exact match. Reference the structure
|
|
// to return it to the caller
|
|
//
|
|
|
|
do {
|
|
LONG localCount;
|
|
localCount = transportInfo->ReferenceCount;
|
|
if (localCount==0) {
|
|
//
|
|
// We hit a small window when the structure is
|
|
// about to be freed. We can't stop this from
|
|
// happenning, so we'll go on to allocate and
|
|
// requery. After all info is not valid anyway,
|
|
// thus we are just loosing on the allocation/dealocation
|
|
// code.
|
|
//
|
|
ASSERT (transportInfo->InfoValid==FALSE);
|
|
goto AllocateInfo;
|
|
}
|
|
|
|
if (InterlockedCompareExchange (
|
|
(PLONG)&transportInfo->ReferenceCount,
|
|
(localCount+1),
|
|
localCount)==localCount) {
|
|
if (transportInfo->InfoValid) {
|
|
//
|
|
// Info is valid return referenced pointer to
|
|
// the caller.
|
|
//
|
|
*TransportInfo = transportInfo;
|
|
ExReleaseResourceLite( AfdResource );
|
|
KeLeaveCriticalRegion ();
|
|
return STATUS_SUCCESS;
|
|
}
|
|
else {
|
|
//
|
|
// We found match, but info is not valid
|
|
//
|
|
goto QueryInfo;
|
|
}
|
|
}
|
|
}
|
|
while (1);
|
|
}
|
|
} // for
|
|
|
|
AllocateInfo:
|
|
if (resourceShared) {
|
|
//
|
|
// If we do not own resource exlusively, we will
|
|
// have to release and reacquire it and then
|
|
// rescan the list
|
|
//
|
|
ExReleaseResourceLite ( AfdResource);
|
|
ExAcquireResourceExclusiveLite( AfdResource, TRUE );
|
|
resourceShared = FALSE;
|
|
goto ScanTransportList;
|
|
}
|
|
//
|
|
// This is a brand new device name, allocate transport info
|
|
// structure for it.
|
|
//
|
|
|
|
structureLength = sizeof(AFD_TRANSPORT_INFO) +
|
|
TransportDeviceName->Length + sizeof(WCHAR);
|
|
|
|
transportInfo = AFD_ALLOCATE_POOL_PRIORITY(
|
|
NonPagedPool,
|
|
structureLength,
|
|
AFD_TRANSPORT_INFO_POOL_TAG,
|
|
NormalPoolPriority
|
|
);
|
|
|
|
if ( transportInfo == NULL ) {
|
|
ExReleaseResourceLite( AfdResource );
|
|
KeLeaveCriticalRegion ();
|
|
return STATUS_INSUFFICIENT_RESOURCES;
|
|
}
|
|
|
|
//
|
|
// Initialize the structure
|
|
//
|
|
transportInfo->ReferenceCount = 1;
|
|
transportInfo->InfoValid = FALSE;
|
|
|
|
//
|
|
// Fill in the transport device name.
|
|
//
|
|
|
|
transportInfo->TransportDeviceName.MaximumLength =
|
|
TransportDeviceName->Length + sizeof(WCHAR);
|
|
transportInfo->TransportDeviceName.Buffer =
|
|
(PWSTR)(transportInfo + 1);
|
|
|
|
RtlCopyUnicodeString(
|
|
&transportInfo->TransportDeviceName,
|
|
TransportDeviceName
|
|
);
|
|
//
|
|
// Insert the structure into the list so that the successive callers
|
|
// can reuse it.
|
|
//
|
|
InsertHeadList (&AfdTransportInfoListHead,
|
|
&transportInfo->TransportInfoListEntry);
|
|
}
|
|
else {
|
|
transportInfo = *TransportInfo;
|
|
//
|
|
// Caller has already referenced info in the list
|
|
// but transport device was not available at the
|
|
// time of the call. Recheck if it is valid under the lock
|
|
//
|
|
if (transportInfo->InfoValid) {
|
|
//
|
|
// Yes, it is, return success
|
|
//
|
|
ExReleaseResourceLite( AfdResource );
|
|
KeLeaveCriticalRegion ();
|
|
return STATUS_SUCCESS;
|
|
}
|
|
}
|
|
|
|
|
|
QueryInfo:
|
|
//
|
|
// Release the resource and leave critical region to let
|
|
// the IRP's in AfdQueryProviderInfo complete
|
|
//
|
|
ExReleaseResourceLite (AfdResource);
|
|
KeLeaveCriticalRegion ();
|
|
|
|
status = AfdQueryProviderInfo (TransportDeviceName, &localProviderInfo);
|
|
|
|
//
|
|
// Make sure the thread in which we execute cannot get
|
|
// suspeneded in APC while we own the global resource.
|
|
//
|
|
KeEnterCriticalRegion ();
|
|
ExAcquireResourceExclusiveLite( AfdResource, TRUE );
|
|
|
|
if (NT_SUCCESS (status)) {
|
|
//
|
|
// Check if someone did not get the info in parallel with us.
|
|
//
|
|
if (!transportInfo->InfoValid) {
|
|
//
|
|
// Copy local info structure to the one in the list.
|
|
//
|
|
transportInfo->ProviderInfo = localProviderInfo;
|
|
|
|
//
|
|
// Bump the reference count on this info structure
|
|
// as we know that it is a valid TDI provider and we
|
|
// want to cache, it so it stays even of all endpoints
|
|
// that use it are gone.
|
|
//
|
|
InterlockedIncrement ((PLONG)&transportInfo->ReferenceCount);
|
|
|
|
//
|
|
// Set the flag so that everyone knows it is now valid.
|
|
//
|
|
transportInfo->InfoValid = TRUE;
|
|
}
|
|
|
|
*TransportInfo = transportInfo;
|
|
}
|
|
else {
|
|
if (status==STATUS_OBJECT_NAME_NOT_FOUND ||
|
|
status==STATUS_OBJECT_PATH_NOT_FOUND ||
|
|
status==STATUS_NO_SUCH_DEVICE) {
|
|
//
|
|
// Transport driver must not have been loaded yet
|
|
// Return transport info structure anyway
|
|
// Caller will know that info structure is not
|
|
// valid because we did not set the flag
|
|
//
|
|
*TransportInfo = transportInfo;
|
|
}
|
|
else {
|
|
//
|
|
// Something else went wrong, free the strucuture
|
|
// if it was allocated in this routine
|
|
//
|
|
|
|
if (*TransportInfo==NULL) {
|
|
if (InterlockedDecrement ((PLONG)&transportInfo->ReferenceCount)==0) {
|
|
RemoveEntryList (&transportInfo->TransportInfoListEntry);
|
|
AFD_FREE_POOL(
|
|
transportInfo,
|
|
AFD_TRANSPORT_INFO_POOL_TAG
|
|
);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
ExReleaseResourceLite( AfdResource );
|
|
KeLeaveCriticalRegion ();
|
|
return status;
|
|
|
|
} // AfdGetTransportInfo
|
|
|