forked from sin365/AxibugEmuOnline
192 lines
5.4 KiB
C#
192 lines
5.4 KiB
C#
using System;
|
|
using VirtualNes.Core.Debug;
|
|
|
|
namespace VirtualNes.Core
|
|
{
|
|
public class APU
|
|
{
|
|
public const uint QUEUE_LENGTH = 8192;
|
|
|
|
private NES nes;
|
|
private byte exsound_select;
|
|
private APU_INTERNAL @internal = new APU_INTERNAL();
|
|
private APU_VRC6 vrc6 = new APU_VRC6();
|
|
private APU_VRC7 vrc7 = new APU_VRC7();
|
|
private APU_MMC5 mmc5 = new APU_MMC5();
|
|
private APU_FDS fds = new APU_FDS();
|
|
private APU_N106 n106 = new APU_N106();
|
|
private APU_FME7 fme7 = new APU_FME7();
|
|
private int last_data;
|
|
private int last_diff;
|
|
protected short[] m_SoundBuffer = new short[256];
|
|
protected int[] lowpass_filter = new int[4];
|
|
protected QUEUE queue = new QUEUE();
|
|
protected QUEUE exqueue = new QUEUE();
|
|
protected bool[] m_bMute = new bool[16];
|
|
protected double elapsed_time;
|
|
|
|
public APU(NES parent)
|
|
{
|
|
exsound_select = 0;
|
|
|
|
nes = parent;
|
|
@internal.SetParent(parent);
|
|
|
|
last_data = last_diff = 0;
|
|
|
|
Array.Clear(m_SoundBuffer, 0, m_SoundBuffer.Length);
|
|
Array.Clear(lowpass_filter, 0, lowpass_filter.Length);
|
|
|
|
for (int i = 0; i < m_bMute.Length; i++)
|
|
m_bMute[i] = true;
|
|
}
|
|
|
|
public void Dispose()
|
|
{
|
|
}
|
|
|
|
internal void SyncDPCM(int cycles)
|
|
{
|
|
@internal.Sync(cycles);
|
|
}
|
|
|
|
internal byte Read(ushort addr)
|
|
{
|
|
return @internal.SyncRead(addr);
|
|
}
|
|
|
|
internal void Write(ushort addr, byte data)
|
|
{
|
|
// $4018偼VirtuaNES屌桳億乕僩
|
|
if (addr >= 0x4000 && addr <= 0x401F)
|
|
{
|
|
@internal.SyncWrite(addr, data);
|
|
SetQueue(nes.cpu.GetTotalCycles(), addr, data);
|
|
}
|
|
}
|
|
|
|
private void SetQueue(int writetime, ushort addr, byte data)
|
|
{
|
|
queue.data[queue.wrptr].time = writetime;
|
|
queue.data[queue.wrptr].addr = addr;
|
|
queue.data[queue.wrptr].data = data;
|
|
queue.wrptr++;
|
|
|
|
var newwrptr = (int)(queue.wrptr & (QUEUE_LENGTH - 1));
|
|
queue.wrptr = newwrptr;
|
|
|
|
if (queue.wrptr == queue.rdptr)
|
|
{
|
|
Debuger.LogError("queue overflow.");
|
|
}
|
|
}
|
|
|
|
private bool GetQueue(int writetime, ref QUEUEDATA ret)
|
|
{
|
|
if (queue.wrptr == queue.rdptr)
|
|
{
|
|
return false;
|
|
}
|
|
if (queue.data[queue.rdptr].time <= writetime)
|
|
{
|
|
ret = queue.data[queue.rdptr];
|
|
queue.rdptr++;
|
|
var newrdptr = (int)(queue.rdptr & (QUEUE_LENGTH - 1));
|
|
queue.rdptr = newrdptr;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
public void SoundSetup()
|
|
{
|
|
float fClock = nes.nescfg.CpuClock;
|
|
int nRate = Supporter.Config.sound.nRate;
|
|
|
|
@internal.Setup(fClock, nRate);
|
|
vrc6.Setup(fClock, nRate);
|
|
vrc7.Setup(fClock, nRate);
|
|
mmc5.Setup(fClock, nRate);
|
|
fds.Setup(fClock, nRate);
|
|
n106.Setup(fClock, nRate);
|
|
fme7.Setup(fClock, nRate);
|
|
}
|
|
|
|
internal void SelectExSound(byte data)
|
|
{
|
|
exsound_select = data;
|
|
}
|
|
|
|
internal void Reset()
|
|
{
|
|
queue = new QUEUE();
|
|
exqueue = new QUEUE();
|
|
|
|
elapsed_time = 0;
|
|
|
|
float fClock = nes.nescfg.CpuClock;
|
|
int nRate = Supporter.Config.sound.nRate;
|
|
|
|
@internal.Reset(fClock, nRate);
|
|
vrc6.Reset(fClock, nRate);
|
|
vrc7.Reset(fClock, nRate);
|
|
mmc5.Reset(fClock, nRate);
|
|
fds.Reset(fClock, nRate);
|
|
n106.Reset(fClock, nRate);
|
|
fme7.Reset(fClock, nRate);
|
|
|
|
SoundSetup();
|
|
}
|
|
|
|
internal void ExWrite(ushort addr, byte data)
|
|
{
|
|
SetExQueue(nes.cpu.GetTotalCycles(), addr, data);
|
|
|
|
if ((exsound_select & 0x04) != 0)
|
|
{
|
|
if (addr >= 0x4040 && addr < 0x4100)
|
|
{
|
|
fds.SyncWrite(addr, data);
|
|
}
|
|
}
|
|
|
|
if ((exsound_select & 0x08) != 0)
|
|
{
|
|
if (addr >= 0x5000 && addr <= 0x5015)
|
|
{
|
|
mmc5.SyncWrite(addr, data);
|
|
}
|
|
}
|
|
}
|
|
|
|
private void SetExQueue(int writetime, ushort addr, byte data)
|
|
{
|
|
exqueue.data[exqueue.wrptr].time = writetime;
|
|
exqueue.data[exqueue.wrptr].addr = addr;
|
|
exqueue.data[exqueue.wrptr].data = data;
|
|
exqueue.wrptr++;
|
|
var temp = QUEUE_LENGTH - 1;
|
|
exqueue.wrptr = (int)(exqueue.wrptr & temp);
|
|
if (exqueue.wrptr == exqueue.rdptr)
|
|
{
|
|
Debuger.LogError("exqueue overflow.");
|
|
}
|
|
}
|
|
}
|
|
|
|
public struct QUEUEDATA
|
|
{
|
|
public int time;
|
|
public ushort addr;
|
|
public byte data;
|
|
public byte reserved;
|
|
}
|
|
|
|
public class QUEUE
|
|
{
|
|
public int rdptr;
|
|
public int wrptr;
|
|
public QUEUEDATA[] data = new QUEUEDATA[8192];
|
|
}
|
|
}
|