Files
archived-pcsx2/pcsx2/SPU2/spu2sys.cpp
Ziemas 10fc9a790d SPU: Emulate voice decode buffers
This makes the timing of NAX advancing more similar to console since it
emulates the decode buffer behaviour of it rushing ahead of playback
until the buffer is full.

It also makes interpolation of the first four samples more correct by
using real data instead of the zero filled previous values.

[SAVEVERSION+]
2026-01-28 12:18:43 -05:00

1639 lines
48 KiB
C++

// SPDX-FileCopyrightText: 2002-2026 PCSX2 Dev Team
// SPDX-License-Identifier: GPL-3.0+
// ======================================================================================
// spu2sys.cpp -- Emulation module for the SPU2 'virtual machine'
// ======================================================================================
// This module contains (most!) stuff which is directly related to SPU2 emulation.
// Contents should be cross-platform compatible whenever possible.
#include "IopCounters.h"
#include "IopDma.h"
#include "IopHw.h"
#include "R3000A.h"
#include "SPU2/Debug.h"
#include "SPU2/defs.h"
#include "SPU2/Dma.h"
#include "SPU2/regs.h"
#include "SPU2/spu2.h"
#include "common/Console.h"
s16 spu2regs[0x010000 / sizeof(s16)];
s16 _spu2mem[0x200000 / sizeof(s16)];
V_CoreDebug DebugCores[2];
V_Core Cores[2];
V_SPDIF Spdif;
StereoOut32 DCFilterIn, DCFilterOut;
u16 OutPos;
u16 InputPos;
u32 Cycles;
int PlayMode;
static bool has_to_call_irq[2] = { false, false };
static bool has_to_call_irq_dma[2] = { false, false };
StereoOut32 (*ReverbUpsample)(V_Core& core);
s32 (*ReverbDownsample)(V_Core& core, bool right);
static bool psxmode = false;
void SetIrqCall(int core)
{
// reset by an irq disable/enable cycle, behaviour found by
// test programs that bizarrely only fired one interrupt
has_to_call_irq[core] = true;
}
void SetIrqCallDMA(int core)
{
// reset by an irq disable/enable cycle, behaviour found by
// test programs that bizarrely only fired one interrupt
has_to_call_irq_dma[core] = true;
}
__forceinline s16* GetMemPtr(u32 addr)
{
#ifndef DEBUG_FAST
// In case you're wondering, this assert is the reason SPU2
// runs so incrediously slow in Debug mode. :P
pxAssume(addr < 0x100000);
#endif
return (_spu2mem + addr);
}
__forceinline s16 spu2M_Read(u32 addr)
{
return *GetMemPtr(addr & 0xfffff);
}
// writes a signed value to the SPU2 ram
// Invalidates the ADPCM cache in the process.
__forceinline void spu2M_Write(u32 addr, s16 value)
{
// Make sure the cache is invalidated:
// (note to self : addr address WORDs, not bytes)
addr &= 0xfffff;
if (addr >= SPU2_DYN_MEMLINE)
{
const int cacheIdx = addr / pcm_WordsPerBlock;
pcm_cache_data[cacheIdx].Validated = false;
if (SPU2::MsgToConsole() && SPU2::MsgCache())
SPU2::ConLog("* SPU2: PcmCache Block Clear at 0x%x (cacheIdx=0x%x)\n", addr, cacheIdx);
}
*GetMemPtr(addr) = value;
}
// writes an unsigned value to the SPU2 ram
__forceinline void spu2M_Write(u32 addr, u16 value)
{
spu2M_Write(addr, (s16)value);
}
V_VolumeLR V_VolumeLR::Max(0x7FFF);
V_VolumeSlideLR V_VolumeSlideLR::Max(0x3FFF, 0x7FFF);
void V_Core::Init(int index)
{
if (SPU2::MsgToConsole())
SPU2::ConLog("* SPU2: Init SPU2 core %d \n", index);
ReverbDownsample = MULTI_ISA_SELECT(ReverbDownsample);
ReverbUpsample = MULTI_ISA_SELECT(ReverbUpsample);
//memset(this, 0, sizeof(V_Core));
// Explicitly initializing variables instead.
Mute = false;
DMABits = 0;
NoiseClk = 0;
NoiseCnt = 0;
NoiseOut = 0;
AutoDMACtrl = 0;
InputDataLeft = 0;
InputPosWrite = 0x100;
InputDataProgress = 0;
InputDataTransferred = 0;
LastEffect.Left = 0;
LastEffect.Right = 0;
CoreEnabled = 0;
AttrBit0 = 0;
DmaMode = 0;
DMAPtr = nullptr;
KeyOn = 0;
OutPos = 0;
DCFilterIn = {};
DCFilterOut = {};
psxmode = false;
psxSoundDataTransferControl = 0;
psxSPUSTAT = 0;
const int c = Index = index;
Regs.STATX = 0;
Regs.ATTR = 0;
ExtVol = V_VolumeLR::Max;
InpVol = V_VolumeLR::Max;
FxVol = V_VolumeLR(0);
MasterVol = V_VolumeSlideLR(0, 0);
memset(&DryGate, -1, sizeof(DryGate));
memset(&WetGate, -1, sizeof(WetGate));
DryGate.ExtL = 0;
DryGate.ExtR = 0;
if (!c)
{
WetGate.ExtL = 0;
WetGate.ExtR = 0;
}
Regs.MMIX = c ? 0xFFC : 0xFF0; // PS2 confirmed (f3c and f30 after BIOS ran, ffc and ff0 after sdinit)
Regs.VMIXL = 0xFFFFFF;
Regs.VMIXR = 0xFFFFFF;
Regs.VMIXEL = 0xFFFFFF;
Regs.VMIXER = 0xFFFFFF;
EffectsStartA = c ? 0xFFFF8 : 0xEFFF8;
EffectsEndA = c ? 0xFFFFF : 0xEFFFF;
FxEnable = false; // Uninitialized it's 0 for both cores. Resetting libs however may set this to 0 or 1.
// These are real PS2 values, mainly constant apart from a few bits: 0x3220EAA4, 0x40505E9C.
// These values mean nothing. They do not reflect the actual address the SPU2 is testing,
// it would seem that reading the IRQA register returns the last written value, not the
// value of the internal register. Rewriting the registers with their current values changes
// whether interrupts fire (they do while uninitialised, but do not when rewritten).
// The exact boot value is unknown and probably unknowable, but it seems to be somewhere
// in the input or output areas, so we're using 0x800.
// F1 2005 is known to rely on an uninitialised IRQA being an address which will be hit.
IRQA = 0x800;
IRQEnable = false; // PS2 confirmed
for (uint v = 0; v < NumVoices; ++v)
{
VoiceGates[v].DryL = -1;
VoiceGates[v].DryR = -1;
VoiceGates[v].WetL = -1;
VoiceGates[v].WetR = -1;
Voices[v].Volume = V_VolumeSlideLR(0, 0); // V_VolumeSlideLR::Max;
Voices[v].ADSR.Counter = 0;
Voices[v].ADSR.Value = 0;
Voices[v].ADSR.Phase = 0;
Voices[v].Pitch = 0x3FFF;
Voices[v].NextA = 0x2801;
Voices[v].StartA = 0x2800;
Voices[v].LoopStartA = 0x2800;
memset(Voices[v].DecodeFifo, 0, sizeof(Voices[v].DecodeFifo));
Voices[v].DecPosRead = 0;
Voices[v].DecPosWrite = 0;
}
DMAICounter = 0;
AdmaInProgress = false;
Regs.STATX = 0x80;
Regs.ENDX = 0xffffff; // PS2 confirmed
RevbSampleBufPos = 0;
memset(RevbDownBuf, 0, sizeof(RevbDownBuf));
memset(RevbUpBuf, 0, sizeof(RevbUpBuf));
}
void V_Voice::Start()
{
if (StartA & 7)
{
fprintf(stderr, " *** Misaligned StartA %05x!\n", StartA);
StartA = (StartA + 0xFFFF8) + 0x8;
}
ADSR.Attack();
LoopMode = 0;
SP = 0;
LoopFlags = 0;
NextA = StartA | 1;
Prev1 = 0;
Prev2 = 0;
SBuffer = nullptr;
DecPosRead = 0;
DecPosWrite = 0;
}
void V_Voice::Stop()
{
ADSR.Value = 0;
ADSR.Phase = V_ADSR::PHASE_STOPPED;
}
__forceinline void CounterUpdate(u32 DMAICounter)
{
if (((psxCounters[6].startCycle + psxCounters[6].deltaCycles) - psxRegs.cycle) > (u32)DMAICounter)
{
psxCounters[6].startCycle = psxRegs.cycle;
psxCounters[6].deltaCycles = DMAICounter;
psxNextDeltaCounter -= (psxRegs.cycle - psxNextStartCounter);
psxNextStartCounter = psxRegs.cycle;
if (psxCounters[6].deltaCycles < psxNextDeltaCounter)
psxNextDeltaCounter = psxCounters[6].deltaCycles;
}
}
__forceinline void CheckDMAProgress(int cid)
{
V_Core& core = Cores[cid];
int adma_cbit = 1 << cid;
if (core.DMAICounter > 0 && (psxRegs.cycle - core.LastClock) > 0)
{
const u32 amt = std::min(psxRegs.cycle - core.LastClock, (u32)core.DMAICounter);
core.DMAICounter -= amt;
core.LastClock = psxRegs.cycle;
if (!core.AdmaInProgress)
{
if (cid == 0)
HW_DMA4_MADR += amt / 2;
if (cid == 1)
HW_DMA7_MADR += amt / 2;
}
if (core.DMAICounter > 0)
{
CounterUpdate(core.DMAICounter);
return;
}
for (int i = 0; i < 2; i++)
{
if (has_to_call_irq_dma[i])
{
//ConLog("* SPU2: Irq Called (%04x) at cycle %d.\n", Spdif.Info, Cycles);
has_to_call_irq_dma[i] = false;
if (!(Spdif.Info & (4 << i)) && Cores[i].IRQEnable)
{
Spdif.Info |= (4 << i);
spu2Irq();
}
}
}
if (((core.AutoDMACtrl & adma_cbit) != adma_cbit) && core.ReadSize)
{
if (core.IsDMARead)
core.FinishDMAread();
else
core.FinishDMAwrite();
}
if (core.DMAICounter <= 0)
{
if (cid == 0)
{
HW_DMA4_MADR = HW_DMA4_TADR;
spu2DMA4Irq();
}
if (cid == 1)
{
HW_DMA7_MADR = HW_DMA7_TADR;
spu2DMA7Irq();
}
}
}
}
static constexpr uint TickInterval = 768;
static constexpr int SanityInterval = 4800;
__forceinline void TimeUpdate(u32 cClocks)
{
u32 dClocks = cClocks - lClocks;
// Sanity Checks:
// It's not totally uncommon for the IOP's clock to jump backwards a cycle or two, and in
// such cases we just want to ignore the TimeUpdate call.
if (dClocks > (u32)-15)
return;
// But if for some reason our clock value seems way off base (typically due to bad dma
// timings from PCSX2), just mix out a little bit, skip the rest, and hope the ship
// "rights" itself later on.
if (dClocks > static_cast<u32>(TickInterval * SanityInterval))
{
if (SPU2::MsgToConsole())
SPU2::ConLog(" * SPU2 > TimeUpdate Sanity Check (Tick Delta: %d) (PS2 Ticks: %d)\n", dClocks / TickInterval, cClocks / TickInterval);
dClocks = TickInterval * SanityInterval;
lClocks = cClocks - dClocks;
}
//Update Mixing Progress
while (dClocks >= TickInterval)
{
for (int i = 0; i < 2; i++)
{
if (has_to_call_irq[i])
{
//ConLog("* SPU2: Irq Called (%04x) at cycle %d.\n", Spdif.Info, Cycles);
has_to_call_irq[i] = false;
if (!(Spdif.Info & (4 << i)) && Cores[i].IRQEnable)
{
Spdif.Info |= (4 << i);
spu2Irq();
}
}
}
dClocks -= TickInterval;
lClocks += TickInterval;
Cycles++;
for(int c = 0; c < 2; c++)
{
if (Cores[c].KeyOff)
{
StopVoices(c, Cores[c].KeyOff);
Cores[c].KeyOff = 0;
}
if (Cores[c].KeyOn)
{
StartVoices(c, Cores[c].KeyOn);
Cores[c].KeyOn = 0;
}
}
spu2Mix();
}
CheckDMAProgress(0);
CheckDMAProgress(1);
}
__forceinline void UpdateSpdifMode()
{
const int OPM = PlayMode;
if (Spdif.Out & 0x4) // use 24/32bit PCM data streaming
{
PlayMode = 8;
if (SPU2::MsgToConsole())
SPU2::ConLog("* SPU2: WARNING: Possibly CDDA mode set!\n");
return;
}
if (Spdif.Out & SPDIF_OUT_BYPASS)
{
PlayMode = 2;
if (!(Spdif.Mode & SPDIF_MODE_BYPASS_BITSTREAM))
PlayMode = 4; //bitstream bypass
}
else
{
PlayMode = 0; //normal processing
if (Spdif.Out & SPDIF_OUT_PCM)
{
PlayMode = 1;
}
}
if (OPM != PlayMode && SPU2::MsgToConsole())
{
SPU2::ConLog("* SPU2: Play Mode Set to %s (%d).\n",
(PlayMode == 0) ? "Normal" : ((PlayMode == 1) ? "PCM Clone" : ((PlayMode == 2) ? "PCM Bypass" : "BitStream Bypass")), PlayMode);
}
}
static u32 map_spu1to2(u32 addr)
{
return addr * 4 + (addr >= 0x200 ? 0xc0000 : 0);
}
static u32 map_spu2to1(u32 addr)
{
// if (addr >= 0x800 && addr < 0xc0000) oh dear
return (addr - (addr >= 0xc0000 ? 0xc0000 : 0)) / 4;
}
void V_Core::WriteRegPS1(u32 mem, u16 value)
{
pxAssume(Index == 0); // Valid on Core 0 only!
bool show = true;
const u32 reg = mem & 0xffff;
if ((reg >= 0x1c00) && (reg < 0x1d80))
{
//voice values
u8 voice = ((reg - 0x1c00) >> 4);
const u8 vval = reg & 0xf;
switch (vval)
{
case 0x0: //VOLL (Volume L)
case 0x2: //VOLR (Volume R)
{
V_VolumeSlide& thisvol = vval == 0 ? Voices[voice].Volume.Left : Voices[voice].Volume.Right;
thisvol.RegSet(value);
//ConLog("voice %x VOL%c write: %x\n", voice, vval == 0 ? 'L' : 'R', value);
break;
}
case 0x4:
Voices[voice].Pitch = value;
//ConLog("voice %x Pitch write: %x\n", voice, Voices[voice].Pitch);
break;
case 0x6:
Voices[voice].StartA = map_spu1to2(value);
//ConLog("voice %x StartA write: %x\n", voice, Voices[voice].StartA);
break;
case 0x8: // ADSR1 (Envelope)
Voices[voice].ADSR.regADSR1 = value;
Voices[voice].ADSR.UpdateCache();
//ConLog("voice %x regADSR1 write: %x\n", voice, Voices[voice].ADSR.regADSR1);
break;
case 0xa: // ADSR2 (Envelope)
Voices[voice].ADSR.regADSR2 = value;
Voices[voice].ADSR.UpdateCache();
//ConLog("voice %x regADSR2 write: %x\n", voice, Voices[voice].ADSR.regADSR2);
break;
case 0xc: // Voice 0..23 ADSR Current Volume
// not commonly set by games
Voices[voice].ADSR.Value = value;
if (SPU2::MsgToConsole())
SPU2::ConLog("voice %x ADSR.Value write: %x\n", voice, Voices[voice].ADSR.Value);
break;
case 0xe:
Voices[voice].LoopStartA = map_spu1to2(value);
//ConLog("voice %x LoopStartA write: %x\n", voice, Voices[voice].LoopStartA);
break;
jNO_DEFAULT;
}
}
else
switch (reg)
{
case 0x1d80: // Mainvolume left
MasterVol.Left.RegSet(value);
break;
case 0x1d82: // Mainvolume right
MasterVol.Right.RegSet(value);
break;
case 0x1d84: // Reverberation depth left
FxVol.Left = SignExtend16(value);
break;
case 0x1d86: // Reverberation depth right
FxVol.Right = SignExtend16(value);
break;
case 0x1d88: // Voice ON (0-15)
SPU2_FastWrite(REG_S_KON, value);
break;
case 0x1d8a: // Voice ON (16-23)
SPU2_FastWrite(REG_S_KON + 2, value);
break;
case 0x1d8c: // Voice OFF (0-15)
SPU2_FastWrite(REG_S_KOFF, value);
break;
case 0x1d8e: // Voice OFF (16-23)
SPU2_FastWrite(REG_S_KOFF + 2, value);
break;
case 0x1d90: // Channel FM (pitch lfo) mode (0-15)
SPU2_FastWrite(REG_S_PMON, value);
if (value != 0 && SPU2::MsgToConsole())
SPU2::ConLog("SPU2 warning: wants to set Pitch Modulation reg1 to %x \n", value);
break;
case 0x1d92: // Channel FM (pitch lfo) mode (16-23)
SPU2_FastWrite(REG_S_PMON + 2, value);
if (value != 0 && SPU2::MsgToConsole())
SPU2::ConLog("SPU2 warning: wants to set Pitch Modulation reg2 to %x \n", value);
break;
case 0x1d94: // Channel Noise mode (0-15)
SPU2_FastWrite(REG_S_NON, value);
if (value != 0 && SPU2::MsgToConsole())
SPU2::ConLog("SPU2 warning: wants to set Channel Noise mode reg1 to %x\n", value);
break;
case 0x1d96: // Channel Noise mode (16-23)
SPU2_FastWrite(REG_S_NON + 2, value);
if (value != 0 && SPU2::MsgToConsole())
SPU2::ConLog("SPU2 warning: wants to set Channel Noise mode reg2 to %x\n", value);
break;
case 0x1d98: // 1F801D98h - Voice 0..23 Reverb mode aka Echo On (EON) (R/W)
//Regs.VMIXEL = value & 0xFFFF;
SPU2_FastWrite(REG_S_VMIXEL, value);
SPU2_FastWrite(REG_S_VMIXER, value);
//ConLog("SPU2 warning: setting reverb mode reg1 to %x \n", Regs.VMIXEL);
break;
case 0x1d9a: // 1F801D98h + 2 - Voice 0..23 Reverb mode aka Echo On (EON) (R/W)
//Regs.VMIXEL = value << 16;
SPU2_FastWrite(REG_S_VMIXEL + 2, value);
SPU2_FastWrite(REG_S_VMIXER + 2, value);
//ConLog("SPU2 warning: setting reverb mode reg2 to %x \n", Regs.VMIXEL);
break;
// this was wrong? // edit: appears so!
//case 0x1d9c:// Channel Reverb mode (0-15)
// SPU2_FastWrite(REG_S_VMIXL,value);
// SPU2_FastWrite(REG_S_VMIXR,value);
//break;
//case 0x1d9e:// Channel Reverb mode (16-23)
// SPU2_FastWrite(REG_S_VMIXL+2,value);
// SPU2_FastWrite(REG_S_VMIXR+2,value);
//break;
case 0x1d9c: // Voice 0..15 ON/OFF (status) (ENDX) (R) // writeable but hw overrides it shortly after
//Regs.ENDX &= 0xff0000;
if (SPU2::MsgToConsole())
SPU2::ConLog("SPU2 warning: wants to set ENDX reg1 to %x \n", value);
break;
case 0x1d9e: // // Voice 15..23 ON/OFF (status) (ENDX) (R) // writeable but hw overrides it shortly after
//Regs.ENDX &= 0xffff;
if (SPU2::MsgToConsole())
SPU2::ConLog("SPU2 warning: wants to set ENDX reg2 to %x \n", value);
break;
case 0x1da2: // Reverb work area start
EffectsStartA = map_spu1to2(value);
break;
case 0x1da4:
IRQA = map_spu1to2(value);
//ConLog("SPU2 Setting IRQA to %x \n", IRQA);
break;
case 0x1da6:
TSA = map_spu1to2(value);
if (SPU2::MsgToConsole())
SPU2::ConLog("SPU2 Setting TSA to %x \n", TSA);
break;
case 0x1da8: // Spu Write to Memory
//ConLog("SPU direct DMA Write. Current TSA = %x\n", TSA);
Cores[0].ActiveTSA = Cores[0].TSA;
if (Cores[0].IRQEnable && (Cores[0].IRQA <= Cores[0].ActiveTSA))
{
SetIrqCall(0);
spu2Irq();
}
DmaWrite(value);
show = false;
break;
case 0x1daa:
SPU2_FastWrite(REG_C_ATTR, value);
break;
case 0x1dac: // 1F801DACh - Sound RAM Data Transfer Control (should be 0004h)
if (SPU2::MsgToConsole())
SPU2::ConLog("SPU Sound RAM Data Transfer Control (should be 4) : value = %x \n", value);
psxSoundDataTransferControl = value;
break;
case 0x1dae: // 1F801DAEh - SPU Status Register (SPUSTAT) (R)
// The SPUSTAT register should be treated read-only (writing is possible in so far that the written
// value can be read-back for a short moment, however, thereafter the hardware is overwriting that value).
//Regs.STATX = value;
break;
case 0x1DB0: // 1F801DB0h 4 CD Volume Left/Right
break; // cd left?
case 0x1DB2:
break; // cd right?
case 0x1DB4: // 1F801DB4h 4 Extern Volume Left / Right
break; // Extern left?
case 0x1DB6:
break; // Extern right?
case 0x1DB8: // 1F801DB8h 4 Current Main Volume Left/Right
break; // Current left?
case 0x1DBA:
break; // Current right?
case 0x1DBC: // 1F801DBCh 4 Unknown? (R/W)
break;
case 0x1DBE:
break;
case 0x1DC0:
Revb.APF1_SIZE = value * 4;
break;
case 0x1DC2:
Revb.APF2_SIZE = value * 4;
break;
case 0x1DC4:
Revb.IIR_VOL = value;
break;
case 0x1DC6:
Revb.COMB1_VOL = value;
break;
case 0x1DC8:
Revb.COMB2_VOL = value;
break;
case 0x1DCA:
Revb.COMB3_VOL = value;
break;
case 0x1DCC:
Revb.COMB4_VOL = value;
break;
case 0x1DCE:
Revb.WALL_VOL = value;
break;
case 0x1DD0:
Revb.APF1_VOL = value;
break;
case 0x1DD2:
Revb.APF2_VOL = value;
break;
case 0x1DD4:
Revb.SAME_L_DST = value * 4;
break;
case 0x1DD6:
Revb.SAME_R_DST = value * 4;
break;
case 0x1DD8:
Revb.COMB1_L_SRC = value * 4;
break;
case 0x1DDA:
Revb.COMB1_R_SRC = value * 4;
break;
case 0x1DDC:
Revb.COMB2_L_SRC = value * 4;
break;
case 0x1DDE:
Revb.COMB2_R_SRC = value * 4;
break;
case 0x1DE0:
Revb.SAME_L_SRC = value * 4;
break;
case 0x1DE2:
Revb.SAME_R_SRC = value * 4;
break;
case 0x1DE4:
Revb.DIFF_L_DST = value * 4;
break;
case 0x1DE6:
Revb.DIFF_R_DST = value * 4;
break;
case 0x1DE8:
Revb.COMB3_L_SRC = value * 4;
break;
case 0x1DEA:
Revb.COMB3_R_SRC = value * 4;
break;
case 0x1DEC:
Revb.COMB4_L_SRC = value * 4;
break;
case 0x1DEE:
Revb.COMB4_R_SRC = value * 4;
break;
case 0x1DF0:
Revb.DIFF_L_SRC = value * 4;
break; // DIFF_R_SRC and DIFF_L_SRC supposedly swapped on SPU2
case 0x1DF2:
Revb.DIFF_R_SRC = value * 4;
break; // but I don't believe it! (games in psxmode sound better unswapped)
case 0x1DF4:
Revb.APF1_L_DST = value * 4;
break;
case 0x1DF6:
Revb.APF1_R_DST = value * 4;
break;
case 0x1DF8:
Revb.APF2_L_DST = value * 4;
break;
case 0x1DFA:
Revb.APF2_R_DST = value * 4;
break;
case 0x1DFC:
Revb.IN_COEF_L = value;
break;
case 0x1DFE:
Revb.IN_COEF_R = value;
break;
}
if (show)
SPU2::FileLog("[%10d] (!) SPU write mem %08x value %04x\n", Cycles, mem, value);
spu2Ru16(mem) = value;
}
u16 V_Core::ReadRegPS1(u32 mem)
{
pxAssume(Index == 0); // Valid on Core 0 only!
bool show = true;
u16 value = spu2Ru16(mem);
const u32 reg = mem & 0xffff;
if ((reg >= 0x1c00) && (reg < 0x1d80))
{
//voice values
const u8 voice = ((reg - 0x1c00) >> 4);
const u8 vval = reg & 0xf;
switch (vval)
{
case 0x0: //VOLL (Volume L)
//value=Voices[voice].VolumeL.Mode;
//value=Voices[voice].VolumeL.Value;
value = Voices[voice].Volume.Left.Reg_VOL;
break;
case 0x2: //VOLR (Volume R)
//value=Voices[voice].VolumeR.Mode;
//value=Voices[voice].VolumeR.Value;
value = Voices[voice].Volume.Right.Reg_VOL;
break;
case 0x4:
value = Voices[voice].Pitch;
//ConLog("voice %d read pitch result = %x\n", voice, value);
break;
case 0x6:
value = map_spu2to1(Voices[voice].StartA);
//ConLog("voice %d read StartA result = %x\n", voice, value);
break;
case 0x8:
value = Voices[voice].ADSR.regADSR1;
break;
case 0xa:
value = Voices[voice].ADSR.regADSR2;
break;
case 0xc: // Voice 0..23 ADSR Current Volume
value = Voices[voice].ADSR.Value;
//if (value != 0) ConLog("voice %d read ADSR.Value result = %x\n", voice, value);
break;
case 0xe:
value = map_spu2to1(Voices[voice].LoopStartA);
//ConLog("voice %d read LoopStartA result = %x\n", voice, value);
break;
jNO_DEFAULT;
}
}
else
switch (reg)
{
case 0x1d80:
value = MasterVol.Left.Value;
break;
case 0x1d82:
value = MasterVol.Right.Value;
break;
case 0x1d84:
value = FxVol.Left;
break;
case 0x1d86:
value = FxVol.Right;
break;
case 0x1d88:
value = 0;
break; // Voice 0..23 Key ON(Start Attack / Decay / Sustain) (W)
case 0x1d8a:
value = 0;
break;
case 0x1d8c:
value = 0;
break; // Voice 0..23 Key OFF (Start Release) (W)
case 0x1d8e:
value = 0;
break;
case 0x1d90:
value = Regs.PMON & 0xFFFF;
break; // Voice 0..23 Channel FM(pitch lfo) mode(R / W)
case 0x1d92:
value = Regs.PMON >> 16;
break;
case 0x1d94:
value = Regs.NON & 0xFFFF;
break; // Voice 0..23 Channel Noise mode (R/W)
case 0x1d96:
value = Regs.NON >> 16;
break;
case 0x1d98:
value = Regs.VMIXEL & 0xFFFF;
break; // Voice 0..23 Channel Reverb mode (R/W)
case 0x1d9a:
value = Regs.VMIXEL >> 16;
break;
/*case 0x1d9c: value = Regs.VMIXL&0xFFFF; break;*/ // this is wrong?
/*case 0x1d9e: value = Regs.VMIXL >> 16; break;*/
case 0x1d9c:
value = Regs.ENDX & 0xFFFF;
break; // Voice 0..23 Channel ON / OFF(status) (R) (ENDX)
case 0x1d9e:
value = Regs.ENDX >> 16;
break;
case 0x1da2:
value = map_spu2to1(EffectsStartA);
break;
case 0x1da4:
value = map_spu2to1(IRQA);
//ConLog("SPU2 IRQA read: 0x1da4 = %x , (IRQA = %x)\n", value, IRQA);
break;
case 0x1da6:
value = map_spu2to1(TSA);
//ConLog("SPU2 TSA read: 0x1da6 = %x , (TSA = %x)\n", value, TSA);
break;
case 0x1da8:
ActiveTSA = TSA;
value = DmaRead();
show = false;
break;
case 0x1daa:
value = Cores[0].Regs.ATTR;
//ConLog("SPU2 ps1 reg psxSPUCNT read return value: %x\n", value);
break;
case 0x1dac: // 1F801DACh - Sound RAM Data Transfer Control (should be 0004h)
value = psxSoundDataTransferControl;
break;
case 0x1dae:
value = Cores[0].Regs.STATX;
//ConLog("SPU2 ps1 reg REG_P_STATX read return value: %x\n", value);
break;
}
if (show)
SPU2::FileLog("[%10d] (!) SPU read mem %08x value %04x\n", Cycles, mem, value);
return value;
}
// Ah the joys of endian-specific code! :D
static __forceinline void SetHiWord(u32& src, u16 value)
{
((u16*)&src)[1] = value;
}
static __forceinline void SetLoWord(u32& src, u16 value)
{
((u16*)&src)[0] = value;
}
template <int CoreIdx, int VoiceIdx, int param>
static void RegWrite_VoiceParams(u16 value)
{
const int core = CoreIdx;
const int voice = VoiceIdx;
V_Voice& thisvoice = Cores[core].Voices[voice];
switch (param)
{
case 0: //VOLL (Volume L)
case 1: //VOLR (Volume R)
{
V_VolumeSlide& thisvol = (param == 0) ? thisvoice.Volume.Left : thisvoice.Volume.Right;
thisvol.RegSet(value);
}
break;
case 2:
thisvoice.Pitch = value;
break;
case 3: // ADSR1 (Envelope)
thisvoice.ADSR.regADSR1 = value;
thisvoice.ADSR.UpdateCache();
break;
case 4: // ADSR2 (Envelope)
thisvoice.ADSR.regADSR2 = value;
thisvoice.ADSR.UpdateCache();
break;
// REG_VP_ENVX, REG_VP_VOLXL and REG_VP_VOLXR are all writable, only ENVX has any effect when written to.
// Colin McRae Rally 2005 triggers case 5 (ADSR), but it doesn't produce issues enabled or disabled.
case 5:
thisvoice.ADSR.Value = value;
break;
case 6:
//thisvoice.Volume.Left.RegSet(value);
break;
case 7:
//thisvoice.Volume.Right.RegSet(value);
break;
jNO_DEFAULT;
}
}
template <int CoreIdx, int VoiceIdx, int address>
static void RegWrite_VoiceAddr(u16 value)
{
const int core = CoreIdx;
const int voice = VoiceIdx;
V_Voice& thisvoice = Cores[core].Voices[voice];
switch (address)
{
case 0: // SSA (Waveform Start Addr) (hiword, 4 bits only)
thisvoice.StartA = ((u32)(value & 0x0F) << 16) | (thisvoice.StartA & 0xFFF8);
if (IsDevBuild)
DebugCores[core].Voices[voice].lastSetStartA = thisvoice.StartA;
break;
case 1: // SSA (loword)
thisvoice.StartA = (thisvoice.StartA & 0x0F0000) | (value & 0xFFF8);
if (IsDevBuild)
DebugCores[core].Voices[voice].lastSetStartA = thisvoice.StartA;
break;
case 2:
thisvoice.LoopMode = 1;
thisvoice.LoopStartA = ((u32)(value & 0x0F) << 16) | (thisvoice.LoopStartA & 0xFFF8);
break;
case 3:
thisvoice.LoopMode = 1;
thisvoice.LoopStartA = (thisvoice.LoopStartA & 0x0F0000) | (value & 0xFFF8);
break;
case 4:
// NAX is confirmed to be writable on hardware (decoder will start decoding at new location).
//
// Example games:
// FlatOut
// Soul Reaver 2
// Wallace And Gromit: Curse Of The Were-Rabbit.
thisvoice.NextA = ((u32)(value & 0x0F) << 16) | (thisvoice.NextA & 0xFFF8) | 1;
break;
case 5:
thisvoice.NextA = (thisvoice.NextA & 0x0F0000) | (value & 0xFFF8) | 1;
break;
}
}
template <int CoreIdx, int cAddr>
static void RegWrite_Core(u16 value)
{
const int omem = cAddr;
const int core = CoreIdx;
V_Core& thiscore = Cores[core];
switch (omem)
{
case REG__1AC:
// ----------------------------------------------------------------------------
// 0x1ac / 0x5ac : direct-write to DMA address : special register (undocumented)
// ----------------------------------------------------------------------------
// On the GS, DMAs are actually pushed through a hardware register. Chances are the
// SPU works the same way, and "technically" *all* DMA data actually passes through
// the HW registers at 0x1ac (core0) and 0x5ac (core1). We handle normal DMAs in
// optimized block copy fashion elsewhere, but some games will write this register
// directly, so handle those here:
// Performance Note: The PS2 Bios uses this extensively right before booting games,
// causing massive slowdown if we don't shortcut it here.
thiscore.ActiveTSA = thiscore.TSA;
for (int i = 0; i < 2; i++)
{
if (Cores[i].IRQEnable && (Cores[i].IRQA == thiscore.ActiveTSA))
{
SetIrqCall(i);
}
}
thiscore.DmaWrite(value);
break;
case REG_C_ATTR:
{
bool irqe = thiscore.IRQEnable;
int bit0 = thiscore.AttrBit0;
bool oldFXenable = thiscore.FxEnable;
u8 oldDmaMode = thiscore.DmaMode;
thiscore.AttrBit0 = (value >> 0) & 0x01; //1 bit
thiscore.DMABits = (value >> 1) & 0x07; //3 bits
thiscore.DmaMode = (value >> 4) & 0x03; //2 bit (not necessary, we get the direction from the iop)
thiscore.IRQEnable = (value >> 6) & 0x01; //1 bit
thiscore.FxEnable = (value >> 7) & 0x01; //1 bit
thiscore.NoiseClk = (value >> 8) & 0x3f; //6 bits
//thiscore.Mute =(value>>14) & 0x01; //1 bit
thiscore.Mute = 0;
//thiscore.CoreEnabled=(value>>15) & 0x01; //1 bit
// no clue
thiscore.Regs.ATTR = value & 0xffff;
if (thiscore.FxEnable && !oldFXenable)
{
if (SPU2::MsgToConsole())
thiscore.AnalyzeReverbPreset();
}
if (!thiscore.DmaMode && !(thiscore.Regs.STATX & 0x400))
thiscore.Regs.STATX &= ~0x80;
else if (!oldDmaMode && thiscore.DmaMode)
thiscore.Regs.STATX |= 0x80;
thiscore.ActiveTSA = thiscore.TSA;
if (value & 0x000E)
{
if (SPU2::MsgToConsole())
SPU2::ConLog("* SPU2: Core %d ATTR unknown bits SET! value=%04x\n", core, value);
}
if (thiscore.AttrBit0 != bit0)
{
if (SPU2::MsgToConsole())
SPU2::ConLog("* SPU2: ATTR bit 0 set to %d\n", thiscore.AttrBit0);
}
if (thiscore.IRQEnable != irqe)
{
//ConLog("* SPU2: Core%d IRQ %s at cycle %d. Current IRQA = %x Current EffectA = %x\n",
// core, ((thiscore.IRQEnable==0)?"disabled":"enabled"), Cycles, thiscore.IRQA, thiscore.EffectsStartA);
if (!thiscore.IRQEnable)
Spdif.Info &= ~(4 << thiscore.Index);
else
if ((thiscore.IRQA & 0xFFF00000) != 0)
DevCon.Warning("SPU2: Core %d IRQA Outside of SPU2 memory, Addr %x", thiscore.Index, thiscore.IRQA);
}
}
break;
case REG_S_PMON:
for (int vc = 1; vc < 16; ++vc)
thiscore.Voices[vc].Modulated = (value >> vc) & 1;
SetLoWord(thiscore.Regs.PMON, value);
break;
case (REG_S_PMON + 2):
for (int vc = 0; vc < 8; ++vc)
thiscore.Voices[vc + 16].Modulated = (value >> vc) & 1;
SetHiWord(thiscore.Regs.PMON, value);
break;
case REG_S_NON:
for (int vc = 0; vc < 16; ++vc)
thiscore.Voices[vc].Noise = (value >> vc) & 1;
SetLoWord(thiscore.Regs.NON, value);
break;
case (REG_S_NON + 2):
for (int vc = 0; vc < 8; ++vc)
thiscore.Voices[vc + 16].Noise = (value >> vc) & 1;
SetHiWord(thiscore.Regs.NON, value);
break;
// Games like to repeatedly write these regs over and over with the same value, hence
// the shortcut that skips the bitloop if the values are equal.
#define vx_SetSomeBits(reg_out, mask_out, hiword) \
{ \
const u32 result = thiscore.Regs.reg_out; \
if (hiword) \
SetHiWord(thiscore.Regs.reg_out, value); \
else \
SetLoWord(thiscore.Regs.reg_out, value); \
if (result == thiscore.Regs.reg_out) \
break; \
\
const uint start_bit = (hiword) ? 16 : 0; \
const uint end_bit = (hiword) ? 24 : 16; \
for (uint vc = start_bit, vx = 1; vc < end_bit; ++vc, vx <<= 1) \
thiscore.VoiceGates[vc].mask_out = (value & vx) ? -1 : 0; \
}
case REG_S_VMIXL:
vx_SetSomeBits(VMIXL, DryL, false);
break;
case (REG_S_VMIXL + 2):
vx_SetSomeBits(VMIXL, DryL, true);
break;
case REG_S_VMIXEL:
vx_SetSomeBits(VMIXEL, WetL, false);
break;
case (REG_S_VMIXEL + 2):
vx_SetSomeBits(VMIXEL, WetL, true);
break;
case REG_S_VMIXR:
vx_SetSomeBits(VMIXR, DryR, false);
break;
case (REG_S_VMIXR + 2):
vx_SetSomeBits(VMIXR, DryR, true);
break;
case REG_S_VMIXER:
vx_SetSomeBits(VMIXER, WetR, false);
break;
case (REG_S_VMIXER + 2):
vx_SetSomeBits(VMIXER, WetR, true);
break;
case REG_P_MMIX:
{
// Each MMIX gate is assigned either 0 or 0xffffffff depending on the status
// of the MMIX bits. I use -1 below as a shorthand for 0xffffffff. :)
const int vx = value & ((core == 0) ? 0xFF0 : 0xFFF);
thiscore.WetGate.ExtR = (vx & 0x001) ? -1 : 0;
thiscore.WetGate.ExtL = (vx & 0x002) ? -1 : 0;
thiscore.DryGate.ExtR = (vx & 0x004) ? -1 : 0;
thiscore.DryGate.ExtL = (vx & 0x008) ? -1 : 0;
thiscore.WetGate.InpR = (vx & 0x010) ? -1 : 0;
thiscore.WetGate.InpL = (vx & 0x020) ? -1 : 0;
thiscore.DryGate.InpR = (vx & 0x040) ? -1 : 0;
thiscore.DryGate.InpL = (vx & 0x080) ? -1 : 0;
thiscore.WetGate.SndR = (vx & 0x100) ? -1 : 0;
thiscore.WetGate.SndL = (vx & 0x200) ? -1 : 0;
thiscore.DryGate.SndR = (vx & 0x400) ? -1 : 0;
thiscore.DryGate.SndL = (vx & 0x800) ? -1 : 0;
thiscore.Regs.MMIX = value;
}
break;
case REG_S_ENDX:
thiscore.Regs.ENDX &= 0xff0000;
break;
case (REG_S_ENDX + 2):
thiscore.Regs.ENDX &= 0xffff;
break;
case REG_S_ADMAS:
if (SPU2::MsgToConsole())
{
SPU2::ConLog("* SPU2: Core %d AutoDMAControl set to %d (at cycle %d)\n", core, value, Cycles);
if (psxmode)
SPU2::ConLog("* SPU2: Writing to REG_S_ADMAS while in PSX mode! value: %x", value);
}
// hack for ps1driver which writes -1 (and never turns the adma off after psxlogo).
// adma isn't available in psx mode either
if (value == 32767)
{
psxmode = true;
//memset(_spu2mem, 0, 0x200000);
Cores[1].FxEnable = 0;
Cores[1].EffectsStartA = 0x7FFF8; // park core1 effect area in inaccessible mem
Cores[1].EffectsEndA = 0x7FFFF;
for (uint v = 0; v < 24; ++v)
{
Cores[1].Voices[v].Volume = V_VolumeSlideLR(0, 0); // V_VolumeSlideLR::Max;
Cores[1].Voices[v].ADSR.Value = 0;
Cores[1].Voices[v].ADSR.Phase = 0;
Cores[1].Voices[v].Pitch = 0x0;
Cores[1].Voices[v].NextA = 0x6FFFF;
Cores[1].Voices[v].StartA = 0x6FFFF;
Cores[1].Voices[v].LoopStartA = 0x6FFFF;
Cores[1].Voices[v].Modulated = 0;
}
return;
}
thiscore.AutoDMACtrl = value;
if (!(value & 0x3) && thiscore.AdmaInProgress)
{
// Kill the current transfer so it doesn't continue
thiscore.AdmaInProgress = 0;
thiscore.InputDataLeft = 0;
thiscore.DMAICounter = 0;
thiscore.InputDataTransferred = 0;
// Not accurate behaviour but shouldn't hurt for now, need to run some tests
// to see why Prince of Persia Warrior Within buzzes when going in to the map
// since it starts an ADMA of music, then kills ADMA and input DMA
// without disabling ADMA read mode or clearing the buffer.
for (int i = 0; i < 0x200; i++)
{
GetMemPtr(0x2000 + (thiscore.Index << 10))[i] = 0;
GetMemPtr(0x2200 + (thiscore.Index << 10))[i] = 0;
}
}
break;
default:
{
const int addr = omem | ((core == 1) ? 0x400 : 0);
*(regtable[addr >> 1]) = value;
}
break;
}
}
template <int CoreIdx, int addr>
static void RegWrite_CoreExt(u16 value)
{
V_Core& thiscore = Cores[CoreIdx];
const int core = CoreIdx;
switch (addr)
{
// Master Volume Address Write!
case REG_P_MVOLL:
case REG_P_MVOLR:
{
V_VolumeSlide& thisvol = (addr == REG_P_MVOLL) ? thiscore.MasterVol.Left : thiscore.MasterVol.Right;
thisvol.RegSet(value);
}
break;
case REG_P_EVOLL:
thiscore.FxVol.Left = SignExtend16(value);
break;
case REG_P_EVOLR:
thiscore.FxVol.Right = SignExtend16(value);
break;
case REG_P_AVOLL:
thiscore.ExtVol.Left = SignExtend16(value);
break;
case REG_P_AVOLR:
thiscore.ExtVol.Right = SignExtend16(value);
break;
case REG_P_BVOLL:
thiscore.InpVol.Left = SignExtend16(value);
break;
case REG_P_BVOLR:
thiscore.InpVol.Right = SignExtend16(value);
break;
// MVOLX has been confirmed to not be allowed to be written to, so cases have been added as a no-op.
// Tokyo Xtreme Racer Zero triggers this code, caused left side volume to be reduced.
case REG_P_MVOLXL:
case REG_P_MVOLXR:
break;
default:
{
const int raddr = addr + ((core == 1) ? 0x28 : 0);
*(regtable[raddr >> 1]) = value;
}
break;
}
}
template <int addr>
static void RegWrite_SPDIF(u16 value)
{
*(regtable[addr >> 1]) = value;
UpdateSpdifMode();
}
template <int addr>
static void RegWrite_Raw(u16 value)
{
*(regtable[addr >> 1]) = value;
}
static void RegWrite_Null(u16 value)
{
}
// --------------------------------------------------------------------------------------
// Macros for tbl_reg_writes
// --------------------------------------------------------------------------------------
#define VoiceParamsSet(core, voice) \
RegWrite_VoiceParams<core, voice, 0>, RegWrite_VoiceParams<core, voice, 1>, \
RegWrite_VoiceParams<core, voice, 2>, RegWrite_VoiceParams<core, voice, 3>, \
RegWrite_VoiceParams<core, voice, 4>, RegWrite_VoiceParams<core, voice, 5>, \
RegWrite_VoiceParams<core, voice, 6>, RegWrite_VoiceParams<core, voice, 7>
#define VoiceParamsCore(core) \
VoiceParamsSet(core, 0), VoiceParamsSet(core, 1), VoiceParamsSet(core, 2), VoiceParamsSet(core, 3), \
VoiceParamsSet(core, 4), VoiceParamsSet(core, 5), VoiceParamsSet(core, 6), VoiceParamsSet(core, 7), \
VoiceParamsSet(core, 8), VoiceParamsSet(core, 9), VoiceParamsSet(core, 10), VoiceParamsSet(core, 11), \
VoiceParamsSet(core, 12), VoiceParamsSet(core, 13), VoiceParamsSet(core, 14), VoiceParamsSet(core, 15), \
VoiceParamsSet(core, 16), VoiceParamsSet(core, 17), VoiceParamsSet(core, 18), VoiceParamsSet(core, 19), \
VoiceParamsSet(core, 20), VoiceParamsSet(core, 21), VoiceParamsSet(core, 22), VoiceParamsSet(core, 23)
#define VoiceAddrSet(core, voice) \
RegWrite_VoiceAddr<core, voice, 0>, RegWrite_VoiceAddr<core, voice, 1>, \
RegWrite_VoiceAddr<core, voice, 2>, RegWrite_VoiceAddr<core, voice, 3>, \
RegWrite_VoiceAddr<core, voice, 4>, RegWrite_VoiceAddr<core, voice, 5>
#define CoreParamsPair(core, omem) \
RegWrite_Core<core, omem>, RegWrite_Core<core, ((omem) + 2)>
#define REGRAW(addr) RegWrite_Raw<addr>
// --------------------------------------------------------------------------------------
// tbl_reg_writes - Register Write Function Invocation LUT
// --------------------------------------------------------------------------------------
typedef void RegWriteHandler(u16 value);
static RegWriteHandler* const tbl_reg_writes[0x401] =
{
VoiceParamsCore(0), // 0x000 -> 0x180
CoreParamsPair(0, REG_S_PMON),
CoreParamsPair(0, REG_S_NON),
CoreParamsPair(0, REG_S_VMIXL),
CoreParamsPair(0, REG_S_VMIXEL),
CoreParamsPair(0, REG_S_VMIXR),
CoreParamsPair(0, REG_S_VMIXER),
RegWrite_Core<0, REG_P_MMIX>,
RegWrite_Core<0, REG_C_ATTR>,
CoreParamsPair(0, REG_A_IRQA),
CoreParamsPair(0, REG_S_KON),
CoreParamsPair(0, REG_S_KOFF),
CoreParamsPair(0, REG_A_TSA),
CoreParamsPair(0, REG__1AC),
RegWrite_Core<0, REG_S_ADMAS>,
REGRAW(0x1b2),
REGRAW(0x1b4), REGRAW(0x1b6),
REGRAW(0x1b8), REGRAW(0x1ba),
REGRAW(0x1bc), REGRAW(0x1be),
// 0x1c0!
VoiceAddrSet(0, 0), VoiceAddrSet(0, 1), VoiceAddrSet(0, 2), VoiceAddrSet(0, 3), VoiceAddrSet(0, 4), VoiceAddrSet(0, 5),
VoiceAddrSet(0, 6), VoiceAddrSet(0, 7), VoiceAddrSet(0, 8), VoiceAddrSet(0, 9), VoiceAddrSet(0, 10), VoiceAddrSet(0, 11),
VoiceAddrSet(0, 12), VoiceAddrSet(0, 13), VoiceAddrSet(0, 14), VoiceAddrSet(0, 15), VoiceAddrSet(0, 16), VoiceAddrSet(0, 17),
VoiceAddrSet(0, 18), VoiceAddrSet(0, 19), VoiceAddrSet(0, 20), VoiceAddrSet(0, 21), VoiceAddrSet(0, 22), VoiceAddrSet(0, 23),
CoreParamsPair(0, REG_A_ESA),
CoreParamsPair(0, R_APF1_SIZE), // 0x02E4 // Feedback Source A
CoreParamsPair(0, R_APF2_SIZE), // 0x02E8 // Feedback Source B
CoreParamsPair(0, R_SAME_L_DST), // 0x02EC
CoreParamsPair(0, R_SAME_R_DST), // 0x02F0
CoreParamsPair(0, R_COMB1_L_SRC), // 0x02F4
CoreParamsPair(0, R_COMB1_R_SRC), // 0x02F8
CoreParamsPair(0, R_COMB2_L_SRC), // 0x02FC
CoreParamsPair(0, R_COMB2_R_SRC), // 0x0300
CoreParamsPair(0, R_SAME_L_SRC), // 0x0304
CoreParamsPair(0, R_SAME_R_SRC), // 0x0308
CoreParamsPair(0, R_DIFF_L_DST), // 0x030C
CoreParamsPair(0, R_DIFF_R_DST), // 0x0310
CoreParamsPair(0, R_COMB3_L_SRC), // 0x0314
CoreParamsPair(0, R_COMB3_R_SRC), // 0x0318
CoreParamsPair(0, R_COMB4_L_SRC), // 0x031C
CoreParamsPair(0, R_COMB4_R_SRC), // 0x0320
CoreParamsPair(0, R_DIFF_L_SRC), // 0x0324
CoreParamsPair(0, R_DIFF_R_SRC), // 0x0328
CoreParamsPair(0, R_APF1_L_DST), // 0x032C
CoreParamsPair(0, R_APF1_R_DST), // 0x0330
CoreParamsPair(0, R_APF2_L_DST), // 0x0334
CoreParamsPair(0, R_APF2_R_DST), // 0x0338
RegWrite_Core<0, REG_A_EEA>, RegWrite_Null,
CoreParamsPair(0, REG_S_ENDX), // 0x0340 // End Point passed flag
RegWrite_Core<0, REG_P_STATX>, // 0x0344 // Status register?
//0x346 here
REGRAW(0x346),
REGRAW(0x348), REGRAW(0x34A), REGRAW(0x34C), REGRAW(0x34E),
REGRAW(0x350), REGRAW(0x352), REGRAW(0x354), REGRAW(0x356),
REGRAW(0x358), REGRAW(0x35A), REGRAW(0x35C), REGRAW(0x35E),
REGRAW(0x360), REGRAW(0x362), REGRAW(0x364), REGRAW(0x366),
REGRAW(0x368), REGRAW(0x36A), REGRAW(0x36C), REGRAW(0x36E),
REGRAW(0x370), REGRAW(0x372), REGRAW(0x374), REGRAW(0x376),
REGRAW(0x378), REGRAW(0x37A), REGRAW(0x37C), REGRAW(0x37E),
REGRAW(0x380), REGRAW(0x382), REGRAW(0x384), REGRAW(0x386),
REGRAW(0x388), REGRAW(0x38A), REGRAW(0x38C), REGRAW(0x38E),
REGRAW(0x390), REGRAW(0x392), REGRAW(0x394), REGRAW(0x396),
REGRAW(0x398), REGRAW(0x39A), REGRAW(0x39C), REGRAW(0x39E),
REGRAW(0x3A0), REGRAW(0x3A2), REGRAW(0x3A4), REGRAW(0x3A6),
REGRAW(0x3A8), REGRAW(0x3AA), REGRAW(0x3AC), REGRAW(0x3AE),
REGRAW(0x3B0), REGRAW(0x3B2), REGRAW(0x3B4), REGRAW(0x3B6),
REGRAW(0x3B8), REGRAW(0x3BA), REGRAW(0x3BC), REGRAW(0x3BE),
REGRAW(0x3C0), REGRAW(0x3C2), REGRAW(0x3C4), REGRAW(0x3C6),
REGRAW(0x3C8), REGRAW(0x3CA), REGRAW(0x3CC), REGRAW(0x3CE),
REGRAW(0x3D0), REGRAW(0x3D2), REGRAW(0x3D4), REGRAW(0x3D6),
REGRAW(0x3D8), REGRAW(0x3DA), REGRAW(0x3DC), REGRAW(0x3DE),
REGRAW(0x3E0), REGRAW(0x3E2), REGRAW(0x3E4), REGRAW(0x3E6),
REGRAW(0x3E8), REGRAW(0x3EA), REGRAW(0x3EC), REGRAW(0x3EE),
REGRAW(0x3F0), REGRAW(0x3F2), REGRAW(0x3F4), REGRAW(0x3F6),
REGRAW(0x3F8), REGRAW(0x3FA), REGRAW(0x3FC), REGRAW(0x3FE),
// AND... we reached 0x400!
// Last verse, same as the first:
VoiceParamsCore(1), // 0x000 -> 0x180
CoreParamsPair(1, REG_S_PMON),
CoreParamsPair(1, REG_S_NON),
CoreParamsPair(1, REG_S_VMIXL),
CoreParamsPair(1, REG_S_VMIXEL),
CoreParamsPair(1, REG_S_VMIXR),
CoreParamsPair(1, REG_S_VMIXER),
RegWrite_Core<1, REG_P_MMIX>,
RegWrite_Core<1, REG_C_ATTR>,
CoreParamsPair(1, REG_A_IRQA),
CoreParamsPair(1, REG_S_KON),
CoreParamsPair(1, REG_S_KOFF),
CoreParamsPair(1, REG_A_TSA),
CoreParamsPair(1, REG__1AC),
RegWrite_Core<1, REG_S_ADMAS>,
REGRAW(0x5b2),
REGRAW(0x5b4), REGRAW(0x5b6),
REGRAW(0x5b8), REGRAW(0x5ba),
REGRAW(0x5bc), REGRAW(0x5be),
// 0x1c0!
VoiceAddrSet(1, 0), VoiceAddrSet(1, 1), VoiceAddrSet(1, 2), VoiceAddrSet(1, 3), VoiceAddrSet(1, 4), VoiceAddrSet(1, 5),
VoiceAddrSet(1, 6), VoiceAddrSet(1, 7), VoiceAddrSet(1, 8), VoiceAddrSet(1, 9), VoiceAddrSet(1, 10), VoiceAddrSet(1, 11),
VoiceAddrSet(1, 12), VoiceAddrSet(1, 13), VoiceAddrSet(1, 14), VoiceAddrSet(1, 15), VoiceAddrSet(1, 16), VoiceAddrSet(1, 17),
VoiceAddrSet(1, 18), VoiceAddrSet(1, 19), VoiceAddrSet(1, 20), VoiceAddrSet(1, 21), VoiceAddrSet(1, 22), VoiceAddrSet(1, 23),
CoreParamsPair(1, REG_A_ESA),
CoreParamsPair(1, R_APF1_SIZE), // 0x02E4 // Feedback Source A
CoreParamsPair(1, R_APF2_SIZE), // 0x02E8 // Feedback Source B
CoreParamsPair(1, R_SAME_L_DST), // 0x02EC
CoreParamsPair(1, R_SAME_R_DST), // 0x02F0
CoreParamsPair(1, R_COMB1_L_SRC), // 0x02F4
CoreParamsPair(1, R_COMB1_R_SRC), // 0x02F8
CoreParamsPair(1, R_COMB2_L_SRC), // 0x02FC
CoreParamsPair(1, R_COMB2_R_SRC), // 0x0300
CoreParamsPair(1, R_SAME_L_SRC), // 0x0304
CoreParamsPair(1, R_SAME_R_SRC), // 0x0308
CoreParamsPair(1, R_DIFF_L_DST), // 0x030C
CoreParamsPair(1, R_DIFF_R_DST), // 0x0310
CoreParamsPair(1, R_COMB3_L_SRC), // 0x0314
CoreParamsPair(1, R_COMB3_R_SRC), // 0x0318
CoreParamsPair(1, R_COMB4_L_SRC), // 0x031C
CoreParamsPair(1, R_COMB4_R_SRC), // 0x0320
CoreParamsPair(1, R_DIFF_R_SRC), // 0x0324
CoreParamsPair(1, R_DIFF_L_SRC), // 0x0328
CoreParamsPair(1, R_APF1_L_DST), // 0x032C
CoreParamsPair(1, R_APF1_R_DST), // 0x0330
CoreParamsPair(1, R_APF2_L_DST), // 0x0334
CoreParamsPair(1, R_APF2_R_DST), // 0x0338
RegWrite_Core<1, REG_A_EEA>, RegWrite_Null,
CoreParamsPair(1, REG_S_ENDX), // 0x0340 // End Point passed flag
RegWrite_Core<1, REG_P_STATX>, // 0x0344 // Status register?
REGRAW(0x746),
REGRAW(0x748), REGRAW(0x74A), REGRAW(0x74C), REGRAW(0x74E),
REGRAW(0x750), REGRAW(0x752), REGRAW(0x754), REGRAW(0x756),
REGRAW(0x758), REGRAW(0x75A), REGRAW(0x75C), REGRAW(0x75E),
// ------ -------
RegWrite_CoreExt<0, REG_P_MVOLL>, // 0x0760 // Master Volume Left
RegWrite_CoreExt<0, REG_P_MVOLR>, // 0x0762 // Master Volume Right
RegWrite_CoreExt<0, REG_P_EVOLL>, // 0x0764 // Effect Volume Left
RegWrite_CoreExt<0, REG_P_EVOLR>, // 0x0766 // Effect Volume Right
RegWrite_CoreExt<0, REG_P_AVOLL>, // 0x0768 // Core External Input Volume Left (Only Core 1)
RegWrite_CoreExt<0, REG_P_AVOLR>, // 0x076A // Core External Input Volume Right (Only Core 1)
RegWrite_CoreExt<0, REG_P_BVOLL>, // 0x076C // Sound Data Volume Left
RegWrite_CoreExt<0, REG_P_BVOLR>, // 0x076E // Sound Data Volume Right
RegWrite_CoreExt<0, REG_P_MVOLXL>, // 0x0770 // Current Master Volume Left
RegWrite_CoreExt<0, REG_P_MVOLXR>, // 0x0772 // Current Master Volume Right
RegWrite_CoreExt<0, R_IIR_VOL>, // 0x0774 //IIR alpha (% used)
RegWrite_CoreExt<0, R_COMB1_VOL>, // 0x0776
RegWrite_CoreExt<0, R_COMB2_VOL>, // 0x0778
RegWrite_CoreExt<0, R_COMB3_VOL>, // 0x077A
RegWrite_CoreExt<0, R_COMB4_VOL>, // 0x077C
RegWrite_CoreExt<0, R_WALL_VOL>, // 0x077E
RegWrite_CoreExt<0, R_APF1_VOL>, // 0x0780 //feedback alpha (% used)
RegWrite_CoreExt<0, R_APF2_VOL>, // 0x0782 //feedback
RegWrite_CoreExt<0, R_IN_COEF_L>, // 0x0784
RegWrite_CoreExt<0, R_IN_COEF_R>, // 0x0786
// ------ -------
RegWrite_CoreExt<1, REG_P_MVOLL>, // 0x0788 // Master Volume Left
RegWrite_CoreExt<1, REG_P_MVOLR>, // 0x078A // Master Volume Right
RegWrite_CoreExt<1, REG_P_EVOLL>, // 0x0764 // Effect Volume Left
RegWrite_CoreExt<1, REG_P_EVOLR>, // 0x0766 // Effect Volume Right
RegWrite_CoreExt<1, REG_P_AVOLL>, // 0x0768 // Core External Input Volume Left (Only Core 1)
RegWrite_CoreExt<1, REG_P_AVOLR>, // 0x076A // Core External Input Volume Right (Only Core 1)
RegWrite_CoreExt<1, REG_P_BVOLL>, // 0x076C // Sound Data Volume Left
RegWrite_CoreExt<1, REG_P_BVOLR>, // 0x076E // Sound Data Volume Right
RegWrite_CoreExt<1, REG_P_MVOLXL>, // 0x0770 // Current Master Volume Left
RegWrite_CoreExt<1, REG_P_MVOLXR>, // 0x0772 // Current Master Volume Right
RegWrite_CoreExt<1, R_IIR_VOL>, // 0x0774 //IIR alpha (% used)
RegWrite_CoreExt<1, R_COMB1_VOL>, // 0x0776
RegWrite_CoreExt<1, R_COMB2_VOL>, // 0x0778
RegWrite_CoreExt<1, R_COMB3_VOL>, // 0x077A
RegWrite_CoreExt<1, R_COMB4_VOL>, // 0x077C
RegWrite_CoreExt<1, R_WALL_VOL>, // 0x077E
RegWrite_CoreExt<1, R_APF1_VOL>, // 0x0780 //feedback alpha (% used)
RegWrite_CoreExt<1, R_APF2_VOL>, // 0x0782 //feedback
RegWrite_CoreExt<1, R_IN_COEF_L>, // 0x0784
RegWrite_CoreExt<1, R_IN_COEF_R>, // 0x0786
REGRAW(0x7B0), REGRAW(0x7B2), REGRAW(0x7B4), REGRAW(0x7B6),
REGRAW(0x7B8), REGRAW(0x7BA), REGRAW(0x7BC), REGRAW(0x7BE),
// SPDIF interface
RegWrite_SPDIF<SPDIF_OUT>, // 0x07C0 // SPDIF Out: OFF/'PCM'/Bitstream/Bypass
RegWrite_SPDIF<SPDIF_IRQINFO>, // 0x07C2
REGRAW(0x7C4),
RegWrite_SPDIF<SPDIF_MODE>, // 0x07C6
RegWrite_SPDIF<SPDIF_MEDIA>, // 0x07C8 // SPDIF Media: 'CD'/DVD
REGRAW(0x7CA),
RegWrite_SPDIF<SPDIF_PROTECT>, // 0x07CC // SPDIF Copy Protection
REGRAW(0x7CE),
REGRAW(0x7D0), REGRAW(0x7D2), REGRAW(0x7D4), REGRAW(0x7D6),
REGRAW(0x7D8), REGRAW(0x7DA), REGRAW(0x7DC), REGRAW(0x7DE),
REGRAW(0x7E0), REGRAW(0x7E2), REGRAW(0x7E4), REGRAW(0x7E6),
REGRAW(0x7E8), REGRAW(0x7EA), REGRAW(0x7EC), REGRAW(0x7EE),
REGRAW(0x7F0), REGRAW(0x7F2), REGRAW(0x7F4), REGRAW(0x7F6),
REGRAW(0x7F8), REGRAW(0x7FA), REGRAW(0x7FC), REGRAW(0x7FE),
nullptr // should be at 0x400! (we assert check it on startup)
};
void SPU2_FastWrite(u32 rmem, u16 value)
{
tbl_reg_writes[(rmem & 0x7ff) / 2](value);
}
void StartVoices(int core, u32 value)
{
Cores[core].Regs.ENDX &= ~value;
for (u8 vc = 0; vc < V_Core::NumVoices; vc++)
{
if (!((value >> vc) & 1))
continue;
Cores[core].Voices[vc].Start();
if (IsDevBuild)
{
if (SPU2::MsgKeyOnOff())
{
V_Voice& thisvc(Cores[core].Voices[vc]);
SPU2::ConLog("* SPU2: KeyOn: C%dV%02d: SSA: %8x; M: %s%s%s%s; H: %04x; P: %04x V: %04x/%04x; ADSR: %04x%04x\n",
core, vc, thisvc.StartA,
(Cores[core].VoiceGates[vc].DryL) ? "+" : "-", (Cores[core].VoiceGates[vc].DryR) ? "+" : "-",
(Cores[core].VoiceGates[vc].WetL) ? "+" : "-", (Cores[core].VoiceGates[vc].WetR) ? "+" : "-",
*(u16*)GetMemPtr(thisvc.StartA),
thisvc.Pitch,
thisvc.Volume.Left.Value, thisvc.Volume.Right.Value,
thisvc.ADSR.regADSR1, thisvc.ADSR.regADSR2);
}
}
}
}
void StopVoices(int core, u32 value)
{
for (u8 vc = 0; vc < V_Core::NumVoices; vc++)
{
if (!((value >> vc) & 1))
continue;
Cores[core].Voices[vc].ADSR.Release();
if (SPU2::MsgKeyOnOff())
SPU2::ConLog("* SPU2: KeyOff: Core %d; Voice %d.\n", core, vc);
}
}