mirror of
https://github.com/reactos/reactos.git
synced 2024-11-20 06:15:26 +00:00
7a0c3bde06
svn path=/branches/condrv_restructure/; revision=63193
173 lines
4.4 KiB
C
173 lines
4.4 KiB
C
/*
|
|
* COPYRIGHT: GPL - See COPYING in the top level directory
|
|
* PROJECT: ReactOS Virtual DOS Machine
|
|
* FILE: clock.c
|
|
* PURPOSE: Clock for VDM
|
|
* PROGRAMMERS: Aleksandar Andrejevic <theflash AT sdf DOT lonestar DOT org>
|
|
* Hermes Belusca-Maito (hermes.belusca@sfr.fr)
|
|
*/
|
|
|
|
/* INCLUDES *******************************************************************/
|
|
|
|
#define NDEBUG
|
|
|
|
#include "emulator.h"
|
|
|
|
// #include "clock.h"
|
|
|
|
#include "hardware/cmos.h"
|
|
#include "hardware/ps2.h"
|
|
#include "hardware/timer.h"
|
|
#include "hardware/vga.h"
|
|
|
|
/* DEFINES ********************************************************************/
|
|
|
|
/*
|
|
* Activate IPS_DISPLAY if you want to display the
|
|
* number of instructions per second, as well as
|
|
* the computed number of ticks for the PIT.
|
|
*/
|
|
// #define IPS_DISPLAY
|
|
|
|
/*
|
|
* Activate WORKING_TIMER when the PIT timing problem is fixed.
|
|
*/
|
|
// #define WORKING_TIMER
|
|
|
|
|
|
/* Processor speed */
|
|
#define STEPS_PER_CYCLE 256
|
|
#define KBD_INT_CYCLES 16
|
|
|
|
/* VARIABLES ******************************************************************/
|
|
|
|
LARGE_INTEGER StartPerfCount, Frequency;
|
|
|
|
LARGE_INTEGER LastTimerTick, LastRtcTick, Counter;
|
|
LONGLONG TimerTicks;
|
|
DWORD StartTickCount, CurrentTickCount;
|
|
DWORD LastClockUpdate;
|
|
DWORD LastVerticalRefresh;
|
|
INT KeyboardIntCounter = 0;
|
|
|
|
#ifdef IPS_DISPLAY
|
|
DWORD LastCyclePrintout;
|
|
DWORD Cycles = 0;
|
|
#endif
|
|
|
|
/* PUBLIC FUNCTIONS ***********************************************************/
|
|
|
|
VOID ClockUpdate(VOID)
|
|
{
|
|
extern BOOLEAN CpuSimulate;
|
|
UINT i;
|
|
|
|
#ifdef WORKING_TIMER
|
|
DWORD PitResolution = PitGetResolution();
|
|
#endif
|
|
DWORD RtcFrequency = RtcGetTicksPerSecond();
|
|
|
|
/* Get the current number of ticks */
|
|
CurrentTickCount = GetTickCount();
|
|
|
|
#ifdef WORKING_TIMER
|
|
if ((PitResolution <= 1000) && (RtcFrequency <= 1000))
|
|
{
|
|
/* Calculate the approximate performance counter value instead */
|
|
Counter.QuadPart = StartPerfCount.QuadPart
|
|
+ (CurrentTickCount - StartTickCount)
|
|
* (Frequency.QuadPart / 1000);
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
/* Get the current performance counter value */
|
|
QueryPerformanceCounter(&Counter);
|
|
}
|
|
|
|
/* Get the number of PIT ticks that have passed */
|
|
TimerTicks = ((Counter.QuadPart - LastTimerTick.QuadPart)
|
|
* PIT_BASE_FREQUENCY) / Frequency.QuadPart;
|
|
|
|
/* Update the PIT */
|
|
if (TimerTicks > 0)
|
|
{
|
|
PitClock(TimerTicks);
|
|
LastTimerTick = Counter;
|
|
}
|
|
|
|
/* Check for RTC update */
|
|
if ((CurrentTickCount - LastClockUpdate) >= 1000)
|
|
{
|
|
RtcTimeUpdate();
|
|
LastClockUpdate = CurrentTickCount;
|
|
}
|
|
|
|
/* Check for RTC periodic tick */
|
|
if ((Counter.QuadPart - LastRtcTick.QuadPart)
|
|
>= (Frequency.QuadPart / (LONGLONG)RtcFrequency))
|
|
{
|
|
RtcPeriodicTick();
|
|
LastRtcTick = Counter;
|
|
}
|
|
|
|
/* Check for vertical retrace */
|
|
if ((CurrentTickCount - LastVerticalRefresh) >= 15)
|
|
{
|
|
VgaRefreshDisplay();
|
|
LastVerticalRefresh = CurrentTickCount;
|
|
}
|
|
|
|
if (++KeyboardIntCounter == KBD_INT_CYCLES)
|
|
{
|
|
GenerateKeyboardInterrupts();
|
|
KeyboardIntCounter = 0;
|
|
}
|
|
|
|
/* Horizontal retrace occurs as fast as possible */
|
|
VgaHorizontalRetrace();
|
|
|
|
/* Continue CPU emulation */
|
|
for (i = 0; VdmRunning && CpuSimulate && (i < STEPS_PER_CYCLE); i++)
|
|
{
|
|
EmulatorStep();
|
|
#ifdef IPS_DISPLAY
|
|
Cycles++;
|
|
#endif
|
|
}
|
|
|
|
#ifdef IPS_DISPLAY
|
|
if ((CurrentTickCount - LastCyclePrintout) >= 1000)
|
|
{
|
|
DPRINT1("NTVDM: %lu Instructions Per Second; TimerTicks = %I64d\n", Cycles, TimerTicks);
|
|
LastCyclePrintout = CurrentTickCount;
|
|
Cycles = 0;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
BOOLEAN ClockInitialize(VOID)
|
|
{
|
|
/* Initialize the performance counter (needed for hardware timers) */
|
|
if (!QueryPerformanceFrequency(&Frequency))
|
|
{
|
|
wprintf(L"FATAL: Performance counter not available\n");
|
|
return FALSE;
|
|
}
|
|
|
|
/* Find the starting performance and tick count */
|
|
StartTickCount = GetTickCount();
|
|
QueryPerformanceCounter(&StartPerfCount);
|
|
|
|
/* Set the different last counts to the starting count */
|
|
LastClockUpdate = LastVerticalRefresh =
|
|
#ifdef IPS_DISPLAY
|
|
LastCyclePrintout =
|
|
#endif
|
|
StartTickCount;
|
|
|
|
/* Set the last timer ticks to the current time */
|
|
LastTimerTick = LastRtcTick = StartPerfCount;
|
|
|
|
return TRUE;
|
|
}
|