reactos/drivers/filesystems/fastfat_new/deviosup.c

3839 lines
103 KiB
C
Raw Normal View History

/*++
Copyright (c) 1989-2000 Microsoft Corporation
Module Name:
DevIoSup.c
Abstract:
This module implements the low lever disk read/write support for Fat.
--*/
#include "fatprocs.h"
//
// The Bug check file id for this module
//
#define BugCheckFileId (FAT_BUG_CHECK_DEVIOSUP)
//
// Local debug trace level
//
#define Dbg (DEBUG_TRACE_DEVIOSUP)
#define CollectDiskIoStats(VCB,FUNCTION,IS_USER_IO,COUNT) { \
PFILESYSTEM_STATISTICS Stats = &(VCB)->Statistics[KeGetCurrentProcessorNumber() % FatData.NumberProcessors].Common; \
if (IS_USER_IO) { \
if ((FUNCTION) == IRP_MJ_WRITE) { \
Stats->UserDiskWrites += (COUNT); \
} else { \
Stats->UserDiskReads += (COUNT); \
} \
} else { \
if ((FUNCTION) == IRP_MJ_WRITE) { \
Stats->MetaDataDiskWrites += (COUNT); \
} else { \
Stats->MetaDataDiskReads += (COUNT); \
} \
} \
}
typedef struct _FAT_SYNC_CONTEXT {
//
// Io status block for the request
//
IO_STATUS_BLOCK Iosb;
//
// Event to be signaled when the request completes
//
KEVENT Event;
} FAT_SYNC_CONTEXT, *PFAT_SYNC_CONTEXT;
//
// Completion Routine declarations
//
IO_COMPLETION_ROUTINE FatMultiSyncCompletionRoutine;
NTSTATUS
NTAPI
FatMultiSyncCompletionRoutine (
_In_ PDEVICE_OBJECT DeviceObject,
_In_ PIRP Irp,
_In_reads_opt_(_Inexpressible_("varies")) PVOID Contxt
);
IO_COMPLETION_ROUTINE FatMultiAsyncCompletionRoutine;
NTSTATUS
NTAPI
FatMultiAsyncCompletionRoutine (
_In_ PDEVICE_OBJECT DeviceObject,
_In_ PIRP Irp,
_In_reads_opt_(_Inexpressible_("varies")) PVOID Contxt
);
IO_COMPLETION_ROUTINE FatSpecialSyncCompletionRoutine;
NTSTATUS
NTAPI
FatSpecialSyncCompletionRoutine (
_In_ PDEVICE_OBJECT DeviceObject,
_In_ PIRP Irp,
_In_reads_opt_(_Inexpressible_("varies")) PVOID Contxt
);
IO_COMPLETION_ROUTINE FatSingleSyncCompletionRoutine;
NTSTATUS
NTAPI
FatSingleSyncCompletionRoutine (
_In_ PDEVICE_OBJECT DeviceObject,
_In_ PIRP Irp,
_In_reads_opt_(_Inexpressible_("varies")) PVOID Contxt
);
IO_COMPLETION_ROUTINE FatSingleAsyncCompletionRoutine;
NTSTATUS
NTAPI
FatSingleAsyncCompletionRoutine (
_In_ PDEVICE_OBJECT DeviceObject,
_In_ PIRP Irp,
_In_reads_opt_(_Inexpressible_("varies")) PVOID Contxt
);
IO_COMPLETION_ROUTINE FatPagingFileCompletionRoutine;
NTSTATUS
NTAPI
FatPagingFileCompletionRoutine (
_In_ PDEVICE_OBJECT DeviceObject,
_In_ PIRP Irp,
_In_reads_opt_(_Inexpressible_("varies")) PVOID MasterIrp
);
IO_COMPLETION_ROUTINE FatPagingFileCompletionRoutineCatch;
NTSTATUS
NTAPI
FatPagingFileCompletionRoutineCatch (
_In_ PDEVICE_OBJECT DeviceObject,
_In_ PIRP Irp,
_In_reads_opt_(_Inexpressible_("varies")) PVOID Contxt
);
VOID
FatSingleNonAlignedSync (
IN PIRP_CONTEXT IrpContext,
IN PVCB Vcb,
IN PUCHAR Buffer,
IN LBO Lbo,
IN ULONG ByteCount,
IN PIRP Irp
);
//
// The following macro decides whether to send a request directly to
// the device driver, or to other routines. It was meant to
// replace IoCallDriver as transparently as possible. It must only be
// called with a read or write Irp.
//
// NTSTATUS
// FatLowLevelReadWrite (
// PIRP_CONTEXT IrpContext,
// PDEVICE_OBJECT DeviceObject,
// PIRP Irp,
// PVCB Vcb
// );
//
#define FatLowLevelReadWrite(IRPCONTEXT,DO,IRP,VCB) ( \
IoCallDriver((DO),(IRP)) \
)
//
// The following macro handles completion-time zeroing of buffers.
//
#define FatDoCompletionZero( I, C ) \
if ((C)->ZeroMdl) { \
NT_ASSERT( (C)->ZeroMdl->MdlFlags & (MDL_MAPPED_TO_SYSTEM_VA | \
MDL_SOURCE_IS_NONPAGED_POOL));\
if (NT_SUCCESS((I)->IoStatus.Status)) { \
RtlZeroMemory( (C)->ZeroMdl->MappedSystemVa, \
(C)->ZeroMdl->ByteCount ); \
} \
IoFreeMdl((C)->ZeroMdl); \
(C)->ZeroMdl = NULL; \
}
#if (NTDDI_VERSION >= NTDDI_WIN8)
#define FatUpdateIOCountersPCW(IsAWrite,Count) \
FsRtlUpdateDiskCounters( ((IsAWrite) ? 0 : (Count) ), \
((IsAWrite) ? (Count) : 0) )
#else
#define FatUpdateIOCountersPCW(IsAWrite,Count)
#endif
#ifdef ALLOC_PRAGMA
#pragma alloc_text(PAGE, FatMultipleAsync)
#pragma alloc_text(PAGE, FatSingleAsync)
#pragma alloc_text(PAGE, FatSingleNonAlignedSync)
#pragma alloc_text(PAGE, FatWaitSync)
#pragma alloc_text(PAGE, FatLockUserBuffer)
#pragma alloc_text(PAGE, FatBufferUserBuffer)
#pragma alloc_text(PAGE, FatMapUserBuffer)
#pragma alloc_text(PAGE, FatNonCachedIo)
#pragma alloc_text(PAGE, FatNonCachedNonAlignedRead)
#pragma alloc_text(PAGE, FatPerformDevIoCtrl)
#endif
typedef struct FAT_PAGING_FILE_CONTEXT {
KEVENT Event;
PMDL RestoreMdl;
} FAT_PAGING_FILE_CONTEXT, *PFAT_PAGING_FILE_CONTEXT;
VOID
FatPagingFileIo (
IN PIRP Irp,
IN PFCB Fcb
)
/*++
Routine Description:
This routine performs the non-cached disk io described in its parameters.
This routine nevers blocks, and should only be used with the paging
file since no completion processing is performed.
Arguments:
Irp - Supplies the requesting Irp.
Fcb - Supplies the file to act on.
Return Value:
None.
--*/
{
//
// Declare some local variables for enumeration through the
// runs of the file.
//
VBO Vbo;
ULONG ByteCount;
PMDL Mdl;
LBO NextLbo;
VBO NextVbo = 0;
ULONG NextByteCount;
ULONG RemainingByteCount;
BOOLEAN MustSucceed;
ULONG FirstIndex;
ULONG CurrentIndex;
ULONG LastIndex;
LBO LastLbo;
ULONG LastByteCount;
BOOLEAN MdlIsReserve = FALSE;
BOOLEAN IrpIsMaster = FALSE;
FAT_PAGING_FILE_CONTEXT Context;
LONG IrpCount;
PIRP AssocIrp;
PIO_STACK_LOCATION IrpSp;
PIO_STACK_LOCATION NextIrpSp;
ULONG BufferOffset;
PDEVICE_OBJECT DeviceObject;
#ifndef __REACTOS__
BOOLEAN IsAWrite = FALSE;
#endif
DebugTrace(+1, Dbg, "FatPagingFileIo\n", 0);
DebugTrace( 0, Dbg, "Irp = %p\n", Irp );
DebugTrace( 0, Dbg, "Fcb = %p\n", Fcb );
NT_ASSERT( FlagOn( Fcb->FcbState, FCB_STATE_PAGING_FILE ));
//
// Initialize some locals.
//
BufferOffset = 0;
DeviceObject = Fcb->Vcb->TargetDeviceObject;
IrpSp = IoGetCurrentIrpStackLocation( Irp );
Vbo = IrpSp->Parameters.Read.ByteOffset.LowPart;
ByteCount = IrpSp->Parameters.Read.Length;
#ifndef __REACTOS__
IsAWrite = (IrpSp->MajorFunction == IRP_MJ_WRITE);
#endif
MustSucceed = FatLookupMcbEntry( Fcb->Vcb, &Fcb->Mcb,
Vbo,
&NextLbo,
&NextByteCount,
&FirstIndex);
//
// If this run isn't present, something is very wrong.
//
if (!MustSucceed) {
#ifdef _MSC_VER
#pragma prefast( suppress:28159, "things are seriously wrong if we get here" )
#endif
FatBugCheck( Vbo, ByteCount, 0 );
}
#if (NTDDI_VERSION >= NTDDI_WIN8)
//
// Charge the IO to paging file to current thread
//
if (FatDiskAccountingEnabled) {
PETHREAD ThreadIssuingIo = PsGetCurrentThread();
BOOLEAN IsWriteOperation = FALSE;
if (IrpSp->MajorFunction == IRP_MJ_WRITE) {
IsWriteOperation = TRUE;
}
PsUpdateDiskCounters( PsGetThreadProcess( ThreadIssuingIo ),
(IsWriteOperation ? 0 : ByteCount ), // bytes to read
(IsWriteOperation ? ByteCount : 0), // bytes to write
(IsWriteOperation ? 0 : 1), // # of reads
(IsWriteOperation ? 1 : 0), // # of writes
0 );
}
#endif
// See if the write covers a single valid run, and if so pass
// it on.
//
if ( NextByteCount >= ByteCount ) {
DebugTrace( 0, Dbg, "Passing Irp on to Disk Driver\n", 0 );
//
// Setup the next IRP stack location for the disk driver beneath us.
//
NextIrpSp = IoGetNextIrpStackLocation( Irp );
NextIrpSp->MajorFunction = IrpSp->MajorFunction;
NextIrpSp->Parameters.Read.Length = ByteCount;
NextIrpSp->Parameters.Read.ByteOffset.QuadPart = NextLbo;
//
// Since this is Paging file IO, we'll just ignore the verify bit.
//
SetFlag( NextIrpSp->Flags, SL_OVERRIDE_VERIFY_VOLUME );
//
// Set up the completion routine address in our stack frame.
// This is only invoked on error or cancel, and just copies
// the error Status into master irp's iosb.
//
// If the error implies a media problem, it also enqueues a
// worker item to write out the dirty bit so that the next
// time we run we will do a autochk /r
//
IoSetCompletionRoutine( Irp,
&FatPagingFileCompletionRoutine,
Irp,
FALSE,
TRUE,
TRUE );
//
// Issue the read/write request
//
// If IoCallDriver returns an error, it has completed the Irp
// and the error will be dealt with as a normal IO error.
//
(VOID)IoCallDriver( DeviceObject, Irp );
//
// We just issued an IO to the storage stack, update the counters indicating so.
//
if (FatDiskAccountingEnabled) {
FatUpdateIOCountersPCW( IsAWrite, ByteCount );
}
DebugTrace(-1, Dbg, "FatPagingFileIo -> VOID\n", 0);
return;
}
//
// Find out how may runs there are.
//
MustSucceed = FatLookupMcbEntry( Fcb->Vcb, &Fcb->Mcb,
Vbo + ByteCount - 1,
&LastLbo,
&LastByteCount,
&LastIndex);
//
// If this run isn't present, something is very wrong.
//
if (!MustSucceed) {
#ifdef _MSC_VER
#pragma prefast( suppress:28159, "things are seriously wrong if we get here" )
#endif
FatBugCheck( Vbo + ByteCount - 1, 1, 0 );
}
CurrentIndex = FirstIndex;
//
// Now set up the Irp->IoStatus. It will be modified by the
// multi-completion routine in case of error or verify required.
//
Irp->IoStatus.Status = STATUS_SUCCESS;
Irp->IoStatus.Information = ByteCount;
//
// Loop while there are still byte writes to satisfy. The way we'll work this
// is to hope for the best - one associated IRP per run, which will let us be
// completely async after launching all the IO.
//
// IrpCount will indicate the remaining number of associated Irps to launch.
//
// All we have to do is make sure IrpCount doesn't hit zero before we're building
// the very last Irp. If it is positive when we're done, it means we have to
// wait for the rest of the associated Irps to come back before we complete the
// master by hand.
//
// This will keep the master from completing early.
//
Irp->AssociatedIrp.IrpCount = IrpCount = LastIndex - FirstIndex + 1;
while (CurrentIndex <= LastIndex) {
//
// Reset this for unwinding purposes
//
AssocIrp = NULL;
//
// If next run is larger than we need, "ya get what ya need".
//
if (NextByteCount > ByteCount) {
NextByteCount = ByteCount;
}
RemainingByteCount = 0;
//
// Allocate and build a partial Mdl for the request.
//
Mdl = IoAllocateMdl( (PCHAR)Irp->UserBuffer + BufferOffset,
NextByteCount,
FALSE,
FALSE,
AssocIrp );
if (Mdl == NULL) {
//
// Pick up the reserve MDL
//
KeWaitForSingleObject( &FatReserveEvent, Executive, KernelMode, FALSE, NULL );
Mdl = FatReserveMdl;
MdlIsReserve = TRUE;
//
// Trim to fit the size of the reserve MDL.
//
if (NextByteCount > FAT_RESERVE_MDL_SIZE * PAGE_SIZE) {
RemainingByteCount = NextByteCount - FAT_RESERVE_MDL_SIZE * PAGE_SIZE;
NextByteCount = FAT_RESERVE_MDL_SIZE * PAGE_SIZE;
}
}
IoBuildPartialMdl( Irp->MdlAddress,
Mdl,
(PCHAR)Irp->UserBuffer + BufferOffset,
NextByteCount );
//
// Now that we have properly bounded this piece of the transfer, it is
// time to read/write it. We can simplify life slightly by always
// re-using the master IRP for cases where we use the reserve MDL,
// since we'll always be synchronous for those and can use a single
// completion context on our local stack.
//
// We also must prevent ourselves from issuing an associated IRP that would
// complete the master UNLESS this is the very last IRP we'll issue.
//
// This logic looks a bit complicated, but is hopefully understandable.
//
if (!MdlIsReserve &&
(IrpCount != 1 ||
(CurrentIndex == LastIndex &&
RemainingByteCount == 0))) {
AssocIrp = IoMakeAssociatedIrp( Irp, (CCHAR)(DeviceObject->StackSize + 1) );
}
if (AssocIrp == NULL) {
AssocIrp = Irp;
IrpIsMaster = TRUE;
//
// We need to drain the associated Irps so we can reliably figure out if
// the master Irp is showing a failed status, in which case we bail out
// immediately - as opposed to putting the value in the status field in
// jeopardy due to our re-use of the master Irp.
//
while (Irp->AssociatedIrp.IrpCount != IrpCount) {
KeDelayExecutionThread (KernelMode, FALSE, &Fat30Milliseconds);
}
//
// Note that since we failed to launch this associated Irp, that the completion
// code at the bottom will take care of completing the master Irp.
//
if (!NT_SUCCESS(Irp->IoStatus.Status)) {
NT_ASSERT( IrpCount );
break;
}
} else {
//
// Indicate we used an associated Irp.
//
IrpCount -= 1;
}
//
// With an associated IRP, we must take over the first stack location so
// we can have one to put the completion routine on. When re-using the
// master IRP, its already there.
//
if (!IrpIsMaster) {
//
// Get the first IRP stack location in the associated Irp
//
IoSetNextIrpStackLocation( AssocIrp );
NextIrpSp = IoGetCurrentIrpStackLocation( AssocIrp );
//
// Setup the Stack location to describe our read.
//
NextIrpSp->MajorFunction = IrpSp->MajorFunction;
NextIrpSp->Parameters.Read.Length = NextByteCount;
NextIrpSp->Parameters.Read.ByteOffset.QuadPart = Vbo;
//
// We also need the VolumeDeviceObject in the Irp stack in case
// we take the failure path.
//
NextIrpSp->DeviceObject = IrpSp->DeviceObject;
} else {
//
// Save the MDL in the IRP and prepare the stack
// context for the completion routine.
//
KeInitializeEvent( &Context.Event, SynchronizationEvent, FALSE );
Context.RestoreMdl = Irp->MdlAddress;
}
//
// And drop our Mdl into the Irp.
//
AssocIrp->MdlAddress = Mdl;
//
// Set up the completion routine address in our stack frame.
// For true associated IRPs, this is only invoked on error or
// cancel, and just copies the error Status into master irp's
// iosb.
//
// If the error implies a media problem, it also enqueues a
// worker item to write out the dirty bit so that the next
// time we run we will do a autochk /r
//
if (IrpIsMaster) {
IoSetCompletionRoutine( AssocIrp,
FatPagingFileCompletionRoutineCatch,
&Context,
TRUE,
TRUE,
TRUE );
} else {
IoSetCompletionRoutine( AssocIrp,
FatPagingFileCompletionRoutine,
Irp,
FALSE,
TRUE,
TRUE );
}
//
// Setup the next IRP stack location for the disk driver beneath us.
//
NextIrpSp = IoGetNextIrpStackLocation( AssocIrp );
//
// Since this is paging file IO, we'll just ignore the verify bit.
//
SetFlag( NextIrpSp->Flags, SL_OVERRIDE_VERIFY_VOLUME );
//
// Setup the Stack location to do a read from the disk driver.
//
NextIrpSp->MajorFunction = IrpSp->MajorFunction;
NextIrpSp->Parameters.Read.Length = NextByteCount;
NextIrpSp->Parameters.Read.ByteOffset.QuadPart = NextLbo;
(VOID)IoCallDriver( DeviceObject, AssocIrp );
//
// We just issued an IO to the storage stack, update the counters indicating so.
//
if (FatDiskAccountingEnabled) {
FatUpdateIOCountersPCW( IsAWrite, (ULONG64)NextByteCount );
}
//
// Wait for the Irp in the catch case and drop the flags.
//
if (IrpIsMaster) {
KeWaitForSingleObject( &Context.Event, Executive, KernelMode, FALSE, NULL );
IrpIsMaster = MdlIsReserve = FALSE;
//
// If the Irp is showing a failed status, there is no point in continuing.
// In doing so, we get to avoid squirreling away the failed status in case
// we were to re-use the master irp again.
//
// Note that since we re-used the master, we must not have issued the "last"
// associated Irp, and thus the completion code at the bottom will take care
// of that for us.
//
if (!NT_SUCCESS(Irp->IoStatus.Status)) {
NT_ASSERT( IrpCount );
break;
}
}
//
// Now adjust everything for the next pass through the loop.
//
Vbo += NextByteCount;
BufferOffset += NextByteCount;
ByteCount -= NextByteCount;
//
// Try to lookup the next run, if we are not done and we got
// all the way through the current run.
//
if (RemainingByteCount) {
//
// Advance the Lbo/Vbo if we have more to do in the current run.
//
NextLbo += NextByteCount;
NextVbo += NextByteCount;
NextByteCount = RemainingByteCount;
} else {
CurrentIndex += 1;
if ( CurrentIndex <= LastIndex ) {
NT_ASSERT( ByteCount != 0 );
FatGetNextMcbEntry( Fcb->Vcb, &Fcb->Mcb,
CurrentIndex,
&NextVbo,
&NextLbo,
&NextByteCount );
NT_ASSERT( NextVbo == Vbo );
}
}
} // while ( CurrentIndex <= LastIndex )
//
// If we didn't get enough associated Irps going to make this asynchronous, we
// twiddle our thumbs and wait for those we did launch to complete.
//
if (IrpCount) {
while (Irp->AssociatedIrp.IrpCount != IrpCount) {
KeDelayExecutionThread (KernelMode, FALSE, &Fat30Milliseconds);
}
IoCompleteRequest( Irp, IO_DISK_INCREMENT );
}
DebugTrace(-1, Dbg, "FatPagingFileIo -> VOID\n", 0);
return;
}
#if (NTDDI_VERSION >= NTDDI_WIN8)
VOID
FatUpdateDiskStats (
IN PIRP_CONTEXT IrpContext,
IN PIRP Irp,
IN ULONG ByteCount
)
/*++
Routine Description:
Charge appropriate process for the IO this IRP will cause.
Arguments:
IrpContext- The Irp Context
Irp - Supplies the requesting Irp.
ByteCount - The lengh of the operation.
Return Value:
None.
--*/
{
PETHREAD OriginatingThread = NULL;
ULONG NumReads = 0;
ULONG NumWrites = 0;
ULONGLONG BytesToRead = 0;
ULONGLONG BytesToWrite = 0;
//
// Here we attempt to charge the IO back to the originating process.
// - These checks are intended to cover following cases:
// o Buffered sync reads
// o Unbuffered sync read
// o Inline metadata reads
// o memory mapped reads (in-line faulting of data)
//
if (IrpContext->MajorFunction == IRP_MJ_READ) {
NumReads++;
BytesToRead = ByteCount;
if ((Irp->Tail.Overlay.Thread != NULL) &&
!IoIsSystemThread( Irp->Tail.Overlay.Thread )) {
OriginatingThread = Irp->Tail.Overlay.Thread;
} else if (!IoIsSystemThread( PsGetCurrentThread() )) {
OriginatingThread = PsGetCurrentThread();
//
// We couldn't find a non-system entity, so this should be charged to system.
// Do so only if we are top level.
// If we are not top-level then the read was initiated by someone like Cc (read ahead)
// who should have already accounted for this IO.
//
} else if (IoIsSystemThread( PsGetCurrentThread() ) &&
(IoGetTopLevelIrp() == Irp)) {
OriginatingThread = PsGetCurrentThread();
}
//
// Charge the write to Originating process.
// Intended to cover the following writes:
// - Unbuffered sync write
// - unbuffered async write
//
// If we re not top-level, then it should already have been accounted for
// somewhere else (Cc).
//
} else if (IrpContext->MajorFunction == IRP_MJ_WRITE) {
NumWrites++;
BytesToWrite = ByteCount;
if (IoGetTopLevelIrp() == Irp) {
if ((Irp->Tail.Overlay.Thread != NULL) &&
!IoIsSystemThread( Irp->Tail.Overlay.Thread )) {
OriginatingThread = Irp->Tail.Overlay.Thread;
} else {
OriginatingThread = PsGetCurrentThread();
}
//
// For mapped page writes
//
} else if (IoGetTopLevelIrp() == (PIRP)FSRTL_MOD_WRITE_TOP_LEVEL_IRP) {
OriginatingThread = PsGetCurrentThread();
}
}
if (OriginatingThread != NULL) {
PsUpdateDiskCounters( PsGetThreadProcess( OriginatingThread ),
BytesToRead,
BytesToWrite,
NumReads,
NumWrites,
0 );
}
}
#endif
_Requires_lock_held_(_Global_critical_region_)
NTSTATUS
FatNonCachedIo (
IN PIRP_CONTEXT IrpContext,
IN PIRP Irp,
IN PFCB FcbOrDcb,
IN ULONG StartingVbo,
IN ULONG ByteCount,
IN ULONG UserByteCount,
IN ULONG StreamFlags
)
/*++
Routine Description:
This routine performs the non-cached disk io described in its parameters.
The choice of a single run is made if possible, otherwise multiple runs
are executed.
Arguments:
IrpContext->MajorFunction - Supplies either IRP_MJ_READ or IRP_MJ_WRITE.
Irp - Supplies the requesting Irp.
FcbOrDcb - Supplies the file to act on.
StartingVbo - The starting point for the operation.
ByteCount - The lengh of the operation.
UserByteCount - The last byte the user can see, rest to be zeroed.
StreamFlags - flag to indicate special attributes for a NonCachedIo.
Return Value:
None.
--*/
{
//
// Declare some local variables for enumeration through the
// runs of the file, and an array to store parameters for
// parallel I/Os
//
BOOLEAN Wait;
LBO NextLbo;
VBO NextVbo;
ULONG NextByteCount;
BOOLEAN NextIsAllocated;
LBO LastLbo;
ULONG LastByteCount;
BOOLEAN LastIsAllocated;
BOOLEAN EndOnMax;
ULONG FirstIndex;
ULONG CurrentIndex;
ULONG LastIndex;
ULONG NextRun;
ULONG BufferOffset;
ULONG OriginalByteCount;
IO_RUN StackIoRuns[FAT_MAX_IO_RUNS_ON_STACK];
PIO_RUN IoRuns;
PAGED_CODE();
UNREFERENCED_PARAMETER( StreamFlags );
DebugTrace(+1, Dbg, "FatNonCachedIo\n", 0);
DebugTrace( 0, Dbg, "Irp = %p\n", Irp );
DebugTrace( 0, Dbg, "MajorFunction = %08lx\n", IrpContext->MajorFunction );
DebugTrace( 0, Dbg, "FcbOrDcb = %p\n", FcbOrDcb );
DebugTrace( 0, Dbg, "StartingVbo = %08lx\n", StartingVbo );
DebugTrace( 0, Dbg, "ByteCount = %08lx\n", ByteCount );
if (!FlagOn(Irp->Flags, IRP_PAGING_IO)) {
PFILE_SYSTEM_STATISTICS Stats =
&FcbOrDcb->Vcb->Statistics[KeGetCurrentProcessorNumber() % FatData.NumberProcessors];
if (IrpContext->MajorFunction == IRP_MJ_READ) {
Stats->Fat.NonCachedReads += 1;
Stats->Fat.NonCachedReadBytes += ByteCount;
} else {
Stats->Fat.NonCachedWrites += 1;
Stats->Fat.NonCachedWriteBytes += ByteCount;
}
}
//
// Initialize some locals.
//
NextRun = 0;
BufferOffset = 0;
OriginalByteCount = ByteCount;
Wait = BooleanFlagOn(IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT);
#if (NTDDI_VERSION >= NTDDI_WIN8)
//
// Disk IO accounting
//
if (FatDiskAccountingEnabled) {
FatUpdateDiskStats( IrpContext,
Irp,
ByteCount );
}
#endif
//
// For nonbuffered I/O, we need the buffer locked in all
// cases.
//
// This call may raise. If this call succeeds and a subsequent
// condition is raised, the buffers are unlocked automatically
// by the I/O system when the request is completed, via the
// Irp->MdlAddress field.
//
FatLockUserBuffer( IrpContext,
Irp,
(IrpContext->MajorFunction == IRP_MJ_READ) ?
IoWriteAccess : IoReadAccess,
ByteCount );
//
// No zeroing for trailing sectors if requested.
// Otherwise setup the required zeroing for read requests.
//
if (UserByteCount != ByteCount) {
PMDL Mdl;
NT_ASSERT( ByteCount > UserByteCount );
_Analysis_assume_(ByteCount > UserByteCount);
Mdl = IoAllocateMdl( (PUCHAR) Irp->UserBuffer + UserByteCount,
ByteCount - UserByteCount,
FALSE,
FALSE,
NULL );
if (Mdl == NULL) {
FatRaiseStatus( IrpContext, STATUS_INSUFFICIENT_RESOURCES );
}
IoBuildPartialMdl( Irp->MdlAddress,
Mdl,
(PUCHAR) Irp->UserBuffer + UserByteCount,
ByteCount - UserByteCount );
IrpContext->FatIoContext->ZeroMdl = Mdl;
//
// Map the MDL now so we can't fail at IO completion time. Note
// that this will be only a single page.
//
if (MmGetSystemAddressForMdlSafe( Mdl, NormalPagePriority | MdlMappingNoExecute ) == NULL) {
FatRaiseStatus( IrpContext, STATUS_INSUFFICIENT_RESOURCES );
}
}
//
// Try to lookup the first run. If there is just a single run,
// we may just be able to pass it on.
//
FatLookupFileAllocation( IrpContext,
FcbOrDcb,
StartingVbo,
&NextLbo,
&NextByteCount,
&NextIsAllocated,
&EndOnMax,
&FirstIndex );
//
// We just added the allocation, thus there must be at least
// one entry in the mcb corresponding to our write, ie.
// NextIsAllocated must be true. If not, the pre-existing file
// must have an allocation error.
//
if ( !NextIsAllocated ) {
FatPopUpFileCorrupt( IrpContext, FcbOrDcb );
FatRaiseStatus( IrpContext, STATUS_FILE_CORRUPT_ERROR );
}
NT_ASSERT( NextByteCount != 0 );
//
// If the request was not aligned correctly, read in the first
// part first.
//
//
// See if the write covers a single valid run, and if so pass
// it on. We must bias this by the byte that is lost at the
// end of the maximal file.
//
if ( NextByteCount >= ByteCount - (EndOnMax ? 1 : 0)) {
if (FlagOn(Irp->Flags, IRP_PAGING_IO)) {
CollectDiskIoStats(FcbOrDcb->Vcb, IrpContext->MajorFunction,
FlagOn(IrpContext->Flags, IRP_CONTEXT_FLAG_USER_IO), 1);
} else {
PFILE_SYSTEM_STATISTICS Stats =
&FcbOrDcb->Vcb->Statistics[KeGetCurrentProcessorNumber() % FatData.NumberProcessors];
if (IrpContext->MajorFunction == IRP_MJ_READ) {
Stats->Fat.NonCachedDiskReads += 1;
} else {
Stats->Fat.NonCachedDiskWrites += 1;
}
}
DebugTrace( 0, Dbg, "Passing 1 Irp on to Disk Driver\n", 0 );
FatSingleAsync( IrpContext,
FcbOrDcb->Vcb,
NextLbo,
ByteCount,
Irp );
} else {
//
// If there we can't wait, and there are more runs than we can handle,
// we will have to post this request.
//
FatLookupFileAllocation( IrpContext,
FcbOrDcb,
StartingVbo + ByteCount - 1,
&LastLbo,
&LastByteCount,
&LastIsAllocated,
&EndOnMax,
&LastIndex );
//
// Since we already added the allocation for the whole
// write, assert that we find runs until ByteCount == 0
// Otherwise this file is corrupt.
//
if ( !LastIsAllocated ) {
FatPopUpFileCorrupt( IrpContext, FcbOrDcb );
FatRaiseStatus( IrpContext, STATUS_FILE_CORRUPT_ERROR );
}
if (LastIndex - FirstIndex + 1 > FAT_MAX_IO_RUNS_ON_STACK) {
IoRuns = FsRtlAllocatePoolWithTag( PagedPool,
(LastIndex - FirstIndex + 1) * sizeof(IO_RUN),
TAG_IO_RUNS );
} else {
IoRuns = StackIoRuns;
}
NT_ASSERT( LastIndex != FirstIndex );
CurrentIndex = FirstIndex;
//
// Loop while there are still byte writes to satisfy.
//
while (CurrentIndex <= LastIndex) {
NT_ASSERT( NextByteCount != 0);
NT_ASSERT( ByteCount != 0);
//
// If next run is larger than we need, "ya get what you need".
//
if (NextByteCount > ByteCount) {
NextByteCount = ByteCount;
}
//
// Now that we have properly bounded this piece of the
// transfer, it is time to write it.
//
// We remember each piece of a parallel run by saving the
// essential information in the IoRuns array. The tranfers
// are started up in parallel below.
//
IoRuns[NextRun].Vbo = StartingVbo;
IoRuns[NextRun].Lbo = NextLbo;
IoRuns[NextRun].Offset = BufferOffset;
IoRuns[NextRun].ByteCount = NextByteCount;
NextRun += 1;
//
// Now adjust everything for the next pass through the loop.
//
StartingVbo += NextByteCount;
BufferOffset += NextByteCount;
ByteCount -= NextByteCount;
//
// Try to lookup the next run (if we are not done).
//
CurrentIndex += 1;
if ( CurrentIndex <= LastIndex ) {
NT_ASSERT( ByteCount != 0 );
FatGetNextMcbEntry( FcbOrDcb->Vcb, &FcbOrDcb->Mcb,
CurrentIndex,
&NextVbo,
&NextLbo,
&NextByteCount );
NT_ASSERT(NextVbo == StartingVbo);
}
} // while ( CurrentIndex <= LastIndex )
//
// Now set up the Irp->IoStatus. It will be modified by the
// multi-completion routine in case of error or verify required.
//
Irp->IoStatus.Status = STATUS_SUCCESS;
Irp->IoStatus.Information = OriginalByteCount;
if (FlagOn(Irp->Flags, IRP_PAGING_IO)) {
CollectDiskIoStats(FcbOrDcb->Vcb, IrpContext->MajorFunction,
FlagOn(IrpContext->Flags, IRP_CONTEXT_FLAG_USER_IO), NextRun);
}
//
// OK, now do the I/O.
//
_SEH2_TRY {
DebugTrace( 0, Dbg, "Passing Multiple Irps on to Disk Driver\n", 0 );
FatMultipleAsync( IrpContext,
FcbOrDcb->Vcb,
Irp,
NextRun,
IoRuns );
} _SEH2_FINALLY {
if (IoRuns != StackIoRuns) {
ExFreePool( IoRuns );
}
} _SEH2_END;
}
if (!Wait) {
DebugTrace(-1, Dbg, "FatNonCachedIo -> STATUS_PENDING\n", 0);
return STATUS_PENDING;
}
FatWaitSync( IrpContext );
DebugTrace(-1, Dbg, "FatNonCachedIo -> 0x%08lx\n", Irp->IoStatus.Status);
return Irp->IoStatus.Status;
}
_Requires_lock_held_(_Global_critical_region_)
VOID
FatNonCachedNonAlignedRead (
IN PIRP_CONTEXT IrpContext,
IN PIRP Irp,
IN PFCB FcbOrDcb,
IN ULONG StartingVbo,
IN ULONG ByteCount
)
/*++
Routine Description:
This routine performs the non-cached disk io described in its parameters.
This routine differs from the above in that the range does not have to be
sector aligned. This accomplished with the use of intermediate buffers.
Arguments:
IrpContext->MajorFunction - Supplies either IRP_MJ_READ or IRP_MJ_WRITE.
Irp - Supplies the requesting Irp.
FcbOrDcb - Supplies the file to act on.
StartingVbo - The starting point for the operation.
ByteCount - The lengh of the operation.
Return Value:
None.
--*/
{
//
// Declare some local variables for enumeration through the
// runs of the file, and an array to store parameters for
// parallel I/Os
//
LBO NextLbo;
ULONG NextByteCount;
BOOLEAN NextIsAllocated;
ULONG SectorSize;
ULONG BytesToCopy;
ULONG OriginalByteCount;
ULONG OriginalStartingVbo;
BOOLEAN EndOnMax;
PUCHAR UserBuffer;
PUCHAR DiskBuffer = NULL;
PMDL Mdl;
PMDL SavedMdl;
PVOID SavedUserBuffer;
PAGED_CODE();
DebugTrace(+1, Dbg, "FatNonCachedNonAlignedRead\n", 0);
DebugTrace( 0, Dbg, "Irp = %p\n", Irp );
DebugTrace( 0, Dbg, "MajorFunction = %08lx\n", IrpContext->MajorFunction );
DebugTrace( 0, Dbg, "FcbOrDcb = %p\n", FcbOrDcb );
DebugTrace( 0, Dbg, "StartingVbo = %08lx\n", StartingVbo );
DebugTrace( 0, Dbg, "ByteCount = %08lx\n", ByteCount );
//
// Initialize some locals.
//
OriginalByteCount = ByteCount;
OriginalStartingVbo = StartingVbo;
SectorSize = FcbOrDcb->Vcb->Bpb.BytesPerSector;
NT_ASSERT( FlagOn(IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT) );
//
// For nonbuffered I/O, we need the buffer locked in all
// cases.
//
// This call may raise. If this call succeeds and a subsequent
// condition is raised, the buffers are unlocked automatically
// by the I/O system when the request is completed, via the
// Irp->MdlAddress field.
//
FatLockUserBuffer( IrpContext,
Irp,
IoWriteAccess,
ByteCount );
UserBuffer = FatMapUserBuffer( IrpContext, Irp );
//
// Allocate the local buffer
//
DiskBuffer = FsRtlAllocatePoolWithTag( NonPagedPoolNxCacheAligned,
(ULONG) ROUND_TO_PAGES( SectorSize ),
TAG_IO_BUFFER );
//
// We use a try block here to ensure the buffer is freed, and to
// fill in the correct byte count in the Iosb.Information field.
//
_SEH2_TRY {
//
// If the beginning of the request was not aligned correctly, read in
// the first part first.
//
if ( StartingVbo & (SectorSize - 1) ) {
VBO Hole;
//
// Try to lookup the first run.
//
FatLookupFileAllocation( IrpContext,
FcbOrDcb,
StartingVbo,
&NextLbo,
&NextByteCount,
&NextIsAllocated,
&EndOnMax,
NULL );
//
// We just added the allocation, thus there must be at least
// one entry in the mcb corresponding to our write, ie.
// NextIsAllocated must be true. If not, the pre-existing file
// must have an allocation error.
//
if ( !NextIsAllocated ) {
FatPopUpFileCorrupt( IrpContext, FcbOrDcb );
FatRaiseStatus( IrpContext, STATUS_FILE_CORRUPT_ERROR );
}
FatSingleNonAlignedSync( IrpContext,
FcbOrDcb->Vcb,
DiskBuffer,
NextLbo & ~((LONG)SectorSize - 1),
SectorSize,
Irp );
if (!NT_SUCCESS( Irp->IoStatus.Status )) {
try_return( NOTHING );
}
//
// Now copy the part of the first sector that we want to the user
// buffer.
//
Hole = StartingVbo & (SectorSize - 1);
BytesToCopy = ByteCount >= SectorSize - Hole ?
SectorSize - Hole : ByteCount;
RtlCopyMemory( UserBuffer, DiskBuffer + Hole, BytesToCopy );
StartingVbo += BytesToCopy;
ByteCount -= BytesToCopy;
if ( ByteCount == 0 ) {
try_return( NOTHING );
}
}
NT_ASSERT( (StartingVbo & (SectorSize - 1)) == 0 );
//
// If there is a tail part that is not sector aligned, read it.
//
if ( ByteCount & (SectorSize - 1) ) {
VBO LastSectorVbo;
LastSectorVbo = StartingVbo + (ByteCount & ~(SectorSize - 1));
//
// Try to lookup the last part of the requested range.
//
FatLookupFileAllocation( IrpContext,
FcbOrDcb,
LastSectorVbo,
&NextLbo,
&NextByteCount,
&NextIsAllocated,
&EndOnMax,
NULL );
//
// We just added the allocation, thus there must be at least
// one entry in the mcb corresponding to our write, ie.
// NextIsAllocated must be true. If not, the pre-existing file
// must have an allocation error.
//
if ( !NextIsAllocated ) {
FatPopUpFileCorrupt( IrpContext, FcbOrDcb );
FatRaiseStatus( IrpContext, STATUS_FILE_CORRUPT_ERROR );
}
FatSingleNonAlignedSync( IrpContext,
FcbOrDcb->Vcb,
DiskBuffer,
NextLbo,
SectorSize,
Irp );
if (!NT_SUCCESS( Irp->IoStatus.Status )) {
try_return( NOTHING );
}
//
// Now copy over the part of this last sector that we need.
//
BytesToCopy = ByteCount & (SectorSize - 1);
UserBuffer += LastSectorVbo - OriginalStartingVbo;
RtlCopyMemory( UserBuffer, DiskBuffer, BytesToCopy );
ByteCount -= BytesToCopy;
if ( ByteCount == 0 ) {
try_return( NOTHING );
}
}
NT_ASSERT( ((StartingVbo | ByteCount) & (SectorSize - 1)) == 0 );
//
// Now build a Mdl describing the sector aligned balance of the transfer,
// and put it in the Irp, and read that part.
//
SavedMdl = Irp->MdlAddress;
Irp->MdlAddress = NULL;
SavedUserBuffer = Irp->UserBuffer;
Irp->UserBuffer = (PUCHAR)MmGetMdlVirtualAddress( SavedMdl ) +
(StartingVbo - OriginalStartingVbo);
Mdl = IoAllocateMdl( Irp->UserBuffer,
ByteCount,
FALSE,
FALSE,
Irp );
if (Mdl == NULL) {
Irp->MdlAddress = SavedMdl;
Irp->UserBuffer = SavedUserBuffer;
FatRaiseStatus( IrpContext, STATUS_INSUFFICIENT_RESOURCES );
}
IoBuildPartialMdl( SavedMdl,
Mdl,
Irp->UserBuffer,
ByteCount );
//
// Try to read in the pages.
//
_SEH2_TRY {
FatNonCachedIo( IrpContext,
Irp,
FcbOrDcb,
StartingVbo,
ByteCount,
ByteCount,
0 );
} _SEH2_FINALLY {
IoFreeMdl( Irp->MdlAddress );
Irp->MdlAddress = SavedMdl;
Irp->UserBuffer = SavedUserBuffer;
} _SEH2_END;
try_exit: NOTHING;
} _SEH2_FINALLY {
ExFreePool( DiskBuffer );
if ( !_SEH2_AbnormalTermination() && NT_SUCCESS(Irp->IoStatus.Status) ) {
Irp->IoStatus.Information = OriginalByteCount;
//
// We now flush the user's buffer to memory.
//
KeFlushIoBuffers( Irp->MdlAddress, TRUE, FALSE );
}
} _SEH2_END;
DebugTrace(-1, Dbg, "FatNonCachedNonAlignedRead -> VOID\n", 0);
return;
}
VOID
FatMultipleAsync (
IN PIRP_CONTEXT IrpContext,
IN PVCB Vcb,
IN PIRP MasterIrp,
IN ULONG MultipleIrpCount,
IN PIO_RUN IoRuns
)
/*++
Routine Description:
This routine first does the initial setup required of a Master IRP that is
going to be completed using associated IRPs. This routine should not
be used if only one async request is needed, instead the single read/write
async routines should be called.
A context parameter is initialized, to serve as a communications area
between here and the common completion routine. This initialization
includes allocation of a spinlock. The spinlock is deallocated in the
FatWaitSync routine, so it is essential that the caller insure that
this routine is always called under all circumstances following a call
to this routine.
Next this routine reads or writes one or more contiguous sectors from
a device asynchronously, and is used if there are multiple reads for a
master IRP. A completion routine is used to synchronize with the
completion of all of the I/O requests started by calls to this routine.
Also, prior to calling this routine the caller must initialize the
IoStatus field in the Context, with the correct success status and byte
count which are expected if all of the parallel transfers complete
successfully. After return this status will be unchanged if all requests
were, in fact, successful. However, if one or more errors occur, the
IoStatus will be modified to reflect the error status and byte count
from the first run (by Vbo) which encountered an error. I/O status
from all subsequent runs will not be indicated.
Arguments:
IrpContext->MajorFunction - Supplies either IRP_MJ_READ or IRP_MJ_WRITE.
Vcb - Supplies the device to be read
MasterIrp - Supplies the master Irp.
MulitpleIrpCount - Supplies the number of multiple async requests
that will be issued against the master irp.
IoRuns - Supplies an array containing the Vbo, Lbo, BufferOffset, and
ByteCount for all the runs to executed in parallel.
Return Value:
None.
--*/
{
PIRP Irp;
PIO_STACK_LOCATION IrpSp;
PMDL Mdl;
BOOLEAN Wait;
PFAT_IO_CONTEXT Context;
#ifndef __REACTOS__
BOOLEAN IsAWrite = FALSE;
ULONG Length = 0;
#endif
ULONG UnwindRunCount = 0;
BOOLEAN ExceptionExpected = TRUE;
PAGED_CODE();
DebugTrace(+1, Dbg, "FatMultipleAsync\n", 0);
DebugTrace( 0, Dbg, "MajorFunction = %08lx\n", IrpContext->MajorFunction );
DebugTrace( 0, Dbg, "Vcb = %p\n", Vcb );
DebugTrace( 0, Dbg, "MasterIrp = %p\n", MasterIrp );
DebugTrace( 0, Dbg, "MultipleIrpCount = %08lx\n", MultipleIrpCount );
DebugTrace( 0, Dbg, "IoRuns = %08lx\n", IoRuns );
//
// If this I/O originating during FatVerifyVolume, bypass the
// verify logic.
//
if (Vcb->VerifyThread == KeGetCurrentThread()) {
SetFlag( IrpContext->Flags, IRP_CONTEXT_FLAG_OVERRIDE_VERIFY );
}
//
// Set up things according to whether this is truely async.
//
Wait = BooleanFlagOn( IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT );
Context = IrpContext->FatIoContext;
//
// Finish initializing Context, for use in Read/Write Multiple Asynch.
//
Context->MasterIrp = MasterIrp;
IrpSp = IoGetCurrentIrpStackLocation( MasterIrp );
#ifndef __REACTOS__
IsAWrite = (IrpSp->MajorFunction == IRP_MJ_WRITE);
Length = IrpSp->Parameters.Read.Length;
#endif
_SEH2_TRY {
//
// Itterate through the runs, doing everything that can fail
//
for ( UnwindRunCount = 0;
UnwindRunCount < MultipleIrpCount;
UnwindRunCount++ ) {
//
// Create an associated IRP, making sure there is one stack entry for
// us, as well.
//
IoRuns[UnwindRunCount].SavedIrp = 0;
Irp = IoMakeAssociatedIrp( MasterIrp,
(CCHAR)(Vcb->TargetDeviceObject->StackSize + 1) );
if (Irp == NULL) {
FatRaiseStatus( IrpContext, STATUS_INSUFFICIENT_RESOURCES );
}
IoRuns[UnwindRunCount].SavedIrp = Irp;
//
// Allocate and build a partial Mdl for the request.
//
Mdl = IoAllocateMdl( (PCHAR)MasterIrp->UserBuffer +
IoRuns[UnwindRunCount].Offset,
IoRuns[UnwindRunCount].ByteCount,
FALSE,
FALSE,
Irp );
if (Mdl == NULL) {
FatRaiseStatus( IrpContext, STATUS_INSUFFICIENT_RESOURCES );
}
//
// Sanity Check
//
NT_ASSERT( Mdl == Irp->MdlAddress );
IoBuildPartialMdl( MasterIrp->MdlAddress,
Mdl,
(PCHAR)MasterIrp->UserBuffer +
IoRuns[UnwindRunCount].Offset,
IoRuns[UnwindRunCount].ByteCount );
//
// Get the first IRP stack location in the associated Irp
//
IoSetNextIrpStackLocation( Irp );
IrpSp = IoGetCurrentIrpStackLocation( Irp );
//
// Setup the Stack location to describe our read.
//
IrpSp->MajorFunction = IrpContext->MajorFunction;
IrpSp->Parameters.Read.Length = IoRuns[UnwindRunCount].ByteCount;
IrpSp->Parameters.Read.ByteOffset.QuadPart = IoRuns[UnwindRunCount].Vbo;
//
// Set up the completion routine address in our stack frame.
//
IoSetCompletionRoutine( Irp,
Wait ?
&FatMultiSyncCompletionRoutine :
&FatMultiAsyncCompletionRoutine,
Context,
TRUE,
TRUE,
TRUE );
//
// Setup the next IRP stack location in the associated Irp for the disk
// driver beneath us.
//
IrpSp = IoGetNextIrpStackLocation( Irp );
//
// Setup the Stack location to do a read from the disk driver.
//
IrpSp->MajorFunction = IrpContext->MajorFunction;
IrpSp->Parameters.Read.Length = IoRuns[UnwindRunCount].ByteCount;
IrpSp->Parameters.Read.ByteOffset.QuadPart = IoRuns[UnwindRunCount].Lbo;
//
// If this Irp is the result of a WriteThough operation,
// tell the device to write it through.
//
if (FlagOn( IrpContext->Flags, IRP_CONTEXT_FLAG_WRITE_THROUGH )) {
SetFlag( IrpSp->Flags, SL_WRITE_THROUGH );
}
//
// If this I/O requires override verify, bypass the verify logic.
//
if (FlagOn( IrpContext->Flags, IRP_CONTEXT_FLAG_OVERRIDE_VERIFY )) {
SetFlag( IrpSp->Flags, SL_OVERRIDE_VERIFY_VOLUME );
}
}
//
// Now we no longer expect an exception. If the driver raises, we
// must bugcheck, because we do not know how to recover from that
// case.
//
ExceptionExpected = FALSE;
//
// We only need to set the associated IRP count in the master irp to
// make it a master IRP. But we set the count to one more than our
// caller requested, because we do not want the I/O system to complete
// the I/O. We also set our own count.
//
Context->IrpCount = MultipleIrpCount;
MasterIrp->AssociatedIrp.IrpCount = MultipleIrpCount;
if (Wait) {
MasterIrp->AssociatedIrp.IrpCount += 1;
}
else if (FlagOn( Context->Wait.Async.ResourceThreadId, 3 )) {
//
// For async requests if we acquired locks, transition the lock owners to an
// object, since when we return this thread could go away before request
// completion, and the resource package may try to boost priority.
//
if (Context->Wait.Async.Resource != NULL) {
ExSetResourceOwnerPointer( Context->Wait.Async.Resource,
(PVOID)Context->Wait.Async.ResourceThreadId );
}
if (Context->Wait.Async.Resource2 != NULL) {
ExSetResourceOwnerPointer( Context->Wait.Async.Resource2,
(PVOID)Context->Wait.Async.ResourceThreadId );
}
}
//
// Back up a copy of the IrpContext flags for later use in async completion.
//
Context->IrpContextFlags = IrpContext->Flags;
//
// Now that all the dangerous work is done, issue the read requests
//
for (UnwindRunCount = 0;
UnwindRunCount < MultipleIrpCount;
UnwindRunCount++) {
Irp = IoRuns[UnwindRunCount].SavedIrp;
DebugDoit( FatIoCallDriverCount += 1);
//
// If IoCallDriver returns an error, it has completed the Irp
// and the error will be caught by our completion routines
// and dealt with as a normal IO error.
//
(VOID)FatLowLevelReadWrite( IrpContext,
Vcb->TargetDeviceObject,
Irp,
Vcb );
}
//
// We just issued an IO to the storage stack, update the counters indicating so.
//
if (FatDiskAccountingEnabled) {
FatUpdateIOCountersPCW( IsAWrite, Length );
}
} _SEH2_FINALLY {
ULONG i;
DebugUnwind( FatMultipleAsync );
//
// Only allocating the spinlock, making the associated Irps
// and allocating the Mdls can fail.
//
if ( _SEH2_AbnormalTermination() ) {
//
// If the driver raised, we are hosed. He is not supposed to raise,
// and it is impossible for us to figure out how to clean up.
//
if (!ExceptionExpected) {
NT_ASSERT( ExceptionExpected );
#ifdef _MSC_VER
#pragma prefast( suppress:28159, "things are seriously wrong if we get here" )
#endif
FatBugCheck( 0, 0, 0 );
}
//
// Unwind
//
for (i = 0; i <= UnwindRunCount; i++) {
if ( (Irp = IoRuns[i].SavedIrp) != NULL ) {
if ( Irp->MdlAddress != NULL ) {
IoFreeMdl( Irp->MdlAddress );
}
IoFreeIrp( Irp );
}
}
}
//
// And return to our caller
//
DebugTrace(-1, Dbg, "FatMultipleAsync -> VOID\n", 0);
} _SEH2_END;
return;
}
VOID
FatSingleAsync (
IN PIRP_CONTEXT IrpContext,
IN PVCB Vcb,
IN LBO Lbo,
IN ULONG ByteCount,
IN PIRP Irp
)
/*++
Routine Description:
This routine reads or writes one or more contiguous sectors from a device
asynchronously, and is used if there is only one read necessary to
complete the IRP. It implements the read by simply filling
in the next stack frame in the Irp, and passing it on. The transfer
occurs to the single buffer originally specified in the user request.
Arguments:
IrpContext->MajorFunction - Supplies either IRP_MJ_READ or IRP_MJ_WRITE.
Vcb - Supplies the device to read
Lbo - Supplies the starting Logical Byte Offset to begin reading from
ByteCount - Supplies the number of bytes to read from the device
Irp - Supplies the master Irp to associated with the async
request.
Return Value:
None.
--*/
{
PIO_STACK_LOCATION IrpSp;
PFAT_IO_CONTEXT Context;
#ifndef __REACTOS__
BOOLEAN IsAWrite = FALSE;
#endif
PAGED_CODE();
DebugTrace(+1, Dbg, "FatSingleAsync\n", 0);
DebugTrace( 0, Dbg, "MajorFunction = %08lx\n", IrpContext->MajorFunction );
DebugTrace( 0, Dbg, "Vcb = %p\n", Vcb );
DebugTrace( 0, Dbg, "Lbo = %08lx\n", Lbo);
DebugTrace( 0, Dbg, "ByteCount = %08lx\n", ByteCount);
DebugTrace( 0, Dbg, "Irp = %p\n", Irp );
//
// If this I/O originating during FatVerifyVolume, bypass the
// verify logic.
//
if (Vcb->VerifyThread == KeGetCurrentThread()) {
SetFlag( IrpContext->Flags, IRP_CONTEXT_FLAG_OVERRIDE_VERIFY );
}
//
// Set up the completion routine address in our stack frame.
//
IoSetCompletionRoutine( Irp,
FlagOn(IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT) ?
&FatSingleSyncCompletionRoutine :
&FatSingleAsyncCompletionRoutine,
IrpContext->FatIoContext,
TRUE,
TRUE,
TRUE );
//
// Setup the next IRP stack location in the associated Irp for the disk
// driver beneath us.
//
IrpSp = IoGetNextIrpStackLocation( Irp );
//
// Setup the Stack location to do a read from the disk driver.
//
IrpSp->MajorFunction = IrpContext->MajorFunction;
IrpSp->Parameters.Read.Length = ByteCount;
IrpSp->Parameters.Read.ByteOffset.QuadPart = Lbo;
#ifndef __REACTOS__
IsAWrite = (IrpSp->MajorFunction == IRP_MJ_WRITE);
#endif
//
// If this Irp is the result of a WriteThough operation,
// tell the device to write it through.
//
if (FlagOn( IrpContext->Flags, IRP_CONTEXT_FLAG_WRITE_THROUGH )) {
SetFlag( IrpSp->Flags, SL_WRITE_THROUGH );
}
//
// If this I/O requires override verify, bypass the verify logic.
//
if (FlagOn( IrpContext->Flags, IRP_CONTEXT_FLAG_OVERRIDE_VERIFY )) {
SetFlag( IrpSp->Flags, SL_OVERRIDE_VERIFY_VOLUME );
}
//
// For async requests if we acquired locks, transition the lock owners to an
// object, since when we return this thread could go away before request
// completion, and the resource package may try to boost priority.
//
if (!FlagOn( IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT ) &&
FlagOn( IrpContext->FatIoContext->Wait.Async.ResourceThreadId, 3 )) {
Context = IrpContext->FatIoContext;
if (Context->Wait.Async.Resource != NULL) {
ExSetResourceOwnerPointer( Context->Wait.Async.Resource,
(PVOID)Context->Wait.Async.ResourceThreadId );
}
if (Context->Wait.Async.Resource2 != NULL) {
ExSetResourceOwnerPointer( Context->Wait.Async.Resource2,
(PVOID)Context->Wait.Async.ResourceThreadId );
}
}
//
// Back up a copy of the IrpContext flags for later use in async completion.
//
IrpContext->FatIoContext->IrpContextFlags = IrpContext->Flags;
//
// Issue the read request
//
DebugDoit( FatIoCallDriverCount += 1);
//
// If IoCallDriver returns an error, it has completed the Irp
// and the error will be caught by our completion routines
// and dealt with as a normal IO error.
//
(VOID)FatLowLevelReadWrite( IrpContext,
Vcb->TargetDeviceObject,
Irp,
Vcb );
//
// We just issued an IO to the storage stack, update the counters indicating so.
//
if (FatDiskAccountingEnabled) {
FatUpdateIOCountersPCW( IsAWrite, ByteCount );
}
//
// And return to our caller
//
DebugTrace(-1, Dbg, "FatSingleAsync -> VOID\n", 0);
return;
}
VOID
FatSingleNonAlignedSync (
IN PIRP_CONTEXT IrpContext,
IN PVCB Vcb,
IN PUCHAR Buffer,
IN LBO Lbo,
IN ULONG ByteCount,
IN PIRP Irp
)
/*++
Routine Description:
This routine reads or writes one or more contiguous sectors from a device
Synchronously, and does so to a buffer that must come from non paged
pool. It saves a pointer to the Irp's original Mdl, and creates a new
one describing the given buffer. It implements the read by simply filling
in the next stack frame in the Irp, and passing it on. The transfer
occurs to the single buffer originally specified in the user request.
Arguments:
IrpContext->MajorFunction - Supplies either IRP_MJ_READ or IRP_MJ_WRITE.
Vcb - Supplies the device to read
Buffer - Supplies a buffer from non-paged pool.
Lbo - Supplies the starting Logical Byte Offset to begin reading from
ByteCount - Supplies the number of bytes to read from the device
Irp - Supplies the master Irp to associated with the async
request.
Return Value:
None.
--*/
{
PIO_STACK_LOCATION IrpSp;
PMDL Mdl;
PMDL SavedMdl;
#ifndef __REACTOS__
BOOLEAN IsAWrite = FALSE;
#endif
PAGED_CODE();
DebugTrace(+1, Dbg, "FatSingleNonAlignedAsync\n", 0);
DebugTrace( 0, Dbg, "MajorFunction = %08lx\n", IrpContext->MajorFunction );
DebugTrace( 0, Dbg, "Vcb = %p\n", Vcb );
DebugTrace( 0, Dbg, "Buffer = %p\n", Buffer );
DebugTrace( 0, Dbg, "Lbo = %08lx\n", Lbo);
DebugTrace( 0, Dbg, "ByteCount = %08lx\n", ByteCount);
DebugTrace( 0, Dbg, "Irp = %p\n", Irp );
//
// Create a new Mdl describing the buffer, saving the current one in the
// Irp
//
SavedMdl = Irp->MdlAddress;
Irp->MdlAddress = 0;
Mdl = IoAllocateMdl( Buffer,
ByteCount,
FALSE,
FALSE,
Irp );
if (Mdl == NULL) {
Irp->MdlAddress = SavedMdl;
FatRaiseStatus( IrpContext, STATUS_INSUFFICIENT_RESOURCES );
}
//
// Lock the new Mdl in memory.
//
_SEH2_TRY {
MmProbeAndLockPages( Mdl, KernelMode, IoWriteAccess );
} _SEH2_FINALLY {
if ( _SEH2_AbnormalTermination() ) {
IoFreeMdl( Mdl );
Irp->MdlAddress = SavedMdl;
}
} _SEH2_END;
//
// Set up the completion routine address in our stack frame.
//
IoSetCompletionRoutine( Irp,
&FatSingleSyncCompletionRoutine,
IrpContext->FatIoContext,
TRUE,
TRUE,
TRUE );
//
// Setup the next IRP stack location in the associated Irp for the disk
// driver beneath us.
//
IrpSp = IoGetNextIrpStackLocation( Irp );
//
// Setup the Stack location to do a read from the disk driver.
//
IrpSp->MajorFunction = IrpContext->MajorFunction;
IrpSp->Parameters.Read.Length = ByteCount;
IrpSp->Parameters.Read.ByteOffset.QuadPart = Lbo;
#ifndef __REACTOS__
IsAWrite = (IrpSp->MajorFunction == IRP_MJ_WRITE);
#endif
//
// If this I/O originating during FatVerifyVolume, bypass the
// verify logic.
//
if (Vcb->VerifyThread == KeGetCurrentThread()) {
SetFlag( IrpContext->Flags, IRP_CONTEXT_FLAG_OVERRIDE_VERIFY );
}
//
// If this I/O requires override verify, bypass the verify logic.
//
if (FlagOn( IrpContext->Flags, IRP_CONTEXT_FLAG_OVERRIDE_VERIFY )) {
SetFlag( IrpSp->Flags, SL_OVERRIDE_VERIFY_VOLUME );
}
//
// Issue the read request
//
DebugDoit( FatIoCallDriverCount += 1);
//
// If IoCallDriver returns an error, it has completed the Irp
// and the error will be caught by our completion routines
// and dealt with as a normal IO error.
//
_SEH2_TRY {
(VOID)FatLowLevelReadWrite( IrpContext,
Vcb->TargetDeviceObject,
Irp,
Vcb );
FatWaitSync( IrpContext );
} _SEH2_FINALLY {
MmUnlockPages( Mdl );
IoFreeMdl( Mdl );
Irp->MdlAddress = SavedMdl;
} _SEH2_END;
//
// We just issued an IO to the storage stack, update the counters indicating so.
//
if (FatDiskAccountingEnabled) {
FatUpdateIOCountersPCW( IsAWrite, ByteCount );
}
//
// And return to our caller
//
DebugTrace(-1, Dbg, "FatSingleNonAlignedSync -> VOID\n", 0);
return;
}
VOID
FatWaitSync (
IN PIRP_CONTEXT IrpContext
)
/*++
Routine Description:
This routine waits for one or more previously started I/O requests
from the above routines, by simply waiting on the event.
Arguments:
Return Value:
None
--*/
{
PAGED_CODE();
DebugTrace(+1, Dbg, "FatWaitSync, Context = %p\n", IrpContext->FatIoContext );
KeWaitForSingleObject( &IrpContext->FatIoContext->Wait.SyncEvent,
Executive, KernelMode, FALSE, NULL );
KeClearEvent( &IrpContext->FatIoContext->Wait.SyncEvent );
DebugTrace(-1, Dbg, "FatWaitSync -> VOID\n", 0 );
}
//
// Internal Support Routine
//
NTSTATUS
NTAPI
FatMultiSyncCompletionRoutine (
_In_ PDEVICE_OBJECT DeviceObject,
_In_ PIRP Irp,
_In_reads_opt_(_Inexpressible_("varies")) PVOID Contxt
)
/*++
Routine Description:
This is the completion routine for all reads and writes started via
FatRead/WriteMultipleAsynch. It must synchronize its operation for
multiprocessor environments with itself on all other processors, via
a spin lock found via the Context parameter.
The completion routine has the following responsibilities:
If the individual request was completed with an error, then
this completion routine must see if this is the first error
(essentially by Vbo), and if so it must correctly reduce the
byte count and remember the error status in the Context.
If the IrpCount goes to 1, then it sets the event in the Context
parameter to signal the caller that all of the asynch requests
are done.
Arguments:
DeviceObject - Pointer to the file system device object.
Irp - Pointer to the associated Irp which is being completed. (This
Irp will no longer be accessible after this routine returns.)
Contxt - The context parameter which was specified for all of
the multiple asynch I/O requests for this MasterIrp.
Return Value:
The routine returns STATUS_MORE_PROCESSING_REQUIRED so that we can
immediately complete the Master Irp without being in a race condition
with the IoCompleteRequest thread trying to decrement the IrpCount in
the Master Irp.
--*/
{
PFAT_IO_CONTEXT Context = Contxt;
PIRP MasterIrp = Context->MasterIrp;
DebugTrace(+1, Dbg, "FatMultiSyncCompletionRoutine, Context = %p\n", Context );
//
// If we got an error (or verify required), remember it in the Irp
//
if (!NT_SUCCESS( Irp->IoStatus.Status )) {
#if DBG
if(!( NT_SUCCESS( FatBreakOnInterestingIoCompletion ) || Irp->IoStatus.Status != FatBreakOnInterestingIoCompletion )) {
DbgBreakPoint();
}
#endif
#ifdef SYSCACHE_COMPILE
DbgPrint( "FAT SYSCACHE: MultiSync (IRP %08x for Master %08x) -> %08x\n", Irp, MasterIrp, Irp->IoStatus );
#endif
MasterIrp->IoStatus = Irp->IoStatus;
}
NT_ASSERT( !(NT_SUCCESS( Irp->IoStatus.Status ) && Irp->IoStatus.Information == 0 ));
//
// We must do this here since IoCompleteRequest won't get a chance
// on this associated Irp.
//
IoFreeMdl( Irp->MdlAddress );
IoFreeIrp( Irp );
if (InterlockedDecrement(&Context->IrpCount) == 0) {
FatDoCompletionZero( MasterIrp, Context );
KeSetEvent( &Context->Wait.SyncEvent, 0, FALSE );
}
DebugTrace(-1, Dbg, "FatMultiSyncCompletionRoutine -> SUCCESS\n", 0 );
UNREFERENCED_PARAMETER( DeviceObject );
return STATUS_MORE_PROCESSING_REQUIRED;
}
//
// Internal Support Routine
//
NTSTATUS
NTAPI
FatMultiAsyncCompletionRoutine (
_In_ PDEVICE_OBJECT DeviceObject,
_In_ PIRP Irp,
_In_reads_opt_(_Inexpressible_("varies")) PVOID Contxt
)
/*++
Routine Description:
This is the completion routine for all reads and writes started via
FatRead/WriteMultipleAsynch. It must synchronize its operation for
multiprocessor environments with itself on all other processors, via
a spin lock found via the Context parameter.
The completion routine has has the following responsibilities:
If the individual request was completed with an error, then
this completion routine must see if this is the first error
(essentially by Vbo), and if so it must correctly reduce the
byte count and remember the error status in the Context.
If the IrpCount goes to 1, then it sets the event in the Context
parameter to signal the caller that all of the asynch requests
are done.
Arguments:
DeviceObject - Pointer to the file system device object.
Irp - Pointer to the associated Irp which is being completed. (This
Irp will no longer be accessible after this routine returns.)
Contxt - The context parameter which was specified for all of
the multiple asynch I/O requests for this MasterIrp.
Return Value:
The routine returns STATUS_MORE_PROCESSING_REQUIRED so that we can
immediately complete the Master Irp without being in a race condition
with the IoCompleteRequest thread trying to decrement the IrpCount in
the Master Irp.
--*/
{
NTSTATUS Status = STATUS_SUCCESS;
PFAT_IO_CONTEXT Context = Contxt;
PIRP MasterIrp = Context->MasterIrp;
BOOLEAN PostRequest = FALSE;
DebugTrace(+1, Dbg, "FatMultiAsyncCompletionRoutine, Context = %p\n", Context );
//
// If we got an error (or verify required), remember it in the Irp
//
if (!NT_SUCCESS( Irp->IoStatus.Status )) {
#if DBG
if (!( NT_SUCCESS( FatBreakOnInterestingIoCompletion ) || Irp->IoStatus.Status != FatBreakOnInterestingIoCompletion )) {
DbgBreakPoint();
}
#endif
#ifdef SYSCACHE_COMPILE
DbgPrint( "FAT SYSCACHE: MultiAsync (IRP %08x for Master %08x) -> %08x\n", Irp, MasterIrp, Irp->IoStatus );
#endif
MasterIrp->IoStatus = Irp->IoStatus;
}
NT_ASSERT( !(NT_SUCCESS( Irp->IoStatus.Status ) && Irp->IoStatus.Information == 0 ));
if (InterlockedDecrement(&Context->IrpCount) == 0) {
FatDoCompletionZero( MasterIrp, Context );
if (NT_SUCCESS(MasterIrp->IoStatus.Status)) {
MasterIrp->IoStatus.Information =
Context->Wait.Async.RequestedByteCount;
NT_ASSERT(MasterIrp->IoStatus.Information != 0);
//
// Now if this wasn't PagingIo, set either the read or write bit.
//
if (!FlagOn(MasterIrp->Flags, IRP_PAGING_IO)) {
SetFlag( Context->Wait.Async.FileObject->Flags,
IoGetCurrentIrpStackLocation(MasterIrp)->MajorFunction == IRP_MJ_READ ?
FO_FILE_FAST_IO_READ : FO_FILE_MODIFIED );
}
} else {
//
// Post STATUS_VERIFY_REQUIRED failures. Only post top level IRPs, because recursive I/Os
// cannot process volume verification.
//
if (!FlagOn(Context->IrpContextFlags, IRP_CONTEXT_FLAG_RECURSIVE_CALL) &&
(MasterIrp->IoStatus.Status == STATUS_VERIFY_REQUIRED)) {
PostRequest = TRUE;
}
}
//
// If this was a special async write, decrement the count. Set the
// event if this was the final outstanding I/O for the file. We will
// also want to queue an APC to deal with any error conditionions.
//
_Analysis_assume_(!(Context->Wait.Async.NonPagedFcb) &&
(ExInterlockedAddUlong( &Context->Wait.Async.NonPagedFcb->OutstandingAsyncWrites,
0xffffffff,
&FatData.GeneralSpinLock ) != 1));
if ((Context->Wait.Async.NonPagedFcb) &&
(ExInterlockedAddUlong( &Context->Wait.Async.NonPagedFcb->OutstandingAsyncWrites,
0xffffffff,
&FatData.GeneralSpinLock ) == 1)) {
KeSetEvent( Context->Wait.Async.NonPagedFcb->OutstandingAsyncEvent, 0, FALSE );
}
//
// Now release the resources.
//
if (Context->Wait.Async.Resource != NULL) {
ExReleaseResourceForThreadLite( Context->Wait.Async.Resource,
Context->Wait.Async.ResourceThreadId );
}
if (Context->Wait.Async.Resource2 != NULL) {
ExReleaseResourceForThreadLite( Context->Wait.Async.Resource2,
Context->Wait.Async.ResourceThreadId );
}
//
// Mark the master Irp pending
//
IoMarkIrpPending( MasterIrp );
//
// and finally, free the context record.
//
ExFreePool( Context );
if (PostRequest) {
PIRP_CONTEXT IrpContext = NULL;
_SEH2_TRY {
IrpContext = FatCreateIrpContext(Irp, TRUE );
ClearFlag(IrpContext->Flags, IRP_CONTEXT_FLAG_RECURSIVE_CALL);
FatFsdPostRequest( IrpContext, Irp );
Status = STATUS_MORE_PROCESSING_REQUIRED;
} _SEH2_EXCEPT( FatExceptionFilter(NULL, _SEH2_GetExceptionInformation()) ) {
//
// If we failed to post the IRP, we just have to return the failure
// to the user. :(
//
NOTHING;
} _SEH2_END;
}
}
DebugTrace(-1, Dbg, "FatMultiAsyncCompletionRoutine -> SUCCESS\n", 0 );
UNREFERENCED_PARAMETER( DeviceObject );
return Status;
}
NTSTATUS
FatPagingFileErrorHandler (
IN PIRP Irp,
IN PKEVENT Event OPTIONAL
)
/*++
Routine Description:
This routine attempts to guarantee that the media is marked dirty
with the surface test bit if a paging file IO fails.
The work done here has several basic problems
1) when paging file writes start failing, this is a good sign
that the rest of the system is about to fall down around us
2) it has no forward progress guarantee
With Whistler, it is actually quite intentional that we're rejiggering
the paging file write path to make forward progress at all times. This
means that the cases where it *does* fail, we're truly seeing media errors
and this is probably going to mean the paging file is going to stop working
very soon.
It'd be nice to make this guarantee progress. It would need
1) a guaranteed worker thread which can only be used by items which
will make forward progress (i.e., not block out this one)
2) the virtual volume file's pages containing the boot sector and
1st FAT entry would have to be pinned resident and have a guaranteed
mapping address
3) mark volume would have to have a stashed irp/mdl and roll the write
irp, or use a generalized mechanism to guarantee issue of the irp
4) the lower stack would have to guarantee progress
Of these, 1 and 4 may actually exist shortly.
Arguments:
Irp - Pointer to the associated Irp which is being failed.
Event - Pointer to optional event to be signalled instead of completing
the IRP
Return Value:
Returns STATUS_MORE_PROCESSING_REQUIRED if we managed to queue off the workitem,
STATUS_SUCCESS otherwise.
--*/
{
NTSTATUS Status;
//
// If this was a media error, we want to chkdsk /r the next time we boot.
//
if (FsRtlIsTotalDeviceFailure(Irp->IoStatus.Status)) {
Status = STATUS_SUCCESS;
} else {
PCLEAN_AND_DIRTY_VOLUME_PACKET Packet;
//
// We are going to try to mark the volume needing recover.
// If we can't get pool, oh well....
//
Packet = ExAllocatePoolWithTag(NonPagedPoolNx, sizeof(CLEAN_AND_DIRTY_VOLUME_PACKET), ' taF');
if ( Packet ) {
Packet->Vcb = &((PVOLUME_DEVICE_OBJECT)IoGetCurrentIrpStackLocation(Irp)->DeviceObject)->Vcb;
Packet->Irp = Irp;
Packet->Event = Event;
ExInitializeWorkItem( &Packet->Item,
&FatFspMarkVolumeDirtyWithRecover,
Packet );
#ifdef _MSC_VER
#pragma prefast( suppress:28159, "prefast indicates this is obsolete, but it is ok for fastfat to use it" )
#endif
ExQueueWorkItem( &Packet->Item, CriticalWorkQueue );
Status = STATUS_MORE_PROCESSING_REQUIRED;
} else {
Status = STATUS_SUCCESS;
}
}
return Status;
}
//
// Internal Support Routine
//
NTSTATUS
NTAPI
FatPagingFileCompletionRoutineCatch (
_In_ PDEVICE_OBJECT DeviceObject,
_In_ PIRP Irp,
_In_reads_opt_(_Inexpressible_("varies")) PVOID Contxt
)
/*++
Routine Description:
This is the completion routine for all reads and writes started via
FatPagingFileIo that reuse the master irp (that we have to catch
on the way back). It is always invoked.
The completion routine has has the following responsibility:
If the error implies a media problem, it enqueues a
worker item to write out the dirty bit so that the next
time we run we will do a autochk /r. This is not forward
progress guaranteed at the moment.
Clean up the Mdl used for this partial request.
Note that if the Irp is failing, the error code is already where
we want it.
Arguments:
DeviceObject - Pointer to the file system device object.
Irp - Pointer to the associated Irp which is being completed. (This
Irp will no longer be accessible after this routine returns.)
MasterIrp - Pointer to the master Irp.
Return Value:
Always returns STATUS_MORE_PROCESSING_REQUIRED.
--*/
{
PFAT_PAGING_FILE_CONTEXT Context = (PFAT_PAGING_FILE_CONTEXT) Contxt;
UNREFERENCED_PARAMETER( DeviceObject );
DebugTrace(+1, Dbg, "FatPagingFileCompletionRoutineCatch, Context = %p\n", Context );
//
// Cleanup the existing Mdl, perhaps by returning the reserve.
//
if (Irp->MdlAddress == FatReserveMdl) {
MmPrepareMdlForReuse( Irp->MdlAddress );
KeSetEvent( &FatReserveEvent, 0, FALSE );
} else {
IoFreeMdl( Irp->MdlAddress );
}
//
// Restore the original Mdl.
//
Irp->MdlAddress = Context->RestoreMdl;
DebugTrace(-1, Dbg, "FatPagingFileCompletionRoutine => (done)\n", 0 );
//
// If the IRP is succeeding or the failure handler did not post off the
// completion, we're done and should set the event to let the master
// know the IRP is his again.
//
if (NT_SUCCESS( Irp->IoStatus.Status ) ||
FatPagingFileErrorHandler( Irp, &Context->Event ) == STATUS_SUCCESS) {
KeSetEvent( &Context->Event, 0, FALSE );
}
return STATUS_MORE_PROCESSING_REQUIRED;
}
//
// Internal Support Routine
//
NTSTATUS
NTAPI
FatPagingFileCompletionRoutine (
_In_ PDEVICE_OBJECT DeviceObject,
_In_ PIRP Irp,
_In_reads_opt_(_Inexpressible_("varies")) PVOID MasterIrp
)
/*++
Routine Description:
This is the completion routine for all reads and writes started via
FatPagingFileIo. It should only be invoked on error or cancel.
The completion routine has has the following responsibility:
Since the individual request was completed with an error,
this completion routine must stuff it into the master irp.
If the error implies a media problem, it also enqueues a
worker item to write out the dirty bit so that the next
time we run we will do a autochk /r
Arguments:
DeviceObject - Pointer to the file system device object.
Irp - Pointer to the associated Irp which is being completed. (This
Irp will no longer be accessible after this routine returns.)
MasterIrp - Pointer to the master Irp.
Return Value:
Always returns STATUS_SUCCESS.
--*/
{
DebugTrace(+1, Dbg, "FatPagingFileCompletionRoutine, MasterIrp = %p\n", MasterIrp );
//
// If we got an error (or verify required), remember it in the Irp
//
NT_ASSERT( !NT_SUCCESS( Irp->IoStatus.Status ));
//
// If we were invoked with an assoicated Irp, copy the error over.
//
if (Irp != MasterIrp) {
((PIRP)MasterIrp)->IoStatus = Irp->IoStatus;
}
DebugTrace(-1, Dbg, "FatPagingFileCompletionRoutine => (done)\n", 0 );
UNREFERENCED_PARAMETER( DeviceObject );
return FatPagingFileErrorHandler( Irp, NULL );
}
//
// Internal Support Routine
//
NTSTATUS
NTAPI
FatSpecialSyncCompletionRoutine (
_In_ PDEVICE_OBJECT DeviceObject,
_In_ PIRP Irp,
_In_reads_opt_(_Inexpressible_("varies")) PVOID Contxt
)
/*++
Routine Description:
This is the completion routine for a special set of sub irps
that have to work at APC level.
The completion routine has has the following responsibilities:
It sets the event passed as the context to signal that the
request is done.
By doing this, the caller will be released before final APC
completion with knowledge that the IRP is finished. Final
completion will occur at an indeterminate time after this
occurs, and by using this completion routine the caller expects
to not have any output or status returned. A junk user Iosb
should be used to capture the status without forcing Io to take
an exception on NULL.
Arguments:
DeviceObject - Pointer to the file system device object.
Irp - Pointer to the Irp for this request. (This Irp will no longer
be accessible after this routine returns.)
Contxt - The context parameter which was specified in the call to
FatRead/WriteSingleAsynch.
Return Value:
Currently always returns STATUS_SUCCESS.
--*/
{
PFAT_SYNC_CONTEXT SyncContext = (PFAT_SYNC_CONTEXT)Contxt;
UNREFERENCED_PARAMETER( Irp );
DebugTrace(+1, Dbg, "FatSpecialSyncCompletionRoutine, Context = %p\n", Contxt );
SyncContext->Iosb = Irp->IoStatus;
KeSetEvent( &SyncContext->Event, 0, FALSE );
DebugTrace(-1, Dbg, "FatSpecialSyncCompletionRoutine -> STATUS_SUCCESS\n", 0 );
UNREFERENCED_PARAMETER( DeviceObject );
return STATUS_SUCCESS;
}
//
// Internal Support Routine
//
NTSTATUS
NTAPI
FatSingleSyncCompletionRoutine (
_In_ PDEVICE_OBJECT DeviceObject,
_In_ PIRP Irp,
_In_reads_opt_(_Inexpressible_("varies")) PVOID Contxt
)
/*++
Routine Description:
This is the completion routine for all reads and writes started via
FatRead/WriteSingleAsynch.
The completion routine has has the following responsibilities:
Copy the I/O status from the Irp to the Context, since the Irp
will no longer be accessible.
It sets the event in the Context parameter to signal the caller
that all of the asynch requests are done.
Arguments:
DeviceObject - Pointer to the file system device object.
Irp - Pointer to the Irp for this request. (This Irp will no longer
be accessible after this routine returns.)
Contxt - The context parameter which was specified in the call to
FatRead/WriteSingleAsynch.
Return Value:
Currently always returns STATUS_SUCCESS.
--*/
{
PFAT_IO_CONTEXT Context = Contxt;
DebugTrace(+1, Dbg, "FatSingleSyncCompletionRoutine, Context = %p\n", Context );
FatDoCompletionZero( Irp, Context );
if (!NT_SUCCESS( Irp->IoStatus.Status )) {
#if DBG
if(!( NT_SUCCESS( FatBreakOnInterestingIoCompletion ) || Irp->IoStatus.Status != FatBreakOnInterestingIoCompletion )) {
DbgBreakPoint();
}
#endif
}
NT_ASSERT( !(NT_SUCCESS( Irp->IoStatus.Status ) && Irp->IoStatus.Information == 0 ));
KeSetEvent( &Context->Wait.SyncEvent, 0, FALSE );
DebugTrace(-1, Dbg, "FatSingleSyncCompletionRoutine -> STATUS_MORE_PROCESSING_REQUIRED\n", 0 );
UNREFERENCED_PARAMETER( DeviceObject );
return STATUS_MORE_PROCESSING_REQUIRED;
}
//
// Internal Support Routine
//
NTSTATUS
NTAPI
FatSingleAsyncCompletionRoutine (
_In_ PDEVICE_OBJECT DeviceObject,
_In_ PIRP Irp,
_In_reads_opt_(_Inexpressible_("varies")) PVOID Contxt
)
/*++
Routine Description:
This is the completion routine for all reads and writes started via
FatRead/WriteSingleAsynch.
The completion routine has has the following responsibilities:
Copy the I/O status from the Irp to the Context, since the Irp
will no longer be accessible.
It sets the event in the Context parameter to signal the caller
that all of the asynch requests are done.
Arguments:
DeviceObject - Pointer to the file system device object.
Irp - Pointer to the Irp for this request. (This Irp will no longer
be accessible after this routine returns.)
Contxt - The context parameter which was specified in the call to
FatRead/WriteSingleAsynch.
Return Value:
Currently always returns STATUS_SUCCESS.
--*/
{
NTSTATUS Status = STATUS_SUCCESS;
PFAT_IO_CONTEXT Context = Contxt;
BOOLEAN PostRequest = FALSE;
DebugTrace(+1, Dbg, "FatSingleAsyncCompletionRoutine, Context = %p\n", Context );
//
// Fill in the information field correctedly if this worked.
//
FatDoCompletionZero( Irp, Context );
if (NT_SUCCESS(Irp->IoStatus.Status)) {
NT_ASSERT( Irp->IoStatus.Information != 0 );
Irp->IoStatus.Information = Context->Wait.Async.RequestedByteCount;
NT_ASSERT( Irp->IoStatus.Information != 0 );
//
// Now if this wasn't PagingIo, set either the read or write bit.
//
if (!FlagOn(Irp->Flags, IRP_PAGING_IO)) {
SetFlag( Context->Wait.Async.FileObject->Flags,
IoGetCurrentIrpStackLocation(Irp)->MajorFunction == IRP_MJ_READ ?
FO_FILE_FAST_IO_READ : FO_FILE_MODIFIED );
}
} else {
#if DBG
if(!( NT_SUCCESS( FatBreakOnInterestingIoCompletion ) || Irp->IoStatus.Status != FatBreakOnInterestingIoCompletion )) {
DbgBreakPoint();
}
#endif
#ifdef SYSCACHE_COMPILE
DbgPrint( "FAT SYSCACHE: SingleAsync (IRP %08x) -> %08x\n", Irp, Irp->IoStatus );
#endif
//
// Post STATUS_VERIFY_REQUIRED failures. Only post top level IRPs, because recursive I/Os
// cannot process volume verification.
//
if (!FlagOn(Context->IrpContextFlags, IRP_CONTEXT_FLAG_RECURSIVE_CALL) &&
(Irp->IoStatus.Status == STATUS_VERIFY_REQUIRED)) {
PostRequest = TRUE;
}
}
//
// If this was a special async write, decrement the count. Set the
// event if this was the final outstanding I/O for the file. We will
// also want to queue an APC to deal with any error conditionions.
//
_Analysis_assume_(!(Context->Wait.Async.NonPagedFcb) &&
(ExInterlockedAddUlong( &Context->Wait.Async.NonPagedFcb->OutstandingAsyncWrites,
0xffffffff,
&FatData.GeneralSpinLock ) != 1));
if ((Context->Wait.Async.NonPagedFcb) &&
(ExInterlockedAddUlong( &Context->Wait.Async.NonPagedFcb->OutstandingAsyncWrites,
0xffffffff,
&FatData.GeneralSpinLock ) == 1)) {
KeSetEvent( Context->Wait.Async.NonPagedFcb->OutstandingAsyncEvent, 0, FALSE );
}
//
// Now release the resources
//
if (Context->Wait.Async.Resource != NULL) {
ExReleaseResourceForThreadLite( Context->Wait.Async.Resource,
Context->Wait.Async.ResourceThreadId );
}
if (Context->Wait.Async.Resource2 != NULL) {
ExReleaseResourceForThreadLite( Context->Wait.Async.Resource2,
Context->Wait.Async.ResourceThreadId );
}
//
// Mark the Irp pending
//
IoMarkIrpPending( Irp );
//
// and finally, free the context record.
//
ExFreePool( Context );
if (PostRequest) {
PIRP_CONTEXT IrpContext = NULL;
_SEH2_TRY {
IrpContext = FatCreateIrpContext(Irp, TRUE );
ClearFlag(IrpContext->Flags, IRP_CONTEXT_FLAG_RECURSIVE_CALL);
FatFsdPostRequest( IrpContext, Irp );
Status = STATUS_MORE_PROCESSING_REQUIRED;
} _SEH2_EXCEPT( FatExceptionFilter(NULL, _SEH2_GetExceptionInformation()) ) {
//
// If we failed to post the IRP, we just have to return the failure
// to the user. :(
//
NOTHING;
} _SEH2_END;
}
DebugTrace(-1, Dbg, "FatSingleAsyncCompletionRoutine -> STATUS_MORE_PROCESSING_REQUIRED\n", 0 );
UNREFERENCED_PARAMETER( DeviceObject );
return Status;
}
VOID
FatLockUserBuffer (
IN PIRP_CONTEXT IrpContext,
IN OUT PIRP Irp,
IN LOCK_OPERATION Operation,
IN ULONG BufferLength
)
/*++
Routine Description:
This routine locks the specified buffer for the specified type of
access. The file system requires this routine since it does not
ask the I/O system to lock its buffers for direct I/O. This routine
may only be called from the Fsd while still in the user context.
Note that this is the *input/output* buffer.
Arguments:
Irp - Pointer to the Irp for which the buffer is to be locked.
Operation - IoWriteAccess for read operations, or IoReadAccess for
write operations.
BufferLength - Length of user buffer.
Return Value:
None
--*/
{
PMDL Mdl = NULL;
PAGED_CODE();
if (Irp->MdlAddress == NULL) {
//
// Allocate the Mdl, and Raise if we fail.
//
Mdl = IoAllocateMdl( Irp->UserBuffer, BufferLength, FALSE, FALSE, Irp );
if (Mdl == NULL) {
FatRaiseStatus( IrpContext, STATUS_INSUFFICIENT_RESOURCES );
}
//
// Now probe the buffer described by the Irp. If we get an exception,
// deallocate the Mdl and return the appropriate "expected" status.
//
_SEH2_TRY {
MmProbeAndLockPages( Mdl,
Irp->RequestorMode,
Operation );
} _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) {
NTSTATUS Status;
Status = _SEH2_GetExceptionCode();
IoFreeMdl( Mdl );
Irp->MdlAddress = NULL;
FatRaiseStatus( IrpContext,
FsRtlIsNtstatusExpected(Status) ? Status : STATUS_INVALID_USER_BUFFER );
} _SEH2_END;
}
UNREFERENCED_PARAMETER( IrpContext );
}
PVOID
FatMapUserBuffer (
IN PIRP_CONTEXT IrpContext,
IN OUT PIRP Irp
)
/*++
Routine Description:
This routine conditionally maps the user buffer for the current I/O
request in the specified mode. If the buffer is already mapped, it
just returns its address.
Note that this is the *input/output* buffer.
Arguments:
Irp - Pointer to the Irp for the request.
Return Value:
Mapped address
--*/
{
UNREFERENCED_PARAMETER( IrpContext );
PAGED_CODE();
//
// If there is no Mdl, then we must be in the Fsd, and we can simply
// return the UserBuffer field from the Irp.
//
if (Irp->MdlAddress == NULL) {
return Irp->UserBuffer;
} else {
PVOID Address = MmGetSystemAddressForMdlSafe( Irp->MdlAddress, NormalPagePriority | MdlMappingNoExecute );
if (Address == NULL) {
ExRaiseStatus( STATUS_INSUFFICIENT_RESOURCES );
}
return Address;
}
}
PVOID
FatBufferUserBuffer (
IN PIRP_CONTEXT IrpContext,
IN OUT PIRP Irp,
IN ULONG BufferLength
)
/*++
Routine Description:
This routine conditionally buffers the user buffer for the current I/O
request. If the buffer is already buffered, it just returns its address.
Note that this is the *input* buffer.
Arguments:
Irp - Pointer to the Irp for the request.
BufferLength - Length of user buffer.
Return Value:
Buffered address.
--*/
{
PUCHAR UserBuffer;
UNREFERENCED_PARAMETER( IrpContext );
PAGED_CODE();
//
// Handle the no buffer case.
//
if (BufferLength == 0) {
return NULL;
}
//
// If there is no system buffer we must have been supplied an Mdl
// describing the users input buffer, which we will now snapshot.
//
if (Irp->AssociatedIrp.SystemBuffer == NULL) {
UserBuffer = FatMapUserBuffer( IrpContext, Irp );
Irp->AssociatedIrp.SystemBuffer = FsRtlAllocatePoolWithQuotaTag( NonPagedPoolNx,
BufferLength,
TAG_IO_USER_BUFFER );
//
// Set the flags so that the completion code knows to deallocate the
// buffer.
//
Irp->Flags |= (IRP_BUFFERED_IO | IRP_DEALLOCATE_BUFFER);
_SEH2_TRY {
RtlCopyMemory( Irp->AssociatedIrp.SystemBuffer,
UserBuffer,
BufferLength );
} _SEH2_EXCEPT (EXCEPTION_EXECUTE_HANDLER) {
NTSTATUS Status;
Status = _SEH2_GetExceptionCode();
FatRaiseStatus( IrpContext,
FsRtlIsNtstatusExpected(Status) ? Status : STATUS_INVALID_USER_BUFFER );
} _SEH2_END;
}
return Irp->AssociatedIrp.SystemBuffer;
}
NTSTATUS
FatToggleMediaEjectDisable (
IN PIRP_CONTEXT IrpContext,
IN PVCB Vcb,
IN BOOLEAN PreventRemoval
)
/*++
Routine Description:
The routine either enables or disables the eject button on removable
media.
Arguments:
Vcb - Descibes the volume to operate on
PreventRemoval - TRUE if we should disable the media eject button. FALSE
if we want to enable it.
Return Value:
Status of the operation.
--*/
{
PIRP Irp;
KIRQL SavedIrql;
NTSTATUS Status;
FAT_SYNC_CONTEXT SyncContext;
PREVENT_MEDIA_REMOVAL Prevent;
UNREFERENCED_PARAMETER( IrpContext );
//
// If PreventRemoval is the same as VCB_STATE_FLAG_REMOVAL_PREVENTED,
// no-op this call, otherwise toggle the state of the flag.
//
KeAcquireSpinLock( &FatData.GeneralSpinLock, &SavedIrql );
if ((PreventRemoval ^
BooleanFlagOn(Vcb->VcbState, VCB_STATE_FLAG_REMOVAL_PREVENTED)) == 0) {
KeReleaseSpinLock( &FatData.GeneralSpinLock, SavedIrql );
return STATUS_SUCCESS;
} else {
Vcb->VcbState ^= VCB_STATE_FLAG_REMOVAL_PREVENTED;
KeReleaseSpinLock( &FatData.GeneralSpinLock, SavedIrql );
}
Prevent.PreventMediaRemoval = PreventRemoval;
KeInitializeEvent( &SyncContext.Event, NotificationEvent, FALSE );
//
// We build this IRP using a junk Iosb that will receive the final
// completion status since we won't be around for it.
//
// We fill in the UserIosb manually below,
// So passing NULL for the final parameter is ok in this special case.
//
#ifdef _MSC_VER
#pragma warning(suppress: 6387)
#endif
Irp = IoBuildDeviceIoControlRequest( IOCTL_DISK_MEDIA_REMOVAL,
Vcb->TargetDeviceObject,
&Prevent,
sizeof(PREVENT_MEDIA_REMOVAL),
NULL,
0,
FALSE,
NULL,
NULL );
if ( Irp != NULL ) {
//
// Use our special completion routine which will remove the requirement that
// the caller must be below APC level. All it tells us is that the Irp got
// back, but will not tell us if it was succesful or not. We don't care,
// and there is of course no fallback if the attempt to prevent removal
// doesn't work for some mysterious reason.
//
// Normally, all IO is done at passive level. However, MM needs to be able
// to issue IO with fast mutexes locked down, which raises us to APC. The
// overlying IRP is set up to complete in yet another magical fashion even
// though APCs are disabled, and any IRPage we do in these cases has to do
// the same. Marking media dirty (and toggling eject state) is one.
//
Irp->UserIosb = &Irp->IoStatus;
IoSetCompletionRoutine( Irp,
FatSpecialSyncCompletionRoutine,
&SyncContext,
TRUE,
TRUE,
TRUE );
Status = IoCallDriver( Vcb->TargetDeviceObject, Irp );
if (Status == STATUS_PENDING) {
(VOID) KeWaitForSingleObject( &SyncContext.Event,
Executive,
KernelMode,
FALSE,
NULL );
Status = SyncContext.Iosb.Status;
}
return Status;
}
return STATUS_INSUFFICIENT_RESOURCES;
}
NTSTATUS
FatPerformDevIoCtrl (
IN PIRP_CONTEXT IrpContext,
IN ULONG IoControlCode,
IN PDEVICE_OBJECT Device,
IN PVOID InputBuffer OPTIONAL,
IN ULONG InputBufferLength,
OUT PVOID OutputBuffer OPTIONAL,
IN ULONG OutputBufferLength,
IN BOOLEAN InternalDeviceIoControl,
IN BOOLEAN OverrideVerify,
OUT PIO_STATUS_BLOCK Iosb OPTIONAL
)
/*++
Routine Description:
This routine is called to perform DevIoCtrl functions internally within
the filesystem. We take the status from the driver and return it to our
caller.
Arguments:
IoControlCode - Code to send to driver.
Device - This is the device to send the request to.
OutPutBuffer - Pointer to output buffer.
OutputBufferLength - Length of output buffer above.
InternalDeviceIoControl - Indicates if this is an internal or external
Io control code.
OverrideVerify - Indicates if we should tell the driver not to return
STATUS_VERIFY_REQUIRED for mount and verify.
Iosb - If specified, we return the results of the operation here.
Return Value:
NTSTATUS - Status returned by next lower driver.
--*/
{
NTSTATUS Status;
PIRP Irp;
KEVENT Event;
IO_STATUS_BLOCK LocalIosb;
PIO_STATUS_BLOCK IosbToUse = &LocalIosb;
PAGED_CODE();
UNREFERENCED_PARAMETER( IrpContext );
//
// Check if the user gave us an Iosb.
//
if (ARGUMENT_PRESENT( Iosb )) {
IosbToUse = Iosb;
}
IosbToUse->Status = 0;
IosbToUse->Information = 0;
KeInitializeEvent( &Event, NotificationEvent, FALSE );
Irp = IoBuildDeviceIoControlRequest( IoControlCode,
Device,
InputBuffer,
InputBufferLength,
OutputBuffer,
OutputBufferLength,
InternalDeviceIoControl,
&Event,
IosbToUse );
if (Irp == NULL) {
return STATUS_INSUFFICIENT_RESOURCES;
}
if (OverrideVerify) {
SetFlag( IoGetNextIrpStackLocation( Irp )->Flags, SL_OVERRIDE_VERIFY_VOLUME );
}
Status = IoCallDriver( Device, Irp );
//
// We check for device not ready by first checking Status
// and then if status pending was returned, the Iosb status
// value.
//
if (Status == STATUS_PENDING) {
(VOID) KeWaitForSingleObject( &Event,
Executive,
KernelMode,
FALSE,
(PLARGE_INTEGER)NULL );
Status = IosbToUse->Status;
}
return Status;
}
PMDL
FatBuildZeroMdl (
__in PIRP_CONTEXT IrpContext,
__in ULONG Length
)
/*++
Routine Description:
Create an efficient mdl that describe a given length of zeros. We'll only
use a one page buffer and make a mdl that maps all the pages back to the single
physical page. We'll default to a smaller size buffer down to 1 PAGE if memory
is tight. The caller should check the Mdl->ByteCount to see the true size
Arguments:
Length - The desired length of the zero buffer. We may return less than this
Return Value:
a MDL if successful / NULL if not
--*/
{
PMDL ZeroMdl;
ULONG SavedByteCount;
PPFN_NUMBER Page;
ULONG i;
UNREFERENCED_PARAMETER( IrpContext );
//
// Spin down trying to get an MDL which can describe our operation.
//
while (TRUE) {
ZeroMdl = IoAllocateMdl( FatData.ZeroPage, Length, FALSE, FALSE, NULL );
//
// Throttle ourselves to what we've physically allocated. Note that
// we could have started with an odd multiple of this number. If we
// tried for exactly that size and failed, we're toast.
//
if (ZeroMdl || (Length <= PAGE_SIZE)) {
break;
}
//
// Fallback by half and round down to a page multiple.
//
ASSERT( IrpContext->Vcb->Bpb.BytesPerSector <= PAGE_SIZE );
Length = BlockAlignTruncate( Length / 2, PAGE_SIZE );
if (Length < PAGE_SIZE) {
Length = PAGE_SIZE;
}
}
if (ZeroMdl == NULL) {
return NULL;
}
//
// If we have throttled all the way down, stop and just build a
// simple MDL describing our previous allocation.
//
if (Length == PAGE_SIZE) {
MmBuildMdlForNonPagedPool( ZeroMdl );
return ZeroMdl;
}
//
// Now we will temporarily lock the allocated pages
// only, and then replicate the page frame numbers through
// the entire Mdl to keep writing the same pages of zeros.
//
// It would be nice if Mm exported a way for us to not have
// to pull the Mdl apart and rebuild it ourselves, but this
// is so bizarre a purpose as to be tolerable.
//
SavedByteCount = ZeroMdl->ByteCount;
ZeroMdl->ByteCount = PAGE_SIZE;
MmBuildMdlForNonPagedPool( ZeroMdl );
ZeroMdl->MdlFlags &= ~MDL_SOURCE_IS_NONPAGED_POOL;
ZeroMdl->MdlFlags |= MDL_PAGES_LOCKED;
ZeroMdl->MappedSystemVa = NULL;
ZeroMdl->StartVa = NULL;
ZeroMdl->ByteCount = SavedByteCount;
Page = MmGetMdlPfnArray( ZeroMdl );
for (i = 1; i < (ADDRESS_AND_SIZE_TO_SPAN_PAGES( 0, SavedByteCount )); i++) {
*(Page + i) = *(Page);
}
return ZeroMdl;
}