2005-08-24 18:29:45 +00:00
|
|
|
/*
|
2001-03-07 13:44:41 +00:00
|
|
|
* COPYRIGHT: See COPYING in the top level directory
|
|
|
|
* PROJECT: ReactOS kernel
|
2003-10-11 17:51:56 +00:00
|
|
|
* FILE: drivers/fs/vfat/cleanup.c
|
2001-03-07 13:44:41 +00:00
|
|
|
* PURPOSE: VFAT Filesystem
|
|
|
|
* PROGRAMMER: Jason Filby (jasonfilby@yahoo.com)
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* INCLUDES *****************************************************************/
|
|
|
|
|
|
|
|
#include "vfat.h"
|
|
|
|
|
2013-12-19 16:20:28 +00:00
|
|
|
#define NDEBUG
|
|
|
|
#include <debug.h>
|
|
|
|
|
2001-03-07 13:44:41 +00:00
|
|
|
/* FUNCTIONS ****************************************************************/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* FUNCTION: Cleans up after a file has been closed.
|
|
|
|
*/
|
2013-12-09 18:48:13 +00:00
|
|
|
static
|
|
|
|
NTSTATUS
|
|
|
|
VfatCleanupFile(
|
|
|
|
PVFAT_IRP_CONTEXT IrpContext)
|
2001-03-07 13:44:41 +00:00
|
|
|
{
|
2008-01-12 17:20:11 +00:00
|
|
|
PVFATFCB pFcb;
|
2017-05-02 19:33:14 +00:00
|
|
|
PVFATCCB pCcb;
|
2017-09-19 21:19:55 +00:00
|
|
|
BOOLEAN IsVolume;
|
2014-10-30 20:56:40 +00:00
|
|
|
PDEVICE_EXTENSION DeviceExt = IrpContext->DeviceExt;
|
2008-01-12 17:20:11 +00:00
|
|
|
PFILE_OBJECT FileObject = IrpContext->FileObject;
|
|
|
|
|
|
|
|
DPRINT("VfatCleanupFile(DeviceExt %p, FileObject %p)\n",
|
|
|
|
IrpContext->DeviceExt, FileObject);
|
2005-05-08 02:16:32 +00:00
|
|
|
|
2008-01-12 17:20:11 +00:00
|
|
|
/* FIXME: handle file/directory deletion here */
|
2013-12-09 18:48:13 +00:00
|
|
|
pFcb = (PVFATFCB)FileObject->FsContext;
|
2008-01-12 17:20:11 +00:00
|
|
|
if (!pFcb)
|
|
|
|
return STATUS_SUCCESS;
|
2005-05-08 02:16:32 +00:00
|
|
|
|
2017-09-19 21:19:55 +00:00
|
|
|
IsVolume = BooleanFlagOn(pFcb->Flags, FCB_IS_VOLUME);
|
|
|
|
if (IsVolume)
|
2003-10-11 17:51:56 +00:00
|
|
|
{
|
2008-01-12 17:20:11 +00:00
|
|
|
pFcb->OpenHandleCount--;
|
2006-12-31 16:43:40 +00:00
|
|
|
|
2008-01-12 17:20:11 +00:00
|
|
|
if (pFcb->OpenHandleCount != 0)
|
|
|
|
{
|
|
|
|
IoRemoveShareAccess(FileObject, &pFcb->FCBShareAccess);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2013-12-09 18:48:13 +00:00
|
|
|
if(!ExAcquireResourceExclusiveLite(&pFcb->MainResource,
|
2015-05-25 17:30:01 +00:00
|
|
|
BooleanFlagOn(IrpContext->Flags, IRPCONTEXT_CANWAIT)))
|
2008-01-12 17:20:11 +00:00
|
|
|
{
|
|
|
|
return STATUS_PENDING;
|
2006-12-31 16:43:40 +00:00
|
|
|
}
|
2013-12-09 18:48:13 +00:00
|
|
|
if(!ExAcquireResourceExclusiveLite(&pFcb->PagingIoResource,
|
2015-05-25 17:30:01 +00:00
|
|
|
BooleanFlagOn(IrpContext->Flags, IRPCONTEXT_CANWAIT)))
|
2006-12-31 16:43:40 +00:00
|
|
|
{
|
2013-12-09 18:48:13 +00:00
|
|
|
ExReleaseResourceLite(&pFcb->MainResource);
|
2008-01-12 17:20:11 +00:00
|
|
|
return STATUS_PENDING;
|
|
|
|
}
|
|
|
|
|
2017-05-02 19:33:14 +00:00
|
|
|
pCcb = FileObject->FsContext2;
|
|
|
|
if (BooleanFlagOn(pCcb->Flags, CCB_DELETE_ON_CLOSE))
|
|
|
|
{
|
|
|
|
pFcb->Flags |= FCB_DELETE_PENDING;
|
|
|
|
}
|
|
|
|
|
2014-03-12 19:58:11 +00:00
|
|
|
/* Notify about the cleanup */
|
|
|
|
FsRtlNotifyCleanup(IrpContext->DeviceExt->NotifySync,
|
|
|
|
&(IrpContext->DeviceExt->NotifyList),
|
|
|
|
FileObject->FsContext2);
|
|
|
|
|
2008-01-12 17:20:11 +00:00
|
|
|
pFcb->OpenHandleCount--;
|
2014-10-30 20:56:40 +00:00
|
|
|
DeviceExt->OpenHandleCount--;
|
2006-12-31 16:43:40 +00:00
|
|
|
|
2016-08-11 11:41:25 +00:00
|
|
|
if (!vfatFCBIsDirectory(pFcb) &&
|
2008-01-12 17:20:11 +00:00
|
|
|
FsRtlAreThereCurrentFileLocks(&pFcb->FileLock))
|
|
|
|
{
|
|
|
|
/* remove all locks this process have on this file */
|
|
|
|
FsRtlFastUnlockAll(&pFcb->FileLock,
|
|
|
|
FileObject,
|
|
|
|
IoGetRequestorProcess(IrpContext->Irp),
|
|
|
|
NULL);
|
|
|
|
}
|
2006-12-31 16:43:40 +00:00
|
|
|
|
2017-02-17 17:58:18 +00:00
|
|
|
if (BooleanFlagOn(pFcb->Flags, FCB_IS_DIRTY))
|
2008-01-12 17:20:11 +00:00
|
|
|
{
|
2017-02-17 21:24:12 +00:00
|
|
|
VfatUpdateEntry (pFcb, vfatVolumeIsFatX(DeviceExt));
|
2008-01-12 17:20:11 +00:00
|
|
|
}
|
|
|
|
|
2017-02-17 17:58:18 +00:00
|
|
|
if (BooleanFlagOn(pFcb->Flags, FCB_DELETE_PENDING) &&
|
2008-01-12 17:29:26 +00:00
|
|
|
pFcb->OpenHandleCount == 0)
|
2008-01-12 17:20:11 +00:00
|
|
|
{
|
2016-08-11 11:41:25 +00:00
|
|
|
if (vfatFCBIsDirectory(pFcb) &&
|
2017-02-17 22:25:03 +00:00
|
|
|
!VfatIsDirectoryEmpty(DeviceExt, pFcb))
|
2006-12-31 16:43:40 +00:00
|
|
|
{
|
2016-08-11 11:41:25 +00:00
|
|
|
pFcb->Flags &= ~FCB_DELETE_PENDING;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
PFILE_OBJECT tmpFileObject;
|
|
|
|
tmpFileObject = pFcb->FileObject;
|
|
|
|
if (tmpFileObject != NULL)
|
|
|
|
{
|
|
|
|
pFcb->FileObject = NULL;
|
|
|
|
CcUninitializeCacheMap(tmpFileObject, NULL, NULL);
|
|
|
|
ObDereferenceObject(tmpFileObject);
|
|
|
|
}
|
|
|
|
|
|
|
|
pFcb->RFCB.ValidDataLength.QuadPart = 0;
|
|
|
|
pFcb->RFCB.FileSize.QuadPart = 0;
|
|
|
|
pFcb->RFCB.AllocationSize.QuadPart = 0;
|
2013-12-09 18:48:13 +00:00
|
|
|
}
|
2008-01-12 17:20:11 +00:00
|
|
|
}
|
2008-11-21 10:17:14 +00:00
|
|
|
|
2008-11-21 13:45:40 +00:00
|
|
|
/* Uninitialize the cache (should be done even if caching was never initialized) */
|
|
|
|
CcUninitializeCacheMap(FileObject, &pFcb->RFCB.FileSize, NULL);
|
2008-11-21 10:17:14 +00:00
|
|
|
|
2017-02-17 17:58:18 +00:00
|
|
|
if (BooleanFlagOn(pFcb->Flags, FCB_DELETE_PENDING) &&
|
2015-03-28 16:58:53 +00:00
|
|
|
pFcb->OpenHandleCount == 0)
|
|
|
|
{
|
|
|
|
VfatDelEntry(DeviceExt, pFcb, NULL);
|
|
|
|
|
|
|
|
FsRtlNotifyFullReportChange(DeviceExt->NotifySync,
|
|
|
|
&(DeviceExt->NotifyList),
|
|
|
|
(PSTRING)&pFcb->PathNameU,
|
|
|
|
pFcb->PathNameU.Length - pFcb->LongNameU.Length,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
2017-02-17 18:06:38 +00:00
|
|
|
vfatFCBIsDirectory(pFcb) ?
|
2017-02-17 17:58:18 +00:00
|
|
|
FILE_NOTIFY_CHANGE_DIR_NAME : FILE_NOTIFY_CHANGE_FILE_NAME,
|
2015-03-28 16:58:53 +00:00
|
|
|
FILE_ACTION_REMOVED,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
|
2008-01-12 17:20:11 +00:00
|
|
|
if (pFcb->OpenHandleCount != 0)
|
|
|
|
{
|
|
|
|
IoRemoveShareAccess(FileObject, &pFcb->FCBShareAccess);
|
|
|
|
}
|
2006-12-31 16:43:40 +00:00
|
|
|
|
2008-01-12 17:20:11 +00:00
|
|
|
FileObject->Flags |= FO_CLEANUP_COMPLETE;
|
2006-12-31 16:43:40 +00:00
|
|
|
|
2013-12-09 18:48:13 +00:00
|
|
|
ExReleaseResourceLite(&pFcb->PagingIoResource);
|
|
|
|
ExReleaseResourceLite(&pFcb->MainResource);
|
2003-10-11 17:51:56 +00:00
|
|
|
}
|
2008-01-12 17:20:11 +00:00
|
|
|
|
2014-11-29 20:26:07 +00:00
|
|
|
#ifdef ENABLE_SWAPOUT
|
2017-09-19 21:19:55 +00:00
|
|
|
if (IsVolume && BooleanFlagOn(DeviceExt->Flags, VCB_DISMOUNT_PENDING))
|
2014-11-10 22:11:36 +00:00
|
|
|
{
|
|
|
|
VfatCheckForDismount(DeviceExt, FALSE);
|
|
|
|
}
|
2014-11-29 20:26:07 +00:00
|
|
|
#endif
|
2014-11-10 22:11:36 +00:00
|
|
|
|
2008-01-12 17:20:11 +00:00
|
|
|
return STATUS_SUCCESS;
|
2001-03-07 13:44:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* FUNCTION: Cleans up after a file has been closed.
|
|
|
|
*/
|
2013-12-09 18:48:13 +00:00
|
|
|
NTSTATUS
|
|
|
|
VfatCleanup(
|
|
|
|
PVFAT_IRP_CONTEXT IrpContext)
|
2001-03-07 13:44:41 +00:00
|
|
|
{
|
2008-01-12 17:20:11 +00:00
|
|
|
NTSTATUS Status;
|
2001-03-07 13:44:41 +00:00
|
|
|
|
2008-01-12 17:20:11 +00:00
|
|
|
DPRINT("VfatCleanup(DeviceObject %p, Irp %p)\n", IrpContext->DeviceObject, IrpContext->Irp);
|
2001-03-07 13:44:41 +00:00
|
|
|
|
2008-01-12 17:20:11 +00:00
|
|
|
if (IrpContext->DeviceObject == VfatGlobalData->DeviceObject)
|
|
|
|
{
|
2015-05-15 16:03:29 +00:00
|
|
|
IrpContext->Irp->IoStatus.Information = 0;
|
|
|
|
return STATUS_SUCCESS;
|
2008-01-12 17:20:11 +00:00
|
|
|
}
|
2002-05-15 18:05:00 +00:00
|
|
|
|
2013-12-09 18:48:13 +00:00
|
|
|
if (!ExAcquireResourceExclusiveLite(&IrpContext->DeviceExt->DirResource,
|
2015-05-25 17:30:01 +00:00
|
|
|
BooleanFlagOn(IrpContext->Flags, IRPCONTEXT_CANWAIT)))
|
2008-01-12 17:20:11 +00:00
|
|
|
{
|
2015-05-15 16:03:29 +00:00
|
|
|
return VfatMarkIrpContextForQueue(IrpContext);
|
2008-01-12 17:20:11 +00:00
|
|
|
}
|
2001-03-07 13:44:41 +00:00
|
|
|
|
2008-01-12 17:20:11 +00:00
|
|
|
Status = VfatCleanupFile(IrpContext);
|
2001-03-07 13:44:41 +00:00
|
|
|
|
2013-12-09 18:48:13 +00:00
|
|
|
ExReleaseResourceLite(&IrpContext->DeviceExt->DirResource);
|
2001-11-02 22:47:36 +00:00
|
|
|
|
2008-01-12 17:20:11 +00:00
|
|
|
if (Status == STATUS_PENDING)
|
|
|
|
{
|
2015-05-15 16:03:29 +00:00
|
|
|
return VfatMarkIrpContextForQueue(IrpContext);
|
2008-01-12 17:20:11 +00:00
|
|
|
}
|
2006-12-31 16:43:40 +00:00
|
|
|
|
2008-01-12 17:20:11 +00:00
|
|
|
IrpContext->Irp->IoStatus.Information = 0;
|
2013-12-09 18:48:13 +00:00
|
|
|
return Status;
|
2001-03-07 13:44:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* EOF */
|