2020-09-30 17:17:25 +02:00

319 lines
8.8 KiB
C

/*++
Copyright (c) 2000-2001 Microsoft Corporation
Module Name:
read.c
Abstract:
This module implements routines related to handling IRP_MJ_READ.
--*/
#include "gdfx.h"
NTSTATUS
GdfxNonCachedReadCompletion(
IN PDEVICE_OBJECT DeviceObject,
IN PIRP Irp,
IN PVOID Context
)
/*++
Routine Description:
This routine is called to process the completion of a non-cached file read.
Arguments:
DeviceObject - Specifies the device object that the I/O request is for.
Irp - Specifies the packet that describes the I/O request.
Context - Specifies the context that was supplied to IoSetCompletionRoutine.
Return Value:
Status of operation.
--*/
{
PIO_STACK_LOCATION IrpSp;
PFILE_OBJECT FileObject;
ULONG ReadLength;
//
// This completion routine is set to be called only on IRP success.
//
ASSERT(NT_SUCCESS(Irp->IoStatus.Status));
IrpSp = IoGetCurrentIrpStackLocation(Irp);
FileObject = IrpSp->FileObject;
ReadLength = IrpSp->Parameters.Read.Length;
//
// Assert that the device handled as many bytes as we programmed it to.
//
ASSERT(Irp->IoStatus.Information == GDF_CD_SECTOR_ALIGN_UP(ReadLength));
//
// Propagate the pending flag up the IRP stack.
//
if (Irp->PendingReturned) {
IoMarkIrpPending(Irp);
}
//
// Fix the number of bytes read to the number that we computed inside
// GdfxFsdRead. This number may be less than the actual number of bytes
// read from the device if we're at the end of file.
//
Irp->IoStatus.Information = ReadLength;
//
// If the file is open for synchronous I/O, then we need to update the
// current file position.
//
if (GdfxIsFlagSet(FileObject->Flags, FO_SYNCHRONOUS_IO)) {
FileObject->CurrentByteOffset.QuadPart =
IrpSp->Parameters.Read.ByteOffset.QuadPart + ReadLength;
}
return STATUS_SUCCESS;
}
NTSTATUS
GdfxFsdRead(
IN PDEVICE_OBJECT DeviceObject,
IN PIRP Irp
)
/*++
Routine Description:
This routine is called by the I/O manager to handle IRP_MJ_READ requests.
Arguments:
DeviceObject - Specifies the device object that the I/O request is for.
Irp - Specifies the packet that describes the I/O request.
Return Value:
Status of operation.
--*/
{
NTSTATUS status;
PGDF_VOLUME_EXTENSION VolumeExtension;
PIO_STACK_LOCATION IrpSp;
PIO_STACK_LOCATION NextIrpSp;
PFILE_OBJECT FileObject;
PGDF_FCB Fcb;
ULONG ReadLength;
BOOLEAN NonCachedEndOfFileTransfer;
ULONGLONG PartitionBytesRemaining;
ULONG FileBytesRemaining;
ULONGLONG PhysicalByteOffset;
VolumeExtension = (PGDF_VOLUME_EXTENSION)DeviceObject->DeviceExtension;
IrpSp = IoGetCurrentIrpStackLocation(Irp);
NextIrpSp = IoGetNextIrpStackLocation(Irp);
FileObject = IrpSp->FileObject;
Fcb = (PGDF_FCB)FileObject->FsContext;
ReadLength = IrpSp->Parameters.Read.Length;
NonCachedEndOfFileTransfer = FALSE;
//
// Check if the volume has been dismounted.
//
if (VolumeExtension->Dismounted) {
status = STATUS_VOLUME_DISMOUNTED;
goto CleanupAndExit;
}
//
// Ensure that the file object is not for a directory.
//
if (GdfxIsFlagSet(Fcb->Flags, GDF_FCB_DIRECTORY)) {
status = STATUS_INVALID_DEVICE_REQUEST;
goto CleanupAndExit;
}
//
// If this is a zero length request, then we can complete the IRP now.
//
if (ReadLength == 0) {
Irp->IoStatus.Information = 0;
status = STATUS_SUCCESS;
goto CleanupAndExit;
}
if (GdfxIsFlagSet(Fcb->Flags, GDF_FCB_VOLUME)) {
//
// Check if the starting offset is beyond the end of the volume.
//
if ((ULONGLONG)IrpSp->Parameters.Read.ByteOffset.QuadPart >=
(ULONGLONG)VolumeExtension->PartitionLength.QuadPart) {
status = STATUS_END_OF_FILE;
goto CleanupAndExit;
}
//
// If the number of bytes to read is greater than the number of bytes
// remaining in the file, then truncate the number of bytes we'll actually
// read.
//
PartitionBytesRemaining =
(ULONGLONG)VolumeExtension->PartitionLength.QuadPart -
(ULONGLONG)IrpSp->Parameters.Read.ByteOffset.QuadPart;
if ((ULONGLONG)ReadLength >= PartitionBytesRemaining) {
ReadLength = (ULONG)PartitionBytesRemaining;
}
} else {
//
// Check if the starting offset is beyond the end of file.
//
if ((IrpSp->Parameters.Read.ByteOffset.HighPart != 0) ||
(IrpSp->Parameters.Read.ByteOffset.LowPart >= Fcb->FileSize)) {
status = STATUS_END_OF_FILE;
goto CleanupAndExit;
}
//
// If the number of bytes to read is greater than the number of bytes
// remaining in the file, then truncate the number of bytes we'll actually
// read.
//
FileBytesRemaining = Fcb->FileSize -
IrpSp->Parameters.Read.ByteOffset.LowPart;
if (ReadLength >= FileBytesRemaining) {
//
// If the user's buffer is large enough to hold the logical read
// length rounded up to a sector boundary, then set a flag so that
// the below code will potentially read this part of the file as
// non cached.
//
if (ReadLength >= (GDF_CD_SECTOR_ALIGN_UP(Fcb->FileSize) -
IrpSp->Parameters.Read.ByteOffset.LowPart)) {
NonCachedEndOfFileTransfer = TRUE;
}
ReadLength = FileBytesRemaining;
}
}
ASSERT(ReadLength > 0);
//
// Check if we're supposed to bypass the file system cache.
//
// If this is a physical volume file control block, also bypass the cache.
// This was the behavior of the old file systems and it also let's us avoid
// having to deal with the volume case in the cached path.
//
if (GdfxIsFlagSet(Irp->Flags, IRP_NOCACHE) ||
GdfxIsFlagSet(Fcb->Flags, GDF_FCB_VOLUME)) {
IoCopyCurrentIrpStackLocationToNext(Irp);
//
// Lock the user's buffer into memory if necessary.
//
IoLockUserBuffer(Irp, IrpSp->Parameters.Read.Length);
//
// The file is contiguous on disk, so we can pass this IRP down to the
// target device after adjusting the starting byte offset assuming this
// isn't a volume file control block.
//
if (GdfxIsFlagClear(Fcb->Flags, GDF_FCB_VOLUME)) {
NextIrpSp->Parameters.Read.ByteOffset.QuadPart +=
GdfxSectorToPhysicalByteOffset(Fcb->FirstSector);
}
//
// Limit the number of bytes physically read to the end of the file or
// volume.
//
NextIrpSp->Parameters.Read.Length = GDF_CD_SECTOR_ALIGN_UP(ReadLength);
//
// If the actual number of bytes we're to read is less than the number
// of bytes we'll read from the physical device, then set a completion
// routine to fix IO_STATUS_BLOCK.Information to be the actual number
// of bytes. We'll store the actual number of bytes back in the IRP so
// that we don't have to calculate it again.
//
// If this is a synchronous I/O operation, then set a completion routine
// so that we can update the current file position.
//
// We don't have to do either of the above if the IRP completes with an
// error.
//
if ((IrpSp->Parameters.Read.Length != ReadLength) ||
GdfxIsFlagSet(FileObject->Flags, FO_SYNCHRONOUS_IO)) {
IrpSp->Parameters.Read.Length = ReadLength;
IoSetCompletionRoutine(Irp, GdfxNonCachedReadCompletion, NULL, TRUE,
FALSE, FALSE);
}
//
// Call down to the target device.
//
return IoCallDriver(VolumeExtension->TargetDeviceObject, Irp);
}
//
// Scatter/gather operations are always non-cached.
//
ASSERT(GdfxIsFlagClear(Irp->Flags, IRP_SCATTER_GATHER_OPERATION));
//
// Otherwise, handle reading the data using the file cache.
//
PhysicalByteOffset = IrpSp->Parameters.Read.ByteOffset.QuadPart +
GdfxSectorToPhysicalByteOffset(Fcb->FirstSector);
status = FscCachedRead(&VolumeExtension->CacheExtension, Irp,
PhysicalByteOffset, ReadLength, NonCachedEndOfFileTransfer);
CleanupAndExit:
Irp->IoStatus.Status = status;
IoCompleteRequest(Irp, IO_NO_INCREMENT);
return status;
}