2008-12-11 18:38:29 +01:00
|
|
|
/****************************************************************************
|
2009-04-15 17:33:51 +02:00
|
|
|
* gx_audio.c
|
2008-08-07 14:26:07 +02:00
|
|
|
*
|
2008-12-11 18:38:29 +01:00
|
|
|
* Genesis Plus GX audio support
|
2008-08-07 14:26:07 +02:00
|
|
|
*
|
2009-05-01 14:56:48 +02:00
|
|
|
* code by Eke-Eke (2007,2009)
|
2008-08-07 14:26:07 +02:00
|
|
|
*
|
2008-12-11 18:38:29 +01:00
|
|
|
* This program 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.
|
2008-08-07 14:26:07 +02:00
|
|
|
*
|
2008-12-11 18:38:29 +01:00
|
|
|
* This program 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 this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
2008-08-07 14:26:07 +02:00
|
|
|
*
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
#include "shared.h"
|
2009-04-05 20:20:43 +02:00
|
|
|
|
|
|
|
#include <asndlib.h>
|
2009-04-20 00:31:08 +02:00
|
|
|
#include <oggplayer.h>
|
2008-08-07 14:26:07 +02:00
|
|
|
|
2008-12-19 16:38:26 +01:00
|
|
|
/* DMA soundbuffers (required to be 32-bytes aligned)
|
|
|
|
Length is dimensionned for one frame of emulation (see below)
|
|
|
|
To prevent audio clashes, we use double buffering technique:
|
|
|
|
one buffer is the active DMA buffer
|
|
|
|
the other one is the current work buffer (updated during frame emulation)
|
2009-02-22 20:57:41 +01:00
|
|
|
We do not need more since frame emulation and DMA operation are synchronized
|
2008-12-19 16:38:26 +01:00
|
|
|
*/
|
|
|
|
u8 soundbuffer[2][3840] ATTRIBUTE_ALIGN(32);
|
2008-08-07 14:26:07 +02:00
|
|
|
|
2008-12-19 16:38:26 +01:00
|
|
|
/* Current work soundbuffer */
|
2009-01-14 17:32:35 +01:00
|
|
|
u8 mixbuffer;
|
2008-12-19 16:38:26 +01:00
|
|
|
|
2009-02-22 20:57:41 +01:00
|
|
|
/* Next DMA length */
|
|
|
|
static u32 dma_len;
|
|
|
|
|
|
|
|
/* Current delta between output & expected sample counts */
|
|
|
|
static int delta;
|
|
|
|
|
|
|
|
/* Expected sample count x 100 */
|
|
|
|
static u32 dma_sync;
|
|
|
|
|
|
|
|
/* audio DMA status */
|
2009-04-20 00:31:08 +02:00
|
|
|
static u8 audioStarted = 0;
|
2008-12-19 16:38:26 +01:00
|
|
|
|
2009-04-24 01:24:40 +02:00
|
|
|
/* Background music */
|
|
|
|
static u8 *Bg_music_ogg = NULL;
|
|
|
|
static u32 Bg_music_ogg_size = 0;
|
|
|
|
|
2009-05-01 14:56:48 +02:00
|
|
|
/***************************************************************************************/
|
|
|
|
/* Audio engine */
|
|
|
|
/***************************************************************************************/
|
2009-02-22 20:57:41 +01:00
|
|
|
|
2009-05-01 14:56:48 +02:00
|
|
|
/* Audio DMA callback */
|
|
|
|
static void ai_callback(void)
|
2008-08-07 14:26:07 +02:00
|
|
|
{
|
2008-12-18 23:34:55 +01:00
|
|
|
frameticker++;
|
2008-08-07 14:26:07 +02:00
|
|
|
}
|
|
|
|
|
2009-04-24 01:24:40 +02:00
|
|
|
void gx_audio_Init(void)
|
2008-08-07 14:26:07 +02:00
|
|
|
{
|
2008-12-10 19:16:30 +01:00
|
|
|
AUDIO_Init (NULL);
|
|
|
|
AUDIO_SetDSPSampleRate (AI_SAMPLERATE_48KHZ);
|
2009-04-24 01:24:40 +02:00
|
|
|
|
|
|
|
/* load background music from FAT device */
|
|
|
|
char fname[MAXPATHLEN];
|
|
|
|
sprintf(fname,"%s/Bg_music.ogg",DEFAULT_PATH);
|
|
|
|
FILE *f = fopen(fname,"rb");
|
|
|
|
if (f)
|
|
|
|
{
|
|
|
|
struct stat filestat;
|
|
|
|
stat(fname, &filestat);
|
|
|
|
Bg_music_ogg_size = filestat.st_size;
|
|
|
|
Bg_music_ogg = memalign(32,Bg_music_ogg_size);
|
|
|
|
if (Bg_music_ogg) fread(Bg_music_ogg,1,Bg_music_ogg_size,f);
|
|
|
|
fclose(f);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void gx_audio_Shutdown(void)
|
|
|
|
{
|
|
|
|
if (Bg_music_ogg) free(Bg_music_ogg);
|
2008-08-07 14:26:07 +02:00
|
|
|
}
|
|
|
|
|
2008-12-19 16:38:26 +01:00
|
|
|
/***
|
2009-04-15 17:33:51 +02:00
|
|
|
gx_audio__update
|
2008-12-19 16:38:26 +01:00
|
|
|
|
2009-02-22 20:57:41 +01:00
|
|
|
This function is called at the end of each frame
|
|
|
|
Genesis Plus only provides sound data on completion of each frame.
|
|
|
|
DMA sync and switching ensure we never access the active DMA buffer and sound clashes never happen
|
|
|
|
This function retrieves samples for the frame then set the next DMA parameters
|
|
|
|
Parameters will be taken in account only when current DMA operation is over
|
2008-12-19 16:38:26 +01:00
|
|
|
***/
|
2009-04-24 01:24:40 +02:00
|
|
|
void gx_audio_Update(void)
|
2008-08-07 14:26:07 +02:00
|
|
|
{
|
2009-02-22 20:57:41 +01:00
|
|
|
u32 size = dma_len;
|
|
|
|
|
|
|
|
/* get audio samples */
|
|
|
|
audio_update(dma_len);
|
|
|
|
|
|
|
|
/* update DMA parameters */
|
|
|
|
s16 *sb = (s16 *)(soundbuffer[mixbuffer]);
|
2008-12-18 23:34:55 +01:00
|
|
|
mixbuffer ^= 1;
|
2009-02-22 20:57:41 +01:00
|
|
|
size = size << 2;
|
|
|
|
DCFlushRange((void *)sb, size);
|
|
|
|
AUDIO_InitDMA((u32) sb, size);
|
|
|
|
|
|
|
|
/* Start Audio DMA */
|
|
|
|
/* this is only called once, DMA is automatically restarted when previous one is over */
|
|
|
|
/* When DMA parameters are not updated, same soundbuffer is played again */
|
|
|
|
if (!audioStarted)
|
|
|
|
{
|
|
|
|
audioStarted = 1;
|
|
|
|
AUDIO_StartDMA();
|
|
|
|
if (frameticker > 1) frameticker = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* VIDEO interrupt sync: we approximate DMA length (see below) */
|
|
|
|
/* DMA length should be 32 bytes so we use 800 or 808 samples */
|
|
|
|
if (dma_sync)
|
|
|
|
{
|
|
|
|
delta += (dma_len * 100) - dma_sync;
|
|
|
|
if (delta < 0) dma_len = 808;
|
|
|
|
else dma_len = 800;
|
|
|
|
}
|
2008-12-18 18:13:47 +01:00
|
|
|
}
|
|
|
|
|
2008-12-19 16:38:26 +01:00
|
|
|
/***
|
2009-04-15 17:33:51 +02:00
|
|
|
gx_audio__start
|
2008-12-17 18:18:07 +01:00
|
|
|
|
2009-02-22 20:57:41 +01:00
|
|
|
This function resets the audio engine
|
|
|
|
This is called when coming back from Main Menu
|
2008-12-19 16:38:26 +01:00
|
|
|
***/
|
2009-04-24 01:24:40 +02:00
|
|
|
void gx_audio_Start(void)
|
2008-12-18 18:13:47 +01:00
|
|
|
{
|
2009-04-05 20:20:43 +02:00
|
|
|
/* shutdown menu audio */
|
2009-04-20 00:31:08 +02:00
|
|
|
PauseOgg(1);
|
|
|
|
StopOgg();
|
2009-04-05 20:20:43 +02:00
|
|
|
ASND_Pause(1);
|
|
|
|
ASND_End();
|
|
|
|
|
2009-02-22 20:57:41 +01:00
|
|
|
/* initialize default DMA length */
|
2009-03-06 16:57:31 +01:00
|
|
|
/* PAL (50Hz): 20000 us period --> 960 samples/frame @48kHz */
|
|
|
|
/* NTSC (60Hz): 16667 us period --> 800 samples/frame @48kHz */
|
2009-02-22 20:57:41 +01:00
|
|
|
dma_len = vdp_pal ? 960 : 800;
|
|
|
|
dma_sync = 0;
|
|
|
|
mixbuffer = 0;
|
|
|
|
delta = 0;
|
|
|
|
|
2009-03-06 16:57:31 +01:00
|
|
|
/* reset sound buffers */
|
|
|
|
memset(soundbuffer, 0, 2 * 3840);
|
|
|
|
|
2009-04-05 20:20:43 +02:00
|
|
|
/* default */
|
|
|
|
AUDIO_SetDSPSampleRate (AI_SAMPLERATE_48KHZ);
|
2009-03-06 16:57:31 +01:00
|
|
|
AUDIO_RegisterDMACallback(NULL);
|
2009-04-05 20:20:43 +02:00
|
|
|
|
|
|
|
/* let's use audio DMA to synchronize frame emulation */
|
2009-05-01 14:56:48 +02:00
|
|
|
if (vdp_pal | gc_pal) AUDIO_RegisterDMACallback(ai_callback);
|
2009-02-22 20:57:41 +01:00
|
|
|
|
|
|
|
/* 60hz video mode requires synchronization with Video interrupt */
|
|
|
|
/* VSYNC period is 16715 us which is approx. 802.32 samples */
|
|
|
|
/* to prevent audio/video desynchronization, we approximate the exact */
|
|
|
|
/* number of samples by using alternate audio length */
|
|
|
|
else dma_sync = 80232;
|
2008-12-17 18:18:07 +01:00
|
|
|
}
|
2008-12-19 16:38:26 +01:00
|
|
|
|
|
|
|
/***
|
2009-04-15 17:33:51 +02:00
|
|
|
gx_audio__stop
|
2008-12-19 16:38:26 +01:00
|
|
|
|
2009-02-22 20:57:41 +01:00
|
|
|
This function stops current Audio DMA process
|
2008-12-19 16:38:26 +01:00
|
|
|
This is called when going back to Main Menu
|
|
|
|
DMA need to be restarted when going back to the game (see above)
|
|
|
|
***/
|
2009-04-24 01:24:40 +02:00
|
|
|
void gx_audio_Stop(void)
|
2008-12-19 16:38:26 +01:00
|
|
|
{
|
2009-04-20 00:31:08 +02:00
|
|
|
/* stop emulator audio */
|
2009-01-08 20:01:16 +01:00
|
|
|
AUDIO_StopDMA ();
|
2009-02-22 20:57:41 +01:00
|
|
|
audioStarted = 0;
|
2009-04-05 20:20:43 +02:00
|
|
|
|
2009-04-20 00:31:08 +02:00
|
|
|
/* restart menu audio */
|
2009-04-05 20:20:43 +02:00
|
|
|
ASND_Init();
|
|
|
|
ASND_Pause(0);
|
2009-04-21 03:05:56 +02:00
|
|
|
if (Bg_music_ogg)
|
|
|
|
{
|
|
|
|
PauseOgg(0);
|
|
|
|
PlayOgg(mem_open((char *)Bg_music_ogg, Bg_music_ogg_size), 0, OGG_INFINITE_TIME);
|
2009-04-23 03:01:07 +02:00
|
|
|
SetVolumeOgg(((int)config.bgm_volume * 255) / 100);
|
2009-04-21 03:05:56 +02:00
|
|
|
}
|
2008-12-19 16:38:26 +01:00
|
|
|
}
|