601 lines
16 KiB
C
601 lines
16 KiB
C
/*++
|
||
|
||
Copyright (c) 1989-2000 Microsoft Corporation
|
||
|
||
Module Name:
|
||
|
||
Read.c
|
||
|
||
Abstract:
|
||
|
||
This module implements the File Read routine for Read called by the
|
||
Fsd/Fsp dispatch drivers.
|
||
|
||
// @@BEGIN_DDKSPLIT
|
||
|
||
Author:
|
||
|
||
Dan Lovinger [DanLo] 22-Sep-1996
|
||
Tom Jolly [tomjolly] 21-Jan-2000
|
||
|
||
Revision History:
|
||
|
||
// @@END_DDKSPLIT
|
||
|
||
--*/
|
||
|
||
#include "UdfProcs.h"
|
||
|
||
//
|
||
// The Bug check file id for this module
|
||
//
|
||
|
||
#define BugCheckFileId (UDFS_BUG_CHECK_READ)
|
||
|
||
//
|
||
// The local debug trace level
|
||
//
|
||
|
||
#define Dbg (UDFS_DEBUG_LEVEL_READ)
|
||
|
||
//
|
||
// VOID
|
||
// SafeZeroMemory (
|
||
// IN PUCHAR At,
|
||
// IN ULONG ByteCount
|
||
// );
|
||
//
|
||
|
||
//
|
||
// This macro just puts a nice little try-except around RtlZeroMemory
|
||
//
|
||
|
||
#define SafeZeroMemory(IC,AT,BYTE_COUNT) { \
|
||
try { \
|
||
RtlZeroMemory( (AT), (BYTE_COUNT) ); \
|
||
} except( EXCEPTION_EXECUTE_HANDLER ) { \
|
||
UdfRaiseStatus( IC, STATUS_INVALID_USER_BUFFER ); \
|
||
} \
|
||
}
|
||
|
||
//
|
||
// Read ahead amount used for normal data files
|
||
//
|
||
|
||
#define READ_AHEAD_GRANULARITY (0x10000)
|
||
|
||
#ifdef ALLOC_PRAGMA
|
||
#pragma alloc_text(PAGE, UdfCommonRead)
|
||
#endif
|
||
|
||
|
||
NTSTATUS
|
||
UdfCommonRead (
|
||
IN PIRP_CONTEXT IrpContext,
|
||
IN PIRP Irp
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This is the common entry point for NtReadFile calls. For synchronous requests,
|
||
CommonRead will complete the request in the current thread. If not
|
||
synchronous the request will be passed to the Fsp if there is a need to
|
||
block.
|
||
|
||
Arguments:
|
||
|
||
Irp - Supplies the Irp to process
|
||
|
||
Return Value:
|
||
|
||
NTSTATUS - The result of this operation.
|
||
|
||
--*/
|
||
|
||
{
|
||
NTSTATUS Status;
|
||
PIO_STACK_LOCATION IrpSp = IoGetCurrentIrpStackLocation( Irp );
|
||
|
||
TYPE_OF_OPEN TypeOfOpen;
|
||
PFCB Fcb;
|
||
PCCB Ccb;
|
||
PVCB Vcb;
|
||
|
||
BOOLEAN Wait;
|
||
ULONG PagingIo;
|
||
ULONG SynchronousIo;
|
||
ULONG NonCachedIo;
|
||
|
||
LONGLONG StartingOffset;
|
||
LONGLONG ByteRange;
|
||
ULONG ByteCount;
|
||
ULONG ReadByteCount;
|
||
ULONG OriginalByteCount;
|
||
|
||
PVOID SystemBuffer, UserBuffer;
|
||
|
||
BOOLEAN ReleaseFile = TRUE;
|
||
BOOLEAN ReleaseVmcbMap = FALSE;
|
||
|
||
PFILE_OBJECT MappingFileObject;
|
||
|
||
UDF_IO_CONTEXT LocalIoContext;
|
||
|
||
PAGED_CODE();
|
||
|
||
//
|
||
// If this is a zero length read then return SUCCESS immediately.
|
||
//
|
||
|
||
if (IrpSp->Parameters.Read.Length == 0) {
|
||
|
||
UdfCompleteRequest( IrpContext, Irp, STATUS_SUCCESS );
|
||
return STATUS_SUCCESS;
|
||
}
|
||
|
||
//
|
||
// Decode the file object and verify we support read on this. It
|
||
// must be a user file, stream file or volume file (for a data disk).
|
||
//
|
||
|
||
TypeOfOpen = UdfDecodeFileObject( IrpSp->FileObject, &Fcb, &Ccb );
|
||
|
||
if ((TypeOfOpen == UnopenedFileObject) || (TypeOfOpen == UserDirectoryOpen)) {
|
||
|
||
UdfCompleteRequest( IrpContext, Irp, STATUS_INVALID_DEVICE_REQUEST );
|
||
return STATUS_INVALID_DEVICE_REQUEST;
|
||
}
|
||
|
||
Vcb = Fcb->Vcb;
|
||
|
||
//
|
||
// Examine our input parameters to determine if this is noncached and/or
|
||
// a paging io operation.
|
||
//
|
||
|
||
Wait = BooleanFlagOn( IrpContext->Flags, IRP_CONTEXT_FLAG_WAIT );
|
||
PagingIo = FlagOn( Irp->Flags, IRP_PAGING_IO );
|
||
NonCachedIo = FlagOn( Irp->Flags, IRP_NOCACHE );
|
||
SynchronousIo = FlagOn( IrpSp->FileObject->Flags, FO_SYNCHRONOUS_IO );
|
||
|
||
//
|
||
// Extract the range of the Io.
|
||
//
|
||
|
||
StartingOffset = IrpSp->Parameters.Read.ByteOffset.QuadPart;
|
||
OriginalByteCount = ByteCount = IrpSp->Parameters.Read.Length;
|
||
|
||
ByteRange = StartingOffset + ByteCount;
|
||
|
||
//
|
||
// Make sure that Dasd access is always non-cached.
|
||
//
|
||
|
||
if (TypeOfOpen == UserVolumeOpen) {
|
||
|
||
NonCachedIo = TRUE;
|
||
}
|
||
|
||
//
|
||
// Acquire the file shared to perform the read. If we are doing paging IO,
|
||
// it may be the case that we would have a deadlock imminent because we may
|
||
// block on shared access, so starve out any exclusive waiters. This requires
|
||
// a degree of caution - we believe that any paging IO bursts will recede and
|
||
// allow the exclusive waiter in.
|
||
//
|
||
|
||
if (PagingIo) {
|
||
|
||
UdfAcquireFileSharedStarveExclusive( IrpContext, Fcb );
|
||
|
||
} else {
|
||
|
||
UdfAcquireFileShared( IrpContext, Fcb );
|
||
}
|
||
|
||
//
|
||
// Use a try-finally to facilitate cleanup.
|
||
//
|
||
|
||
try {
|
||
|
||
//
|
||
// Verify the Fcb. Allow reads if this is a DASD handle that is
|
||
// dismounting the volume.
|
||
//
|
||
|
||
if ((TypeOfOpen != UserVolumeOpen) || (NULL == Ccb) ||
|
||
!FlagOn( Ccb->Flags, CCB_FLAG_DISMOUNT_ON_CLOSE)) {
|
||
|
||
UdfVerifyFcbOperation( IrpContext, Fcb );
|
||
}
|
||
|
||
//
|
||
// If this is a user request then verify the oplock and filelock state.
|
||
//
|
||
|
||
if (TypeOfOpen == UserFileOpen) {
|
||
|
||
//
|
||
// We check whether we can proceed
|
||
// based on the state of the file oplocks.
|
||
//
|
||
|
||
Status = FsRtlCheckOplock( &Fcb->Oplock,
|
||
Irp,
|
||
IrpContext,
|
||
UdfOplockComplete,
|
||
UdfPrePostIrp );
|
||
|
||
//
|
||
// If the result is not STATUS_SUCCESS then the Irp was completed
|
||
// elsewhere.
|
||
//
|
||
|
||
if (Status != STATUS_SUCCESS) {
|
||
|
||
Irp = NULL;
|
||
IrpContext = NULL;
|
||
|
||
try_leave( Status );
|
||
}
|
||
|
||
if (!PagingIo &&
|
||
(Fcb->FileLock != NULL) &&
|
||
!FsRtlCheckLockForReadAccess( Fcb->FileLock, Irp )) {
|
||
|
||
try_leave( Status = STATUS_FILE_LOCK_CONFLICT );
|
||
}
|
||
}
|
||
|
||
|
||
|
||
if ((TypeOfOpen != UserVolumeOpen) || !FlagOn( Ccb->Flags, CCB_FLAG_ALLOW_EXTENDED_DASD_IO )) {
|
||
|
||
//
|
||
// Complete the request if it begins beyond the end of file.
|
||
//
|
||
|
||
if (StartingOffset >= Fcb->FileSize.QuadPart) {
|
||
|
||
try_leave( Status = STATUS_END_OF_FILE );
|
||
}
|
||
|
||
//
|
||
// Truncate the read if it extends beyond the end of the file.
|
||
//
|
||
|
||
if (ByteRange > Fcb->FileSize.QuadPart) {
|
||
|
||
ByteCount = (ULONG) (Fcb->FileSize.QuadPart - StartingOffset);
|
||
ByteRange = Fcb->FileSize.QuadPart;
|
||
}
|
||
|
||
}
|
||
|
||
//
|
||
// Now if the data is embedded in the ICB, map through the metadata
|
||
// stream to retrieve the bytes.
|
||
//
|
||
|
||
if (FlagOn( Fcb->FcbState, FCB_STATE_EMBEDDED_DATA )) {
|
||
|
||
//
|
||
// The metadata stream better be here by now.
|
||
//
|
||
|
||
ASSERT( Vcb->MetadataFcb->FileObject != NULL );
|
||
|
||
//
|
||
// Bias our starting offset by the offset of the ICB in the metadata
|
||
// stream plus the offset of the data bytes in that ICB. Obviously,
|
||
// we aren't doing non-cached IO here.
|
||
//
|
||
|
||
StartingOffset += (BytesFromSectors( Vcb, Fcb->EmbeddedVsn ) + Fcb->EmbeddedOffset);
|
||
MappingFileObject = Vcb->MetadataFcb->FileObject;
|
||
NonCachedIo = FALSE;
|
||
|
||
//
|
||
// Ensure that we're mapping within the range of the metadata stream
|
||
//
|
||
|
||
ASSERT( (StartingOffset + ByteCount) <= Vcb->MetadataFcb->FileSize.QuadPart);
|
||
|
||
} else {
|
||
|
||
//
|
||
// We are mapping through the caller's fileobject
|
||
//
|
||
|
||
MappingFileObject = IrpSp->FileObject;
|
||
}
|
||
|
||
//
|
||
// Handle the non-cached read first.
|
||
//
|
||
|
||
if (NonCachedIo) {
|
||
|
||
//
|
||
// If we have an unaligned transfer then post this request if
|
||
// we can't wait. Unaligned means that the starting offset
|
||
// is not on a sector boundary or the read is not integral
|
||
// sectors.
|
||
//
|
||
|
||
ReadByteCount = SectorAlign( Vcb, ByteCount );
|
||
|
||
if (SectorOffset( Vcb, StartingOffset ) ||
|
||
(ReadByteCount > OriginalByteCount)) {
|
||
|
||
if (!Wait) {
|
||
|
||
UdfRaiseStatus( IrpContext, STATUS_CANT_WAIT );
|
||
}
|
||
|
||
//
|
||
// Make sure we don't overwrite the buffer.
|
||
//
|
||
|
||
ReadByteCount = ByteCount;
|
||
}
|
||
|
||
//
|
||
// Initialize the IoContext for the read.
|
||
// If there is a context pointer, we need to make sure it was
|
||
// allocated and not a stale stack pointer.
|
||
//
|
||
|
||
if (IrpContext->IoContext == NULL ||
|
||
!FlagOn( IrpContext->Flags, IRP_CONTEXT_FLAG_ALLOC_IO )) {
|
||
|
||
//
|
||
// If we can wait, use the context on the stack. Otherwise
|
||
// we need to allocate one.
|
||
//
|
||
|
||
if (Wait) {
|
||
|
||
IrpContext->IoContext = &LocalIoContext;
|
||
ClearFlag( IrpContext->Flags, IRP_CONTEXT_FLAG_ALLOC_IO );
|
||
|
||
} else {
|
||
|
||
IrpContext->IoContext = UdfAllocateIoContext();
|
||
SetFlag( IrpContext->Flags, IRP_CONTEXT_FLAG_ALLOC_IO );
|
||
}
|
||
}
|
||
|
||
RtlZeroMemory( IrpContext->IoContext, sizeof( UDF_IO_CONTEXT ));
|
||
|
||
//
|
||
// Store whether we allocated this context structure in the structure
|
||
// itself.
|
||
//
|
||
|
||
IrpContext->IoContext->AllocatedContext =
|
||
BooleanFlagOn( IrpContext->Flags, IRP_CONTEXT_FLAG_ALLOC_IO );
|
||
|
||
if (Wait) {
|
||
|
||
KeInitializeEvent( &IrpContext->IoContext->SyncEvent,
|
||
NotificationEvent,
|
||
FALSE );
|
||
|
||
} else {
|
||
|
||
IrpContext->IoContext->ResourceThreadId = ExGetCurrentResourceThread();
|
||
IrpContext->IoContext->Resource = Fcb->Resource;
|
||
IrpContext->IoContext->RequestedByteCount = ByteCount;
|
||
}
|
||
|
||
Irp->IoStatus.Information = ReadByteCount;
|
||
|
||
//
|
||
// Call the NonCacheIo routine to perform the actual read.
|
||
//
|
||
|
||
Status = UdfNonCachedRead( IrpContext, Fcb, Ccb, StartingOffset, ReadByteCount );
|
||
|
||
//
|
||
// Don't complete this request now if STATUS_PENDING was returned.
|
||
//
|
||
|
||
if (Status == STATUS_PENDING) {
|
||
|
||
Irp = NULL;
|
||
ReleaseFile = FALSE;
|
||
|
||
//
|
||
// Test is we should zero part of the buffer or update the
|
||
// synchronous file position.
|
||
//
|
||
|
||
} else {
|
||
|
||
//
|
||
// Convert any unknown error code to IO_ERROR.
|
||
//
|
||
|
||
if (!NT_SUCCESS( Status )) {
|
||
|
||
//
|
||
// Set the information field to zero.
|
||
//
|
||
|
||
Irp->IoStatus.Information = 0;
|
||
|
||
//
|
||
// Raise if this is a user induced error.
|
||
//
|
||
|
||
if (IoIsErrorUserInduced( Status )) {
|
||
|
||
UdfRaiseStatus( IrpContext, Status );
|
||
}
|
||
|
||
Status = FsRtlNormalizeNtstatus( Status, STATUS_UNEXPECTED_IO_ERROR );
|
||
|
||
//
|
||
// Check if there is any portion of the user's buffer to zero.
|
||
//
|
||
|
||
} else if (ReadByteCount != ByteCount) {
|
||
|
||
UdfMapUserBuffer( IrpContext, &UserBuffer );
|
||
|
||
SafeZeroMemory( IrpContext,
|
||
Add2Ptr( UserBuffer,
|
||
ByteCount,
|
||
PVOID ),
|
||
ReadByteCount - ByteCount );
|
||
|
||
Irp->IoStatus.Information = ByteCount;
|
||
}
|
||
|
||
//
|
||
// Update the file position if this is a synchronous request.
|
||
//
|
||
|
||
if (SynchronousIo && !PagingIo && NT_SUCCESS( Status )) {
|
||
|
||
IrpSp->FileObject->CurrentByteOffset.QuadPart = ByteRange;
|
||
}
|
||
}
|
||
|
||
try_leave( NOTHING );
|
||
}
|
||
|
||
//
|
||
// Handle the cached case. Start by initializing the private
|
||
// cache map.
|
||
//
|
||
|
||
if (MappingFileObject->PrivateCacheMap == NULL) {
|
||
|
||
//
|
||
// The metadata Fcb stream was fired up before any data read. We should never
|
||
// see it here.
|
||
//
|
||
|
||
ASSERT( MappingFileObject != Vcb->MetadataFcb->FileObject );
|
||
|
||
//
|
||
// Now initialize the cache map.
|
||
//
|
||
|
||
CcInitializeCacheMap( IrpSp->FileObject,
|
||
(PCC_FILE_SIZES) &Fcb->AllocationSize,
|
||
FALSE,
|
||
&UdfData.CacheManagerCallbacks,
|
||
Fcb );
|
||
|
||
CcSetReadAheadGranularity( IrpSp->FileObject, READ_AHEAD_GRANULARITY );
|
||
}
|
||
|
||
//
|
||
// Read from the cache if this is not an Mdl read.
|
||
//
|
||
|
||
if (!FlagOn( IrpContext->MinorFunction, IRP_MN_MDL )) {
|
||
|
||
//
|
||
// If we are in the Fsp now because we had to wait earlier,
|
||
// we must map the user buffer, otherwise we can use the
|
||
// user's buffer directly.
|
||
//
|
||
|
||
UdfMapUserBuffer( IrpContext, &SystemBuffer);
|
||
|
||
//
|
||
// Now try to do the copy.
|
||
//
|
||
|
||
if (MappingFileObject == Vcb->MetadataFcb->FileObject) {
|
||
|
||
UdfAcquireVmcbForCcMap( IrpContext, Vcb);
|
||
ReleaseVmcbMap = TRUE;
|
||
}
|
||
|
||
if (!CcCopyRead( MappingFileObject,
|
||
(PLARGE_INTEGER) &StartingOffset,
|
||
ByteCount,
|
||
Wait,
|
||
SystemBuffer,
|
||
&Irp->IoStatus )) {
|
||
|
||
try_leave( Status = STATUS_CANT_WAIT );
|
||
}
|
||
|
||
//
|
||
// If the call didn't succeed, raise the error status
|
||
//
|
||
|
||
if (!NT_SUCCESS( Irp->IoStatus.Status )) {
|
||
|
||
UdfNormalizeAndRaiseStatus( IrpContext, Irp->IoStatus.Status );
|
||
}
|
||
|
||
Status = Irp->IoStatus.Status;
|
||
|
||
//
|
||
// Otherwise perform the MdlRead operation.
|
||
//
|
||
|
||
} else {
|
||
|
||
CcMdlRead( MappingFileObject,
|
||
(PLARGE_INTEGER) &StartingOffset,
|
||
ByteCount,
|
||
&Irp->MdlAddress,
|
||
&Irp->IoStatus );
|
||
|
||
Status = Irp->IoStatus.Status;
|
||
}
|
||
|
||
//
|
||
// Update the current file position in the user file object.
|
||
//
|
||
|
||
if (SynchronousIo && !PagingIo && NT_SUCCESS( Status )) {
|
||
|
||
IrpSp->FileObject->CurrentByteOffset.QuadPart = ByteRange;
|
||
}
|
||
|
||
} finally {
|
||
|
||
DebugUnwind( "UdfCommonRead" );
|
||
|
||
//
|
||
// Release the Fcb / Vmcb mapping resource
|
||
//
|
||
|
||
if (ReleaseFile) { UdfReleaseFile( IrpContext, Fcb ); }
|
||
|
||
if (ReleaseVmcbMap) { UdfReleaseVmcb( IrpContext, Vcb); }
|
||
}
|
||
|
||
//
|
||
// Post the request if we got CANT_WAIT.
|
||
//
|
||
|
||
if (Status == STATUS_CANT_WAIT) {
|
||
|
||
Status = UdfFsdPostRequest( IrpContext, Irp );
|
||
|
||
//
|
||
// Otherwise complete the request.
|
||
//
|
||
|
||
} else {
|
||
|
||
UdfCompleteRequest( IrpContext, Irp, Status );
|
||
}
|
||
|
||
return Status;
|
||
}
|
||
|