mirror of
https://github.com/reactos/reactos.git
synced 2025-01-07 14:51:00 +00:00
0daa5547d9
This implies that a sample for W10. It has been backported to NT5.2; not sure how it would work on a W2K3 (feel free to test!)
1189 lines
38 KiB
C
1189 lines
38 KiB
C
/*++
|
||
|
||
Copyright (c) 1989-2000 Microsoft Corporation
|
||
|
||
Module Name:
|
||
|
||
Cleanup.c
|
||
|
||
Abstract:
|
||
|
||
This module implements the File Cleanup routine for Fat called by the
|
||
dispatch driver.
|
||
|
||
|
||
--*/
|
||
|
||
#include "fatprocs.h"
|
||
|
||
//
|
||
// The Bug check file id for this module
|
||
//
|
||
|
||
#define BugCheckFileId (FAT_BUG_CHECK_CLEANUP)
|
||
|
||
//
|
||
// The local debug trace level
|
||
//
|
||
|
||
#define Dbg (DEBUG_TRACE_CLEANUP)
|
||
|
||
//
|
||
// The following little routine exists solely because it need a spin lock.
|
||
//
|
||
|
||
VOID
|
||
FatAutoUnlock (
|
||
IN PIRP_CONTEXT IrpContext,
|
||
IN PVCB Vcb
|
||
);
|
||
|
||
#ifdef ALLOC_PRAGMA
|
||
#pragma alloc_text(PAGE, FatCommonCleanup)
|
||
#pragma alloc_text(PAGE, FatFsdCleanup)
|
||
#endif
|
||
|
||
|
||
_Function_class_(IRP_MJ_CLEANUP)
|
||
_Function_class_(DRIVER_DISPATCH)
|
||
NTSTATUS
|
||
NTAPI
|
||
FatFsdCleanup (
|
||
_In_ PVOLUME_DEVICE_OBJECT VolumeDeviceObject,
|
||
_Inout_ PIRP Irp
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This routine implements the FSD part of closing down a handle to a
|
||
file object.
|
||
|
||
Arguments:
|
||
|
||
VolumeDeviceObject - Supplies the volume device object where the
|
||
file being Cleanup exists
|
||
|
||
Irp - Supplies the Irp being processed
|
||
|
||
Return Value:
|
||
|
||
NTSTATUS - The FSD status for the IRP
|
||
|
||
--*/
|
||
|
||
{
|
||
NTSTATUS Status;
|
||
PIRP_CONTEXT IrpContext = NULL;
|
||
|
||
BOOLEAN TopLevel;
|
||
|
||
PAGED_CODE();
|
||
|
||
//
|
||
// If we were called with our file system device object instead of a
|
||
// volume device object, just complete this request with STATUS_SUCCESS
|
||
//
|
||
|
||
if ( FatDeviceIsFatFsdo( VolumeDeviceObject)) {
|
||
|
||
Irp->IoStatus.Status = STATUS_SUCCESS;
|
||
Irp->IoStatus.Information = FILE_OPENED;
|
||
|
||
IoCompleteRequest( Irp, IO_DISK_INCREMENT );
|
||
|
||
return STATUS_SUCCESS;
|
||
}
|
||
|
||
DebugTrace(+1, Dbg, "FatFsdCleanup\n", 0);
|
||
|
||
//
|
||
// Call the common Cleanup routine, with blocking allowed.
|
||
//
|
||
|
||
FsRtlEnterFileSystem();
|
||
|
||
TopLevel = FatIsIrpTopLevel( Irp );
|
||
|
||
_SEH2_TRY {
|
||
|
||
IrpContext = FatCreateIrpContext( Irp, TRUE );
|
||
|
||
Status = FatCommonCleanup( IrpContext, Irp );
|
||
|
||
} _SEH2_EXCEPT(FatExceptionFilter( IrpContext, _SEH2_GetExceptionInformation() )) {
|
||
|
||
//
|
||
// We had some trouble trying to perform the requested
|
||
// operation, so we'll abort the I/O request with
|
||
// the error status that we get back from the
|
||
// execption code
|
||
//
|
||
|
||
Status = FatProcessException( IrpContext, Irp, _SEH2_GetExceptionCode() );
|
||
} _SEH2_END;
|
||
|
||
if (TopLevel) { IoSetTopLevelIrp( NULL ); }
|
||
|
||
FsRtlExitFileSystem();
|
||
|
||
//
|
||
// And return to our caller
|
||
//
|
||
|
||
DebugTrace(-1, Dbg, "FatFsdCleanup -> %08lx\n", Status);
|
||
|
||
UNREFERENCED_PARAMETER( VolumeDeviceObject );
|
||
|
||
return Status;
|
||
}
|
||
|
||
|
||
_Requires_lock_held_(_Global_critical_region_)
|
||
NTSTATUS
|
||
FatCommonCleanup (
|
||
IN PIRP_CONTEXT IrpContext,
|
||
IN PIRP Irp
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This is the common routine for cleanup of a file/directory called by both
|
||
the fsd and fsp threads.
|
||
|
||
Cleanup is invoked whenever the last handle to a file object is closed.
|
||
This is different than the Close operation which is invoked when the last
|
||
reference to a file object is deleted.
|
||
|
||
The function of cleanup is to essentially "cleanup" the file/directory
|
||
after a user is done with it. The Fcb/Dcb remains around (because MM
|
||
still has the file object referenced) but is now available for another
|
||
user to open (i.e., as far as the user is concerned the is now closed).
|
||
|
||
See close for a more complete description of what close does.
|
||
|
||
Arguments:
|
||
|
||
Irp - Supplies the Irp to process
|
||
|
||
Return Value:
|
||
|
||
NTSTATUS - The return status for the operation
|
||
|
||
--*/
|
||
|
||
{
|
||
NTSTATUS Status = STATUS_SUCCESS;
|
||
|
||
PIO_STACK_LOCATION IrpSp;
|
||
|
||
PFILE_OBJECT FileObject;
|
||
|
||
TYPE_OF_OPEN TypeOfOpen;
|
||
PVCB Vcb;
|
||
PFCB Fcb;
|
||
PCCB Ccb;
|
||
|
||
BOOLEAN SendUnlockNotification = FALSE;
|
||
|
||
PSHARE_ACCESS ShareAccess;
|
||
|
||
PLARGE_INTEGER TruncateSize = NULL;
|
||
LARGE_INTEGER LocalTruncateSize;
|
||
|
||
BOOLEAN AcquiredVcb = FALSE;
|
||
BOOLEAN AcquiredFcb = FALSE;
|
||
|
||
#if (NTDDI_VERSION >= NTDDI_WIN8)
|
||
BOOLEAN ProcessingDeleteOnClose = FALSE;
|
||
#endif
|
||
|
||
PAGED_CODE();
|
||
|
||
IrpSp = IoGetCurrentIrpStackLocation( Irp );
|
||
|
||
DebugTrace(+1, Dbg, "FatCommonCleanup\n", 0);
|
||
DebugTrace( 0, Dbg, "Irp = %p\n", Irp);
|
||
DebugTrace( 0, Dbg, "->FileObject = %p\n", IrpSp->FileObject);
|
||
|
||
//
|
||
// Extract and decode the file object
|
||
//
|
||
|
||
FileObject = IrpSp->FileObject;
|
||
TypeOfOpen = FatDecodeFileObject( FileObject, &Vcb, &Fcb, &Ccb );
|
||
|
||
//
|
||
// Special case the unopened file object. This will occur only when
|
||
// we are initializing Vcb and IoCreateStreamFileObject is being
|
||
// called.
|
||
//
|
||
|
||
if (TypeOfOpen == UnopenedFileObject) {
|
||
|
||
DebugTrace(0, Dbg, "Unopened File Object\n", 0);
|
||
|
||
FatCompleteRequest( IrpContext, Irp, STATUS_SUCCESS );
|
||
|
||
DebugTrace(-1, Dbg, "FatCommonCleanup -> STATUS_SUCCESS\n", 0);
|
||
return STATUS_SUCCESS;
|
||
}
|
||
|
||
//
|
||
// If this is not our first time through (for whatever reason)
|
||
// only see if we have to flush the file.
|
||
//
|
||
|
||
if (FlagOn( FileObject->Flags, FO_CLEANUP_COMPLETE )) {
|
||
|
||
if (FlagOn(Vcb->VcbState, VCB_STATE_FLAG_DEFERRED_FLUSH) &&
|
||
FlagOn(FileObject->Flags, FO_FILE_MODIFIED) &&
|
||
!FlagOn(Vcb->VcbState, VCB_STATE_FLAG_WRITE_PROTECTED) &&
|
||
(TypeOfOpen == UserFileOpen)) {
|
||
|
||
//
|
||
// Flush the file.
|
||
//
|
||
|
||
Status = FatFlushFile( IrpContext, Fcb, Flush );
|
||
|
||
if (!NT_SUCCESS(Status)) {
|
||
|
||
FatNormalizeAndRaiseStatus( IrpContext, Status );
|
||
}
|
||
}
|
||
|
||
FatCompleteRequest( IrpContext, Irp, STATUS_SUCCESS );
|
||
|
||
DebugTrace(-1, Dbg, "FatCommonCleanup -> STATUS_SUCCESS\n", 0);
|
||
return STATUS_SUCCESS;
|
||
}
|
||
|
||
//
|
||
// If we call change the allocation or call CcUninitialize,
|
||
// we have to take the Fcb exclusive
|
||
//
|
||
|
||
if ((TypeOfOpen == UserFileOpen) || (TypeOfOpen == UserDirectoryOpen)) {
|
||
|
||
NT_ASSERT( Fcb != NULL );
|
||
|
||
(VOID)FatAcquireExclusiveFcb( IrpContext, Fcb );
|
||
|
||
AcquiredFcb = TRUE;
|
||
|
||
//
|
||
// Do a check here if this was a DELETE_ON_CLOSE FileObject, and
|
||
// set the Fcb flag appropriately.
|
||
//
|
||
|
||
if (FlagOn(Ccb->Flags, CCB_FLAG_DELETE_ON_CLOSE)) {
|
||
|
||
NT_ASSERT( NodeType(Fcb) != FAT_NTC_ROOT_DCB );
|
||
|
||
//
|
||
// Transfer the delete-on-close state to the FCB. We do this rather
|
||
// than leave the CCB_FLAG_DELETE_ON_CLOSE flag set so that if we
|
||
// end up breaking an oplock and come in again we won't try to break
|
||
// the oplock again (and again, and again...).
|
||
//
|
||
|
||
SetFlag( Fcb->FcbState, FCB_STATE_DELETE_ON_CLOSE );
|
||
ClearFlag( Ccb->Flags, CCB_FLAG_DELETE_ON_CLOSE );
|
||
|
||
#if (NTDDI_VERSION >= NTDDI_WIN8)
|
||
ProcessingDeleteOnClose = TRUE;
|
||
#endif
|
||
|
||
//
|
||
// Report this to the dir notify package for a directory.
|
||
//
|
||
|
||
if (TypeOfOpen == UserDirectoryOpen) {
|
||
|
||
#ifdef _MSC_VER
|
||
#pragma prefast( suppress:6309, "FullDirectoryName may be NULL if NotifyIrp is also NULL. this indicates the object is being deleted." )
|
||
#endif
|
||
FsRtlNotifyFullChangeDirectory( Vcb->NotifySync,
|
||
&Vcb->DirNotifyList,
|
||
FileObject->FsContext,
|
||
NULL,
|
||
FALSE,
|
||
FALSE,
|
||
0,
|
||
NULL,
|
||
NULL,
|
||
NULL );
|
||
}
|
||
}
|
||
|
||
//
|
||
// Now if we may delete the file, drop the Fcb and acquire the Vcb
|
||
// first. Note that while we own the Fcb exclusive, a file cannot
|
||
// become DELETE_ON_CLOSE and cannot be opened via CommonCreate.
|
||
//
|
||
|
||
if ((Fcb->UncleanCount == 1) &&
|
||
FlagOn(Fcb->FcbState, FCB_STATE_DELETE_ON_CLOSE) &&
|
||
(Fcb->FcbCondition != FcbBad) &&
|
||
!FlagOn(Vcb->VcbState, VCB_STATE_FLAG_WRITE_PROTECTED)) {
|
||
|
||
FatReleaseFcb( IrpContext, Fcb );
|
||
AcquiredFcb = FALSE;
|
||
|
||
(VOID)FatAcquireExclusiveVcb( IrpContext, Vcb );
|
||
AcquiredVcb = TRUE;
|
||
|
||
(VOID)FatAcquireExclusiveFcb( IrpContext, Fcb );
|
||
AcquiredFcb = TRUE;
|
||
}
|
||
}
|
||
|
||
//
|
||
// For user DASD cleanups, grab the Vcb exclusive.
|
||
//
|
||
|
||
if (TypeOfOpen == UserVolumeOpen) {
|
||
|
||
(VOID)FatAcquireExclusiveVcb( IrpContext, Vcb );
|
||
AcquiredVcb = TRUE;
|
||
}
|
||
|
||
//
|
||
// Complete any Notify Irps on this file handle.
|
||
//
|
||
|
||
if (TypeOfOpen == UserDirectoryOpen) {
|
||
|
||
FsRtlNotifyCleanup( Vcb->NotifySync,
|
||
&Vcb->DirNotifyList,
|
||
Ccb );
|
||
}
|
||
|
||
//
|
||
// Determine the Fcb state, Good or Bad, for better or for worse.
|
||
//
|
||
// We can only read the volume file if VcbCondition is good.
|
||
//
|
||
|
||
if ( Fcb != NULL) {
|
||
|
||
//
|
||
// Stop any raises from FatVerifyFcb, unless it is REAL bad.
|
||
//
|
||
|
||
_SEH2_TRY {
|
||
|
||
_SEH2_TRY {
|
||
|
||
FatVerifyFcb( IrpContext, Fcb );
|
||
|
||
} _SEH2_EXCEPT( FsRtlIsNtstatusExpected(_SEH2_GetExceptionCode()) ?
|
||
EXCEPTION_EXECUTE_HANDLER : EXCEPTION_CONTINUE_SEARCH ) {
|
||
|
||
FatResetExceptionState( IrpContext );
|
||
} _SEH2_END;
|
||
|
||
} _SEH2_FINALLY {
|
||
|
||
if ( _SEH2_AbnormalTermination() ) {
|
||
|
||
//
|
||
// We will be raising out of here.
|
||
//
|
||
|
||
if (AcquiredFcb) { FatReleaseFcb( IrpContext, Fcb ); }
|
||
if (AcquiredVcb) { FatReleaseVcb( IrpContext, Vcb ); }
|
||
}
|
||
} _SEH2_END;
|
||
}
|
||
|
||
_SEH2_TRY {
|
||
|
||
#if (NTDDI_VERSION >= NTDDI_WIN8)
|
||
|
||
//
|
||
// See if this is a delete-on-close handle on a file or empty directory.
|
||
// If so we may need to break an oplock. We do this in the try block
|
||
// so that resources will be properly released.
|
||
//
|
||
|
||
if (ProcessingDeleteOnClose &&
|
||
FatIsFileOplockable( Fcb ) &&
|
||
((NodeType( Fcb ) != FAT_NTC_DCB) ||
|
||
FatIsDirectoryEmpty( IrpContext, Fcb ))) {
|
||
|
||
Status = FsRtlCheckOplockEx( FatGetFcbOplock(Fcb),
|
||
Irp,
|
||
OPLOCK_FLAG_CLOSING_DELETE_ON_CLOSE,
|
||
IrpContext,
|
||
FatOplockComplete,
|
||
FatPrePostIrp );
|
||
|
||
if (Status != STATUS_SUCCESS) {
|
||
|
||
if (Status == STATUS_PENDING) {
|
||
|
||
SetFlag( IrpContext->Flags, IRP_CONTEXT_FLAG_CLEANUP_BREAKING_OPLOCK );
|
||
try_return( Status );
|
||
|
||
} else {
|
||
|
||
FatNormalizeAndRaiseStatus( IrpContext, Status );
|
||
}
|
||
}
|
||
}
|
||
#endif
|
||
|
||
//
|
||
// Case on the type of open that we are trying to cleanup.
|
||
// For all cases we need to set the share access to point to the
|
||
// share access variable (if there is one). After the switch
|
||
// we then remove the share access and complete the Irp.
|
||
// In the case of UserFileOpen we actually have a lot more work
|
||
// to do and we have the FsdLockControl complete the Irp for us.
|
||
//
|
||
|
||
switch (TypeOfOpen) {
|
||
|
||
case DirectoryFile:
|
||
case VirtualVolumeFile:
|
||
|
||
DebugTrace(0, Dbg, "Cleanup VirtualVolumeFile/DirectoryFile\n", 0);
|
||
|
||
ShareAccess = NULL;
|
||
|
||
break;
|
||
|
||
|
||
case UserVolumeOpen:
|
||
|
||
DebugTrace(0, Dbg, "Cleanup UserVolumeOpen\n", 0);
|
||
|
||
if (FlagOn( Ccb->Flags, CCB_FLAG_COMPLETE_DISMOUNT )) {
|
||
|
||
FatCheckForDismount( IrpContext, Vcb, TRUE );
|
||
|
||
//
|
||
// If this handle had write access, and actually wrote something,
|
||
// flush the device buffers, and then set the verify bit now
|
||
// just to be safe (in case there is no dismount).
|
||
//
|
||
|
||
} else if (FileObject->WriteAccess &&
|
||
FlagOn(FileObject->Flags, FO_FILE_MODIFIED)) {
|
||
|
||
(VOID)FatHijackIrpAndFlushDevice( IrpContext,
|
||
Irp,
|
||
Vcb->TargetDeviceObject );
|
||
|
||
SetFlag(Vcb->Vpb->RealDevice->Flags, DO_VERIFY_VOLUME);
|
||
}
|
||
|
||
//
|
||
// If the volume is locked by this file object then release
|
||
// the volume and send notification.
|
||
//
|
||
|
||
if (FlagOn(Vcb->VcbState, VCB_STATE_FLAG_LOCKED) &&
|
||
(Vcb->FileObjectWithVcbLocked == FileObject)) {
|
||
|
||
FatAutoUnlock( IrpContext, Vcb );
|
||
SendUnlockNotification = TRUE;
|
||
}
|
||
|
||
ShareAccess = &Vcb->ShareAccess;
|
||
|
||
break;
|
||
|
||
case EaFile:
|
||
|
||
DebugTrace(0, Dbg, "Cleanup EaFileObject\n", 0);
|
||
|
||
ShareAccess = NULL;
|
||
|
||
break;
|
||
|
||
case UserDirectoryOpen:
|
||
|
||
DebugTrace(0, Dbg, "Cleanup UserDirectoryOpen\n", 0);
|
||
|
||
ShareAccess = &Fcb->ShareAccess;
|
||
|
||
//
|
||
// Determine here if we should try do delayed close.
|
||
//
|
||
|
||
if ((Fcb->UncleanCount == 1) &&
|
||
(Fcb->OpenCount == 1) &&
|
||
(Fcb->Specific.Dcb.DirectoryFileOpenCount == 0) &&
|
||
!FlagOn(Fcb->FcbState, FCB_STATE_DELETE_ON_CLOSE) &&
|
||
Fcb->FcbCondition == FcbGood) {
|
||
|
||
//
|
||
// Delay our close.
|
||
//
|
||
|
||
SetFlag( Fcb->FcbState, FCB_STATE_DELAY_CLOSE );
|
||
}
|
||
|
||
//
|
||
// Clear the deny defrag bit, if the handle we're cleaning up was the one that set it.
|
||
//
|
||
|
||
if( FlagOn(Fcb->FcbState, FCB_STATE_DENY_DEFRAG) && FlagOn(Ccb->Flags, CCB_FLAG_DENY_DEFRAG) ) {
|
||
|
||
ClearFlag(Ccb->Flags, CCB_FLAG_DENY_DEFRAG);
|
||
ClearFlag(Fcb->FcbState, FCB_STATE_DENY_DEFRAG );
|
||
}
|
||
|
||
if ((VcbGood == Vcb->VcbCondition) &&
|
||
!FlagOn(Vcb->VcbState, VCB_STATE_FLAG_SHUTDOWN)) {
|
||
|
||
FatUpdateDirentFromFcb( IrpContext, FileObject, Fcb, Ccb );
|
||
|
||
//
|
||
// If the directory has a unclean count of 1 then we know
|
||
// that this is the last handle for the file object. If
|
||
// we are supposed to delete it, do so.
|
||
//
|
||
|
||
if ((Fcb->UncleanCount == 1) &&
|
||
(NodeType(Fcb) == FAT_NTC_DCB) &&
|
||
(FlagOn(Fcb->FcbState, FCB_STATE_DELETE_ON_CLOSE)) &&
|
||
(Fcb->FcbCondition == FcbGood) &&
|
||
!FlagOn(Vcb->VcbState, VCB_STATE_FLAG_WRITE_PROTECTED)) {
|
||
|
||
if (!FatIsDirectoryEmpty(IrpContext, Fcb)) {
|
||
|
||
//
|
||
// If there are files in the directory at this point,
|
||
// forget that we were trying to delete it.
|
||
//
|
||
|
||
ClearFlag( Fcb->FcbState, FCB_STATE_DELETE_ON_CLOSE );
|
||
|
||
} else {
|
||
|
||
#if (NTDDI_VERSION >= NTDDI_WIN8)
|
||
NTSTATUS BreakStatus;
|
||
#endif
|
||
|
||
//
|
||
// Even if something goes wrong, we cannot turn back!
|
||
//
|
||
|
||
_SEH2_TRY {
|
||
|
||
DELETE_CONTEXT DeleteContext;
|
||
|
||
|
||
//
|
||
// Before truncating file allocation remember this
|
||
// info for FatDeleteDirent.
|
||
//
|
||
|
||
DeleteContext.FileSize = Fcb->Header.FileSize.LowPart;
|
||
DeleteContext.FirstClusterOfFile = Fcb->FirstClusterOfFile;
|
||
|
||
//
|
||
// Synchronize here with paging IO
|
||
//
|
||
|
||
(VOID)ExAcquireResourceExclusiveLite( Fcb->Header.PagingIoResource,
|
||
TRUE );
|
||
|
||
Fcb->Header.FileSize.LowPart = 0;
|
||
|
||
ExReleaseResourceLite( Fcb->Header.PagingIoResource );
|
||
|
||
//
|
||
// Truncate the file allocation down to zero
|
||
//
|
||
|
||
DebugTrace(0, Dbg, "Delete File allocation\n", 0);
|
||
|
||
FatTruncateFileAllocation( IrpContext, Fcb, 0 );
|
||
|
||
if (Fcb->Header.AllocationSize.LowPart == 0) {
|
||
|
||
//
|
||
// Tunnel and remove the dirent for the directory
|
||
//
|
||
|
||
DebugTrace(0, Dbg, "Delete the directory dirent\n", 0);
|
||
|
||
FatTunnelFcbOrDcb( Fcb, NULL );
|
||
|
||
FatDeleteDirent( IrpContext, Fcb, &DeleteContext, TRUE );
|
||
|
||
//
|
||
// Report that we have removed an entry.
|
||
//
|
||
|
||
FatNotifyReportChange( IrpContext,
|
||
Vcb,
|
||
Fcb,
|
||
FILE_NOTIFY_CHANGE_DIR_NAME,
|
||
FILE_ACTION_REMOVED );
|
||
}
|
||
|
||
} _SEH2_EXCEPT( FsRtlIsNtstatusExpected(_SEH2_GetExceptionCode()) ?
|
||
EXCEPTION_EXECUTE_HANDLER : EXCEPTION_CONTINUE_SEARCH ) {
|
||
|
||
FatResetExceptionState( IrpContext );
|
||
} _SEH2_END;
|
||
|
||
//
|
||
// Remove the entry from the name table.
|
||
// This will ensure that
|
||
// we will not collide with the Dcb if the user wants
|
||
// to recreate the same file over again before we
|
||
// get a close irp.
|
||
//
|
||
|
||
FatRemoveNames( IrpContext, Fcb );
|
||
|
||
#if (NTDDI_VERSION >= NTDDI_WIN8)
|
||
//
|
||
// We've removed the names so break any parent directory oplock.
|
||
// Directory oplock breaks are always advisory, so we will never
|
||
// block/get STATUS_PENDING here.
|
||
//
|
||
|
||
BreakStatus = FsRtlCheckOplockEx( FatGetFcbOplock(Fcb->ParentDcb),
|
||
Irp,
|
||
(OPLOCK_FLAG_PARENT_OBJECT |
|
||
OPLOCK_FLAG_REMOVING_FILE_OR_LINK),
|
||
NULL,
|
||
NULL,
|
||
NULL );
|
||
|
||
ASSERT( BreakStatus != STATUS_PENDING );
|
||
#endif
|
||
}
|
||
}
|
||
}
|
||
|
||
//
|
||
// Decrement the unclean count.
|
||
//
|
||
|
||
NT_ASSERT( Fcb->UncleanCount != 0 );
|
||
Fcb->UncleanCount -= 1;
|
||
|
||
break;
|
||
|
||
case UserFileOpen:
|
||
|
||
DebugTrace(0, Dbg, "Cleanup UserFileOpen\n", 0);
|
||
|
||
ShareAccess = &Fcb->ShareAccess;
|
||
|
||
//
|
||
// Determine here if we should do a delayed close.
|
||
//
|
||
|
||
if ((FileObject->SectionObjectPointer->DataSectionObject == NULL) &&
|
||
(FileObject->SectionObjectPointer->ImageSectionObject == NULL) &&
|
||
(Fcb->UncleanCount == 1) &&
|
||
(Fcb->OpenCount == 1) &&
|
||
!FlagOn(Fcb->FcbState, FCB_STATE_DELETE_ON_CLOSE) &&
|
||
!FlagOn(Fcb->FcbState, FCB_STATE_PAGING_FILE) &&
|
||
Fcb->FcbCondition == FcbGood) {
|
||
|
||
//
|
||
// Delay our close.
|
||
//
|
||
|
||
SetFlag( Fcb->FcbState, FCB_STATE_DELAY_CLOSE );
|
||
}
|
||
|
||
//
|
||
// Clear the deny defrag bit, if the handle we're cleaning up was the one that set it.
|
||
//
|
||
|
||
if( FlagOn(Fcb->FcbState, FCB_STATE_DENY_DEFRAG) && FlagOn(Ccb->Flags, CCB_FLAG_DENY_DEFRAG) ) {
|
||
|
||
ClearFlag(Ccb->Flags, CCB_FLAG_DENY_DEFRAG);
|
||
ClearFlag(Fcb->FcbState, FCB_STATE_DENY_DEFRAG );
|
||
}
|
||
|
||
//
|
||
// Unlock all outstanding file locks.
|
||
//
|
||
|
||
(VOID) FsRtlFastUnlockAll( &Fcb->Specific.Fcb.FileLock,
|
||
FileObject,
|
||
IoGetRequestorProcess( Irp ),
|
||
NULL );
|
||
|
||
|
||
|
||
//
|
||
// We can proceed with on-disk updates only if the volume is mounted
|
||
// and we can still write to it if it hasn't been shutdown. Remember that
|
||
// we toss all sections in the failed-verify and dismount cases.
|
||
//
|
||
|
||
if ((Vcb->VcbCondition == VcbGood) &&
|
||
!FlagOn(Vcb->VcbState, VCB_STATE_FLAG_SHUTDOWN)) {
|
||
|
||
if (Fcb->FcbCondition == FcbGood) {
|
||
|
||
|
||
FatUpdateDirentFromFcb( IrpContext, FileObject, Fcb, Ccb );
|
||
|
||
}
|
||
|
||
//
|
||
// If the file has a unclean count of 1 then we know
|
||
// that this is the last handle for the file object.
|
||
//
|
||
|
||
if ( (Fcb->UncleanCount == 1) && (Fcb->FcbCondition == FcbGood) ) {
|
||
|
||
DELETE_CONTEXT DeleteContext;
|
||
|
||
//
|
||
// Check if we should be deleting the file. The
|
||
// delete operation really deletes the file but
|
||
// keeps the Fcb around for close to do away with.
|
||
//
|
||
|
||
if (FlagOn(Fcb->FcbState, FCB_STATE_DELETE_ON_CLOSE) &&
|
||
!FlagOn(Vcb->VcbState, VCB_STATE_FLAG_WRITE_PROTECTED)) {
|
||
|
||
//
|
||
// Before truncating file allocation remember this
|
||
// info for FatDeleteDirent.
|
||
//
|
||
|
||
DeleteContext.FileSize = Fcb->Header.FileSize.LowPart;
|
||
DeleteContext.FirstClusterOfFile = Fcb->FirstClusterOfFile;
|
||
|
||
DebugTrace(0, Dbg, "Delete File allocation\n", 0);
|
||
|
||
//
|
||
// Synchronize here with paging IO
|
||
//
|
||
|
||
(VOID)ExAcquireResourceExclusiveLite( Fcb->Header.PagingIoResource,
|
||
TRUE );
|
||
|
||
Fcb->Header.FileSize.LowPart = 0;
|
||
Fcb->Header.ValidDataLength.LowPart = 0;
|
||
Fcb->ValidDataToDisk = 0;
|
||
|
||
ExReleaseResourceLite( Fcb->Header.PagingIoResource );
|
||
|
||
_SEH2_TRY {
|
||
|
||
FatSetFileSizeInDirent( IrpContext, Fcb, NULL );
|
||
|
||
} _SEH2_EXCEPT( FsRtlIsNtstatusExpected(_SEH2_GetExceptionCode()) ?
|
||
EXCEPTION_EXECUTE_HANDLER : EXCEPTION_CONTINUE_SEARCH ) {
|
||
|
||
FatResetExceptionState( IrpContext );
|
||
} _SEH2_END;
|
||
|
||
Fcb->FcbState |= FCB_STATE_TRUNCATE_ON_CLOSE;
|
||
|
||
} else {
|
||
|
||
//
|
||
// We must zero between ValidDataLength and FileSize
|
||
//
|
||
|
||
if (!FlagOn(Fcb->FcbState, FCB_STATE_PAGING_FILE) &&
|
||
(Fcb->Header.ValidDataLength.LowPart < Fcb->Header.FileSize.LowPart)) {
|
||
|
||
ULONG ValidDataLength;
|
||
|
||
ValidDataLength = Fcb->Header.ValidDataLength.LowPart;
|
||
|
||
if (ValidDataLength < Fcb->ValidDataToDisk) {
|
||
ValidDataLength = Fcb->ValidDataToDisk;
|
||
}
|
||
|
||
//
|
||
// Recheck, VDD can be >= FS
|
||
//
|
||
|
||
if (ValidDataLength < Fcb->Header.FileSize.LowPart) {
|
||
|
||
_SEH2_TRY {
|
||
|
||
(VOID)FatZeroData( IrpContext,
|
||
Vcb,
|
||
FileObject,
|
||
ValidDataLength,
|
||
Fcb->Header.FileSize.LowPart -
|
||
ValidDataLength );
|
||
|
||
//
|
||
// Since we just zeroed this, we can now bump
|
||
// up VDL in the Fcb.
|
||
//
|
||
|
||
Fcb->ValidDataToDisk =
|
||
Fcb->Header.ValidDataLength.LowPart =
|
||
Fcb->Header.FileSize.LowPart;
|
||
|
||
//
|
||
// We inform Cc of the motion so that the cache map is updated.
|
||
// This prevents optimized zero-page faults in case the cache
|
||
// structures are re-used for another handle before they are torn
|
||
// down by our soon-to-occur uninitialize. If they were, a noncached
|
||
// producer could write into the region we just zeroed and Cc would
|
||
// be none the wiser, then our async cached reader comes in and takes
|
||
// the optimized path, and we get bad (zero) data.
|
||
//
|
||
// If this was memory mapped, we don't have to (can't) tell Cc, it'll
|
||
// figure it out when a cached handle is opened.
|
||
//
|
||
|
||
if (CcIsFileCached( FileObject )) {
|
||
CcSetFileSizes( FileObject, (PCC_FILE_SIZES)&Fcb->Header.AllocationSize );
|
||
}
|
||
|
||
} _SEH2_EXCEPT( FsRtlIsNtstatusExpected(_SEH2_GetExceptionCode()) ?
|
||
EXCEPTION_EXECUTE_HANDLER : EXCEPTION_CONTINUE_SEARCH ) {
|
||
|
||
FatResetExceptionState( IrpContext );
|
||
} _SEH2_END;
|
||
}
|
||
}
|
||
}
|
||
|
||
//
|
||
// See if we are supposed to truncate the file on the last
|
||
// close. If we cannot wait we'll ship this off to the fsp
|
||
//
|
||
|
||
_SEH2_TRY {
|
||
|
||
if (FlagOn(Fcb->FcbState, FCB_STATE_TRUNCATE_ON_CLOSE)) {
|
||
|
||
DebugTrace(0, Dbg, "truncate file allocation\n", 0);
|
||
|
||
if (Vcb->VcbCondition == VcbGood) {
|
||
|
||
|
||
FatTruncateFileAllocation( IrpContext,
|
||
Fcb,
|
||
Fcb->Header.FileSize.LowPart );
|
||
|
||
|
||
}
|
||
|
||
//
|
||
// We also have to get rid of the Cache Map because
|
||
// this is the only way we have of trashing the
|
||
// truncated pages.
|
||
//
|
||
|
||
LocalTruncateSize = Fcb->Header.FileSize;
|
||
TruncateSize = &LocalTruncateSize;
|
||
|
||
//
|
||
// Mark the Fcb as having now been truncated, just incase
|
||
// we have to reship this off to the fsp.
|
||
//
|
||
|
||
Fcb->FcbState &= ~FCB_STATE_TRUNCATE_ON_CLOSE;
|
||
}
|
||
|
||
//
|
||
// Now check again if we are to delete the file and if
|
||
// so then we remove the file from the disk.
|
||
//
|
||
|
||
if (FlagOn(Fcb->FcbState, FCB_STATE_DELETE_ON_CLOSE) &&
|
||
Fcb->Header.AllocationSize.LowPart == 0) {
|
||
|
||
DebugTrace(0, Dbg, "Delete File\n", 0);
|
||
|
||
//
|
||
// Now tunnel and delete the dirent
|
||
//
|
||
|
||
FatTunnelFcbOrDcb( Fcb, Ccb );
|
||
|
||
FatDeleteDirent( IrpContext, Fcb, &DeleteContext, TRUE );
|
||
|
||
//
|
||
// Report that we have removed an entry.
|
||
//
|
||
|
||
FatNotifyReportChange( IrpContext,
|
||
Vcb,
|
||
Fcb,
|
||
FILE_NOTIFY_CHANGE_FILE_NAME,
|
||
FILE_ACTION_REMOVED );
|
||
}
|
||
|
||
} _SEH2_EXCEPT( FsRtlIsNtstatusExpected(_SEH2_GetExceptionCode()) ?
|
||
EXCEPTION_EXECUTE_HANDLER : EXCEPTION_CONTINUE_SEARCH ) {
|
||
|
||
FatResetExceptionState( IrpContext );
|
||
} _SEH2_END;
|
||
|
||
if (FlagOn(Fcb->FcbState, FCB_STATE_DELETE_ON_CLOSE)) {
|
||
|
||
#if (NTDDI_VERSION >= NTDDI_WIN8)
|
||
NTSTATUS BreakStatus;
|
||
#endif
|
||
//
|
||
// Remove the entry from the splay table. This will
|
||
// ensure that we will not collide with the Fcb if the
|
||
// user wants to recreate the same file over again
|
||
// before we get a close irp.
|
||
//
|
||
// Note that we remove the name even if we couldn't
|
||
// truncate the allocation and remove the dirent above.
|
||
//
|
||
|
||
FatRemoveNames( IrpContext, Fcb );
|
||
|
||
#if (NTDDI_VERSION >= NTDDI_WIN8)
|
||
//
|
||
// We've removed the names so break any parent directory oplock.
|
||
// Directory oplock breaks are always advisory, so we will never
|
||
// block/get STATUS_PENDING here.
|
||
//
|
||
|
||
BreakStatus = FsRtlCheckOplockEx( FatGetFcbOplock(Fcb->ParentDcb),
|
||
Irp,
|
||
(OPLOCK_FLAG_PARENT_OBJECT |
|
||
OPLOCK_FLAG_REMOVING_FILE_OR_LINK),
|
||
NULL,
|
||
NULL,
|
||
NULL );
|
||
|
||
ASSERT( BreakStatus != STATUS_PENDING );
|
||
#endif
|
||
}
|
||
}
|
||
}
|
||
|
||
//
|
||
// We've just finished everything associated with an unclean
|
||
// fcb so now decrement the unclean count before releasing
|
||
// the resource.
|
||
//
|
||
|
||
NT_ASSERT( Fcb->UncleanCount != 0 );
|
||
Fcb->UncleanCount -= 1;
|
||
if (!FlagOn( FileObject->Flags, FO_CACHE_SUPPORTED )) {
|
||
NT_ASSERT( Fcb->NonCachedUncleanCount != 0 );
|
||
Fcb->NonCachedUncleanCount -= 1;
|
||
}
|
||
|
||
//
|
||
// If this was the last cached open, and there are open
|
||
// non-cached handles, attempt a flush and purge operation
|
||
// to avoid cache coherency overhead from these non-cached
|
||
// handles later. We ignore any I/O errors from the flush.
|
||
//
|
||
|
||
if (FlagOn( FileObject->Flags, FO_CACHE_SUPPORTED ) &&
|
||
(Fcb->NonCachedUncleanCount != 0) &&
|
||
(Fcb->NonCachedUncleanCount == Fcb->UncleanCount) &&
|
||
(Fcb->NonPaged->SectionObjectPointers.DataSectionObject != NULL)) {
|
||
|
||
CcFlushCache( &Fcb->NonPaged->SectionObjectPointers, NULL, 0, NULL );
|
||
|
||
//
|
||
// Grab and release PagingIo to serialize ourselves with the lazy writer.
|
||
// This will work to ensure that all IO has completed on the cached
|
||
// data and we will succesfully tear away the cache section.
|
||
//
|
||
|
||
ExAcquireResourceExclusiveLite( Fcb->Header.PagingIoResource, TRUE);
|
||
ExReleaseResourceLite( Fcb->Header.PagingIoResource );
|
||
|
||
CcPurgeCacheSection( &Fcb->NonPaged->SectionObjectPointers,
|
||
NULL,
|
||
0,
|
||
FALSE );
|
||
}
|
||
|
||
//
|
||
// If the file is invalid, hint to the cache that we should throw everything out.
|
||
//
|
||
|
||
if ( Fcb->FcbCondition == FcbBad ) {
|
||
|
||
TruncateSize = &FatLargeZero;
|
||
}
|
||
|
||
//
|
||
// Cleanup the cache map
|
||
//
|
||
|
||
CcUninitializeCacheMap( FileObject, TruncateSize, NULL );
|
||
|
||
break;
|
||
|
||
default:
|
||
|
||
#ifdef _MSC_VER
|
||
#pragma prefast( suppress: 28159, "if the type of open is unknown then things are very bad." )
|
||
#endif
|
||
FatBugCheck( TypeOfOpen, 0, 0 );
|
||
}
|
||
|
||
//
|
||
// We must clean up the share access at this time, since we may not
|
||
// get a Close call for awhile if the file was mapped through this
|
||
// File Object.
|
||
//
|
||
|
||
if (ShareAccess != NULL) {
|
||
|
||
DebugTrace(0, Dbg, "Cleanup the Share access\n", 0);
|
||
IoRemoveShareAccess( FileObject, ShareAccess );
|
||
}
|
||
|
||
if ((TypeOfOpen == UserFileOpen)
|
||
#if (NTDDI_VERSION >= NTDDI_WIN8)
|
||
||
|
||
(TypeOfOpen == UserDirectoryOpen)
|
||
#endif
|
||
) {
|
||
|
||
//
|
||
// Coordinate the cleanup operation with the oplock state.
|
||
// Cleanup operations can always cleanup immediately.
|
||
//
|
||
|
||
FsRtlCheckOplock( FatGetFcbOplock(Fcb),
|
||
Irp,
|
||
IrpContext,
|
||
NULL,
|
||
NULL );
|
||
|
||
Fcb->Header.IsFastIoPossible = FatIsFastIoPossible( Fcb );
|
||
}
|
||
|
||
//
|
||
// First set the FO_CLEANUP_COMPLETE flag.
|
||
//
|
||
|
||
SetFlag( FileObject->Flags, FO_CLEANUP_COMPLETE );
|
||
|
||
Status = STATUS_SUCCESS;
|
||
|
||
//
|
||
// Now unpin any repinned Bcbs.
|
||
//
|
||
|
||
FatUnpinRepinnedBcbs( IrpContext );
|
||
|
||
//
|
||
// If this was deferred flush media, flush the volume.
|
||
// We used to do this in lieu of write through for all removable
|
||
// media.
|
||
//
|
||
|
||
if (FlagOn(Vcb->VcbState, VCB_STATE_FLAG_DEFERRED_FLUSH) &&
|
||
!FlagOn(Vcb->VcbState, VCB_STATE_FLAG_WRITE_PROTECTED)) {
|
||
|
||
//
|
||
// Flush the file.
|
||
//
|
||
|
||
if ((TypeOfOpen == UserFileOpen) &&
|
||
FlagOn(FileObject->Flags, FO_FILE_MODIFIED)) {
|
||
|
||
Status = FatFlushFile( IrpContext, Fcb, Flush );
|
||
}
|
||
|
||
//
|
||
// If that worked ok, then see if we should flush the FAT as well.
|
||
//
|
||
|
||
if (NT_SUCCESS(Status) && Fcb && !FatIsFat12( Vcb) &&
|
||
FlagOn( Fcb->FcbState, FCB_STATE_FLUSH_FAT)) {
|
||
|
||
Status = FatFlushFat( IrpContext, Vcb);
|
||
|
||
//
|
||
// Also flush the parent directory.
|
||
//
|
||
|
||
if (NT_SUCCESS(Status) && (Fcb->ParentDcb != NULL)) {
|
||
|
||
Status = FatFlushFile( IrpContext, Fcb->ParentDcb, Flush );
|
||
}
|
||
}
|
||
|
||
if (!NT_SUCCESS(Status)) {
|
||
|
||
FatNormalizeAndRaiseStatus( IrpContext, Status );
|
||
}
|
||
}
|
||
|
||
#if (NTDDI_VERSION >= NTDDI_WIN8)
|
||
|
||
try_exit: NOTHING;
|
||
|
||
#endif
|
||
|
||
} _SEH2_FINALLY {
|
||
|
||
DebugUnwind( FatCommonCleanup );
|
||
|
||
if (AcquiredFcb) { FatReleaseFcb( IrpContext, Fcb ); }
|
||
if (AcquiredVcb) { FatReleaseVcb( IrpContext, Vcb ); }
|
||
|
||
if (SendUnlockNotification) {
|
||
|
||
FsRtlNotifyVolumeEvent( FileObject, FSRTL_VOLUME_UNLOCK );
|
||
}
|
||
|
||
//
|
||
// If this is a normal termination then complete the request
|
||
//
|
||
|
||
if (!_SEH2_AbnormalTermination() &&
|
||
(Status != STATUS_PENDING)) {
|
||
|
||
FatCompleteRequest( IrpContext, Irp, Status );
|
||
}
|
||
|
||
DebugTrace(-1, Dbg, "FatCommonCleanup -> %08lx\n", Status);
|
||
} _SEH2_END;
|
||
|
||
return Status;
|
||
}
|
||
|
||
VOID
|
||
FatAutoUnlock (
|
||
IN PIRP_CONTEXT IrpContext,
|
||
IN PVCB Vcb
|
||
)
|
||
{
|
||
KIRQL SavedIrql;
|
||
|
||
//
|
||
// Unlock the volume.
|
||
//
|
||
|
||
UNREFERENCED_PARAMETER( IrpContext );
|
||
|
||
IoAcquireVpbSpinLock( &SavedIrql );
|
||
|
||
ClearFlag( Vcb->Vpb->Flags, (VPB_LOCKED | VPB_DIRECT_WRITES_ALLOWED) );
|
||
|
||
Vcb->VcbState &= ~VCB_STATE_FLAG_LOCKED;
|
||
Vcb->FileObjectWithVcbLocked = NULL;
|
||
|
||
IoReleaseVpbSpinLock( SavedIrql );
|
||
}
|
||
|
||
|