mirror of
https://github.com/reactos/reactos.git
synced 2024-11-20 14:30:57 +00:00
692 lines
22 KiB
C
692 lines
22 KiB
C
/*
|
|
* ReactOS kernel
|
|
* Copyright (C) 2017 ReactOS Team
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*/
|
|
/*
|
|
* COPYRIGHT: See COPYING in the top level directory
|
|
* PROJECT: ReactOS kernel
|
|
* FILE: sdk/lib/drivers/copysup/copysup.c
|
|
* PURPOSE: CopySup library
|
|
* PROGRAMMER: Pierre Schweitzer (pierre@reactos.org)
|
|
*/
|
|
|
|
/* INCLUDES *****************************************************************/
|
|
|
|
#include "copysup.h"
|
|
#include <pseh/pseh2.h>
|
|
#define NDEBUG
|
|
#include <debug.h>
|
|
|
|
/* FUNCTIONS ****************************************************************/
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
BOOLEAN
|
|
FsRtlCopyRead2(
|
|
IN PFILE_OBJECT FileObject,
|
|
IN PLARGE_INTEGER FileOffset,
|
|
IN ULONG Length,
|
|
IN BOOLEAN Wait,
|
|
IN ULONG LockKey,
|
|
OUT PVOID Buffer,
|
|
OUT PIO_STATUS_BLOCK IoStatus,
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
IN PVOID TopLevelContext)
|
|
{
|
|
BOOLEAN Ret;
|
|
ULONG PageCount;
|
|
LARGE_INTEGER FinalOffset;
|
|
PFSRTL_COMMON_FCB_HEADER Fcb;
|
|
PFAST_IO_DISPATCH FastIoDispatch;
|
|
PDEVICE_OBJECT RelatedDeviceObject;
|
|
|
|
PAGED_CODE();
|
|
|
|
Ret = TRUE;
|
|
PageCount = ADDRESS_AND_SIZE_TO_SPAN_PAGES(FileOffset, Length);
|
|
|
|
/* Null-length read is always OK */
|
|
if (Length == 0)
|
|
{
|
|
IoStatus->Information = 0;
|
|
IoStatus->Status = STATUS_SUCCESS;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/* Check we don't overflow */
|
|
FinalOffset.QuadPart = FileOffset->QuadPart + Length;
|
|
if (FinalOffset.QuadPart <= 0)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
/* Get the FCB (at least, its header) */
|
|
Fcb = FileObject->FsContext;
|
|
|
|
FsRtlEnterFileSystem();
|
|
|
|
/* Acquire its resource (shared) */
|
|
if (Wait)
|
|
{
|
|
ExAcquireResourceSharedLite(Fcb->Resource, TRUE);
|
|
}
|
|
else
|
|
{
|
|
if (!ExAcquireResourceSharedLite(Fcb->Resource, FALSE))
|
|
{
|
|
Ret = FALSE;
|
|
goto CriticalSection;
|
|
}
|
|
}
|
|
|
|
/* If cache wasn't initialized, or FastIO isn't possible, fail */
|
|
if (FileObject->PrivateCacheMap == NULL || Fcb->IsFastIoPossible == FastIoIsNotPossible)
|
|
{
|
|
Ret = FALSE;
|
|
goto Resource;
|
|
}
|
|
|
|
/* If FastIO is questionable, then, question! */
|
|
if (Fcb->IsFastIoPossible == FastIoIsQuestionable)
|
|
{
|
|
RelatedDeviceObject = IoGetRelatedDeviceObject(FileObject);
|
|
FastIoDispatch = RelatedDeviceObject->DriverObject->FastIoDispatch;
|
|
ASSERT(FastIoDispatch != NULL);
|
|
ASSERT(FastIoDispatch->FastIoCheckIfPossible != NULL);
|
|
|
|
/* If it's not possible, then fail */
|
|
if (!FastIoDispatch->FastIoCheckIfPossible(FileObject, FileOffset, Length,
|
|
Wait, LockKey, TRUE, IoStatus, RelatedDeviceObject))
|
|
{
|
|
Ret = FALSE;
|
|
goto Resource;
|
|
}
|
|
}
|
|
|
|
/* If we get beyond file end... */
|
|
if (FinalOffset.QuadPart > Fcb->FileSize.QuadPart)
|
|
{
|
|
/* Fail if the offset was already beyond file end */
|
|
if (FileOffset->QuadPart >= Fcb->FileSize.QuadPart)
|
|
{
|
|
IoStatus->Information = 0;
|
|
IoStatus->Status = STATUS_END_OF_FILE;
|
|
goto Resource;
|
|
}
|
|
|
|
/* Otherwise, just fix read length */
|
|
Length = (ULONG)(Fcb->FileSize.QuadPart - FileOffset->QuadPart);
|
|
}
|
|
|
|
/* Set caller provided context as TLI */
|
|
IoSetTopLevelIrp(TopLevelContext);
|
|
|
|
_SEH2_TRY
|
|
{
|
|
/* If we cannot wait, or if file is bigger than 4GB */
|
|
if (!Wait || (FinalOffset.HighPart | Fcb->FileSize.HighPart) != 0)
|
|
{
|
|
/* Forward to Cc */
|
|
Ret = CcCopyRead(FileObject, FileOffset, Length, Wait, Buffer, IoStatus);
|
|
SetFlag(FileObject->Flags, FO_FILE_FAST_IO_READ);
|
|
|
|
/* Validate output */
|
|
ASSERT(!Ret || (IoStatus->Status == STATUS_END_OF_FILE) || (((ULONGLONG)FileOffset->QuadPart + IoStatus->Information) <= (ULONGLONG)Fcb->FileSize.QuadPart));
|
|
}
|
|
else
|
|
{
|
|
/* Forward to Cc */
|
|
CcFastCopyRead(FileObject, FileOffset->LowPart, Length, PageCount, Buffer, IoStatus);
|
|
SetFlag(FileObject->Flags, FO_FILE_FAST_IO_READ);
|
|
|
|
/* Validate output */
|
|
ASSERT((IoStatus->Status == STATUS_END_OF_FILE) || ((FileOffset->LowPart + IoStatus->Information) <= Fcb->FileSize.LowPart));
|
|
}
|
|
|
|
/* If read was successful, update the byte offset in the FO */
|
|
if (Ret)
|
|
{
|
|
FileObject->CurrentByteOffset.QuadPart = FileOffset->QuadPart + IoStatus->Information;
|
|
}
|
|
}
|
|
_SEH2_EXCEPT(FsRtlIsNtstatusExpected(_SEH2_GetExceptionCode()) ?
|
|
EXCEPTION_EXECUTE_HANDLER : EXCEPTION_CONTINUE_SEARCH)
|
|
{
|
|
Ret = FALSE;
|
|
}
|
|
_SEH2_END;
|
|
|
|
/* Reset TLI */
|
|
IoSetTopLevelIrp(NULL);
|
|
|
|
Resource:
|
|
ExReleaseResourceLite(Fcb->Resource);
|
|
CriticalSection:
|
|
FsRtlExitFileSystem();
|
|
|
|
return Ret;
|
|
}
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
BOOLEAN
|
|
FsRtlCopyWrite2(
|
|
IN PFILE_OBJECT FileObject,
|
|
IN PLARGE_INTEGER FileOffset,
|
|
IN ULONG Length,
|
|
IN BOOLEAN Wait,
|
|
IN ULONG LockKey,
|
|
IN PVOID Buffer,
|
|
OUT PIO_STATUS_BLOCK IoStatus,
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
IN PVOID TopLevelContext)
|
|
{
|
|
IO_STATUS_BLOCK LocalIoStatus;
|
|
PFSRTL_ADVANCED_FCB_HEADER Fcb;
|
|
BOOLEAN WriteToEof, AcquiredShared, FileSizeChanged, Ret;
|
|
LARGE_INTEGER WriteOffset, LastOffset, InitialFileSize, InitialValidDataLength;
|
|
|
|
PAGED_CODE();
|
|
|
|
/* First, check whether we're writing to EOF */
|
|
WriteToEof = ((FileOffset->LowPart == FILE_WRITE_TO_END_OF_FILE) &&
|
|
(FileOffset->HighPart == -1));
|
|
|
|
/* If Cc says we cannot write, fail now */
|
|
if (!CcCanIWrite(FileObject, Length, Wait, FALSE))
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
/* Write through means no cache */
|
|
if (BooleanFlagOn(FileObject->Flags, FO_WRITE_THROUGH))
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
/* If write is > 64Kb, don't use FastIO */
|
|
if (!CcCopyWriteWontFlush(FileObject, FileOffset, Length))
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
/* Initialize the IO_STATUS_BLOCK */
|
|
IoStatus->Status = STATUS_SUCCESS;
|
|
IoStatus->Information = Length;
|
|
|
|
/* No length, it's already written! */
|
|
if (Length == 0)
|
|
{
|
|
return TRUE;
|
|
}
|
|
|
|
AcquiredShared = FALSE;
|
|
FileSizeChanged = FALSE;
|
|
Fcb = FileObject->FsContext;
|
|
|
|
FsRtlEnterFileSystem();
|
|
|
|
/* If we cannot wait, or deal with files bigger then 4GB */
|
|
if (!Wait || (Fcb->AllocationSize.HighPart != 0))
|
|
{
|
|
/* If we're to extend the file, then, acquire exclusively */
|
|
if (WriteToEof || FileOffset->QuadPart + Length > Fcb->ValidDataLength.QuadPart)
|
|
{
|
|
if (!ExAcquireResourceExclusiveLite(Fcb->Resource, Wait))
|
|
{
|
|
FsRtlExitFileSystem();
|
|
return FALSE;
|
|
}
|
|
}
|
|
/* Otherwise, a shared lock is enough */
|
|
else
|
|
{
|
|
if (!ExAcquireResourceSharedLite(Fcb->Resource, Wait))
|
|
{
|
|
FsRtlExitFileSystem();
|
|
return FALSE;
|
|
}
|
|
|
|
AcquiredShared = TRUE;
|
|
}
|
|
|
|
/* Get first write offset, and last */
|
|
if (WriteToEof)
|
|
{
|
|
WriteOffset.QuadPart = Fcb->FileSize.QuadPart;
|
|
LastOffset.QuadPart = WriteOffset.QuadPart + Length;
|
|
}
|
|
else
|
|
{
|
|
WriteOffset.QuadPart = FileOffset->QuadPart;
|
|
LastOffset.QuadPart = WriteOffset.QuadPart + Length;
|
|
}
|
|
|
|
/* If cache wasn't initialized, fail */
|
|
if (FileObject->PrivateCacheMap == NULL ||
|
|
Fcb->IsFastIoPossible == FastIoIsNotPossible)
|
|
{
|
|
ExReleaseResourceLite(Fcb->Resource);
|
|
FsRtlExitFileSystem();
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
/* If we're to write beyond allocation size, it's no go,
|
|
* same is we create a hole bigger than 8kb
|
|
*/
|
|
if ((Fcb->ValidDataLength.QuadPart + 0x2000 <= WriteOffset.QuadPart) ||
|
|
(Length > MAXLONGLONG - WriteOffset.QuadPart) ||
|
|
(Fcb->AllocationSize.QuadPart < LastOffset.QuadPart))
|
|
{
|
|
ExReleaseResourceLite(Fcb->Resource);
|
|
FsRtlExitFileSystem();
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
/* If we have to extend the VDL, shared lock isn't enough */
|
|
if (AcquiredShared && LastOffset.QuadPart > Fcb->ValidDataLength.QuadPart)
|
|
{
|
|
/* So release, and attempt to acquire exclusively */
|
|
ExReleaseResourceLite(Fcb->Resource);
|
|
if (!ExAcquireResourceExclusiveLite(Fcb->Resource, Wait))
|
|
{
|
|
FsRtlExitFileSystem();
|
|
return FALSE;
|
|
}
|
|
|
|
/* Get again EOF, in case file size changed in between */
|
|
if (WriteToEof)
|
|
{
|
|
WriteOffset.QuadPart = Fcb->FileSize.QuadPart;
|
|
LastOffset.QuadPart = WriteOffset.QuadPart + Length;
|
|
}
|
|
|
|
/* Make sure caching is still enabled */
|
|
if (FileObject->PrivateCacheMap == NULL ||
|
|
Fcb->IsFastIoPossible == FastIoIsNotPossible)
|
|
{
|
|
ExReleaseResourceLite(Fcb->Resource);
|
|
FsRtlExitFileSystem();
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
/* And that we're not writting beyond allocation size */
|
|
if (Fcb->AllocationSize.QuadPart < LastOffset.QuadPart)
|
|
{
|
|
ExReleaseResourceLite(Fcb->Resource);
|
|
FsRtlExitFileSystem();
|
|
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
/* If FastIO is questionable, then question */
|
|
if (Fcb->IsFastIoPossible == FastIoIsQuestionable)
|
|
{
|
|
PFAST_IO_DISPATCH FastIoDispatch;
|
|
PDEVICE_OBJECT RelatedDeviceObject;
|
|
|
|
RelatedDeviceObject = IoGetRelatedDeviceObject(FileObject);
|
|
FastIoDispatch = RelatedDeviceObject->DriverObject->FastIoDispatch;
|
|
ASSERT(FastIoDispatch != NULL);
|
|
ASSERT(FastIoDispatch->FastIoCheckIfPossible != NULL);
|
|
|
|
if (!FastIoDispatch->FastIoCheckIfPossible(FileObject,
|
|
&WriteOffset,
|
|
Length, Wait, LockKey,
|
|
FALSE, &LocalIoStatus,
|
|
RelatedDeviceObject))
|
|
{
|
|
ExReleaseResourceLite(Fcb->Resource);
|
|
FsRtlExitFileSystem();
|
|
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
/* If we write beyond EOF, then, save previous sizes (in case of failure)
|
|
* and update file size, to allow writing
|
|
*/
|
|
if (LastOffset.QuadPart > Fcb->FileSize.QuadPart)
|
|
{
|
|
FileSizeChanged = TRUE;
|
|
InitialFileSize.QuadPart = Fcb->FileSize.QuadPart;
|
|
InitialValidDataLength.QuadPart = Fcb->ValidDataLength.QuadPart;
|
|
|
|
if (LastOffset.HighPart != Fcb->FileSize.HighPart &&
|
|
Fcb->PagingIoResource != NULL)
|
|
{
|
|
ExAcquireResourceExclusiveLite(Fcb->PagingIoResource, TRUE);
|
|
Fcb->FileSize.QuadPart = LastOffset.QuadPart;
|
|
ExReleaseResourceLite(Fcb->PagingIoResource);
|
|
}
|
|
else
|
|
{
|
|
Fcb->FileSize.QuadPart = LastOffset.QuadPart;
|
|
}
|
|
}
|
|
|
|
/* Set caller provided context as top level IRP */
|
|
IoSetTopLevelIrp(TopLevelContext);
|
|
|
|
Ret = TRUE;
|
|
|
|
/* And perform the writing */
|
|
_SEH2_TRY
|
|
{
|
|
/* Check whether we've to create a hole first */
|
|
if (LastOffset.QuadPart > Fcb->ValidDataLength.QuadPart)
|
|
{
|
|
Ret = CcZeroData(FileObject, &Fcb->ValidDataLength,
|
|
&WriteOffset, Wait);
|
|
}
|
|
|
|
/* If not needed, or if it worked, write data */
|
|
if (Ret)
|
|
{
|
|
Ret = CcCopyWrite(FileObject, &WriteOffset,
|
|
Length, Wait, Buffer);
|
|
}
|
|
}
|
|
_SEH2_EXCEPT(FsRtlIsNtstatusExpected(_SEH2_GetExceptionCode()) ?
|
|
EXCEPTION_EXECUTE_HANDLER :
|
|
EXCEPTION_CONTINUE_SEARCH)
|
|
{
|
|
Ret = FALSE;
|
|
}
|
|
_SEH2_END;
|
|
|
|
/* Restore top level IRP */
|
|
IoSetTopLevelIrp(NULL);
|
|
|
|
/* If writing succeed */
|
|
if (Ret)
|
|
{
|
|
/* If we wrote beyond VDL, update it */
|
|
if (LastOffset.QuadPart > Fcb->ValidDataLength.QuadPart)
|
|
{
|
|
if (LastOffset.HighPart != Fcb->ValidDataLength.HighPart &&
|
|
Fcb->PagingIoResource != NULL)
|
|
{
|
|
ExAcquireResourceExclusiveLite(Fcb->PagingIoResource, TRUE);
|
|
Fcb->ValidDataLength.QuadPart = LastOffset.QuadPart;
|
|
ExReleaseResourceLite(Fcb->PagingIoResource);
|
|
}
|
|
else
|
|
{
|
|
Fcb->ValidDataLength.QuadPart = LastOffset.QuadPart;
|
|
}
|
|
}
|
|
|
|
/* File was obviously modified */
|
|
SetFlag(FileObject->Flags, FO_FILE_MODIFIED);
|
|
|
|
/* And if we increased it, modify size in Cc and update FO */
|
|
if (FileSizeChanged)
|
|
{
|
|
(*CcGetFileSizePointer(FileObject)).QuadPart = LastOffset.QuadPart;
|
|
SetFlag(FileObject->Flags, FO_FILE_SIZE_CHANGED);
|
|
}
|
|
|
|
/* Update offset */
|
|
FileObject->CurrentByteOffset.QuadPart = WriteOffset.QuadPart + Length;
|
|
}
|
|
else
|
|
{
|
|
/* We failed, we need to restore previous sizes */
|
|
if (FileSizeChanged)
|
|
{
|
|
if (Fcb->PagingIoResource != NULL)
|
|
{
|
|
ExAcquireResourceExclusiveLite(Fcb->PagingIoResource, TRUE);
|
|
Fcb->FileSize.QuadPart = InitialFileSize.QuadPart;
|
|
Fcb->ValidDataLength.QuadPart = InitialValidDataLength.QuadPart;
|
|
ExReleaseResourceLite(Fcb->PagingIoResource);
|
|
}
|
|
else
|
|
{
|
|
Fcb->FileSize.QuadPart = InitialFileSize.QuadPart;
|
|
Fcb->ValidDataLength.QuadPart = InitialValidDataLength.QuadPart;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
BOOLEAN AboveFour;
|
|
|
|
WriteOffset.HighPart = 0;
|
|
LastOffset.HighPart = 0;
|
|
|
|
/* If we're to extend the file, then, acquire exclusively
|
|
* Here, easy stuff, we know we can wait, no return to check!
|
|
*/
|
|
if (WriteToEof || FileOffset->QuadPart + Length > Fcb->ValidDataLength.QuadPart)
|
|
{
|
|
ExAcquireResourceExclusiveLite(Fcb->Resource, TRUE);
|
|
}
|
|
/* Otherwise, a shared lock is enough */
|
|
else
|
|
{
|
|
ExAcquireResourceSharedLite(Fcb->Resource, TRUE);
|
|
AcquiredShared = TRUE;
|
|
}
|
|
|
|
/* Get first write offset, and last
|
|
* Also check whether our writing will bring us
|
|
* beyond the 4GB
|
|
*/
|
|
if (WriteToEof)
|
|
{
|
|
WriteOffset.LowPart = Fcb->FileSize.LowPart;
|
|
LastOffset.LowPart = WriteOffset.LowPart + Length;
|
|
AboveFour = (LastOffset.LowPart < Fcb->FileSize.LowPart);
|
|
}
|
|
else
|
|
{
|
|
WriteOffset.LowPart = FileOffset->LowPart;
|
|
LastOffset.LowPart = WriteOffset.LowPart + Length;
|
|
AboveFour = (LastOffset.LowPart < FileOffset->LowPart) ||
|
|
(FileOffset->HighPart != 0);
|
|
}
|
|
|
|
/* If cache wasn't initialized, fail */
|
|
if (FileObject->PrivateCacheMap == NULL ||
|
|
Fcb->IsFastIoPossible == FastIoIsNotPossible)
|
|
{
|
|
ExReleaseResourceLite(Fcb->Resource);
|
|
FsRtlExitFileSystem();
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
/* If we're to write beyond allocation size, it's no go,
|
|
* same is we create a hole bigger than 8kb
|
|
* same if we end writing beyond 4GB
|
|
*/
|
|
if ((Fcb->AllocationSize.LowPart < LastOffset.LowPart) ||
|
|
(WriteOffset.LowPart >= Fcb->ValidDataLength.LowPart + 0x2000) ||
|
|
AboveFour)
|
|
{
|
|
ExReleaseResourceLite(Fcb->Resource);
|
|
FsRtlExitFileSystem();
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
/* If we have to extend the VDL, shared lock isn't enough */
|
|
if (AcquiredShared && LastOffset.LowPart > Fcb->ValidDataLength.LowPart)
|
|
{
|
|
/* So release, and acquire exclusively */
|
|
ExReleaseResourceLite(Fcb->Resource);
|
|
ExAcquireResourceExclusiveLite(Fcb->Resource, TRUE);
|
|
|
|
/* Get again EOF, in case file size changed in between and
|
|
* recheck we won't go beyond 4GB
|
|
*/
|
|
if (WriteToEof)
|
|
{
|
|
WriteOffset.LowPart = Fcb->FileSize.LowPart;
|
|
LastOffset.LowPart = WriteOffset.LowPart + Length;
|
|
AboveFour = (LastOffset.LowPart < Fcb->FileSize.LowPart);
|
|
}
|
|
|
|
/* Make sure caching is still enabled */
|
|
if (FileObject->PrivateCacheMap == NULL ||
|
|
Fcb->IsFastIoPossible == FastIoIsNotPossible)
|
|
{
|
|
ExReleaseResourceLite(Fcb->Resource);
|
|
FsRtlExitFileSystem();
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
/* And that we're not writting beyond allocation size
|
|
* and that we're not going above 4GB
|
|
*/
|
|
if ((Fcb->AllocationSize.LowPart < LastOffset.LowPart) ||
|
|
(Fcb->AllocationSize.HighPart != 0) || AboveFour)
|
|
{
|
|
ExReleaseResourceLite(Fcb->Resource);
|
|
FsRtlExitFileSystem();
|
|
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
/* If FastIO is questionable, then question */
|
|
if (Fcb->IsFastIoPossible == FastIoIsQuestionable)
|
|
{
|
|
PFAST_IO_DISPATCH FastIoDispatch;
|
|
PDEVICE_OBJECT RelatedDeviceObject;
|
|
|
|
RelatedDeviceObject = IoGetRelatedDeviceObject(FileObject);
|
|
FastIoDispatch = RelatedDeviceObject->DriverObject->FastIoDispatch;
|
|
ASSERT(FastIoDispatch != NULL);
|
|
ASSERT(FastIoDispatch->FastIoCheckIfPossible != NULL);
|
|
|
|
if (!FastIoDispatch->FastIoCheckIfPossible(FileObject,
|
|
&WriteOffset,
|
|
Length, Wait, LockKey,
|
|
FALSE, &LocalIoStatus,
|
|
RelatedDeviceObject))
|
|
{
|
|
ExReleaseResourceLite(Fcb->Resource);
|
|
FsRtlExitFileSystem();
|
|
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
/* If we write beyond EOF, then, save previous sizes (in case of failure)
|
|
* and update file size, to allow writing
|
|
*/
|
|
if (LastOffset.LowPart > Fcb->FileSize.LowPart)
|
|
{
|
|
FileSizeChanged = TRUE;
|
|
InitialFileSize.LowPart = Fcb->FileSize.LowPart;
|
|
InitialValidDataLength.LowPart = Fcb->ValidDataLength.LowPart;
|
|
Fcb->FileSize.LowPart = LastOffset.LowPart;
|
|
}
|
|
|
|
/* Set caller provided context as top level IRP */
|
|
IoSetTopLevelIrp(TopLevelContext);
|
|
|
|
Ret = TRUE;
|
|
|
|
/* And perform the writing */
|
|
_SEH2_TRY
|
|
{
|
|
/* Check whether we've to create a hole first -
|
|
* it cannot fail, we can wait
|
|
*/
|
|
if (LastOffset.LowPart > Fcb->ValidDataLength.LowPart)
|
|
{
|
|
CcZeroData(FileObject, &Fcb->ValidDataLength, &WriteOffset, TRUE);
|
|
}
|
|
|
|
/* Write data */
|
|
CcFastCopyWrite(FileObject, WriteOffset.LowPart, Length, Buffer);
|
|
}
|
|
_SEH2_EXCEPT(FsRtlIsNtstatusExpected(_SEH2_GetExceptionCode()) ?
|
|
EXCEPTION_EXECUTE_HANDLER :
|
|
EXCEPTION_CONTINUE_SEARCH)
|
|
{
|
|
Ret = FALSE;
|
|
}
|
|
_SEH2_END;
|
|
|
|
/* Restore top level IRP */
|
|
IoSetTopLevelIrp(NULL);
|
|
|
|
/* If writing succeed */
|
|
if (Ret)
|
|
{
|
|
/* If we wrote beyond VDL, update it */
|
|
if (LastOffset.LowPart > Fcb->ValidDataLength.LowPart)
|
|
{
|
|
Fcb->ValidDataLength.LowPart = LastOffset.LowPart;
|
|
}
|
|
|
|
/* File was obviously modified */
|
|
SetFlag(FileObject->Flags, FO_FILE_MODIFIED);
|
|
|
|
/* And if we increased it, modify size in Cc and update FO */
|
|
if (FileSizeChanged)
|
|
{
|
|
(*CcGetFileSizePointer(FileObject)).LowPart = LastOffset.LowPart;
|
|
SetFlag(FileObject->Flags, FO_FILE_SIZE_CHANGED);
|
|
}
|
|
|
|
/* Update offset - we're still below 4GB, so high part must be 0 */
|
|
FileObject->CurrentByteOffset.LowPart = WriteOffset.LowPart + Length;
|
|
FileObject->CurrentByteOffset.HighPart = 0;
|
|
}
|
|
else
|
|
{
|
|
/* We failed, we need to restore previous sizes */
|
|
if (FileSizeChanged)
|
|
{
|
|
if (Fcb->PagingIoResource != NULL)
|
|
{
|
|
ExAcquireResourceExclusiveLite(Fcb->PagingIoResource, TRUE);
|
|
Fcb->FileSize.LowPart = InitialFileSize.LowPart;
|
|
Fcb->ValidDataLength.LowPart = InitialValidDataLength.LowPart;
|
|
ExReleaseResourceLite(Fcb->PagingIoResource);
|
|
}
|
|
else
|
|
{
|
|
Fcb->FileSize.LowPart = InitialFileSize.LowPart;
|
|
Fcb->ValidDataLength.LowPart = InitialValidDataLength.LowPart;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Release our resource and leave */
|
|
ExReleaseResourceLite(Fcb->Resource);
|
|
|
|
FsRtlExitFileSystem();
|
|
|
|
return Ret;
|
|
}
|