mirror of
https://github.com/libretro/Mesen.git
synced 2025-01-22 17:04:42 +00:00
257 lines
7.2 KiB
C++
257 lines
7.2 KiB
C++
#include "stdafx.h"
|
|
#include "APU.h"
|
|
#include "CPU.h"
|
|
#include "SquareChannel.h"
|
|
#include "TriangleChannel.h"
|
|
#include "NoiseChannel.h"
|
|
#include "DeltaModulationChannel.h"
|
|
#include "ApuFrameCounter.h"
|
|
#include "EmulationSettings.h"
|
|
#include "SoundMixer.h"
|
|
#include "MemoryManager.h"
|
|
|
|
APU* APU::Instance = nullptr;
|
|
bool APU::_apuEnabled = true;
|
|
|
|
APU::APU(MemoryManager* memoryManager)
|
|
{
|
|
APU::Instance = this;
|
|
|
|
_memoryManager = memoryManager;
|
|
|
|
_nesModel = NesModel::Auto;
|
|
|
|
_apuEnabled = true;
|
|
|
|
_mixer.reset(new SoundMixer());
|
|
|
|
_squareChannel[0].reset(new SquareChannel(AudioChannel::Square1, _mixer.get(), true));
|
|
_squareChannel[1].reset(new SquareChannel(AudioChannel::Square2, _mixer.get(), false));
|
|
_triangleChannel.reset(new TriangleChannel(AudioChannel::Triangle, _mixer.get()));
|
|
_noiseChannel.reset(new NoiseChannel(AudioChannel::Noise, _mixer.get()));
|
|
_deltaModulationChannel.reset(new DeltaModulationChannel(AudioChannel::DMC, _mixer.get(), _memoryManager));
|
|
_frameCounter.reset(new ApuFrameCounter(&APU::FrameCounterTick));
|
|
|
|
_memoryManager->RegisterIODevice(_squareChannel[0].get());
|
|
_memoryManager->RegisterIODevice(_squareChannel[1].get());
|
|
_memoryManager->RegisterIODevice(_frameCounter.get());
|
|
_memoryManager->RegisterIODevice(_triangleChannel.get());
|
|
_memoryManager->RegisterIODevice(_noiseChannel.get());
|
|
_memoryManager->RegisterIODevice(_deltaModulationChannel.get());
|
|
|
|
Reset(false);
|
|
}
|
|
|
|
APU::~APU()
|
|
{
|
|
}
|
|
|
|
void APU::SetNesModel(NesModel model, bool forceInit)
|
|
{
|
|
if(_nesModel != model || forceInit) {
|
|
//Finish the current apu frame before switching model
|
|
Run();
|
|
|
|
_nesModel = model;
|
|
_squareChannel[0]->SetNesModel(model);
|
|
_squareChannel[1]->SetNesModel(model);
|
|
_triangleChannel->SetNesModel(model);
|
|
_noiseChannel->SetNesModel(model);
|
|
_deltaModulationChannel->SetNesModel(model);
|
|
_frameCounter->SetNesModel(model);
|
|
|
|
_mixer->SetNesModel(model);
|
|
}
|
|
}
|
|
|
|
void APU::FrameCounterTick(FrameType type)
|
|
{
|
|
//Quarter & half frame clock envelope & linear counter
|
|
Instance->_squareChannel[0]->TickEnvelope();
|
|
Instance->_squareChannel[1]->TickEnvelope();
|
|
Instance->_triangleChannel->TickLinearCounter();
|
|
Instance->_noiseChannel->TickEnvelope();
|
|
|
|
if(type == FrameType::HalfFrame) {
|
|
//Half frames clock length counter & sweep
|
|
Instance->_squareChannel[0]->TickLengthCounter();
|
|
Instance->_squareChannel[1]->TickLengthCounter();
|
|
Instance->_triangleChannel->TickLengthCounter();
|
|
Instance->_noiseChannel->TickLengthCounter();
|
|
|
|
Instance->_squareChannel[0]->TickSweep();
|
|
Instance->_squareChannel[1]->TickSweep();
|
|
}
|
|
}
|
|
|
|
uint8_t APU::ReadRAM(uint16_t addr)
|
|
{
|
|
//$4015 read
|
|
Run();
|
|
|
|
uint8_t status = 0;
|
|
status |= _squareChannel[0]->GetStatus() ? 0x01 : 0x00;
|
|
status |= _squareChannel[1]->GetStatus() ? 0x02 : 0x00;
|
|
status |= _triangleChannel->GetStatus() ? 0x04 : 0x00;
|
|
status |= _noiseChannel->GetStatus() ? 0x08 : 0x00;
|
|
status |= _deltaModulationChannel->GetStatus() ? 0x10 : 0x00;
|
|
status |= CPU::HasIRQSource(IRQSource::FrameCounter) ? 0x40 : 0x00;
|
|
status |= CPU::HasIRQSource(IRQSource::DMC) ? 0x80 : 0x00;
|
|
|
|
//Reading $4015 clears the Frame Counter interrupt flag.
|
|
CPU::ClearIRQSource(IRQSource::FrameCounter);
|
|
|
|
return status;
|
|
}
|
|
|
|
void APU::WriteRAM(uint16_t addr, uint8_t value)
|
|
{
|
|
//$4015 write
|
|
Run();
|
|
|
|
//Writing to $4015 clears the DMC interrupt flag.
|
|
//This needs to be done before setting the enabled flag for the DMC (because doing so can trigger an IRQ)
|
|
CPU::ClearIRQSource(IRQSource::DMC);
|
|
|
|
_squareChannel[0]->SetEnabled((value & 0x01) == 0x01);
|
|
_squareChannel[1]->SetEnabled((value & 0x02) == 0x02);
|
|
_triangleChannel->SetEnabled((value & 0x04) == 0x04);
|
|
_noiseChannel->SetEnabled((value & 0x08) == 0x08);
|
|
_deltaModulationChannel->SetEnabled((value & 0x10) == 0x10);
|
|
}
|
|
|
|
void APU::GetMemoryRanges(MemoryRanges &ranges)
|
|
{
|
|
ranges.AddHandler(MemoryOperation::Read, 0x4015);
|
|
ranges.AddHandler(MemoryOperation::Write, 0x4015);
|
|
}
|
|
|
|
void APU::Run()
|
|
{
|
|
//Update framecounter and all channels
|
|
//This is called:
|
|
//-At the end of a frame
|
|
//-Before APU registers are read/written to
|
|
//-When a DMC or FrameCounter interrupt needs to be fired
|
|
int32_t cyclesToRun = _currentCycle - _previousCycle;
|
|
|
|
while(cyclesToRun > 0) {
|
|
_previousCycle += _frameCounter->Run(cyclesToRun);
|
|
|
|
//Reload counters set by writes to 4003/4008/400B/400F after running the frame counter to allow the length counter to be clocked first
|
|
//This fixes the test "len_reload_timing" (tests 4 & 5)
|
|
_squareChannel[0]->ReloadCounter();
|
|
_squareChannel[1]->ReloadCounter();
|
|
_noiseChannel->ReloadCounter();
|
|
_triangleChannel->ReloadCounter();
|
|
|
|
_squareChannel[0]->Run(_previousCycle);
|
|
_squareChannel[1]->Run(_previousCycle);
|
|
_noiseChannel->Run(_previousCycle);
|
|
_triangleChannel->Run(_previousCycle);
|
|
_deltaModulationChannel->Run(_previousCycle);
|
|
}
|
|
}
|
|
|
|
void APU::StaticRun()
|
|
{
|
|
Instance->Run();
|
|
}
|
|
|
|
bool APU::NeedToRun(uint32_t currentCycle)
|
|
{
|
|
if(ApuLengthCounter::NeedToRun()) {
|
|
return true;
|
|
}
|
|
|
|
if(_deltaModulationChannel->NeedToRun()) {
|
|
//Need to run every cycle when DMC is running to get accurate emulation (CPU stalling, interaction with sprite DMA, etc.)
|
|
return true;
|
|
}
|
|
|
|
uint32_t cyclesToRun = currentCycle - _previousCycle;
|
|
if(_frameCounter->IrqPending(cyclesToRun)) {
|
|
return true;
|
|
} else if(_deltaModulationChannel->IrqPending(cyclesToRun)) {
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void APU::Exec()
|
|
{
|
|
_currentCycle++;
|
|
if(_currentCycle == SoundMixer::CycleLength - 1) {
|
|
EndFrame();
|
|
} else if(NeedToRun(_currentCycle)) {
|
|
Run();
|
|
}
|
|
}
|
|
|
|
void APU::EndFrame()
|
|
{
|
|
Run();
|
|
_squareChannel[0]->EndFrame();
|
|
_squareChannel[1]->EndFrame();
|
|
_triangleChannel->EndFrame();
|
|
_noiseChannel->EndFrame();
|
|
_deltaModulationChannel->EndFrame();
|
|
|
|
_mixer->PlayAudioBuffer(_currentCycle);
|
|
|
|
_currentCycle = 0;
|
|
_previousCycle = 0;
|
|
}
|
|
|
|
void APU::Reset(bool softReset)
|
|
{
|
|
_apuEnabled = true;
|
|
_cyclesNeeded = 0;
|
|
_currentCycle = 0;
|
|
_previousCycle = 0;
|
|
_squareChannel[0]->Reset(softReset);
|
|
_squareChannel[1]->Reset(softReset);
|
|
_triangleChannel->Reset(softReset);
|
|
_noiseChannel->Reset(softReset);
|
|
_deltaModulationChannel->Reset(softReset);
|
|
_frameCounter->Reset(softReset);
|
|
}
|
|
|
|
void APU::StreamState(bool saving)
|
|
{
|
|
if(saving) {
|
|
//End the APU frame - makes it simpler to restore sound after a state reload
|
|
EndFrame();
|
|
} else {
|
|
_previousCycle = 0;
|
|
_currentCycle = 0;
|
|
}
|
|
|
|
SnapshotInfo squareChannel0{ _squareChannel[0].get() };
|
|
SnapshotInfo squareChannel1{ _squareChannel[1].get() };
|
|
SnapshotInfo triangleChannel{ _triangleChannel.get() };
|
|
SnapshotInfo noiseChannel{ _noiseChannel.get() };
|
|
SnapshotInfo deltaModulationChannel{ _deltaModulationChannel.get() };
|
|
SnapshotInfo frameCounter{ _frameCounter.get() };
|
|
SnapshotInfo mixer{ _mixer.get() };
|
|
Stream(_nesModel, squareChannel0, squareChannel1, triangleChannel, noiseChannel, deltaModulationChannel, frameCounter, mixer, _cyclesNeeded);
|
|
}
|
|
|
|
void APU::AddExpansionAudioDelta(AudioChannel channel, int16_t delta)
|
|
{
|
|
Instance->_mixer->AddDelta(channel, Instance->_currentCycle, delta);
|
|
}
|
|
|
|
void APU::SetApuStatus(bool enabled)
|
|
{
|
|
_apuEnabled = enabled;
|
|
}
|
|
|
|
bool APU::IsApuEnabled()
|
|
{
|
|
//Adding extra lines before/after NMI temporarely turns off the APU
|
|
//This appears to result in less side-effects than spreading out the APU's
|
|
//load over the entire PPU frame, like what was done before.
|
|
//This is most likely due to the timing of the Frame Counter & DMC IRQs.
|
|
return _apuEnabled;
|
|
} |