2020-11-26 21:50:06 +00:00
/*
AppleWin : An Apple //e emulator for Windows
Copyright ( C ) 1994 - 1996 , Michael O ' Brien
Copyright ( C ) 1999 - 2001 , Oliver Schmidt
Copyright ( C ) 2002 - 2005 , Tom Charlesworth
Copyright ( C ) 2006 - 2014 , Tom Charlesworth , Michael Pohoreski
AppleWin is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 2 of the License , or
( at your option ) any later version .
AppleWin is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with AppleWin ; if not , write to the Free Software
Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
/* Description: main
*
* Author : Various
*/
# include "StdAfx.h"
# include "Windows/AppleWin.h"
2020-12-20 15:32:51 +00:00
# include "Interface.h"
2020-11-26 21:50:06 +00:00
# include "Utilities.h"
# include "CmdLine.h"
# include "Debug.h"
# include "Log.h"
# include "Memory.h"
# include "Mockingboard.h"
# include "MouseInterface.h"
# include "ParallelPrinter.h"
# include "Registry.h"
# include "Riff.h"
# include "SaveState.h"
# include "SoundCore.h"
# include "Speaker.h"
# ifdef USE_SPEECH_API
# include "Speech.h"
# endif
2020-12-24 15:08:50 +00:00
# include "Windows/Win32Frame.h"
2020-11-26 21:50:06 +00:00
# include "RGBMonitor.h"
# include "NTSC.h"
# include "Configuration/About.h"
# include "Configuration/PropertySheet.h"
# include "Tfe/Tfe.h"
//=================================================
static bool g_bLoadedSaveState = false ;
static bool g_bSysClkOK = false ;
bool g_bRestartFullScreen = false ;
//===========================================================================
bool GetLoadedSaveStateFlag ( void )
{
return g_bLoadedSaveState ;
}
2020-12-24 15:08:50 +00:00
void Win32Frame : : SetLoadedSaveStateFlag ( const bool bFlag )
2020-11-26 21:50:06 +00:00
{
g_bLoadedSaveState = bFlag ;
}
bool GetHookAltGrControl ( void )
{
return g_bHookAltGrControl ;
}
static void ResetToLogoMode ( void )
{
g_nAppMode = MODE_LOGO ;
2020-12-24 15:08:50 +00:00
GetFrame ( ) . SetLoadedSaveStateFlag ( false ) ;
2020-11-26 21:50:06 +00:00
}
//---------------------------------------------------------------------------
static bool g_bPriorityNormal = true ;
// Make APPLEWIN process higher priority
void SetPriorityAboveNormal ( void )
{
if ( ! g_bPriorityNormal )
return ;
if ( SetPriorityClass ( GetCurrentProcess ( ) , ABOVE_NORMAL_PRIORITY_CLASS ) )
{
SetThreadPriority ( GetCurrentThread ( ) , THREAD_PRIORITY_ABOVE_NORMAL ) ;
g_bPriorityNormal = false ;
}
}
// Make APPLEWIN process normal priority
void SetPriorityNormal ( void )
{
if ( g_bPriorityNormal )
return ;
if ( SetPriorityClass ( GetCurrentProcess ( ) , NORMAL_PRIORITY_CLASS ) )
{
SetThreadPriority ( GetCurrentThread ( ) , THREAD_PRIORITY_NORMAL ) ;
g_bPriorityNormal = true ;
}
}
//---------------------------------------------------------------------------
static UINT g_uModeStepping_Cycles = 0 ;
static bool g_uModeStepping_LastGetKey_ScrollLock = false ;
static void ContinueExecution ( void )
{
# ifdef LOG_PERF_TIMINGS
PerfMarker * pPerfMarkerTotal = new PerfMarker ( g_timeTotal ) ;
# endif
_ASSERT ( g_nAppMode = = MODE_RUNNING | | g_nAppMode = = MODE_STEPPING ) ;
const double fUsecPerSec = 1.e6 ;
# if 1
const UINT nExecutionPeriodUsec = 1000 ; // 1.0ms
// const UINT nExecutionPeriodUsec = 100; // 0.1ms
const double fExecutionPeriodClks = g_fCurrentCLK6502 * ( ( double ) nExecutionPeriodUsec / fUsecPerSec ) ;
# else
const double fExecutionPeriodClks = 1800.0 ;
const UINT nExecutionPeriodUsec = ( UINT ) ( fUsecPerSec * ( fExecutionPeriodClks / g_fCurrentCLK6502 ) ) ;
# endif
//
bool bScrollLock_FullSpeed = false ;
2020-12-20 15:32:51 +00:00
if ( GetPropertySheet ( ) . GetScrollLockToggle ( ) )
2020-11-26 21:50:06 +00:00
{
2021-01-16 21:57:28 +00:00
bScrollLock_FullSpeed = Win32Frame : : GetWin32Frame ( ) . g_bScrollLock_FullSpeed ;
2020-11-26 21:50:06 +00:00
}
else
{
if ( g_nAppMode = = MODE_RUNNING )
{
bScrollLock_FullSpeed = GetKeyState ( VK_SCROLL ) < 0 ;
}
else if ( ! IsDebugSteppingAtFullSpeed ( ) ) // Implicitly: MODE_STEPPING
{
// NB. For MODE_STEPPING: GetKeyState() is slow, so only call periodically
// . 0x3FFF is roughly the number of cycles in a video frame, which seems a reasonable rate to call GetKeyState()
if ( ( g_uModeStepping_Cycles & 0x3FFF ) = = 0 )
g_uModeStepping_LastGetKey_ScrollLock = GetKeyState ( VK_SCROLL ) < 0 ;
bScrollLock_FullSpeed = g_uModeStepping_LastGetKey_ScrollLock ;
}
}
const bool bWasFullSpeed = g_bFullSpeed ;
g_bFullSpeed = ( g_dwSpeed = = SPEED_MAX ) | |
bScrollLock_FullSpeed | |
( GetCardMgr ( ) . GetDisk2CardMgr ( ) . IsConditionForFullSpeed ( ) & & ! Spkr_IsActive ( ) & & ! MB_IsActive ( ) ) | |
IsDebugSteppingAtFullSpeed ( ) ;
if ( g_bFullSpeed )
{
if ( ! bWasFullSpeed )
2021-01-03 16:21:24 +00:00
GetFrame ( ) . VideoRedrawScreenDuringFullSpeed ( 0 , true ) ; // Init for full-speed mode
2020-11-26 21:50:06 +00:00
// Don't call Spkr_Mute() - will get speaker clicks
MB_Mute ( ) ;
SysClk_StopTimer ( ) ;
# ifdef USE_SPEECH_API
g_Speech . Reset ( ) ; // TODO: Put this on a timer (in emulated cycles)... otherwise CATALOG cuts out
# endif
g_nCpuCyclesFeedback = 0 ; // For the case when this is a big -ve number
// Switch to normal priority so that APPLEWIN process doesn't hog machine!
//. EG: No disk in Drive-1, and boot Apple: Windows will start to crawl!
SetPriorityNormal ( ) ;
}
else
{
if ( bWasFullSpeed )
2021-01-03 16:21:24 +00:00
GetFrame ( ) . VideoRedrawScreenAfterFullSpeed ( g_dwCyclesThisFrame ) ;
2020-11-26 21:50:06 +00:00
// Don't call Spkr_Demute()
MB_Demute ( ) ;
SysClk_StartTimerUsec ( nExecutionPeriodUsec ) ;
// Switch to higher priority, eg. for audio (BUG #015394)
SetPriorityAboveNormal ( ) ;
}
//
int nCyclesWithFeedback = ( int ) fExecutionPeriodClks + g_nCpuCyclesFeedback ;
const UINT uCyclesToExecuteWithFeedback = ( nCyclesWithFeedback > = 0 ) ? nCyclesWithFeedback
: 0 ;
const DWORD uCyclesToExecute = ( g_nAppMode = = MODE_RUNNING ) ? uCyclesToExecuteWithFeedback
/* MODE_STEPPING */ : 0 ;
const bool bVideoUpdate = ! g_bFullSpeed ;
const DWORD uActualCyclesExecuted = CpuExecute ( uCyclesToExecute , bVideoUpdate ) ;
g_dwCyclesThisFrame + = uActualCyclesExecuted ;
GetCardMgr ( ) . GetDisk2CardMgr ( ) . UpdateDriveState ( uActualCyclesExecuted ) ;
JoyUpdateButtonLatch ( nExecutionPeriodUsec ) ; // Button latch time is independent of CPU clock frequency
PrintUpdate ( uActualCyclesExecuted ) ;
MB_PeriodicUpdate ( uActualCyclesExecuted ) ;
//
DWORD uSpkrActualCyclesExecuted = uActualCyclesExecuted ;
bool bModeStepping_WaitTimer = false ;
if ( g_nAppMode = = MODE_STEPPING & & ! IsDebugSteppingAtFullSpeed ( ) )
{
g_uModeStepping_Cycles + = uActualCyclesExecuted ;
if ( g_uModeStepping_Cycles > = uCyclesToExecuteWithFeedback )
{
uSpkrActualCyclesExecuted = g_uModeStepping_Cycles ;
g_uModeStepping_Cycles - = uCyclesToExecuteWithFeedback ;
bModeStepping_WaitTimer = true ;
}
}
// For MODE_STEPPING: do this speaker update periodically
// - Otherwise kills performance due to sound-buffer lock/unlock for every 6502 opcode!
if ( g_nAppMode = = MODE_RUNNING | | bModeStepping_WaitTimer )
SpkrUpdate ( uSpkrActualCyclesExecuted ) ;
//
const UINT dwClksPerFrame = NTSC_GetCyclesPerFrame ( ) ;
2020-12-28 16:25:29 +00:00
if ( g_dwCyclesThisFrame > = dwClksPerFrame & & ! GetVideo ( ) . VideoGetVblBarEx ( g_dwCyclesThisFrame ) )
2020-11-26 21:50:06 +00:00
{
# ifdef LOG_PERF_TIMINGS
PerfMarker perfMarkerVideoRefresh ( g_timeVideoRefresh ) ;
# endif
g_dwCyclesThisFrame - = dwClksPerFrame ;
if ( g_bFullSpeed )
2021-01-03 16:21:24 +00:00
GetFrame ( ) . VideoRedrawScreenDuringFullSpeed ( g_dwCyclesThisFrame ) ;
2020-11-26 21:50:06 +00:00
else
2021-01-03 16:21:24 +00:00
GetFrame ( ) . VideoPresentScreen ( ) ; // Just copy the output of our Apple framebuffer to the system Back Buffer
2020-11-26 21:50:06 +00:00
}
# ifdef LOG_PERF_TIMINGS
delete pPerfMarkerTotal ; // Explicitly call dtor *before* SysClk_WaitTimer()
# endif
if ( ( g_nAppMode = = MODE_RUNNING & & ! g_bFullSpeed ) | | bModeStepping_WaitTimer )
{
SysClk_WaitTimer ( ) ;
}
}
void SingleStep ( bool bReinit )
{
if ( bReinit )
{
g_uModeStepping_Cycles = 0 ;
g_uModeStepping_LastGetKey_ScrollLock = false ;
}
ContinueExecution ( ) ;
}
//===========================================================================
void EnterMessageLoop ( void )
{
MSG message ;
PeekMessage ( & message , NULL , 0 , 0 , PM_NOREMOVE ) ;
while ( message . message ! = WM_QUIT )
{
if ( PeekMessage ( & message , NULL , 0 , 0 , PM_REMOVE ) )
{
TranslateMessage ( & message ) ;
DispatchMessage ( & message ) ;
while ( ( g_nAppMode = = MODE_RUNNING ) | | ( g_nAppMode = = MODE_STEPPING ) )
{
if ( PeekMessage ( & message , 0 , 0 , 0 , PM_REMOVE ) )
{
if ( message . message = = WM_QUIT )
return ;
TranslateMessage ( & message ) ;
DispatchMessage ( & message ) ;
}
else if ( g_nAppMode = = MODE_STEPPING )
{
DebugContinueStepping ( ) ;
}
else
{
ContinueExecution ( ) ;
if ( g_nAppMode ! = MODE_DEBUG )
{
if ( g_bFullSpeed )
ContinueExecution ( ) ;
}
}
}
}
else
{
if ( g_nAppMode = = MODE_DEBUG )
DebuggerUpdate ( ) ;
else if ( g_nAppMode = = MODE_PAUSED )
Sleep ( 1 ) ; // Stop process hogging CPU - 1ms, as need to fade-out speaker sound buffer
else if ( g_nAppMode = = MODE_LOGO )
Sleep ( 1 ) ; // Stop process hogging CPU (NB. don't delay for too long otherwise key input can be slow in other apps - GH#569)
}
}
}
//===========================================================================
static void GetProgramDirectory ( void )
{
TCHAR programDir [ MAX_PATH ] ;
GetModuleFileName ( ( HINSTANCE ) 0 , programDir , MAX_PATH ) ;
programDir [ MAX_PATH - 1 ] = 0 ;
g_sProgramDir = programDir ;
int loop = g_sProgramDir . size ( ) ;
while ( loop - - )
{
if ( ( g_sProgramDir [ loop ] = = TEXT ( ' \\ ' ) ) | | ( g_sProgramDir [ loop ] = = TEXT ( ' : ' ) ) )
{
g_sProgramDir . resize ( loop + 1 ) ; // this reduces the size
break ;
}
}
}
//===========================================================================
void RegisterExtensions ( void )
{
TCHAR szCommandTmp [ MAX_PATH ] ;
GetModuleFileName ( ( HMODULE ) 0 , szCommandTmp , MAX_PATH ) ;
TCHAR command [ MAX_PATH ] ;
wsprintf ( command , " \" %s \" " , szCommandTmp ) ; // Wrap path & filename in quotes & null terminate
TCHAR icon [ MAX_PATH ] ;
wsprintf ( icon , TEXT ( " %s,1 " ) , ( LPCTSTR ) command ) ;
_tcscat ( command , TEXT ( " \" %1 \" " ) ) ; // Append "%1"
// _tcscat(command,TEXT("-d1 %1\"")); // Append "%1"
// sprintf(command, "\"%s\" \"-d1 %%1\"", szCommandTmp); // Wrap path & filename in quotes & null terminate
// NB. Registry access to HKLM typically results in ErrorCode 5(ACCESS DENIED), as UAC requires elevated permissions (Run as administrator).
// . HKEY_CLASSES_ROOT\CLSID is a merged view of HKLM\SOFTWARE\Classes and HKCU\SOFTWARE\Classes
// NB. Reflect extensions in DELREG.INF
// RegSetValue(HKEY_CLASSES_ROOT,".bin",REG_SZ,"DiskImage",0); // Removed as .bin is too generic
const char * pValueName = " .bin " ;
LSTATUS res = RegDeleteValue ( HKEY_CLASSES_ROOT , pValueName ) ;
if ( res ! = NOERROR & & res ! = ERROR_FILE_NOT_FOUND ) LogFileOutput ( " RegDeleteValue(%s) failed (0x%08X) \n " , pValueName , res ) ;
pValueName = " .do " ;
res = RegSetValue ( HKEY_CLASSES_ROOT , pValueName , REG_SZ , " DiskImage " , 0 ) ;
if ( res ! = NOERROR ) LogFileOutput ( " RegSetValue(%s) failed (0x%08X) \n " , pValueName , res ) ;
pValueName = " .dsk " ;
res = RegSetValue ( HKEY_CLASSES_ROOT , pValueName , REG_SZ , " DiskImage " , 0 ) ;
if ( res ! = NOERROR ) LogFileOutput ( " RegSetValue(%s) failed (0x%08X) \n " , pValueName , res ) ;
pValueName = " .nib " ;
res = RegSetValue ( HKEY_CLASSES_ROOT , pValueName , REG_SZ , " DiskImage " , 0 ) ;
if ( res ! = NOERROR ) LogFileOutput ( " RegSetValue(%s) failed (0x%08X) \n " , pValueName , res ) ;
pValueName = " .po " ;
res = RegSetValue ( HKEY_CLASSES_ROOT , pValueName , REG_SZ , " DiskImage " , 0 ) ;
if ( res ! = NOERROR ) LogFileOutput ( " RegSetValue(%s) failed (0x%08X) \n " , pValueName , res ) ;
pValueName = " .woz " ;
res = RegSetValue ( HKEY_CLASSES_ROOT , pValueName , REG_SZ , " DiskImage " , 0 ) ;
if ( res ! = NOERROR ) LogFileOutput ( " RegSetValue(%s) failed (0x%08X) \n " , pValueName , res ) ;
// RegSetValue(HKEY_CLASSES_ROOT,".2mg",REG_SZ,"DiskImage",0); // Don't grab this, as not all .2mg images are supported (so defer to CiderPress)
// RegSetValue(HKEY_CLASSES_ROOT,".2img",REG_SZ,"DiskImage",0); // Don't grab this, as not all .2mg images are supported (so defer to CiderPress)
// RegSetValue(HKEY_CLASSES_ROOT,".aws.yaml",REG_SZ,"DiskImage",0); // NB. Can't grab this extension (even though it returns 0!) with embedded period (and .yaml is too generic) - GH#548
// RegSetValue(HKEY_CLASSES_ROOT,".hdv",REG_SZ,"DiskImage",0); // TO DO
pValueName = " DiskImage " ;
res = RegSetValue ( HKEY_CLASSES_ROOT ,
pValueName ,
REG_SZ , " Disk Image " , 0 ) ;
if ( res ! = NOERROR ) LogFileOutput ( " RegSetValue(%s) failed (0x%08X) \n " , pValueName , res ) ;
pValueName = " DiskImage \\ DefaultIcon " ;
res = RegSetValue ( HKEY_CLASSES_ROOT ,
pValueName ,
REG_SZ , icon , 0 ) ;
if ( res ! = NOERROR ) LogFileOutput ( " RegSetValue(%s) failed (0x%08X) \n " , pValueName , res ) ;
// This key can interfere....
// HKEY_CURRENT_USER\Software\Microsoft\Windows\CurrentVersion\Explorer\FileExt\.dsk
pValueName = " DiskImage \\ shell \\ open \\ command " ;
res = RegSetValue ( HKEY_CLASSES_ROOT ,
pValueName ,
REG_SZ , command , _tcslen ( command ) + 1 ) ;
if ( res ! = NOERROR ) LogFileOutput ( " RegSetValue(%s) failed (0x%08X) \n " , pValueName , res ) ;
pValueName = " DiskImage \\ shell \\ open \\ ddeexec " ;
res = RegSetValue ( HKEY_CLASSES_ROOT ,
pValueName ,
REG_SZ , " %1 " , 3 ) ;
if ( res ! = NOERROR ) LogFileOutput ( " RegSetValue(%s) failed (0x%08X) \n " , pValueName , res ) ;
pValueName = " DiskImage \\ shell \\ open \\ ddeexec \\ application " ;
res = RegSetValue ( HKEY_CLASSES_ROOT ,
pValueName ,
REG_SZ , " applewin " , _tcslen ( " applewin " ) + 1 ) ;
// REG_SZ,szCommandTmp,_tcslen(szCommandTmp)+1);
if ( res ! = NOERROR ) LogFileOutput ( " RegSetValue(%s) failed (0x%08X) \n " , pValueName , res ) ;
pValueName = " DiskImage \\ shell \\ open \\ ddeexec \\ topic " ;
res = RegSetValue ( HKEY_CLASSES_ROOT ,
pValueName ,
REG_SZ , " system " , _tcslen ( " system " ) + 1 ) ;
if ( res ! = NOERROR ) LogFileOutput ( " RegSetValue(%s) failed (0x%08X) \n " , pValueName , res ) ;
}
//===========================================================================
// NB. On a restart, it's OK to call RegisterHotKey() again since the old g_hFrameWindow has been destroyed
static void RegisterHotKeys ( void )
{
BOOL bStatus [ 3 ] = { 0 , 0 , 0 } ;
bStatus [ 0 ] = RegisterHotKey (
2020-12-24 15:08:50 +00:00
GetFrame ( ) . g_hFrameWindow , // HWND hWnd
2020-11-26 21:50:06 +00:00
VK_SNAPSHOT_560 , // int id (user/custom id)
0 , // UINT fsModifiers
VK_SNAPSHOT // UINT vk = PrintScreen
) ;
bStatus [ 1 ] = RegisterHotKey (
2020-12-24 15:08:50 +00:00
GetFrame ( ) . g_hFrameWindow , // HWND hWnd
2020-11-26 21:50:06 +00:00
VK_SNAPSHOT_280 , // int id (user/custom id)
MOD_SHIFT , // UINT fsModifiers
VK_SNAPSHOT // UINT vk = PrintScreen
) ;
bStatus [ 2 ] = RegisterHotKey (
2020-12-24 15:08:50 +00:00
GetFrame ( ) . g_hFrameWindow , // HWND hWnd
2020-11-26 21:50:06 +00:00
VK_SNAPSHOT_TEXT , // int id (user/custom id)
MOD_CONTROL , // UINT fsModifiers
VK_SNAPSHOT // UINT vk = PrintScreen
) ;
if ( ( ! bStatus [ 0 ] | | ! bStatus [ 1 ] | | ! bStatus [ 2 ] ) )
{
std : : string msg ( " Unable to register for PrintScreen key(s): \n " ) ;
if ( ! bStatus [ 0 ] )
msg + = " \n . PrintScreen " ;
if ( ! bStatus [ 1 ] )
msg + = " \n . Shift+PrintScreen " ;
if ( ! bStatus [ 2 ] )
msg + = " \n . Ctrl+PrintScreen " ;
2021-01-03 16:21:24 +00:00
if ( GetFrame ( ) . GetShowPrintScreenWarningDialog ( ) )
2020-12-27 13:21:46 -06:00
SHMessageBoxCheck ( GetFrame ( ) . g_hFrameWindow , msg . c_str ( ) , " Warning " , MB_ICONASTERISK | MB_OK , MB_OK , " AppleWin-75097740-8e59-444c-bc94-2d4915132599 " ) ;
2020-11-26 21:50:06 +00:00
msg + = " \n " ;
LogFileOutput ( msg . c_str ( ) ) ;
}
}
//---------------------------------------------------------------------------
static HINSTANCE g_hinstDLL = 0 ;
static HHOOK g_hhook = 0 ;
static HANDLE g_hHookThread = NULL ;
static DWORD g_HookThreadId = 0 ;
// Pre: g_hFrameWindow must be valid
static bool HookFilterForKeyboard ( )
{
g_hinstDLL = LoadLibrary ( TEXT ( " HookFilter.dll " ) ) ;
2020-12-24 15:08:50 +00:00
_ASSERT ( GetFrame ( ) . g_hFrameWindow ) ;
2020-11-26 21:50:06 +00:00
typedef void ( * RegisterHWNDProc ) ( HWND , bool , bool ) ;
RegisterHWNDProc RegisterHWND = ( RegisterHWNDProc ) GetProcAddress ( g_hinstDLL , " RegisterHWND " ) ;
if ( RegisterHWND )
2020-12-24 15:08:50 +00:00
RegisterHWND ( GetFrame ( ) . g_hFrameWindow , g_bHookAltTab , g_bHookAltGrControl ) ;
2020-11-26 21:50:06 +00:00
HOOKPROC hkprcLowLevelKeyboardProc = ( HOOKPROC ) GetProcAddress ( g_hinstDLL , " LowLevelKeyboardProc " ) ;
g_hhook = SetWindowsHookEx (
WH_KEYBOARD_LL ,
hkprcLowLevelKeyboardProc ,
g_hinstDLL ,
0 ) ;
2020-12-24 15:08:50 +00:00
if ( g_hhook ! = 0 & & GetFrame ( ) . g_hFrameWindow ! = 0 )
2020-11-26 21:50:06 +00:00
return true ;
std : : string msg ( " Failed to install hook filter for system keys " ) ;
DWORD dwErr = GetLastError ( ) ;
2021-01-19 20:37:43 +00:00
GetFrame ( ) . FrameMessageBox ( msg . c_str ( ) , " Warning " , MB_ICONASTERISK | MB_OK ) ;
2020-11-26 21:50:06 +00:00
msg + = " \n " ;
LogFileOutput ( msg . c_str ( ) ) ;
return false ;
}
static void UnhookFilterForKeyboard ( )
{
UnhookWindowsHookEx ( g_hhook ) ;
FreeLibrary ( g_hinstDLL ) ;
}
static DWORD WINAPI HookThread ( LPVOID lpParameter )
{
if ( ! HookFilterForKeyboard ( ) )
return - 1 ;
MSG msg ;
while ( GetMessage ( & msg , NULL , 0 , 0 ) > 0 )
{
if ( msg . message = = WM_QUIT )
break ;
TranslateMessage ( & msg ) ;
DispatchMessage ( & msg ) ;
}
UnhookFilterForKeyboard ( ) ;
return 0 ;
}
static bool InitHookThread ( )
{
g_hHookThread = CreateThread ( NULL , // lpThreadAttributes
0 , // dwStackSize
( LPTHREAD_START_ROUTINE ) HookThread ,
0 , // lpParameter
0 , // dwCreationFlags : 0 = Run immediately
& g_HookThreadId ) ; // lpThreadId
if ( g_hHookThread = = NULL )
return false ;
return true ;
}
static void UninitHookThread ( )
{
if ( g_hHookThread )
{
if ( ! PostThreadMessage ( g_HookThreadId , WM_QUIT , 0 , 0 ) )
{
_ASSERT ( 0 ) ;
return ;
}
do
{
DWORD dwExitCode ;
if ( GetExitCodeThread ( g_hHookThread , & dwExitCode ) )
{
if ( dwExitCode = = STILL_ACTIVE )
Sleep ( 10 ) ;
else
break ;
}
}
while ( 1 ) ;
CloseHandle ( g_hHookThread ) ;
g_hHookThread = NULL ;
g_HookThreadId = 0 ;
}
}
static void ExceptionHandler ( const char * pError )
{
2021-01-19 20:37:43 +00:00
GetFrame ( ) . FrameMessageBox (
2020-11-26 21:50:06 +00:00
pError ,
TEXT ( " Runtime Exception " ) ,
MB_ICONEXCLAMATION | MB_SETFOREGROUND ) ;
LogFileOutput ( " Runtime Exception: %s \n " , pError ) ;
}
//---------------------------------------------------------------------------
static void GetAppleWinVersion ( void ) ;
static void OneTimeInitialization ( HINSTANCE passinstance ) ;
static void RepeatInitialization ( void ) ;
static void Shutdown ( void ) ;
int APIENTRY WinMain ( HINSTANCE passinstance , HINSTANCE , LPSTR lpCmdLine , int )
{
char startDir [ _MAX_PATH ] ;
GetCurrentDirectory ( sizeof ( startDir ) , startDir ) ;
g_sStartDir = startDir ;
if ( * ( g_sStartDir . end ( ) - 1 ) ! = ' \\ ' ) g_sStartDir + = ' \\ ' ;
if ( ! ProcessCmdLine ( lpCmdLine ) )
return 0 ;
LogFileOutput ( " g_sStartDir = %s \n " , g_sStartDir . c_str ( ) ) ;
GetAppleWinVersion ( ) ;
OneTimeInitialization ( passinstance ) ;
try
{
do
{
g_bRestart = false ;
RepeatInitialization ( ) ;
// ENTER THE MAIN MESSAGE LOOP
LogFileOutput ( " Main: EnterMessageLoop() \n " ) ;
EnterMessageLoop ( ) ;
LogFileOutput ( " Main: LeaveMessageLoop() \n " ) ;
if ( g_bRestart )
{
g_cmdLine . bSetFullScreen = g_bRestartFullScreen ;
g_bRestartFullScreen = false ;
2021-02-15 21:47:21 +00:00
MB_Reset ( true ) ;
LogFileOutput ( " Main: MB_Reset() \n " ) ;
2020-11-26 21:50:06 +00:00
2020-12-12 17:46:36 +00:00
CMouseInterface * pMouseCard = GetCardMgr ( ) . GetMouseCard ( ) ;
if ( pMouseCard )
{
// dtor removes event from g_SynchronousEventMgr - do before g_SynchronousEventMgr.Reset()
GetCardMgr ( ) . Remove ( pMouseCard - > GetSlot ( ) ) ;
LogFileOutput ( " Main: CMouseInterface::dtor \n " ) ;
}
_ASSERT ( g_SynchronousEventMgr . GetHead ( ) = = NULL ) ;
g_SynchronousEventMgr . Reset ( ) ;
2020-11-26 21:50:06 +00:00
}
DSUninit ( ) ;
LogFileOutput ( " Main: DSUninit() \n " ) ;
if ( g_bHookSystemKey )
{
UninitHookThread ( ) ;
LogFileOutput ( " Main: UnhookFilterForKeyboard() \n " ) ;
}
}
while ( g_bRestart ) ;
}
catch ( std : : runtime_error exception )
{
ExceptionHandler ( exception . what ( ) ) ;
}
catch ( std : : exception exception )
{
ExceptionHandler ( exception . what ( ) ) ;
}
Shutdown ( ) ;
return 0 ;
}
static void GetAppleWinVersion ( void )
{
char szPath [ _MAX_PATH ] ;
if ( 0 = = GetModuleFileName ( NULL , szPath , sizeof ( szPath ) ) )
strcpy_s ( szPath , sizeof ( szPath ) , __argv [ 0 ] ) ;
// Extract application version and store in a global variable
DWORD dwHandle , dwVerInfoSize ;
dwVerInfoSize = GetFileVersionInfoSize ( szPath , & dwHandle ) ;
if ( dwVerInfoSize > 0 )
{
char * pVerInfoBlock = new char [ dwVerInfoSize ] ;
if ( GetFileVersionInfo ( szPath , NULL , dwVerInfoSize , pVerInfoBlock ) )
{
VS_FIXEDFILEINFO * pFixedFileInfo ;
UINT pFixedFileInfoLen ;
VerQueryValue ( pVerInfoBlock , TEXT ( " \\ " ) , ( LPVOID * ) & pFixedFileInfo , ( PUINT ) & pFixedFileInfoLen ) ;
// Construct version string from fixed file info block
UINT16 major = pFixedFileInfo - > dwFileVersionMS > > 16 ;
UINT16 minor = pFixedFileInfo - > dwFileVersionMS & 0xffff ;
UINT16 fix = pFixedFileInfo - > dwFileVersionLS > > 16 ;
UINT16 fix_minor = pFixedFileInfo - > dwFileVersionLS & 0xffff ;
SetAppleWinVersion ( major , minor , fix , fix_minor ) ;
}
delete [ ] pVerInfoBlock ;
}
LogFileOutput ( " AppleWin version: %s \n " , VERSIONSTRING ) ;
}
// DO ONE-TIME INITIALIZATION
static void OneTimeInitialization ( HINSTANCE passinstance )
{
#if 0
# ifdef RIFF_SPKR
RiffInitWriteFile ( " Spkr.wav " , SPKR_SAMPLE_RATE , 1 ) ;
# endif
# ifdef RIFF_MB
RiffInitWriteFile ( " Mockingboard.wav " , 44100 , 2 ) ;
# endif
# endif
// Initialize COM - so we can use CoCreateInstance
2020-12-28 16:25:29 +00:00
// . DSInit() & DIMouse::DirectInputInit are done when g_hFrameWindow is created (WM_CREATE)
// . DDInit() is done in RepeatInitialization() by GetVideo().Initialize()
2020-11-26 21:50:06 +00:00
HRESULT hr = CoInitializeEx ( NULL , COINIT_APARTMENTTHREADED ) ;
LogFileOutput ( " Init: CoInitializeEx(), hr=0x%08X \n " , hr ) ;
g_bSysClkOK = SysClk_InitTimer ( ) ;
LogFileOutput ( " Init: SysClk_InitTimer(), res=%d \n " , g_bSysClkOK ? 1 : 0 ) ;
# ifdef USE_SPEECH_API
if ( g_bEnableSpeech )
{
const bool bSpeechOK = g_Speech . Init ( ) ;
LogFileOutput ( " Init: SysClk_InitTimer(), res=%d \n " , bSpeechOK ? 1 : 0 ) ;
}
# endif
2020-12-24 15:08:50 +00:00
GetFrame ( ) . g_hInstance = passinstance ;
2020-11-26 21:50:06 +00:00
GdiSetBatchLimit ( 512 ) ;
LogFileOutput ( " Init: GdiSetBatchLimit() \n " ) ;
GetProgramDirectory ( ) ;
LogFileOutput ( " Init: GetProgramDirectory() \n " ) ;
if ( g_bRegisterFileTypes )
{
RegisterExtensions ( ) ;
LogFileOutput ( " Init: RegisterExtensions() \n " ) ;
}
2021-01-10 16:33:06 +00:00
Win32Frame : : GetWin32Frame ( ) . FrameRegisterClass ( ) ;
2020-11-26 21:50:06 +00:00
LogFileOutput ( " Init: FrameRegisterClass() \n " ) ;
}
// DO INITIALIZATION THAT MUST BE REPEATED FOR A RESTART
static void RepeatInitialization ( void )
{
ResetToLogoMode ( ) ;
// NB. g_OldAppleWinVersion needed by LoadConfiguration() -> Config_Load_Video()
const bool bShowAboutDlg = CheckOldAppleWinVersion ( ) ; // Post: g_OldAppleWinVersion
LoadConfiguration ( ) ;
LogFileOutput ( " Main: LoadConfiguration() \n " ) ;
if ( g_cmdLine . model ! = A2TYPE_MAX )
SetApple2Type ( g_cmdLine . model ) ;
RGB_SetVideocard ( g_cmdLine . rgbCard , g_cmdLine . rgbCardForegroundColor , g_cmdLine . rgbCardBackgroundColor ) ;
if ( g_cmdLine . newVideoType > = 0 )
{
2020-12-28 16:25:29 +00:00
GetVideo ( ) . SetVideoType ( ( VideoType_e ) g_cmdLine . newVideoType ) ;
2020-11-26 21:50:06 +00:00
g_cmdLine . newVideoType = - 1 ; // Don't reapply after a restart
}
2020-12-28 16:25:29 +00:00
GetVideo ( ) . SetVideoStyle ( ( VideoStyle_e ) ( ( GetVideo ( ) . GetVideoStyle ( ) | g_cmdLine . newVideoStyleEnableMask ) & ~ g_cmdLine . newVideoStyleDisableMask ) ) ;
2020-11-26 21:50:06 +00:00
if ( g_cmdLine . newVideoRefreshRate ! = VR_NONE )
{
2020-12-28 16:25:29 +00:00
GetVideo ( ) . SetVideoRefreshRate ( g_cmdLine . newVideoRefreshRate ) ;
2020-11-26 21:50:06 +00:00
g_cmdLine . newVideoRefreshRate = VR_NONE ; // Don't reapply after a restart
SetCurrentCLK6502 ( ) ;
}
UseClockMultiplier ( g_cmdLine . clockMultiplier ) ;
g_cmdLine . clockMultiplier = 0.0 ;
// Apply the memory expansion switches after loading the Apple II machine type
# ifdef RAMWORKS
if ( g_cmdLine . uRamWorksExPages )
{
SetRamWorksMemorySize ( g_cmdLine . uRamWorksExPages ) ;
SetExpansionMemType ( CT_RamWorksIII ) ;
g_cmdLine . uRamWorksExPages = 0 ; // Don't reapply after a restart
}
# endif
if ( g_cmdLine . uSaturnBanks )
{
SetSaturnMemorySize ( g_cmdLine . uSaturnBanks ) ; // Set number of banks before constructing Saturn card
SetExpansionMemType ( CT_Saturn128K ) ;
g_cmdLine . uSaturnBanks = 0 ; // Don't reapply after a restart
}
if ( g_cmdLine . bSlot0LanguageCard )
{
SetExpansionMemType ( CT_LanguageCard ) ;
g_cmdLine . bSlot0LanguageCard = false ; // Don't reapply after a restart
}
if ( g_cmdLine . bSwapButtons0and1 )
{
2020-12-20 15:32:51 +00:00
GetPropertySheet ( ) . SetButtonsSwapState ( true ) ;
2020-11-26 21:50:06 +00:00
// Reapply after a restart - TODO: grey-out the Config UI for "Swap 0/1" when this cmd line is passed in
}
DebugInitialize ( ) ;
LogFileOutput ( " Main: DebugInitialize() \n " ) ;
JoyInitialize ( ) ;
LogFileOutput ( " Main: JoyInitialize() \n " ) ;
2021-01-03 16:21:24 +00:00
GetFrame ( ) . Initialize ( ) ; // g_pFramebufferinfo been created now & COM init'ed
2020-11-26 21:50:06 +00:00
LogFileOutput ( " Main: VideoInitialize() \n " ) ;
LogFileOutput ( " Main: FrameCreateWindow() - pre \n " ) ;
2021-01-10 16:33:06 +00:00
Win32Frame : : GetWin32Frame ( ) . FrameCreateWindow ( ) ; // GetFrame().g_hFrameWindow is now valid
2020-11-26 21:50:06 +00:00
LogFileOutput ( " Main: FrameCreateWindow() - post \n " ) ;
// Init palette color
2020-12-28 16:25:29 +00:00
VideoSwitchVideocardPalette ( RGB_GetVideocard ( ) , GetVideo ( ) . GetVideoType ( ) ) ;
2020-11-26 21:50:06 +00:00
// Allow the 4 hardcoded slots to be configurated as empty
// NB. this state is not persisted to the Registry/conf.ini (just as '-s7 empty' isn't)
// TODO: support bSlotEmpty[] for slots: 0,4,5
if ( g_cmdLine . bSlotEmpty [ SLOT1 ] )
GetCardMgr ( ) . Remove ( SLOT1 ) ;
if ( g_cmdLine . bSlotEmpty [ SLOT2 ] )
GetCardMgr ( ) . Remove ( SLOT2 ) ;
if ( g_cmdLine . bSlotEmpty [ SLOT3 ] )
GetCardMgr ( ) . Remove ( SLOT3 ) ;
if ( g_cmdLine . bSlotEmpty [ SLOT6 ] )
GetCardMgr ( ) . Remove ( SLOT6 ) ;
if ( g_cmdLine . slotInsert [ SLOT5 ] ! = CT_Empty )
{
if ( GetCardMgr ( ) . QuerySlot ( SLOT4 ) = = CT_MockingboardC & & g_cmdLine . slotInsert [ SLOT5 ] ! = CT_MockingboardC ) // Currently MB occupies slot4+5 when enabled
{
GetCardMgr ( ) . Remove ( SLOT4 ) ;
GetCardMgr ( ) . Remove ( SLOT5 ) ;
}
GetCardMgr ( ) . Insert ( SLOT5 , g_cmdLine . slotInsert [ SLOT5 ] ) ;
}
// Pre: may need g_hFrameWindow for MessageBox errors
// Post: may enable HDD, required for MemInitialize()->MemInitializeIO()
{
bool temp = false ;
2020-12-12 20:49:46 +00:00
InsertFloppyDisks ( SLOT5 , g_cmdLine . szImageName_drive [ SLOT5 ] , g_cmdLine . driveConnected [ SLOT5 ] , temp ) ;
2020-11-26 21:50:06 +00:00
//g_cmdLine.szImageName_drive[SLOT5][DRIVE_1] = g_cmdLine.szImageName_drive[SLOT5][DRIVE_2] = NULL; // *Do* insert on a restart (since no way they could have changed)
2020-12-12 20:49:46 +00:00
InsertFloppyDisks ( SLOT6 , g_cmdLine . szImageName_drive [ SLOT6 ] , g_cmdLine . driveConnected [ SLOT6 ] , g_cmdLine . bBoot ) ;
2020-11-26 21:50:06 +00:00
g_cmdLine . szImageName_drive [ SLOT6 ] [ DRIVE_1 ] = g_cmdLine . szImageName_drive [ SLOT6 ] [ DRIVE_2 ] = NULL ; // Don't insert on a restart
InsertHardDisks ( g_cmdLine . szImageName_harddisk , g_cmdLine . bBoot ) ;
g_cmdLine . szImageName_harddisk [ HARDDISK_1 ] = g_cmdLine . szImageName_harddisk [ HARDDISK_2 ] = NULL ; // Don't insert on a restart
if ( g_cmdLine . bSlotEmpty [ 7 ] )
{
HD_SetEnabled ( false ) ; // Disable HDD controller, but don't persist this to Registry/conf.ini (consistent with other '-sn empty' cmds)
Snapshot_UpdatePath ( ) ; // If save-state's filename is a harddisk, and the floppy is in the same path, then the filename won't be updated
}
}
// Set *after* InsertFloppyDisks() & InsertHardDisks(), which both update g_sCurrentDir
if ( ! g_cmdLine . strCurrentDir . empty ( ) )
SetCurrentImageDir ( g_cmdLine . strCurrentDir ) ;
if ( g_cmdLine . bRemoveNoSlotClock )
MemRemoveNoSlotClock ( ) ;
MemInitialize ( ) ;
LogFileOutput ( " Main: MemInitialize() \n " ) ;
// Show About dialog after creating main window (need g_hFrameWindow)
if ( bShowAboutDlg )
{
if ( ! AboutDlg ( ) )
g_cmdLine . bShutdown = true ; // Close everything down
else
RegSaveString ( TEXT ( REG_CONFIG ) , TEXT ( REGVALUE_VERSION ) , 1 , VERSIONSTRING ) ; // Only save version after user accepts license
}
if ( g_bCapturePrintScreenKey )
{
RegisterHotKeys ( ) ; // needs valid g_hFrameWindow
LogFileOutput ( " Main: RegisterHotKeys() \n " ) ;
}
if ( g_bHookSystemKey )
{
if ( InitHookThread ( ) ) // needs valid g_hFrameWindow (for message pump)
LogFileOutput ( " Main: HookFilterForKeyboard() \n " ) ;
}
// Need to test if it's safe to call ResetMachineState(). In the meantime, just call Disk2Card's Reset():
GetCardMgr ( ) . GetDisk2CardMgr ( ) . Reset ( true ) ; // Switch from a booting A][+ to a non-autostart A][, so need to turn off floppy motor
LogFileOutput ( " Main: DiskReset() \n " ) ;
HD_Reset ( ) ; // GH#515
LogFileOutput ( " Main: HDDReset() \n " ) ;
if ( ! g_bSysClkOK )
{
2021-01-19 20:37:43 +00:00
GetFrame ( ) . FrameMessageBox ( " DirectX failed to create SystemClock instance " , TEXT ( " AppleWin Error " ) , MB_OK ) ;
2020-11-26 21:50:06 +00:00
g_cmdLine . bShutdown = true ;
}
if ( g_bCustomRomF8Failed | | g_bCustomRomFailed | | ( g_hCustomRomF8 ! = INVALID_HANDLE_VALUE & & g_hCustomRom ! = INVALID_HANDLE_VALUE ) )
{
std : : string msg = g_bCustomRomF8Failed ? " Failed to load custom F8 rom (not found or not exactly 2KiB) \n "
: g_bCustomRomFailed ? " Failed to load custom rom (not found or not exactly 12KiB or 16KiB) \n "
: " Unsupported -rom and -f8rom being used at the same time \n " ;
LogFileOutput ( " %s " , msg . c_str ( ) ) ;
2021-01-19 20:37:43 +00:00
GetFrame ( ) . FrameMessageBox ( msg . c_str ( ) , TEXT ( " AppleWin Error " ) , MB_OK ) ;
2020-11-26 21:50:06 +00:00
g_cmdLine . bShutdown = true ;
}
tfe_init ( ) ;
LogFileOutput ( " Main: tfe_init() \n " ) ;
if ( g_cmdLine . szSnapshotName )
{
std : : string strPathname ( g_cmdLine . szSnapshotName ) ;
int nIdx = strPathname . find_last_of ( ' \\ ' ) ;
if ( nIdx > = 0 & & nIdx + 1 < ( int ) strPathname . length ( ) ) // path exists?
{
const std : : string strPath = strPathname . substr ( 0 , nIdx + 1 ) ;
SetCurrentImageDir ( strPath ) ;
}
// Override value just loaded from Registry by LoadConfiguration()
// . NB. Registry value is not updated with this cmd-line value
Snapshot_SetFilename ( g_cmdLine . szSnapshotName ) ;
Snapshot_LoadState ( ) ;
g_cmdLine . bBoot = true ;
g_cmdLine . szSnapshotName = NULL ;
}
else
{
Snapshot_Startup ( ) ; // Do this after everything has been init'ed
LogFileOutput ( " Main: Snapshot_Startup() \n " ) ;
}
if ( g_cmdLine . szScreenshotFilename )
{
2021-01-03 16:21:24 +00:00
GetFrame ( ) . Video_RedrawAndTakeScreenShot ( g_cmdLine . szScreenshotFilename ) ;
2020-11-26 21:50:06 +00:00
g_cmdLine . bShutdown = true ;
}
if ( g_cmdLine . bShutdown )
{
2020-12-24 15:08:50 +00:00
PostMessage ( GetFrame ( ) . g_hFrameWindow , WM_DESTROY , 0 , 0 ) ; // Close everything down
2020-11-26 21:50:06 +00:00
// NB. If shutting down, then don't post any other messages (GH#286)
}
else
{
2020-12-12 18:23:23 +00:00
if ( g_cmdLine . bestWidth & & g_cmdLine . bestHeight )
2020-11-26 21:50:06 +00:00
{
2020-12-12 18:23:23 +00:00
DEVMODE devMode ;
memset ( & devMode , 0 , sizeof ( devMode ) ) ;
devMode . dmSize = sizeof ( devMode ) ;
devMode . dmPelsWidth = g_cmdLine . bestWidth ;
devMode . dmPelsHeight = g_cmdLine . bestHeight ;
devMode . dmFields = DM_PELSWIDTH | DM_PELSHEIGHT ;
DWORD dwFlags = 0 ;
LONG res = ChangeDisplaySettings ( & devMode , dwFlags ) ;
if ( res = = 0 )
g_cmdLine . bChangedDisplayResolution = true ;
}
2020-11-26 21:50:06 +00:00
2020-12-12 18:23:23 +00:00
if ( g_cmdLine . bSetFullScreen )
{
2020-12-24 15:08:50 +00:00
PostMessage ( GetFrame ( ) . g_hFrameWindow , WM_USER_FULLSCREEN , 0 , 0 ) ;
2020-11-26 21:50:06 +00:00
g_cmdLine . bSetFullScreen = false ;
}
if ( g_cmdLine . bBoot )
{
2020-12-24 15:08:50 +00:00
PostMessage ( GetFrame ( ) . g_hFrameWindow , WM_USER_BOOT , 0 , 0 ) ;
2020-11-26 21:50:06 +00:00
g_cmdLine . bBoot = false ;
}
}
}
static void Shutdown ( void )
{
if ( g_cmdLine . bChangedDisplayResolution )
ChangeDisplaySettings ( NULL , 0 ) ; // restore default
// Release COM
SysClk_UninitTimer ( ) ;
LogFileOutput ( " Exit: SysClk_UninitTimer() \n " ) ;
CoUninitialize ( ) ;
LogFileOutput ( " Exit: CoUninitialize() \n " ) ;
tfe_shutdown ( ) ;
LogFileOutput ( " Exit: tfe_shutdown() \n " ) ;
LogDone ( ) ;
RiffFinishWriteFile ( ) ;
if ( g_hCustomRomF8 ! = INVALID_HANDLE_VALUE )
CloseHandle ( g_hCustomRomF8 ) ;
if ( g_hCustomRom ! = INVALID_HANDLE_VALUE )
CloseHandle ( g_hCustomRom ) ;
if ( g_cmdLine . bSlot7EmptyOnExit )
UnplugHardDiskControllerCard ( ) ;
}
2020-12-20 15:32:51 +00:00
2020-12-28 16:25:29 +00:00
IPropertySheet & GetPropertySheet ( void )
2020-12-20 15:32:51 +00:00
{
static CPropertySheet sg_PropertySheet ;
return sg_PropertySheet ;
}
2020-12-24 15:08:50 +00:00
2020-12-28 16:25:29 +00:00
FrameBase & GetFrame ( void )
2020-12-24 15:08:50 +00:00
{
static Win32Frame sg_Win32Frame ;
return sg_Win32Frame ;
}
2020-12-28 16:25:29 +00:00
Video & GetVideo ( void )
{
2021-01-03 16:21:24 +00:00
static Video video ;
2020-12-28 16:25:29 +00:00
return video ;
}