Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

255 lines
6.6 KiB

/******************************Module*Header*******************************\
* Module Name: Perf.c
*
* Performance counter functions. Uses the Pentium performance counters
* if they are available, otherwise falls back to the system QueryPerformance
* api's.
*
* InitPerfCounter MUST be called before using the QUERY_PERFORMANCE_XXX macros
* as it initializes the two global functions pointers.
*
*
*
* Created: 13-10-95
* Author: Stephen Estrop [StephenE]
*
* Copyright (c) 1994 - 1995 Microsoft Corporation. All Rights Reserved.
\**************************************************************************/
#include <windows.h>
#include "Perf.h"
PERFFUNCTION lpQueryPerfCounter;
PERFFUNCTION lpQueryPerfFreqency;
void
GetFrequencyEstimate(
LARGE_INTEGER *li
);
#ifdef TEST
#include <stdio.h>
/******************************Public*Routine******************************\
* main
*
* Program entry point.
*
* History:
* dd-mm-95 - StephenE - Created
*
\**************************************************************************/
int __cdecl main( void )
{
LARGE_INTEGER liP1;
LARGE_INTEGER liP2;
LARGE_INTEGER liPf;
InitPerfCounter();
QUERY_PERFORMANCE_FREQUENCY(&liPf);
// Time a 50 milli second sleep
QUERY_PERFORMANCE_COUNTER(&liP1);
Sleep(50);
QUERY_PERFORMANCE_COUNTER(&liP2);
printf("Pentium counter frequency = %u\n", liPf.LowPart );
printf("Pentium counter %#X%X - %#X%X = %u\n",
liP2.HighPart, liP2.LowPart, liP1.HighPart, liP1.LowPart,
liP2.LowPart - liP1.LowPart
);
printf("Time taken = %6.6f seconds\n",
(double)(liP2.LowPart - liP1.LowPart) / (double)liPf.QuadPart);
return 0;
}
#endif
/******************************Public*Routine******************************\
* InitPerfCounter
*
* Determine (at runtime) if it is possible to use the Pentium performance
* counter. If it is not fall back to the system performance counter.
*
* History:
* dd-mm-95 - StephenE - Created
*
\**************************************************************************/
void
InitPerfCounter(
void
)
{
SYSTEM_INFO sysInfo;
GetSystemInfo(&sysInfo);
if (sysInfo.dwProcessorType == PROCESSOR_INTEL_PENTIUM) {
lpQueryPerfFreqency = QueryPerfFrequency;
lpQueryPerfCounter = QueryPerfCounter;
}
else {
lpQueryPerfFreqency = (PERFFUNCTION)QueryPerformanceFrequency;
lpQueryPerfCounter = (PERFFUNCTION)QueryPerformanceCounter;
}
}
/******************************Public*Routine******************************\
* QueryPerfFrequency
*
* Determines the clock frequency of a (Pentium) microprocessor. Takes an
* averaged estimate of the clk frequency and then matches it to known
* Pentium clock frequencies. Returns the estimate if a match is not found.
*
* This is an expensive call in cpu terms as it takes at least 16 milli seconds
* just to calculate an averaged estimate of the clock speed. You only need
* to call this function once, make sure you don't call it more times.
*
* History:
* 13-10-95 - StephenE - Created
*
\**************************************************************************/
void WINAPI
QueryPerfFrequency(
LARGE_INTEGER *li
)
{
#ifdef _X86_
#define SAMPLE_SIZE 8
LARGE_INTEGER est;
int i;
li->QuadPart = 0;
for (i = 0; i < SAMPLE_SIZE; i++) {
GetFrequencyEstimate(&est);
li->QuadPart += est.QuadPart;
}
li->QuadPart /= SAMPLE_SIZE;
//
// At the moment Pentiums come in 60, 66, 75, 90, 100, 120 and 133 MHz
// clock speeds. So use the above estimation of the clock frequency
// to determine the real clock frequency.
//
// 59Mhz to 61Mhz assume its a 60 Mhz
if (li->QuadPart >= 59000000 && li->QuadPart < 61000000) {
li->QuadPart = 60000000;
}
// 65Mhz to 67Mhz assume its a 66 Mhz
else if (li->QuadPart >= 65000000 && li->QuadPart < 67000000) {
li->QuadPart = 66000000;
}
// 74Mhz to 76Mhz assume its a 75 Mhz
else if (li->QuadPart >= 74000000 && li->QuadPart < 76000000) {
li->QuadPart = 75000000;
}
// 89Mhz to 91Mhz assume its a 90 Mhz
else if (li->QuadPart >= 89000000 && li->QuadPart < 91000000) {
li->QuadPart = 90000000;
}
// 99Mhz to 101Mhz assume its a 100 Mhz
else if (li->QuadPart >= 99000000 && li->QuadPart < 101000000) {
li->QuadPart = 100000000;
}
// 119Mhz to 121Mhz assume its a 120 Mhz
else if (li->QuadPart >= 119000000 && li->QuadPart < 121000000) {
li->QuadPart = 120000000;
}
// 132Mhz to 134Mhz assume its a 133 Mhz
else if (li->QuadPart >= 132000000 && li->QuadPart < 134000000) {
li->QuadPart = 133000000;
}
// if use our estimate.
#else
li->QuadPart = -1;
#endif
}
/*****************************Private*Routine******************************\
* GetFrequencyEstimate
*
* Uses the system QueryPerformance counter to estimate the Pentium
* cpu clock * frequency
*
* History:
* 13-10-95 - StephenE - Created
*
\**************************************************************************/
void
GetFrequencyEstimate(
LARGE_INTEGER *li
)
{
LARGE_INTEGER liP1; // Pentium clk start
LARGE_INTEGER liP2; // Pentium clk end
LARGE_INTEGER liS1; // System clk end
LARGE_INTEGER liS2; // System clk end
LARGE_INTEGER liSf; // System clk frequency
QueryPerformanceFrequency(&liSf);
QueryPerformanceCounter(&liS1);
QueryPerfCounter(&liP1);
Sleep(2); // Sleep for approx 2 milli- seconds
QueryPerfCounter(&liP2);
QueryPerformanceCounter(&liS2);
//
// Determine the time recorded by both clocks.
//
liP2.QuadPart = liP2.QuadPart - liP1.QuadPart;
liS2.QuadPart = liS2.QuadPart - liS1.QuadPart;
li->QuadPart = (liP2.QuadPart * liSf.QuadPart) / liS2.QuadPart;
}
/******************************Public*Routine******************************\
* QueryPerfCounter
*
* Query the internal clock counter on the Pentium, uses the undocumented
* rdtsc instruction, which copies the current 64 bit clock count into
* edx:eax.
*
* History:
* 13-10-95 - StephenE - Created
*
\**************************************************************************/
void WINAPI
QueryPerfCounter(
LARGE_INTEGER *li
)
{
#ifdef _X86_
_asm mov ecx, dword ptr li // copy li pointer value to ecx
_asm _emit 0x0f // opcode 0x0F31 is rdtsc
_asm _emit 0x31
_asm mov dword ptr [ecx], eax // save result in li->LowPart
_asm mov dword ptr [ecx+4], edx // and li->HighPart
#else
;
#endif
}