931 lines
23 KiB
C
931 lines
23 KiB
C
|
/*++
|
|||
|
|
|||
|
Copyright (c) 1990 Microsoft Corporation
|
|||
|
|
|||
|
Module Name:
|
|||
|
|
|||
|
blktrans.c
|
|||
|
|
|||
|
Abstract:
|
|||
|
|
|||
|
This module implements routines for managing transaction blocks.
|
|||
|
|
|||
|
Author:
|
|||
|
|
|||
|
Chuck Lenzmeier (chuckl) 23-Feb-1990
|
|||
|
|
|||
|
Revision History:
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
#include "precomp.h"
|
|||
|
#pragma hdrstop
|
|||
|
|
|||
|
#define BugCheckFileId SRV_FILE_BLKTRANS
|
|||
|
|
|||
|
//
|
|||
|
// If a transaction block has no space for extra data, and its name is
|
|||
|
// the null string, it is eligible to be cached when it is free. This
|
|||
|
// means that instead of freeing the transaction block, a pointer to the
|
|||
|
// block is stored in the connection block.
|
|||
|
//
|
|||
|
// An eligible transaction will be four bytes longer than the base
|
|||
|
// transaction block size. This allows for a Unicode string terminator
|
|||
|
// padded to a longword.
|
|||
|
//
|
|||
|
|
|||
|
#define CACHED_TRANSACTION_BLOCK_SIZE sizeof(TRANSACTION) + 4
|
|||
|
|
|||
|
//
|
|||
|
// We allow up to four transactions to be cached.
|
|||
|
//
|
|||
|
// !!! This should be a configuration parameter.
|
|||
|
//
|
|||
|
|
|||
|
#define CACHED_TRANSACTION_LIMIT 4
|
|||
|
|
|||
|
#ifdef ALLOC_PRAGMA
|
|||
|
#pragma alloc_text( PAGE, SrvCloseTransaction )
|
|||
|
#pragma alloc_text( PAGE, SrvCloseTransactionsOnSession )
|
|||
|
#pragma alloc_text( PAGE, SrvCloseTransactionsOnTree )
|
|||
|
#pragma alloc_text( PAGE, SrvDereferenceTransaction )
|
|||
|
#pragma alloc_text( PAGE, SrvAllocateTransaction )
|
|||
|
#pragma alloc_text( PAGE, SrvFreeTransaction )
|
|||
|
#endif
|
|||
|
#if 0
|
|||
|
#endif
|
|||
|
|
|||
|
|
|||
|
VOID
|
|||
|
SrvAllocateTransaction (
|
|||
|
OUT PTRANSACTION *Transaction,
|
|||
|
OUT PVOID *TrailingBytes,
|
|||
|
IN PCONNECTION Connection,
|
|||
|
IN CLONG TrailingByteCount,
|
|||
|
IN PVOID TransactionName,
|
|||
|
IN PVOID EndOfSourceBuffer OPTIONAL,
|
|||
|
IN BOOLEAN SourceIsUnicode,
|
|||
|
IN BOOLEAN RemoteApiRequest
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function allocates a Transaction block from the FSP heap.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Transaction - Returns a pointer to the transaction block, or NULL if
|
|||
|
no heap space was available.
|
|||
|
|
|||
|
TrailingBytes - Returns a pointer to the trailing bytes allocated at
|
|||
|
the end of the transaction block. Invalid if *Transaction is
|
|||
|
NULL.
|
|||
|
|
|||
|
TrailingByteCount - Supplies the count of bytes (not including the
|
|||
|
transaction name) to be allocated at the tail end of the
|
|||
|
transaction block.
|
|||
|
|
|||
|
TransactionName - Supplies a pointer to the null-terminated
|
|||
|
transaction name string. Is SourceIsUnicode is TRUE, this must
|
|||
|
be an aligned pointer.
|
|||
|
|
|||
|
EndOfSourceBuffer - A pointer to the end of the SMB buffer. Used to
|
|||
|
protect the server from accessing beyond the end of the SMB buffer,
|
|||
|
if the format is invalid. If NULL, indicates that checking is not
|
|||
|
necessary.
|
|||
|
|
|||
|
SourceIsUnicode - Indicates whether the TransactionName buffer contains
|
|||
|
Unicode characters.
|
|||
|
|
|||
|
RemoteApiRequest - TRUE if this is a remote API request and should
|
|||
|
hence be allocated from the shared memory that XACTSRV can see.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
USHORT nameLength;
|
|||
|
CLONG extraLength;
|
|||
|
CLONG blockSize;
|
|||
|
PSINGLE_LIST_ENTRY listEntry;
|
|||
|
PNONPAGED_HEADER header;
|
|||
|
PTRANSACTION transaction;
|
|||
|
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
//
|
|||
|
// Get the length of the name (in bytes) including the null terminator.
|
|||
|
//
|
|||
|
|
|||
|
if ( EndOfSourceBuffer == NULL ) {
|
|||
|
|
|||
|
//
|
|||
|
// No checking is required
|
|||
|
//
|
|||
|
|
|||
|
if ( SourceIsUnicode ) {
|
|||
|
nameLength = (USHORT)(wcslen( (PWCH)TransactionName ) + 1);
|
|||
|
} else {
|
|||
|
nameLength = (USHORT)(strlen( (PCHAR)TransactionName ) + 1);
|
|||
|
}
|
|||
|
nameLength *= sizeof(WCHAR);
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
nameLength = SrvGetStringLength(
|
|||
|
TransactionName,
|
|||
|
EndOfSourceBuffer,
|
|||
|
SourceIsUnicode,
|
|||
|
TRUE // include null terminator
|
|||
|
);
|
|||
|
|
|||
|
if ( nameLength == (USHORT)-1 ) {
|
|||
|
|
|||
|
//
|
|||
|
// If the name is screwed up, assume L'\0'
|
|||
|
//
|
|||
|
|
|||
|
nameLength = sizeof(WCHAR);
|
|||
|
|
|||
|
} else if ( !SourceIsUnicode ) {
|
|||
|
|
|||
|
nameLength *= sizeof(WCHAR);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
extraLength = ((nameLength + 3) & ~3) + TrailingByteCount;
|
|||
|
blockSize = sizeof(TRANSACTION) + extraLength;
|
|||
|
|
|||
|
//
|
|||
|
// Attempt to allocate from the heap.
|
|||
|
//
|
|||
|
|
|||
|
if ( !RemoteApiRequest ) {
|
|||
|
|
|||
|
//
|
|||
|
// If the extra length required allows us to use a cached
|
|||
|
// transaction block, try to get one of those first.
|
|||
|
//
|
|||
|
|
|||
|
if ( blockSize == CACHED_TRANSACTION_BLOCK_SIZE ) {
|
|||
|
|
|||
|
listEntry = ExInterlockedPopEntrySList( &Connection->CachedTransactionList, &Connection->SpinLock );
|
|||
|
|
|||
|
if ( listEntry != NULL ) {
|
|||
|
|
|||
|
ASSERT( Connection->CachedTransactionCount > 0 );
|
|||
|
InterlockedDecrement( &Connection->CachedTransactionCount );
|
|||
|
|
|||
|
header = CONTAINING_RECORD(
|
|||
|
listEntry,
|
|||
|
NONPAGED_HEADER,
|
|||
|
ListEntry
|
|||
|
);
|
|||
|
transaction = header->PagedBlock;
|
|||
|
|
|||
|
IF_DEBUG(HEAP) {
|
|||
|
SrvPrint1( "SrvAllocateTransaction: Found cached "
|
|||
|
"transaction block at %lx\n", transaction );
|
|||
|
}
|
|||
|
|
|||
|
*Transaction = transaction;
|
|||
|
goto got_cached_transaction;
|
|||
|
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
transaction = ALLOCATE_HEAP( blockSize, BlockTypeTransaction );
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
NTSTATUS status; // ignore this
|
|||
|
transaction = SrvXsAllocateHeap( blockSize, &status );
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
*Transaction = transaction;
|
|||
|
|
|||
|
if ( transaction == NULL ) {
|
|||
|
INTERNAL_ERROR(
|
|||
|
ERROR_LEVEL_EXPECTED,
|
|||
|
"SrvAllocateTransaction: Unable to allocate %d bytes from heap.",
|
|||
|
blockSize,
|
|||
|
NULL
|
|||
|
);
|
|||
|
|
|||
|
// An error will be logged by the caller
|
|||
|
|
|||
|
return;
|
|||
|
}
|
|||
|
IF_DEBUG(HEAP) {
|
|||
|
SrvPrint1( "SrvAllocateTransaction: Allocated transaction block "
|
|||
|
"at %lx\n", transaction );
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Allocate the nonpaged header.
|
|||
|
//
|
|||
|
|
|||
|
header = ALLOCATE_NONPAGED_POOL(
|
|||
|
sizeof(NONPAGED_HEADER),
|
|||
|
BlockTypeNonpagedHeader
|
|||
|
);
|
|||
|
if ( header == NULL ) {
|
|||
|
INTERNAL_ERROR(
|
|||
|
ERROR_LEVEL_EXPECTED,
|
|||
|
"SrvAllocateTransaction: Unable to allocate %d bytes from pool.",
|
|||
|
sizeof( NONPAGED_HEADER ),
|
|||
|
NULL
|
|||
|
);
|
|||
|
if ( !RemoteApiRequest ) {
|
|||
|
FREE_HEAP( transaction );
|
|||
|
} else {
|
|||
|
SrvXsFreeHeap( transaction );
|
|||
|
}
|
|||
|
*Transaction = NULL;
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
header->Type = BlockTypeTransaction;
|
|||
|
header->PagedBlock = transaction;
|
|||
|
|
|||
|
INCREMENT_DEBUG_STAT( SrvDbgStatistics.TransactionInfo.Allocations );
|
|||
|
|
|||
|
#if SRVDBG2
|
|||
|
transaction->BlockHeader.ReferenceCount = 2; // for INITIALIZE_REFERENCE_HISTORY
|
|||
|
#endif
|
|||
|
INITIALIZE_REFERENCE_HISTORY( transaction );
|
|||
|
|
|||
|
got_cached_transaction:
|
|||
|
|
|||
|
RtlZeroMemory( transaction, sizeof(TRANSACTION) );
|
|||
|
|
|||
|
transaction->NonpagedHeader = header;
|
|||
|
|
|||
|
SET_BLOCK_TYPE_STATE_SIZE( transaction, BlockTypeTransaction, BlockStateActive, blockSize );
|
|||
|
header->ReferenceCount = 2; // allow for Active status and caller's pointer
|
|||
|
|
|||
|
transaction->RemoteApiRequest = RemoteApiRequest;
|
|||
|
|
|||
|
//
|
|||
|
// Put transaction name after main part of transaction block.
|
|||
|
//
|
|||
|
|
|||
|
transaction->TransactionName.Buffer = (PWCH)( transaction + 1 );
|
|||
|
transaction->TransactionName.MaximumLength = (USHORT)nameLength;
|
|||
|
transaction->TransactionName.Length = (USHORT)(nameLength - sizeof(WCHAR));
|
|||
|
|
|||
|
if ( nameLength == sizeof(WCHAR) ) {
|
|||
|
|
|||
|
transaction->TransactionName.Buffer = L'\0';
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
if ( SourceIsUnicode ) {
|
|||
|
|
|||
|
RtlCopyMemory(
|
|||
|
(PVOID)transaction->TransactionName.Buffer,
|
|||
|
TransactionName,
|
|||
|
nameLength
|
|||
|
);
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
ANSI_STRING ansiName;
|
|||
|
|
|||
|
ansiName.Buffer = (PCHAR)TransactionName;
|
|||
|
ansiName.Length = (nameLength / sizeof(WCHAR)) - 1;
|
|||
|
|
|||
|
RtlOemStringToUnicodeString(
|
|||
|
&transaction->TransactionName,
|
|||
|
&ansiName,
|
|||
|
FALSE
|
|||
|
);
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Set address of trailing bytes.
|
|||
|
//
|
|||
|
|
|||
|
*TrailingBytes = (PCHAR)transaction + sizeof(TRANSACTION) +
|
|||
|
((nameLength + 3) & ~3);
|
|||
|
|
|||
|
return;
|
|||
|
|
|||
|
} // SrvAllocateTransaction
|
|||
|
|
|||
|
|
|||
|
VOID
|
|||
|
SrvCloseTransaction (
|
|||
|
IN PTRANSACTION Transaction
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This routine closes a pending transaction. It sets the state of the
|
|||
|
transaction to Closing and dereferences the transaction block. The
|
|||
|
block will be destroyed as soon as all other references to it are
|
|||
|
eliminated.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Transaction - Supplies a pointer to the transaction block that is
|
|||
|
to be closed.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
PAGED_CODE( );
|
|||
|
|
|||
|
ACQUIRE_LOCK( &Transaction->Connection->Lock );
|
|||
|
|
|||
|
if ( GET_BLOCK_STATE(Transaction) == BlockStateActive ) {
|
|||
|
|
|||
|
IF_DEBUG(BLOCK1) {
|
|||
|
SrvPrint1( "Closing transaction at %lx\n", Transaction );
|
|||
|
}
|
|||
|
|
|||
|
SET_BLOCK_STATE( Transaction, BlockStateClosing );
|
|||
|
|
|||
|
RELEASE_LOCK( &Transaction->Connection->Lock );
|
|||
|
|
|||
|
//
|
|||
|
// If the transaction request indicated that the tree connect
|
|||
|
// should be closed on completion, do so now.
|
|||
|
//
|
|||
|
|
|||
|
if ( Transaction->Flags & SMB_TRANSACTION_DISCONNECT ) {
|
|||
|
SrvCloseTreeConnect( Transaction->TreeConnect );
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Dereference the transaction (to indicate that it's no longer
|
|||
|
// open).
|
|||
|
//
|
|||
|
|
|||
|
SrvDereferenceTransaction( Transaction );
|
|||
|
|
|||
|
INCREMENT_DEBUG_STAT( SrvDbgStatistics.TransactionInfo.Closes );
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
RELEASE_LOCK( &Transaction->Connection->Lock );
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
return;
|
|||
|
|
|||
|
} // SrvCloseTransaction
|
|||
|
|
|||
|
|
|||
|
VOID
|
|||
|
SrvCloseTransactionsOnSession (
|
|||
|
PSESSION Session
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This routine closes all pending transactions that are "owned" by the
|
|||
|
specified session. It walks the transaction list of the connection
|
|||
|
that owns the session. Each transaction in that list that is owned
|
|||
|
by the session is closed.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Session - Supplies a pointer to the session block for which
|
|||
|
transactions are to be closed.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
PCONNECTION connection;
|
|||
|
PPAGED_CONNECTION pagedConnection;
|
|||
|
PLIST_ENTRY entry;
|
|||
|
PTRANSACTION previousTransaction;
|
|||
|
PTRANSACTION transaction;
|
|||
|
|
|||
|
PAGED_CODE( );
|
|||
|
|
|||
|
//
|
|||
|
// Get the address of the owning connection.
|
|||
|
//
|
|||
|
|
|||
|
connection = Session->Connection;
|
|||
|
pagedConnection = connection->PagedConnection;
|
|||
|
|
|||
|
//
|
|||
|
// Walk the transaction list, looking for transactions owned by the
|
|||
|
// specified session.
|
|||
|
//
|
|||
|
// *** This routine is complicated by the following requirements:
|
|||
|
//
|
|||
|
// 1) We must hold the transaction lock while looking at the
|
|||
|
// list, and we must ensure the integrity of the list as
|
|||
|
// we walk it.
|
|||
|
//
|
|||
|
// 2) The transaction lock must NOT be held when closing or
|
|||
|
// dereferencing a transaction, because its lock level is
|
|||
|
// higher than that of other locks that may need to be
|
|||
|
// taken out as a result of the close or dereference.
|
|||
|
//
|
|||
|
// We work around these problems in the following way:
|
|||
|
//
|
|||
|
// 1) We hold the transaction lock while we search for a
|
|||
|
// transaction to close.
|
|||
|
//
|
|||
|
// 2) We reference the transaction we're about to close, then
|
|||
|
// release the lock. This prevents someone else from
|
|||
|
// invalidating the transaction after we release the lock
|
|||
|
// but before we close it ourselves.
|
|||
|
//
|
|||
|
// 3) We close the transaction. Our extra reference to the
|
|||
|
// transaction prevents it from being deleted. This also
|
|||
|
// keeps it on the transaction list.
|
|||
|
//
|
|||
|
// 4) We retake the lock, find another transaction (using the
|
|||
|
// previous transaction as a starting point), reference it,
|
|||
|
// then release the lock.
|
|||
|
//
|
|||
|
// 5) We dereference the original transaction and go to step 3.
|
|||
|
//
|
|||
|
// Note that the loop below is NOT structured in exactly the same
|
|||
|
// way as the steps above are listed.
|
|||
|
//
|
|||
|
|
|||
|
entry = &pagedConnection->TransactionList;
|
|||
|
previousTransaction = NULL;
|
|||
|
|
|||
|
while ( TRUE ) {
|
|||
|
|
|||
|
ACQUIRE_LOCK( &connection->Lock );
|
|||
|
|
|||
|
//
|
|||
|
// Find a transaction that is owned by the specified session.
|
|||
|
//
|
|||
|
|
|||
|
while ( TRUE ) {
|
|||
|
|
|||
|
//
|
|||
|
// Get the address of the next list entry. If we hit the
|
|||
|
// end of the list, exit the inner loop.
|
|||
|
//
|
|||
|
|
|||
|
entry = entry->Flink;
|
|||
|
if ( entry == &pagedConnection->TransactionList ) break;
|
|||
|
|
|||
|
//
|
|||
|
// Get the address of the transaction. If it's owned by
|
|||
|
// the specified session and currently active, exit the
|
|||
|
// inner loop. If it is closing don't touch it.
|
|||
|
//
|
|||
|
|
|||
|
transaction = CONTAINING_RECORD(
|
|||
|
entry,
|
|||
|
TRANSACTION,
|
|||
|
ConnectionListEntry
|
|||
|
);
|
|||
|
|
|||
|
if ( transaction->Session == Session &&
|
|||
|
GET_BLOCK_STATE(transaction) == BlockStateActive) {
|
|||
|
|
|||
|
break;
|
|||
|
}
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// If we hit the end of the list without finding a transaction
|
|||
|
// that was owned by the specified session, exit the main loop.
|
|||
|
//
|
|||
|
|
|||
|
if ( entry == &pagedConnection->TransactionList ) break;
|
|||
|
|
|||
|
//
|
|||
|
// Reference the transaction to ensure that it isn't deleted
|
|||
|
// when we close it.
|
|||
|
//
|
|||
|
|
|||
|
SrvReferenceTransaction( transaction );
|
|||
|
|
|||
|
//
|
|||
|
// Unlock the transaction list, so that we can dereference the
|
|||
|
// previous transaction and close the current one.
|
|||
|
//
|
|||
|
|
|||
|
RELEASE_LOCK( &connection->Lock );
|
|||
|
|
|||
|
//
|
|||
|
// If this is not the first matching transaction that we've
|
|||
|
// found, dereference the previous one now.
|
|||
|
//
|
|||
|
|
|||
|
if ( previousTransaction != NULL ) {
|
|||
|
SrvDereferenceTransaction( previousTransaction );
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Close the current transaction and mark that we need to
|
|||
|
// dereference it.
|
|||
|
//
|
|||
|
|
|||
|
SrvCloseTransaction( transaction );
|
|||
|
|
|||
|
previousTransaction = transaction;
|
|||
|
|
|||
|
//
|
|||
|
// Go find another matching transaction.
|
|||
|
//
|
|||
|
|
|||
|
} // while ( TRUE )
|
|||
|
|
|||
|
//
|
|||
|
// We have hit the end of the transaction list. Release the
|
|||
|
// transaction lock. If we have a transaction that needs to be
|
|||
|
// dereferenced, do so. Then return to the caller.
|
|||
|
//
|
|||
|
|
|||
|
RELEASE_LOCK( &connection->Lock );
|
|||
|
|
|||
|
if ( previousTransaction != NULL ) {
|
|||
|
SrvDereferenceTransaction( previousTransaction );
|
|||
|
}
|
|||
|
|
|||
|
return;
|
|||
|
|
|||
|
} // SrvCloseTransactionsOnSession
|
|||
|
|
|||
|
|
|||
|
VOID
|
|||
|
SrvCloseTransactionsOnTree (
|
|||
|
IN PTREE_CONNECT TreeConnect
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This routine closes all pending transactions that are "owned" by the
|
|||
|
specified tree connect. It walks the transaction list of the
|
|||
|
connection that owns the tree connect. Each transaction in that
|
|||
|
list that is owned by the tree connect is closed.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
TreeConnect - Supplies a pointer to the tree connect block for which
|
|||
|
transactions are to be closed.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
PCONNECTION connection;
|
|||
|
PPAGED_CONNECTION pagedConnection;
|
|||
|
PLIST_ENTRY entry;
|
|||
|
PTRANSACTION previousTransaction;
|
|||
|
PTRANSACTION transaction;
|
|||
|
|
|||
|
PAGED_CODE( );
|
|||
|
|
|||
|
//
|
|||
|
// Get the address of the owning connection.
|
|||
|
//
|
|||
|
|
|||
|
connection = TreeConnect->Connection;
|
|||
|
pagedConnection = connection->PagedConnection;
|
|||
|
|
|||
|
//
|
|||
|
// Walk the transaction list, looking for transactions owned by the
|
|||
|
// specified tree connect.
|
|||
|
//
|
|||
|
// *** See the description of SrvCloseTransactionsOnSession, which
|
|||
|
// explains why this loop is so complicated.
|
|||
|
//
|
|||
|
|
|||
|
entry = &pagedConnection->TransactionList;
|
|||
|
previousTransaction = NULL;
|
|||
|
|
|||
|
while ( TRUE ) {
|
|||
|
|
|||
|
ACQUIRE_LOCK( &connection->Lock );
|
|||
|
|
|||
|
//
|
|||
|
// Find a transaction that is owned by the specified tree
|
|||
|
// connect.
|
|||
|
//
|
|||
|
|
|||
|
while ( TRUE ) {
|
|||
|
|
|||
|
//
|
|||
|
// Get the address of the next list entry. If we hit the
|
|||
|
// end of the list, exit the inner loop.
|
|||
|
//
|
|||
|
|
|||
|
entry = entry->Flink;
|
|||
|
if ( entry == &pagedConnection->TransactionList ) break;
|
|||
|
|
|||
|
//
|
|||
|
// Get the address of the transaction. If it's owned by
|
|||
|
// the specified tree connect and currently active, exit
|
|||
|
// the inner loop.
|
|||
|
//
|
|||
|
|
|||
|
transaction = CONTAINING_RECORD(
|
|||
|
entry,
|
|||
|
TRANSACTION,
|
|||
|
ConnectionListEntry
|
|||
|
);
|
|||
|
if ( transaction->TreeConnect == TreeConnect &&
|
|||
|
GET_BLOCK_STATE(transaction) == BlockStateActive) {
|
|||
|
|
|||
|
break;
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// If we hit the end of the list without finding a transaction
|
|||
|
// that was owned by the specified tree connect, exit the main
|
|||
|
// loop.
|
|||
|
//
|
|||
|
|
|||
|
if ( entry == &pagedConnection->TransactionList ) break;
|
|||
|
|
|||
|
//
|
|||
|
// Reference the transaction to ensure that it isn't deleted
|
|||
|
// when we close it.
|
|||
|
//
|
|||
|
|
|||
|
SrvReferenceTransaction( transaction );
|
|||
|
|
|||
|
//
|
|||
|
// Unlock the transaction list, so that we can dereference the
|
|||
|
// previous transaction and close the current one.
|
|||
|
//
|
|||
|
|
|||
|
RELEASE_LOCK( &connection->Lock );
|
|||
|
|
|||
|
//
|
|||
|
// If this is not the first matching transaction that we've
|
|||
|
// found, dereference the previous one now.
|
|||
|
//
|
|||
|
|
|||
|
if ( previousTransaction != NULL ) {
|
|||
|
SrvDereferenceTransaction( previousTransaction );
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Close the current transaction and mark that we need to
|
|||
|
// dereference it.
|
|||
|
//
|
|||
|
|
|||
|
SrvCloseTransaction( transaction );
|
|||
|
|
|||
|
previousTransaction = transaction;
|
|||
|
|
|||
|
//
|
|||
|
// Go find another matching transaction.
|
|||
|
//
|
|||
|
|
|||
|
} // while ( TRUE )
|
|||
|
|
|||
|
//
|
|||
|
// We have hit the end of the transaction list. Release the
|
|||
|
// transaction lock. If we have a transaction that needs to be
|
|||
|
// dereferenced, do so. Then return to the caller.
|
|||
|
//
|
|||
|
|
|||
|
RELEASE_LOCK( &connection->Lock );
|
|||
|
|
|||
|
if ( previousTransaction != NULL ) {
|
|||
|
SrvDereferenceTransaction( previousTransaction );
|
|||
|
}
|
|||
|
|
|||
|
return;
|
|||
|
|
|||
|
} // SrvCloseTransactionsOnTree
|
|||
|
|
|||
|
|
|||
|
VOID
|
|||
|
SrvDereferenceTransaction (
|
|||
|
IN PTRANSACTION Transaction
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function decrements the reference count on a transaction. If
|
|||
|
the reference count goes to zero, the transaction block is deleted.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Transaction - Address of transaction
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
PCONNECTION connection;
|
|||
|
LONG result;
|
|||
|
|
|||
|
PAGED_CODE( );
|
|||
|
|
|||
|
//
|
|||
|
// Decrement the reference count on the block.
|
|||
|
//
|
|||
|
|
|||
|
connection = Transaction->Connection;
|
|||
|
|
|||
|
IF_DEBUG(REFCNT) {
|
|||
|
SrvPrint2( "Dereferencing transaction %lx; old refcnt %lx\n",
|
|||
|
Transaction, Transaction->NonpagedHeader->ReferenceCount );
|
|||
|
}
|
|||
|
|
|||
|
ASSERT( GET_BLOCK_TYPE( Transaction ) == BlockTypeTransaction );
|
|||
|
ASSERT( Transaction->NonpagedHeader->ReferenceCount > 0 );
|
|||
|
UPDATE_REFERENCE_HISTORY( Transaction, TRUE );
|
|||
|
|
|||
|
result = InterlockedDecrement(
|
|||
|
&Transaction->NonpagedHeader->ReferenceCount
|
|||
|
);
|
|||
|
|
|||
|
if ( result == 0 ) {
|
|||
|
|
|||
|
//
|
|||
|
// The new reference count is 0, meaning that it's time to
|
|||
|
// delete this block.
|
|||
|
//
|
|||
|
// If the transaction is on the connection's pending transaction
|
|||
|
// list, remove it and dereference the connection, session, and
|
|||
|
// tree connect. If the transaction isn't on the list, then the
|
|||
|
// session and tree connect pointers are not referenced
|
|||
|
// pointers, but just copies from the (single) work context
|
|||
|
// block associated with the transaction.
|
|||
|
//
|
|||
|
|
|||
|
if ( Transaction->Inserted ) {
|
|||
|
|
|||
|
ACQUIRE_LOCK( &connection->Lock );
|
|||
|
|
|||
|
SrvRemoveEntryList(
|
|||
|
&connection->PagedConnection->TransactionList,
|
|||
|
&Transaction->ConnectionListEntry
|
|||
|
);
|
|||
|
|
|||
|
RELEASE_LOCK( &connection->Lock );
|
|||
|
|
|||
|
if ( Transaction->Session != NULL ) {
|
|||
|
SrvDereferenceSession( Transaction->Session );
|
|||
|
DEBUG Transaction->Session = NULL;
|
|||
|
}
|
|||
|
|
|||
|
if ( Transaction->TreeConnect != NULL ) {
|
|||
|
SrvDereferenceTreeConnect( Transaction->TreeConnect );
|
|||
|
DEBUG Transaction->TreeConnect = NULL;
|
|||
|
}
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
DEBUG Transaction->Session = NULL;
|
|||
|
DEBUG Transaction->TreeConnect = NULL;
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Free the transaction block, then release the transaction's
|
|||
|
// reference to the connection. Note that we have to do the
|
|||
|
// dereference after calling SrvFreeConnection because that
|
|||
|
// routine may try to put the transaction on the connection's
|
|||
|
// cached transaction list.
|
|||
|
//
|
|||
|
|
|||
|
SrvFreeTransaction( Transaction );
|
|||
|
|
|||
|
SrvDereferenceConnection( connection );
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
return;
|
|||
|
|
|||
|
} // SrvDereferenceTransaction
|
|||
|
|
|||
|
|
|||
|
VOID
|
|||
|
SrvFreeTransaction (
|
|||
|
IN PTRANSACTION Transaction
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function returns a Transaction block to the server heap.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Transaction - Address of Transaction block
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
ULONG blockSize;
|
|||
|
PCONNECTION connection;
|
|||
|
PNONPAGED_HEADER header;
|
|||
|
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
blockSize = GET_BLOCK_SIZE( Transaction );
|
|||
|
|
|||
|
DEBUG SET_BLOCK_TYPE_STATE_SIZE( Transaction, BlockTypeGarbage, BlockStateDead, -1 );
|
|||
|
DEBUG Transaction->NonpagedHeader->ReferenceCount = -1;
|
|||
|
TERMINATE_REFERENCE_HISTORY( Transaction );
|
|||
|
|
|||
|
connection = Transaction->Connection;
|
|||
|
|
|||
|
//
|
|||
|
// If the transaction was not allocated from the XACTSRV heap and
|
|||
|
// its block size is correct, cache this transaction instead of
|
|||
|
// freeing it back to pool.
|
|||
|
//
|
|||
|
|
|||
|
header = Transaction->NonpagedHeader;
|
|||
|
|
|||
|
if ( !Transaction->RemoteApiRequest ) {
|
|||
|
|
|||
|
if ( blockSize == CACHED_TRANSACTION_BLOCK_SIZE ) {
|
|||
|
//
|
|||
|
// Check the count of cached transactions on the connection.
|
|||
|
// If there aren't already enough transactions cached, link
|
|||
|
// this transaction to the list. Otherwise, free the
|
|||
|
// transaction block.
|
|||
|
//
|
|||
|
|
|||
|
if ( connection->CachedTransactionCount < CACHED_TRANSACTION_LIMIT ) {
|
|||
|
|
|||
|
if ( connection->CachedTransactionCount < CACHED_TRANSACTION_LIMIT ) {
|
|||
|
|
|||
|
ExInterlockedPushEntrySList(
|
|||
|
&connection->CachedTransactionList,
|
|||
|
&header->ListEntry,
|
|||
|
&connection->SpinLock
|
|||
|
);
|
|||
|
InterlockedIncrement( &connection->CachedTransactionCount );
|
|||
|
|
|||
|
return;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
FREE_HEAP( Transaction );
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
SrvXsFreeHeap( Transaction );
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
DEALLOCATE_NONPAGED_POOL( header );
|
|||
|
|
|||
|
IF_DEBUG(HEAP) {
|
|||
|
SrvPrint1( "SrvFreeTransaction: Freed transaction block at %lx\n",
|
|||
|
Transaction );
|
|||
|
}
|
|||
|
|
|||
|
INCREMENT_DEBUG_STAT( SrvDbgStatistics.TransactionInfo.Frees );
|
|||
|
|
|||
|
return;
|
|||
|
|
|||
|
} // SrvFreeTransaction
|
|||
|
|