mirror of
https://github.com/libretro/blueMSX-libretro.git
synced 2024-11-27 10:40:23 +00:00
228 lines
5.8 KiB
C++
Executable File
228 lines
5.8 KiB
C++
Executable File
/*****************************************************************************
|
|
** $Source: /cygdrive/d/Private/_SVNROOT/bluemsx/blueMSX/Src/SoundChips/MsxAudio.cpp,v $
|
|
**
|
|
** $Revision: 1.8 $
|
|
**
|
|
** $Date: 2006-09-21 04:28:08 $
|
|
**
|
|
** More info: http://www.bluemsx.com
|
|
**
|
|
** Copyright (C) 2003-2006 Daniel Vik
|
|
**
|
|
** 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.
|
|
**
|
|
** 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., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
**
|
|
******************************************************************************
|
|
*/
|
|
#include "MsxAudio.h"
|
|
#include <string.h>
|
|
#include "OpenMsxY8950.h"
|
|
|
|
extern "C" {
|
|
#include "Board.h"
|
|
#include "SaveState.h"
|
|
#include "IoPort.h"
|
|
#include "MediaDb.h"
|
|
#include "DeviceManager.h"
|
|
}
|
|
|
|
#define FREQUENCY 3579545
|
|
|
|
extern "C" Int32* msxaudioSync(void* ref, UInt32 count);
|
|
|
|
#define OFFSETOF(s, a) ((int)(&((s*)0)->a))
|
|
|
|
struct MsxAudio {
|
|
MsxAudio() :
|
|
timer1(0), timer2(0), timerRef1(-1), timerRef2(-1) {
|
|
memset(defaultBuffer, 0, sizeof(defaultBuffer));
|
|
}
|
|
|
|
Mixer* mixer;
|
|
Int32 handle;
|
|
|
|
Int32 deviceHandle;
|
|
Y8950* y8950;
|
|
Int32 buffer[AUDIO_MONO_BUFFER_SIZE];
|
|
Int32 defaultBuffer[AUDIO_MONO_BUFFER_SIZE];
|
|
UInt32 timer1;
|
|
UInt32 counter1;
|
|
UInt8 timerRef1;
|
|
UInt32 timer2;
|
|
UInt32 counter2;
|
|
UInt8 timerRef2;
|
|
UInt8 registerLatch;
|
|
};
|
|
|
|
MsxAudio* theMsxAudio = NULL;
|
|
|
|
|
|
extern "C" Int32* msxaudioSync(void* ref, UInt32 count)
|
|
{
|
|
MsxAudio* msxaudio = (MsxAudio*)ref;
|
|
Int32* genBuf = NULL;
|
|
|
|
genBuf = (Int32*)msxaudio->y8950->updateBuffer(count);
|
|
if (genBuf == NULL) {
|
|
genBuf = msxaudio->defaultBuffer;
|
|
}
|
|
return genBuf;
|
|
}
|
|
|
|
void msxaudioTimerSet(int timer, int count)
|
|
{
|
|
if (timer == 1) {
|
|
if (theMsxAudio->counter1 != -1) {
|
|
theMsxAudio->counter1 = count;
|
|
}
|
|
theMsxAudio->timer1 = count;
|
|
}
|
|
else {
|
|
if (theMsxAudio->counter2 != -1) {
|
|
theMsxAudio->counter2 = count;
|
|
}
|
|
theMsxAudio->timer2 = count;
|
|
}
|
|
}
|
|
|
|
void msxaudioTimerStart(int timer, int start, UInt8 ref)
|
|
{
|
|
if (timer == 1) {
|
|
theMsxAudio->timerRef1 = ref;
|
|
theMsxAudio->counter1 = start ? theMsxAudio->timer1 : -1;
|
|
}
|
|
else {
|
|
theMsxAudio->timerRef2 = ref;
|
|
theMsxAudio->counter2 = start ? theMsxAudio->timer2 : -1;
|
|
}
|
|
}
|
|
|
|
extern "C" void msxaudioTick2() {
|
|
if (theMsxAudio != NULL) {
|
|
theMsxAudio->y8950->pushTime(boardSystemTime());
|
|
}
|
|
}
|
|
|
|
extern "C" void msxaudioTick(UInt32 elapsedTime)
|
|
{
|
|
if (theMsxAudio != NULL) {
|
|
while (elapsedTime--) {
|
|
if (theMsxAudio->counter1 != -1) {
|
|
if (theMsxAudio->counter1-- == 0) {
|
|
theMsxAudio->counter1 = theMsxAudio->timer1;
|
|
theMsxAudio->y8950->callback(theMsxAudio->timerRef1);
|
|
}
|
|
}
|
|
|
|
if (theMsxAudio->counter2 != -1) {
|
|
if (theMsxAudio->counter2-- == 0) {
|
|
theMsxAudio->counter2 = theMsxAudio->timer2;
|
|
theMsxAudio->y8950->callback(theMsxAudio->timerRef2);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
extern "C" void msxaudioDestroy(void* rm) {
|
|
MsxAudio* msxaudio = (MsxAudio*)rm;
|
|
deviceManagerUnregister(msxaudio->deviceHandle);
|
|
|
|
ioPortUnregister(0xc0);
|
|
ioPortUnregister(0xc1);
|
|
|
|
mixerUnregisterChannel(msxaudio->mixer, msxaudio->handle);
|
|
|
|
delete msxaudio->y8950;
|
|
|
|
theMsxAudio = NULL;
|
|
}
|
|
|
|
|
|
|
|
extern "C" void msxaudioSaveState(void* rm)
|
|
{
|
|
}
|
|
|
|
extern "C" void msxaudioLoadState(void* rm)
|
|
{
|
|
}
|
|
|
|
extern "C" UInt8 msxaudioRead(MsxAudio* msxaudio, UInt16 ioPort)
|
|
{
|
|
UInt8 result = 0xff;
|
|
UInt32 systemTime = boardSystemTime();
|
|
|
|
switch (ioPort & 0x01) {
|
|
case 0:
|
|
result = msxaudio->y8950->readStatus();
|
|
break;
|
|
case 1:
|
|
mixerSync(msxaudio->mixer);
|
|
result = msxaudio->y8950->readReg(msxaudio->registerLatch, systemTime);
|
|
break;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
|
|
extern "C" void msxaudioWrite(MsxAudio* msxaudio, UInt16 ioPort, UInt8 value)
|
|
{
|
|
UInt32 systemTime = boardSystemTime();
|
|
switch (ioPort & 0x01) {
|
|
case 0:
|
|
msxaudio->registerLatch = value;
|
|
break;
|
|
case 1:
|
|
mixerSync(msxaudio->mixer);
|
|
msxaudio->y8950->writeReg(msxaudio->registerLatch, value, systemTime);
|
|
break;
|
|
}
|
|
}
|
|
|
|
void msxaudioSetSampleRate(void* ref, UInt32 rate)
|
|
{
|
|
MsxAudio* msxaudio = (MsxAudio*)ref;
|
|
msxaudio->y8950->setSampleRate(rate, boardGetY8950Oversampling());
|
|
}
|
|
|
|
extern "C" int msxaudioCreate(Mixer* mixer)
|
|
{
|
|
DeviceCallbacks callbacks = { msxaudioDestroy, NULL, msxaudioSaveState, msxaudioLoadState };
|
|
MsxAudio* msxaudio = new MsxAudio;
|
|
UInt32 systemTime = boardSystemTime();
|
|
|
|
theMsxAudio = msxaudio;
|
|
|
|
msxaudio->mixer = mixer;
|
|
msxaudio->counter1 = -1;
|
|
msxaudio->counter2 = -1;
|
|
msxaudio->registerLatch = 0;
|
|
|
|
msxaudio->handle = mixerRegisterChannel(mixer, MIXER_CHANNEL_MSXAUDIO, 0, msxaudioSync, msxaudioSetSampleRate, msxaudio);
|
|
|
|
msxaudio->deviceHandle = deviceManagerRegister(ROM_MSXAUDIO, &callbacks, msxaudio);
|
|
|
|
msxaudio->y8950 = new Y8950("MsxAudio", 256*1024, systemTime);
|
|
msxaudio->y8950->setSampleRate(mixerGetSampleRate(mixer), boardGetY8950Oversampling());
|
|
msxaudio->y8950->setVolume(32767);
|
|
|
|
ioPortRegister(0xc0, (IoPortRead)msxaudioRead, (IoPortWrite)msxaudioWrite, msxaudio);
|
|
ioPortRegister(0xc1, (IoPortRead)msxaudioRead, (IoPortWrite)msxaudioWrite, msxaudio);
|
|
|
|
return 1;
|
|
}
|
|
|
|
|