AxibugEmuOnline_old/AxibugEmuOnline.Client/Assets/VirtualNes.Core/APU.cs

110 lines
2.9 KiB
C#
Raw Normal View History

2024-07-25 18:34:52 +08:00
using System;
using VirtualNes.Core.Debug;
2024-07-23 18:31:59 +08:00
namespace VirtualNes.Core
{
public class APU
{
2024-07-25 18:34:52 +08:00
public const uint QUEUE_LENGTH = 8192;
2024-07-23 18:31:59 +08:00
private NES nes;
private byte exsound_select;
2024-07-25 11:03:58 +08:00
private APU_INTERNAL @internal = new APU_INTERNAL();
2024-07-23 18:31:59 +08:00
private int last_data;
private int last_diff;
protected short[] m_SoundBuffer = new short[256];
protected int[] lowpass_filter = new int[4];
2024-07-25 18:34:52 +08:00
protected QUEUE queue = new QUEUE();
protected QUEUE exqueue = new QUEUE();
2024-07-25 14:03:52 +08:00
protected bool[] m_bMute = new bool[16];
2024-07-23 18:31:59 +08:00
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;
}
2024-07-25 11:03:58 +08:00
public void Dispose()
{
}
internal void SyncDPCM(int cycles)
{
@internal.Sync(cycles);
}
2024-07-25 14:03:52 +08:00
internal byte Read(ushort addr)
{
return @internal.SyncRead(addr);
}
2024-07-25 18:34:52 +08:00
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;
}
2024-07-23 18:31:59 +08:00
}
public struct QUEUEDATA
{
public int time;
public ushort addr;
public byte data;
public byte reserved;
}
2024-07-25 18:34:52 +08:00
public class QUEUE
2024-07-23 18:31:59 +08:00
{
public int rdptr;
public int wrptr;
2024-07-25 18:34:52 +08:00
public QUEUEDATA[] data = new QUEUEDATA[8192];
2024-07-23 18:31:59 +08:00
}
}