From be81458284e5199494e62cda08d0aff6bc9599d5 Mon Sep 17 00:00:00 2001 From: tomcw Date: Sun, 24 Feb 2019 15:37:15 +0000 Subject: [PATCH] Save-state: correctly stop & play Mockingboard sound buffer across save-state loading transition (#609) --- source/Mockingboard.cpp | 51 +++++++++++++++++++++++++---------------- source/Mockingboard.h | 1 - source/SoundCore.cpp | 2 +- 3 files changed, 32 insertions(+), 22 deletions(-) diff --git a/source/Mockingboard.cpp b/source/Mockingboard.cpp index 5f92766a..9bf96736 100644 --- a/source/Mockingboard.cpp +++ b/source/Mockingboard.cpp @@ -1418,10 +1418,15 @@ void MB_Initialize() g_bCritSectionValid = true; } +void MB_SetSoundcardType(SS_CARDTYPE NewSoundcardType); + +// NB. Mockingboard voice is *already* muted because showing 'Select Load State file' dialog +// . and voice will be demuted when dialog is closed void MB_InitializeForLoadingSnapshot() // GH#609 { MB_Reset(); InitSoundcardType(); + MockingboardVoice.lpDSBvoice->Stop(); // Reason: 'MB voice is playing' then loading a save-state where 'no MB present' } //----------------------------------------------------------------------------- @@ -1629,6 +1634,26 @@ static BYTE __stdcall PhasorIO(WORD PC, WORD nAddr, BYTE bWrite, BYTE nValue, UL //----------------------------------------------------------------------------- +SS_CARDTYPE MB_GetSoundcardType() +{ + return g_SoundcardType; +} + +static void MB_SetSoundcardType(const SS_CARDTYPE NewSoundcardType) +{ + if (NewSoundcardType == g_SoundcardType) + return; + + if (NewSoundcardType == CT_Empty) + MB_Mute(); // Call MB_Mute() before setting g_SoundcardType = CT_Empty + + g_SoundcardType = NewSoundcardType; + + g_bPhasorEnable = (g_SoundcardType == CT_Phasor); +} + +//----------------------------------------------------------------------------- + void MB_InitializeIO(LPBYTE pCxRomPeripheral, UINT uSlot4, UINT uSlot5) { // Mockingboard: Slot 4 & 5 @@ -1650,6 +1675,12 @@ void MB_InitializeIO(LPBYTE pCxRomPeripheral, UINT uSlot4, UINT uSlot5) RegisterIoHandler(uSlot5, IO_Null, IO_Null, MB_Read, MB_Write, NULL, NULL); MB_SetSoundcardType(g_Slot4); + + // Sound buffer may have been stopped by MB_InitializeForLoadingSnapshot(). + // NB. DSZeroVoiceBuffer() also zeros the sound buffer, so it's better than directly calling IDirectSoundBuffer::Play(): + // - without zeroing, then the previous sound buffer can be heard for a fraction of a second + // - eg. when doing Mockingboard playback, then loading a save-state which is also doing Mockingboard playback + DSZeroVoiceBuffer(&MockingboardVoice, "MB", g_dwDSBufferSize); } //----------------------------------------------------------------------------- @@ -1789,26 +1820,6 @@ void MB_UpdateCycles(ULONG uExecutedCycles) //----------------------------------------------------------------------------- -SS_CARDTYPE MB_GetSoundcardType() -{ - return g_SoundcardType; -} - -void MB_SetSoundcardType(SS_CARDTYPE NewSoundcardType) -{ - if (g_SoundcardType == NewSoundcardType) - return; - - g_SoundcardType = NewSoundcardType; - - if(g_SoundcardType == CT_Empty) - MB_Mute(); - - g_bPhasorEnable = (g_SoundcardType == CT_Phasor); -} - -//----------------------------------------------------------------------------- - static double MB_GetFramePeriod(void) { // TODO: Ideally remove this (slot-4) Phasor-IFR check: [*1] diff --git a/source/Mockingboard.h b/source/Mockingboard.h index 1486adc1..c5dd6db2 100644 --- a/source/Mockingboard.h +++ b/source/Mockingboard.h @@ -13,7 +13,6 @@ void MB_EndOfVideoFrame(); void MB_CheckIRQ(); void MB_UpdateCycles(ULONG uExecutedCycles); SS_CARDTYPE MB_GetSoundcardType(); -void MB_SetSoundcardType(SS_CARDTYPE NewSoundcardType); bool MB_IsActive(); DWORD MB_GetVolume(); void MB_SetVolume(DWORD dwVolume, DWORD dwVolumeMax); diff --git a/source/SoundCore.cpp b/source/SoundCore.cpp index 9a236214..61fc46c7 100644 --- a/source/SoundCore.cpp +++ b/source/SoundCore.cpp @@ -419,7 +419,7 @@ void SoundCore_SetFade(eFADE FadeType) if(g_nAppMode == MODE_DEBUG) return; - // Fade in/out just for speaker, the others are demuted/muted + // Fade in/out for speaker, the others are demuted/muted here if(FadeType != FADE_NONE) { for(UINT i=0; i