1211 lines
32 KiB
C
1211 lines
32 KiB
C
|
/*++
|
|||
|
|
|||
|
Copyright (c) 1989 Microsoft Corporation
|
|||
|
|
|||
|
Module Name:
|
|||
|
|
|||
|
blkconn.c
|
|||
|
|
|||
|
Abstract:
|
|||
|
|
|||
|
This module implements routines for managing connection blocks.
|
|||
|
|
|||
|
Author:
|
|||
|
|
|||
|
Chuck Lenzmeier (chuckl) 4-Oct-1989
|
|||
|
|
|||
|
Revision History:
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
#include "precomp.h"
|
|||
|
#pragma hdrstop
|
|||
|
|
|||
|
#define BugCheckFileId SRV_FILE_BLKCONN
|
|||
|
|
|||
|
#ifdef ALLOC_PRAGMA
|
|||
|
#pragma alloc_text( PAGE, SrvAllocateConnection )
|
|||
|
#pragma alloc_text( PAGE, SrvCloseConnectionsFromClient )
|
|||
|
#pragma alloc_text( PAGE, SrvFreeConnection )
|
|||
|
#endif
|
|||
|
#if 0
|
|||
|
NOT PAGEABLE -- SrvCloseConnection
|
|||
|
NOT PAGEABLE -- SrvCloseFreeConnection
|
|||
|
NOT PAGEABLE -- SrvDereferenceConnection
|
|||
|
NOT PAGEABLE -- SrvQueryConnections
|
|||
|
#endif
|
|||
|
|
|||
|
VOID
|
|||
|
SrvAllocateConnection (
|
|||
|
OUT PCONNECTION *Connection
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function allocates a Connection Block from the system nonpaged
|
|||
|
pool.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Connection - Returns a pointer to the connection block, or NULL if
|
|||
|
no pool was available.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
PCONNECTION connection;
|
|||
|
ULONG i;
|
|||
|
PPAGED_CONNECTION pagedConnection;
|
|||
|
|
|||
|
PAGED_CODE( );
|
|||
|
|
|||
|
//
|
|||
|
// Attempt to allocate from nonpaged pool.
|
|||
|
//
|
|||
|
|
|||
|
connection = ALLOCATE_NONPAGED_POOL( sizeof(CONNECTION), BlockTypeConnection );
|
|||
|
*Connection = connection;
|
|||
|
|
|||
|
if ( connection == NULL ) {
|
|||
|
INTERNAL_ERROR(
|
|||
|
ERROR_LEVEL_EXPECTED,
|
|||
|
"SrvAllocateConnection: Unable to allocate %d bytes from"
|
|||
|
"nonpaged pool",
|
|||
|
sizeof( CONNECTION ),
|
|||
|
NULL
|
|||
|
);
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
RtlZeroMemory( connection, sizeof(CONNECTION) );
|
|||
|
|
|||
|
pagedConnection = ALLOCATE_HEAP(
|
|||
|
sizeof(PAGED_CONNECTION),
|
|||
|
BlockTypePagedConnection );
|
|||
|
|
|||
|
if ( pagedConnection == NULL ) {
|
|||
|
INTERNAL_ERROR(
|
|||
|
ERROR_LEVEL_EXPECTED,
|
|||
|
"SrvAllocateConnection: Unable to allocate %d bytes from"
|
|||
|
"paged pool",
|
|||
|
sizeof( PAGED_CONNECTION ),
|
|||
|
NULL
|
|||
|
);
|
|||
|
goto error_exit;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
IF_DEBUG(HEAP) {
|
|||
|
KdPrint(( "SrvAllocateConnection: Allocated connection at %lx\n",
|
|||
|
connection ));
|
|||
|
}
|
|||
|
|
|||
|
RtlZeroMemory( pagedConnection, sizeof(PAGED_CONNECTION) );
|
|||
|
|
|||
|
SET_BLOCK_TYPE_STATE_SIZE( connection, BlockTypeConnection, BlockStateInitializing, sizeof( CONNECTION ) );
|
|||
|
|
|||
|
connection->PagedConnection = pagedConnection;
|
|||
|
pagedConnection->PagedHeader.NonPagedBlock = connection;
|
|||
|
pagedConnection->PagedHeader.Type = BlockTypePagedConnection;
|
|||
|
|
|||
|
connection->BlockHeader.ReferenceCount = 2; // allow for Active status
|
|||
|
// and caller's pointer
|
|||
|
|
|||
|
InitializeListHead( &pagedConnection->TransactionList );
|
|||
|
|
|||
|
connection->SmbDialect = SmbDialectIllegal;
|
|||
|
connection->CachedFid = (ULONG)-1;
|
|||
|
|
|||
|
//
|
|||
|
// Allocate session table.
|
|||
|
//
|
|||
|
|
|||
|
SrvAllocateTable(
|
|||
|
&pagedConnection->SessionTable,
|
|||
|
SrvInitialSessionTableSize,
|
|||
|
FALSE
|
|||
|
);
|
|||
|
if ( pagedConnection->SessionTable.Table == NULL ) {
|
|||
|
goto error_exit;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Allocate tree connect table.
|
|||
|
//
|
|||
|
|
|||
|
SrvAllocateTable(
|
|||
|
&pagedConnection->TreeConnectTable,
|
|||
|
SrvInitialTreeTableSize,
|
|||
|
FALSE
|
|||
|
);
|
|||
|
if ( pagedConnection->TreeConnectTable.Table == NULL ) {
|
|||
|
goto error_exit;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Allocate file table.
|
|||
|
//
|
|||
|
|
|||
|
SrvAllocateTable(
|
|||
|
&connection->FileTable,
|
|||
|
SrvInitialFileTableSize,
|
|||
|
TRUE
|
|||
|
);
|
|||
|
if ( connection->FileTable.Table == NULL ) {
|
|||
|
goto error_exit;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Allocate search table.
|
|||
|
//
|
|||
|
|
|||
|
SrvAllocateTable(
|
|||
|
&pagedConnection->SearchTable,
|
|||
|
SrvInitialSearchTableSize,
|
|||
|
FALSE
|
|||
|
);
|
|||
|
if ( pagedConnection->SearchTable.Table == NULL ) {
|
|||
|
goto error_exit;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Initialize core search list heads
|
|||
|
//
|
|||
|
|
|||
|
InitializeListHead( &pagedConnection->CoreSearchList );
|
|||
|
|
|||
|
//
|
|||
|
// Initialize the locks that protect the connection and its data.
|
|||
|
//
|
|||
|
|
|||
|
INITIALIZE_SPIN_LOCK( &connection->SpinLock );
|
|||
|
INITIALIZE_SPIN_LOCK( &connection->Interlock );
|
|||
|
|
|||
|
INITIALIZE_LOCK( &connection->Lock, CONNECTION_LOCK_LEVEL, "ConnectionLock" );
|
|||
|
INITIALIZE_LOCK( &connection->LicenseLock, LICENSE_LOCK_LEVEL, "LicenseLock" );
|
|||
|
|
|||
|
//
|
|||
|
// Initialize the client machine name string.
|
|||
|
//
|
|||
|
|
|||
|
pagedConnection->ClientMachineNameString.Buffer =
|
|||
|
pagedConnection->LeadingSlashes;
|
|||
|
pagedConnection->ClientMachineNameString.Length = 0;
|
|||
|
pagedConnection->ClientMachineNameString.MaximumLength =
|
|||
|
(USHORT)((2 + COMPUTER_NAME_LENGTH + 1) * sizeof(WCHAR));
|
|||
|
|
|||
|
pagedConnection->LeadingSlashes[0] = '\\';
|
|||
|
pagedConnection->LeadingSlashes[1] = '\\';
|
|||
|
|
|||
|
//
|
|||
|
// Initialize the oem client machine name string
|
|||
|
//
|
|||
|
|
|||
|
connection->OemClientMachineNameString.Buffer =
|
|||
|
connection->OemClientMachineName;
|
|||
|
|
|||
|
connection->OemClientMachineNameString.MaximumLength =
|
|||
|
(USHORT)(COMPUTER_NAME_LENGTH + 1);
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Initialize count of sessions.
|
|||
|
//
|
|||
|
// *** Already done by RtlZeroMemory.
|
|||
|
|
|||
|
//pagedConnection->CurrentNumberOfSessions = 0;
|
|||
|
|
|||
|
//
|
|||
|
// Initialize the in-progress work item list, the outstanding
|
|||
|
// oplock breaks list, and the cached-after-close lists.
|
|||
|
//
|
|||
|
|
|||
|
InitializeListHead( &connection->InProgressWorkItemList );
|
|||
|
InitializeListHead( &connection->OplockWorkList );
|
|||
|
InitializeListHead( &connection->CachedOpenList );
|
|||
|
InitializeListHead( &connection->CachedDirectoryList );
|
|||
|
|
|||
|
//
|
|||
|
// Indicate that no IPX saved response buffer has been allocated.
|
|||
|
//
|
|||
|
|
|||
|
connection->LastResponse = connection->BuiltinSavedResponse;
|
|||
|
|
|||
|
//
|
|||
|
// Initialize the search hash table list.
|
|||
|
//
|
|||
|
|
|||
|
for ( i = 0; i < SEARCH_HASH_TABLE_SIZE ; i++ ) {
|
|||
|
InitializeListHead( &pagedConnection->SearchHashTable[i].ListHead );
|
|||
|
}
|
|||
|
|
|||
|
INITIALIZE_REFERENCE_HISTORY( connection );
|
|||
|
|
|||
|
INCREMENT_DEBUG_STAT2( SrvDbgStatistics.ConnectionInfo.Allocations );
|
|||
|
|
|||
|
return;
|
|||
|
|
|||
|
error_exit:
|
|||
|
|
|||
|
if ( pagedConnection != NULL ) {
|
|||
|
if ( pagedConnection->SessionTable.Table != NULL ) {
|
|||
|
SrvFreeTable( &pagedConnection->SessionTable );
|
|||
|
}
|
|||
|
if ( pagedConnection->TreeConnectTable.Table != NULL ) {
|
|||
|
SrvFreeTable( &pagedConnection->TreeConnectTable );
|
|||
|
}
|
|||
|
if ( pagedConnection->SearchTable.Table != NULL ) {
|
|||
|
SrvFreeTable( &pagedConnection->SearchTable );
|
|||
|
}
|
|||
|
FREE_HEAP( pagedConnection );
|
|||
|
}
|
|||
|
|
|||
|
if ( connection != NULL ) {
|
|||
|
if ( connection->FileTable.Table != NULL ) {
|
|||
|
SrvFreeTable( &connection->FileTable );
|
|||
|
}
|
|||
|
DEALLOCATE_NONPAGED_POOL( connection );
|
|||
|
*Connection = NULL;
|
|||
|
}
|
|||
|
|
|||
|
return;
|
|||
|
|
|||
|
} // SrvAllocateConnection
|
|||
|
|
|||
|
VOID
|
|||
|
SrvCloseConnection (
|
|||
|
IN PCONNECTION Connection,
|
|||
|
IN BOOLEAN RemoteDisconnect
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function closes a connection (virtual circuit).
|
|||
|
|
|||
|
*** This routine must NOT be entered with the connection lock held!
|
|||
|
It may be entered with the endpoint lock held.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Connection - Supplies a pointer to a Connection Block
|
|||
|
|
|||
|
RemoteDisconnect - Indicates whether this call is being made in
|
|||
|
response to a notification by the transport provider.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
PTABLE_HEADER tableHeader;
|
|||
|
PLIST_ENTRY listEntry;
|
|||
|
CSHORT i;
|
|||
|
KIRQL oldIrql;
|
|||
|
PRFCB rfcb;
|
|||
|
|
|||
|
ASSERT( !ExIsResourceAcquiredExclusive(&RESOURCE_OF(Connection->Lock)) );
|
|||
|
|
|||
|
ACQUIRE_LOCK( &Connection->Lock );
|
|||
|
|
|||
|
//
|
|||
|
// If the connection hasn't already been closed, do so now.
|
|||
|
//
|
|||
|
|
|||
|
if ( GET_BLOCK_STATE(Connection) == BlockStateActive ) {
|
|||
|
|
|||
|
#if SRVDBG29
|
|||
|
{
|
|||
|
ULONG conn = (ULONG)Connection;
|
|||
|
if (RemoteDisconnect) conn |= 1;
|
|||
|
if (Connection->DisconnectPending) conn |= 2;
|
|||
|
UpdateConnectionHistory( "CLOS", Connection->Endpoint, Connection );
|
|||
|
}
|
|||
|
#endif
|
|||
|
IF_DEBUG(TDI) KdPrint(( "Closing connection at %lx for %Z\n",
|
|||
|
Connection, &Connection->OemClientMachineNameString ));
|
|||
|
|
|||
|
SET_BLOCK_STATE( Connection, BlockStateClosing );
|
|||
|
|
|||
|
RELEASE_LOCK( &Connection->Lock );
|
|||
|
|
|||
|
//
|
|||
|
// If the connection is on the need-resource queue (waiting for
|
|||
|
// a work item) or the disconnect queue (a Disconnect having
|
|||
|
// been indicated by the transport), remove it now, and
|
|||
|
// dereference it. (Note that the connection can't go away yet,
|
|||
|
// because the initial reference is still there.)
|
|||
|
//
|
|||
|
|
|||
|
ACQUIRE_GLOBAL_SPIN_LOCK( Fsd, &oldIrql );
|
|||
|
|
|||
|
//
|
|||
|
// Invalidate the cached connection
|
|||
|
//
|
|||
|
|
|||
|
if ( Connection->OnNeedResourceQueue ) {
|
|||
|
|
|||
|
SrvRemoveEntryList(
|
|||
|
&SrvNeedResourceQueue,
|
|||
|
&Connection->ListEntry
|
|||
|
);
|
|||
|
|
|||
|
Connection->OnNeedResourceQueue = FALSE;
|
|||
|
DEBUG Connection->ReceivePending = FALSE;
|
|||
|
|
|||
|
RELEASE_GLOBAL_SPIN_LOCK( Fsd, oldIrql );
|
|||
|
|
|||
|
SrvDereferenceConnection( Connection );
|
|||
|
|
|||
|
} else if ( Connection->DisconnectPending ) {
|
|||
|
|
|||
|
SrvRemoveEntryList(
|
|||
|
&SrvDisconnectQueue,
|
|||
|
&Connection->ListEntry
|
|||
|
);
|
|||
|
|
|||
|
DEBUG Connection->DisconnectPending = FALSE;
|
|||
|
|
|||
|
RELEASE_GLOBAL_SPIN_LOCK( Fsd, oldIrql );
|
|||
|
|
|||
|
SrvDereferenceConnection( Connection );
|
|||
|
|
|||
|
//
|
|||
|
// If there's a disconnect pending, then don't try to shut
|
|||
|
// down the connection later.
|
|||
|
//
|
|||
|
|
|||
|
RemoteDisconnect = TRUE;
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
RELEASE_GLOBAL_SPIN_LOCK( Fsd, oldIrql );
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// If this is not a remote disconnect, issue a TdiDisconnect
|
|||
|
// request now.
|
|||
|
//
|
|||
|
// *** This is currently done as a synchronous request. It may
|
|||
|
// be better to do this asynchronously.
|
|||
|
//
|
|||
|
|
|||
|
if ( !RemoteDisconnect && !Connection->Endpoint->IsConnectionless ) {
|
|||
|
SrvDoDisconnect( Connection );
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Close all active sessions. (This also causes all open files
|
|||
|
// and pending transactions to be closed.)
|
|||
|
//
|
|||
|
|
|||
|
SrvCloseSessionsOnConnection( Connection, NULL );
|
|||
|
|
|||
|
//
|
|||
|
// Close all active tree connects.
|
|||
|
//
|
|||
|
// *** Reference the tree connect for the same reasons as we
|
|||
|
// referenced the session; see above.
|
|||
|
|
|||
|
tableHeader = &Connection->PagedConnection->TreeConnectTable;
|
|||
|
|
|||
|
ACQUIRE_LOCK( &Connection->Lock );
|
|||
|
|
|||
|
for ( i = 0; i < tableHeader->TableSize; i++ ) {
|
|||
|
|
|||
|
PTREE_CONNECT treeConnect =
|
|||
|
(PTREE_CONNECT)tableHeader->Table[i].Owner;
|
|||
|
|
|||
|
if ( treeConnect != NULL &&
|
|||
|
GET_BLOCK_STATE( treeConnect ) == BlockStateActive ) {
|
|||
|
|
|||
|
SrvReferenceTreeConnect( treeConnect );
|
|||
|
RELEASE_LOCK( &Connection->Lock );
|
|||
|
|
|||
|
SrvCloseTreeConnect( treeConnect );
|
|||
|
|
|||
|
SrvDereferenceTreeConnect( treeConnect );
|
|||
|
ACQUIRE_LOCK( &Connection->Lock );
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
RELEASE_LOCK( &Connection->Lock );
|
|||
|
|
|||
|
//
|
|||
|
// Cancel all outstanding oplock break requests.
|
|||
|
//
|
|||
|
|
|||
|
while ( (listEntry = ExInterlockedRemoveHeadList(
|
|||
|
&Connection->OplockWorkList,
|
|||
|
Connection->EndpointSpinLock
|
|||
|
)) != NULL ) {
|
|||
|
|
|||
|
//
|
|||
|
// Remove this work item from the connection queue and
|
|||
|
// return it to the free queue.
|
|||
|
//
|
|||
|
|
|||
|
rfcb = CONTAINING_RECORD( listEntry, RFCB, ListEntry );
|
|||
|
|
|||
|
#if DBG
|
|||
|
rfcb->ListEntry.Flink = rfcb->ListEntry.Blink = NULL;
|
|||
|
#endif
|
|||
|
SrvDereferenceRfcb( rfcb );
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Close RFCBs that are cached after having been closed by the
|
|||
|
// client.
|
|||
|
//
|
|||
|
|
|||
|
SrvCloseCachedRfcbsOnConnection( Connection );
|
|||
|
|
|||
|
//
|
|||
|
// Dereference the connection (to indicate that it's no longer
|
|||
|
// open). This may cause the connection block to be deleted.
|
|||
|
//
|
|||
|
|
|||
|
SrvDereferenceConnection( Connection );
|
|||
|
|
|||
|
INCREMENT_DEBUG_STAT2( SrvDbgStatistics.ConnectionInfo.Closes );
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
RELEASE_LOCK( &Connection->Lock );
|
|||
|
}
|
|||
|
|
|||
|
return;
|
|||
|
|
|||
|
} // SrvCloseConnection
|
|||
|
|
|||
|
|
|||
|
VOID
|
|||
|
SrvCloseConnectionsFromClient (
|
|||
|
IN PCONNECTION Connection
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This routine closes all connections from a given remote machine name
|
|||
|
except the connection passed in. This is used in the Session Setup
|
|||
|
SMB when the client indicates that he believes that he has exactly
|
|||
|
one connection to this server; if there are others, we know that
|
|||
|
they are not valid.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Connection - Address of connection to keep around. This is used
|
|||
|
for the machine name.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
CSHORT index;
|
|||
|
PENDPOINT endpoint;
|
|||
|
PLIST_ENTRY listEntry;
|
|||
|
PPAGED_CONNECTION pagedConnection = Connection->PagedConnection;
|
|||
|
PPAGED_CONNECTION testPagedConnection;
|
|||
|
PCONNECTION testConnection;
|
|||
|
BOOLEAN Connectionless = Connection->Endpoint->IsConnectionless == 1;
|
|||
|
|
|||
|
PAGED_CODE( );
|
|||
|
|
|||
|
//
|
|||
|
// We need to look at the name of every client for which the server
|
|||
|
// has a connection. Connection lists are stored off endpoints, so
|
|||
|
// walk the global endpoint list and the list of connections on each
|
|||
|
// endpoint.
|
|||
|
//
|
|||
|
|
|||
|
IF_DEBUG(TDI) {
|
|||
|
KdPrint(( "SrvCloseConnectionsFromClient entered for connection "
|
|||
|
"%lx, OemName %Z, looking for %Z\n", Connection,
|
|||
|
&Connection->OemClientMachineNameString,
|
|||
|
&pagedConnection->ClientMachineNameString));
|
|||
|
}
|
|||
|
|
|||
|
ACQUIRE_LOCK( &SrvEndpointLock );
|
|||
|
|
|||
|
listEntry = SrvEndpointList.ListHead.Flink;
|
|||
|
|
|||
|
while ( listEntry != &SrvEndpointList.ListHead ) {
|
|||
|
|
|||
|
|
|||
|
endpoint = CONTAINING_RECORD(
|
|||
|
listEntry,
|
|||
|
ENDPOINT,
|
|||
|
GlobalEndpointListEntry
|
|||
|
);
|
|||
|
|
|||
|
//
|
|||
|
// If this endpoint is closing, or if the types don't match,
|
|||
|
// skip to the next one.
|
|||
|
// Otherwise, reference the endpoint so that it can't go away.
|
|||
|
//
|
|||
|
|
|||
|
if ( GET_BLOCK_STATE(endpoint) != BlockStateActive ||
|
|||
|
endpoint->IsConnectionless != Connectionless ) {
|
|||
|
listEntry = listEntry->Flink;
|
|||
|
continue;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// If this endpoint doesn't have the same netbios name as the
|
|||
|
// endpoint of the passed-in connection then skip it. This is
|
|||
|
// to allow servers to have more than one name on the network.
|
|||
|
//
|
|||
|
|
|||
|
if( Connection->Endpoint->TransportAddress.Length !=
|
|||
|
endpoint->TransportAddress.Length ||
|
|||
|
|
|||
|
!RtlEqualMemory( Connection->Endpoint->TransportAddress.Buffer,
|
|||
|
endpoint->TransportAddress.Buffer,
|
|||
|
endpoint->TransportAddress.Length ) ) {
|
|||
|
|
|||
|
//
|
|||
|
// This connection is for an endpoint having a different network
|
|||
|
// name than does this endpoint. Skip this endpoint.
|
|||
|
//
|
|||
|
|
|||
|
listEntry = listEntry->Flink;
|
|||
|
continue;
|
|||
|
}
|
|||
|
|
|||
|
SrvReferenceEndpoint( endpoint );
|
|||
|
|
|||
|
//
|
|||
|
// Walk the endpoint's connection table.
|
|||
|
//
|
|||
|
|
|||
|
index = (CSHORT)-1;
|
|||
|
|
|||
|
while ( TRUE ) {
|
|||
|
|
|||
|
//
|
|||
|
// Get the next active connection in the table. If no more
|
|||
|
// are available, WalkConnectionTable returns NULL.
|
|||
|
// Otherwise, it returns a referenced pointer to a
|
|||
|
// connection.
|
|||
|
//
|
|||
|
|
|||
|
testConnection = WalkConnectionTable( endpoint, &index );
|
|||
|
if ( testConnection == NULL ) {
|
|||
|
break;
|
|||
|
}
|
|||
|
|
|||
|
if( testConnection == Connection ) {
|
|||
|
//
|
|||
|
// Skip ourselves!
|
|||
|
//
|
|||
|
SrvDereferenceConnection( testConnection );
|
|||
|
continue;
|
|||
|
}
|
|||
|
|
|||
|
testPagedConnection = testConnection->PagedConnection;
|
|||
|
|
|||
|
if( Connectionless ) {
|
|||
|
//
|
|||
|
// Connectionless clients match on IPX address...
|
|||
|
//
|
|||
|
|
|||
|
if( !RtlEqualMemory( &Connection->IpxAddress,
|
|||
|
&testConnection->IpxAddress,
|
|||
|
sizeof(Connection->IpxAddress) ) ) {
|
|||
|
|
|||
|
SrvDereferenceConnection( testConnection );
|
|||
|
continue;
|
|||
|
}
|
|||
|
|
|||
|
} else {
|
|||
|
//
|
|||
|
// All other clients match on computername
|
|||
|
//
|
|||
|
if ( RtlCompareUnicodeString(
|
|||
|
&testPagedConnection->ClientMachineNameString,
|
|||
|
&pagedConnection->ClientMachineNameString,
|
|||
|
TRUE
|
|||
|
) != 0 ) {
|
|||
|
SrvDereferenceConnection( testConnection );
|
|||
|
continue;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// We found a connection that we need to kill. We
|
|||
|
// have to release the lock in order to close it.
|
|||
|
//
|
|||
|
|
|||
|
RELEASE_LOCK( &SrvEndpointLock );
|
|||
|
|
|||
|
IF_DEBUG(TDI) {
|
|||
|
KdPrint(( "SrvCloseConnectionsFromClient closing "
|
|||
|
"connection %lx, MachineNameString %Z\n",
|
|||
|
testConnection,
|
|||
|
&testPagedConnection->ClientMachineNameString ));
|
|||
|
}
|
|||
|
|
|||
|
#if SRVDBG29
|
|||
|
UpdateConnectionHistory( "CFC1", testConnection->Endpoint, testConnection );
|
|||
|
UpdateConnectionHistory( "CFC2", testConnection->Endpoint, Connection );
|
|||
|
#endif
|
|||
|
SrvCloseConnection( testConnection, FALSE );
|
|||
|
|
|||
|
ACQUIRE_LOCK( &SrvEndpointLock );
|
|||
|
|
|||
|
//
|
|||
|
// Dereference the connection to account for the reference
|
|||
|
// from WalkConnectionTable.
|
|||
|
//
|
|||
|
|
|||
|
SrvDereferenceConnection( testConnection );
|
|||
|
|
|||
|
} // walk connection table
|
|||
|
|
|||
|
//
|
|||
|
// Capture a pointer to the next endpoint in the list (that one
|
|||
|
// can't go away because we hold the endpoint list), then
|
|||
|
// dereference the current endpoint.
|
|||
|
//
|
|||
|
|
|||
|
listEntry = listEntry->Flink;
|
|||
|
SrvDereferenceEndpoint( endpoint );
|
|||
|
|
|||
|
} // walk endpoint list
|
|||
|
|
|||
|
RELEASE_LOCK( &SrvEndpointLock );
|
|||
|
|
|||
|
} // SrvCloseConnectionsFromClient
|
|||
|
|
|||
|
|
|||
|
VOID
|
|||
|
SrvCloseFreeConnection (
|
|||
|
IN PCONNECTION Connection
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function closes a free connection. This is a connection that
|
|||
|
is not active -- has already been closed via SrvCloseConnection --
|
|||
|
and is no longer needed.
|
|||
|
|
|||
|
*** WARNING! This routine frees the storage occupied by Connection!
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Connection - Supplies a pointer to a Connection Block
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
PENDPOINT endpoint;
|
|||
|
PPAGED_CONNECTION pagedConnection = Connection->PagedConnection;
|
|||
|
KIRQL oldIrql;
|
|||
|
|
|||
|
ASSERT( Connection->BlockHeader.ReferenceCount == 0 );
|
|||
|
|
|||
|
endpoint = Connection->Endpoint;
|
|||
|
|
|||
|
ACQUIRE_LOCK( &SrvEndpointLock );
|
|||
|
|
|||
|
//
|
|||
|
// Remove the connection from the endpoint's connection table.
|
|||
|
//
|
|||
|
|
|||
|
if ( Connection->Sid != 0 ) {
|
|||
|
ACQUIRE_SPIN_LOCK( Connection->EndpointSpinLock, &oldIrql );
|
|||
|
SrvRemoveEntryTable(
|
|||
|
&endpoint->ConnectionTable,
|
|||
|
IPXSID_INDEX(Connection->Sid)
|
|||
|
);
|
|||
|
RELEASE_SPIN_LOCK( Connection->EndpointSpinLock, oldIrql );
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Decrement the count of connections for the endpoint.
|
|||
|
//
|
|||
|
|
|||
|
ExInterlockedAddUlong(
|
|||
|
&endpoint->TotalConnectionCount,
|
|||
|
(ULONG)-1,
|
|||
|
&GLOBAL_SPIN_LOCK(Fsd)
|
|||
|
);
|
|||
|
|
|||
|
RELEASE_LOCK( &SrvEndpointLock );
|
|||
|
|
|||
|
//
|
|||
|
// Close the connection file object. Remove the additional
|
|||
|
// reference.
|
|||
|
//
|
|||
|
|
|||
|
if ( !endpoint->IsConnectionless ) {
|
|||
|
SRVDBG_RELEASE_HANDLE( pagedConnection->ConnectionHandle, "CON", 1, Connection );
|
|||
|
SrvNtClose( pagedConnection->ConnectionHandle, FALSE );
|
|||
|
ObDereferenceObject( Connection->FileObject );
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Dereference the endpoint.
|
|||
|
//
|
|||
|
|
|||
|
SrvDereferenceEndpoint( endpoint );
|
|||
|
|
|||
|
//
|
|||
|
// Free the storage occupied by the connection.
|
|||
|
//
|
|||
|
|
|||
|
SrvFreeConnection( Connection );
|
|||
|
|
|||
|
return;
|
|||
|
|
|||
|
} // SrvCloseFreeConnection
|
|||
|
|
|||
|
|
|||
|
VOID
|
|||
|
SrvDereferenceConnection (
|
|||
|
IN PCONNECTION Connection
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function decrements the reference count on a connection. If the
|
|||
|
reference count goes to zero, the connection block is deleted.
|
|||
|
|
|||
|
The connection lock must not be held when this routine is called,
|
|||
|
because the global endpoint lock, which has a lower level, must be
|
|||
|
acquired.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Connection - Address of connection
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
ULONG oldCount;
|
|||
|
PENDPOINT endpoint;
|
|||
|
KIRQL oldIrql;
|
|||
|
PPAGED_CONNECTION pagedConnection = Connection->PagedConnection;
|
|||
|
|
|||
|
ASSERT( GET_BLOCK_TYPE( Connection ) == BlockTypeConnection );
|
|||
|
ASSERT( (LONG)Connection->BlockHeader.ReferenceCount > 0 );
|
|||
|
UPDATE_REFERENCE_HISTORY( Connection, TRUE );
|
|||
|
|
|||
|
//
|
|||
|
// Perform an interlocked decrement of the connection block's
|
|||
|
// reference count.
|
|||
|
//
|
|||
|
// *** Note that we don't hold a lock between the time we decrement
|
|||
|
// the reference count and the time we delete the connection
|
|||
|
// block. Normally this would imply that the FSD could
|
|||
|
// reference the block in between. However, the transport
|
|||
|
// provider guarantees that it won't deliver any more events
|
|||
|
// after a remote Disconnect event or after a local
|
|||
|
// TdiDisconnect request, and one of those two things has to
|
|||
|
// happen before the reference count can go to 0 (see
|
|||
|
// SrvCloseConnection).
|
|||
|
//
|
|||
|
|
|||
|
oldCount = ExInterlockedAddUlong(
|
|||
|
&Connection->BlockHeader.ReferenceCount,
|
|||
|
(ULONG)-1,
|
|||
|
Connection->EndpointSpinLock
|
|||
|
);
|
|||
|
IF_DEBUG(REFCNT) {
|
|||
|
KdPrint(( "Dereferencing connection %lx; old refcnt %lx\n",
|
|||
|
Connection, oldCount ));
|
|||
|
}
|
|||
|
|
|||
|
if ( oldCount == 1 ) {
|
|||
|
|
|||
|
//
|
|||
|
// The new reference count is 0, meaning that it's time to
|
|||
|
// delete this block.
|
|||
|
//
|
|||
|
|
|||
|
ASSERT( GET_BLOCK_STATE(Connection) != BlockStateActive );
|
|||
|
#if SRVDBG29
|
|||
|
if ( GET_BLOCK_STATE(Connection) != BlockStateClosing ) {
|
|||
|
KdPrint(( "SRV: Connection is not CLOSING with refcnt 0!\n" ));
|
|||
|
DbgBreakPoint( );
|
|||
|
}
|
|||
|
#endif
|
|||
|
|
|||
|
//
|
|||
|
// Free the space allocated for client Domain, OS Name, and
|
|||
|
// LAN type.
|
|||
|
//
|
|||
|
|
|||
|
if ( Connection->ClientOSType.Buffer != NULL ) {
|
|||
|
DEALLOCATE_NONPAGED_POOL( Connection->ClientOSType.Buffer );
|
|||
|
Connection->ClientOSType.Buffer = NULL;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Keep the WORK_QUEUE statistic correct
|
|||
|
//
|
|||
|
if( Connection->CurrentWorkQueue )
|
|||
|
InterlockedDecrement( &Connection->CurrentWorkQueue->CurrentClients );
|
|||
|
|
|||
|
ASSERT( Connection->CurrentWorkQueue->CurrentClients >= 0 );
|
|||
|
|
|||
|
endpoint = Connection->Endpoint;
|
|||
|
|
|||
|
ACQUIRE_LOCK( &SrvEndpointLock );
|
|||
|
|
|||
|
//
|
|||
|
// If the connection hasn't been marked as not reusable (e.g.,
|
|||
|
// because a disconnect failed), and the endpoint isn't closing,
|
|||
|
// and it isn't already "full" of free connections, put this
|
|||
|
// connection on the endpoint's free connection list.
|
|||
|
// Otherwise, close the connection file object and free the
|
|||
|
// connection block.
|
|||
|
//
|
|||
|
|
|||
|
if ( !Connection->NotReusable &&
|
|||
|
(GET_BLOCK_STATE(endpoint) == BlockStateActive) &&
|
|||
|
(endpoint->FreeConnectionCount < SrvFreeConnectionMaximum) ) {
|
|||
|
|
|||
|
//
|
|||
|
// Reinitialize the connection state.
|
|||
|
//
|
|||
|
// !!! Should probably reset the connection's table sizes,
|
|||
|
// if they've grown.
|
|||
|
//
|
|||
|
|
|||
|
SET_BLOCK_STATE( Connection, BlockStateInitializing );
|
|||
|
pagedConnection->LinkInfoValidTime.QuadPart = 0;
|
|||
|
pagedConnection->Throughput.QuadPart = 0;
|
|||
|
pagedConnection->Delay.QuadPart = 0;
|
|||
|
pagedConnection->ClientMachineNameString.Length = 0;
|
|||
|
Connection->ClientCapabilities = 0;
|
|||
|
Connection->SmbDialect = SmbDialectIllegal;
|
|||
|
Connection->DisconnectPending = FALSE;
|
|||
|
Connection->ReceivePending = FALSE;
|
|||
|
Connection->OplocksAlwaysDisabled = FALSE;
|
|||
|
Connection->CachedFid = (ULONG)-1;
|
|||
|
|
|||
|
//
|
|||
|
// Put the connection on the free list.
|
|||
|
//
|
|||
|
|
|||
|
ACQUIRE_GLOBAL_SPIN_LOCK( Fsd, &oldIrql );
|
|||
|
|
|||
|
#if SRVDBG29
|
|||
|
UpdateConnectionHistory( "KEEP", endpoint, Connection );
|
|||
|
#endif
|
|||
|
SrvInsertTailList(
|
|||
|
&endpoint->FreeConnectionList,
|
|||
|
&Connection->EndpointFreeListEntry
|
|||
|
);
|
|||
|
|
|||
|
endpoint->FreeConnectionCount++;
|
|||
|
|
|||
|
RELEASE_GLOBAL_SPIN_LOCK( Fsd, oldIrql );
|
|||
|
|
|||
|
RELEASE_LOCK( &SrvEndpointLock );
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
RELEASE_LOCK( &SrvEndpointLock );
|
|||
|
|
|||
|
SrvCloseFreeConnection( Connection );
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
return;
|
|||
|
|
|||
|
} // SrvDereferenceConnection
|
|||
|
|
|||
|
|
|||
|
VOID
|
|||
|
SrvFreeConnection (
|
|||
|
IN PCONNECTION Connection
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function returns a Connection Block to the system nonpaged
|
|||
|
pool.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Connection - Address of connection
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
PSINGLE_LIST_ENTRY listEntry;
|
|||
|
PNONPAGED_HEADER header;
|
|||
|
PTRANSACTION transaction;
|
|||
|
PPAGED_CONNECTION pagedConnection = Connection->PagedConnection;
|
|||
|
|
|||
|
PAGED_CODE( );
|
|||
|
|
|||
|
//
|
|||
|
// Free cached transactions.
|
|||
|
//
|
|||
|
|
|||
|
listEntry = ExInterlockedPopEntrySList( &Connection->CachedTransactionList,
|
|||
|
&Connection->SpinLock );
|
|||
|
|
|||
|
while ( listEntry != NULL ) {
|
|||
|
|
|||
|
header = CONTAINING_RECORD( listEntry, NONPAGED_HEADER, ListEntry );
|
|||
|
transaction = header->PagedBlock;
|
|||
|
|
|||
|
DEALLOCATE_NONPAGED_POOL( header );
|
|||
|
FREE_HEAP( transaction );
|
|||
|
INCREMENT_DEBUG_STAT( SrvDbgStatistics.TransactionInfo.Frees );
|
|||
|
|
|||
|
listEntry = ExInterlockedPopEntrySList(
|
|||
|
&Connection->CachedTransactionList,
|
|||
|
&Connection->SpinLock );
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Free the search, session, tree, and file tables.
|
|||
|
//
|
|||
|
|
|||
|
SrvFreeTable( &pagedConnection->SearchTable );
|
|||
|
SrvFreeTable( &Connection->FileTable );
|
|||
|
SrvFreeTable( &pagedConnection->TreeConnectTable );
|
|||
|
SrvFreeTable( &pagedConnection->SessionTable );
|
|||
|
|
|||
|
//
|
|||
|
// Free the IPX saved response buffer, if there is one.
|
|||
|
//
|
|||
|
|
|||
|
if ( Connection->LastResponse != Connection->BuiltinSavedResponse ) {
|
|||
|
DEALLOCATE_NONPAGED_POOL( Connection->LastResponse );
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Delete the lock on the connection.
|
|||
|
//
|
|||
|
|
|||
|
DELETE_LOCK( &Connection->Lock );
|
|||
|
|
|||
|
//
|
|||
|
// Delete the license server lock
|
|||
|
//
|
|||
|
DELETE_LOCK( &Connection->LicenseLock );
|
|||
|
|
|||
|
//
|
|||
|
// Free the connection block.
|
|||
|
//
|
|||
|
|
|||
|
DEBUG SET_BLOCK_TYPE_STATE_SIZE( Connection, BlockTypeGarbage, BlockStateDead, -1 );
|
|||
|
DEBUG Connection->BlockHeader.ReferenceCount = (ULONG)-1;
|
|||
|
TERMINATE_REFERENCE_HISTORY( Connection );
|
|||
|
|
|||
|
FREE_HEAP( pagedConnection );
|
|||
|
DEALLOCATE_NONPAGED_POOL( Connection );
|
|||
|
IF_DEBUG(HEAP) {
|
|||
|
KdPrint(( "SrvFreeConnection: Freed connection block at %lx\n",
|
|||
|
Connection ));
|
|||
|
}
|
|||
|
|
|||
|
INCREMENT_DEBUG_STAT2( SrvDbgStatistics.ConnectionInfo.Frees );
|
|||
|
|
|||
|
return;
|
|||
|
|
|||
|
} // SrvFreeConnection
|
|||
|
|
|||
|
#if DBG
|
|||
|
|
|||
|
NTSTATUS
|
|||
|
SrvQueryConnections (
|
|||
|
OUT PVOID Buffer,
|
|||
|
IN ULONG BufferLength,
|
|||
|
OUT PULONG BytesWritten
|
|||
|
)
|
|||
|
|
|||
|
{
|
|||
|
CSHORT index;
|
|||
|
PLIST_ENTRY listEntry;
|
|||
|
PLIST_ENTRY connectionListEntry;
|
|||
|
PBLOCK_INFORMATION blockInfo = Buffer;
|
|||
|
PENDPOINT endpoint;
|
|||
|
PCONNECTION connection;
|
|||
|
KIRQL oldIrql;
|
|||
|
|
|||
|
*BytesWritten = 0;
|
|||
|
|
|||
|
//
|
|||
|
// We need to look at the name of every client for which the server
|
|||
|
// has a connection. Connection lists are stored off endpoints, so
|
|||
|
// walk the global endpoint list and the list of connections on each
|
|||
|
// endpoint.
|
|||
|
//
|
|||
|
|
|||
|
ACQUIRE_LOCK( &SrvEndpointLock );
|
|||
|
|
|||
|
listEntry = SrvEndpointList.ListHead.Flink;
|
|||
|
|
|||
|
while ( listEntry != &SrvEndpointList.ListHead ) {
|
|||
|
|
|||
|
endpoint = CONTAINING_RECORD(
|
|||
|
listEntry,
|
|||
|
ENDPOINT,
|
|||
|
GlobalEndpointListEntry
|
|||
|
);
|
|||
|
|
|||
|
//
|
|||
|
// If this endpoint is closing, skip to the next one.
|
|||
|
// Otherwise, reference the endpoint so that it can't go away.
|
|||
|
//
|
|||
|
|
|||
|
if ( GET_BLOCK_STATE(endpoint) != BlockStateActive ) {
|
|||
|
listEntry = listEntry->Flink;
|
|||
|
continue;
|
|||
|
}
|
|||
|
|
|||
|
SrvReferenceEndpoint( endpoint );
|
|||
|
|
|||
|
//
|
|||
|
// Put information about the endpoint into the output buffer.
|
|||
|
//
|
|||
|
|
|||
|
if ( (PCHAR)(blockInfo + 1) <= (PCHAR)Buffer + BufferLength ) {
|
|||
|
blockInfo->Block = endpoint;
|
|||
|
blockInfo->BlockType = (ULONG)BlockTypeEndpoint;
|
|||
|
blockInfo->BlockState = (ULONG)endpoint->BlockHeader.State;
|
|||
|
blockInfo->ReferenceCount = endpoint->BlockHeader.ReferenceCount;
|
|||
|
blockInfo++;
|
|||
|
} else {
|
|||
|
SrvDereferenceEndpoint( endpoint );
|
|||
|
RELEASE_LOCK( &SrvEndpointLock );
|
|||
|
return STATUS_BUFFER_OVERFLOW;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Walk the connection table, writing information about each
|
|||
|
// connection to the output buffer.
|
|||
|
//
|
|||
|
|
|||
|
index = (CSHORT)-1;
|
|||
|
|
|||
|
while ( TRUE ) {
|
|||
|
|
|||
|
//
|
|||
|
// Get the next active connection in the table. If no more
|
|||
|
// are available, WalkConnectionTable returns NULL.
|
|||
|
// Otherwise, it returns a referenced pointer to a
|
|||
|
// connection.
|
|||
|
//
|
|||
|
|
|||
|
connection = WalkConnectionTable( endpoint, &index );
|
|||
|
if ( connection == NULL ) {
|
|||
|
break;
|
|||
|
}
|
|||
|
|
|||
|
if ( (PCHAR)(blockInfo + 1) <= (PCHAR)Buffer + BufferLength ) {
|
|||
|
blockInfo->Block = connection;
|
|||
|
blockInfo->BlockType = (ULONG)BlockTypeConnection;
|
|||
|
blockInfo->BlockState = (ULONG)connection->BlockHeader.State;
|
|||
|
blockInfo->ReferenceCount =
|
|||
|
connection->BlockHeader.ReferenceCount;
|
|||
|
blockInfo++;
|
|||
|
SrvDereferenceConnection( connection );
|
|||
|
} else {
|
|||
|
SrvDereferenceConnection( connection );
|
|||
|
SrvDereferenceEndpoint( endpoint );
|
|||
|
RELEASE_LOCK( &SrvEndpointLock );
|
|||
|
return STATUS_BUFFER_OVERFLOW;
|
|||
|
}
|
|||
|
|
|||
|
} // walk connection list
|
|||
|
|
|||
|
//
|
|||
|
// Walk the free connection list, writing information about each
|
|||
|
// connection to the output buffer.
|
|||
|
//
|
|||
|
|
|||
|
ACQUIRE_GLOBAL_SPIN_LOCK( Fsd, &oldIrql );
|
|||
|
|
|||
|
for ( connectionListEntry = endpoint->FreeConnectionList.Flink;
|
|||
|
connectionListEntry != &endpoint->FreeConnectionList;
|
|||
|
connectionListEntry = connectionListEntry->Flink ) {
|
|||
|
|
|||
|
connection = CONTAINING_RECORD(
|
|||
|
connectionListEntry,
|
|||
|
CONNECTION,
|
|||
|
EndpointFreeListEntry
|
|||
|
);
|
|||
|
|
|||
|
if ( (PCHAR)(blockInfo + 1) <= (PCHAR)Buffer + BufferLength ) {
|
|||
|
blockInfo->Block = connection;
|
|||
|
blockInfo->BlockType = (ULONG)BlockTypeConnection;
|
|||
|
blockInfo->BlockState = (ULONG)connection->BlockHeader.State;
|
|||
|
blockInfo->ReferenceCount =
|
|||
|
connection->BlockHeader.ReferenceCount;
|
|||
|
blockInfo++;
|
|||
|
} else {
|
|||
|
RELEASE_GLOBAL_SPIN_LOCK( Fsd, oldIrql );
|
|||
|
RELEASE_LOCK( &SrvEndpointLock );
|
|||
|
return STATUS_BUFFER_OVERFLOW;
|
|||
|
}
|
|||
|
|
|||
|
} // walk free connection list
|
|||
|
|
|||
|
RELEASE_GLOBAL_SPIN_LOCK( Fsd, oldIrql );
|
|||
|
|
|||
|
//
|
|||
|
// Capture a pointer to the next endpoint in the list (that one
|
|||
|
// can't go away because we hold the endpoint list), then
|
|||
|
// dereference the current endpoint.
|
|||
|
//
|
|||
|
|
|||
|
listEntry = listEntry->Flink;
|
|||
|
SrvDereferenceEndpoint( endpoint );
|
|||
|
|
|||
|
} // walk endpoint list
|
|||
|
|
|||
|
RELEASE_LOCK( &SrvEndpointLock );
|
|||
|
|
|||
|
*BytesWritten = (ULONG)blockInfo - (ULONG)Buffer;
|
|||
|
|
|||
|
return STATUS_SUCCESS;
|
|||
|
|
|||
|
} // SrvQueryConnections
|
|||
|
#endif // if DBG
|
|||
|
|