reactos/ntoskrnl/io/iomgr/rawfs.c

1276 lines
38 KiB
C

/*
* PROJECT: ReactOS Kernel
* LICENSE: GPL - See COPYING in the top level directory
* FILE: ntoskrnl/io/iomgr/rawfs.c
* PURPOSE: Raw File System Driver
* PROGRAMMERS: Alex Ionescu (alex.ionescu@reactos.org)
*/
/* INCLUDES *****************************************************************/
#include <ntoskrnl.h>
#define NDEBUG
#include <debug.h>
/* TYPES *******************************************************************/
typedef struct _VCB
{
USHORT NodeTypeCode;
USHORT NodeByteSize;
PDEVICE_OBJECT TargetDeviceObject;
PVPB Vpb;
PVPB LocalVpb;
ULONG VcbState;
KMUTEX Mutex;
CLONG OpenCount;
SHARE_ACCESS ShareAccess;
ULONG BytesPerSector;
LARGE_INTEGER SectorsOnDisk;
} VCB, *PVCB;
typedef struct _VOLUME_DEVICE_OBJECT
{
DEVICE_OBJECT DeviceObject;
VCB Vcb;
} VOLUME_DEVICE_OBJECT, *PVOLUME_DEVICE_OBJECT;
#define VCB_STATE_LOCKED 0x00000001
#define VCB_STATE_DISMOUNTED 0x00000002
/* GLOBALS *******************************************************************/
PDEVICE_OBJECT RawDiskDeviceObject, RawCdromDeviceObject, RawTapeDeviceObject;
/* FUNCTIONS *****************************************************************/
NTSTATUS
NTAPI
RawInitializeVcb(IN OUT PVCB Vcb,
IN PDEVICE_OBJECT TargetDeviceObject,
IN PVPB Vpb)
{
NTSTATUS Status = STATUS_SUCCESS;
PAGED_CODE();
DPRINT("RawInitializeVcb(%p, %p, %p)\n", Vcb, TargetDeviceObject, Vpb);
/* Clear it */
RtlZeroMemory(Vcb, sizeof(VCB));
/* Associate to system objects */
Vcb->TargetDeviceObject = TargetDeviceObject;
Vcb->Vpb = Vpb;
/* Initialize the lock */
KeInitializeMutex(&Vcb->Mutex, 0);
Vcb->LocalVpb = ExAllocatePoolWithTag(NonPagedPool, sizeof(VPB), ' waR');
if (Vcb->LocalVpb == NULL)
{
Status = STATUS_INSUFFICIENT_RESOURCES;
}
return Status;
}
BOOLEAN
NTAPI
RawCheckForDismount(IN PVCB Vcb,
IN BOOLEAN CreateOperation)
{
KIRQL OldIrql;
PVPB Vpb;
BOOLEAN Delete;
DPRINT("RawCheckForDismount(%p, %lu)\n", Vcb, CreateOperation);
ASSERT(KeReadStateMutant(&Vcb->Mutex) == 0);
/* Lock VPB */
IoAcquireVpbSpinLock(&OldIrql);
/* Reference it and check if a create is being done */
Vpb = Vcb->Vpb;
if (Vcb->Vpb->ReferenceCount != CreateOperation)
{
/* Copy the VPB to our local own to prepare later dismount */
if (Vcb->LocalVpb != NULL)
{
RtlZeroMemory(Vcb->LocalVpb, sizeof(VPB));
Vcb->LocalVpb->Type = IO_TYPE_VPB;
Vcb->LocalVpb->Size = sizeof(VPB);
Vcb->LocalVpb->RealDevice = Vcb->Vpb->RealDevice;
Vcb->LocalVpb->DeviceObject = NULL;
Vcb->LocalVpb->Flags = Vcb->Vpb->Flags & VPB_REMOVE_PENDING;
Vcb->Vpb->RealDevice->Vpb = Vcb->LocalVpb;
Vcb->LocalVpb = NULL;
Vcb->Vpb->Flags |= VPB_PERSISTENT;
}
/* Don't do anything */
Delete = FALSE;
}
else
{
/* Otherwise, delete the volume */
Delete = TRUE;
/* Check if it has a VPB and unmount it */
if (Vpb->RealDevice->Vpb == Vpb)
{
Vpb->DeviceObject = NULL;
Vpb->Flags &= ~VPB_MOUNTED;
}
}
/* Release lock and return status */
IoReleaseVpbSpinLock(OldIrql);
/* If we were to delete, delete volume */
if (Delete)
{
PVPB DelVpb;
/* Release our Vcb lock to be able delete us */
KeReleaseMutex(&Vcb->Mutex, 0);
/* If we have a local VPB, we'll have to delete it
* but we won't dismount us - something went bad before
*/
if (Vcb->LocalVpb)
{
DelVpb = Vcb->LocalVpb;
}
/* Otherwise, dismount our device if possible */
else
{
if (Vcb->Vpb->ReferenceCount)
{
ObfDereferenceObject(Vcb->TargetDeviceObject);
IoDeleteDevice((PDEVICE_OBJECT)CONTAINING_RECORD(Vcb,
VOLUME_DEVICE_OBJECT,
Vcb));
return Delete;
}
DelVpb = Vcb->Vpb;
}
/* Delete any of the available VPB and dismount */
ExFreePool(DelVpb);
ObfDereferenceObject(Vcb->TargetDeviceObject);
IoDeleteDevice((PDEVICE_OBJECT)CONTAINING_RECORD(Vcb,
VOLUME_DEVICE_OBJECT,
Vcb));
return Delete;
}
return Delete;
}
NTSTATUS
NTAPI
RawCompletionRoutine(IN PDEVICE_OBJECT DeviceObject,
IN PIRP Irp,
IN PVOID Context)
{
PIO_STACK_LOCATION IoStackLocation = IoGetCurrentIrpStackLocation(Irp);
DPRINT("RawCompletionRoutine(%p, %p, %p)\n", DeviceObject, Irp, Context);
/* Check if this was a valid sync R/W request */
if (((IoStackLocation->MajorFunction == IRP_MJ_READ) ||
(IoStackLocation->MajorFunction == IRP_MJ_WRITE)) &&
((IoStackLocation->FileObject)) &&
(FlagOn(IoStackLocation->FileObject->Flags, FO_SYNCHRONOUS_IO)) &&
(NT_SUCCESS(Irp->IoStatus.Status)))
{
/* Update byte offset */
IoStackLocation->FileObject->CurrentByteOffset.QuadPart +=
Irp->IoStatus.Information;
}
/* Mark the IRP Pending if it was */
if (Irp->PendingReturned) IoMarkIrpPending(Irp);
return STATUS_SUCCESS;
}
NTSTATUS
NTAPI
RawClose(IN PVCB Vcb,
IN PIRP Irp,
IN PIO_STACK_LOCATION IoStackLocation)
{
NTSTATUS Status;
PAGED_CODE();
DPRINT("RawClose(%p, %p, %p)\n", Vcb, Irp, IoStackLocation);
/* If its a stream, not much to do */
if (IoStackLocation->FileObject->Flags & FO_STREAM_FILE)
{
Irp->IoStatus.Status = STATUS_SUCCESS;
IoCompleteRequest(Irp, IO_DISK_INCREMENT);
return STATUS_SUCCESS;
}
/* Make sure we can clean up */
Status = KeWaitForSingleObject(&Vcb->Mutex,
Executive,
KernelMode,
FALSE,
NULL);
ASSERT(NT_SUCCESS(Status));
/* Decrease the open count and check if this is a dismount */
Vcb->OpenCount--;
if (!Vcb->OpenCount || !RawCheckForDismount(Vcb, FALSE))
{
KeReleaseMutex(&Vcb->Mutex, FALSE);
}
/* Complete the request */
Irp->IoStatus.Status = STATUS_SUCCESS;
IoCompleteRequest(Irp, IO_DISK_INCREMENT);
return STATUS_SUCCESS;
}
NTSTATUS
NTAPI
RawCreate(IN PVCB Vcb,
IN PIRP Irp,
IN PIO_STACK_LOCATION IoStackLocation)
{
NTSTATUS Status;
USHORT ShareAccess;
ACCESS_MASK DesiredAccess;
BOOLEAN Deleted = FALSE;
PAGED_CODE();
DPRINT("RawCreate(%p, %p, %p)\n", Vcb, Irp, IoStackLocation);
/* Make sure we can clean up */
Status = KeWaitForSingleObject(&Vcb->Mutex,
Executive,
KernelMode,
FALSE,
NULL);
ASSERT(NT_SUCCESS(Status));
/* Check if this is a valid non-directory file open */
if ((!(IoStackLocation->FileObject) ||
!(IoStackLocation->FileObject->FileName.Length)) &&
((IoStackLocation->Parameters.Create.Options >> 24) == FILE_OPEN) &&
(!(IoStackLocation->Parameters.Create.Options & FILE_DIRECTORY_FILE)))
{
/* Make sure the VCB isn't locked */
if (Vcb->VcbState & VCB_STATE_LOCKED)
{
/* Refuse the operation */
Status = STATUS_ACCESS_DENIED;
Irp->IoStatus.Information = 0;
}
else if (Vcb->VcbState & VCB_STATE_DISMOUNTED)
{
/* Refuse the operation */
Status = STATUS_VOLUME_DISMOUNTED;
Irp->IoStatus.Information = 0;
}
else
{
/* Setup share access */
ShareAccess = IoStackLocation->Parameters.Create.ShareAccess;
DesiredAccess = IoStackLocation->Parameters.Create.
SecurityContext->DesiredAccess;
/* Check if this VCB was already opened */
if (Vcb->OpenCount > 0)
{
/* Try to see if we have access to it */
Status = IoCheckShareAccess(DesiredAccess,
ShareAccess,
IoStackLocation->FileObject,
&Vcb->ShareAccess,
TRUE);
if (!NT_SUCCESS(Status)) Irp->IoStatus.Information = 0;
}
/* Make sure we have access */
if (NT_SUCCESS(Status))
{
/* Check if this is the first open */
if (!Vcb->OpenCount)
{
/* Set the share access */
IoSetShareAccess(DesiredAccess,
ShareAccess,
IoStackLocation->FileObject,
&Vcb->ShareAccess);
}
/* Increase the open count and set the VPB */
Vcb->OpenCount += 1;
IoStackLocation->FileObject->Vpb = Vcb->Vpb;
/* Set IRP status and disable intermediate buffering */
Status = STATUS_SUCCESS;
Irp->IoStatus.Information = FILE_OPENED;
IoStackLocation->FileObject->Flags |=
FO_NO_INTERMEDIATE_BUFFERING;
}
}
}
else
{
/* Invalid create request */
Status = STATUS_INVALID_PARAMETER;
Irp->IoStatus.Information = 0;
}
/* Check if the request failed */
if (!(NT_SUCCESS(Status)) && !(Vcb->OpenCount))
{
/* Check if we can dismount the device */
Deleted = RawCheckForDismount(Vcb, TRUE);
}
/* In case of deletion, the mutex is already released */
if (!Deleted)
{
KeReleaseMutex(&Vcb->Mutex, FALSE);
}
/* Complete the request */
Irp->IoStatus.Status = Status;
IoCompleteRequest(Irp, IO_DISK_INCREMENT);
return Status;
}
NTSTATUS
NTAPI
RawReadWriteDeviceControl(IN PVCB Vcb,
IN PIRP Irp,
IN PIO_STACK_LOCATION IoStackLocation)
{
NTSTATUS Status;
PAGED_CODE();
DPRINT("RawReadWriteDeviceControl(%p, %p, %p)\n", Vcb, Irp, IoStackLocation);
/* Don't do anything if the request was 0 bytes */
if (((IoStackLocation->MajorFunction == IRP_MJ_READ) ||
(IoStackLocation->MajorFunction == IRP_MJ_WRITE)) &&
!(IoStackLocation->Parameters.Read.Length))
{
/* Complete it */
Irp->IoStatus.Status = STATUS_SUCCESS;
IoCompleteRequest(Irp, IO_DISK_INCREMENT);
return STATUS_SUCCESS;
}
/* Copy the IRP stack location */
IoCopyCurrentIrpStackLocationToNext(Irp);
/* Disable verifies */
IoGetNextIrpStackLocation(Irp)->Flags |= SL_OVERRIDE_VERIFY_VOLUME;
/* Setup a completion routine */
IoSetCompletionRoutine(Irp,
RawCompletionRoutine,
NULL,
TRUE,
TRUE,
TRUE);
/* Call the next driver and exit */
Status = IoCallDriver(Vcb->TargetDeviceObject, Irp);
return Status;
}
NTSTATUS
NTAPI
RawMountVolume(IN PIO_STACK_LOCATION IoStackLocation)
{
NTSTATUS Status;
PDEVICE_OBJECT DeviceObject;
PVOLUME_DEVICE_OBJECT Volume;
PFILE_OBJECT FileObject = NULL;
PAGED_CODE();
DPRINT("RawMountVolume(%p)\n", IoStackLocation);
/* Remember our owner */
DeviceObject = IoStackLocation->Parameters.MountVolume.DeviceObject;
/* Create the volume */
Status = IoCreateDevice(RawDiskDeviceObject->DriverObject,
sizeof(VOLUME_DEVICE_OBJECT) -
sizeof(DEVICE_OBJECT),
NULL,
FILE_DEVICE_DISK_FILE_SYSTEM,
0,
FALSE,
(PDEVICE_OBJECT*)&Volume);
if (!NT_SUCCESS(Status)) return Status;
/* Use highest alignment requirement */
Volume->DeviceObject.AlignmentRequirement = max(DeviceObject->
AlignmentRequirement,
Volume->DeviceObject.
AlignmentRequirement);
/* Setup the VCB */
Status = RawInitializeVcb(&Volume->Vcb,
IoStackLocation->Parameters.MountVolume.DeviceObject,
IoStackLocation->Parameters.MountVolume.Vpb);
if (!NT_SUCCESS(Status))
{
IoDeleteDevice((PDEVICE_OBJECT)Volume);
return Status;
}
/* Set dummy label and serial number */
Volume->Vcb.Vpb->SerialNumber = 0xFFFFFFFF;
Volume->Vcb.Vpb->VolumeLabelLength = 0;
/* Setup the DO */
Volume->Vcb.Vpb->DeviceObject = &Volume->DeviceObject;
Volume->DeviceObject.StackSize = DeviceObject->StackSize + 1;
Volume->DeviceObject.SectorSize = DeviceObject->SectorSize;
Volume->DeviceObject.Flags |= DO_DIRECT_IO;
Volume->DeviceObject.Flags &= ~DO_DEVICE_INITIALIZING;
/* Try to get associated FO (for notification) */
_SEH2_TRY
{
FileObject = IoCreateStreamFileObjectLite(NULL,
&(Volume->DeviceObject));
}
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
{
/* Get the exception code */
Status = _SEH2_GetExceptionCode();
}
_SEH2_END;
/* If failed, delete devive */
if (!NT_SUCCESS(Status))
{
IoDeleteDevice((PDEVICE_OBJECT)Volume);
return Status;
}
/* Increment OpenCount by two to avoid dismount when RawClose() will be called on ObDereferenceObject() */
Volume->Vcb.OpenCount += 2;
/* Notify for sucessful mount */
FsRtlNotifyVolumeEvent(FileObject, FSRTL_VOLUME_MOUNT);
/* Decrease refcount to 0 to make FileObject being released */
ObDereferenceObject(FileObject);
/* It's not open anymore, go back to 0 */
Volume->Vcb.OpenCount -= 2;
return Status;
}
NTSTATUS
NTAPI
RawUserFsCtrl(IN PIO_STACK_LOCATION IoStackLocation,
IN PVCB Vcb)
{
NTSTATUS Status;
PAGED_CODE();
DPRINT("RawUserFsCtrl(%p, %p)\n", IoStackLocation, Vcb);
/* Lock the device */
Status = KeWaitForSingleObject(&Vcb->Mutex,
Executive,
KernelMode,
FALSE,
NULL);
ASSERT(NT_SUCCESS(Status));
/* Check what kind of request this is */
switch (IoStackLocation->Parameters.FileSystemControl.FsControlCode)
{
/* Oplock requests */
case FSCTL_REQUEST_OPLOCK_LEVEL_1:
case FSCTL_REQUEST_OPLOCK_LEVEL_2:
case FSCTL_OPLOCK_BREAK_ACKNOWLEDGE:
case FSCTL_OPLOCK_BREAK_NOTIFY:
/* We don't handle them */
Status = STATUS_NOT_IMPLEMENTED;
break;
/* Lock request */
case FSCTL_LOCK_VOLUME:
/* Make sure we're not locked, and that we're alone */
if (!(Vcb->VcbState & 1) && (Vcb->OpenCount == 1))
{
/* Lock the VCB */
Vcb->VcbState |= 1;
Status = STATUS_SUCCESS;
}
else
{
/* Otherwise, we can't do this */
Status = STATUS_ACCESS_DENIED;
}
break;
/* Unlock request */
case FSCTL_UNLOCK_VOLUME:
/* Make sure we're locked */
if (!(Vcb->VcbState & 1))
{
/* Let caller know we're not */
Status = STATUS_NOT_LOCKED;
}
else
{
/* Unlock the VCB */
Vcb->VcbState &= ~1;
Status = STATUS_SUCCESS;
}
break;
/* Dismount request */
case FSCTL_DISMOUNT_VOLUME:
/* Make sure we're locked */
if (Vcb->VcbState & 1)
{
/* Do nothing, just return success */
Status = STATUS_SUCCESS;
}
else
{
/* We can't dismount, device not locked */
Status = STATUS_ACCESS_DENIED;
}
break;
/* Unknown request */
default:
/* Fail */
Status = STATUS_INVALID_PARAMETER;
break;
}
/* Unlock device */
KeReleaseMutex(&Vcb->Mutex, FALSE);
/* In case of status change, notify */
switch (IoStackLocation->Parameters.FileSystemControl.FsControlCode)
{
case FSCTL_LOCK_VOLUME:
FsRtlNotifyVolumeEvent(IoStackLocation->FileObject, (NT_SUCCESS(Status) ? FSRTL_VOLUME_LOCK : FSRTL_VOLUME_LOCK_FAILED));
break;
case FSCTL_UNLOCK_VOLUME:
if (NT_SUCCESS(Status))
{
FsRtlNotifyVolumeEvent(IoStackLocation->FileObject, FSRTL_VOLUME_UNLOCK);
}
break;
case FSCTL_DISMOUNT_VOLUME:
FsRtlNotifyVolumeEvent(IoStackLocation->FileObject, (NT_SUCCESS(Status) ? FSRTL_VOLUME_DISMOUNT : FSRTL_VOLUME_DISMOUNT_FAILED));
break;
}
return Status;
}
NTSTATUS
NTAPI
RawFileSystemControl(IN PVCB Vcb,
IN PIRP Irp,
IN PIO_STACK_LOCATION IoStackLocation)
{
NTSTATUS Status;
PAGED_CODE();
DPRINT("RawFileSystemControl(%p, %p, %p)\n", Vcb, Irp, IoStackLocation);
/* Check the kinds of FSCTLs that we support */
switch (IoStackLocation->MinorFunction)
{
/* User-mode request */
case IRP_MN_USER_FS_REQUEST:
/* Handle it */
Status = RawUserFsCtrl(IoStackLocation, Vcb);
break;
/* Mount request */
case IRP_MN_MOUNT_VOLUME:
/* Mount the volume */
Status = RawMountVolume(IoStackLocation);
break;
case IRP_MN_VERIFY_VOLUME:
/* We don't do verifies */
Status = STATUS_WRONG_VOLUME;
Vcb->Vpb->RealDevice->Flags &= ~DO_VERIFY_VOLUME;
/* Check if we should delete the device */
if (RawCheckForDismount(Vcb, FALSE))
{
/* Do it */
IoDeleteDevice((PDEVICE_OBJECT)
CONTAINING_RECORD(Vcb,
VOLUME_DEVICE_OBJECT,
Vcb));
}
/* We're done */
break;
/* Invalid request */
default:
/* Fail it */
Status = STATUS_INVALID_DEVICE_REQUEST;
break;
}
/* Complete the request */
Irp->IoStatus.Status = Status;
IoCompleteRequest(Irp, IO_DISK_INCREMENT);
return Status;
}
NTSTATUS
NTAPI
RawQueryInformation(IN PVCB Vcb,
IN PIRP Irp,
IN PIO_STACK_LOCATION IoStackLocation)
{
NTSTATUS Status = STATUS_INVALID_DEVICE_REQUEST;
PULONG Length;
PFILE_POSITION_INFORMATION Buffer;
PAGED_CODE();
DPRINT("RawQueryInformation(%p, %p, %p)\n", Vcb, Irp, IoStackLocation);
/* Get information from the IRP */
Length = &IoStackLocation->Parameters.QueryFile.Length;
Buffer = Irp->AssociatedIrp.SystemBuffer;
/* We only handle this request */
if (IoStackLocation->Parameters.QueryFile.FileInformationClass ==
FilePositionInformation)
{
/* Validate buffer size */
if (*Length < sizeof(FILE_POSITION_INFORMATION))
{
/* Invalid, fail */
Irp->IoStatus.Information = 0;
Status = STATUS_BUFFER_OVERFLOW;
}
else
{
/* Get offset and update length */
Buffer->CurrentByteOffset = IoStackLocation->FileObject->
CurrentByteOffset;
*Length -= sizeof(FILE_POSITION_INFORMATION);
/* Set IRP Status information */
Irp->IoStatus.Information = sizeof(FILE_POSITION_INFORMATION);
Status = STATUS_SUCCESS;
}
}
/* Complete it */
Irp->IoStatus.Status = Status;
IoCompleteRequest(Irp, IO_DISK_INCREMENT);
return Status;
}
NTSTATUS
NTAPI
RawSetInformation(IN PVCB Vcb,
IN PIRP Irp,
IN PIO_STACK_LOCATION IoStackLocation)
{
NTSTATUS Status = STATUS_INVALID_DEVICE_REQUEST;
PFILE_POSITION_INFORMATION Buffer;
PDEVICE_OBJECT DeviceObject;
PAGED_CODE();
DPRINT("RawSetInformation(%p, %p, %p)\n", Vcb, Irp, IoStackLocation);
/* Get information from the IRP */
Buffer = Irp->AssociatedIrp.SystemBuffer;
/* We only handle this request */
if (IoStackLocation->Parameters.QueryFile.FileInformationClass ==
FilePositionInformation)
{
/* Get the DO */
DeviceObject = IoGetRelatedDeviceObject(IoStackLocation->FileObject);
/* Make sure the offset is aligned */
if ((Buffer->CurrentByteOffset.LowPart &
DeviceObject->AlignmentRequirement))
{
/* It's not, fail */
Status = STATUS_INVALID_PARAMETER;
}
else
{
/* Otherwise, set offset */
IoStackLocation->FileObject->CurrentByteOffset = Buffer->
CurrentByteOffset;
/* Set IRP Status information */
Status = STATUS_SUCCESS;
}
}
/* Complete it */
Irp->IoStatus.Status = Status;
IoCompleteRequest(Irp, IO_DISK_INCREMENT);
return Status;
}
NTSTATUS
NTAPI
RawQueryFsVolumeInfo(IN PVCB Vcb,
IN PFILE_FS_VOLUME_INFORMATION Buffer,
IN OUT PULONG Length)
{
PAGED_CODE();
DPRINT("RawQueryFsVolumeInfo(%p, %p, %p)\n", Vcb, Buffer, Length);
/* Clear the buffer and stub it out */
RtlZeroMemory( Buffer, sizeof(FILE_FS_VOLUME_INFORMATION));
Buffer->VolumeSerialNumber = Vcb->Vpb->SerialNumber;
Buffer->SupportsObjects = FALSE;
Buffer->VolumeLabelLength = 0;
/* Return length and success */
*Length -= FIELD_OFFSET(FILE_FS_VOLUME_INFORMATION, VolumeLabel[0]);
return STATUS_SUCCESS;
}
NTSTATUS
NTAPI
RawQueryFsSizeInfo(IN PVCB Vcb,
IN PFILE_FS_SIZE_INFORMATION Buffer,
IN OUT PULONG Length)
{
PIRP Irp;
KEVENT Event;
NTSTATUS Status;
IO_STATUS_BLOCK IoStatusBlock;
PDEVICE_OBJECT RealDevice;
DISK_GEOMETRY DiskGeometry;
PARTITION_INFORMATION PartitionInformation;
BOOLEAN DiskHasPartitions;
PAGED_CODE();
DPRINT("RawQueryFsSizeInfo(%p, %p, %p)\n", Vcb, Buffer, Length);
/* Validate the buffer */
if (*Length < sizeof(FILE_FS_SIZE_INFORMATION))
{
/* Fail */
return STATUS_BUFFER_OVERFLOW;
}
/* Clear the buffer, initialize the event and set the DO */
RtlZeroMemory(Buffer, sizeof(FILE_FS_SIZE_INFORMATION));
KeInitializeEvent(&Event, NotificationEvent, FALSE);
RealDevice = Vcb->Vpb->RealDevice;
/* Build query IRP */
Irp = IoBuildDeviceIoControlRequest(IOCTL_DISK_GET_DRIVE_GEOMETRY,
RealDevice,
NULL,
0,
&DiskGeometry,
sizeof(DISK_GEOMETRY),
FALSE,
&Event,
&IoStatusBlock);
if (!Irp) return STATUS_INSUFFICIENT_RESOURCES;
/* Call driver and check if we're pending */
Status = IoCallDriver(RealDevice, Irp);
if (Status == STATUS_PENDING)
{
/* Wait on driver to finish */
KeWaitForSingleObject(&Event,
Executive,
KernelMode,
FALSE,
NULL);
Status = IoStatusBlock.Status;
}
/* Fail if we couldn't get CHS data */
if (!NT_SUCCESS(Status))
{
*Length = 0;
return Status;
}
/* Check if this is a floppy */
if (FlagOn(RealDevice->Characteristics, FILE_FLOPPY_DISKETTE))
{
/* Floppies don't have partitions */
DiskHasPartitions = FALSE;
}
else
{
/* Setup query IRP */
KeResetEvent(&Event);
Irp = IoBuildDeviceIoControlRequest(IOCTL_DISK_GET_PARTITION_INFO,
RealDevice,
NULL,
0,
&PartitionInformation,
sizeof(PARTITION_INFORMATION),
FALSE,
&Event,
&IoStatusBlock);
if (!Irp) return STATUS_INSUFFICIENT_RESOURCES;
/* Call driver and check if we're pending */
Status = IoCallDriver(RealDevice, Irp);
if (Status == STATUS_PENDING)
{
/* Wait on driver to finish */
KeWaitForSingleObject(&Event,
Executive,
KernelMode,
FALSE,
NULL);
Status = IoStatusBlock.Status;
}
/* If this was an invalid request, then the disk is not partitioned */
if (Status == STATUS_INVALID_DEVICE_REQUEST)
{
DiskHasPartitions = FALSE;
}
else
{
/* Otherwise, it must be */
ASSERT(NT_SUCCESS(Status));
DiskHasPartitions = TRUE;
}
}
/* Set sector data */
Buffer->BytesPerSector = DiskGeometry.BytesPerSector;
Buffer->SectorsPerAllocationUnit = 1;
/* Calculate allocation units */
if (DiskHasPartitions)
{
/* Use partition data */
Buffer->TotalAllocationUnits =
RtlExtendedLargeIntegerDivide(PartitionInformation.PartitionLength,
DiskGeometry.BytesPerSector,
NULL);
}
else
{
/* Use CHS */
Buffer->TotalAllocationUnits =
RtlExtendedIntegerMultiply(DiskGeometry.Cylinders,
DiskGeometry.TracksPerCylinder *
DiskGeometry.SectorsPerTrack);
}
/* Set available units */
Buffer->AvailableAllocationUnits = Buffer->TotalAllocationUnits;
/* Return length and success */
*Length -= sizeof(FILE_FS_SIZE_INFORMATION);
return STATUS_SUCCESS;
}
NTSTATUS
NTAPI
RawQueryFsDeviceInfo(IN PVCB Vcb,
IN PFILE_FS_DEVICE_INFORMATION Buffer,
IN OUT PULONG Length)
{
PAGED_CODE();
DPRINT("RawQueryFsDeviceInfo(%p, %p, %p)\n", Vcb, Buffer, Length);
/* Validate buffer */
if (*Length < sizeof(FILE_FS_DEVICE_INFORMATION))
{
/* Fail */
return STATUS_BUFFER_OVERFLOW;
}
/* Clear buffer and write information */
RtlZeroMemory(Buffer, sizeof(FILE_FS_DEVICE_INFORMATION));
Buffer->DeviceType = FILE_DEVICE_DISK;
Buffer->Characteristics = Vcb->TargetDeviceObject->Characteristics;
/* Return length and success */
*Length -= sizeof(FILE_FS_DEVICE_INFORMATION);
return STATUS_SUCCESS;
}
NTSTATUS
NTAPI
RawQueryFsAttributeInfo(IN PVCB Vcb,
IN PFILE_FS_ATTRIBUTE_INFORMATION Buffer,
IN OUT PULONG Length)
{
const WCHAR szRawFSName[] = L"RAW";
ULONG ReturnLength;
PAGED_CODE();
DPRINT("RawQueryFsAttributeInfo(%p, %p, %p)\n", Vcb, Buffer, Length);
/* Check if the buffer is large enough for our name ("RAW") */
ReturnLength = FIELD_OFFSET(FILE_FS_ATTRIBUTE_INFORMATION,
FileSystemName[sizeof(szRawFSName) / sizeof(szRawFSName[0])]);
if (*Length < ReturnLength) return STATUS_BUFFER_OVERFLOW;
/* Output the data */
Buffer->FileSystemAttributes = 0;
Buffer->MaximumComponentNameLength = 0;
Buffer->FileSystemNameLength = 6;
RtlCopyMemory(&Buffer->FileSystemName[0], szRawFSName, sizeof(szRawFSName));
/* Return length and success */
*Length -= ReturnLength;
return STATUS_SUCCESS;
}
NTSTATUS
NTAPI
RawQueryVolumeInformation(IN PVCB Vcb,
IN PIRP Irp,
IN PIO_STACK_LOCATION IoStackLocation)
{
NTSTATUS Status;
ULONG Length;
PVOID Buffer;
PAGED_CODE();
DPRINT("RawQueryVolumeInformation(%p, %p, %p)\n", Vcb, Irp, IoStackLocation);
/* Get IRP Data */
Length = IoStackLocation->Parameters.QueryVolume.Length;
Buffer = Irp->AssociatedIrp.SystemBuffer;
/* Check the kind of request */
switch (IoStackLocation->Parameters.QueryVolume.FsInformationClass)
{
/* Volume information request */
case FileFsVolumeInformation:
Status = RawQueryFsVolumeInfo(Vcb, Buffer, &Length);
break;
/* File system size invormation */
case FileFsSizeInformation:
Status = RawQueryFsSizeInfo(Vcb, Buffer, &Length);
break;
/* Device information */
case FileFsDeviceInformation:
Status = RawQueryFsDeviceInfo(Vcb, Buffer, &Length);
break;
/* Attribute information */
case FileFsAttributeInformation:
Status = RawQueryFsAttributeInfo(Vcb, Buffer, &Length);
break;
/* Invalid request */
default:
/* Fail it */
Status = STATUS_INVALID_PARAMETER;
break;
}
/* Set status and complete the request */
Irp->IoStatus.Information = IoStackLocation->
Parameters.QueryVolume.Length - Length;
Irp->IoStatus.Status = Status;
IoCompleteRequest(Irp, IO_DISK_INCREMENT);
return Status;
}
NTSTATUS
NTAPI
RawCleanup(IN PVCB Vcb,
IN PIRP Irp,
IN PIO_STACK_LOCATION IoStackLocation)
{
NTSTATUS Status;
PAGED_CODE();
DPRINT("RawCleanup(%p, %p, %p)\n", Vcb, Irp, IoStackLocation);
/* Make sure we can clean up */
Status = KeWaitForSingleObject(&Vcb->Mutex,
Executive,
KernelMode,
FALSE,
NULL);
ASSERT(NT_SUCCESS(Status));
/* Remove shared access */
IoRemoveShareAccess(IoStackLocation->FileObject, &Vcb->ShareAccess);
/* Check if we're to dismount */
if (Vcb->VcbState & VCB_STATE_DISMOUNTED)
{
ASSERT(Vcb->OpenCount == 1);
RawCheckForDismount(Vcb, FALSE);
}
KeReleaseMutex(&Vcb->Mutex, FALSE);
Irp->IoStatus.Status = STATUS_SUCCESS;
IoCompleteRequest(Irp, IO_DISK_INCREMENT);
return STATUS_SUCCESS;
}
NTSTATUS
NTAPI
RawDispatch(IN PDEVICE_OBJECT DeviceObject,
IN PIRP Irp)
{
PVOLUME_DEVICE_OBJECT VolumeDeviceObject = (PVOLUME_DEVICE_OBJECT)DeviceObject;
NTSTATUS Status = STATUS_INVALID_DEVICE_REQUEST;
PIO_STACK_LOCATION IoStackLocation;
PVCB Vcb;
PAGED_CODE();
DPRINT("RawDispatch(%p, %p)\n", DeviceObject, Irp);
/* Get the stack location */
IoStackLocation = IoGetCurrentIrpStackLocation(Irp);
/* Differentiate between Volume DO and FS DO */
if ((DeviceObject->Size == sizeof(DEVICE_OBJECT)) &&
!((IoStackLocation->MajorFunction == IRP_MJ_FILE_SYSTEM_CONTROL) &&
(IoStackLocation->MinorFunction == IRP_MN_MOUNT_VOLUME)))
{
/* This is an FS DO. Stub out the common calls */
if ((IoStackLocation->MajorFunction == IRP_MJ_CREATE) ||
(IoStackLocation->MajorFunction == IRP_MJ_CLEANUP) ||
(IoStackLocation->MajorFunction == IRP_MJ_CLOSE))
{
/* Return success for them */
Status = STATUS_SUCCESS;
}
else
{
/* Anything else, we don't support */
Status = STATUS_INVALID_DEVICE_REQUEST;
}
/* Complete the request */
Irp->IoStatus.Status = Status;
IoCompleteRequest(Irp, IO_DISK_INCREMENT);
return Status;
}
/* Otherwise, get our VCB and start handling the IRP */
FsRtlEnterFileSystem();
Vcb = &VolumeDeviceObject->Vcb;
/* Check what kind of IRP this is */
switch (IoStackLocation->MajorFunction)
{
/* Cleanup request */
case IRP_MJ_CLEANUP:
Status = RawCleanup(Vcb, Irp, IoStackLocation);
break;
/* Close request */
case IRP_MJ_CLOSE:
Status = RawClose(Vcb, Irp, IoStackLocation);
break;
/* Create request */
case IRP_MJ_CREATE:
Status = RawCreate(Vcb, Irp, IoStackLocation);
break;
/* FSCTL request */
case IRP_MJ_FILE_SYSTEM_CONTROL:
Status = RawFileSystemControl(Vcb, Irp, IoStackLocation);
break;
/* R/W or IOCTL request */
case IRP_MJ_READ:
case IRP_MJ_WRITE:
case IRP_MJ_DEVICE_CONTROL:
Status = RawReadWriteDeviceControl(Vcb, Irp, IoStackLocation);
break;
/* Information query request */
case IRP_MJ_QUERY_INFORMATION:
Status = RawQueryInformation(Vcb, Irp, IoStackLocation);
break;
/* Information set request */
case IRP_MJ_SET_INFORMATION:
Status = RawSetInformation(Vcb, Irp, IoStackLocation);
break;
/* Volume information request */
case IRP_MJ_QUERY_VOLUME_INFORMATION:
Status = RawQueryVolumeInformation(Vcb, Irp, IoStackLocation);
break;
/* Unexpected request */
default:
/* Anything else is pretty bad */
KeBugCheck(FILE_SYSTEM);
}
/* Return the status */
FsRtlExitFileSystem();
return Status;
}
NTSTATUS
NTAPI
RawShutdown(IN PDEVICE_OBJECT DeviceObject,
IN PIRP Irp)
{
/* Unregister file systems */
#if 0 // FIXME: This freezes ROS at shutdown. PnP Problem?
IoUnregisterFileSystem(RawDiskDeviceObject);
IoUnregisterFileSystem(RawCdromDeviceObject);
IoUnregisterFileSystem(RawTapeDeviceObject);
/* Delete the devices */
IoDeleteDevice(RawDiskDeviceObject);
IoDeleteDevice(RawCdromDeviceObject);
IoDeleteDevice(RawTapeDeviceObject);
#endif
/* Complete the request */
Irp->IoStatus.Status = STATUS_SUCCESS;
IoCompleteRequest(Irp, IO_DISK_INCREMENT);
return STATUS_SUCCESS;
}
VOID
NTAPI
RawUnload(IN PDRIVER_OBJECT DriverObject)
{
#if 0 // FIXME: DriverUnload is never called
/* Dereference device objects */
ObDereferenceObject(RawDiskDeviceObject);
ObDereferenceObject(RawCdromDeviceObject);
ObDereferenceObject(RawTapeDeviceObject);
#endif
}
NTSTATUS
NTAPI
INIT_FUNCTION
RawFsDriverEntry(IN PDRIVER_OBJECT DriverObject,
IN PUNICODE_STRING RegistryPath)
{
UNICODE_STRING DeviceName;
NTSTATUS Status;
/* Create the raw disk device */
RtlInitUnicodeString(&DeviceName, L"\\Device\\RawDisk");
Status = IoCreateDevice(DriverObject,
0,
NULL,
FILE_DEVICE_DISK_FILE_SYSTEM,
0,
FALSE,
&RawDiskDeviceObject);
if (!NT_SUCCESS(Status)) return Status;
/* Create the raw CDROM device */
RtlInitUnicodeString(&DeviceName, L"\\Device\\RawCdRom");
Status = IoCreateDevice(DriverObject,
0,
NULL,
FILE_DEVICE_CD_ROM_FILE_SYSTEM,
0,
FALSE,
&RawCdromDeviceObject);
if (!NT_SUCCESS(Status)) return Status;
/* Create the raw tape device */
RtlInitUnicodeString(&DeviceName, L"\\Device\\RawTape");
Status = IoCreateDevice(DriverObject,
0,
NULL,
FILE_DEVICE_TAPE_FILE_SYSTEM,
0,
FALSE,
&RawTapeDeviceObject);
if (!NT_SUCCESS(Status)) return Status;
/* Set Direct I/O for all devices */
RawDiskDeviceObject->Flags |= DO_DIRECT_IO;
RawCdromDeviceObject->Flags |= DO_DIRECT_IO;
RawTapeDeviceObject->Flags |= DO_DIRECT_IO;
/* Set generic stubs */
DriverObject->MajorFunction[IRP_MJ_CREATE] =
DriverObject->MajorFunction[IRP_MJ_CLEANUP] =
DriverObject->MajorFunction[IRP_MJ_CLOSE] =
DriverObject->MajorFunction[IRP_MJ_READ] =
DriverObject->MajorFunction[IRP_MJ_WRITE] =
DriverObject->MajorFunction[IRP_MJ_QUERY_INFORMATION] =
DriverObject->MajorFunction[IRP_MJ_SET_INFORMATION] =
DriverObject->MajorFunction[IRP_MJ_QUERY_VOLUME_INFORMATION] =
DriverObject->MajorFunction[IRP_MJ_FILE_SYSTEM_CONTROL] =
DriverObject->MajorFunction[IRP_MJ_DEVICE_CONTROL] = RawDispatch;
/* Shutdown and unload */
DriverObject->MajorFunction[IRP_MJ_SHUTDOWN] = RawShutdown;
DriverObject->DriverUnload = RawUnload;
/* Register the file systems */
IoRegisterFileSystem(RawDiskDeviceObject);
IoRegisterFileSystem(RawCdromDeviceObject);
IoRegisterFileSystem(RawTapeDeviceObject);
#if 0 // FIXME: DriverUnload is never called
/* Reference device objects */
ObReferenceObject(RawDiskDeviceObject);
ObReferenceObject(RawCdromDeviceObject);
ObReferenceObject(RawTapeDeviceObject);
#endif
return STATUS_SUCCESS;
}
/* EOF */