2013-04-21 19:00:10 +00:00
|
|
|
/*
|
2017-09-17 08:08:39 +00:00
|
|
|
* PROJECT: ReactOS kernel-mode tests
|
|
|
|
* LICENSE: LGPL-2.1+ (https://spdx.org/licenses/LGPL-2.1+)
|
|
|
|
* PURPOSE: Test driver for kernel32 filesystem tests
|
|
|
|
* COPYRIGHT: Copyright 2013-2017 Thomas Faber <thomas.faber@reactos.org>
|
2013-04-21 19:00:10 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <kmt_test.h>
|
|
|
|
|
|
|
|
#define NDEBUG
|
|
|
|
#include <debug.h>
|
|
|
|
|
2017-06-29 15:30:23 +00:00
|
|
|
#include "kernel32_test.h"
|
2013-04-21 19:00:10 +00:00
|
|
|
|
|
|
|
static KMT_MESSAGE_HANDLER TestMessageHandler;
|
2017-06-29 15:30:23 +00:00
|
|
|
static KMT_IRP_HANDLER TestDirectoryControl;
|
2017-06-29 16:16:20 +00:00
|
|
|
static KMT_IRP_HANDLER TestQueryInformation;
|
|
|
|
static KMT_IRP_HANDLER TestSetInformation;
|
2013-04-21 19:00:10 +00:00
|
|
|
|
|
|
|
static UNICODE_STRING ExpectedExpression = RTL_CONSTANT_STRING(L"<not set>");
|
|
|
|
static WCHAR ExpressionBuffer[MAX_PATH];
|
2017-09-17 08:08:39 +00:00
|
|
|
static BOOLEAN ExpectingSetAttributes = FALSE;
|
2017-06-29 16:16:20 +00:00
|
|
|
static ULONG ExpectedSetAttributes = -1;
|
2017-09-17 08:08:39 +00:00
|
|
|
static BOOLEAN ExpectingQueryAttributes = FALSE;
|
2017-06-29 16:16:20 +00:00
|
|
|
static ULONG ReturnQueryAttributes = -1;
|
2013-04-21 19:00:10 +00:00
|
|
|
|
|
|
|
NTSTATUS
|
|
|
|
TestEntry(
|
|
|
|
IN PDRIVER_OBJECT DriverObject,
|
|
|
|
IN PCUNICODE_STRING RegistryPath,
|
|
|
|
OUT PCWSTR *DeviceName,
|
|
|
|
IN OUT INT *Flags)
|
|
|
|
{
|
|
|
|
NTSTATUS Status = STATUS_SUCCESS;
|
|
|
|
|
|
|
|
PAGED_CODE();
|
|
|
|
|
|
|
|
UNREFERENCED_PARAMETER(RegistryPath);
|
|
|
|
|
2017-06-29 15:30:23 +00:00
|
|
|
*DeviceName = L"kernel32";
|
2013-04-21 19:00:10 +00:00
|
|
|
*Flags = TESTENTRY_NO_EXCLUSIVE_DEVICE;
|
|
|
|
|
2017-06-29 15:30:23 +00:00
|
|
|
KmtRegisterIrpHandler(IRP_MJ_DIRECTORY_CONTROL, NULL, TestDirectoryControl);
|
2017-06-29 16:16:20 +00:00
|
|
|
KmtRegisterIrpHandler(IRP_MJ_QUERY_INFORMATION, NULL, TestQueryInformation);
|
|
|
|
KmtRegisterIrpHandler(IRP_MJ_SET_INFORMATION, NULL, TestSetInformation);
|
2013-04-21 19:00:10 +00:00
|
|
|
KmtRegisterMessageHandler(0, NULL, TestMessageHandler);
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
|
|
TestUnload(
|
|
|
|
IN PDRIVER_OBJECT DriverObject)
|
|
|
|
{
|
|
|
|
PAGED_CODE();
|
|
|
|
}
|
|
|
|
|
|
|
|
static
|
|
|
|
NTSTATUS
|
|
|
|
TestMessageHandler(
|
|
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
|
|
IN ULONG ControlCode,
|
|
|
|
IN PVOID Buffer OPTIONAL,
|
|
|
|
IN SIZE_T InLength,
|
|
|
|
IN OUT PSIZE_T OutLength)
|
|
|
|
{
|
|
|
|
NTSTATUS Status = STATUS_SUCCESS;
|
|
|
|
|
|
|
|
PAGED_CODE();
|
|
|
|
|
|
|
|
switch (ControlCode)
|
|
|
|
{
|
2017-06-29 15:30:23 +00:00
|
|
|
case IOCTL_EXPECT_EXPRESSION:
|
2013-04-21 19:00:10 +00:00
|
|
|
{
|
|
|
|
C_ASSERT(sizeof(ExpressionBuffer) <= UNICODE_STRING_MAX_BYTES);
|
2017-06-29 15:30:23 +00:00
|
|
|
DPRINT("IOCTL_EXPECT_EXPRESSION, InLength = %lu\n", InLength);
|
2013-04-21 19:00:10 +00:00
|
|
|
if (InLength > sizeof(ExpressionBuffer))
|
|
|
|
return STATUS_BUFFER_OVERFLOW;
|
|
|
|
|
|
|
|
if (InLength % sizeof(WCHAR) != 0)
|
|
|
|
return STATUS_INVALID_PARAMETER;
|
|
|
|
|
|
|
|
RtlInitEmptyUnicodeString(&ExpectedExpression, ExpressionBuffer, sizeof(ExpressionBuffer));
|
|
|
|
RtlCopyMemory(ExpressionBuffer, Buffer, InLength);
|
|
|
|
ExpectedExpression.Length = (USHORT)InLength;
|
2017-06-29 15:30:23 +00:00
|
|
|
DPRINT("IOCTL_EXPECT_EXPRESSION: %wZ\n", &ExpectedExpression);
|
2013-04-21 19:00:10 +00:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
2017-06-29 16:16:20 +00:00
|
|
|
case IOCTL_RETURN_QUERY_ATTRIBUTES:
|
|
|
|
{
|
|
|
|
DPRINT("IOCTL_RETURN_QUERY_ATTRIBUTES, InLength = %lu\n", InLength);
|
|
|
|
if (InLength != sizeof(ULONG))
|
|
|
|
return STATUS_INVALID_PARAMETER;
|
|
|
|
|
|
|
|
ReturnQueryAttributes = *(PULONG)Buffer;
|
2017-09-17 08:08:39 +00:00
|
|
|
ExpectingQueryAttributes = TRUE;
|
2017-06-29 16:16:20 +00:00
|
|
|
DPRINT("IOCTL_RETURN_QUERY_ATTRIBUTES: %lu\n", ReturnQueryAttributes);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case IOCTL_EXPECT_SET_ATTRIBUTES:
|
|
|
|
{
|
|
|
|
DPRINT("IOCTL_EXPECT_SET_ATTRIBUTES, InLength = %lu\n", InLength);
|
|
|
|
if (InLength != sizeof(ULONG))
|
|
|
|
return STATUS_INVALID_PARAMETER;
|
|
|
|
|
|
|
|
ExpectedSetAttributes = *(PULONG)Buffer;
|
2017-09-17 08:08:39 +00:00
|
|
|
ExpectingSetAttributes = TRUE;
|
2017-06-29 16:16:20 +00:00
|
|
|
DPRINT("IOCTL_EXPECT_SET_ATTRIBUTES: %lu\n", ExpectedSetAttributes);
|
|
|
|
break;
|
|
|
|
}
|
2013-04-21 19:00:10 +00:00
|
|
|
default:
|
|
|
|
return STATUS_NOT_SUPPORTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
static
|
|
|
|
NTSTATUS
|
2017-06-29 15:30:23 +00:00
|
|
|
TestDirectoryControl(
|
2013-04-21 19:00:10 +00:00
|
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
|
|
IN PIRP Irp,
|
|
|
|
IN PIO_STACK_LOCATION IoStackLocation)
|
|
|
|
{
|
|
|
|
NTSTATUS Status = STATUS_NOT_SUPPORTED;
|
|
|
|
|
|
|
|
PAGED_CODE();
|
|
|
|
|
|
|
|
DPRINT("IRP %x/%x\n", IoStackLocation->MajorFunction, IoStackLocation->MinorFunction);
|
|
|
|
ASSERT(IoStackLocation->MajorFunction == IRP_MJ_DIRECTORY_CONTROL);
|
|
|
|
|
|
|
|
ok(IoStackLocation->MinorFunction == IRP_MN_QUERY_DIRECTORY, "Minor function: %u\n", IoStackLocation->MinorFunction);
|
|
|
|
if (IoStackLocation->MinorFunction == IRP_MN_QUERY_DIRECTORY)
|
|
|
|
{
|
|
|
|
ok(IoStackLocation->Parameters.QueryDirectory.FileInformationClass == FileBothDirectoryInformation,
|
|
|
|
"FileInformationClass: %d\n", IoStackLocation->Parameters.QueryDirectory.FileInformationClass);
|
|
|
|
if (IoStackLocation->Parameters.QueryDirectory.FileInformationClass == FileBothDirectoryInformation)
|
|
|
|
{
|
|
|
|
ok(RtlEqualUnicodeString(IoStackLocation->Parameters.QueryDirectory.FileName, &ExpectedExpression, FALSE),
|
|
|
|
"Expression is '%wZ', expected '%wZ'\n", IoStackLocation->Parameters.QueryDirectory.FileName, &ExpectedExpression);
|
|
|
|
RtlZeroMemory(Irp->UserBuffer, IoStackLocation->Parameters.QueryDirectory.Length);
|
|
|
|
Status = STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Irp->IoStatus.Status = Status;
|
|
|
|
Irp->IoStatus.Information = 0;
|
|
|
|
|
|
|
|
IoCompleteRequest(Irp, IO_NO_INCREMENT);
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
2017-06-29 16:16:20 +00:00
|
|
|
|
|
|
|
static
|
|
|
|
NTSTATUS
|
|
|
|
TestQueryInformation(
|
|
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
|
|
IN PIRP Irp,
|
|
|
|
IN PIO_STACK_LOCATION IoStackLocation)
|
|
|
|
{
|
|
|
|
NTSTATUS Status = STATUS_NOT_SUPPORTED;
|
|
|
|
PFILE_BASIC_INFORMATION BasicInfo;
|
|
|
|
|
|
|
|
PAGED_CODE();
|
|
|
|
|
|
|
|
DPRINT("IRP %x/%x\n", IoStackLocation->MajorFunction, IoStackLocation->MinorFunction);
|
|
|
|
ASSERT(IoStackLocation->MajorFunction == IRP_MJ_QUERY_INFORMATION);
|
|
|
|
|
|
|
|
Irp->IoStatus.Information = 0;
|
|
|
|
|
|
|
|
ok_eq_ulong(IoStackLocation->Parameters.QueryFile.FileInformationClass, FileBasicInformation);
|
|
|
|
if (IoStackLocation->Parameters.QueryFile.FileInformationClass == FileBasicInformation)
|
|
|
|
{
|
2017-09-17 08:08:39 +00:00
|
|
|
ok(ExpectingQueryAttributes, "Unexpected QUERY_INFORMATION call\n");
|
2017-06-29 16:16:20 +00:00
|
|
|
BasicInfo = Irp->AssociatedIrp.SystemBuffer;
|
|
|
|
BasicInfo->CreationTime.QuadPart = 126011664000000000;
|
|
|
|
BasicInfo->LastAccessTime.QuadPart = 130899112800000000;
|
|
|
|
BasicInfo->LastWriteTime.QuadPart = 130899112800000000;
|
|
|
|
BasicInfo->ChangeTime.QuadPart = 130899112800000000;
|
|
|
|
BasicInfo->FileAttributes = ReturnQueryAttributes;
|
|
|
|
ReturnQueryAttributes = -1;
|
2017-09-17 08:08:39 +00:00
|
|
|
ExpectingQueryAttributes = FALSE;
|
2017-06-29 16:16:20 +00:00
|
|
|
Status = STATUS_SUCCESS;
|
|
|
|
Irp->IoStatus.Information = sizeof(*BasicInfo);
|
|
|
|
}
|
|
|
|
|
|
|
|
Irp->IoStatus.Status = Status;
|
|
|
|
|
|
|
|
IoCompleteRequest(Irp, IO_NO_INCREMENT);
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
static
|
|
|
|
NTSTATUS
|
|
|
|
TestSetInformation(
|
|
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
|
|
IN PIRP Irp,
|
|
|
|
IN PIO_STACK_LOCATION IoStackLocation)
|
|
|
|
{
|
|
|
|
NTSTATUS Status = STATUS_NOT_SUPPORTED;
|
|
|
|
PFILE_BASIC_INFORMATION BasicInfo;
|
|
|
|
|
|
|
|
PAGED_CODE();
|
|
|
|
|
|
|
|
DPRINT("IRP %x/%x\n", IoStackLocation->MajorFunction, IoStackLocation->MinorFunction);
|
|
|
|
ASSERT(IoStackLocation->MajorFunction == IRP_MJ_SET_INFORMATION);
|
|
|
|
|
|
|
|
ok_eq_ulong(IoStackLocation->Parameters.SetFile.FileInformationClass, FileBasicInformation);
|
|
|
|
if (IoStackLocation->Parameters.SetFile.FileInformationClass == FileBasicInformation)
|
|
|
|
{
|
2017-09-17 08:08:39 +00:00
|
|
|
ok(ExpectingSetAttributes, "Unexpected SET_INFORMATION call\n");
|
2017-06-29 16:16:20 +00:00
|
|
|
BasicInfo = Irp->AssociatedIrp.SystemBuffer;
|
|
|
|
ok_eq_longlong(BasicInfo->CreationTime.QuadPart, 0LL);
|
|
|
|
ok_eq_longlong(BasicInfo->LastAccessTime.QuadPart, 0LL);
|
|
|
|
ok_eq_longlong(BasicInfo->LastWriteTime.QuadPart, 0LL);
|
|
|
|
ok_eq_longlong(BasicInfo->ChangeTime.QuadPart, 0LL);
|
|
|
|
ok_eq_ulong(BasicInfo->FileAttributes, ExpectedSetAttributes);
|
|
|
|
ExpectedSetAttributes = -1;
|
2017-09-17 08:08:39 +00:00
|
|
|
ExpectingSetAttributes = FALSE;
|
2017-06-29 16:16:20 +00:00
|
|
|
Status = STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
Irp->IoStatus.Status = Status;
|
|
|
|
Irp->IoStatus.Information = 0;
|
|
|
|
|
|
|
|
IoCompleteRequest(Irp, IO_NO_INCREMENT);
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|