mirror of
https://github.com/reactos/reactos.git
synced 2025-01-01 03:54:02 +00:00
0daa5547d9
This implies that a sample for W10. It has been backported to NT5.2; not sure how it would work on a W2K3 (feel free to test!)
3858 lines
104 KiB
C
3858 lines
104 KiB
C
/*++
|
||
|
||
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.
|
||
//
|
||
|
||
#ifndef __REACTOS__
|
||
if (MmGetSystemAddressForMdlSafe( Mdl, NormalPagePriority | MdlMappingNoExecute ) == NULL) {
|
||
#else
|
||
if (MmGetSystemAddressForMdlSafe( Mdl, NormalPagePriority ) == NULL) {
|
||
#endif
|
||
|
||
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
|
||
//
|
||
|
||
#ifndef __REACTOS__
|
||
DiskBuffer = FsRtlAllocatePoolWithTag( NonPagedPoolNxCacheAligned,
|
||
#else
|
||
DiskBuffer = FsRtlAllocatePoolWithTag( NonPagedPoolCacheAligned,
|
||
#endif
|
||
(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....
|
||
//
|
||
|
||
#ifndef __REACTOS__
|
||
Packet = ExAllocatePoolWithTag(NonPagedPoolNx, sizeof(CLEAN_AND_DIRTY_VOLUME_PACKET), ' taF');
|
||
#else
|
||
Packet = ExAllocatePoolWithTag(NonPagedPool, sizeof(CLEAN_AND_DIRTY_VOLUME_PACKET), ' taF');
|
||
#endif
|
||
|
||
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 {
|
||
|
||
#ifndef __REACTOS__
|
||
PVOID Address = MmGetSystemAddressForMdlSafe( Irp->MdlAddress, NormalPagePriority | MdlMappingNoExecute );
|
||
#else
|
||
PVOID Address = MmGetSystemAddressForMdlSafe( Irp->MdlAddress, NormalPagePriority );
|
||
#endif
|
||
|
||
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 );
|
||
|
||
#ifndef __REACTOS__
|
||
Irp->AssociatedIrp.SystemBuffer = FsRtlAllocatePoolWithQuotaTag( NonPagedPoolNx,
|
||
#else
|
||
Irp->AssociatedIrp.SystemBuffer = FsRtlAllocatePoolWithQuotaTag( NonPagedPool,
|
||
#endif
|
||
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 bizzare 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;
|
||
}
|
||
|
||
|