2008-09-02 01:57:21 +00:00
|
|
|
/****************************************************************************
|
2009-07-22 02:05:49 +00:00
|
|
|
* FCE Ultra
|
2008-09-02 01:57:21 +00:00
|
|
|
* Nintendo Wii/Gamecube Port
|
|
|
|
*
|
2021-01-06 21:13:49 +01:00
|
|
|
* Tantric 2008-2021
|
2008-10-19 17:26:40 +00:00
|
|
|
* eke-eke October 2008
|
2008-09-02 01:57:21 +00:00
|
|
|
*
|
2009-03-28 17:23:08 +00:00
|
|
|
* gcaudio.cpp
|
2008-09-02 01:57:21 +00:00
|
|
|
*
|
|
|
|
* Audio driver
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#include <gccore.h>
|
2008-10-01 06:03:16 +00:00
|
|
|
#include <string.h>
|
2009-03-28 17:23:08 +00:00
|
|
|
#include <asndlib.h>
|
2011-05-15 18:33:57 +00:00
|
|
|
#include "fceusupport.h"
|
2008-10-18 15:58:54 +00:00
|
|
|
|
2018-08-22 20:52:19 -06:00
|
|
|
extern int ScreenshotRequested;
|
|
|
|
extern int ConfigRequested;
|
2008-10-18 15:58:54 +00:00
|
|
|
static u8 soundbuffer[2][3840] ATTRIBUTE_ALIGN(32);
|
|
|
|
static u8 mixbuffer[16000];
|
|
|
|
static int mixhead = 0;
|
|
|
|
static int mixtail = 0;
|
|
|
|
static int whichab = 0;
|
2008-12-30 00:05:57 +00:00
|
|
|
static int IsPlaying = 0;
|
2011-05-15 18:33:57 +00:00
|
|
|
static int samplerate;
|
2008-10-18 15:58:54 +00:00
|
|
|
|
2008-10-22 16:46:44 +00:00
|
|
|
/****************************************************************************
|
|
|
|
* MixerCollect
|
|
|
|
*
|
|
|
|
* Collects sound samples from mixbuffer and puts them into outbuffer
|
|
|
|
* Makes sure to align them to 32 bytes for AUDIO_InitDMA
|
|
|
|
***************************************************************************/
|
|
|
|
static int MixerCollect( u8 *outbuffer, int len )
|
2008-10-18 15:58:54 +00:00
|
|
|
{
|
2008-10-19 17:26:40 +00:00
|
|
|
u32 *dst = (u32 *)outbuffer;
|
|
|
|
u32 *src = (u32 *)mixbuffer;
|
|
|
|
int done = 0;
|
2008-10-18 15:58:54 +00:00
|
|
|
|
2008-10-19 17:26:40 +00:00
|
|
|
// Always clear output buffer
|
|
|
|
memset(outbuffer, 0, len);
|
2008-10-18 15:58:54 +00:00
|
|
|
|
2008-10-19 17:26:40 +00:00
|
|
|
while ( ( mixtail != mixhead ) && ( done < len ) )
|
|
|
|
{
|
|
|
|
*dst++ = src[mixtail++];
|
|
|
|
if (mixtail == 4000) mixtail = 0;
|
|
|
|
done += 4;
|
|
|
|
}
|
2008-10-18 15:58:54 +00:00
|
|
|
|
2008-10-19 17:26:40 +00:00
|
|
|
// Realign to 32 bytes for DMA
|
|
|
|
mixtail -= ((done&0x1f) >> 2);
|
|
|
|
if (mixtail < 0)
|
|
|
|
mixtail += 4000;
|
|
|
|
done &= ~0x1f;
|
|
|
|
if (!done)
|
|
|
|
return len >> 1;
|
|
|
|
|
|
|
|
return done;
|
2008-10-18 15:58:54 +00:00
|
|
|
}
|
2008-09-02 01:57:21 +00:00
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* AudioSwitchBuffers
|
|
|
|
*
|
|
|
|
* Manages which buffer is played next
|
2008-10-22 16:46:44 +00:00
|
|
|
***************************************************************************/
|
2008-12-30 00:05:57 +00:00
|
|
|
static void AudioSwitchBuffers()
|
2008-09-02 01:57:21 +00:00
|
|
|
{
|
2018-08-22 20:52:19 -06:00
|
|
|
if ( !ScreenshotRequested && !ConfigRequested ) {
|
|
|
|
IsPlaying = 1;
|
2008-10-22 16:46:44 +00:00
|
|
|
int len = MixerCollect( soundbuffer[whichab], 3840 );
|
2008-10-19 17:26:40 +00:00
|
|
|
DCFlushRange(soundbuffer[whichab], len);
|
|
|
|
AUDIO_InitDMA((u32)soundbuffer[whichab], len);
|
2018-08-22 20:52:19 -06:00
|
|
|
whichab ^= 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
IsPlaying = 0;
|
2008-10-19 17:26:40 +00:00
|
|
|
}
|
2008-09-02 01:57:21 +00:00
|
|
|
}
|
|
|
|
|
2008-10-22 16:46:44 +00:00
|
|
|
/****************************************************************************
|
|
|
|
* InitialiseAudio
|
|
|
|
*
|
|
|
|
* Initializes sound system on first load of emulator
|
|
|
|
***************************************************************************/
|
|
|
|
void InitialiseAudio()
|
2008-09-02 01:57:21 +00:00
|
|
|
{
|
2009-06-13 08:38:27 +00:00
|
|
|
#ifdef NO_SOUND
|
|
|
|
AUDIO_Init (NULL);
|
2010-05-27 21:08:02 +00:00
|
|
|
AUDIO_SetDSPSampleRate(AI_SAMPLERATE_48KHZ);
|
2009-07-01 17:39:55 +00:00
|
|
|
AUDIO_RegisterDMACallback(AudioSwitchBuffers);
|
2009-06-13 08:38:27 +00:00
|
|
|
#else
|
|
|
|
ASND_Init();
|
|
|
|
#endif
|
2008-10-19 17:26:40 +00:00
|
|
|
memset(soundbuffer, 0, 3840*2);
|
|
|
|
memset(mixbuffer, 0, 16000);
|
2008-09-02 01:57:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
2008-10-22 16:46:44 +00:00
|
|
|
* ResetAudio
|
|
|
|
*
|
|
|
|
* Reset audio output when loading a new game
|
|
|
|
***************************************************************************/
|
|
|
|
void ResetAudio()
|
|
|
|
{
|
|
|
|
memset(soundbuffer, 0, 3840*2);
|
|
|
|
memset(mixbuffer, 0, 16000);
|
|
|
|
mixhead = mixtail = 0;
|
|
|
|
}
|
|
|
|
|
2009-03-28 17:23:08 +00:00
|
|
|
/****************************************************************************
|
|
|
|
* SwitchAudioMode
|
|
|
|
*
|
|
|
|
* Switches between menu sound and emulator sound
|
|
|
|
***************************************************************************/
|
|
|
|
void
|
|
|
|
SwitchAudioMode(int mode)
|
|
|
|
{
|
|
|
|
if(mode == 0) // emulator
|
|
|
|
{
|
|
|
|
#ifndef NO_SOUND
|
|
|
|
ASND_Pause(1);
|
2018-08-14 15:39:52 -06:00
|
|
|
ASND_End();
|
2009-06-13 08:38:27 +00:00
|
|
|
AUDIO_StopDMA();
|
2018-08-14 15:39:52 -06:00
|
|
|
AUDIO_RegisterDMACallback(NULL);
|
|
|
|
DSP_Halt();
|
2009-03-28 17:23:08 +00:00
|
|
|
AUDIO_RegisterDMACallback(AudioSwitchBuffers);
|
2010-05-21 01:36:10 +00:00
|
|
|
#endif
|
2009-03-28 17:23:08 +00:00
|
|
|
}
|
|
|
|
else // menu
|
|
|
|
{
|
2009-03-28 18:26:47 +00:00
|
|
|
IsPlaying = 0;
|
2009-03-28 17:23:08 +00:00
|
|
|
#ifndef NO_SOUND
|
2018-08-14 15:39:52 -06:00
|
|
|
DSP_Unhalt();
|
2009-06-13 17:43:24 +00:00
|
|
|
ASND_Init();
|
2009-03-28 17:23:08 +00:00
|
|
|
ASND_Pause(0);
|
2009-06-13 17:43:24 +00:00
|
|
|
#else
|
|
|
|
AUDIO_StopDMA();
|
2009-03-28 17:23:08 +00:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* ShutdownAudio
|
|
|
|
*
|
|
|
|
* Shuts down audio subsystem. Useful to avoid unpleasant sounds if a
|
|
|
|
* crash occurs during shutdown.
|
|
|
|
***************************************************************************/
|
|
|
|
void ShutdownAudio()
|
|
|
|
{
|
|
|
|
AUDIO_StopDMA();
|
|
|
|
}
|
|
|
|
|
2008-10-22 16:46:44 +00:00
|
|
|
/****************************************************************************
|
|
|
|
* PlaySound
|
2008-09-02 01:57:21 +00:00
|
|
|
*
|
2008-10-22 16:46:44 +00:00
|
|
|
* Puts incoming mono samples into mixbuffer
|
|
|
|
* Splits mono samples into two channels (stereo)
|
2008-09-02 01:57:21 +00:00
|
|
|
****************************************************************************/
|
2018-07-28 15:45:29 -05:00
|
|
|
void PlaySound( int32 *Buffer, int count )
|
2008-09-02 01:57:21 +00:00
|
|
|
{
|
2008-10-18 15:58:54 +00:00
|
|
|
int i;
|
2008-10-21 07:50:37 +00:00
|
|
|
u16 sample;
|
2008-10-19 17:26:40 +00:00
|
|
|
u32 *dst = (u32 *)mixbuffer;
|
2008-10-18 15:58:54 +00:00
|
|
|
|
2018-08-22 20:52:19 -06:00
|
|
|
for( i = 0; i < count; i++ ) {
|
2008-10-19 17:26:40 +00:00
|
|
|
sample = Buffer[i] & 0xffff;
|
|
|
|
dst[mixhead++] = sample | ( sample << 16);
|
2018-08-22 20:52:19 -06:00
|
|
|
if (mixhead == 4000) {
|
2008-10-19 17:26:40 +00:00
|
|
|
mixhead = 0;
|
2018-08-22 20:52:19 -06:00
|
|
|
}
|
2008-10-19 17:26:40 +00:00
|
|
|
}
|
2008-10-18 15:58:54 +00:00
|
|
|
|
2008-10-19 17:26:40 +00:00
|
|
|
// Restart Sound Processing if stopped
|
2018-08-22 20:52:19 -06:00
|
|
|
if (IsPlaying == 0) {
|
|
|
|
AUDIO_StartDMA();
|
2008-10-19 17:26:40 +00:00
|
|
|
}
|
2008-09-02 01:57:21 +00:00
|
|
|
}
|
2011-05-15 18:33:57 +00:00
|
|
|
|
|
|
|
void UpdateSampleRate(int rate)
|
|
|
|
{
|
2018-08-23 08:57:30 -06:00
|
|
|
if(samplerate != rate) {
|
|
|
|
samplerate = rate;
|
|
|
|
FCEUI_Sound(samplerate);
|
|
|
|
}
|
2011-05-15 18:33:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SetSampleRate()
|
|
|
|
{
|
|
|
|
FCEUI_Sound(samplerate);
|
|
|
|
}
|