mirror of
https://github.com/reactos/reactos.git
synced 2025-01-05 13:59:25 +00:00
1023 lines
25 KiB
C
1023 lines
25 KiB
C
/*++
|
|
|
|
Copyright (c) 1989-2000 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
Close.c
|
|
|
|
Abstract:
|
|
|
|
This module implements the File Close routine for Cdfs called by the
|
|
Fsd/Fsp dispatch routines.
|
|
|
|
The close operation interacts with both the async and delayed close queues
|
|
in the CdData structure. Since close may be called recursively we may
|
|
violate the locking order in acquiring the Vcb or Fcb. In this case
|
|
we may move the request to the async close queue. If this is the last
|
|
reference on the Fcb and there is a chance the user may reopen this
|
|
file again soon we would like to defer the close. In this case we
|
|
may move the request to the async close queue.
|
|
|
|
Once we are past the decode file operation there is no need for the
|
|
file object. If we are moving the request to either of the work
|
|
queues then we remember all of the information from the file object and
|
|
complete the request with STATUS_SUCCESS. The Io system can then
|
|
reuse the file object and we can complete the request when convenient.
|
|
|
|
The async close queue consists of requests which we would like to
|
|
complete as soon as possible. They are queued using the original
|
|
IrpContext where some of the fields have been overwritten with
|
|
information from the file object. We will extract this information,
|
|
cleanup the IrpContext and then call the close worker routine.
|
|
|
|
The delayed close queue consists of requests which we would like to
|
|
defer the close for. We keep size of this list within a range
|
|
determined by the size of the system. We let it grow to some maximum
|
|
value and then shrink to some minimum value. We allocate a small
|
|
structure which contains the key information from the file object
|
|
and use this information along with an IrpContext on the stack
|
|
to complete the request.
|
|
|
|
|
|
--*/
|
|
|
|
#include "cdprocs.h"
|
|
|
|
//
|
|
// The Bug check file id for this module
|
|
//
|
|
|
|
#define BugCheckFileId (CDFS_BUG_CHECK_CLOSE)
|
|
|
|
//
|
|
// Local support routines
|
|
//
|
|
|
|
_Requires_lock_held_(_Global_critical_region_)
|
|
BOOLEAN
|
|
CdCommonClosePrivate (
|
|
_In_ PIRP_CONTEXT IrpContext,
|
|
_In_ PVCB Vcb,
|
|
_In_ PFCB Fcb,
|
|
_In_ ULONG UserReference,
|
|
_In_ BOOLEAN FromFsd
|
|
);
|
|
|
|
VOID
|
|
CdQueueClose (
|
|
_In_ PIRP_CONTEXT IrpContext,
|
|
_In_ PFCB Fcb,
|
|
_In_ ULONG UserReference,
|
|
_In_ BOOLEAN DelayedClose
|
|
);
|
|
|
|
PIRP_CONTEXT
|
|
CdRemoveClose (
|
|
_In_opt_ PVCB Vcb
|
|
);
|
|
|
|
// Tell prefast this is a workitem routine
|
|
IO_WORKITEM_ROUTINE CdCloseWorker;
|
|
|
|
VOID
|
|
NTAPI /* ReactOS Change: GCC Does not support STDCALL by default */
|
|
CdCloseWorker (
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_In_opt_ PVOID Context
|
|
);
|
|
|
|
#ifdef ALLOC_PRAGMA
|
|
#pragma alloc_text(PAGE, CdFspClose)
|
|
#pragma alloc_text(PAGE, CdCommonClose)
|
|
#pragma alloc_text(PAGE, CdCommonClosePrivate)
|
|
#pragma alloc_text(PAGE, CdQueueClose)
|
|
#pragma alloc_text(PAGE, CdRemoveClose)
|
|
#pragma alloc_text(PAGE, CdCloseWorker)
|
|
#endif
|
|
|
|
|
|
VOID
|
|
CdFspClose (
|
|
_In_opt_ PVCB Vcb
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is called to process the close queues in the CdData. If the
|
|
Vcb is passed then we want to remove all of the closes for this Vcb.
|
|
Otherwise we will do as many of the delayed closes as we need to do.
|
|
|
|
Arguments:
|
|
|
|
Vcb - If specified then we are looking for all of the closes for the
|
|
given Vcb.
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
|
|
{
|
|
PIRP_CONTEXT IrpContext;
|
|
IRP_CONTEXT StackIrpContext;
|
|
|
|
THREAD_CONTEXT ThreadContext = {0};
|
|
|
|
PFCB Fcb;
|
|
ULONG UserReference;
|
|
|
|
ULONG VcbHoldCount = 0;
|
|
PVCB CurrentVcb = NULL;
|
|
|
|
BOOLEAN PotentialVcbTeardown = FALSE;
|
|
|
|
PAGED_CODE();
|
|
|
|
FsRtlEnterFileSystem();
|
|
|
|
//
|
|
// Continue processing until there are no more closes to process.
|
|
//
|
|
|
|
while ((IrpContext = CdRemoveClose( Vcb )) != NULL) {
|
|
|
|
//
|
|
// If we don't have an IrpContext then use the one on the stack.
|
|
// Initialize it for this request.
|
|
//
|
|
|
|
if (SafeNodeType( IrpContext ) != CDFS_NTC_IRP_CONTEXT ) {
|
|
|
|
//
|
|
// Update the local values from the IrpContextLite.
|
|
//
|
|
|
|
Fcb = ((PIRP_CONTEXT_LITE) IrpContext)->Fcb;
|
|
UserReference = ((PIRP_CONTEXT_LITE) IrpContext)->UserReference;
|
|
|
|
//
|
|
// Update the stack irp context with the values from the
|
|
// IrpContextLite.
|
|
//
|
|
|
|
CdInitializeStackIrpContext( &StackIrpContext,
|
|
(PIRP_CONTEXT_LITE) IrpContext );
|
|
|
|
//
|
|
// Free the IrpContextLite.
|
|
//
|
|
|
|
CdFreeIrpContextLite( *(PVOID*)&IrpContext ); /* ReactOS Change: GCC "error: invalid lvalue in unary '&'" */
|
|
|
|
//
|
|
// Remember we have the IrpContext from the stack.
|
|
//
|
|
|
|
IrpContext = &StackIrpContext;
|
|
|
|
//
|
|
// Otherwise cleanup the existing IrpContext.
|
|
//
|
|
|
|
} else {
|
|
|
|
//
|
|
// Remember the Fcb and user reference count.
|
|
//
|
|
|
|
Fcb = (PFCB) IrpContext->Irp;
|
|
IrpContext->Irp = NULL;
|
|
|
|
UserReference = (ULONG) IrpContext->ExceptionStatus;
|
|
IrpContext->ExceptionStatus = STATUS_SUCCESS;
|
|
}
|
|
|
|
_Analysis_assume_(Fcb != NULL && Fcb->Vcb != NULL);
|
|
|
|
//
|
|
// We have an IrpContext. Now we need to set the top level thread
|
|
// context.
|
|
//
|
|
|
|
SetFlag( IrpContext->Flags, IRP_CONTEXT_FSP_FLAGS );
|
|
|
|
//
|
|
// If we were given a Vcb then there is a request on top of this.
|
|
//
|
|
|
|
if (ARGUMENT_PRESENT( Vcb )) {
|
|
|
|
ClearFlag( IrpContext->Flags,
|
|
IRP_CONTEXT_FLAG_TOP_LEVEL | IRP_CONTEXT_FLAG_TOP_LEVEL_CDFS );
|
|
}
|
|
|
|
CdSetThreadContext( IrpContext, &ThreadContext );
|
|
|
|
//
|
|
// If we have hit the maximum number of requests to process without
|
|
// releasing the Vcb then release the Vcb now. If we are holding
|
|
// a different Vcb to this one then release the previous Vcb.
|
|
//
|
|
// In either case acquire the current Vcb.
|
|
//
|
|
// We use the MinDelayedCloseCount from the CdData since it is
|
|
// a convenient value based on the system size. Only thing we are trying
|
|
// to do here is prevent this routine starving other threads which
|
|
// may need this Vcb exclusively.
|
|
//
|
|
// Note that the check for potential teardown below is unsafe. We'll
|
|
// repeat later within the cddata lock.
|
|
//
|
|
|
|
PotentialVcbTeardown = !ARGUMENT_PRESENT( Vcb ) &&
|
|
(Fcb->Vcb->VcbCondition != VcbMounted) &&
|
|
(Fcb->Vcb->VcbCondition != VcbMountInProgress) &&
|
|
(Fcb->Vcb->VcbCleanup == 0);
|
|
|
|
if (PotentialVcbTeardown ||
|
|
(VcbHoldCount > CdData.MinDelayedCloseCount) ||
|
|
(Fcb->Vcb != CurrentVcb)) {
|
|
|
|
if (CurrentVcb != NULL) {
|
|
|
|
CdReleaseVcb( IrpContext, CurrentVcb );
|
|
}
|
|
|
|
if (PotentialVcbTeardown) {
|
|
|
|
CdAcquireCdData( IrpContext );
|
|
|
|
//
|
|
// Repeat the checks with global lock held. The volume could have
|
|
// been remounted while we didn't hold the lock.
|
|
//
|
|
|
|
PotentialVcbTeardown = !ARGUMENT_PRESENT( Vcb ) &&
|
|
(Fcb->Vcb->VcbCondition != VcbMounted) &&
|
|
(Fcb->Vcb->VcbCondition != VcbMountInProgress) &&
|
|
(Fcb->Vcb->VcbCleanup == 0);
|
|
|
|
if (!PotentialVcbTeardown) {
|
|
|
|
CdReleaseCdData( IrpContext);
|
|
}
|
|
}
|
|
|
|
CurrentVcb = Fcb->Vcb;
|
|
|
|
_Analysis_assume_( CurrentVcb != NULL );
|
|
|
|
CdAcquireVcbShared( IrpContext, CurrentVcb, FALSE );
|
|
|
|
VcbHoldCount = 0;
|
|
|
|
} else {
|
|
|
|
VcbHoldCount += 1;
|
|
}
|
|
|
|
//
|
|
// Call our worker routine to perform the close operation.
|
|
//
|
|
|
|
CdCommonClosePrivate( IrpContext, CurrentVcb, Fcb, UserReference, FALSE );
|
|
|
|
//
|
|
// If the reference count on this Vcb is below our residual reference
|
|
// then check if we should dismount the volume.
|
|
//
|
|
|
|
if (PotentialVcbTeardown) {
|
|
|
|
CdReleaseVcb( IrpContext, CurrentVcb );
|
|
CdCheckForDismount( IrpContext, CurrentVcb, FALSE );
|
|
|
|
CurrentVcb = NULL;
|
|
|
|
CdReleaseCdData( IrpContext );
|
|
PotentialVcbTeardown = FALSE;
|
|
}
|
|
|
|
//
|
|
// Complete the current request to cleanup the IrpContext.
|
|
//
|
|
|
|
CdCompleteRequest( IrpContext, NULL, STATUS_SUCCESS );
|
|
}
|
|
|
|
//
|
|
// Release any Vcb we may still hold.
|
|
//
|
|
|
|
if (CurrentVcb != NULL) {
|
|
|
|
CdReleaseVcb( IrpContext, CurrentVcb );
|
|
|
|
}
|
|
|
|
#ifdef _MSC_VER
|
|
#pragma prefast(suppress:26165, "Esp:1153")
|
|
#endif
|
|
FsRtlExitFileSystem();
|
|
}
|
|
|
|
_Requires_lock_held_(_Global_critical_region_)
|
|
NTSTATUS
|
|
CdCommonClose (
|
|
_Inout_ PIRP_CONTEXT IrpContext,
|
|
_Inout_ PIRP Irp
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is the Fsd entry for the close operation. We decode the file
|
|
object to find the CDFS structures and type of open. We call our internal
|
|
worker routine to perform the actual work. If the work wasn't completed
|
|
then we post to one of our worker queues. The Ccb isn't needed after this
|
|
point so we delete the Ccb and return STATUS_SUCCESS to our caller in all
|
|
cases.
|
|
|
|
Arguments:
|
|
|
|
Irp - Supplies the Irp to process
|
|
|
|
Return Value:
|
|
|
|
STATUS_SUCCESS
|
|
|
|
--*/
|
|
|
|
{
|
|
TYPE_OF_OPEN TypeOfOpen;
|
|
|
|
PVCB Vcb;
|
|
PFCB Fcb;
|
|
PCCB Ccb;
|
|
ULONG UserReference = 0;
|
|
|
|
BOOLEAN PotentialVcbTeardown = FALSE;
|
|
|
|
PAGED_CODE();
|
|
|
|
ASSERT_IRP_CONTEXT( IrpContext );
|
|
ASSERT_IRP( Irp );
|
|
|
|
//
|
|
// If we were called with our file system device object instead of a
|
|
// volume device object, just complete this request with STATUS_SUCCESS.
|
|
//
|
|
|
|
if (IrpContext->Vcb == NULL) {
|
|
|
|
CdCompleteRequest( IrpContext, Irp, STATUS_SUCCESS );
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
//
|
|
// Decode the file object to get the type of open and Fcb/Ccb.
|
|
//
|
|
|
|
TypeOfOpen = CdDecodeFileObject( IrpContext,
|
|
IoGetCurrentIrpStackLocation( Irp )->FileObject,
|
|
&Fcb,
|
|
&Ccb );
|
|
|
|
//
|
|
// No work to do for unopened file objects.
|
|
//
|
|
|
|
if (TypeOfOpen == UnopenedFileObject) {
|
|
|
|
CdCompleteRequest( IrpContext, Irp, STATUS_SUCCESS );
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
Vcb = Fcb->Vcb;
|
|
|
|
//
|
|
// Clean up any CCB associated with this open.
|
|
//
|
|
|
|
if (Ccb != NULL) {
|
|
|
|
UserReference = 1;
|
|
|
|
//
|
|
// We can always deallocate the Ccb if present.
|
|
//
|
|
|
|
CdDeleteCcb( IrpContext, Ccb );
|
|
}
|
|
|
|
//
|
|
// If this is the last reference to a user file or directory on a
|
|
// currently mounted volume, then post it to the delayed close queue. Note
|
|
// that the VcbCondition check is unsafe, but it doesn't really matter -
|
|
// we just might delay the volume teardown a little by posting this close.
|
|
//
|
|
|
|
if ((Vcb->VcbCondition == VcbMounted) &&
|
|
(Fcb->FcbReference == 1) &&
|
|
((TypeOfOpen == UserFileOpen) ||
|
|
(TypeOfOpen == UserDirectoryOpen))) {
|
|
|
|
CdQueueClose( IrpContext, Fcb, UserReference, TRUE );
|
|
IrpContext = NULL;
|
|
|
|
//
|
|
// Otherwise try to process this close. Post to the async close queue
|
|
// if we can't acquire all of the resources.
|
|
//
|
|
|
|
}
|
|
else {
|
|
|
|
//
|
|
// If we may be dismounting this volume then acquire the CdData
|
|
// resource.
|
|
//
|
|
// Since we now must make volumes go away as soon as reasonable after
|
|
// the last user handles closes, key off of the cleanup count. It is
|
|
// OK to do this more than neccesary. Since this Fcb could be holding
|
|
// a number of other Fcbs (and thus their references), a simple check
|
|
// on reference count is not appropriate.
|
|
//
|
|
// Do an unsafe check first to avoid taking the (global) cddata lock in the
|
|
// common case.
|
|
//
|
|
|
|
if ((Vcb->VcbCleanup == 0) &&
|
|
(Vcb->VcbCondition != VcbMounted)) {
|
|
|
|
//
|
|
// Possible dismount. Acquire CdData to synchronise with the remount path
|
|
// before looking at the vcb condition again.
|
|
//
|
|
|
|
CdAcquireCdData( IrpContext );
|
|
|
|
if ((Vcb->VcbCleanup == 0) &&
|
|
(Vcb->VcbCondition != VcbMounted) &&
|
|
(Vcb->VcbCondition != VcbMountInProgress) &&
|
|
FlagOn( IrpContext->Flags, IRP_CONTEXT_FLAG_TOP_LEVEL_CDFS )) {
|
|
|
|
PotentialVcbTeardown = TRUE;
|
|
}
|
|
else {
|
|
|
|
//
|
|
// We can't dismount this volume now, there are other references or
|
|
// it's just been remounted.
|
|
//
|
|
}
|
|
|
|
//
|
|
// Drop the global lock if we don't need it anymore.
|
|
//
|
|
|
|
if (!PotentialVcbTeardown) {
|
|
|
|
CdReleaseCdData( IrpContext );
|
|
}
|
|
}
|
|
|
|
//
|
|
// Call the worker routine to perform the actual work. This routine
|
|
// should never raise except for a fatal error.
|
|
//
|
|
|
|
if (!CdCommonClosePrivate( IrpContext, Vcb, Fcb, UserReference, TRUE )) {
|
|
|
|
//
|
|
// If we didn't complete the request then post the request as needed.
|
|
//
|
|
|
|
CdQueueClose( IrpContext, Fcb, UserReference, FALSE );
|
|
IrpContext = NULL;
|
|
|
|
//
|
|
// Check whether we should be dismounting the volume and then complete
|
|
// the request.
|
|
//
|
|
|
|
}
|
|
else if (PotentialVcbTeardown) {
|
|
|
|
CdCheckForDismount( IrpContext, Vcb, FALSE );
|
|
}
|
|
}
|
|
|
|
//
|
|
// Always complete this request with STATUS_SUCCESS.
|
|
//
|
|
|
|
CdCompleteRequest( IrpContext, Irp, STATUS_SUCCESS );
|
|
|
|
if (PotentialVcbTeardown) {
|
|
|
|
CdReleaseCdData( IrpContext );
|
|
}
|
|
|
|
//
|
|
// Always return STATUS_SUCCESS for closes.
|
|
//
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
//
|
|
// Local support routine
|
|
//
|
|
|
|
_Requires_lock_held_(_Global_critical_region_)
|
|
BOOLEAN
|
|
CdCommonClosePrivate (
|
|
_In_ PIRP_CONTEXT IrpContext,
|
|
_In_ PVCB Vcb,
|
|
_In_ PFCB Fcb,
|
|
_In_ ULONG UserReference,
|
|
_In_ BOOLEAN FromFsd
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This is the worker routine for the close operation. We can be called in
|
|
an Fsd thread or from a worker Fsp thread. If called from the Fsd thread
|
|
then we acquire the resources without waiting. Otherwise we know it is
|
|
safe to wait.
|
|
|
|
We check to see whether we should post this request to the delayed close
|
|
queue. If we are to process the close here then we acquire the Vcb and
|
|
Fcb. We will adjust the counts and call our teardown routine to see
|
|
if any of the structures should go away.
|
|
|
|
Arguments:
|
|
|
|
Vcb - Vcb for this volume.
|
|
|
|
Fcb - Fcb for this request.
|
|
|
|
UserReference - Number of user references for this file object. This is
|
|
zero for an internal stream.
|
|
|
|
FromFsd - This request was called from an Fsd thread. Indicates whether
|
|
we should wait to acquire resources.
|
|
|
|
DelayedClose - Address to store whether we should try to put this on
|
|
the delayed close queue. Ignored if this routine can process this
|
|
close.
|
|
|
|
Return Value:
|
|
|
|
BOOLEAN - TRUE if this thread processed the close, FALSE otherwise.
|
|
|
|
--*/
|
|
|
|
{
|
|
BOOLEAN RemovedFcb;
|
|
|
|
PAGED_CODE();
|
|
|
|
ASSERT_IRP_CONTEXT( IrpContext );
|
|
ASSERT_FCB( Fcb );
|
|
|
|
//
|
|
// Try to acquire the Vcb and Fcb. If we can't acquire them then return
|
|
// and let our caller know he should post the request to the async
|
|
// queue.
|
|
//
|
|
|
|
if (CdAcquireVcbShared( IrpContext, Vcb, FromFsd )) {
|
|
|
|
if (!CdAcquireFcbExclusive( IrpContext, Fcb, FromFsd )) {
|
|
|
|
//
|
|
// We couldn't get the Fcb. Release the Vcb and let our caller
|
|
// know to post this request.
|
|
//
|
|
|
|
CdReleaseVcb( IrpContext, Vcb );
|
|
return FALSE;
|
|
}
|
|
|
|
//
|
|
// We didn't get the Vcb. Let our caller know to post this request.
|
|
//
|
|
|
|
} else {
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
//
|
|
// Lock the Vcb and decrement the reference counts.
|
|
//
|
|
|
|
CdLockVcb( IrpContext, Vcb );
|
|
CdDecrementReferenceCounts( IrpContext, Fcb, 1, UserReference );
|
|
CdUnlockVcb( IrpContext, Vcb );
|
|
|
|
//
|
|
// Call our teardown routine to see if this object can go away.
|
|
// If we don't remove the Fcb then release it.
|
|
//
|
|
|
|
CdTeardownStructures( IrpContext, Fcb, &RemovedFcb );
|
|
|
|
if (!RemovedFcb) {
|
|
|
|
CdReleaseFcb( IrpContext, Fcb );
|
|
}
|
|
else {
|
|
_Analysis_assume_lock_not_held_(Fcb->FcbNonpaged->FcbResource);
|
|
}
|
|
|
|
//
|
|
// Release the Vcb and return to our caller. Let him know we completed
|
|
// this request.
|
|
//
|
|
|
|
CdReleaseVcb( IrpContext, Vcb );
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
VOID
|
|
NTAPI /* ReactOS Change: GCC Does not support STDCALL by default */
|
|
CdCloseWorker (
|
|
_In_ PDEVICE_OBJECT DeviceObject,
|
|
_In_opt_ PVOID Context
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Worker routine to call CsFspClose.
|
|
|
|
Arguments:
|
|
|
|
DeviceObject - Filesystem registration device object
|
|
|
|
Context - Callers context
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
|
|
{
|
|
PAGED_CODE();
|
|
|
|
UNREFERENCED_PARAMETER( DeviceObject );
|
|
UNREFERENCED_PARAMETER( Context );
|
|
|
|
CdFspClose (NULL);
|
|
}
|
|
|
|
|
|
VOID
|
|
CdQueueClose (
|
|
_In_ PIRP_CONTEXT IrpContext,
|
|
_In_ PFCB Fcb,
|
|
_In_ ULONG UserReference,
|
|
_In_ BOOLEAN DelayedClose
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is called to queue a request to either the async or delayed
|
|
close queue. For the delayed queue we need to allocate a smaller
|
|
structure to contain the information about the file object. We do
|
|
that so we don't put the larger IrpContext structures into this long
|
|
lived queue. If we can allocate this structure then we put this
|
|
on the async queue instead.
|
|
|
|
Arguments:
|
|
|
|
Fcb - Fcb for this file object.
|
|
|
|
UserReference - Number of user references for this file object. This is
|
|
zero for an internal stream.
|
|
|
|
DelayedClose - Indicates whether this should go on the async or delayed
|
|
close queue.
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
|
|
{
|
|
PIRP_CONTEXT_LITE IrpContextLite = NULL;
|
|
BOOLEAN StartWorker = FALSE;
|
|
|
|
PAGED_CODE();
|
|
|
|
ASSERT_IRP_CONTEXT( IrpContext );
|
|
ASSERT_FCB( Fcb );
|
|
|
|
//
|
|
// Start with the delayed queue request. We can move this to the async
|
|
// queue if there is an allocation failure.
|
|
//
|
|
|
|
if (DelayedClose) {
|
|
|
|
//
|
|
// Try to allocate non-paged pool for the IRP_CONTEXT_LITE.
|
|
//
|
|
|
|
IrpContextLite = CdCreateIrpContextLite( IrpContext );
|
|
}
|
|
|
|
//
|
|
// We want to clear the top level context in this thread if
|
|
// necessary. Call our cleanup routine to do the work.
|
|
//
|
|
|
|
SetFlag( IrpContext->Flags, IRP_CONTEXT_FLAG_MORE_PROCESSING );
|
|
CdCleanupIrpContext( IrpContext, TRUE );
|
|
|
|
//
|
|
// Synchronize with the CdData lock.
|
|
//
|
|
|
|
CdLockCdData();
|
|
|
|
//
|
|
// If we have an IrpContext then put the request on the delayed close queue.
|
|
//
|
|
|
|
if (IrpContextLite != NULL) {
|
|
|
|
//
|
|
// Initialize the IrpContextLite.
|
|
//
|
|
|
|
IrpContextLite->NodeTypeCode = CDFS_NTC_IRP_CONTEXT_LITE;
|
|
IrpContextLite->NodeByteSize = sizeof( IRP_CONTEXT_LITE );
|
|
IrpContextLite->Fcb = Fcb;
|
|
IrpContextLite->UserReference = UserReference;
|
|
IrpContextLite->RealDevice = IrpContext->RealDevice;
|
|
|
|
//
|
|
// Add this to the delayed close list and increment
|
|
// the count.
|
|
//
|
|
|
|
InsertTailList( &CdData.DelayedCloseQueue,
|
|
&IrpContextLite->DelayedCloseLinks );
|
|
|
|
CdData.DelayedCloseCount += 1;
|
|
|
|
//
|
|
// If we are above our threshold then start the delayed
|
|
// close operation.
|
|
//
|
|
|
|
if (CdData.DelayedCloseCount > CdData.MaxDelayedCloseCount) {
|
|
|
|
CdData.ReduceDelayedClose = TRUE;
|
|
|
|
if (!CdData.FspCloseActive) {
|
|
|
|
CdData.FspCloseActive = TRUE;
|
|
StartWorker = TRUE;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Unlock the CdData.
|
|
//
|
|
|
|
CdUnlockCdData();
|
|
|
|
//
|
|
// Cleanup the IrpContext.
|
|
//
|
|
|
|
CdCompleteRequest( IrpContext, NULL, STATUS_SUCCESS );
|
|
|
|
//
|
|
// Otherwise drop into the async case below.
|
|
//
|
|
|
|
} else {
|
|
|
|
//
|
|
// Store the information about the file object into the IrpContext.
|
|
//
|
|
|
|
IrpContext->Irp = (PIRP) Fcb;
|
|
IrpContext->ExceptionStatus = (NTSTATUS) UserReference;
|
|
|
|
//
|
|
// Add this to the async close list and increment the count.
|
|
//
|
|
|
|
InsertTailList( &CdData.AsyncCloseQueue,
|
|
&IrpContext->WorkQueueItem.List );
|
|
|
|
CdData.AsyncCloseCount += 1;
|
|
|
|
//
|
|
// Remember to start the Fsp close thread if not currently started.
|
|
//
|
|
|
|
if (!CdData.FspCloseActive) {
|
|
|
|
CdData.FspCloseActive = TRUE;
|
|
|
|
StartWorker = TRUE;
|
|
}
|
|
|
|
//
|
|
// Unlock the CdData.
|
|
//
|
|
|
|
CdUnlockCdData();
|
|
}
|
|
|
|
//
|
|
// Start the FspClose thread if we need to.
|
|
//
|
|
|
|
if (StartWorker) {
|
|
|
|
IoQueueWorkItem( CdData.CloseItem, CdCloseWorker, CriticalWorkQueue, NULL );
|
|
}
|
|
|
|
//
|
|
// Return to our caller.
|
|
//
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
//
|
|
// Local support routine
|
|
//
|
|
|
|
PIRP_CONTEXT
|
|
CdRemoveClose (
|
|
_In_opt_ PVCB Vcb
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Arguments:
|
|
|
|
This routine is called to scan the async and delayed close queues looking
|
|
for a suitable entry. If the Vcb is specified then we scan both queues
|
|
looking for an entry with the same Vcb. Otherwise we will look in the
|
|
async queue first for any close item. If none found there then we look
|
|
in the delayed close queue provided that we have triggered the delayed
|
|
close operation.
|
|
|
|
Return Value:
|
|
|
|
PIRP_CONTEXT - NULL if no work item found. Otherwise it is the pointer to
|
|
either the IrpContext or IrpContextLite for this request.
|
|
|
|
--*/
|
|
|
|
{
|
|
PIRP_CONTEXT IrpContext = NULL;
|
|
PIRP_CONTEXT NextIrpContext;
|
|
PIRP_CONTEXT_LITE NextIrpContextLite;
|
|
|
|
PLIST_ENTRY Entry;
|
|
|
|
PAGED_CODE();
|
|
|
|
ASSERT_OPTIONAL_VCB( Vcb );
|
|
|
|
//
|
|
// Lock the CdData to perform the scan.
|
|
//
|
|
|
|
CdLockCdData();
|
|
|
|
//
|
|
// First check the list of async closes.
|
|
//
|
|
|
|
Entry = CdData.AsyncCloseQueue.Flink;
|
|
|
|
while (Entry != &CdData.AsyncCloseQueue) {
|
|
|
|
//
|
|
// Extract the IrpContext.
|
|
//
|
|
|
|
NextIrpContext = CONTAINING_RECORD( Entry,
|
|
IRP_CONTEXT,
|
|
WorkQueueItem.List );
|
|
|
|
//
|
|
// If no Vcb was specified or this Vcb is for our volume
|
|
// then perform the close.
|
|
//
|
|
|
|
if (!ARGUMENT_PRESENT( Vcb ) || (NextIrpContext->Vcb == Vcb)) {
|
|
|
|
RemoveEntryList( Entry );
|
|
CdData.AsyncCloseCount -= 1;
|
|
|
|
IrpContext = NextIrpContext;
|
|
break;
|
|
}
|
|
|
|
//
|
|
// Move to the next entry.
|
|
//
|
|
|
|
Entry = Entry->Flink;
|
|
}
|
|
|
|
//
|
|
// If we didn't find anything look through the delayed close
|
|
// queue.
|
|
//
|
|
// We will only check the delayed close queue if we were given
|
|
// a Vcb or the delayed close operation is active.
|
|
//
|
|
|
|
if ((IrpContext == NULL) &&
|
|
(ARGUMENT_PRESENT( Vcb ) ||
|
|
(CdData.ReduceDelayedClose &&
|
|
(CdData.DelayedCloseCount > CdData.MinDelayedCloseCount)))) {
|
|
|
|
Entry = CdData.DelayedCloseQueue.Flink;
|
|
|
|
while (Entry != &CdData.DelayedCloseQueue) {
|
|
|
|
//
|
|
// Extract the IrpContext.
|
|
//
|
|
|
|
NextIrpContextLite = CONTAINING_RECORD( Entry,
|
|
IRP_CONTEXT_LITE,
|
|
DelayedCloseLinks );
|
|
|
|
//
|
|
// If no Vcb was specified or this Vcb is for our volume
|
|
// then perform the close.
|
|
//
|
|
|
|
if (!ARGUMENT_PRESENT( Vcb ) || (NextIrpContextLite->Fcb->Vcb == Vcb)) {
|
|
|
|
RemoveEntryList( Entry );
|
|
CdData.DelayedCloseCount -= 1;
|
|
|
|
IrpContext = (PIRP_CONTEXT) NextIrpContextLite;
|
|
break;
|
|
}
|
|
|
|
//
|
|
// Move to the next entry.
|
|
//
|
|
|
|
Entry = Entry->Flink;
|
|
}
|
|
}
|
|
|
|
//
|
|
// If the Vcb wasn't specified and we couldn't find an entry
|
|
// then turn off the Fsp thread.
|
|
//
|
|
|
|
if (!ARGUMENT_PRESENT( Vcb ) && (IrpContext == NULL)) {
|
|
|
|
CdData.FspCloseActive = FALSE;
|
|
CdData.ReduceDelayedClose = FALSE;
|
|
}
|
|
|
|
//
|
|
// Unlock the CdData.
|
|
//
|
|
|
|
CdUnlockCdData();
|
|
|
|
return IrpContext;
|
|
}
|
|
|
|
|
|
|