mirror of
https://github.com/reactos/reactos.git
synced 2025-01-03 21:09:19 +00:00
217 lines
6.1 KiB
C
217 lines
6.1 KiB
C
/*
|
|
* PROJECT: ReactOS Kernel
|
|
* LICENSE: GPL-2.0+ (https://spdx.org/licenses/GPL-2.0+)
|
|
* PURPOSE: Null Device Driver
|
|
* COPYRIGHT: Copyright 1998-2018 David Welch (welch@mcmail.com)
|
|
* Copyright 2007-2018 Alex Ionescu (alex.ionescu@reactos.org)
|
|
*/
|
|
|
|
/* INCLUDES ******************************************************************/
|
|
|
|
#include <wdm.h>
|
|
|
|
/* GLOBALS *******************************************************************/
|
|
|
|
FAST_IO_DISPATCH FastIoDispatch;
|
|
|
|
/* FUNCTIONS *****************************************************************/
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
NullQueryFileInformation(OUT PVOID Buffer,
|
|
IN PULONG Length,
|
|
IN FILE_INFORMATION_CLASS InformationClass)
|
|
{
|
|
PFILE_STANDARD_INFORMATION StandardInfo = Buffer;
|
|
|
|
PAGED_CODE();
|
|
|
|
/* We only support one class */
|
|
if (InformationClass != FileStandardInformation)
|
|
{
|
|
/* Fail */
|
|
return STATUS_INVALID_INFO_CLASS;
|
|
}
|
|
|
|
/* Fill out the information */
|
|
RtlZeroMemory(StandardInfo, sizeof(FILE_STANDARD_INFORMATION));
|
|
StandardInfo->NumberOfLinks = 1;
|
|
|
|
/* Return the length and success */
|
|
*Length = sizeof(FILE_STANDARD_INFORMATION);
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
BOOLEAN
|
|
NTAPI
|
|
NullRead(IN PFILE_OBJECT FileObject,
|
|
IN PLARGE_INTEGER FileOffset,
|
|
IN ULONG Length,
|
|
IN BOOLEAN Wait,
|
|
IN ULONG LockKey,
|
|
OUT PVOID Buffer,
|
|
OUT PIO_STATUS_BLOCK IoStatus,
|
|
IN PDEVICE_OBJECT DeviceObject)
|
|
{
|
|
PAGED_CODE();
|
|
|
|
/* Complete successfully */
|
|
IoStatus->Status = STATUS_END_OF_FILE;
|
|
IoStatus->Information = 0;
|
|
return TRUE;
|
|
}
|
|
|
|
BOOLEAN
|
|
NTAPI
|
|
NullWrite(IN PFILE_OBJECT FileObject,
|
|
IN PLARGE_INTEGER FileOffset,
|
|
IN ULONG Length,
|
|
IN BOOLEAN Wait,
|
|
IN ULONG LockKey,
|
|
IN PVOID Buffer,
|
|
OUT PIO_STATUS_BLOCK IoStatus,
|
|
IN PDEVICE_OBJECT DeviceObject)
|
|
{
|
|
PAGED_CODE();
|
|
|
|
/* Complete successfully */
|
|
IoStatus->Status = STATUS_SUCCESS;
|
|
IoStatus->Information = Length;
|
|
return TRUE;
|
|
}
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
NullDispatch(IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp)
|
|
{
|
|
NTSTATUS Status;
|
|
PIO_STACK_LOCATION IoStack = IoGetCurrentIrpStackLocation(Irp);
|
|
PFILE_OBJECT FileObject;
|
|
ULONG Length;
|
|
|
|
PAGED_CODE();
|
|
|
|
/* Get the file object and check what kind of request this is */
|
|
FileObject = IoStack->FileObject;
|
|
switch (IoStack->MajorFunction)
|
|
{
|
|
case IRP_MJ_CREATE:
|
|
case IRP_MJ_CLOSE:
|
|
|
|
/* Check if this is synch I/O */
|
|
if (FileObject->Flags & FO_SYNCHRONOUS_IO)
|
|
{
|
|
/* Set distinguished value for Cc */
|
|
FileObject->PrivateCacheMap = (PVOID)1;
|
|
}
|
|
|
|
/* Complete successfully */
|
|
Irp->IoStatus.Status = STATUS_SUCCESS;
|
|
Irp->IoStatus.Information = 0;
|
|
break;
|
|
|
|
case IRP_MJ_READ:
|
|
|
|
/* Return as if we read the entire file */
|
|
Irp->IoStatus.Status = STATUS_END_OF_FILE;
|
|
Irp->IoStatus.Information = 0;
|
|
break;
|
|
|
|
case IRP_MJ_WRITE:
|
|
|
|
/* Return as if we wrote the entire request */
|
|
Irp->IoStatus.Status = STATUS_SUCCESS;
|
|
Irp->IoStatus.Information = IoStack->Parameters.Write.Length;
|
|
break;
|
|
|
|
case IRP_MJ_LOCK_CONTROL:
|
|
|
|
/* Dummy */
|
|
Irp->IoStatus.Status = STATUS_SUCCESS;
|
|
Irp->IoStatus.Information = 0;
|
|
break;
|
|
|
|
case IRP_MJ_QUERY_INFORMATION:
|
|
|
|
/* Get the length inputted and do the request */
|
|
Length = IoStack->Parameters.QueryFile.Length;
|
|
Irp->IoStatus.Status = NullQueryFileInformation(Irp->AssociatedIrp.
|
|
SystemBuffer,
|
|
&Length,
|
|
IoStack->
|
|
Parameters.
|
|
QueryFile.
|
|
FileInformationClass);
|
|
|
|
/* Return the actual length */
|
|
Irp->IoStatus.Information = Length;
|
|
break;
|
|
}
|
|
|
|
/* Complete the request */
|
|
Status = Irp->IoStatus.Status;
|
|
IoCompleteRequest(Irp, IO_NO_INCREMENT);
|
|
return Status;
|
|
}
|
|
|
|
VOID
|
|
NTAPI
|
|
NullUnload(IN PDRIVER_OBJECT DriverObject)
|
|
{
|
|
PDEVICE_OBJECT DeviceObject = DriverObject->DeviceObject;
|
|
|
|
/* Delete the Null device */
|
|
IoDeleteDevice(DeviceObject);
|
|
}
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
DriverEntry(IN PDRIVER_OBJECT DriverObject,
|
|
IN PUNICODE_STRING RegistryPath)
|
|
{
|
|
NTSTATUS Status;
|
|
PDEVICE_OBJECT DeviceObject;
|
|
UNICODE_STRING DeviceName = RTL_CONSTANT_STRING(L"\\Device\\Null");
|
|
|
|
PAGED_CODE();
|
|
|
|
UNREFERENCED_PARAMETER(RegistryPath);
|
|
|
|
/* Page the driver */
|
|
MmPageEntireDriver(DriverEntry);
|
|
|
|
/* Create the Null device */
|
|
Status = IoCreateDevice(DriverObject,
|
|
0,
|
|
&DeviceName,
|
|
FILE_DEVICE_NULL,
|
|
FILE_DEVICE_SECURE_OPEN,
|
|
FALSE,
|
|
&DeviceObject);
|
|
if (!NT_SUCCESS(Status))
|
|
return Status;
|
|
|
|
/* Register driver routines */
|
|
DriverObject->MajorFunction[IRP_MJ_CLOSE] = NullDispatch;
|
|
DriverObject->MajorFunction[IRP_MJ_CREATE] = NullDispatch;
|
|
DriverObject->MajorFunction[IRP_MJ_WRITE] = NullDispatch;
|
|
DriverObject->MajorFunction[IRP_MJ_READ] = NullDispatch;
|
|
DriverObject->MajorFunction[IRP_MJ_LOCK_CONTROL] = NullDispatch;
|
|
DriverObject->MajorFunction[IRP_MJ_QUERY_INFORMATION] = NullDispatch;
|
|
DriverObject->DriverUnload = NullUnload;
|
|
|
|
/* Initialize the fast I/O dispatch table */
|
|
RtlZeroMemory(&FastIoDispatch, sizeof(FastIoDispatch));
|
|
FastIoDispatch.SizeOfFastIoDispatch = sizeof(FastIoDispatch);
|
|
|
|
/* Setup our pointers */
|
|
FastIoDispatch.FastIoRead = NullRead;
|
|
FastIoDispatch.FastIoWrite = NullWrite;
|
|
DriverObject->FastIoDispatch = &FastIoDispatch;
|
|
|
|
/* Return success */
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
/* EOF */
|