/* * PROJECT: ReactOS API tests * LICENSE: LGPLv2.1+ - See COPYING.LIB in the top level directory * PURPOSE: Tests for the NtQueryInformationProcess API * PROGRAMMER: Thomas Faber * George Bișoc */ #include "precomp.h" #include 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(); }