xbox-kernel/private/ntos/io/qsfs.c
2020-09-30 17:17:25 +02:00

234 lines
6.9 KiB
C

/*++
Copyright (c) 1989-2001 Microsoft Corporation
Module Name:
qsfs.c
Abstract:
This module contains the code to implement the NtQueryVolumeInformationFile
and NtSetVolumeInformationFile system services for the NT I/O system.
--*/
#include "iop.h"
NTSTATUS
NtQueryVolumeInformationFile(
IN HANDLE FileHandle,
OUT PIO_STATUS_BLOCK IoStatusBlock,
OUT PVOID FsInformation,
IN ULONG Length,
IN FS_INFORMATION_CLASS FsInformationClass
)
/*++
Routine Description:
This service returns information about the volume associated with the
FileHandle parameter. The information returned in the buffer is defined
by the FsInformationClass parameter. The legal values for this parameter
are as follows:
o FileFsVolumeInformation
o FileFsSizeInformation
o FileFsDeviceInformation
o FileFsAttributeInformation
Arguments:
FileHandle - Supplies a handle to an open volume, directory, or file
for which information about the volume is returned.
IoStatusBlock - Address of the caller's I/O status block.
FsInformation - Supplies a buffer to receive the requested information
returned about the volume.
Length - Supplies the length, in bytes, of the FsInformation buffer.
FsInformationClass - Specifies the type of information which should be
returned about the volume.
Return Value:
The status returned is the final completion status of the operation.
--*/
{
PIRP irp;
NTSTATUS status;
PFILE_OBJECT fileObject;
PDEVICE_OBJECT deviceObject;
PIO_STACK_LOCATION irpSp;
KEVENT localEvent;
IO_STATUS_BLOCK localIoStatus;
BOOLEAN synchronousIo;
PAGED_CODE();
//
// Ensure that the FsInformationClass parameter is legal for querying
// information about the volume.
//
if ((ULONG) FsInformationClass >= FileFsMaximumInformation ||
IopQueryFsOperationLength[FsInformationClass] == 0) {
return STATUS_INVALID_INFO_CLASS;
}
//
// Finally, ensure that the supplied buffer is large enough to contain
// the information associated with the specified query operation that
// is to be performed.
//
if (Length < (ULONG) IopQueryFsOperationLength[FsInformationClass]) {
return STATUS_INFO_LENGTH_MISMATCH;
}
//
// There were no blatant errors so far, so reference the file object so
// the target device object can be found. Note that if the handle does
// not refer to a file object, or if the caller does not have the required
// access to the file, then it will fail.
//
status = ObReferenceObjectByHandle( FileHandle,
&IoFileObjectType,
(PVOID *) &fileObject);
if (!NT_SUCCESS( status )) {
return status;
}
if ((IopQueryFsOperationAccess[FsInformationClass] & FILE_READ_DATA) &&
!fileObject->ReadAccess) {
ObDereferenceObject( fileObject );
return STATUS_ACCESS_DENIED;
}
//
// Make a special check here to determine whether this is a synchronous
// I/O operation. If it is, then wait here until the file is owned by
// the current thread. If this is not a (serialized) synchronous I/O
// operation, then allocate and initialize the local event.
//
if (fileObject->Flags & FO_SYNCHRONOUS_IO) {
IopAcquireFileObjectLock(fileObject);
synchronousIo = TRUE;
} else {
synchronousIo = FALSE;
}
//
// Set the file object to the Not-Signaled state.
//
KeClearEvent( &fileObject->Event );
//
// Get a pointer to the device object for the target device.
//
deviceObject = fileObject->DeviceObject;
//
// If this I/O operation is not being performed as synchronous I/O,
// then allocate an event that will be used to synchronize the
// completion of this operation. That is, this system service is
// a synchronous API being invoked for a file that is opened for
// asynchronous I/O.
//
if (!(fileObject->Flags & FO_SYNCHRONOUS_IO)) {
KeInitializeEvent( &localEvent, SynchronizationEvent, FALSE );
}
//
// Allocate and initialize the I/O Request Packet (IRP) for this
// operation. The allocation is performed with an exception handler
// in case the caller does not have enough quota to allocate the packet.
irp = IoAllocateIrp( deviceObject->StackSize );
if (!irp) {
//
// An IRP could not be allocated. Cleanup and return an
// appropriate error status code.
//
return IopAllocateIrpCleanup( fileObject, (PKEVENT) NULL );
}
irp->Tail.Overlay.OriginalFileObject = fileObject;
irp->Tail.Overlay.Thread = PsGetCurrentThread();
//
// Fill in the service independent parameters in the IRP.
//
irp->UserBuffer = FsInformation;
if (synchronousIo) {
irp->UserIosb = IoStatusBlock;
} else {
irp->UserEvent = &localEvent;
irp->UserIosb = &localIoStatus;
irp->Flags = IRP_SYNCHRONOUS_API;
}
irp->Flags |= IRP_DEFER_IO_COMPLETION;
//
// Get a pointer to the stack location for the first driver. This will
// be used to pass the original function codes and parameters.
//
irpSp = IoGetNextIrpStackLocation( irp );
irpSp->MajorFunction = IRP_MJ_QUERY_VOLUME_INFORMATION;
irpSp->FileObject = fileObject;
//
// Copy the caller's parameters to the service-specific portion of the
// IRP.
//
irpSp->Parameters.QueryVolume.Length = Length;
irpSp->Parameters.QueryVolume.FsInformationClass = FsInformationClass;
//
// Queue the packet, call the driver, and synchronize appopriately with
// I/O completion.
//
status = IopSynchronousServiceTail( deviceObject,
irp,
fileObject,
TRUE,
synchronousIo );
//
// If the file for this operation was not opened for synchronous I/O, then
// synchronization of completion of the I/O operation has not yet occurred
// since the allocated event must be used for synchronous APIs on files
// opened for asynchronous I/O. Synchronize the completion of the I/O
// operation now.
//
if (!synchronousIo) {
status = IopSynchronousApiServiceTail( status,
&localEvent,
irp,
&localIoStatus,
IoStatusBlock );
}
return status;
}