mirror of
https://github.com/reactos/reactos.git
synced 2025-01-03 21:09:19 +00:00
724 lines
31 KiB
C
724 lines
31 KiB
C
/*
|
|
* PROJECT: ReactOS API tests
|
|
* LICENSE: LGPLv2.1+ - See COPYING.LIB in the top level directory
|
|
* PURPOSE: Tests for the NtQueryInformationProcess API
|
|
* PROGRAMMER: Thomas Faber <thomas.faber@reactos.org>
|
|
* George Bișoc <george.bisoc@reactos.org>
|
|
*/
|
|
|
|
#include "precomp.h"
|
|
#include <internal/ps_i.h>
|
|
|
|
static LARGE_INTEGER TestStartTime;
|
|
|
|
static
|
|
void
|
|
Test_ProcessTimes(void)
|
|
{
|
|
#define SPIN_TIME 1000000
|
|
NTSTATUS Status;
|
|
KERNEL_USER_TIMES Times1;
|
|
KERNEL_USER_TIMES Times2;
|
|
ULONG Length;
|
|
LARGE_INTEGER Time1, Time2;
|
|
|
|
/* Everything is NULL */
|
|
Status = NtQueryInformationProcess(NULL,
|
|
ProcessTimes,
|
|
NULL,
|
|
0,
|
|
NULL);
|
|
ok_hex(Status, STATUS_INFO_LENGTH_MISMATCH);
|
|
|
|
/* Right size, invalid process */
|
|
Status = NtQueryInformationProcess(NULL,
|
|
ProcessTimes,
|
|
NULL,
|
|
sizeof(KERNEL_USER_TIMES),
|
|
NULL);
|
|
ok_hex(Status, STATUS_INVALID_HANDLE);
|
|
|
|
/* Valid process, no buffer */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessTimes,
|
|
NULL,
|
|
0,
|
|
NULL);
|
|
ok_hex(Status, STATUS_INFO_LENGTH_MISMATCH);
|
|
|
|
/* Unaligned buffer, wrong size */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessTimes,
|
|
(PVOID)2,
|
|
0,
|
|
NULL);
|
|
ok_hex(Status, STATUS_INFO_LENGTH_MISMATCH);
|
|
|
|
/* Unaligned buffer, correct size */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessTimes,
|
|
(PVOID)2,
|
|
sizeof(KERNEL_USER_TIMES),
|
|
NULL);
|
|
ok_hex(Status, STATUS_DATATYPE_MISALIGNMENT);
|
|
|
|
/* Buffer too small */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessTimes,
|
|
NULL,
|
|
sizeof(KERNEL_USER_TIMES) - 1,
|
|
NULL);
|
|
ok_hex(Status, STATUS_INFO_LENGTH_MISMATCH);
|
|
|
|
/* Right buffer size but NULL pointer */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessTimes,
|
|
NULL,
|
|
sizeof(KERNEL_USER_TIMES),
|
|
NULL);
|
|
ok_hex(Status, STATUS_ACCESS_VIOLATION);
|
|
|
|
/* Buffer too large */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessTimes,
|
|
NULL,
|
|
sizeof(KERNEL_USER_TIMES) + 1,
|
|
NULL);
|
|
ok_hex(Status, STATUS_INFO_LENGTH_MISMATCH);
|
|
|
|
/* Buffer too small, ask for length */
|
|
Length = 0x55555555;
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessTimes,
|
|
NULL,
|
|
sizeof(KERNEL_USER_TIMES) - 1,
|
|
&Length);
|
|
ok_hex(Status, STATUS_INFO_LENGTH_MISMATCH);
|
|
ok_dec(Length, 0x55555555);
|
|
|
|
Status = NtQuerySystemTime(&Time1);
|
|
ok_hex(Status, STATUS_SUCCESS);
|
|
|
|
/* Do some busy waiting to increase UserTime */
|
|
do
|
|
{
|
|
Status = NtQuerySystemTime(&Time2);
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
ok(0, "NtQuerySystemTime failed with %lx\n", Status);
|
|
break;
|
|
}
|
|
} while (Time2.QuadPart - Time1.QuadPart < SPIN_TIME);
|
|
|
|
/* Valid parameters, no return length */
|
|
Status = NtQuerySystemTime(&Time1);
|
|
ok_hex(Status, STATUS_SUCCESS);
|
|
|
|
RtlFillMemory(&Times1, sizeof(Times1), 0x55);
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessTimes,
|
|
&Times1,
|
|
sizeof(KERNEL_USER_TIMES),
|
|
NULL);
|
|
ok_hex(Status, STATUS_SUCCESS);
|
|
ok(Times1.CreateTime.QuadPart < TestStartTime.QuadPart,
|
|
"CreateTime is %I64u, expected < %I64u\n", Times1.CreateTime.QuadPart, TestStartTime.QuadPart);
|
|
ok(Times1.CreateTime.QuadPart > TestStartTime.QuadPart - 100000000LL,
|
|
"CreateTime is %I64u, expected > %I64u\n", Times1.CreateTime.QuadPart, TestStartTime.QuadPart - 100000000LL);
|
|
ok(Times1.ExitTime.QuadPart == 0,
|
|
"ExitTime is %I64u, expected 0\n", Times1.ExitTime.QuadPart);
|
|
ok(Times1.KernelTime.QuadPart != 0, "KernelTime is 0\n");
|
|
ros_skip_flaky
|
|
ok(Times1.UserTime.QuadPart != 0, "UserTime is 0\n");
|
|
|
|
/* Do some busy waiting to increase UserTime */
|
|
do
|
|
{
|
|
Status = NtQuerySystemTime(&Time2);
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
ok(0, "NtQuerySystemTime failed with %lx\n", Status);
|
|
break;
|
|
}
|
|
} while (Time2.QuadPart - Time1.QuadPart < SPIN_TIME);
|
|
|
|
/* Again, this time with a return length */
|
|
Length = 0x55555555;
|
|
RtlFillMemory(&Times2, sizeof(Times2), 0x55);
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessTimes,
|
|
&Times2,
|
|
sizeof(KERNEL_USER_TIMES),
|
|
&Length);
|
|
ok_hex(Status, STATUS_SUCCESS);
|
|
ok_dec(Length, sizeof(KERNEL_USER_TIMES));
|
|
ok(Times1.CreateTime.QuadPart == Times2.CreateTime.QuadPart,
|
|
"CreateTimes not equal: %I64u != %I64u\n", Times1.CreateTime.QuadPart, Times2.CreateTime.QuadPart);
|
|
ok(Times2.ExitTime.QuadPart == 0,
|
|
"ExitTime is %I64u, expected 0\n", Times2.ExitTime.QuadPart);
|
|
ok(Times2.KernelTime.QuadPart != 0, "KernelTime is 0\n");
|
|
ok(Times2.UserTime.QuadPart != 0, "UserTime is 0\n");
|
|
|
|
/* Compare the two sets of KernelTime/UserTime values */
|
|
Status = NtQuerySystemTime(&Time2);
|
|
ok_hex(Status, STATUS_SUCCESS);
|
|
/* Time values must have increased */
|
|
ok(Times2.KernelTime.QuadPart > Times1.KernelTime.QuadPart,
|
|
"KernelTime values inconsistent. Expected %I64u > %I64u\n", Times2.KernelTime.QuadPart, Times1.KernelTime.QuadPart);
|
|
ros_skip_flaky
|
|
ok(Times2.UserTime.QuadPart > Times1.UserTime.QuadPart,
|
|
"UserTime values inconsistent. Expected %I64u > %I64u\n", Times2.UserTime.QuadPart, Times1.UserTime.QuadPart);
|
|
/* They can't have increased by more than wall clock time difference (we only have one thread) */
|
|
ros_skip_flaky
|
|
ok(Times2.KernelTime.QuadPart - Times1.KernelTime.QuadPart < Time2.QuadPart - Time1.QuadPart,
|
|
"KernelTime values inconsistent. Expected %I64u - %I64u < %I64u\n",
|
|
Times2.KernelTime.QuadPart, Times1.KernelTime.QuadPart, Time2.QuadPart - Time1.QuadPart);
|
|
ok(Times2.UserTime.QuadPart - Times1.UserTime.QuadPart < Time2.QuadPart - Time1.QuadPart,
|
|
"UserTime values inconsistent. Expected %I64u - %I64u < %I64u\n",
|
|
Times2.UserTime.QuadPart, Times1.UserTime.QuadPart, Time2.QuadPart - Time1.QuadPart);
|
|
|
|
trace("KernelTime1 = %I64u\n", Times1.KernelTime.QuadPart);
|
|
trace("KernelTime2 = %I64u\n", Times2.KernelTime.QuadPart);
|
|
trace("UserTime1 = %I64u\n", Times1.UserTime.QuadPart);
|
|
trace("UserTime2 = %I64u\n", Times2.UserTime.QuadPart);
|
|
|
|
/* TODO: Test ExitTime on a terminated process */
|
|
#undef SPIN_TIME
|
|
}
|
|
|
|
static
|
|
void
|
|
Test_ProcessBasicInformation(void)
|
|
{
|
|
NTSTATUS Status;
|
|
ULONG Length;
|
|
PROCESS_BASIC_INFORMATION BasicInfo;
|
|
|
|
/* Everything is NULL */
|
|
Status = NtQueryInformationProcess(NULL,
|
|
ProcessBasicInformation,
|
|
NULL,
|
|
0,
|
|
NULL);
|
|
ok_hex(Status, STATUS_INFO_LENGTH_MISMATCH);
|
|
|
|
/* Right size, invalid process handle */
|
|
Status = NtQueryInformationProcess(NULL,
|
|
ProcessBasicInformation,
|
|
NULL,
|
|
sizeof(BasicInfo),
|
|
NULL);
|
|
ok_hex(Status, STATUS_INVALID_HANDLE);
|
|
|
|
/* Valid process handle, no buffer */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessBasicInformation,
|
|
NULL,
|
|
0,
|
|
NULL);
|
|
ok_hex(Status, STATUS_INFO_LENGTH_MISMATCH);
|
|
|
|
/* Unaligned buffer, wrong size */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessBasicInformation,
|
|
(PVOID)2,
|
|
0,
|
|
NULL);
|
|
ok_hex(Status, STATUS_INFO_LENGTH_MISMATCH);
|
|
|
|
/* Unaligned buffer, correct size */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessBasicInformation,
|
|
(PVOID)2,
|
|
sizeof(BasicInfo),
|
|
NULL);
|
|
ok_hex(Status, STATUS_DATATYPE_MISALIGNMENT);
|
|
|
|
/* Buffer too small */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessBasicInformation,
|
|
NULL,
|
|
sizeof(BasicInfo) - 1,
|
|
NULL);
|
|
ok_hex(Status, STATUS_INFO_LENGTH_MISMATCH);
|
|
|
|
/* Right buffer size but NULL pointer */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessBasicInformation,
|
|
NULL,
|
|
sizeof(BasicInfo),
|
|
NULL);
|
|
ok_hex(Status, STATUS_ACCESS_VIOLATION);
|
|
|
|
/* Buffer too large */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessBasicInformation,
|
|
NULL,
|
|
sizeof(BasicInfo) + 1,
|
|
NULL);
|
|
ok_hex(Status, STATUS_INFO_LENGTH_MISMATCH);
|
|
|
|
/* Buffer too small, ask for length */
|
|
Length = 0x55555555;
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessBasicInformation,
|
|
NULL,
|
|
sizeof(BasicInfo) - 1,
|
|
&Length);
|
|
ok_hex(Status, STATUS_INFO_LENGTH_MISMATCH);
|
|
ok_dec(Length, 0x55555555);
|
|
|
|
/* Valid parameters, no return length */
|
|
RtlFillMemory(&BasicInfo, sizeof(BasicInfo), 0x55);
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessBasicInformation,
|
|
&BasicInfo,
|
|
sizeof(BasicInfo),
|
|
NULL);
|
|
ok_hex(Status, STATUS_SUCCESS);
|
|
|
|
/* Trace the returned data (1) */
|
|
trace("[1] BasicInfo.ExitStatus = %lx\n", BasicInfo.ExitStatus);
|
|
trace("[1] BasicInfo.PebBaseAddress = %p\n", BasicInfo.PebBaseAddress);
|
|
trace("[1] BasicInfo.AffinityMask = %Ix\n", BasicInfo.AffinityMask);
|
|
trace("[1] BasicInfo.BasePriority = %ld\n", BasicInfo.BasePriority);
|
|
trace("[1] BasicInfo.UniqueProcessId = %Iu\n", BasicInfo.UniqueProcessId);
|
|
trace("[1] BasicInfo.InheritedFromUniqueProcessId = %Iu\n", BasicInfo.InheritedFromUniqueProcessId);
|
|
|
|
/* Again, this time with a return length */
|
|
Length = 0x55555555;
|
|
RtlFillMemory(&BasicInfo, sizeof(BasicInfo), 0x55);
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessBasicInformation,
|
|
&BasicInfo,
|
|
sizeof(BasicInfo),
|
|
&Length);
|
|
ok_hex(Status, STATUS_SUCCESS);
|
|
ok_dec(Length, sizeof(BasicInfo));
|
|
|
|
/* Trace the returned data (2) */
|
|
trace("[2] BasicInfo.ExitStatus = %lx\n", BasicInfo.ExitStatus);
|
|
trace("[2] BasicInfo.PebBaseAddress = %p\n", BasicInfo.PebBaseAddress);
|
|
trace("[2] BasicInfo.AffinityMask = %Ix\n", BasicInfo.AffinityMask);
|
|
trace("[2] BasicInfo.BasePriority = %ld\n", BasicInfo.BasePriority);
|
|
trace("[2] BasicInfo.UniqueProcessId = %Iu\n", BasicInfo.UniqueProcessId);
|
|
trace("[2] BasicInfo.InheritedFromUniqueProcessId = %Iu\n", BasicInfo.InheritedFromUniqueProcessId);
|
|
}
|
|
|
|
static
|
|
void
|
|
Test_ProcessQuotaLimits(void)
|
|
{
|
|
NTSTATUS Status;
|
|
ULONG Length;
|
|
QUOTA_LIMITS QuotaLimits;
|
|
|
|
/* Everything is NULL */
|
|
Status = NtQueryInformationProcess(NULL,
|
|
ProcessQuotaLimits,
|
|
NULL,
|
|
0,
|
|
NULL);
|
|
ok_hex(Status, STATUS_INFO_LENGTH_MISMATCH);
|
|
|
|
/* Right size, invalid process handle */
|
|
Status = NtQueryInformationProcess(NULL,
|
|
ProcessQuotaLimits,
|
|
NULL,
|
|
sizeof(QuotaLimits),
|
|
NULL);
|
|
ok_hex(Status, STATUS_INVALID_HANDLE);
|
|
|
|
/* Valid process handle, no buffer */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessQuotaLimits,
|
|
NULL,
|
|
0,
|
|
NULL);
|
|
ok_hex(Status, STATUS_INFO_LENGTH_MISMATCH);
|
|
|
|
/* Unaligned buffer, wrong size */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessQuotaLimits,
|
|
(PVOID)2,
|
|
0,
|
|
NULL);
|
|
ok_hex(Status, STATUS_INFO_LENGTH_MISMATCH);
|
|
|
|
/* Unaligned buffer, correct size */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessQuotaLimits,
|
|
(PVOID)2,
|
|
sizeof(QuotaLimits),
|
|
NULL);
|
|
ok_hex(Status, STATUS_DATATYPE_MISALIGNMENT);
|
|
|
|
/* Buffer too small */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessQuotaLimits,
|
|
NULL,
|
|
sizeof(QuotaLimits) - 1,
|
|
NULL);
|
|
ok_hex(Status, STATUS_INFO_LENGTH_MISMATCH);
|
|
|
|
/* Right buffer size but NULL pointer */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessQuotaLimits,
|
|
NULL,
|
|
sizeof(QuotaLimits),
|
|
NULL);
|
|
ok_hex(Status, STATUS_ACCESS_VIOLATION);
|
|
|
|
/* Buffer too large */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessQuotaLimits,
|
|
NULL,
|
|
sizeof(QuotaLimits) + 1,
|
|
NULL);
|
|
ok_hex(Status, STATUS_INFO_LENGTH_MISMATCH);
|
|
|
|
/* Buffer too small, ask for length */
|
|
Length = 0x55555555;
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessQuotaLimits,
|
|
NULL,
|
|
sizeof(QuotaLimits) - 1,
|
|
&Length);
|
|
ok_hex(Status, STATUS_INFO_LENGTH_MISMATCH);
|
|
ok_dec(Length, 0x55555555);
|
|
|
|
/* Valid parameters, no return length */
|
|
RtlFillMemory(&QuotaLimits, sizeof(QuotaLimits), 0x55);
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessQuotaLimits,
|
|
&QuotaLimits,
|
|
sizeof(QuotaLimits),
|
|
NULL);
|
|
ok_hex(Status, STATUS_SUCCESS);
|
|
|
|
/* Trace the returned data (1) */
|
|
trace("[1] QuotaLimits.PagedPoolLimit = %Iu\n", QuotaLimits.PagedPoolLimit);
|
|
trace("[1] QuotaLimits.NonPagedPoolLimit = %Iu\n", QuotaLimits.NonPagedPoolLimit);
|
|
trace("[1] QuotaLimits.MinimumWorkingSetSize = %Iu\n", QuotaLimits.MinimumWorkingSetSize);
|
|
trace("[1] QuotaLimits.MaximumWorkingSetSize = %Iu\n", QuotaLimits.MaximumWorkingSetSize);
|
|
trace("[1] QuotaLimits.PagefileLimit = %Iu\n", QuotaLimits.PagefileLimit);
|
|
trace("[1] QuotaLimits.TimeLimit = %I64d\n", QuotaLimits.TimeLimit.QuadPart);
|
|
|
|
/* Again, this time with a return length */
|
|
Length = 0x55555555;
|
|
RtlFillMemory(&QuotaLimits, sizeof(QuotaLimits), 0x55);
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessQuotaLimits,
|
|
&QuotaLimits,
|
|
sizeof(QuotaLimits),
|
|
&Length);
|
|
ok_hex(Status, STATUS_SUCCESS);
|
|
ok_dec(Length, sizeof(QuotaLimits));
|
|
|
|
/* Trace the returned data (2) */
|
|
trace("[2] QuotaLimits.PagedPoolLimit = %Iu\n", QuotaLimits.PagedPoolLimit);
|
|
trace("[2] QuotaLimits.NonPagedPoolLimit = %Iu\n", QuotaLimits.NonPagedPoolLimit);
|
|
trace("[2] QuotaLimits.MinimumWorkingSetSize = %Iu\n", QuotaLimits.MinimumWorkingSetSize);
|
|
trace("[2] QuotaLimits.MaximumWorkingSetSize = %Iu\n", QuotaLimits.MaximumWorkingSetSize);
|
|
trace("[2] QuotaLimits.PagefileLimit = %Iu\n", QuotaLimits.PagefileLimit);
|
|
trace("[2] QuotaLimits.TimeLimit = %I64d\n", QuotaLimits.TimeLimit.QuadPart);
|
|
}
|
|
|
|
static
|
|
void
|
|
Test_ProcessQuotaLimitsEx(void)
|
|
{
|
|
NTSTATUS Status;
|
|
ULONG Length;
|
|
QUOTA_LIMITS_EX QuotaLimitsEx;
|
|
|
|
/* Right size, invalid process handle */
|
|
Status = NtQueryInformationProcess(NULL,
|
|
ProcessQuotaLimits,
|
|
NULL,
|
|
sizeof(QuotaLimitsEx),
|
|
NULL);
|
|
ok_hex(Status, STATUS_INVALID_HANDLE);
|
|
|
|
/* Unaligned buffer, correct size */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessQuotaLimits,
|
|
(PVOID)2,
|
|
sizeof(QuotaLimitsEx),
|
|
NULL);
|
|
ok_hex(Status, STATUS_DATATYPE_MISALIGNMENT);
|
|
|
|
/* Buffer too small */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessQuotaLimits,
|
|
NULL,
|
|
sizeof(QuotaLimitsEx) - 1,
|
|
NULL);
|
|
ok_hex(Status, STATUS_INFO_LENGTH_MISMATCH);
|
|
|
|
/* Right buffer size but NULL pointer */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessQuotaLimits,
|
|
NULL,
|
|
sizeof(QuotaLimitsEx),
|
|
NULL);
|
|
ok_hex(Status, STATUS_ACCESS_VIOLATION);
|
|
|
|
/* Buffer too large */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessQuotaLimits,
|
|
NULL,
|
|
sizeof(QuotaLimitsEx) + 1,
|
|
NULL);
|
|
ok_hex(Status, STATUS_INFO_LENGTH_MISMATCH);
|
|
|
|
/* Buffer too small, ask for length */
|
|
Length = 0x55555555;
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessQuotaLimits,
|
|
NULL,
|
|
sizeof(QuotaLimitsEx) - 1,
|
|
&Length);
|
|
ok_hex(Status, STATUS_INFO_LENGTH_MISMATCH);
|
|
ok_dec(Length, 0x55555555);
|
|
|
|
/* Valid parameters, no return length */
|
|
RtlFillMemory(&QuotaLimitsEx, sizeof(QuotaLimitsEx), 0x55);
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessQuotaLimits,
|
|
&QuotaLimitsEx,
|
|
sizeof(QuotaLimitsEx),
|
|
NULL);
|
|
ok_hex(Status, STATUS_SUCCESS);
|
|
|
|
/* Trace the returned data (1) */
|
|
trace("[1] QuotaLimitsEx.PagedPoolLimit = %Iu\n", QuotaLimitsEx.PagedPoolLimit);
|
|
trace("[1] QuotaLimitsEx.NonPagedPoolLimit = %Iu\n", QuotaLimitsEx.NonPagedPoolLimit);
|
|
trace("[1] QuotaLimitsEx.MinimumWorkingSetSize = %Iu\n", QuotaLimitsEx.MinimumWorkingSetSize);
|
|
trace("[1] QuotaLimitsEx.MaximumWorkingSetSize = %Iu\n", QuotaLimitsEx.MaximumWorkingSetSize);
|
|
trace("[1] QuotaLimitsEx.PagefileLimit = %Iu\n", QuotaLimitsEx.PagefileLimit);
|
|
trace("[1] QuotaLimitsEx.TimeLimit = %I64d\n", QuotaLimitsEx.TimeLimit.QuadPart);
|
|
//trace("[1] QuotaLimitsEx.WorkingSetLimit = %Iu\n", QuotaLimitsEx.WorkingSetLimit); // Not used on Win2k3
|
|
trace("[1] QuotaLimitsEx.Flags = %lx\n", QuotaLimitsEx.Flags);
|
|
trace("[1] QuotaLimitsEx.CpuRateLimit.RateData = %lx\n", QuotaLimitsEx.CpuRateLimit.RateData);
|
|
|
|
/* Again, this time with a return length */
|
|
Length = 0x55555555;
|
|
RtlFillMemory(&QuotaLimitsEx, sizeof(QuotaLimitsEx), 0x55);
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessQuotaLimits,
|
|
&QuotaLimitsEx,
|
|
sizeof(QuotaLimitsEx),
|
|
&Length);
|
|
ok_hex(Status, STATUS_SUCCESS);
|
|
ok_dec(Length, sizeof(QuotaLimitsEx));
|
|
|
|
/* Trace the returned data (2) */
|
|
trace("[2] QuotaLimitsEx.PagedPoolLimit = %Iu\n", QuotaLimitsEx.PagedPoolLimit);
|
|
trace("[2] QuotaLimitsEx.NonPagedPoolLimit = %Iu\n", QuotaLimitsEx.NonPagedPoolLimit);
|
|
trace("[2] QuotaLimitsEx.MinimumWorkingSetSize = %Iu\n", QuotaLimitsEx.MinimumWorkingSetSize);
|
|
trace("[2] QuotaLimitsEx.MaximumWorkingSetSize = %Iu\n", QuotaLimitsEx.MaximumWorkingSetSize);
|
|
trace("[2] QuotaLimitsEx.PagefileLimit = %Iu\n", QuotaLimitsEx.PagefileLimit);
|
|
trace("[2] QuotaLimitsEx.TimeLimit = %I64d\n", QuotaLimitsEx.TimeLimit.QuadPart);
|
|
//trace("[2] QuotaLimitsEx.WorkingSetLimit = %Iu\n", QuotaLimitsEx.WorkingSetLimit); // Not used on Win2k3
|
|
trace("[2] QuotaLimitsEx.Flags = %lx\n", QuotaLimitsEx.Flags);
|
|
trace("[2] QuotaLimitsEx.CpuRateLimit.RateData = %lx\n", QuotaLimitsEx.CpuRateLimit.RateData);
|
|
}
|
|
|
|
static
|
|
void
|
|
Test_ProcessPriorityClassAlignment(void)
|
|
{
|
|
NTSTATUS Status;
|
|
PPROCESS_PRIORITY_CLASS ProcPriority;
|
|
|
|
/* Allocate some memory for the priority class structure */
|
|
ProcPriority = malloc(sizeof(PROCESS_PRIORITY_CLASS));
|
|
if (ProcPriority == NULL)
|
|
{
|
|
skip("Failed to allocate memory for PROCESS_PRIORITY_CLASS!\n");
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* Initialize the PriorityClass member to ensure the test won't randomly succeed (if such data is uninitialized).
|
|
* Filling 85 to the data member makes sure that if the test fails continously then NtQueryInformationProcess()
|
|
* didn't initialize the structure with data.
|
|
*/
|
|
RtlFillMemory(&ProcPriority->PriorityClass, sizeof(ProcPriority->PriorityClass), 0x55);
|
|
|
|
/* Unaligned buffer -- wrong size */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessPriorityClass,
|
|
(PVOID)1,
|
|
0,
|
|
NULL);
|
|
ok_hex(Status, STATUS_INFO_LENGTH_MISMATCH);
|
|
|
|
/* Unaligned buffer -- correct size */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessPriorityClass,
|
|
(PVOID)1,
|
|
sizeof(PROCESS_PRIORITY_CLASS),
|
|
NULL);
|
|
ok_hex(Status, STATUS_DATATYPE_MISALIGNMENT);
|
|
|
|
/* Unaligned buffer -- wrong size (but this time do with an alignment of 2) */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessPriorityClass,
|
|
(PVOID)2,
|
|
0,
|
|
NULL);
|
|
ok_hex(Status, STATUS_INFO_LENGTH_MISMATCH);
|
|
|
|
/* Unaligned buffer -- correct size (but this time do with an alignment of 2) */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessPriorityClass,
|
|
(PVOID)2,
|
|
sizeof(PROCESS_PRIORITY_CLASS),
|
|
NULL);
|
|
ok_hex(Status, STATUS_DATATYPE_MISALIGNMENT);
|
|
|
|
/* Do not care for the length but expect to return the priority class */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessPriorityClass,
|
|
ProcPriority,
|
|
sizeof(PROCESS_PRIORITY_CLASS),
|
|
NULL);
|
|
ok_hex(Status, STATUS_SUCCESS);
|
|
|
|
/* Make sure the returned priority class is a valid number (non negative) but also it should be within the PROCESS_PRIORITY_CLASS range */
|
|
ok(ProcPriority->PriorityClass > PROCESS_PRIORITY_CLASS_INVALID && ProcPriority->PriorityClass <= PROCESS_PRIORITY_CLASS_ABOVE_NORMAL,
|
|
"Expected a valid number from priority class range but got %d\n", ProcPriority->PriorityClass);
|
|
free(ProcPriority);
|
|
}
|
|
|
|
static
|
|
void
|
|
Test_ProcessWx86Information(void)
|
|
{
|
|
NTSTATUS Status;
|
|
ULONG VdmPower = 1, ReturnLength;
|
|
|
|
/* Everything is NULL */
|
|
Status = NtQueryInformationProcess(NULL,
|
|
ProcessWx86Information,
|
|
NULL,
|
|
0,
|
|
NULL);
|
|
ok_hex(Status, STATUS_INFO_LENGTH_MISMATCH);
|
|
|
|
/* Given an invalid process handle */
|
|
Status = NtQueryInformationProcess(NULL,
|
|
ProcessWx86Information,
|
|
&VdmPower,
|
|
sizeof(VdmPower),
|
|
NULL);
|
|
ok_hex(Status, STATUS_INVALID_HANDLE);
|
|
|
|
/* Don't query anything */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessWx86Information,
|
|
NULL,
|
|
sizeof(VdmPower),
|
|
NULL);
|
|
ok_hex(Status, STATUS_ACCESS_VIOLATION);
|
|
|
|
/* The buffer is misaligned and information length is wrong */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessWx86Information,
|
|
(PVOID)1,
|
|
0,
|
|
NULL);
|
|
ok_hex(Status, STATUS_INFO_LENGTH_MISMATCH);
|
|
|
|
/* The buffer is misaligned */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessWx86Information,
|
|
(PVOID)1,
|
|
sizeof(VdmPower),
|
|
NULL);
|
|
ok_hex(Status, STATUS_DATATYPE_MISALIGNMENT);
|
|
|
|
/* The buffer is misaligned -- try with an alignment size of 2 */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessWx86Information,
|
|
(PVOID)2,
|
|
sizeof(VdmPower),
|
|
NULL);
|
|
ok_hex(Status, STATUS_DATATYPE_MISALIGNMENT);
|
|
|
|
/* Query the VDM power */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessWx86Information,
|
|
&VdmPower,
|
|
sizeof(VdmPower),
|
|
NULL);
|
|
ok_hex(Status, STATUS_SUCCESS);
|
|
ok(VdmPower == 0 || VdmPower == 1, "The VDM power value must be within the boundary between 0 and 1, not anything else! Got %lu\n", VdmPower);
|
|
|
|
/* Same but with ReturnLength */
|
|
Status = NtQueryInformationProcess(NtCurrentProcess(),
|
|
ProcessWx86Information,
|
|
&VdmPower,
|
|
sizeof(VdmPower),
|
|
&ReturnLength);
|
|
ok_hex(Status, STATUS_SUCCESS);
|
|
ok(ReturnLength != 0, "ReturnLength shouldn't be 0!\n");
|
|
ok(VdmPower == 0 || VdmPower == 1, "The VDM power value must be within the boundary between 0 and 1, not anything else! Got %lu\n", VdmPower);
|
|
|
|
/* Trace the VDM power value and returned length */
|
|
trace("ReturnLength = %lu\n", ReturnLength);
|
|
trace("VdmPower = %lu\n", VdmPower);
|
|
}
|
|
|
|
static
|
|
void
|
|
Test_ProcQueryAlignmentProbe(void)
|
|
{
|
|
ULONG InfoClass;
|
|
|
|
/* Iterate over the process info classes and begin the tests */
|
|
for (InfoClass = 0; InfoClass < _countof(PsProcessInfoClass); InfoClass++)
|
|
{
|
|
/* The buffer is misaligned */
|
|
QuerySetProcessValidator(QUERY,
|
|
InfoClass,
|
|
(PVOID)(ULONG_PTR)1,
|
|
PsProcessInfoClass[InfoClass].RequiredSizeQUERY,
|
|
STATUS_DATATYPE_MISALIGNMENT);
|
|
|
|
/* We query an invalid buffer address */
|
|
QuerySetProcessValidator(QUERY,
|
|
InfoClass,
|
|
(PVOID)(ULONG_PTR)PsProcessInfoClass[InfoClass].AlignmentQUERY,
|
|
PsProcessInfoClass[InfoClass].RequiredSizeQUERY,
|
|
STATUS_ACCESS_VIOLATION);
|
|
|
|
/* The information length is wrong */
|
|
QuerySetProcessValidator(QUERY,
|
|
InfoClass,
|
|
(PVOID)(ULONG_PTR)PsProcessInfoClass[InfoClass].AlignmentQUERY,
|
|
PsProcessInfoClass[InfoClass].RequiredSizeQUERY - 1,
|
|
STATUS_INFO_LENGTH_MISMATCH);
|
|
}
|
|
}
|
|
|
|
START_TEST(NtQueryInformationProcess)
|
|
{
|
|
NTSTATUS Status;
|
|
|
|
/* Make sure that some time has passed since process creation, even if the resolution of our NtQuerySystemTime is low. */
|
|
Sleep(1);
|
|
|
|
Status = NtQuerySystemTime(&TestStartTime);
|
|
ok_hex(Status, STATUS_SUCCESS);
|
|
|
|
Test_ProcessTimes();
|
|
Test_ProcessBasicInformation();
|
|
Test_ProcessQuotaLimits();
|
|
Test_ProcessQuotaLimitsEx();
|
|
Test_ProcessPriorityClassAlignment();
|
|
Test_ProcessWx86Information();
|
|
Test_ProcQueryAlignmentProbe();
|
|
}
|