/* Copyright 2016-2021 Arisotura This file is part of melonDS. melonDS 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 3 of the License, or (at your option) any later version. melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/. */ #ifndef SPU_H #define SPU_H #include "Savestate.h" namespace SPU { bool Init(); void DeInit(); void Reset(); void Stop(); void DoSavestate(Savestate* file); void SetBias(u16 bias); void Mix(u32 dummy); void TrimOutput(); void DrainOutput(); void InitOutput(); int GetOutputSize(); void Sync(bool wait); int ReadOutput(s16* data, int samples); void TransferOutput(); u8 Read8(u32 addr); u16 Read16(u32 addr); u32 Read32(u32 addr); void Write8(u32 addr, u8 val); void Write16(u32 addr, u16 val); void Write32(u32 addr, u32 val); class Channel { public: Channel(u32 num); ~Channel(); void Reset(); void DoSavestate(Savestate* file); u32 Num; u32 Cnt; u32 SrcAddr; u16 TimerReload; u32 LoopPos; u32 Length; u8 Volume; u8 VolumeShift; u8 Pan; bool KeyOn; u32 Timer; s32 Pos; s16 CurSample; u16 NoiseVal; s32 ADPCMVal; s32 ADPCMIndex; s32 ADPCMValLoop; s32 ADPCMIndexLoop; u8 ADPCMCurByte; u32 FIFO[8]; u32 FIFOReadPos; u32 FIFOWritePos; u32 FIFOReadOffset; u32 FIFOLevel; void FIFO_BufferData(); template T FIFO_ReadData(); void SetCnt(u32 val) { u32 oldcnt = Cnt; Cnt = val & 0xFF7F837F; Volume = Cnt & 0x7F; if (Volume == 127) Volume++; const u8 volshift[4] = {4, 3, 2, 0}; VolumeShift = volshift[(Cnt >> 8) & 0x3]; Pan = (Cnt >> 16) & 0x7F; if (Pan == 127) Pan++; if ((val & (1<<31)) && !(oldcnt & (1<<31))) { KeyOn = true; } } void SetSrcAddr(u32 val) { SrcAddr = val & 0x07FFFFFC; } void SetTimerReload(u32 val) { TimerReload = val & 0xFFFF; } void SetLoopPos(u32 val) { LoopPos = (val & 0xFFFF) << 2; } void SetLength(u32 val) { Length = (val & 0x001FFFFF) << 2; } void Start(); void NextSample_PCM8(); void NextSample_PCM16(); void NextSample_ADPCM(); void NextSample_PSG(); void NextSample_Noise(); template s32 Run(); s32 DoRun() { switch ((Cnt >> 29) & 0x3) { case 0: return Run<0>(); break; case 1: return Run<1>(); break; case 2: return Run<2>(); break; case 3: if (Num >= 14) { return Run<4>(); break; } else if (Num >= 8) { return Run<3>(); break; } [[fallthrough]]; default: return 0; } } void PanOutput(s32 in, s32& left, s32& right); private: u32 (*BusRead32)(u32 addr); }; class CaptureUnit { public: CaptureUnit(u32 num); ~CaptureUnit(); void Reset(); void DoSavestate(Savestate* file); u32 Num; u8 Cnt; u32 DstAddr; u16 TimerReload; u32 Length; u32 Timer; s32 Pos; u32 FIFO[4]; u32 FIFOReadPos; u32 FIFOWritePos; u32 FIFOWriteOffset; u32 FIFOLevel; void FIFO_FlushData(); template void FIFO_WriteData(T val); void SetCnt(u8 val) { if ((val & 0x80) && !(Cnt & 0x80)) Start(); val &= 0x8F; if (!(val & 0x80)) val &= ~0x01; Cnt = val; } void SetDstAddr(u32 val) { DstAddr = val & 0x07FFFFFC; } void SetTimerReload(u32 val) { TimerReload = val & 0xFFFF; } void SetLength(u32 val) { Length = val << 2; if (Length == 0) Length = 4; } void Start() { Timer = TimerReload; Pos = 0; FIFOReadPos = 0; FIFOWritePos = 0; FIFOWriteOffset = 0; FIFOLevel = 0; } void Run(s32 sample); private: void (*BusWrite32)(u32 addr, u32 val); }; } #endif // SPU_H