2008-08-06 01:09:59 +00:00
|
|
|
/****************************************************************************
|
2010-01-27 22:20:37 +00:00
|
|
|
* Snes9x Nintendo Wii/Gamecube Port
|
2008-08-06 01:09:59 +00:00
|
|
|
*
|
2020-07-02 08:50:35 +02:00
|
|
|
* Tantric 2008-2020
|
2008-08-06 01:09:59 +00:00
|
|
|
*
|
2008-09-12 05:28:40 +00:00
|
|
|
* audio.cpp
|
2008-08-06 01:09:59 +00:00
|
|
|
*
|
2008-09-12 05:28:40 +00:00
|
|
|
* Audio driver
|
|
|
|
* Audio is fixed to 32Khz/16bit/Stereo
|
|
|
|
***************************************************************************/
|
|
|
|
|
2008-08-06 01:09:59 +00:00
|
|
|
#include <gccore.h>
|
|
|
|
#include <ogcsys.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2009-03-11 17:28:37 +00:00
|
|
|
#include <asndlib.h>
|
2008-09-12 05:28:40 +00:00
|
|
|
|
2008-08-06 01:09:59 +00:00
|
|
|
#include "video.h"
|
|
|
|
|
2010-03-21 23:43:54 +00:00
|
|
|
#include "snes9x/snes9x.h"
|
|
|
|
#include "snes9x/memmap.h"
|
|
|
|
#include "snes9x/cpuexec.h"
|
|
|
|
#include "snes9x/ppu.h"
|
|
|
|
#include "snes9x/apu/apu.h"
|
|
|
|
#include "snes9x/display.h"
|
|
|
|
#include "snes9x/gfx.h"
|
|
|
|
#include "snes9x/spc7110.h"
|
|
|
|
#include "snes9x/controls.h"
|
|
|
|
|
2018-08-22 15:19:11 -06:00
|
|
|
extern int ScreenshotRequested;
|
2008-12-30 00:08:17 +00:00
|
|
|
extern int ConfigRequested;
|
|
|
|
|
2008-08-06 01:09:59 +00:00
|
|
|
/*** Double buffered audio ***/
|
2018-08-22 15:19:11 -06:00
|
|
|
#define SAMPLES_TO_PROCESS 1024
|
2008-08-06 01:09:59 +00:00
|
|
|
#define AUDIOBUFFER 2048
|
2018-08-22 15:19:11 -06:00
|
|
|
#define BUFFERCOUNT 16
|
|
|
|
static u8 soundbuffer[BUFFERCOUNT][AUDIOBUFFER] __attribute__ ((__aligned__ (32)));
|
|
|
|
static int playab = 0;
|
|
|
|
static int nextab = 0;
|
2018-08-22 17:24:37 -06:00
|
|
|
static int unplayed = 0;
|
2008-08-06 01:09:59 +00:00
|
|
|
|
2018-08-22 17:24:37 -06:00
|
|
|
static inline void updateUnplayed(int diff) {
|
|
|
|
unplayed += diff;
|
2008-08-06 01:09:59 +00:00
|
|
|
|
2018-08-22 17:24:37 -06:00
|
|
|
if(unplayed < 0) {
|
|
|
|
unplayed = 0;
|
2018-08-22 15:19:11 -06:00
|
|
|
}
|
2018-08-22 17:24:37 -06:00
|
|
|
else if(unplayed > BUFFERCOUNT-1) {
|
|
|
|
unplayed = BUFFERCOUNT-1;
|
2008-08-06 01:09:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-22 15:19:11 -06:00
|
|
|
static void DMACallback () {
|
|
|
|
if (!ScreenshotRequested && !ConfigRequested) {
|
2018-08-22 17:24:37 -06:00
|
|
|
updateUnplayed(-1);
|
2018-08-22 15:19:11 -06:00
|
|
|
AUDIO_InitDMA ((u32) soundbuffer[playab], AUDIOBUFFER);
|
|
|
|
playab = (playab + 1) % BUFFERCOUNT;
|
2009-03-11 17:28:37 +00:00
|
|
|
}
|
2008-08-06 01:09:59 +00:00
|
|
|
}
|
|
|
|
|
2018-08-22 15:19:11 -06:00
|
|
|
static void S9xAudioCallback (void *data) {
|
2018-08-22 17:24:37 -06:00
|
|
|
double rate = 1.0;
|
|
|
|
|
|
|
|
if(unplayed > 8) {
|
|
|
|
rate = 1.005;
|
|
|
|
}
|
|
|
|
else if(unplayed < 4) {
|
|
|
|
rate = 0.995;
|
|
|
|
}
|
|
|
|
|
|
|
|
S9xUpdateDynamicRate(rate);
|
2010-04-13 23:34:01 +00:00
|
|
|
S9xFinalizeSamples();
|
2018-08-22 15:19:11 -06:00
|
|
|
|
|
|
|
if (ScreenshotRequested || ConfigRequested) {
|
|
|
|
AUDIO_StopDMA();
|
|
|
|
}
|
2018-08-22 17:24:37 -06:00
|
|
|
else if(S9xGetSampleCount() >= SAMPLES_TO_PROCESS) {
|
2018-08-22 15:19:11 -06:00
|
|
|
S9xMixSamples (soundbuffer[nextab], SAMPLES_TO_PROCESS);
|
|
|
|
DCFlushRange (soundbuffer[nextab], AUDIOBUFFER);
|
2018-08-22 17:24:37 -06:00
|
|
|
updateUnplayed(1);
|
|
|
|
nextab = (nextab + 1) % BUFFERCOUNT;
|
|
|
|
|
2018-08-23 10:27:26 -06:00
|
|
|
if(!Settings.TurboMode && ((nextab + 1) % BUFFERCOUNT) == playab) {
|
|
|
|
// quick and dirty attempt to prevent reading and writing from/to the same buffer
|
|
|
|
nextab = (nextab + BUFFERCOUNT/2) % BUFFERCOUNT;
|
|
|
|
}
|
|
|
|
|
2018-08-22 15:19:11 -06:00
|
|
|
if(playab == -1) {
|
2018-08-22 17:24:37 -06:00
|
|
|
if(unplayed > 2) {
|
2018-08-22 15:19:11 -06:00
|
|
|
playab = 0;
|
|
|
|
AUDIO_StartDMA();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-01-27 22:08:56 +00:00
|
|
|
}
|
|
|
|
|
2008-09-12 05:28:40 +00:00
|
|
|
/****************************************************************************
|
2009-03-12 07:07:52 +00:00
|
|
|
* InitAudio
|
2008-09-12 05:28:40 +00:00
|
|
|
***************************************************************************/
|
2008-08-06 01:09:59 +00:00
|
|
|
void
|
2009-03-12 07:07:52 +00:00
|
|
|
InitAudio ()
|
2008-08-06 01:09:59 +00:00
|
|
|
{
|
2009-06-13 08:40:45 +00:00
|
|
|
#ifdef NO_SOUND
|
|
|
|
AUDIO_Init (NULL);
|
2018-08-18 12:44:28 -06:00
|
|
|
AUDIO_SetDSPSampleRate(AI_SAMPLERATE_48KHZ);
|
2018-08-22 15:19:11 -06:00
|
|
|
AUDIO_RegisterDMACallback(DMACallback);
|
2009-06-13 08:40:45 +00:00
|
|
|
#else
|
|
|
|
ASND_Init();
|
|
|
|
#endif
|
2008-08-06 01:09:59 +00:00
|
|
|
}
|
|
|
|
|
2009-03-28 19:03:35 +00:00
|
|
|
/****************************************************************************
|
|
|
|
* SwitchAudioMode
|
|
|
|
*
|
|
|
|
* Switches between menu sound and emulator sound
|
|
|
|
***************************************************************************/
|
2009-03-11 17:28:37 +00:00
|
|
|
void
|
|
|
|
SwitchAudioMode(int mode)
|
|
|
|
{
|
|
|
|
if(mode == 0) // emulator
|
|
|
|
{
|
|
|
|
#ifndef NO_SOUND
|
|
|
|
ASND_Pause(1);
|
2018-08-14 15:38:54 -06:00
|
|
|
ASND_End();
|
2009-06-13 08:40:45 +00:00
|
|
|
AUDIO_StopDMA();
|
2018-08-14 15:38:54 -06:00
|
|
|
AUDIO_RegisterDMACallback(NULL);
|
|
|
|
DSP_Halt();
|
2018-08-22 15:19:11 -06:00
|
|
|
AUDIO_RegisterDMACallback(DMACallback);
|
2010-05-21 01:36:02 +00:00
|
|
|
#endif
|
2018-08-22 15:19:11 -06:00
|
|
|
S9xSetSamplesAvailableCallback(S9xAudioCallback, NULL);
|
2009-03-11 17:28:37 +00:00
|
|
|
}
|
|
|
|
else // menu
|
|
|
|
{
|
2010-01-27 22:08:56 +00:00
|
|
|
S9xSetSamplesAvailableCallback(NULL, NULL);
|
2009-03-11 17:28:37 +00:00
|
|
|
#ifndef NO_SOUND
|
2018-08-14 15:38:54 -06:00
|
|
|
DSP_Unhalt();
|
2009-06-13 17:48:03 +00:00
|
|
|
ASND_Init();
|
2009-03-11 17:28:37 +00:00
|
|
|
ASND_Pause(0);
|
2009-06-13 17:48:03 +00:00
|
|
|
#else
|
2010-05-21 01:36:02 +00:00
|
|
|
AUDIO_StopDMA();
|
2009-03-11 17:28:37 +00:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-03-28 19:03:35 +00:00
|
|
|
/****************************************************************************
|
|
|
|
* ShutdownAudio
|
|
|
|
*
|
|
|
|
* Shuts down audio subsystem. Useful to avoid unpleasant sounds if a
|
|
|
|
* crash occurs during shutdown.
|
|
|
|
***************************************************************************/
|
2009-03-12 07:07:52 +00:00
|
|
|
void ShutdownAudio()
|
|
|
|
{
|
|
|
|
AUDIO_StopDMA();
|
|
|
|
}
|
|
|
|
|
2008-09-12 05:28:40 +00:00
|
|
|
/****************************************************************************
|
2008-08-06 01:09:59 +00:00
|
|
|
* AudioStart
|
|
|
|
*
|
|
|
|
* Called to kick off the Audio Queue
|
2008-09-12 05:28:40 +00:00
|
|
|
***************************************************************************/
|
2008-08-06 01:09:59 +00:00
|
|
|
void
|
|
|
|
AudioStart ()
|
|
|
|
{
|
2018-08-22 17:24:37 -06:00
|
|
|
unplayed = 0;
|
2018-08-22 15:19:11 -06:00
|
|
|
nextab = 0;
|
|
|
|
playab = -1;
|
2008-08-06 01:09:59 +00:00
|
|
|
}
|