mirror of
https://github.com/winfsp/winfsp.git
synced 2025-06-08 13:02:10 -05:00
sys: FspFsvolWriteCached: update file size prior to a cached write to ensure that the user-mode file system knows the correct file size
This commit is contained in:
parent
bd413b9c9b
commit
12463043e4
@ -389,6 +389,8 @@ PVOID FspAllocateIrpMustSucceed(CCHAR StackSize);
|
|||||||
BOOLEAN FspUnicodePathIsValid(PUNICODE_STRING Path, BOOLEAN AllowStreams);
|
BOOLEAN FspUnicodePathIsValid(PUNICODE_STRING Path, BOOLEAN AllowStreams);
|
||||||
VOID FspUnicodePathSuffix(PUNICODE_STRING Path, PUNICODE_STRING Remain, PUNICODE_STRING Suffix);
|
VOID FspUnicodePathSuffix(PUNICODE_STRING Path, PUNICODE_STRING Remain, PUNICODE_STRING Suffix);
|
||||||
NTSTATUS FspCreateGuid(GUID *Guid);
|
NTSTATUS FspCreateGuid(GUID *Guid);
|
||||||
|
NTSTATUS FspSendSetInformationIrp(PFILE_OBJECT FileObject,
|
||||||
|
FILE_INFORMATION_CLASS FileInformationClass, PVOID FileInformation, ULONG Length);
|
||||||
NTSTATUS FspLockUserBuffer(PVOID UserBuffer, ULONG Length,
|
NTSTATUS FspLockUserBuffer(PVOID UserBuffer, ULONG Length,
|
||||||
KPROCESSOR_MODE RequestorMode, LOCK_OPERATION Operation, PMDL *PMdl);
|
KPROCESSOR_MODE RequestorMode, LOCK_OPERATION Operation, PMDL *PMdl);
|
||||||
NTSTATUS FspMapLockedPagesInUserMode(PMDL Mdl, PVOID *PAddress);
|
NTSTATUS FspMapLockedPagesInUserMode(PMDL Mdl, PVOID *PAddress);
|
||||||
|
@ -9,6 +9,10 @@
|
|||||||
BOOLEAN FspUnicodePathIsValid(PUNICODE_STRING Path, BOOLEAN AllowStreams);
|
BOOLEAN FspUnicodePathIsValid(PUNICODE_STRING Path, BOOLEAN AllowStreams);
|
||||||
VOID FspUnicodePathSuffix(PUNICODE_STRING Path, PUNICODE_STRING Remain, PUNICODE_STRING Suffix);
|
VOID FspUnicodePathSuffix(PUNICODE_STRING Path, PUNICODE_STRING Remain, PUNICODE_STRING Suffix);
|
||||||
NTSTATUS FspCreateGuid(GUID *Guid);
|
NTSTATUS FspCreateGuid(GUID *Guid);
|
||||||
|
NTSTATUS FspSendSetInformationIrp(PFILE_OBJECT FileObject,
|
||||||
|
FILE_INFORMATION_CLASS FileInformationClass, PVOID FileInformation, ULONG Length);
|
||||||
|
static NTSTATUS FspSendSetInformationIrpCompletion(
|
||||||
|
PDEVICE_OBJECT DeviceObject, PIRP Irp, PVOID Context0);
|
||||||
NTSTATUS FspLockUserBuffer(PVOID UserBuffer, ULONG Length,
|
NTSTATUS FspLockUserBuffer(PVOID UserBuffer, ULONG Length,
|
||||||
KPROCESSOR_MODE RequestorMode, LOCK_OPERATION Operation, PMDL *PMdl);
|
KPROCESSOR_MODE RequestorMode, LOCK_OPERATION Operation, PMDL *PMdl);
|
||||||
NTSTATUS FspMapLockedPagesInUserMode(PMDL Mdl, PVOID *PAddress);
|
NTSTATUS FspMapLockedPagesInUserMode(PMDL Mdl, PVOID *PAddress);
|
||||||
@ -45,6 +49,7 @@ VOID FspSafeMdlDelete(FSP_SAFE_MDL *SafeMdl);
|
|||||||
#pragma alloc_text(PAGE, FspUnicodePathIsValid)
|
#pragma alloc_text(PAGE, FspUnicodePathIsValid)
|
||||||
#pragma alloc_text(PAGE, FspUnicodePathSuffix)
|
#pragma alloc_text(PAGE, FspUnicodePathSuffix)
|
||||||
#pragma alloc_text(PAGE, FspCreateGuid)
|
#pragma alloc_text(PAGE, FspCreateGuid)
|
||||||
|
#pragma alloc_text(PAGE, FspSendSetInformationIrp)
|
||||||
#pragma alloc_text(PAGE, FspLockUserBuffer)
|
#pragma alloc_text(PAGE, FspLockUserBuffer)
|
||||||
#pragma alloc_text(PAGE, FspMapLockedPagesInUserMode)
|
#pragma alloc_text(PAGE, FspMapLockedPagesInUserMode)
|
||||||
#pragma alloc_text(PAGE, FspCcInitializeCacheMap)
|
#pragma alloc_text(PAGE, FspCcInitializeCacheMap)
|
||||||
@ -185,6 +190,66 @@ NTSTATUS FspCreateGuid(GUID *Guid)
|
|||||||
return Result;
|
return Result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
typedef struct
|
||||||
|
{
|
||||||
|
IO_STATUS_BLOCK IoStatus;
|
||||||
|
KEVENT Event;
|
||||||
|
} FSP_SEND_SET_INFORMATION_IRP_CONTEXT;
|
||||||
|
|
||||||
|
NTSTATUS FspSendSetInformationIrp(PFILE_OBJECT FileObject,
|
||||||
|
FILE_INFORMATION_CLASS FileInformationClass, PVOID FileInformation, ULONG Length)
|
||||||
|
{
|
||||||
|
PAGED_CODE();
|
||||||
|
|
||||||
|
ASSERT(
|
||||||
|
FileAllocationInformation == FileInformationClass ||
|
||||||
|
FileEndOfFileInformation == FileInformationClass);
|
||||||
|
|
||||||
|
NTSTATUS Result;
|
||||||
|
PDEVICE_OBJECT DeviceObject;
|
||||||
|
PIRP Irp;
|
||||||
|
PIO_STACK_LOCATION IrpSp;
|
||||||
|
FSP_SEND_SET_INFORMATION_IRP_CONTEXT Context;
|
||||||
|
|
||||||
|
DeviceObject = IoGetRelatedDeviceObject(FileObject);
|
||||||
|
|
||||||
|
Irp = IoAllocateIrp(DeviceObject->StackSize, FALSE);
|
||||||
|
if (0 == Irp)
|
||||||
|
return STATUS_INSUFFICIENT_RESOURCES;
|
||||||
|
|
||||||
|
IrpSp = IoGetNextIrpStackLocation(Irp);
|
||||||
|
Irp->RequestorMode = KernelMode;
|
||||||
|
Irp->AssociatedIrp.SystemBuffer = FileInformation;
|
||||||
|
IrpSp->MajorFunction = IRP_MJ_SET_INFORMATION;
|
||||||
|
IrpSp->FileObject = FileObject;
|
||||||
|
IrpSp->Parameters.SetFile.FileInformationClass = FileInformationClass;
|
||||||
|
IrpSp->Parameters.SetFile.Length = FileInformationClass;
|
||||||
|
|
||||||
|
IoSetCompletionRoutine(Irp, FspSendSetInformationIrpCompletion, &Context, TRUE, TRUE, TRUE);
|
||||||
|
|
||||||
|
KeInitializeEvent(&Context.Event, NotificationEvent, FALSE);
|
||||||
|
Result = IoCallDriver(DeviceObject, Irp);
|
||||||
|
if (STATUS_PENDING == Result)
|
||||||
|
KeWaitForSingleObject(&Context.Event, Executive, KernelMode, FALSE, 0);
|
||||||
|
|
||||||
|
return NT_SUCCESS(Result) ? Context.IoStatus.Status : Result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static NTSTATUS FspSendSetInformationIrpCompletion(
|
||||||
|
PDEVICE_OBJECT DeviceObject, PIRP Irp, PVOID Context0)
|
||||||
|
{
|
||||||
|
// !PAGED_CODE();
|
||||||
|
|
||||||
|
FSP_SEND_SET_INFORMATION_IRP_CONTEXT *Context = Context0;
|
||||||
|
|
||||||
|
Context->IoStatus = Irp->IoStatus;
|
||||||
|
KeSetEvent(&Context->Event, 1, FALSE);
|
||||||
|
|
||||||
|
IoFreeIrp(Irp);
|
||||||
|
|
||||||
|
return STATUS_MORE_PROCESSING_REQUIRED;
|
||||||
|
}
|
||||||
|
|
||||||
NTSTATUS FspLockUserBuffer(PVOID UserBuffer, ULONG Length,
|
NTSTATUS FspLockUserBuffer(PVOID UserBuffer, ULONG Length,
|
||||||
KPROCESSOR_MODE RequestorMode, LOCK_OPERATION Operation, PMDL *PMdl)
|
KPROCESSOR_MODE RequestorMode, LOCK_OPERATION Operation, PMDL *PMdl)
|
||||||
{
|
{
|
||||||
|
@ -106,7 +106,7 @@ static NTSTATUS FspFsvolWriteCached(
|
|||||||
BOOLEAN SynchronousIo = BooleanFlagOn(FileObject->Flags, FO_SYNCHRONOUS_IO);
|
BOOLEAN SynchronousIo = BooleanFlagOn(FileObject->Flags, FO_SYNCHRONOUS_IO);
|
||||||
FSP_FSCTL_FILE_INFO FileInfo;
|
FSP_FSCTL_FILE_INFO FileInfo;
|
||||||
CC_FILE_SIZES FileSizes;
|
CC_FILE_SIZES FileSizes;
|
||||||
UINT64 OriginalFileSize;
|
FILE_END_OF_FILE_INFORMATION EndOfFileInformation;
|
||||||
UINT64 WriteEndOffset;
|
UINT64 WriteEndOffset;
|
||||||
BOOLEAN ExtendingFile;
|
BOOLEAN ExtendingFile;
|
||||||
BOOLEAN Success;
|
BOOLEAN Success;
|
||||||
@ -133,26 +133,18 @@ static NTSTATUS FspFsvolWriteCached(
|
|||||||
if (!Success)
|
if (!Success)
|
||||||
return FspWqRepostIrpWorkItem(Irp, FspFsvolWriteCached, 0);
|
return FspWqRepostIrpWorkItem(Irp, FspFsvolWriteCached, 0);
|
||||||
|
|
||||||
/* compute new file size and allocation size */
|
/* compute new file size */
|
||||||
ASSERT(FspTimeoutInfinity32 ==
|
ASSERT(FspTimeoutInfinity32 ==
|
||||||
FspFsvolDeviceExtension(FsvolDeviceObject)->VolumeParams.FileInfoTimeout);
|
FspFsvolDeviceExtension(FsvolDeviceObject)->VolumeParams.FileInfoTimeout);
|
||||||
FspFileNodeGetFileInfo(FileNode, &FileInfo);
|
FspFileNodeGetFileInfo(FileNode, &FileInfo);
|
||||||
OriginalFileSize = FileInfo.FileSize;
|
|
||||||
WriteEndOffset = WriteToEndOfFile ?
|
WriteEndOffset = WriteToEndOfFile ?
|
||||||
FileInfo.FileSize + WriteLength : WriteOffset.QuadPart + WriteLength;
|
FileInfo.FileSize + WriteLength : WriteOffset.QuadPart + WriteLength;
|
||||||
ExtendingFile = FileInfo.FileSize < WriteEndOffset;
|
ExtendingFile = FileInfo.FileSize < WriteEndOffset;
|
||||||
if (ExtendingFile)
|
if (ExtendingFile && !CanWait)
|
||||||
{
|
{
|
||||||
FileInfo.FileSize = WriteEndOffset;
|
/* need CanWait==TRUE for FspSendSetInformationIrp */
|
||||||
if (FileInfo.FileSize > FileInfo.AllocationSize)
|
FspFileNodeRelease(FileNode, Main);
|
||||||
{
|
return FspWqRepostIrpWorkItem(Irp, FspFsvolWriteCached, 0);
|
||||||
FSP_FSVOL_DEVICE_EXTENSION *FsvolDeviceExtension =
|
|
||||||
FspFsvolDeviceExtension(FsvolDeviceObject);
|
|
||||||
UINT64 AllocationUnit = FsvolDeviceExtension->VolumeParams.SectorSize *
|
|
||||||
FsvolDeviceExtension->VolumeParams.SectorsPerAllocationUnit;
|
|
||||||
FileInfo.AllocationSize = (FileInfo.FileSize + AllocationUnit - 1)
|
|
||||||
/ AllocationUnit * AllocationUnit;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* initialize cache if not already initialized! */
|
/* initialize cache if not already initialized! */
|
||||||
@ -170,14 +162,15 @@ static NTSTATUS FspFsvolWriteCached(
|
|||||||
return Result;
|
return Result;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (ExtendingFile)
|
|
||||||
{
|
|
||||||
FileSizes.AllocationSize.QuadPart = FileInfo.AllocationSize;
|
|
||||||
FileSizes.FileSize.QuadPart = FileInfo.FileSize;
|
|
||||||
FileSizes.ValidDataLength.QuadPart = MAXLONGLONG;
|
|
||||||
|
|
||||||
/* file is being extended */
|
/* are we extending the file? */
|
||||||
Result = FspCcSetFileSizes(FileObject, &FileSizes);
|
if (ExtendingFile)
|
||||||
|
{
|
||||||
|
ASSERT(CanWait);
|
||||||
|
|
||||||
|
EndOfFileInformation.EndOfFile.QuadPart = WriteEndOffset;
|
||||||
|
Result = FspSendSetInformationIrp(FileObject, FileEndOfFileInformation,
|
||||||
|
&EndOfFileInformation, sizeof EndOfFileInformation);
|
||||||
if (!NT_SUCCESS(Result))
|
if (!NT_SUCCESS(Result))
|
||||||
{
|
{
|
||||||
FspFileNodeRelease(FileNode, Main);
|
FspFileNodeRelease(FileNode, Main);
|
||||||
@ -223,18 +216,11 @@ static NTSTATUS FspFsvolWriteCached(
|
|||||||
if (SynchronousIo)
|
if (SynchronousIo)
|
||||||
FileObject->CurrentByteOffset.QuadPart = WriteEndOffset;
|
FileObject->CurrentByteOffset.QuadPart = WriteEndOffset;
|
||||||
|
|
||||||
if (ExtendingFile)
|
|
||||||
FspFileNodeSetFileInfo(FileNode, 0, &FileInfo);
|
|
||||||
|
|
||||||
FspFileNodeRelease(FileNode, Main);
|
FspFileNodeRelease(FileNode, Main);
|
||||||
|
|
||||||
return STATUS_SUCCESS;
|
return STATUS_SUCCESS;
|
||||||
|
|
||||||
cleanup:
|
cleanup:
|
||||||
/* pull back the cache file size if we extended it */
|
|
||||||
if (ExtendingFile)
|
|
||||||
CcGetFileSizePointer(FileObject)->QuadPart = OriginalFileSize;
|
|
||||||
|
|
||||||
FspFileNodeRelease(FileNode, Main);
|
FspFileNodeRelease(FileNode, Main);
|
||||||
|
|
||||||
if (STATUS_PENDING == Result)
|
if (STATUS_PENDING == Result)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user