218 lines
5.6 KiB
C
218 lines
5.6 KiB
C
#include <nt.h>
|
||
#include <ntrtl.h>
|
||
#include <nturtl.h>
|
||
#include <assert.h>
|
||
#include <stdlib.h>
|
||
#include <stdio.h>
|
||
#include <windows.h>
|
||
#include <string.h>
|
||
#include <memory.h>
|
||
#include <process.h>
|
||
|
||
#define WAIT_MULTIPLE_ITERATIONS 10000
|
||
HANDLE WaitHandles[64];
|
||
|
||
//
|
||
// Define local types.
|
||
//
|
||
|
||
typedef struct _PERFINFO {
|
||
LARGE_INTEGER StartTime;
|
||
LARGE_INTEGER StopTime;
|
||
ULONG ContextSwitches;
|
||
ULONG FirstLevelFills;
|
||
ULONG SecondLevelFills;
|
||
ULONG SystemCalls;
|
||
PCHAR Title;
|
||
ULONG Iterations;
|
||
} PERFINFO, *PPERFINFO;
|
||
|
||
VOID
|
||
FinishBenchMark (
|
||
IN PPERFINFO PerfInfo
|
||
)
|
||
|
||
{
|
||
|
||
ULONG ContextSwitches;
|
||
LARGE_INTEGER Duration;
|
||
ULONG FirstLevelFills;
|
||
ULONG Length;
|
||
ULONG Performance;
|
||
ULONG SecondLevelFills;
|
||
NTSTATUS Status;
|
||
ULONG SystemCalls;
|
||
SYSTEM_PERFORMANCE_INFORMATION SystemInfo;
|
||
|
||
|
||
//
|
||
// Print results and announce end of test.
|
||
//
|
||
|
||
NtQuerySystemTime((PLARGE_INTEGER)&PerfInfo->StopTime);
|
||
Status = NtQuerySystemInformation(SystemPerformanceInformation,
|
||
(PVOID)&SystemInfo,
|
||
sizeof(SYSTEM_PERFORMANCE_INFORMATION),
|
||
NULL);
|
||
|
||
if (NT_SUCCESS(Status) == FALSE) {
|
||
printf("Failed to query performance information, status = %lx\n", Status);
|
||
return;
|
||
}
|
||
|
||
Duration = RtlLargeIntegerSubtract(PerfInfo->StopTime, PerfInfo->StartTime);
|
||
Length = Duration.LowPart / 10000;
|
||
printf(" Test time in milliseconds %d\n", Length);
|
||
printf(" Number of iterations %d\n", PerfInfo->Iterations);
|
||
|
||
Performance = PerfInfo->Iterations * 1000 / Length;
|
||
printf(" Iterations per second %d\n", Performance);
|
||
|
||
ContextSwitches = SystemInfo.ContextSwitches - PerfInfo->ContextSwitches;
|
||
FirstLevelFills = SystemInfo.FirstLevelTbFills - PerfInfo->FirstLevelFills;
|
||
SecondLevelFills = SystemInfo.SecondLevelTbFills - PerfInfo->SecondLevelFills;
|
||
SystemCalls = SystemInfo.SystemCalls - PerfInfo->SystemCalls;
|
||
printf(" First Level TB Fills %d\n", FirstLevelFills);
|
||
printf(" Second Level TB Fills %d\n", SecondLevelFills);
|
||
printf(" Total Context Switches %d\n", ContextSwitches);
|
||
printf(" Number of System Calls %d\n", SystemCalls);
|
||
|
||
printf("*** End of Test ***\n\n");
|
||
return;
|
||
}
|
||
|
||
VOID
|
||
StartBenchMark (
|
||
IN PCHAR Title,
|
||
IN ULONG Iterations,
|
||
IN PPERFINFO PerfInfo
|
||
)
|
||
|
||
{
|
||
|
||
NTSTATUS Status;
|
||
SYSTEM_PERFORMANCE_INFORMATION SystemInfo;
|
||
|
||
//
|
||
// Announce start of test and the number of iterations.
|
||
//
|
||
|
||
printf("*** Start of test ***\n %s\n", Title);
|
||
PerfInfo->Title = Title;
|
||
PerfInfo->Iterations = Iterations;
|
||
NtQuerySystemTime((PLARGE_INTEGER)&PerfInfo->StartTime);
|
||
Status = NtQuerySystemInformation(SystemPerformanceInformation,
|
||
(PVOID)&SystemInfo,
|
||
sizeof(SYSTEM_PERFORMANCE_INFORMATION),
|
||
NULL);
|
||
|
||
if (NT_SUCCESS(Status) == FALSE) {
|
||
printf("Failed to query performance information, status = %lx\n", Status);
|
||
return;
|
||
}
|
||
|
||
PerfInfo->ContextSwitches = SystemInfo.ContextSwitches;
|
||
PerfInfo->FirstLevelFills = SystemInfo.FirstLevelTbFills;
|
||
PerfInfo->SecondLevelFills = SystemInfo.SecondLevelTbFills;
|
||
PerfInfo->SystemCalls = SystemInfo.SystemCalls;
|
||
return;
|
||
}
|
||
|
||
|
||
VOID
|
||
WaitMultipleTest()
|
||
{
|
||
PERFINFO PerfInfo;
|
||
int i;
|
||
|
||
StartBenchMark(
|
||
"Wait Single Object",
|
||
WAIT_MULTIPLE_ITERATIONS,
|
||
&PerfInfo
|
||
);
|
||
|
||
for ( i=0;i<WAIT_MULTIPLE_ITERATIONS;i++) {
|
||
WaitForSingleObject(WaitHandles[0],INFINITE); // Wait Single Object
|
||
}
|
||
|
||
FinishBenchMark(&PerfInfo);
|
||
|
||
StartBenchMark(
|
||
"Wait Multiple Test 1 Object",
|
||
WAIT_MULTIPLE_ITERATIONS,
|
||
&PerfInfo
|
||
);
|
||
|
||
for ( i=0;i<WAIT_MULTIPLE_ITERATIONS;i++) {
|
||
WaitForMultipleObjects(1,WaitHandles,FALSE,INFINITE); // Wait Any, 1 Object
|
||
}
|
||
|
||
FinishBenchMark(&PerfInfo);
|
||
|
||
StartBenchMark(
|
||
"Wait Multiple Test 8 Objects",
|
||
WAIT_MULTIPLE_ITERATIONS,
|
||
&PerfInfo
|
||
);
|
||
|
||
for ( i=0;i<WAIT_MULTIPLE_ITERATIONS;i++) {
|
||
WaitForMultipleObjects(8,WaitHandles,FALSE,INFINITE); // Wait Any, 8 Objects
|
||
}
|
||
|
||
FinishBenchMark(&PerfInfo);
|
||
|
||
StartBenchMark(
|
||
"Wait Multiple Test 16 Objects",
|
||
WAIT_MULTIPLE_ITERATIONS,
|
||
&PerfInfo
|
||
);
|
||
|
||
for ( i=0;i<WAIT_MULTIPLE_ITERATIONS;i++) {
|
||
WaitForMultipleObjects(16,WaitHandles,FALSE,INFINITE); // Wait Any, 16 Objects
|
||
}
|
||
|
||
FinishBenchMark(&PerfInfo);
|
||
|
||
StartBenchMark(
|
||
"Wait Multiple Test 32 Objects",
|
||
WAIT_MULTIPLE_ITERATIONS,
|
||
&PerfInfo
|
||
);
|
||
|
||
for ( i=0;i<WAIT_MULTIPLE_ITERATIONS;i++) {
|
||
WaitForMultipleObjects(32,WaitHandles,FALSE,INFINITE); // Wait Any, 32 Objects
|
||
}
|
||
|
||
FinishBenchMark(&PerfInfo);
|
||
|
||
StartBenchMark(
|
||
"Wait Multiple Test 64 Objects",
|
||
WAIT_MULTIPLE_ITERATIONS,
|
||
&PerfInfo
|
||
);
|
||
|
||
for ( i=0;i<WAIT_MULTIPLE_ITERATIONS;i++) {
|
||
WaitForMultipleObjects(64,WaitHandles,FALSE,INFINITE); // Wait Any, 64 Objects
|
||
}
|
||
|
||
FinishBenchMark(&PerfInfo);
|
||
}
|
||
|
||
DWORD
|
||
_cdecl
|
||
main(
|
||
int argc,
|
||
char *argv[],
|
||
char *envp[]
|
||
)
|
||
{
|
||
|
||
int i;
|
||
|
||
for(i=0;i<64;i++){
|
||
WaitHandles[i] = CreateEvent(NULL,TRUE,TRUE,NULL);
|
||
}
|
||
WaitMultipleTest();
|
||
return 0;
|
||
}
|