mirror of
https://github.com/reactos/reactos.git
synced 2025-01-11 08:38:17 +00:00
912ce51ae6
Sync with trunk head (r48826) svn path=/branches/cmake-bringup/; revision=48831
1153 lines
34 KiB
C
Executable file
1153 lines
34 KiB
C
Executable file
/*
|
|
* 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;
|
|
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;
|
|
|
|
/* GLOBALS *******************************************************************/
|
|
|
|
PDEVICE_OBJECT RawDiskDeviceObject, RawCdromDeviceObject, RawTapeDeviceObject;
|
|
|
|
/* FUNCTIONS *****************************************************************/
|
|
|
|
VOID
|
|
NTAPI
|
|
RawInitializeVcb(IN OUT PVCB Vcb,
|
|
IN PDEVICE_OBJECT TargetDeviceObject,
|
|
IN PVPB Vpb)
|
|
{
|
|
PAGED_CODE();
|
|
|
|
/* Clear it */
|
|
RtlZeroMemory(Vcb, sizeof(VCB));
|
|
|
|
/* Associate to system objects */
|
|
Vcb->TargetDeviceObject = TargetDeviceObject;
|
|
Vcb->Vpb = Vpb;
|
|
|
|
/* Initialize the lock */
|
|
KeInitializeMutex(&Vcb->Mutex, 0);
|
|
}
|
|
|
|
BOOLEAN
|
|
NTAPI
|
|
RawCheckForDismount(IN PVCB Vcb,
|
|
IN BOOLEAN CreateOperation)
|
|
{
|
|
KIRQL OldIrql;
|
|
PVPB Vpb;
|
|
BOOLEAN Delete;
|
|
|
|
/* Lock VPB */
|
|
IoAcquireVpbSpinLock(&OldIrql);
|
|
|
|
/* Reference it and check if a create is being done */
|
|
Vpb = Vcb->Vpb;
|
|
if (Vcb->Vpb->ReferenceCount != CreateOperation)
|
|
{
|
|
/* 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);
|
|
return Delete;
|
|
}
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
RawCompletionRoutine(IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp,
|
|
IN PVOID Context)
|
|
{
|
|
PIO_STACK_LOCATION IoStackLocation = IoGetCurrentIrpStackLocation(Irp);
|
|
|
|
/* 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;
|
|
BOOLEAN Deleted = FALSE;
|
|
PAGED_CODE();
|
|
|
|
/* 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) Deleted = RawCheckForDismount(Vcb, FALSE);
|
|
|
|
/* Check if we should delete the device */
|
|
KeReleaseMutex(&Vcb->Mutex, FALSE);
|
|
if (Deleted)
|
|
{
|
|
/* Delete it */
|
|
IoDeleteDevice((PDEVICE_OBJECT)CONTAINING_RECORD(Vcb,
|
|
VOLUME_DEVICE_OBJECT,
|
|
Vcb));
|
|
}
|
|
|
|
/* 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;
|
|
BOOLEAN Deleted = FALSE;
|
|
USHORT ShareAccess;
|
|
ACCESS_MASK DesiredAccess;
|
|
PAGED_CODE();
|
|
|
|
/* 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 & 1)
|
|
{
|
|
/* Refuse the operation */
|
|
Status = STATUS_ACCESS_DENIED;
|
|
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, FALSE);
|
|
}
|
|
|
|
/* Check if we should delete the device */
|
|
KeReleaseMutex(&Vcb->Mutex, FALSE);
|
|
if (Deleted)
|
|
{
|
|
/* Delete it */
|
|
IoDeleteDevice((PDEVICE_OBJECT)CONTAINING_RECORD(Vcb,
|
|
VOLUME_DEVICE_OBJECT,
|
|
Vcb));
|
|
}
|
|
|
|
/* Complete the request */
|
|
Irp->IoStatus.Status = STATUS_SUCCESS;
|
|
IoCompleteRequest(Irp, IO_DISK_INCREMENT);
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
RawReadWriteDeviceControl(IN PVCB Vcb,
|
|
IN PIRP Irp,
|
|
IN PIO_STACK_LOCATION IoStackLocation)
|
|
{
|
|
NTSTATUS Status;
|
|
PAGED_CODE();
|
|
|
|
/* 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();
|
|
|
|
/* 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 */
|
|
RawInitializeVcb(&Volume->Vcb,
|
|
IoStackLocation->Parameters.MountVolume.DeviceObject,
|
|
IoStackLocation->Parameters.MountVolume.Vpb);
|
|
|
|
/* 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();
|
|
|
|
/* 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();
|
|
|
|
/* 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();
|
|
|
|
/* 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();
|
|
|
|
/* 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();
|
|
|
|
/* 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();
|
|
|
|
/* 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();
|
|
|
|
/* 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();
|
|
|
|
/* 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();
|
|
|
|
/* 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();
|
|
|
|
/* Make sure we can clean up */
|
|
Status = KeWaitForSingleObject(&Vcb->Mutex,
|
|
Executive,
|
|
KernelMode,
|
|
FALSE,
|
|
NULL);
|
|
ASSERT(NT_SUCCESS(Status));
|
|
|
|
/* Remove shared access and complete the request */
|
|
IoRemoveShareAccess(IoStackLocation->FileObject, &Vcb->ShareAccess);
|
|
KeReleaseMutex(&Vcb->Mutex, FALSE);
|
|
Irp->IoStatus.Status = STATUS_SUCCESS;
|
|
IoCompleteRequest(Irp, IO_DISK_INCREMENT);
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
RawDispatch(IN PVOLUME_DEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp)
|
|
{
|
|
NTSTATUS Status = STATUS_INVALID_DEVICE_REQUEST;
|
|
PIO_STACK_LOCATION IoStackLocation;
|
|
PVCB Vcb;
|
|
PAGED_CODE();
|
|
|
|
/* Get the stack location */
|
|
IoStackLocation = IoGetCurrentIrpStackLocation(Irp);
|
|
|
|
/* Differentiate between Volume DO and FS DO */
|
|
if ((((PDEVICE_OBJECT)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 = &DeviceObject->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
|
|
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] = (PDRIVER_DISPATCH)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 */
|