emu48-mirror/Sources/Emu48/ENGINE.C

643 lines
18 KiB
C
Raw Normal View History

2024-03-19 22:24:30 +01:00
/*
* engine.c
*
* This file is part of Emu48
*
* Copyright (C) 1995 Sebastien Carlier
*
*/
#include "pch.h"
#include "Emu48.h"
2024-03-19 22:25:45 +01:00
#include "Opcodes.h"
2024-03-19 22:36:03 +01:00
#include "io.h"
2024-03-19 22:25:45 +01:00
#include "debugger.h"
2024-03-19 22:24:30 +01:00
2024-03-19 22:25:45 +01:00
#define SAMPLE 16384 // speed adjust sample frequency
2024-03-19 22:24:30 +01:00
2024-03-19 22:37:03 +01:00
BOOL bInterrupt = FALSE;
UINT nState = SM_INVALID;
UINT nNextState = SM_RUN;
BOOL bEnableSlow = TRUE; // slow down is enabled
2024-03-19 22:37:03 +01:00
BOOL bRealSpeed = FALSE;
BOOL bKeySlow = FALSE; // slow down for key emulation
BOOL bSoundSlow = FALSE; // slow down for sound emulation
UINT nOpcSlow = 0; // no. of opcodes to slow down
2024-03-19 22:37:03 +01:00
BOOL bCommInit = FALSE; // COM port not open
2024-03-19 22:24:30 +01:00
2024-03-19 22:37:03 +01:00
CHIPSET Chipset;
2024-03-19 22:24:30 +01:00
2024-03-19 22:37:03 +01:00
TCHAR szSerialWire[16]; // devicename for wire port
TCHAR szSerialIr[16]; // devicename for IR port
DWORD dwSXCycles = 82; // SX cpu cycles in interval
DWORD dwGXCycles = 123; // GX cpu cycles in interval
// variables for debugger engine
HANDLE hEventDebug; // event handle to stop cpu thread
2024-03-19 22:37:54 +01:00
BOOL bDbgAutoStateCtrl = TRUE; // debugger suspend control by SwitchToState()
INT nDbgState = DBG_OFF; // state of debugger
2024-03-19 22:37:03 +01:00
BOOL bDbgNOP3 = FALSE; // halt on NOP3 (#420 opcode)
BOOL bDbgRPL = FALSE; // halt on RPL entry
BOOL bDbgCode = FALSE; // halt on DOCODE entry
BOOL bDbgSkipInt = FALSE; // execute interrupt handler
DWORD dwDbgStopPC = -1; // stop address for goto cursor
DWORD dwDbgRplPC = -1; // stop address for RPL breakpoint
DWORD dwDbgRstkp; // stack recursion level of step over end
DWORD dwDbgRstk; // possible return address
DWORD *pdwInstrArray = NULL; // last instruction array
WORD wInstrSize = 256; // size of last instruction array
2024-03-19 22:37:03 +01:00
WORD wInstrWp; // write pointer of instruction array
WORD wInstrRp; // read pointer of instruction array
2024-03-19 22:24:30 +01:00
2024-03-19 22:38:33 +01:00
static INT nDbgRplBreak = BN_ASM; // flag for RPL breakpoint detection
2024-03-19 22:37:54 +01:00
static INT nDbgOldState = DBG_OFF; // old state of debugger for suspend/resume
2024-03-19 22:36:03 +01:00
static BOOL bCpuSlow = FALSE; // enable/disable real speed
static DWORD dwEDbgT2 = 0; // debugger timer2 emulation
static DWORD dwEDbgCycles = 0; // debugger cycle counter
2024-03-19 22:24:30 +01:00
2024-03-19 22:25:45 +01:00
static DWORD dwOldCyc; // cpu cycles at last event
static DWORD dwSpeedRef; // timer value at last event
static DWORD dwTickRef; // sample timer ticks
2024-03-19 22:24:30 +01:00
#include "Ops.h"
2024-03-19 22:36:03 +01:00
// save last instruction in circular instruction buffer
static __inline VOID SaveInstrAddr(DWORD dwAddr)
{
EnterCriticalSection(&csDbgLock);
2024-03-19 22:36:03 +01:00
{
if (pdwInstrArray) // circular buffer allocated
{
pdwInstrArray[wInstrWp] = dwAddr;
wInstrWp = (wInstrWp + 1) % wInstrSize;
if (wInstrWp == wInstrRp)
wInstrRp = (wInstrRp + 1) % wInstrSize;
}
2024-03-19 22:36:03 +01:00
}
LeaveCriticalSection(&csDbgLock);
2024-03-19 22:36:03 +01:00
return;
}
static __inline VOID Debugger(VOID) // debugger part
{
LARGE_INTEGER lDummyInt; // sample timer ticks
2024-03-19 22:37:03 +01:00
BOOL bStopEmulation;
2024-03-19 22:36:03 +01:00
LPBYTE I = FASTPTR(Chipset.pc); // get opcode stream
2024-03-19 22:38:33 +01:00
UpdateDbgCycleCounter(); // update 64 bit cpu cycle counter
2024-03-19 22:36:03 +01:00
SaveInstrAddr(Chipset.pc); // save pc in last instruction buffer
2024-03-19 22:38:33 +01:00
nDbgRplBreak = BN_ASM; // notify ASM breakpoint
2024-03-19 22:37:54 +01:00
2024-03-19 22:36:03 +01:00
// check for code breakpoints
bStopEmulation = CheckBreakpoint(Chipset.pc, 1, BP_EXEC);
// check for memory breakpoints, opcode #14x or #15x
if (I[0] == 0x1 && (I[1] == 0x4 || I[1] == 0x5))
{
DWORD dwData = (I[2] & 0x1) ? Chipset.d1 : Chipset.d0;
UINT nType = (I[2] & 0x2) ? BP_READ : BP_WRITE;
DWORD dwRange;
if (I[1] == 0x4) // B,A
{
dwRange = (I[2] & 0x8) ? 2 : 5;
}
2024-03-19 22:37:54 +01:00
else // number of nibbles, (P,WP,XS,X,S,M,W)
2024-03-19 22:36:03 +01:00
{
2024-03-19 22:37:03 +01:00
dwRange = (I[2] & 0x8) ? (I[3]+1) : (F_l[I[3]]);
2024-03-19 22:36:03 +01:00
}
#if defined DEBUG_DEBUGGER
{
2024-03-19 22:37:03 +01:00
TCHAR buffer[256];
wsprintf(buffer,_T("Memory breakpoint %.5lx, %u\n",dwData,dwRange));
2024-03-19 22:36:03 +01:00
OutputDebugString(buffer);
}
#endif
bStopEmulation |= CheckBreakpoint(dwData, dwRange, nType);
}
2024-03-19 22:37:03 +01:00
// check for step cursor
bStopEmulation |= (dwDbgStopPC == Chipset.pc);
2024-03-19 22:36:03 +01:00
// NOP3, opcode #420 (GOC)
if (bDbgNOP3 && I[0] == 0x4 && I[1] == 0x2 && I[2] == 0x0)
bStopEmulation = TRUE;
2024-03-19 22:37:03 +01:00
// stop on first instruction of DOCODE object
if (bDbgCode && (Chipset.pc == 0x02DDE || Chipset.pc == 0x02E3C))
{
// return address
DWORD dwAddr = Chipset.rstk[(Chipset.rstkp-1)&7];
_ASSERT(I[0] == 0 && I[1] == 1); // stopped at RTN opcode
if (MapData(dwAddr) != M_ROM) // address not in ROM
dwDbgStopPC = dwAddr; // then stop
}
2024-03-19 22:38:33 +01:00
// check for RPL breakpoint
if (dwDbgRplPC == Chipset.pc)
{
dwDbgRplPC = -1;
nDbgRplBreak = bStopEmulation ? BN_ASM_BT : BN_RPL;
bStopEmulation = TRUE;
}
2024-03-19 22:36:03 +01:00
// RPL breakpoints, PC=(A), opcode #808C or PC=(C), opcode #808E
2024-03-19 22:37:03 +01:00
if (I[0] == 0x8 && I[1] == 0x0 && I[2] == 0x8 && (I[3] == 0xC || I[3] == 0xE ))
2024-03-19 22:36:03 +01:00
{
2024-03-19 22:37:03 +01:00
// get next RPL entry
DWORD dwAddr = Npack((I[3] == 0xC) ? Chipset.A : Chipset.C,5);
2024-03-19 22:36:03 +01:00
2024-03-19 22:37:03 +01:00
if (bDbgRPL || CheckBreakpoint(dwAddr, 1, BP_RPL))
2024-03-19 22:36:03 +01:00
{
2024-03-19 22:37:03 +01:00
BYTE byRplPtr[5];
Npeek(byRplPtr,dwAddr,5); // get PC address of next opcode
dwDbgRplPC = Npack(byRplPtr,5); // set RPL breakpoint
2024-03-19 22:36:03 +01:00
}
}
// step over interrupt execution
if (bDbgSkipInt && !bStopEmulation && !Chipset.inte)
return;
// check for step into
bStopEmulation |= (nDbgState == DBG_STEPINTO);
// check for step over
bStopEmulation |= (nDbgState == DBG_STEPOVER) && dwDbgRstkp == Chipset.rstkp;
// check for step out, something was popped from hardware stack
if (nDbgState == DBG_STEPOUT && dwDbgRstkp == Chipset.rstkp)
{
_ASSERT(bStopEmulation == FALSE);
if ((bStopEmulation = (Chipset.pc == dwDbgRstk)) == FALSE)
{
// it was C=RSTK, check for next object popped from hardware stack
dwDbgRstkp = (Chipset.rstkp-1)&7;
dwDbgRstk = Chipset.rstk[dwDbgRstkp];
}
}
if (bStopEmulation) // stop condition
{
StopTimers(); // hold timer values when emulator is stopped
if (Chipset.IORam[TIMER2_CTRL]&RUN) // check if timer running
{
if (dwEDbgT2 == Chipset.t2)
{
// cpu cycles for one timer2 tick elapsed
2024-03-19 22:37:54 +01:00
if ((DWORD) (Chipset.cycles & 0xFFFFFFFF) - dwEDbgCycles
2024-03-19 22:36:03 +01:00
>= (SAMPLE / 8192) * (DWORD) T2CYCLES)
{
--Chipset.t2;
// adjust cycles reference
dwEDbgCycles += (SAMPLE / 8192) * T2CYCLES;
}
}
else // new timer2 value
{
// new cycle reference
dwEDbgCycles = (DWORD) (Chipset.cycles & 0xFFFFFFFF);
}
// check rising edge of Bit 8 of timer2
if ((dwEDbgT2 & 0x100) == 0 && (Chipset.t2 & 0x100) != 0)
Chipset.t1 = (Chipset.t1 - 1) & 0xF;
}
dwEDbgT2 = Chipset.t2; // timer2 check reference value
2024-03-19 22:37:54 +01:00
// redraw debugger window and stop
2024-03-19 22:38:33 +01:00
NotifyDebugger(nDbgRplBreak);
2024-03-19 22:36:03 +01:00
WaitForSingleObject(hEventDebug,INFINITE);
StartTimers(); // continue timers
2024-03-19 22:38:33 +01:00
if (nDbgState >= DBG_OFF) // if debugger is active
2024-03-19 22:37:54 +01:00
{
Chipset.Shutdn = FALSE;
Chipset.bShutdnWake = FALSE;
}
2024-03-19 22:36:03 +01:00
// init slow down part
dwOldCyc = (DWORD) (Chipset.cycles & 0xFFFFFFFF);
QueryPerformanceCounter(&lDummyInt);
dwSpeedRef = lDummyInt.LowPart;
}
2024-03-19 22:37:54 +01:00
return;
}
VOID SuspendDebugger(VOID)
{
// auto control enabled, emulation halted by debugger
if (bDbgAutoStateCtrl && nDbgState > DBG_OFF)
{
nDbgOldState = nDbgState; // save old state
nDbgState = DBG_SUSPEND; // suspend state
SetEvent(hEventDebug); // exit debugger
}
return;
}
VOID ResumeDebugger(VOID)
{
// auto control enabled, debugger is suspended
if (bDbgAutoStateCtrl && nDbgState == DBG_SUSPEND)
{
// active RPL breakpoint
2024-03-19 22:38:33 +01:00
if (nDbgRplBreak) dwDbgRplPC = Chipset.pc;
2024-03-19 22:37:54 +01:00
nDbgState = nDbgOldState; // set to old debugger state
2024-03-19 22:38:33 +01:00
if (Chipset.Shutdn) // inside shutdown
SetEvent(hEventShutdn); // leave it to call debugger
2024-03-19 22:37:54 +01:00
}
return;
2024-03-19 22:36:03 +01:00
}
2024-03-19 22:25:45 +01:00
static __inline VOID CheckDisp(BOOL bSync)
{
if (disp == 0) return; // no display update need
2024-03-19 22:24:30 +01:00
2024-03-19 22:25:45 +01:00
// update display when drawing top line or display is off
if (bSync && GetLineCounter() != 0x3F && (Chipset.IORam[0x00]&8))
return;
2024-03-19 22:24:30 +01:00
2024-03-19 22:25:45 +01:00
_ASSERT((disp & DISP_POINTER) == 0); // display pointer already updated
if (disp & DISP_MAIN) UpdateMainDisplay();
if (disp & DISP_MENUE) UpdateMenuDisplay();
_ASSERT((disp & DISP_ANNUN) == 0); // annunciators already updated
disp = 0; // display updated
return;
}
2024-03-19 22:24:30 +01:00
2024-03-19 22:25:45 +01:00
static __inline VOID AdjustSpeed(VOID) // adjust emulation speed
2024-03-19 22:24:30 +01:00
{
// emulation slow down
if ( bEnableSlow
&& (bCpuSlow || bKeySlow || bSoundSlow || nOpcSlow > 0))
2024-03-19 22:24:30 +01:00
{
2024-03-19 22:36:03 +01:00
DWORD dwCycles,dwTicks;
2024-03-19 22:38:33 +01:00
EnterCriticalSection(&csSlowLock);
2024-03-19 22:24:30 +01:00
{
2024-03-19 22:38:33 +01:00
// cycles elapsed for next check
if ((dwCycles = (DWORD) (Chipset.cycles & 0xFFFFFFFF)-dwOldCyc) >= (DWORD) T2CYCLES)
2024-03-19 22:24:30 +01:00
{
2024-03-19 22:38:33 +01:00
LARGE_INTEGER lAct;
do
{
VERIFY(QueryPerformanceCounter(&lAct));
2024-03-19 22:24:30 +01:00
// get time difference
dwTicks = lAct.LowPart - dwSpeedRef;
2024-03-19 22:38:33 +01:00
}
// ticks elapsed or negative number (workaround for QueryPerformanceCounter() in Win2k)
while (dwTicks <= dwTickRef || (dwTicks & 0x80000000) != 0);
dwOldCyc += T2CYCLES; // adjust cycles reference
dwSpeedRef += dwTickRef; // adjust reference time
2024-03-19 22:36:03 +01:00
}
if (nOpcSlow > 0) --nOpcSlow; // decr. slow down opcode counter
2024-03-19 22:24:30 +01:00
}
2024-03-19 22:38:33 +01:00
LeaveCriticalSection(&csSlowLock);
2024-03-19 22:24:30 +01:00
}
return;
}
2024-03-19 22:37:54 +01:00
VOID CheckSerial(VOID)
{
// COM port closed and serial on
if (bCommInit == FALSE && (Chipset.IORam[IOC] & SON) != 0)
{
bCommInit = CommOpen(szSerialWire,szSerialIr); // open COM ports
}
// COM port opened and serial off
if (bCommInit == TRUE && (Chipset.IORam[IOC] & SON) == 0)
{
CommClose(); // close COM port
bCommInit = FALSE;
}
return;
}
VOID InitAdjustSpeed(VOID)
{
// slow down function not initalized
if (!bEnableSlow || (!bCpuSlow && !bKeySlow && !bSoundSlow && nOpcSlow == 0))
{
LARGE_INTEGER lTime; // sample timer ticks
// save reference cycles
dwOldCyc = (DWORD) (Chipset.cycles & 0xFFFFFFFF);
QueryPerformanceCounter(&lTime); // get timer ticks
dwSpeedRef = lTime.LowPart; // save reference time
}
return;
}
2024-03-19 22:25:45 +01:00
VOID AdjKeySpeed(VOID) // slow down key repeat
2024-03-19 22:24:30 +01:00
{
WORD i;
BOOL bKey;
bKey = FALSE; // search for a pressed key
2024-03-19 22:37:03 +01:00
for (i = 0;i < ARRAYSIZEOF(Chipset.Keyboard_Row) && !bKey;++i)
2024-03-19 22:24:30 +01:00
bKey = (Chipset.Keyboard_Row[i] != 0);
2024-03-19 22:38:33 +01:00
EnterCriticalSection(&csSlowLock);
2024-03-19 22:24:30 +01:00
{
if (bKey) // key pressed
2024-03-19 22:38:33 +01:00
{
InitAdjustSpeed(); // init variables if necessary
2024-03-19 22:38:33 +01:00
}
bKeySlow = bKey; // save new state
2024-03-19 22:24:30 +01:00
}
2024-03-19 22:38:33 +01:00
LeaveCriticalSection(&csSlowLock);
2024-03-19 22:24:30 +01:00
return;
}
2024-03-19 22:25:45 +01:00
VOID SetSpeed(BOOL bAdjust) // set emulation speed
2024-03-19 22:24:30 +01:00
{
2024-03-19 22:38:33 +01:00
EnterCriticalSection(&csSlowLock);
2024-03-19 22:24:30 +01:00
{
2024-03-19 22:38:33 +01:00
if (bAdjust) // switch to real speed
{
InitAdjustSpeed(); // init variables if necessary
2024-03-19 22:38:33 +01:00
}
bCpuSlow = bAdjust; // save emulation speed
2024-03-19 22:24:30 +01:00
}
2024-03-19 22:38:33 +01:00
LeaveCriticalSection(&csSlowLock);
2024-03-19 22:37:54 +01:00
return;
2024-03-19 22:24:30 +01:00
}
2024-03-19 22:25:45 +01:00
VOID UpdateKdnBit(VOID) // update KDN bit
2024-03-19 22:24:30 +01:00
{
2024-03-19 22:38:33 +01:00
if ( Chipset.intk
&& (Chipset.IORam[TIMER2_CTRL]&RUN) != 0
&& (DWORD) (Chipset.cycles & 0xFFFFFFFF) - Chipset.dwKdnCycles > (DWORD) T2CYCLES * 16)
2024-03-19 22:37:54 +01:00
IOBit(SRQ2,KDN,Chipset.in != 0);
return;
2024-03-19 22:24:30 +01:00
}
2024-03-19 22:25:45 +01:00
BOOL WaitForSleepState(VOID) // wait for cpu SHUTDN then sleep state
2024-03-19 22:24:30 +01:00
{
2024-03-19 22:37:03 +01:00
DWORD dwRefTime;
2024-03-19 22:37:54 +01:00
SuspendDebugger(); // suspend debugger
2024-03-19 22:24:30 +01:00
2024-03-19 22:37:54 +01:00
dwRefTime = timeGetTime();
2024-03-19 22:24:30 +01:00
// wait for the SHUTDN command with 1.5 sec timeout
while (timeGetTime() - dwRefTime < 1500L && !Chipset.Shutdn)
Sleep(0);
if (Chipset.Shutdn) // not timeout, cpu is down
2024-03-19 22:37:03 +01:00
SwitchToState(SM_SLEEP); // go to sleep state
2024-03-19 22:37:54 +01:00
else
ResumeDebugger(); // timeout, resume to debugger
2024-03-19 22:24:30 +01:00
2024-03-19 22:37:03 +01:00
return SM_SLEEP != nNextState; // state not changed, emulator was busy
2024-03-19 22:24:30 +01:00
}
UINT SwitchToState(UINT nNewState)
{
2024-03-19 22:37:54 +01:00
UINT nOldState = nState;
2024-03-19 22:24:30 +01:00
if (nState == nNewState) return nOldState;
switch (nState)
{
2024-03-19 22:37:03 +01:00
case SM_RUN: // Run
2024-03-19 22:24:30 +01:00
switch (nNewState)
{
2024-03-19 22:37:03 +01:00
case SM_INVALID: // -> Invalid
nNextState = SM_INVALID;
2024-03-19 22:24:30 +01:00
if (Chipset.Shutdn)
2024-03-19 22:25:45 +01:00
SetEvent(hEventShutdn);
2024-03-19 22:24:30 +01:00
else
bInterrupt = TRUE;
2024-03-19 22:37:54 +01:00
SuspendDebugger(); // suspend debugger
2024-03-19 22:24:30 +01:00
while (nState!=nNextState) Sleep(0);
break;
2024-03-19 22:37:03 +01:00
case SM_RETURN: // -> Return
DisableDebugger(); // disable debugger
nNextState = SM_INVALID;
2024-03-19 22:24:30 +01:00
if (Chipset.Shutdn)
2024-03-19 22:25:45 +01:00
SetEvent(hEventShutdn);
2024-03-19 22:24:30 +01:00
else
bInterrupt = TRUE;
while (nState!=nNextState) Sleep(0);
2024-03-19 22:37:03 +01:00
nNextState = SM_RETURN;
2024-03-19 22:25:45 +01:00
SetEvent(hEventShutdn);
2024-03-19 22:37:03 +01:00
WaitForSingleObject(hThread,INFINITE);
2024-03-19 22:24:30 +01:00
break;
2024-03-19 22:37:03 +01:00
case SM_SLEEP: // -> Sleep
nNextState = SM_SLEEP;
bInterrupt = TRUE; // exit main loop
2024-03-19 22:37:54 +01:00
SuspendDebugger(); // suspend debugger
SetEvent(hEventShutdn); // exit shutdown
2024-03-19 22:24:30 +01:00
while (nState!=nNextState) Sleep(0);
2024-03-19 22:37:03 +01:00
bInterrupt = FALSE;
ResetEvent(hEventDebug);
ResetEvent(hEventShutdn);
2024-03-19 22:24:30 +01:00
break;
}
break;
2024-03-19 22:37:03 +01:00
case SM_INVALID: // Invalid
2024-03-19 22:24:30 +01:00
switch (nNewState)
{
2024-03-19 22:37:03 +01:00
case SM_RUN: // -> Run
nNextState = SM_RUN;
2024-03-19 22:36:03 +01:00
// don't enter opcode loop on interrupt request
bInterrupt = Chipset.Shutdn || Chipset.SoftInt;
2024-03-19 22:37:54 +01:00
ResumeDebugger();
2024-03-19 22:25:45 +01:00
SetEvent(hEventShutdn);
2024-03-19 22:24:30 +01:00
while (nState!=nNextState) Sleep(0);
break;
2024-03-19 22:37:03 +01:00
case SM_RETURN: // -> Return
2024-03-19 22:37:54 +01:00
DisableDebugger(); // disable debugger
2024-03-19 22:37:03 +01:00
nNextState = SM_RETURN;
2024-03-19 22:25:45 +01:00
SetEvent(hEventShutdn);
2024-03-19 22:37:03 +01:00
WaitForSingleObject(hThread,INFINITE);
2024-03-19 22:24:30 +01:00
break;
2024-03-19 22:37:03 +01:00
case SM_SLEEP: // -> Sleep
nNextState = SM_SLEEP;
2024-03-19 22:25:45 +01:00
SetEvent(hEventShutdn);
2024-03-19 22:24:30 +01:00
while (nState!=nNextState) Sleep(0);
break;
}
break;
2024-03-19 22:37:03 +01:00
case SM_SLEEP: // Sleep
2024-03-19 22:24:30 +01:00
switch (nNewState)
{
2024-03-19 22:37:03 +01:00
case SM_RUN: // -> Run
nNextState = SM_RUN;
// don't enter opcode loop on interrupt request
2024-03-19 22:37:54 +01:00
bInterrupt = (nDbgState == DBG_OFF) && (Chipset.Shutdn || Chipset.SoftInt);
ResumeDebugger();
2024-03-19 22:37:03 +01:00
SetEvent(hEventShutdn); // leave sleep state
2024-03-19 22:24:30 +01:00
break;
2024-03-19 22:37:03 +01:00
case SM_INVALID: // -> Invalid
nNextState = SM_INVALID;
2024-03-19 22:25:45 +01:00
SetEvent(hEventShutdn);
2024-03-19 22:24:30 +01:00
while (nState!=nNextState) Sleep(0);
break;
2024-03-19 22:37:03 +01:00
case SM_RETURN: // -> Return
2024-03-19 22:37:54 +01:00
DisableDebugger(); // disable debugger
2024-03-19 22:37:03 +01:00
nNextState = SM_INVALID;
2024-03-19 22:25:45 +01:00
SetEvent(hEventShutdn);
2024-03-19 22:24:30 +01:00
while (nState!=nNextState) Sleep(0);
2024-03-19 22:37:03 +01:00
nNextState = SM_RETURN;
2024-03-19 22:25:45 +01:00
SetEvent(hEventShutdn);
2024-03-19 22:37:03 +01:00
WaitForSingleObject(hThread,INFINITE);
2024-03-19 22:24:30 +01:00
break;
}
break;
}
return nOldState;
}
UINT WorkerThread(LPVOID pParam)
{
2024-03-19 22:25:45 +01:00
LARGE_INTEGER lDummyInt; // sample timer ticks
QueryPerformanceFrequency(&lDummyInt); // init timer ticks
lDummyInt.QuadPart /= SAMPLE; // calculate sample ticks
dwTickRef = lDummyInt.LowPart; // sample timer ticks
_ASSERT(dwTickRef); // tick resolution error
2024-03-19 22:24:30 +01:00
loop:
2024-03-19 22:37:03 +01:00
while (nNextState == SM_INVALID) // go into invalid state
2024-03-19 22:24:30 +01:00
{
OnToolMacroStop(); // close open keyboard macro handler
2024-03-19 22:25:45 +01:00
CommClose(); // close COM port
bCommInit = FALSE; // COM port not open
2024-03-19 22:37:03 +01:00
nState = SM_INVALID; // in invalid state
2024-03-19 22:25:45 +01:00
WaitForSingleObject(hEventShutdn,INFINITE);
2024-03-19 22:37:03 +01:00
if (nNextState == SM_RETURN) // go into return state
2024-03-19 22:24:30 +01:00
{
2024-03-19 22:37:03 +01:00
nState = SM_RETURN; // in return state
2024-03-19 22:24:30 +01:00
return 0; // kill thread
}
2024-03-19 22:37:54 +01:00
CheckSerial(); // test if UART on
2024-03-19 22:24:30 +01:00
}
2024-03-19 22:37:03 +01:00
while (nNextState == SM_RUN)
2024-03-19 22:24:30 +01:00
{
2024-03-19 22:37:03 +01:00
if (nState != SM_RUN)
2024-03-19 22:24:30 +01:00
{
2024-03-19 22:37:03 +01:00
nState = SM_RUN;
2024-03-19 22:25:45 +01:00
// clear port2 status bits
Chipset.cards_status &= ~(PORT2_PRESENT | PORT2_WRITE);
if (pbyPort2 || Port2) // card plugged in port2
2024-03-19 22:24:30 +01:00
{
2024-03-19 22:25:45 +01:00
Chipset.cards_status |= PORT2_PRESENT;
if (bPort2Writeable) // is card writeable
Chipset.cards_status |= PORT2_WRITE;
2024-03-19 22:24:30 +01:00
}
2024-03-19 22:36:03 +01:00
// card detection off and timer running
if ((Chipset.IORam[CARDCTL] & ECDT) == 0 && (Chipset.IORam[TIMER2_CTRL] & RUN) != 0)
{
BOOL bNINT2 = Chipset.IORam[SRQ1] == 0 && (Chipset.IORam[SRQ2] & LSRQ) == 0;
BOOL bNINT = (Chipset.IORam[CARDCTL] & SMP) == 0;
// state of CDT2
bNINT2 = bNINT2 && (Chipset.cards_status & (P2W|P2C)) != P2C;
// state of CDT1
bNINT = bNINT && (Chipset.cards_status & (P1W|P1C)) != P1C;
IOBit(SRQ2,NINT2,bNINT2);
IOBit(SRQ2,NINT,bNINT);
}
2024-03-19 22:25:45 +01:00
RomSwitch(Chipset.Bank_FF); // select HP49G ROM bank and update memory mapping
UpdateContrast(Chipset.contrast);
2024-03-19 22:24:30 +01:00
UpdateDisplayPointers();
UpdateMainDisplay();
UpdateMenuDisplay();
UpdateAnnunciators();
2024-03-19 22:25:45 +01:00
// init speed reference
dwOldCyc = (DWORD) (Chipset.cycles & 0xFFFFFFFF);
2024-03-19 22:24:30 +01:00
QueryPerformanceCounter(&lDummyInt);
dwSpeedRef = lDummyInt.LowPart;
2024-03-19 22:25:45 +01:00
SetHP48Time(); // update HP48 time & date
2024-03-19 22:37:54 +01:00
// start display counter/update engine
StartDisplay((BYTE)(((Chipset.IORam[LINECOUNT+1]<<4)|Chipset.IORam[LINECOUNT])&0x3F));
StartBatMeasure(); // start battery measurement
StartTimers();
2024-03-19 22:24:30 +01:00
}
PCHANGED;
while (!bInterrupt)
{
2024-03-19 22:37:54 +01:00
if (nDbgState > DBG_OFF) // debugger active
{
Debugger();
// if suspended skip next opcode execution
if (nDbgState == DBG_SUSPEND)
{
if (Chipset.Shutdn) break;
continue;
}
}
2024-03-19 22:25:45 +01:00
2024-03-19 22:36:03 +01:00
EvalOpcode(FASTPTR(Chipset.pc)); // execute opcode
2024-03-19 22:25:45 +01:00
// check for display update in BW mode
if (!bGrayscale) CheckDisp(!Chipset.Shutdn);
2024-03-19 22:25:45 +01:00
AdjustSpeed(); // adjust emulation speed
2024-03-19 22:24:30 +01:00
}
2024-03-19 22:25:45 +01:00
bInterrupt = FALSE; // be sure to reenter opcode loop
// enter SHUTDN handler only in RUN mode
if (Chipset.Shutdn && !(nDbgState == DBG_STEPINTO || nDbgState == DBG_STEPOVER))
2024-03-19 22:24:30 +01:00
{
2024-03-19 22:36:03 +01:00
if (!Chipset.SoftInt) // ignore SHUTDN on interrupt request
WaitForSingleObject(hEventShutdn,INFINITE);
else
Chipset.bShutdnWake = TRUE; // waked by interrupt
2024-03-19 22:25:45 +01:00
if (Chipset.bShutdnWake) // waked up by timer, keyboard or serial
{
Chipset.bShutdnWake = FALSE;
Chipset.Shutdn = FALSE;
// init speed reference
dwOldCyc = (DWORD) (Chipset.cycles & 0xFFFFFFFF);
QueryPerformanceCounter(&lDummyInt);
dwSpeedRef = lDummyInt.LowPart;
nOpcSlow = 0; // no opcodes to slow down
2024-03-19 22:25:45 +01:00
}
2024-03-19 22:24:30 +01:00
}
if (Chipset.SoftInt)
{
Chipset.SoftInt = FALSE;
if (Chipset.inte)
{
Chipset.inte = FALSE;
rstkpush(Chipset.pc);
Chipset.pc = 0xf;
}
}
}
2024-03-19 22:37:03 +01:00
_ASSERT(nNextState != SM_RUN);
2024-03-19 22:36:03 +01:00
2024-03-19 22:37:54 +01:00
StopDisplay(); // stop display counter/update
StopBatMeasure(); // stop battery measurement
2024-03-19 22:36:03 +01:00
StopTimers();
2024-03-19 22:37:03 +01:00
while (nNextState == SM_SLEEP) // go into sleep state
2024-03-19 22:24:30 +01:00
{
2024-03-19 22:37:03 +01:00
nState = SM_SLEEP; // in sleep state
2024-03-19 22:25:45 +01:00
WaitForSingleObject(hEventShutdn,INFINITE);
2024-03-19 22:24:30 +01:00
}
goto loop;
UNREFERENCED_PARAMETER(pParam);
}