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.
273 lines
5.3 KiB
273 lines
5.3 KiB
/*++
|
|
|
|
Copyright (c) 2000 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
xxtimer.c
|
|
|
|
Abstract:
|
|
|
|
This module contains the HAL's timer-related APIs
|
|
|
|
Author:
|
|
|
|
Eric Nelson (enelson) July 6, 2000
|
|
|
|
Revision History:
|
|
|
|
--*/
|
|
|
|
#include "halp.h"
|
|
#include "xxtimer.h"
|
|
|
|
#ifdef ALLOC_PRAGMA
|
|
#pragma alloc_text(INIT, HalpSetTimerFunctions)
|
|
#endif
|
|
|
|
//
|
|
// External function prototypes
|
|
//
|
|
ULONG
|
|
HalpAcpiTimerSetTimeIncrement(
|
|
IN ULONG DesiredIncrement
|
|
);
|
|
|
|
VOID
|
|
HalpAcpiTimerStallExecProc(
|
|
IN ULONG MicroSeconds
|
|
);
|
|
|
|
VOID
|
|
HalpAcpiTimerCalibratePerfCount(
|
|
IN LONG volatile *Number,
|
|
IN ULONGLONG NewCount
|
|
);
|
|
VOID
|
|
HalpPmTimerCalibratePerfCount(
|
|
IN LONG volatile *Number,
|
|
IN ULONGLONG NewCount
|
|
);
|
|
|
|
LARGE_INTEGER
|
|
HalpAcpiTimerQueryPerfCount(
|
|
OUT PLARGE_INTEGER PerformanceFrequency OPTIONAL
|
|
);
|
|
|
|
LARGE_INTEGER
|
|
HalpPmTimerQueryPerfCount(
|
|
OUT PLARGE_INTEGER PerformanceFrequency OPTIONAL
|
|
);
|
|
|
|
//
|
|
// Local variables
|
|
//
|
|
static TIMER_FUNCTIONS HalpTimerFunctions = { HalpAcpiTimerStallExecProc,
|
|
#ifdef NO_PM_KEQPC
|
|
HalpAcpiTimerCalibratePerfCount,
|
|
HalpAcpiTimerQueryPerfCount,
|
|
#else
|
|
HalpPmTimerCalibratePerfCount,
|
|
HalpPmTimerQueryPerfCount,
|
|
#endif
|
|
HalpAcpiTimerSetTimeIncrement };
|
|
|
|
|
|
VOID
|
|
HalpSetTimerFunctions(
|
|
IN PTIMER_FUNCTIONS TimerFunctions
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine can be used to override the HALs ACPI-timer functions with
|
|
multimedia event timer functions
|
|
|
|
Arguments:
|
|
|
|
TimerFunctions - Pointer to a table of timer functions
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
{
|
|
HalpTimerFunctions = *TimerFunctions;
|
|
|
|
#if 1
|
|
HalpTimerFunctions.SetTimeIncrement = HalpAcpiTimerSetTimeIncrement;
|
|
#endif
|
|
}
|
|
|
|
|
|
ULONG
|
|
HalSetTimeIncrement(
|
|
IN ULONG DesiredIncrement
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine initialize system time clock to generate an
|
|
interrupt at every DesiredIncrement interval
|
|
|
|
Arguments:
|
|
|
|
DesiredIncrement - Desired interval between every timer tick (in
|
|
100ns unit)
|
|
|
|
Return Value:
|
|
|
|
The *REAL* time increment set
|
|
|
|
--*/
|
|
{
|
|
return (HalpTimerFunctions.SetTimeIncrement)(DesiredIncrement);
|
|
}
|
|
|
|
|
|
VOID
|
|
HalCalibratePerformanceCounter(
|
|
IN LONG volatile *Number,
|
|
IN ULONGLONG NewCount
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine resets the performance counter value for the current
|
|
processor to zero, the reset is done such that the resulting value
|
|
is closely synchronized with other processors in the configuration
|
|
|
|
Arguments:
|
|
|
|
Number - Supplies a pointer to count of the number of processors in
|
|
the configuration
|
|
|
|
NewCount - Supplies the value to synchronize the counter too
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
{
|
|
(HalpTimerFunctions.CalibratePerfCount)(Number, NewCount);
|
|
}
|
|
|
|
|
|
#ifdef TIMER_DBG
|
|
static ULONG HalpQueryPerfLogIndex = 0;
|
|
#define MAX_QUERY_LOG 10
|
|
static LARGE_INTEGER HalpQueryPerfLog[MAX_QUERY_LOG];
|
|
static LARGE_INTEGER HalpQueryPerfTSLog[MAX_QUERY_LOG];
|
|
#endif
|
|
|
|
|
|
LARGE_INTEGER
|
|
KeQueryPerformanceCounter(
|
|
OUT PLARGE_INTEGER PerformanceFrequency OPTIONAL
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine returns current 64-bit performance counter and,
|
|
optionally, the Performance Frequency
|
|
|
|
N.B. The performace counter returned by this routine is
|
|
not necessary the value when this routine is just entered,
|
|
The value returned is actually the counter value at any point
|
|
between the routine is entered and is exited
|
|
|
|
Arguments:
|
|
|
|
PerformanceFrequency - optionally, supplies the address of a
|
|
variable to receive the performance counter
|
|
frequency
|
|
|
|
Return Value:
|
|
|
|
Current value of the performance counter will be returned
|
|
|
|
--*/
|
|
{
|
|
#ifdef TIMER_DBG
|
|
ULONG Index;
|
|
ULONG TSCounterHigh;
|
|
ULONG TSCounterLow;
|
|
KIRQL Irql;
|
|
ULONG flags;
|
|
extern BOOLEAN HalInitialized;
|
|
|
|
_asm {
|
|
pushfd
|
|
pop eax
|
|
mov flags, eax
|
|
cli
|
|
}
|
|
|
|
Index = HalpQueryPerfLogIndex % MAX_QUERY_LOG;
|
|
HalpQueryPerfLogIndex++;
|
|
|
|
HalpQueryPerfLog[Index] =
|
|
(HalpTimerFunctions.QueryPerfCount)(PerformanceFrequency);
|
|
|
|
_asm { rdtsc
|
|
mov TSCounterLow, eax
|
|
mov TSCounterHigh, edx };
|
|
|
|
HalpQueryPerfTSLog[Index].HighPart = TSCounterHigh;
|
|
HalpQueryPerfTSLog[Index].LowPart = TSCounterLow;
|
|
|
|
if (HalInitialized) {
|
|
ULONG PriorIndex;
|
|
|
|
PriorIndex = (Index == 0) ? MAX_QUERY_LOG - 1: Index - 1;
|
|
|
|
if (HalpQueryPerfLog[Index].QuadPart <
|
|
HalpQueryPerfLog[PriorIndex].QuadPart) {
|
|
|
|
ASSERT(FALSE);
|
|
}
|
|
}
|
|
|
|
_asm {
|
|
mov eax, flags
|
|
push eax
|
|
popfd
|
|
}
|
|
|
|
return HalpQueryPerfLog[Index];
|
|
|
|
#else
|
|
return (HalpTimerFunctions.QueryPerfCount)(PerformanceFrequency);
|
|
#endif
|
|
}
|
|
|
|
|
|
VOID
|
|
KeStallExecutionProcessor(
|
|
IN ULONG MicroSeconds
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This function stalls execution for the specified number of microseconds
|
|
|
|
Arguments:
|
|
|
|
MicroSeconds - Supplies the number of microseconds that execution is to be
|
|
stalled
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
{
|
|
(HalpTimerFunctions.StallExecProc)(MicroSeconds);
|
|
}
|