2017-04-04 13:32:50 +00:00
|
|
|
/*
|
2019-01-22 14:58:29 +00:00
|
|
|
Copyright 2016-2019 Arisotura
|
2017-04-04 13:32:50 +00:00
|
|
|
|
|
|
|
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/.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include "NDS.h"
|
|
|
|
#include "SPU.h"
|
|
|
|
|
|
|
|
|
2017-05-09 01:54:37 +00:00
|
|
|
// SPU TODO
|
2019-05-14 13:10:58 +00:00
|
|
|
// * capture addition modes, overflow bugs
|
|
|
|
// * channel hold
|
|
|
|
// * 'length less than 4' glitch
|
2017-05-09 01:54:37 +00:00
|
|
|
|
2017-04-04 13:32:50 +00:00
|
|
|
namespace SPU
|
|
|
|
{
|
|
|
|
|
2017-04-06 23:06:10 +00:00
|
|
|
const s8 ADPCMIndexTable[8] = {-1, -1, -1, -1, 2, 4, 6, 8};
|
|
|
|
|
|
|
|
const u16 ADPCMTable[89] =
|
|
|
|
{
|
|
|
|
0x0007, 0x0008, 0x0009, 0x000A, 0x000B, 0x000C, 0x000D, 0x000E,
|
|
|
|
0x0010, 0x0011, 0x0013, 0x0015, 0x0017, 0x0019, 0x001C, 0x001F,
|
|
|
|
0x0022, 0x0025, 0x0029, 0x002D, 0x0032, 0x0037, 0x003C, 0x0042,
|
|
|
|
0x0049, 0x0050, 0x0058, 0x0061, 0x006B, 0x0076, 0x0082, 0x008F,
|
|
|
|
0x009D, 0x00AD, 0x00BE, 0x00D1, 0x00E6, 0x00FD, 0x0117, 0x0133,
|
|
|
|
0x0151, 0x0173, 0x0198, 0x01C1, 0x01EE, 0x0220, 0x0256, 0x0292,
|
|
|
|
0x02D4, 0x031C, 0x036C, 0x03C3, 0x0424, 0x048E, 0x0502, 0x0583,
|
|
|
|
0x0610, 0x06AB, 0x0756, 0x0812, 0x08E0, 0x09C3, 0x0ABD, 0x0BD0,
|
|
|
|
0x0CFF, 0x0E4C, 0x0FBA, 0x114C, 0x1307, 0x14EE, 0x1706, 0x1954,
|
|
|
|
0x1BDC, 0x1EA5, 0x21B6, 0x2515, 0x28CA, 0x2CDF, 0x315B, 0x364B,
|
|
|
|
0x3BB9, 0x41B2, 0x4844, 0x4F7E, 0x5771, 0x602F, 0x69CE, 0x7462,
|
|
|
|
0x7FFF
|
|
|
|
};
|
|
|
|
|
|
|
|
const s16 PSGTable[8][8] =
|
|
|
|
{
|
|
|
|
{-0x7FFF, -0x7FFF, -0x7FFF, -0x7FFF, -0x7FFF, -0x7FFF, -0x7FFF, 0x7FFF},
|
|
|
|
{-0x7FFF, -0x7FFF, -0x7FFF, -0x7FFF, -0x7FFF, -0x7FFF, 0x7FFF, 0x7FFF},
|
|
|
|
{-0x7FFF, -0x7FFF, -0x7FFF, -0x7FFF, -0x7FFF, 0x7FFF, 0x7FFF, 0x7FFF},
|
|
|
|
{-0x7FFF, -0x7FFF, -0x7FFF, -0x7FFF, 0x7FFF, 0x7FFF, 0x7FFF, 0x7FFF},
|
|
|
|
{-0x7FFF, -0x7FFF, -0x7FFF, 0x7FFF, 0x7FFF, 0x7FFF, 0x7FFF, 0x7FFF},
|
|
|
|
{-0x7FFF, -0x7FFF, 0x7FFF, 0x7FFF, 0x7FFF, 0x7FFF, 0x7FFF, 0x7FFF},
|
|
|
|
{-0x7FFF, 0x7FFF, 0x7FFF, 0x7FFF, 0x7FFF, 0x7FFF, 0x7FFF, 0x7FFF},
|
2017-04-07 15:37:49 +00:00
|
|
|
{-0x7FFF, -0x7FFF, -0x7FFF, -0x7FFF, -0x7FFF, -0x7FFF, -0x7FFF, -0x7FFF}
|
2017-04-06 23:06:10 +00:00
|
|
|
};
|
|
|
|
|
2017-11-11 17:46:42 +00:00
|
|
|
const u32 kSamplesPerRun = 1;
|
|
|
|
|
2019-08-29 13:55:18 +00:00
|
|
|
const u32 OutputBufferSize = 2*1024;
|
2017-04-06 17:44:34 +00:00
|
|
|
s16 OutputBuffer[2 * OutputBufferSize];
|
2019-08-29 13:55:18 +00:00
|
|
|
volatile u32 OutputReadOffset;
|
|
|
|
volatile u32 OutputWriteOffset;
|
2017-04-06 17:44:34 +00:00
|
|
|
|
|
|
|
|
|
|
|
u16 Cnt;
|
|
|
|
u8 MasterVolume;
|
|
|
|
u16 Bias;
|
|
|
|
|
|
|
|
Channel* Channels[16];
|
2017-04-08 20:59:27 +00:00
|
|
|
CaptureUnit* Capture[2];
|
2017-04-04 13:32:50 +00:00
|
|
|
|
|
|
|
|
|
|
|
bool Init()
|
|
|
|
{
|
2017-04-06 17:44:34 +00:00
|
|
|
for (int i = 0; i < 16; i++)
|
|
|
|
Channels[i] = new Channel(i);
|
2017-04-04 13:32:50 +00:00
|
|
|
|
2017-04-08 20:59:27 +00:00
|
|
|
Capture[0] = new CaptureUnit(0);
|
|
|
|
Capture[1] = new CaptureUnit(1);
|
|
|
|
|
2017-04-04 13:32:50 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeInit()
|
|
|
|
{
|
2017-04-06 17:44:34 +00:00
|
|
|
for (int i = 0; i < 16; i++)
|
|
|
|
delete Channels[i];
|
2017-04-08 20:59:27 +00:00
|
|
|
|
|
|
|
delete Capture[0];
|
|
|
|
delete Capture[1];
|
2017-04-04 13:32:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Reset()
|
2017-04-06 17:44:34 +00:00
|
|
|
{
|
|
|
|
memset(OutputBuffer, 0, 2*OutputBufferSize*2);
|
|
|
|
OutputReadOffset = 0;
|
2019-08-18 09:31:03 +00:00
|
|
|
OutputWriteOffset = 0;//OutputBufferSize;
|
2017-04-06 17:44:34 +00:00
|
|
|
|
|
|
|
Cnt = 0;
|
|
|
|
MasterVolume = 0;
|
|
|
|
Bias = 0;
|
|
|
|
|
|
|
|
for (int i = 0; i < 16; i++)
|
|
|
|
Channels[i]->Reset();
|
|
|
|
|
2017-04-08 20:59:27 +00:00
|
|
|
Capture[0]->Reset();
|
|
|
|
Capture[1]->Reset();
|
|
|
|
|
2017-11-11 17:46:42 +00:00
|
|
|
NDS::ScheduleEvent(NDS::Event_SPU, true, 1024*kSamplesPerRun, Mix, kSamplesPerRun);
|
2017-04-06 17:44:34 +00:00
|
|
|
}
|
|
|
|
|
2017-09-21 01:59:12 +00:00
|
|
|
void Stop()
|
|
|
|
{
|
|
|
|
memset(OutputBuffer, 0, 2*OutputBufferSize*2);
|
|
|
|
}
|
|
|
|
|
2018-10-18 00:45:38 +00:00
|
|
|
void DoSavestate(Savestate* file)
|
|
|
|
{
|
|
|
|
file->Section("SPU.");
|
|
|
|
|
|
|
|
file->Var16(&Cnt);
|
|
|
|
file->Var8(&MasterVolume);
|
|
|
|
file->Var16(&Bias);
|
|
|
|
|
|
|
|
for (int i = 0; i < 16; i++)
|
|
|
|
Channels[i]->DoSavestate(file);
|
|
|
|
|
|
|
|
Capture[0]->DoSavestate(file);
|
|
|
|
Capture[1]->DoSavestate(file);
|
|
|
|
}
|
|
|
|
|
2017-04-06 17:44:34 +00:00
|
|
|
|
2017-04-07 15:37:49 +00:00
|
|
|
void SetBias(u16 bias)
|
|
|
|
{
|
|
|
|
Bias = bias;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-04-06 17:44:34 +00:00
|
|
|
Channel::Channel(u32 num)
|
|
|
|
{
|
2017-04-06 20:38:46 +00:00
|
|
|
Num = num;
|
2017-04-06 17:44:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Channel::~Channel()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void Channel::Reset()
|
|
|
|
{
|
|
|
|
SetCnt(0);
|
|
|
|
SrcAddr = 0;
|
|
|
|
TimerReload = 0;
|
|
|
|
LoopPos = 0;
|
|
|
|
Length = 0;
|
2017-04-09 01:35:32 +00:00
|
|
|
|
|
|
|
Timer = 0;
|
2017-11-11 17:46:42 +00:00
|
|
|
|
|
|
|
Pos = 0;
|
|
|
|
FIFOReadPos = 0;
|
|
|
|
FIFOWritePos = 0;
|
|
|
|
FIFOReadOffset = 0;
|
|
|
|
FIFOLevel = 0;
|
|
|
|
}
|
|
|
|
|
2018-10-18 00:45:38 +00:00
|
|
|
void Channel::DoSavestate(Savestate* file)
|
|
|
|
{
|
|
|
|
file->Var32(&Cnt);
|
|
|
|
file->Var32(&SrcAddr);
|
|
|
|
file->Var16(&TimerReload);
|
|
|
|
file->Var32(&LoopPos);
|
|
|
|
file->Var32(&Length);
|
|
|
|
|
|
|
|
file->Var8(&Volume);
|
|
|
|
file->Var8(&VolumeShift);
|
|
|
|
file->Var8(&Pan);
|
|
|
|
|
|
|
|
file->Var32(&Timer);
|
|
|
|
file->Var32((u32*)&Pos);
|
|
|
|
file->Var16((u16*)&CurSample);
|
|
|
|
file->Var16(&NoiseVal);
|
|
|
|
|
|
|
|
file->Var32((u32*)&ADPCMVal);
|
|
|
|
file->Var32((u32*)&ADPCMIndex);
|
|
|
|
file->Var32((u32*)&ADPCMValLoop);
|
|
|
|
file->Var32((u32*)&ADPCMIndexLoop);
|
|
|
|
file->Var8(&ADPCMCurByte);
|
|
|
|
|
|
|
|
file->Var32(&FIFOReadPos);
|
|
|
|
file->Var32(&FIFOWritePos);
|
|
|
|
file->Var32(&FIFOReadOffset);
|
|
|
|
file->Var32(&FIFOLevel);
|
|
|
|
file->VarArray(FIFO, 8*4);
|
|
|
|
}
|
|
|
|
|
2017-11-11 17:46:42 +00:00
|
|
|
void Channel::FIFO_BufferData()
|
|
|
|
{
|
|
|
|
u32 totallen = LoopPos + Length;
|
|
|
|
|
|
|
|
if (FIFOReadOffset >= totallen)
|
|
|
|
{
|
|
|
|
u32 repeatmode = (Cnt >> 27) & 0x3;
|
2018-12-13 23:14:06 +00:00
|
|
|
if (repeatmode & 1) FIFOReadOffset = LoopPos;
|
|
|
|
else if (repeatmode & 2) return; // one-shot sound, we're done
|
2017-11-11 17:46:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
u32 burstlen = 16;
|
|
|
|
if ((FIFOReadOffset + 16) > totallen)
|
|
|
|
burstlen = totallen - FIFOReadOffset;
|
|
|
|
|
|
|
|
for (u32 i = 0; i < burstlen; i += 4)
|
|
|
|
{
|
2018-12-08 21:33:41 +00:00
|
|
|
FIFO[FIFOWritePos] = NDS::ARM7Read32(SrcAddr + FIFOReadOffset);
|
2017-11-11 17:46:42 +00:00
|
|
|
FIFOReadOffset += 4;
|
|
|
|
FIFOWritePos++;
|
|
|
|
FIFOWritePos &= 0x7;
|
|
|
|
}
|
|
|
|
|
|
|
|
FIFOLevel += burstlen;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
T Channel::FIFO_ReadData()
|
|
|
|
{
|
|
|
|
T ret = *(T*)&((u8*)FIFO)[FIFOReadPos];
|
|
|
|
|
|
|
|
FIFOReadPos += sizeof(T);
|
|
|
|
FIFOReadPos &= 0x1F;
|
|
|
|
FIFOLevel -= sizeof(T);
|
|
|
|
|
|
|
|
if (FIFOLevel <= 16)
|
|
|
|
FIFO_BufferData();
|
|
|
|
|
|
|
|
return ret;
|
2017-04-06 17:44:34 +00:00
|
|
|
}
|
|
|
|
|
2017-04-06 20:38:46 +00:00
|
|
|
void Channel::Start()
|
|
|
|
{
|
|
|
|
Timer = TimerReload;
|
|
|
|
|
2017-04-09 01:35:32 +00:00
|
|
|
if (((Cnt >> 29) & 0x3) == 3)
|
2017-04-06 23:06:10 +00:00
|
|
|
Pos = -1;
|
|
|
|
else
|
|
|
|
Pos = -3;
|
|
|
|
|
|
|
|
NoiseVal = 0x7FFF;
|
|
|
|
CurSample = 0;
|
2017-11-11 17:46:42 +00:00
|
|
|
|
|
|
|
FIFOReadPos = 0;
|
|
|
|
FIFOWritePos = 0;
|
|
|
|
FIFOReadOffset = 0;
|
|
|
|
FIFOLevel = 0;
|
|
|
|
|
2018-12-13 23:14:06 +00:00
|
|
|
// when starting a channel, buffer data
|
2017-11-11 17:46:42 +00:00
|
|
|
if (((Cnt >> 29) & 0x3) != 3)
|
|
|
|
{
|
|
|
|
FIFO_BufferData();
|
2018-12-14 21:03:34 +00:00
|
|
|
FIFO_BufferData();
|
2017-11-11 17:46:42 +00:00
|
|
|
}
|
2017-04-06 20:38:46 +00:00
|
|
|
}
|
|
|
|
|
2017-04-06 23:06:10 +00:00
|
|
|
void Channel::NextSample_PCM8()
|
2017-04-06 20:38:46 +00:00
|
|
|
{
|
2017-04-06 23:06:10 +00:00
|
|
|
Pos++;
|
|
|
|
if (Pos < 0) return;
|
|
|
|
if (Pos >= (LoopPos + Length))
|
2017-04-06 20:38:46 +00:00
|
|
|
{
|
2017-04-06 23:06:10 +00:00
|
|
|
u32 repeat = (Cnt >> 27) & 0x3;
|
2018-12-13 23:14:06 +00:00
|
|
|
if (repeat & 1)
|
|
|
|
{
|
|
|
|
Pos = LoopPos;
|
|
|
|
}
|
|
|
|
else if (repeat & 2)
|
2017-04-06 23:06:10 +00:00
|
|
|
{
|
|
|
|
CurSample = 0;
|
|
|
|
Cnt &= ~(1<<31);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-11 17:46:42 +00:00
|
|
|
s8 val = FIFO_ReadData<s8>();
|
2017-04-06 23:06:10 +00:00
|
|
|
CurSample = val << 8;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Channel::NextSample_PCM16()
|
|
|
|
{
|
|
|
|
Pos++;
|
|
|
|
if (Pos < 0) return;
|
|
|
|
if ((Pos<<1) >= (LoopPos + Length))
|
|
|
|
{
|
|
|
|
u32 repeat = (Cnt >> 27) & 0x3;
|
2018-12-13 23:14:06 +00:00
|
|
|
if (repeat & 1)
|
|
|
|
{
|
|
|
|
Pos = LoopPos>>1;
|
|
|
|
}
|
|
|
|
else if (repeat & 2)
|
2017-04-06 23:06:10 +00:00
|
|
|
{
|
|
|
|
CurSample = 0;
|
|
|
|
Cnt &= ~(1<<31);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2017-04-06 20:38:46 +00:00
|
|
|
|
2017-11-11 17:46:42 +00:00
|
|
|
s16 val = FIFO_ReadData<s16>();
|
2017-04-06 23:06:10 +00:00
|
|
|
CurSample = val;
|
2017-04-06 20:38:46 +00:00
|
|
|
}
|
|
|
|
|
2017-04-06 23:06:10 +00:00
|
|
|
void Channel::NextSample_ADPCM()
|
|
|
|
{
|
|
|
|
Pos++;
|
|
|
|
if (Pos < 8)
|
|
|
|
{
|
|
|
|
if (Pos == 0)
|
|
|
|
{
|
|
|
|
// setup ADPCM
|
2017-11-11 17:46:42 +00:00
|
|
|
u32 header = FIFO_ReadData<u32>();
|
2017-04-06 23:06:10 +00:00
|
|
|
ADPCMVal = header & 0xFFFF;
|
|
|
|
ADPCMIndex = (header >> 16) & 0x7F;
|
|
|
|
if (ADPCMIndex > 88) ADPCMIndex = 88;
|
|
|
|
|
|
|
|
ADPCMValLoop = ADPCMVal;
|
|
|
|
ADPCMIndexLoop = ADPCMIndex;
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((Pos>>1) >= (LoopPos + Length))
|
|
|
|
{
|
|
|
|
u32 repeat = (Cnt >> 27) & 0x3;
|
2018-12-13 23:14:06 +00:00
|
|
|
if (repeat & 1)
|
2017-04-06 23:06:10 +00:00
|
|
|
{
|
|
|
|
Pos = LoopPos<<1;
|
|
|
|
ADPCMVal = ADPCMValLoop;
|
|
|
|
ADPCMIndex = ADPCMIndexLoop;
|
2017-11-11 17:46:42 +00:00
|
|
|
ADPCMCurByte = FIFO_ReadData<u8>();
|
2017-04-06 23:06:10 +00:00
|
|
|
}
|
2018-12-13 23:14:06 +00:00
|
|
|
else if (repeat & 2)
|
|
|
|
{
|
|
|
|
CurSample = 0;
|
|
|
|
Cnt &= ~(1<<31);
|
|
|
|
return;
|
|
|
|
}
|
2017-04-06 23:06:10 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!(Pos & 0x1))
|
2017-11-11 17:46:42 +00:00
|
|
|
ADPCMCurByte = FIFO_ReadData<u8>();
|
2017-04-06 23:06:10 +00:00
|
|
|
else
|
|
|
|
ADPCMCurByte >>= 4;
|
|
|
|
|
|
|
|
u16 val = ADPCMTable[ADPCMIndex];
|
|
|
|
u16 diff = val >> 3;
|
|
|
|
if (ADPCMCurByte & 0x1) diff += (val >> 2);
|
|
|
|
if (ADPCMCurByte & 0x2) diff += (val >> 1);
|
|
|
|
if (ADPCMCurByte & 0x4) diff += val;
|
|
|
|
|
|
|
|
if (ADPCMCurByte & 0x8)
|
|
|
|
{
|
|
|
|
ADPCMVal -= diff;
|
|
|
|
if (ADPCMVal < -0x7FFF) ADPCMVal = -0x7FFF;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ADPCMVal += diff;
|
|
|
|
if (ADPCMVal > 0x7FFF) ADPCMVal = 0x7FFF;
|
|
|
|
}
|
|
|
|
|
|
|
|
ADPCMIndex += ADPCMIndexTable[ADPCMCurByte & 0x7];
|
|
|
|
if (ADPCMIndex < 0) ADPCMIndex = 0;
|
|
|
|
else if (ADPCMIndex > 88) ADPCMIndex = 88;
|
|
|
|
|
|
|
|
if (Pos == (LoopPos<<1))
|
|
|
|
{
|
|
|
|
ADPCMValLoop = ADPCMVal;
|
|
|
|
ADPCMIndexLoop = ADPCMIndex;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CurSample = ADPCMVal;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Channel::NextSample_PSG()
|
|
|
|
{
|
|
|
|
Pos++;
|
|
|
|
CurSample = PSGTable[(Cnt >> 24) & 0x7][Pos & 0x7];
|
|
|
|
}
|
|
|
|
|
|
|
|
void Channel::NextSample_Noise()
|
|
|
|
{
|
|
|
|
if (NoiseVal & 0x1)
|
|
|
|
{
|
|
|
|
NoiseVal = (NoiseVal >> 1) ^ 0x6000;
|
|
|
|
CurSample = -0x7FFF;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
NoiseVal >>= 1;
|
|
|
|
CurSample = 0x7FFF;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template<u32 type>
|
2017-04-06 20:38:46 +00:00
|
|
|
void Channel::Run(s32* buf, u32 samples)
|
|
|
|
{
|
2017-11-11 17:46:42 +00:00
|
|
|
if (!(Cnt & (1<<31))) return;
|
|
|
|
|
2017-04-06 20:38:46 +00:00
|
|
|
for (u32 s = 0; s < samples; s++)
|
|
|
|
{
|
|
|
|
Timer += 512; // 1 sample = 512 cycles at 16MHz
|
2017-04-06 23:06:10 +00:00
|
|
|
|
2017-04-06 20:38:46 +00:00
|
|
|
while (Timer >> 16)
|
|
|
|
{
|
|
|
|
Timer = TimerReload + (Timer - 0x10000);
|
2017-04-06 23:06:10 +00:00
|
|
|
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case 0: NextSample_PCM8(); break;
|
|
|
|
case 1: NextSample_PCM16(); break;
|
|
|
|
case 2: NextSample_ADPCM(); break;
|
|
|
|
case 3: NextSample_PSG(); break;
|
|
|
|
case 4: NextSample_Noise(); break;
|
|
|
|
}
|
2017-04-06 20:38:46 +00:00
|
|
|
}
|
|
|
|
|
2017-04-08 20:59:27 +00:00
|
|
|
s32 val = (s32)CurSample;
|
|
|
|
val <<= VolumeShift;
|
|
|
|
val *= Volume;
|
|
|
|
buf[s] = val;
|
|
|
|
|
|
|
|
if (!(Cnt & (1<<31))) break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-11 17:46:42 +00:00
|
|
|
void Channel::PanOutput(s32* inbuf, u32 samples, s32* leftbuf, s32* rightbuf)
|
|
|
|
{
|
|
|
|
for (u32 s = 0; s < samples; s++)
|
|
|
|
{
|
|
|
|
s32 val = (s32)inbuf[s];
|
|
|
|
|
|
|
|
s32 l = ((s64)val * (128-Pan)) >> 10;
|
|
|
|
s32 r = ((s64)val * Pan) >> 10;
|
|
|
|
|
|
|
|
leftbuf[s] += l;
|
|
|
|
rightbuf[s] += r;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-08 20:59:27 +00:00
|
|
|
|
|
|
|
CaptureUnit::CaptureUnit(u32 num)
|
|
|
|
{
|
|
|
|
Num = num;
|
|
|
|
}
|
|
|
|
|
|
|
|
CaptureUnit::~CaptureUnit()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void CaptureUnit::Reset()
|
|
|
|
{
|
|
|
|
SetCnt(0);
|
|
|
|
DstAddr = 0;
|
|
|
|
TimerReload = 0;
|
|
|
|
Length = 0;
|
2017-04-09 01:35:32 +00:00
|
|
|
|
|
|
|
Timer = 0;
|
2017-11-11 17:46:42 +00:00
|
|
|
|
|
|
|
Pos = 0;
|
|
|
|
FIFOReadPos = 0;
|
|
|
|
FIFOWritePos = 0;
|
|
|
|
FIFOWriteOffset = 0;
|
|
|
|
FIFOLevel = 0;
|
|
|
|
}
|
|
|
|
|
2018-10-18 00:45:38 +00:00
|
|
|
void CaptureUnit::DoSavestate(Savestate* file)
|
|
|
|
{
|
|
|
|
file->Var8(&Cnt);
|
|
|
|
file->Var32(&DstAddr);
|
|
|
|
file->Var16(&TimerReload);
|
|
|
|
file->Var32(&Length);
|
|
|
|
|
|
|
|
file->Var32(&Timer);
|
|
|
|
file->Var32((u32*)&Pos);
|
|
|
|
|
|
|
|
file->Var32(&FIFOReadPos);
|
|
|
|
file->Var32(&FIFOWritePos);
|
|
|
|
file->Var32(&FIFOWriteOffset);
|
|
|
|
file->Var32(&FIFOLevel);
|
|
|
|
file->VarArray(FIFO, 4*4);
|
|
|
|
}
|
|
|
|
|
2017-11-11 17:46:42 +00:00
|
|
|
void CaptureUnit::FIFO_FlushData()
|
|
|
|
{
|
|
|
|
for (u32 i = 0; i < 4; i++)
|
|
|
|
{
|
|
|
|
NDS::ARM7Write32(DstAddr + FIFOWriteOffset, FIFO[FIFOReadPos]);
|
|
|
|
|
|
|
|
FIFOReadPos++;
|
|
|
|
FIFOReadPos &= 0x3;
|
|
|
|
FIFOLevel -= 4;
|
|
|
|
|
|
|
|
FIFOWriteOffset += 4;
|
|
|
|
if (FIFOWriteOffset >= Length)
|
|
|
|
{
|
|
|
|
FIFOWriteOffset = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
void CaptureUnit::FIFO_WriteData(T val)
|
|
|
|
{
|
|
|
|
*(T*)&((u8*)FIFO)[FIFOWritePos] = val;
|
|
|
|
|
|
|
|
FIFOWritePos += sizeof(T);
|
|
|
|
FIFOWritePos &= 0xF;
|
|
|
|
FIFOLevel += sizeof(T);
|
|
|
|
|
|
|
|
if (FIFOLevel >= 16)
|
|
|
|
FIFO_FlushData();
|
2017-04-08 20:59:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CaptureUnit::Run(s32 sample)
|
|
|
|
{
|
|
|
|
Timer += 512;
|
|
|
|
|
|
|
|
if (Cnt & 0x08)
|
|
|
|
{
|
|
|
|
while (Timer >> 16)
|
|
|
|
{
|
|
|
|
Timer = TimerReload + (Timer - 0x10000);
|
|
|
|
|
2017-11-11 17:46:42 +00:00
|
|
|
FIFO_WriteData<s8>((s8)(sample >> 8));
|
2017-04-08 20:59:27 +00:00
|
|
|
Pos++;
|
|
|
|
if (Pos >= Length)
|
|
|
|
{
|
2017-11-11 17:46:42 +00:00
|
|
|
if (FIFOLevel >= 4)
|
|
|
|
FIFO_FlushData();
|
|
|
|
|
2017-04-08 20:59:27 +00:00
|
|
|
if (Cnt & 0x04)
|
|
|
|
{
|
|
|
|
Cnt &= 0x7F;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
Pos = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
while (Timer >> 16)
|
|
|
|
{
|
|
|
|
Timer = TimerReload + (Timer - 0x10000);
|
|
|
|
|
2017-11-11 17:46:42 +00:00
|
|
|
FIFO_WriteData<s16>((s16)sample);
|
2017-04-08 20:59:27 +00:00
|
|
|
Pos += 2;
|
|
|
|
if (Pos >= Length)
|
|
|
|
{
|
2017-11-11 17:46:42 +00:00
|
|
|
if (FIFOLevel >= 4)
|
|
|
|
FIFO_FlushData();
|
|
|
|
|
2017-04-08 20:59:27 +00:00
|
|
|
if (Cnt & 0x04)
|
|
|
|
{
|
|
|
|
Cnt &= 0x7F;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
Pos = 0;
|
|
|
|
}
|
|
|
|
}
|
2017-04-06 20:38:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-29 13:55:18 +00:00
|
|
|
|
2017-04-06 17:44:34 +00:00
|
|
|
void Mix(u32 samples)
|
2017-04-04 13:32:50 +00:00
|
|
|
{
|
2017-04-06 20:38:46 +00:00
|
|
|
s32 channelbuf[32];
|
2017-04-08 20:59:27 +00:00
|
|
|
s32 leftbuf[32], rightbuf[32];
|
2017-11-11 17:46:42 +00:00
|
|
|
s32 ch0buf[32], ch1buf[32], ch2buf[32], ch3buf[32];
|
2017-04-08 20:59:27 +00:00
|
|
|
s32 leftoutput[32], rightoutput[32];
|
2017-04-06 20:38:46 +00:00
|
|
|
|
|
|
|
for (u32 s = 0; s < samples; s++)
|
2017-04-06 23:06:10 +00:00
|
|
|
{
|
2017-04-08 20:59:27 +00:00
|
|
|
leftbuf[s] = 0; rightbuf[s] = 0;
|
|
|
|
leftoutput[s] = 0; rightoutput[s] = 0;
|
2017-04-06 23:06:10 +00:00
|
|
|
}
|
2017-04-06 20:38:46 +00:00
|
|
|
|
2017-04-08 20:59:27 +00:00
|
|
|
if (Cnt & (1<<15))
|
2017-04-06 20:38:46 +00:00
|
|
|
{
|
2017-11-11 17:46:42 +00:00
|
|
|
Channels[0]->DoRun(ch0buf, samples);
|
|
|
|
Channels[1]->DoRun(ch1buf, samples);
|
|
|
|
Channels[2]->DoRun(ch2buf, samples);
|
|
|
|
Channels[3]->DoRun(ch3buf, samples);
|
|
|
|
|
|
|
|
// TODO: addition from capture registers
|
|
|
|
Channels[0]->PanOutput(ch0buf, samples, leftbuf, rightbuf);
|
|
|
|
Channels[2]->PanOutput(ch2buf, samples, leftbuf, rightbuf);
|
|
|
|
|
|
|
|
if (!(Cnt & (1<<12))) Channels[1]->PanOutput(ch1buf, samples, leftbuf, rightbuf);
|
|
|
|
if (!(Cnt & (1<<13))) Channels[3]->PanOutput(ch3buf, samples, leftbuf, rightbuf);
|
2017-04-08 20:59:27 +00:00
|
|
|
|
2017-11-11 17:46:42 +00:00
|
|
|
for (int i = 4; i < 16; i++)
|
2017-04-08 20:59:27 +00:00
|
|
|
{
|
|
|
|
Channel* chan = Channels[i];
|
|
|
|
|
|
|
|
chan->DoRun(channelbuf, samples);
|
2017-11-11 17:46:42 +00:00
|
|
|
chan->PanOutput(channelbuf, samples, leftbuf, rightbuf);
|
2017-04-06 23:06:10 +00:00
|
|
|
}
|
2017-04-06 20:38:46 +00:00
|
|
|
|
2017-04-08 20:59:27 +00:00
|
|
|
// sound capture
|
|
|
|
// TODO: other sound capture sources, along with their bugs
|
|
|
|
|
|
|
|
if (Capture[0]->Cnt & (1<<7))
|
2017-04-06 20:38:46 +00:00
|
|
|
{
|
2017-04-08 20:59:27 +00:00
|
|
|
for (u32 s = 0; s < samples; s++)
|
|
|
|
{
|
|
|
|
s32 val = leftbuf[s];
|
2017-04-06 23:06:10 +00:00
|
|
|
|
2017-04-08 20:59:27 +00:00
|
|
|
val >>= 8;
|
|
|
|
if (val < -0x8000) val = -0x8000;
|
|
|
|
else if (val > 0x7FFF) val = 0x7FFF;
|
2017-04-06 23:06:10 +00:00
|
|
|
|
2017-04-08 20:59:27 +00:00
|
|
|
Capture[0]->Run(val);
|
2019-08-29 13:55:18 +00:00
|
|
|
if (!(Capture[0]->Cnt & (1<<7))) break;
|
2017-04-08 20:59:27 +00:00
|
|
|
}
|
|
|
|
}
|
2017-04-06 23:06:10 +00:00
|
|
|
|
2017-04-08 20:59:27 +00:00
|
|
|
if (Capture[1]->Cnt & (1<<7))
|
|
|
|
{
|
|
|
|
for (u32 s = 0; s < samples; s++)
|
|
|
|
{
|
|
|
|
s32 val = rightbuf[s];
|
|
|
|
|
|
|
|
val >>= 8;
|
|
|
|
if (val < -0x8000) val = -0x8000;
|
|
|
|
else if (val > 0x7FFF) val = 0x7FFF;
|
|
|
|
|
|
|
|
Capture[1]->Run(val);
|
2019-08-29 13:55:18 +00:00
|
|
|
if (!(Capture[1]->Cnt & (1<<7))) break;
|
2017-04-08 20:59:27 +00:00
|
|
|
}
|
2017-04-06 20:38:46 +00:00
|
|
|
}
|
|
|
|
|
2017-04-08 20:59:27 +00:00
|
|
|
// final output
|
|
|
|
|
|
|
|
switch (Cnt & 0x0300)
|
|
|
|
{
|
|
|
|
case 0x0000: // left mixer
|
|
|
|
{
|
|
|
|
for (u32 s = 0; s < samples; s++)
|
|
|
|
leftoutput[s] = leftbuf[s];
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 0x0100: // channel 1
|
|
|
|
{
|
|
|
|
s32 pan = 128 - Channels[1]->Pan;
|
|
|
|
for (u32 s = 0; s < samples; s++)
|
|
|
|
leftoutput[s] = ((s64)ch1buf[s] * pan) >> 10;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 0x0200: // channel 3
|
|
|
|
{
|
|
|
|
s32 pan = 128 - Channels[3]->Pan;
|
|
|
|
for (u32 s = 0; s < samples; s++)
|
|
|
|
leftoutput[s] = ((s64)ch3buf[s] * pan) >> 10;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 0x0300: // channel 1+3
|
|
|
|
{
|
|
|
|
s32 pan1 = 128 - Channels[1]->Pan;
|
|
|
|
s32 pan3 = 128 - Channels[3]->Pan;
|
|
|
|
for (u32 s = 0; s < samples; s++)
|
|
|
|
leftoutput[s] = (((s64)ch1buf[s] * pan1) >> 10) + (((s64)ch3buf[s] * pan3) >> 10);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (Cnt & 0x0C00)
|
|
|
|
{
|
|
|
|
case 0x0000: // right mixer
|
|
|
|
{
|
|
|
|
for (u32 s = 0; s < samples; s++)
|
|
|
|
rightoutput[s] = rightbuf[s];
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 0x0400: // channel 1
|
|
|
|
{
|
|
|
|
s32 pan = Channels[1]->Pan;
|
|
|
|
for (u32 s = 0; s < samples; s++)
|
|
|
|
rightoutput[s] = ((s64)ch1buf[s] * pan) >> 10;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 0x0800: // channel 3
|
|
|
|
{
|
|
|
|
s32 pan = Channels[3]->Pan;
|
|
|
|
for (u32 s = 0; s < samples; s++)
|
|
|
|
rightoutput[s] = ((s64)ch3buf[s] * pan) >> 10;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 0x0C00: // channel 1+3
|
|
|
|
{
|
|
|
|
s32 pan1 = Channels[1]->Pan;
|
|
|
|
s32 pan3 = Channels[3]->Pan;
|
|
|
|
for (u32 s = 0; s < samples; s++)
|
|
|
|
rightoutput[s] = (((s64)ch1buf[s] * pan1) >> 10) + (((s64)ch3buf[s] * pan3) >> 10);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2017-04-06 17:44:34 +00:00
|
|
|
|
2017-04-06 20:38:46 +00:00
|
|
|
for (u32 s = 0; s < samples; s++)
|
|
|
|
{
|
2017-04-08 20:59:27 +00:00
|
|
|
s32 l = leftoutput[s];
|
|
|
|
s32 r = rightoutput[s];
|
2017-04-06 20:38:46 +00:00
|
|
|
|
2017-04-06 23:06:10 +00:00
|
|
|
l = ((s64)l * MasterVolume) >> 7;
|
|
|
|
r = ((s64)r * MasterVolume) >> 7;
|
|
|
|
|
2017-04-08 20:59:27 +00:00
|
|
|
l >>= 8;
|
2017-04-07 15:37:49 +00:00
|
|
|
if (l < -0x8000) l = -0x8000;
|
|
|
|
else if (l > 0x7FFF) l = 0x7FFF;
|
2017-04-08 20:59:27 +00:00
|
|
|
r >>= 8;
|
2017-04-07 15:37:49 +00:00
|
|
|
if (r < -0x8000) r = -0x8000;
|
|
|
|
else if (r > 0x7FFF) r = 0x7FFF;
|
|
|
|
|
2017-04-08 20:59:27 +00:00
|
|
|
OutputBuffer[OutputWriteOffset ] = l >> 1;
|
|
|
|
OutputBuffer[OutputWriteOffset + 1] = r >> 1;
|
2017-04-06 20:38:46 +00:00
|
|
|
OutputWriteOffset += 2;
|
|
|
|
OutputWriteOffset &= ((2*OutputBufferSize)-1);
|
2019-08-18 09:31:03 +00:00
|
|
|
if (OutputWriteOffset == OutputReadOffset) printf("!! SOUND FIFO OVERFLOW %d\n", OutputWriteOffset>>1);
|
2017-04-06 20:38:46 +00:00
|
|
|
}
|
|
|
|
|
2017-11-11 17:46:42 +00:00
|
|
|
NDS::ScheduleEvent(NDS::Event_SPU, true, 1024*kSamplesPerRun, Mix, kSamplesPerRun);
|
2017-04-06 17:44:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-08-18 09:31:03 +00:00
|
|
|
int GetOutputSize()
|
|
|
|
{
|
2019-08-29 13:55:18 +00:00
|
|
|
int ret;
|
|
|
|
if (OutputWriteOffset >= OutputReadOffset)
|
|
|
|
ret = OutputWriteOffset - OutputReadOffset;
|
|
|
|
else
|
|
|
|
ret = (OutputBufferSize*2) - OutputReadOffset + OutputWriteOffset;
|
|
|
|
|
|
|
|
ret >>= 1;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Sync(bool wait)
|
|
|
|
{
|
|
|
|
// sync to audio output in case the core is running too fast
|
|
|
|
// * wait=true: wait until enough audio data has been played
|
|
|
|
// * wait=false: merely skip some audio data to avoid a FIFO overflow
|
|
|
|
|
|
|
|
const int halflimit = (OutputBufferSize / 2);
|
|
|
|
|
|
|
|
if (wait)
|
|
|
|
{
|
|
|
|
// TODO: less CPU-intensive wait?
|
|
|
|
while (GetOutputSize() > halflimit);
|
|
|
|
}
|
|
|
|
else if (GetOutputSize() > halflimit)
|
|
|
|
{
|
|
|
|
int readpos = OutputWriteOffset - (halflimit*2);
|
|
|
|
if (readpos < 0) readpos += (OutputBufferSize*2);
|
|
|
|
|
|
|
|
OutputReadOffset = readpos;
|
|
|
|
}
|
2019-08-18 09:31:03 +00:00
|
|
|
}
|
|
|
|
|
2017-12-11 17:41:05 +00:00
|
|
|
int ReadOutput(s16* data, int samples)
|
2017-04-06 23:06:10 +00:00
|
|
|
{
|
2017-12-11 17:41:05 +00:00
|
|
|
if (OutputReadOffset == OutputWriteOffset)
|
|
|
|
return 0;
|
|
|
|
|
2017-04-06 17:44:34 +00:00
|
|
|
for (int i = 0; i < samples; i++)
|
|
|
|
{
|
|
|
|
*data++ = OutputBuffer[OutputReadOffset];
|
|
|
|
*data++ = OutputBuffer[OutputReadOffset + 1];
|
|
|
|
|
2017-12-11 17:41:05 +00:00
|
|
|
//if (OutputReadOffset != OutputWriteOffset)
|
2017-04-06 23:06:10 +00:00
|
|
|
{
|
2017-04-06 17:44:34 +00:00
|
|
|
OutputReadOffset += 2;
|
|
|
|
OutputReadOffset &= ((2*OutputBufferSize)-1);
|
|
|
|
}
|
2017-12-11 17:41:05 +00:00
|
|
|
if (OutputReadOffset == OutputWriteOffset)
|
|
|
|
return i+1;
|
2017-04-06 23:06:10 +00:00
|
|
|
}
|
2017-12-11 17:41:05 +00:00
|
|
|
|
|
|
|
return samples;
|
2017-04-06 17:44:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
u8 Read8(u32 addr)
|
|
|
|
{
|
|
|
|
if (addr < 0x04000500)
|
|
|
|
{
|
|
|
|
Channel* chan = Channels[(addr >> 4) & 0xF];
|
|
|
|
|
|
|
|
switch (addr & 0xF)
|
|
|
|
{
|
|
|
|
case 0x0: return chan->Cnt & 0xFF;
|
|
|
|
case 0x1: return (chan->Cnt >> 8) & 0xFF;
|
|
|
|
case 0x2: return (chan->Cnt >> 16) & 0xFF;
|
|
|
|
case 0x3: return chan->Cnt >> 24;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
switch (addr)
|
|
|
|
{
|
|
|
|
case 0x04000500: return Cnt & 0x7F;
|
|
|
|
case 0x04000501: return Cnt >> 8;
|
2017-04-08 20:59:27 +00:00
|
|
|
|
|
|
|
case 0x04000508: return Capture[0]->Cnt;
|
|
|
|
case 0x04000509: return Capture[1]->Cnt;
|
2017-04-06 17:44:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-08 20:59:27 +00:00
|
|
|
printf("unknown SPU read8 %08X\n", addr);
|
2017-04-06 17:44:34 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
u16 Read16(u32 addr)
|
|
|
|
{
|
|
|
|
if (addr < 0x04000500)
|
|
|
|
{
|
|
|
|
Channel* chan = Channels[(addr >> 4) & 0xF];
|
|
|
|
|
|
|
|
switch (addr & 0xF)
|
|
|
|
{
|
|
|
|
case 0x0: return chan->Cnt & 0xFFFF;
|
|
|
|
case 0x2: return chan->Cnt >> 16;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
switch (addr)
|
|
|
|
{
|
|
|
|
case 0x04000500: return Cnt;
|
|
|
|
case 0x04000504: return Bias;
|
2017-04-08 20:59:27 +00:00
|
|
|
|
|
|
|
case 0x04000508: return Capture[0]->Cnt | (Capture[1]->Cnt << 8);
|
2017-04-06 17:44:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("unknown SPU read16 %08X\n", addr);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
u32 Read32(u32 addr)
|
|
|
|
{
|
|
|
|
if (addr < 0x04000500)
|
|
|
|
{
|
|
|
|
Channel* chan = Channels[(addr >> 4) & 0xF];
|
|
|
|
|
|
|
|
switch (addr & 0xF)
|
|
|
|
{
|
|
|
|
case 0x0: return chan->Cnt;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
switch (addr)
|
|
|
|
{
|
|
|
|
case 0x04000500: return Cnt;
|
|
|
|
case 0x04000504: return Bias;
|
2017-04-08 20:59:27 +00:00
|
|
|
|
|
|
|
case 0x04000508: return Capture[0]->Cnt | (Capture[1]->Cnt << 8);
|
|
|
|
|
|
|
|
case 0x04000510: return Capture[0]->DstAddr;
|
|
|
|
case 0x04000518: return Capture[1]->DstAddr;
|
2017-04-06 17:44:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("unknown SPU read32 %08X\n", addr);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Write8(u32 addr, u8 val)
|
|
|
|
{
|
|
|
|
if (addr < 0x04000500)
|
|
|
|
{
|
|
|
|
Channel* chan = Channels[(addr >> 4) & 0xF];
|
|
|
|
|
|
|
|
switch (addr & 0xF)
|
|
|
|
{
|
|
|
|
case 0x0: chan->SetCnt((chan->Cnt & 0xFFFFFF00) | val); return;
|
|
|
|
case 0x1: chan->SetCnt((chan->Cnt & 0xFFFF00FF) | (val << 8)); return;
|
|
|
|
case 0x2: chan->SetCnt((chan->Cnt & 0xFF00FFFF) | (val << 16)); return;
|
|
|
|
case 0x3: chan->SetCnt((chan->Cnt & 0x00FFFFFF) | (val << 24)); return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
switch (addr)
|
|
|
|
{
|
|
|
|
case 0x04000500:
|
|
|
|
Cnt = (Cnt & 0xBF00) | (val & 0x7F);
|
|
|
|
MasterVolume = Cnt & 0x7F;
|
|
|
|
if (MasterVolume == 127) MasterVolume++;
|
|
|
|
return;
|
|
|
|
case 0x04000501:
|
|
|
|
Cnt = (Cnt & 0x007F) | ((val & 0xBF) << 8);
|
|
|
|
return;
|
2017-04-08 20:59:27 +00:00
|
|
|
|
|
|
|
case 0x04000508:
|
|
|
|
Capture[0]->SetCnt(val);
|
|
|
|
if (val & 0x03) printf("!! UNSUPPORTED SPU CAPTURE MODE %02X\n", val);
|
|
|
|
return;
|
|
|
|
case 0x04000509:
|
|
|
|
Capture[1]->SetCnt(val);
|
|
|
|
if (val & 0x03) printf("!! UNSUPPORTED SPU CAPTURE MODE %02X\n", val);
|
|
|
|
return;
|
2017-04-06 17:44:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("unknown SPU write8 %08X %02X\n", addr, val);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Write16(u32 addr, u16 val)
|
|
|
|
{
|
|
|
|
if (addr < 0x04000500)
|
|
|
|
{
|
|
|
|
Channel* chan = Channels[(addr >> 4) & 0xF];
|
|
|
|
|
|
|
|
switch (addr & 0xF)
|
|
|
|
{
|
|
|
|
case 0x0: chan->SetCnt((chan->Cnt & 0xFFFF0000) | val); return;
|
|
|
|
case 0x2: chan->SetCnt((chan->Cnt & 0x0000FFFF) | (val << 16)); return;
|
2017-04-08 20:59:27 +00:00
|
|
|
case 0x8:
|
|
|
|
chan->SetTimerReload(val);
|
|
|
|
if ((addr & 0xF0) == 0x10) Capture[0]->SetTimerReload(val);
|
|
|
|
else if ((addr & 0xF0) == 0x30) Capture[1]->SetTimerReload(val);
|
|
|
|
return;
|
2017-04-06 17:44:34 +00:00
|
|
|
case 0xA: chan->SetLoopPos(val); return;
|
2017-04-08 20:59:27 +00:00
|
|
|
|
2019-08-18 09:31:03 +00:00
|
|
|
case 0xC: chan->SetLength(((chan->Length >> 2) & 0xFFFF0000) | val); return;
|
|
|
|
case 0xE: chan->SetLength(((chan->Length >> 2) & 0x0000FFFF) | (val << 16)); return;
|
2017-04-06 17:44:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
switch (addr)
|
|
|
|
{
|
|
|
|
case 0x04000500:
|
|
|
|
Cnt = val & 0xBF7F;
|
|
|
|
MasterVolume = Cnt & 0x7F;
|
|
|
|
if (MasterVolume == 127) MasterVolume++;
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 0x04000504:
|
|
|
|
Bias = val & 0x3FF;
|
|
|
|
return;
|
2017-04-08 20:59:27 +00:00
|
|
|
|
|
|
|
case 0x04000508:
|
|
|
|
Capture[0]->SetCnt(val & 0xFF);
|
|
|
|
Capture[1]->SetCnt(val >> 8);
|
|
|
|
if (val & 0x0303) printf("!! UNSUPPORTED SPU CAPTURE MODE %04X\n", val);
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 0x04000514: Capture[0]->SetLength(val); return;
|
|
|
|
case 0x0400051C: Capture[1]->SetLength(val); return;
|
2017-04-06 17:44:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("unknown SPU write16 %08X %04X\n", addr, val);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Write32(u32 addr, u32 val)
|
|
|
|
{
|
|
|
|
if (addr < 0x04000500)
|
|
|
|
{
|
|
|
|
Channel* chan = Channels[(addr >> 4) & 0xF];
|
|
|
|
|
|
|
|
switch (addr & 0xF)
|
|
|
|
{
|
|
|
|
case 0x0: chan->SetCnt(val); return;
|
|
|
|
case 0x4: chan->SetSrcAddr(val); return;
|
|
|
|
case 0x8:
|
|
|
|
chan->SetLoopPos(val >> 16);
|
2017-04-08 20:59:27 +00:00
|
|
|
val &= 0xFFFF;
|
|
|
|
chan->SetTimerReload(val);
|
|
|
|
if ((addr & 0xF0) == 0x10) Capture[0]->SetTimerReload(val);
|
|
|
|
else if ((addr & 0xF0) == 0x30) Capture[1]->SetTimerReload(val);
|
2017-04-06 17:44:34 +00:00
|
|
|
return;
|
|
|
|
case 0xC: chan->SetLength(val); return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
switch (addr)
|
|
|
|
{
|
|
|
|
case 0x04000500:
|
|
|
|
Cnt = val & 0xBF7F;
|
|
|
|
MasterVolume = Cnt & 0x7F;
|
|
|
|
if (MasterVolume == 127) MasterVolume++;
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 0x04000504:
|
|
|
|
Bias = val & 0x3FF;
|
|
|
|
return;
|
2017-04-08 20:59:27 +00:00
|
|
|
|
|
|
|
case 0x04000508:
|
|
|
|
Capture[0]->SetCnt(val & 0xFF);
|
|
|
|
Capture[1]->SetCnt(val >> 8);
|
|
|
|
if (val & 0x0303) printf("!! UNSUPPORTED SPU CAPTURE MODE %04X\n", val);
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 0x04000510: Capture[0]->SetDstAddr(val); return;
|
|
|
|
case 0x04000514: Capture[0]->SetLength(val & 0xFFFF); return;
|
|
|
|
case 0x04000518: Capture[1]->SetDstAddr(val); return;
|
|
|
|
case 0x0400051C: Capture[1]->SetLength(val & 0xFFFF); return;
|
2017-04-06 17:44:34 +00:00
|
|
|
}
|
|
|
|
}
|
2017-04-04 13:32:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|