2006-02-25 20:50:29 +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
|
2007-04-01 15:24:52 +00:00
|
|
|
Copyright (C) 2006-2007, Tom Charlesworth, Michael Pohoreski
|
2006-02-25 20:50:29 +00:00
|
|
|
|
|
|
|
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: Core sound related functionality
|
|
|
|
*
|
|
|
|
* Author: Tom Charlesworth
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "StdAfx.h"
|
2010-02-14 21:11:26 +00:00
|
|
|
|
2020-11-11 21:15:27 +00:00
|
|
|
#include "SoundCore.h"
|
2020-11-26 21:50:06 +00:00
|
|
|
#include "Core.h"
|
2020-11-29 17:30:06 +00:00
|
|
|
#include "Interface.h"
|
2018-04-08 16:48:08 +00:00
|
|
|
#include "Log.h"
|
2014-08-13 20:30:35 +00:00
|
|
|
#include "Speaker.h"
|
2006-02-25 20:50:29 +00:00
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
#define MAX_SOUND_DEVICES 10
|
|
|
|
|
2020-12-29 21:17:03 +00:00
|
|
|
static std::string sound_devices[MAX_SOUND_DEVICES];
|
2006-02-25 20:50:29 +00:00
|
|
|
static GUID sound_device_guid[MAX_SOUND_DEVICES];
|
|
|
|
static int num_sound_devices = 0;
|
|
|
|
|
|
|
|
static LPDIRECTSOUND g_lpDS = NULL;
|
|
|
|
|
|
|
|
//-------------------------------------
|
|
|
|
|
|
|
|
// Used for muting & fading:
|
|
|
|
|
2023-01-28 18:15:28 +00:00
|
|
|
static const UINT uMAX_VOICES = NUM_SLOTS * 2 + 1 + 1; // 8x (2x SSI263) + spkr + MockingboardCardManager
|
2006-02-25 20:50:29 +00:00
|
|
|
static UINT g_uNumVoices = 0;
|
|
|
|
static VOICE* g_pVoices[uMAX_VOICES] = {NULL};
|
|
|
|
|
|
|
|
static VOICE* g_pSpeakerVoice = NULL;
|
|
|
|
|
|
|
|
//-------------------------------------
|
|
|
|
|
|
|
|
bool g_bDSAvailable = false;
|
|
|
|
|
2023-01-28 18:15:28 +00:00
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
// NB. Also similar is done by: MockingboardCardManager::Destroy()
|
|
|
|
// - which is called from WM_DESTROY (when both restarting VM & exiting the app)
|
|
|
|
|
|
|
|
VOICE::~VOICE(void)
|
|
|
|
{
|
|
|
|
if (lpDSBvoice)
|
|
|
|
{
|
|
|
|
DSVoiceStop(this);
|
|
|
|
DSReleaseSoundBuffer(this);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-02-25 20:50:29 +00:00
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
static BOOL CALLBACK DSEnumProc(LPGUID lpGUID, LPCTSTR lpszDesc, LPCTSTR lpszDrvName, LPVOID lpContext)
|
|
|
|
{
|
|
|
|
int i = num_sound_devices;
|
|
|
|
if(i == MAX_SOUND_DEVICES)
|
|
|
|
return TRUE;
|
|
|
|
if(lpGUID != NULL)
|
|
|
|
memcpy(&sound_device_guid[i], lpGUID, sizeof (GUID));
|
2020-12-29 21:17:03 +00:00
|
|
|
else
|
|
|
|
memset(&sound_device_guid[i], 0, sizeof(GUID));
|
|
|
|
sound_devices[i] = lpszDesc;
|
2006-02-25 20:50:29 +00:00
|
|
|
|
|
|
|
if(g_fh) fprintf(g_fh, "%d: %s - %s\n",i,lpszDesc,lpszDrvName);
|
|
|
|
|
|
|
|
num_sound_devices++;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
#ifdef _DEBUG
|
2021-02-10 21:05:00 +00:00
|
|
|
static const char *DirectSound_ErrorText (HRESULT error)
|
2006-02-25 20:50:29 +00:00
|
|
|
{
|
|
|
|
switch( error )
|
|
|
|
{
|
|
|
|
case DSERR_ALLOCATED:
|
|
|
|
return "Allocated";
|
|
|
|
case DSERR_CONTROLUNAVAIL:
|
|
|
|
return "Control Unavailable";
|
|
|
|
case DSERR_INVALIDPARAM:
|
|
|
|
return "Invalid Parameter";
|
|
|
|
case DSERR_INVALIDCALL:
|
|
|
|
return "Invalid Call";
|
|
|
|
case DSERR_GENERIC:
|
|
|
|
return "Generic";
|
|
|
|
case DSERR_PRIOLEVELNEEDED:
|
|
|
|
return "Priority Level Needed";
|
|
|
|
case DSERR_OUTOFMEMORY:
|
|
|
|
return "Out of Memory";
|
|
|
|
case DSERR_BADFORMAT:
|
|
|
|
return "Bad Format";
|
|
|
|
case DSERR_UNSUPPORTED:
|
|
|
|
return "Unsupported";
|
|
|
|
case DSERR_NODRIVER:
|
|
|
|
return "No Driver";
|
|
|
|
case DSERR_ALREADYINITIALIZED:
|
|
|
|
return "Already Initialized";
|
|
|
|
case DSERR_NOAGGREGATION:
|
|
|
|
return "No Aggregation";
|
|
|
|
case DSERR_BUFFERLOST:
|
|
|
|
return "Buffer Lost";
|
|
|
|
case DSERR_OTHERAPPHASPRIO:
|
|
|
|
return "Other Application Has Priority";
|
|
|
|
case DSERR_UNINITIALIZED:
|
|
|
|
return "Uninitialized";
|
|
|
|
case DSERR_NOINTERFACE:
|
|
|
|
return "No Interface";
|
|
|
|
default:
|
|
|
|
return "Unknown";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
2021-05-16 19:03:59 +00:00
|
|
|
HRESULT DSGetLock(LPDIRECTSOUNDBUFFER pVoice, DWORD dwOffset, DWORD dwBytes,
|
2006-02-25 20:50:29 +00:00
|
|
|
SHORT** ppDSLockedBuffer0, DWORD* pdwDSLockedBufferSize0,
|
|
|
|
SHORT** ppDSLockedBuffer1, DWORD* pdwDSLockedBufferSize1)
|
|
|
|
{
|
|
|
|
DWORD nStatus;
|
|
|
|
HRESULT hr = pVoice->GetStatus(&nStatus);
|
|
|
|
if(hr != DS_OK)
|
2021-05-16 19:03:59 +00:00
|
|
|
return hr;
|
2006-02-25 20:50:29 +00:00
|
|
|
|
|
|
|
if(nStatus & DSBSTATUS_BUFFERLOST)
|
|
|
|
{
|
|
|
|
do
|
|
|
|
{
|
|
|
|
hr = pVoice->Restore();
|
|
|
|
if(hr == DSERR_BUFFERLOST)
|
|
|
|
Sleep(10);
|
|
|
|
}
|
|
|
|
while(hr != DS_OK);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get write only pointer(s) to sound buffer
|
|
|
|
if(dwBytes == 0)
|
|
|
|
{
|
|
|
|
if(FAILED(hr = pVoice->Lock(0, 0,
|
|
|
|
(void**)ppDSLockedBuffer0, pdwDSLockedBufferSize0,
|
|
|
|
(void**)ppDSLockedBuffer1, pdwDSLockedBufferSize1,
|
|
|
|
DSBLOCK_ENTIREBUFFER)))
|
2021-05-16 19:03:59 +00:00
|
|
|
return hr;
|
2006-02-25 20:50:29 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if(FAILED(hr = pVoice->Lock(dwOffset, dwBytes,
|
|
|
|
(void**)ppDSLockedBuffer0, pdwDSLockedBufferSize0,
|
|
|
|
(void**)ppDSLockedBuffer1, pdwDSLockedBufferSize1,
|
|
|
|
0)))
|
2021-05-16 19:03:59 +00:00
|
|
|
return hr;
|
2006-02-25 20:50:29 +00:00
|
|
|
}
|
|
|
|
|
2021-05-16 19:03:59 +00:00
|
|
|
return hr;
|
2006-02-25 20:50:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
2021-03-23 22:01:41 +00:00
|
|
|
HRESULT DSGetSoundBuffer(VOICE* pVoice, DWORD dwFlags, DWORD dwBufferSize, DWORD nSampleRate, int nChannels, const char* pszDevName)
|
2006-02-25 20:50:29 +00:00
|
|
|
{
|
2021-03-23 22:01:41 +00:00
|
|
|
pVoice->name = pszDevName;
|
|
|
|
|
2006-02-25 20:50:29 +00:00
|
|
|
WAVEFORMATEX wavfmt;
|
|
|
|
DSBUFFERDESC dsbdesc;
|
|
|
|
|
|
|
|
wavfmt.wFormatTag = WAVE_FORMAT_PCM;
|
|
|
|
wavfmt.nChannels = nChannels;
|
|
|
|
wavfmt.nSamplesPerSec = nSampleRate;
|
|
|
|
wavfmt.wBitsPerSample = 16;
|
|
|
|
wavfmt.nBlockAlign = wavfmt.nChannels==1 ? 2 : 4;
|
|
|
|
wavfmt.nAvgBytesPerSec = wavfmt.nBlockAlign * wavfmt.nSamplesPerSec;
|
|
|
|
|
|
|
|
memset (&dsbdesc, 0, sizeof (dsbdesc));
|
|
|
|
dsbdesc.dwSize = sizeof (dsbdesc);
|
|
|
|
dsbdesc.dwBufferBytes = dwBufferSize;
|
|
|
|
dsbdesc.lpwfxFormat = &wavfmt;
|
|
|
|
dsbdesc.dwFlags = dwFlags | DSBCAPS_GETCURRENTPOSITION2 | DSBCAPS_STICKYFOCUS;
|
|
|
|
|
|
|
|
// Are buffers released when g_lpDS OR pVoice->lpDSBvoice is released?
|
|
|
|
// . From DirectX doc:
|
|
|
|
// "Buffer objects are owned by the device object that created them. When the
|
|
|
|
// device object is released, all buffers created by that object are also released..."
|
|
|
|
HRESULT hr = g_lpDS->CreateSoundBuffer(&dsbdesc, &pVoice->lpDSBvoice, NULL);
|
|
|
|
if(FAILED(hr))
|
|
|
|
return hr;
|
|
|
|
|
|
|
|
//
|
|
|
|
|
|
|
|
_ASSERT(g_uNumVoices < uMAX_VOICES);
|
|
|
|
if(g_uNumVoices < uMAX_VOICES)
|
|
|
|
g_pVoices[g_uNumVoices++] = pVoice;
|
|
|
|
|
|
|
|
if(pVoice->bIsSpeaker)
|
|
|
|
g_pSpeakerVoice = pVoice;
|
|
|
|
|
|
|
|
return hr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DSReleaseSoundBuffer(VOICE* pVoice)
|
|
|
|
{
|
|
|
|
if(pVoice->bIsSpeaker)
|
|
|
|
g_pSpeakerVoice = NULL;
|
|
|
|
|
|
|
|
for(UINT i=0; i<g_uNumVoices; i++)
|
|
|
|
{
|
|
|
|
if(g_pVoices[i] == pVoice)
|
|
|
|
{
|
|
|
|
g_pVoices[i] = g_pVoices[g_uNumVoices-1];
|
|
|
|
g_pVoices[g_uNumVoices-1] = NULL;
|
|
|
|
g_uNumVoices--;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
SAFE_RELEASE(pVoice->lpDSBvoice);
|
|
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
2021-03-23 22:01:41 +00:00
|
|
|
bool DSVoiceStop(PVOICE Voice)
|
2006-02-25 20:50:29 +00:00
|
|
|
{
|
2019-09-01 10:28:24 +00:00
|
|
|
#ifdef NO_DIRECT_X
|
|
|
|
return false;
|
|
|
|
#else
|
2019-11-03 14:23:47 +00:00
|
|
|
_ASSERT(Voice->lpDSBvoice);
|
2006-02-25 20:50:29 +00:00
|
|
|
HRESULT hr = Voice->lpDSBvoice->Stop();
|
|
|
|
if(FAILED(hr))
|
|
|
|
{
|
2021-03-23 22:01:41 +00:00
|
|
|
if(g_fh) fprintf(g_fh, "%s: DSStop failed (%08X)\n", Voice->name.c_str(), hr);
|
2006-02-25 20:50:29 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-03-23 22:01:41 +00:00
|
|
|
Voice->bActive = false;
|
|
|
|
return true;
|
|
|
|
#endif // NO_DIRECT_X
|
|
|
|
}
|
|
|
|
|
|
|
|
// Use this to Play()
|
|
|
|
bool DSZeroVoiceBuffer(PVOICE Voice, DWORD dwBufferSize)
|
|
|
|
{
|
|
|
|
#ifdef NO_DIRECT_X
|
|
|
|
return false;
|
|
|
|
#else
|
|
|
|
|
|
|
|
DWORD dwDSLockedBufferSize = 0; // Size of the locked DirectSound buffer
|
|
|
|
SHORT* pDSLockedBuffer;
|
|
|
|
|
|
|
|
if (!DSVoiceStop(Voice))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
HRESULT hr = DSGetLock(Voice->lpDSBvoice, 0, 0, &pDSLockedBuffer, &dwDSLockedBufferSize, NULL, 0);
|
2006-02-25 20:50:29 +00:00
|
|
|
if(FAILED(hr))
|
|
|
|
{
|
2021-03-23 22:01:41 +00:00
|
|
|
if(g_fh) fprintf(g_fh, "%s: DSGetLock failed (%08X)\n", Voice->name.c_str(), hr);
|
2006-02-25 20:50:29 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
_ASSERT(dwDSLockedBufferSize == dwBufferSize);
|
|
|
|
memset(pDSLockedBuffer, 0x00, dwDSLockedBufferSize);
|
|
|
|
|
|
|
|
hr = Voice->lpDSBvoice->Unlock((void*)pDSLockedBuffer, dwDSLockedBufferSize, NULL, 0);
|
|
|
|
if(FAILED(hr))
|
|
|
|
{
|
2021-03-23 22:01:41 +00:00
|
|
|
if(g_fh) fprintf(g_fh, "%s: DSUnlock failed (%08X)\n", Voice->name.c_str(), hr);
|
2006-02-25 20:50:29 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
hr = Voice->lpDSBvoice->Play(0,0,DSBPLAY_LOOPING);
|
|
|
|
if(FAILED(hr))
|
|
|
|
{
|
2021-03-23 22:01:41 +00:00
|
|
|
if(g_fh) fprintf(g_fh, "%s: DSPlay failed (%08X)\n", Voice->name.c_str(), hr);
|
2006-02-25 20:50:29 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-03-23 22:01:41 +00:00
|
|
|
Voice->bActive = true;
|
2006-02-25 20:50:29 +00:00
|
|
|
return true;
|
2019-09-01 10:28:24 +00:00
|
|
|
#endif // NO_DIRECT_X
|
2006-02-25 20:50:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
2021-03-23 22:01:41 +00:00
|
|
|
bool DSZeroVoiceWritableBuffer(PVOICE Voice, DWORD dwBufferSize)
|
2006-02-25 20:50:29 +00:00
|
|
|
{
|
|
|
|
DWORD dwDSLockedBufferSize0=0, dwDSLockedBufferSize1=0;
|
|
|
|
SHORT *pDSLockedBuffer0, *pDSLockedBuffer1;
|
|
|
|
|
|
|
|
|
|
|
|
HRESULT hr = DSGetLock(Voice->lpDSBvoice,
|
|
|
|
0, dwBufferSize,
|
|
|
|
&pDSLockedBuffer0, &dwDSLockedBufferSize0,
|
|
|
|
&pDSLockedBuffer1, &dwDSLockedBufferSize1);
|
|
|
|
if(FAILED(hr))
|
|
|
|
{
|
2021-03-23 22:01:41 +00:00
|
|
|
if(g_fh) fprintf(g_fh, "%s: DSGetLock failed (%08X)\n", Voice->name.c_str(), hr);
|
2006-02-25 20:50:29 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
memset(pDSLockedBuffer0, 0x00, dwDSLockedBufferSize0);
|
|
|
|
if(pDSLockedBuffer1)
|
|
|
|
memset(pDSLockedBuffer1, 0x00, dwDSLockedBufferSize1);
|
|
|
|
|
|
|
|
hr = Voice->lpDSBvoice->Unlock((void*)pDSLockedBuffer0, dwDSLockedBufferSize0,
|
|
|
|
(void*)pDSLockedBuffer1, dwDSLockedBufferSize1);
|
|
|
|
if(FAILED(hr))
|
|
|
|
{
|
2021-03-23 22:01:41 +00:00
|
|
|
if(g_fh) fprintf(g_fh, "%s: DSUnlock failed (%08X)\n", Voice->name.c_str(), hr);
|
2006-02-25 20:50:29 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
static bool g_bTimerActive = false;
|
|
|
|
static eFADE g_FadeType = FADE_NONE;
|
|
|
|
static UINT_PTR g_nTimerID = 0;
|
|
|
|
|
|
|
|
//-------------------------------------
|
|
|
|
|
|
|
|
static VOID CALLBACK SoundCore_TimerFunc(HWND hwnd, UINT uMsg, UINT_PTR idEvent, DWORD dwTime);
|
|
|
|
|
|
|
|
static bool SoundCore_StartTimer()
|
|
|
|
{
|
|
|
|
if(g_bTimerActive)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
g_nTimerID = SetTimer(NULL, 0, 1, SoundCore_TimerFunc); // 1ms interval
|
|
|
|
if(g_nTimerID == 0)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "Error creating timer\n");
|
|
|
|
_ASSERT(0);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_bTimerActive = true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void SoundCore_StopTimer()
|
|
|
|
{
|
|
|
|
if(!g_bTimerActive)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if(KillTimer(NULL, g_nTimerID) == FALSE)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "Error killing timer\n");
|
|
|
|
_ASSERT(0);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_bTimerActive = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SoundCore_GetTimerState()
|
|
|
|
{
|
|
|
|
return g_bTimerActive;
|
|
|
|
}
|
|
|
|
|
|
|
|
//-------------------------------------
|
|
|
|
|
|
|
|
// [OLD: Used to fade volume in/out]
|
|
|
|
// FADE_OUT : Just keep filling speaker soundbuffer with same value
|
|
|
|
// FADE_IN : Switch to FADE_NONE & StopTimer()
|
|
|
|
|
|
|
|
static VOID CALLBACK SoundCore_TimerFunc(HWND hwnd, UINT uMsg, UINT_PTR idEvent, DWORD dwTime)
|
|
|
|
{
|
|
|
|
if((g_pSpeakerVoice == NULL) || (g_pSpeakerVoice->bActive == false))
|
|
|
|
g_FadeType = FADE_NONE;
|
|
|
|
|
|
|
|
// Timer expired
|
|
|
|
if(g_FadeType == FADE_NONE)
|
|
|
|
{
|
|
|
|
SoundCore_StopTimer();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
|
|
|
|
#if 1
|
|
|
|
if(g_FadeType == FADE_IN)
|
|
|
|
g_FadeType = FADE_NONE;
|
|
|
|
else
|
|
|
|
SpkrUpdate_Timer();
|
|
|
|
#else
|
|
|
|
const LONG nFadeUnit_Fast = (DSBVOLUME_MAX - DSBVOLUME_MIN) / 10;
|
|
|
|
const LONG nFadeUnit_Slow = (DSBVOLUME_MAX - DSBVOLUME_MIN) / 1000; // Less noisy for 'silence'
|
|
|
|
|
|
|
|
LONG nFadeUnit = g_pSpeakerVoice->bRecentlyActive ? nFadeUnit_Fast : nFadeUnit_Slow;
|
|
|
|
LONG nFadeVolume = g_pSpeakerVoice->nFadeVolume;
|
|
|
|
|
|
|
|
if(g_FadeType == FADE_IN)
|
|
|
|
{
|
|
|
|
if(nFadeVolume == g_pSpeakerVoice->nVolume)
|
|
|
|
{
|
|
|
|
g_FadeType = FADE_NONE;
|
|
|
|
SoundCore_StopTimer();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
nFadeVolume += nFadeUnit;
|
|
|
|
|
|
|
|
if(nFadeVolume > g_pSpeakerVoice->nVolume)
|
|
|
|
nFadeVolume = g_pSpeakerVoice->nVolume;
|
|
|
|
}
|
|
|
|
else // FADE_OUT
|
|
|
|
{
|
|
|
|
if(nFadeVolume == DSBVOLUME_MIN)
|
|
|
|
{
|
|
|
|
g_FadeType = FADE_NONE;
|
|
|
|
SoundCore_StopTimer();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
nFadeVolume -= nFadeUnit;
|
|
|
|
|
|
|
|
if(nFadeVolume < DSBVOLUME_MIN)
|
|
|
|
nFadeVolume = DSBVOLUME_MIN;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_pSpeakerVoice->nFadeVolume = nFadeVolume;
|
|
|
|
g_pSpeakerVoice->lpDSBvoice->SetVolume(nFadeVolume);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
void SoundCore_SetFade(eFADE FadeType)
|
|
|
|
{
|
2023-01-03 21:46:51 +00:00
|
|
|
static AppMode_e nLastMode = MODE_UNDEFINED;
|
2006-02-25 20:50:29 +00:00
|
|
|
|
2006-06-12 22:06:50 +00:00
|
|
|
if(g_nAppMode == MODE_DEBUG)
|
2006-02-25 20:50:29 +00:00
|
|
|
return;
|
|
|
|
|
2021-03-23 22:01:41 +00:00
|
|
|
// Fade in/out for speaker, the others are unmuted/muted here
|
2006-02-25 20:50:29 +00:00
|
|
|
if(FadeType != FADE_NONE)
|
|
|
|
{
|
|
|
|
for(UINT i=0; i<g_uNumVoices; i++)
|
|
|
|
{
|
2006-06-12 22:06:50 +00:00
|
|
|
// Note: Kludge for fading speaker if curr/last g_nAppMode is/was MODE_LOGO:
|
2006-02-25 20:50:29 +00:00
|
|
|
// . Bug in DirectSound? SpeakerVoice.lpDSBvoice->SetVolume() doesn't work without this!
|
2009-04-20 21:53:20 +00:00
|
|
|
// . See SoundCore_TweakVolumes() - could be this?
|
2006-06-12 22:06:50 +00:00
|
|
|
if((g_pVoices[i]->bIsSpeaker) && (g_nAppMode != MODE_LOGO) && (nLastMode != MODE_LOGO))
|
2006-02-25 20:50:29 +00:00
|
|
|
{
|
|
|
|
g_pVoices[i]->lpDSBvoice->GetVolume(&g_pVoices[i]->nFadeVolume);
|
|
|
|
g_FadeType = FadeType;
|
|
|
|
SoundCore_StartTimer();
|
|
|
|
}
|
|
|
|
else if(FadeType == FADE_OUT)
|
|
|
|
{
|
|
|
|
g_pVoices[i]->lpDSBvoice->SetVolume(DSBVOLUME_MIN);
|
|
|
|
g_pVoices[i]->bMute = true;
|
|
|
|
}
|
|
|
|
else // FADE_IN
|
|
|
|
{
|
|
|
|
g_pVoices[i]->lpDSBvoice->SetVolume(g_pVoices[i]->nVolume);
|
|
|
|
g_pVoices[i]->bMute = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else // FadeType == FADE_NONE
|
|
|
|
{
|
|
|
|
if( (g_FadeType != FADE_NONE) && // Currently fading-in/out
|
|
|
|
(g_pSpeakerVoice && g_pSpeakerVoice->bActive) )
|
|
|
|
{
|
|
|
|
g_FadeType = FADE_NONE; // TimerFunc will call StopTimer()
|
|
|
|
g_pSpeakerVoice->lpDSBvoice->SetVolume(g_pSpeakerVoice->nVolume);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-06-12 22:06:50 +00:00
|
|
|
nLastMode = g_nAppMode;
|
2006-02-25 20:50:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
2009-12-12 11:04:22 +00:00
|
|
|
// If AppleWin started by double-clicking a .dsk, then our window won't have focus when volumes are set (so gets ignored).
|
|
|
|
// Subsequent setting (to the same volume) will get ignored, as DirectSound thinks that volume is already set.
|
2009-04-20 21:53:20 +00:00
|
|
|
|
|
|
|
void SoundCore_TweakVolumes()
|
|
|
|
{
|
|
|
|
for (UINT i=0; i<g_uNumVoices; i++)
|
|
|
|
{
|
|
|
|
g_pVoices[i]->lpDSBvoice->SetVolume(g_pVoices[i]->nVolume-1);
|
|
|
|
g_pVoices[i]->lpDSBvoice->SetVolume(g_pVoices[i]->nVolume);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
2006-02-25 20:50:29 +00:00
|
|
|
static UINT g_uDSInitRefCount = 0;
|
|
|
|
|
|
|
|
bool DSInit()
|
|
|
|
{
|
|
|
|
if(g_bDSAvailable)
|
|
|
|
{
|
|
|
|
g_uDSInitRefCount++;
|
|
|
|
return true; // Already initialised successfully
|
|
|
|
}
|
|
|
|
|
2020-12-29 21:17:03 +00:00
|
|
|
num_sound_devices = 0;
|
2006-02-25 20:50:29 +00:00
|
|
|
HRESULT hr = DirectSoundEnumerate((LPDSENUMCALLBACK)DSEnumProc, NULL);
|
|
|
|
if(FAILED(hr))
|
|
|
|
{
|
|
|
|
if(g_fh) fprintf(g_fh, "DSEnumerate failed (%08X)\n",hr);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(g_fh)
|
|
|
|
{
|
|
|
|
fprintf(g_fh, "Number of sound devices = %d\n",num_sound_devices);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool bCreatedOK = false;
|
|
|
|
for(int x=0; x<num_sound_devices; x++)
|
|
|
|
{
|
|
|
|
hr = DirectSoundCreate(&sound_device_guid[x], &g_lpDS, NULL);
|
|
|
|
if(SUCCEEDED(hr))
|
|
|
|
{
|
|
|
|
if(g_fh) fprintf(g_fh, "DSCreate succeeded for sound device #%d\n",x);
|
|
|
|
bCreatedOK = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(g_fh) fprintf(g_fh, "DSCreate failed for sound device #%d (%08X)\n",x,hr);
|
|
|
|
}
|
|
|
|
if(!bCreatedOK)
|
|
|
|
{
|
|
|
|
if(g_fh) fprintf(g_fh, "DSCreate failed for all sound devices\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-01-28 18:15:28 +00:00
|
|
|
HWND hwnd = GetFrame().g_hFrameWindow;
|
|
|
|
_ASSERT(hwnd);
|
|
|
|
hr = g_lpDS->SetCooperativeLevel(hwnd, DSSCL_NORMAL);
|
|
|
|
if (FAILED(hr))
|
2006-02-25 20:50:29 +00:00
|
|
|
{
|
|
|
|
if(g_fh) fprintf(g_fh, "SetCooperativeLevel failed (%08X)\n",hr);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
DSCAPS DSCaps;
|
2020-12-10 21:08:15 +00:00
|
|
|
memset(&DSCaps, 0, sizeof(DSCAPS));
|
2009-05-04 22:07:17 +00:00
|
|
|
DSCaps.dwSize = sizeof(DSCAPS);
|
2006-02-25 20:50:29 +00:00
|
|
|
hr = g_lpDS->GetCaps(&DSCaps);
|
|
|
|
if(FAILED(hr))
|
|
|
|
{
|
2009-05-04 22:07:17 +00:00
|
|
|
if(g_fh) fprintf(g_fh, "GetCaps failed (%08X)\n",hr);
|
|
|
|
// Not fatal: so continue...
|
2006-02-25 20:50:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
g_bDSAvailable = true;
|
|
|
|
|
|
|
|
g_uDSInitRefCount = 1;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
void DSUninit()
|
|
|
|
{
|
|
|
|
if(!g_bDSAvailable)
|
|
|
|
return;
|
|
|
|
|
|
|
|
_ASSERT(g_uDSInitRefCount);
|
|
|
|
|
|
|
|
if(g_uDSInitRefCount == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
g_uDSInitRefCount--;
|
|
|
|
|
|
|
|
if(g_uDSInitRefCount)
|
|
|
|
return;
|
|
|
|
|
|
|
|
//
|
|
|
|
|
|
|
|
_ASSERT(g_uNumVoices == 0);
|
|
|
|
|
|
|
|
SAFE_RELEASE(g_lpDS);
|
|
|
|
g_bDSAvailable = false;
|
|
|
|
|
|
|
|
SoundCore_StopTimer();
|
|
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
LONG NewVolume(DWORD dwVolume, DWORD dwVolumeMax)
|
|
|
|
{
|
|
|
|
float fVol = (float) dwVolume / (float) dwVolumeMax; // 0.0=Max, 1.0=Min
|
|
|
|
|
|
|
|
return (LONG) ((float) DSBVOLUME_MIN * fVol);
|
|
|
|
}
|
|
|
|
|
|
|
|
//=============================================================================
|
|
|
|
|
2009-10-07 21:38:42 +00:00
|
|
|
static int g_nErrorInc = 20; // Old: 1
|
|
|
|
static int g_nErrorMax = 200; // Old: 50
|
|
|
|
|
|
|
|
int SoundCore_GetErrorInc()
|
|
|
|
{
|
|
|
|
return g_nErrorInc;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SoundCore_SetErrorInc(const int nErrorInc)
|
|
|
|
{
|
|
|
|
g_nErrorInc = nErrorInc < g_nErrorMax ? nErrorInc : g_nErrorMax;
|
|
|
|
if(g_fh) fprintf(g_fh, "Speaker/MB Error Inc = %d\n", g_nErrorInc);
|
|
|
|
}
|
|
|
|
|
|
|
|
int SoundCore_GetErrorMax()
|
|
|
|
{
|
|
|
|
return g_nErrorMax;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SoundCore_SetErrorMax(const int nErrorMax)
|
|
|
|
{
|
|
|
|
g_nErrorMax = nErrorMax < MAX_SAMPLES ? nErrorMax : MAX_SAMPLES;
|
|
|
|
if(g_fh) fprintf(g_fh, "Speaker/MB Error Max = %d\n", g_nErrorMax);
|
|
|
|
}
|
|
|
|
|
|
|
|
//=============================================================================
|
|
|
|
|
2006-02-25 20:50:29 +00:00
|
|
|
static DWORD g_dwAdviseToken;
|
|
|
|
static IReferenceClock *g_pRefClock = NULL;
|
|
|
|
static HANDLE g_hSemaphore = NULL;
|
|
|
|
static bool g_bRefClockTimerActive = false;
|
|
|
|
static DWORD g_dwLastUsecPeriod = 0;
|
|
|
|
|
|
|
|
|
2007-03-26 21:27:29 +00:00
|
|
|
bool SysClk_InitTimer()
|
2006-02-25 20:50:29 +00:00
|
|
|
{
|
|
|
|
g_hSemaphore = CreateSemaphore(NULL, 0, 1, NULL); // Max count = 1
|
|
|
|
if (g_hSemaphore == NULL)
|
2007-03-31 14:45:44 +00:00
|
|
|
{
|
2006-02-25 20:50:29 +00:00
|
|
|
fprintf(stderr, "Error creating semaphore\n");
|
2007-03-31 14:45:44 +00:00
|
|
|
return false;
|
|
|
|
}
|
2006-02-25 20:50:29 +00:00
|
|
|
|
|
|
|
if (CoCreateInstance(CLSID_SystemClock, NULL, CLSCTX_INPROC,
|
|
|
|
IID_IReferenceClock, (LPVOID*)&g_pRefClock) != S_OK)
|
2007-03-26 21:27:29 +00:00
|
|
|
{
|
2006-02-25 20:50:29 +00:00
|
|
|
fprintf(stderr, "Error initialising COM\n");
|
2007-03-26 21:27:29 +00:00
|
|
|
return false; // Fails for Win95!
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
2006-02-25 20:50:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SysClk_UninitTimer()
|
|
|
|
{
|
|
|
|
SysClk_StopTimer();
|
|
|
|
|
|
|
|
SAFE_RELEASE(g_pRefClock);
|
|
|
|
|
|
|
|
if (CloseHandle(g_hSemaphore) == 0)
|
|
|
|
fprintf(stderr, "Error closing semaphore handle\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
|
|
|
|
void SysClk_WaitTimer()
|
|
|
|
{
|
|
|
|
if(!g_bRefClockTimerActive)
|
|
|
|
return;
|
|
|
|
|
|
|
|
WaitForSingleObject(g_hSemaphore, INFINITE);
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
|
|
|
|
void SysClk_StartTimerUsec(DWORD dwUsecPeriod)
|
|
|
|
{
|
|
|
|
if(g_bRefClockTimerActive && (g_dwLastUsecPeriod == dwUsecPeriod))
|
|
|
|
return;
|
|
|
|
|
|
|
|
SysClk_StopTimer();
|
|
|
|
|
|
|
|
REFERENCE_TIME rtPeriod = (REFERENCE_TIME) (dwUsecPeriod * 10); // In units of 100ns
|
|
|
|
REFERENCE_TIME rtNow;
|
|
|
|
|
|
|
|
HRESULT hr = g_pRefClock->GetTime(&rtNow);
|
|
|
|
// S_FALSE : Returned time is the same as the previous value
|
|
|
|
|
|
|
|
if ((hr != S_OK) && (hr != S_FALSE))
|
|
|
|
{
|
|
|
|
_ASSERT(0);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (g_pRefClock->AdvisePeriodic(rtNow, rtPeriod, g_hSemaphore, &g_dwAdviseToken) != S_OK)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "Error creating timer\n");
|
|
|
|
_ASSERT(0);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_dwLastUsecPeriod = dwUsecPeriod;
|
|
|
|
g_bRefClockTimerActive = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SysClk_StopTimer()
|
|
|
|
{
|
|
|
|
if(!g_bRefClockTimerActive)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (g_pRefClock->Unadvise(g_dwAdviseToken) != S_OK)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "Error deleting timer\n");
|
|
|
|
_ASSERT(0);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_bRefClockTimerActive = false;
|
|
|
|
}
|