Mesen2/Core/NES/Mappers/Sunsoft/SunsoftFme7.h
2024-10-01 20:57:06 +09:00

134 lines
No EOL
3.1 KiB
C++

#pragma once
#include "pch.h"
#include "NES/BaseMapper.h"
#include "NES/NesCpu.h"
#include "NES/Mappers/Audio/Sunsoft5bAudio.h"
class SunsoftFme7 : public BaseMapper
{
private:
unique_ptr<Sunsoft5bAudio> _audio;
uint8_t _command = 0;
uint8_t _workRamValue = 0;
bool _irqEnabled = false;
bool _irqCounterEnabled = false;
uint16_t _irqCounter = 0;
protected:
uint16_t GetPrgPageSize() override { return 0x2000; }
uint16_t GetChrPageSize() override { return 0x400; }
uint32_t GetWorkRamSize() override { return 0x8000; }
uint32_t GetWorkRamPageSize() override { return 0x2000; }
uint32_t GetSaveRamSize() override { return 0x8000; }
uint32_t GetSaveRamPageSize() override { return 0x2000; }
bool EnableCpuClockHook() override { return true; }
void InitMapper() override
{
_audio.reset(new Sunsoft5bAudio(_console));
_command = 0;
_workRamValue = 0;
_irqEnabled = false;
_irqCounterEnabled = false;
_irqCounter = 0;
SelectPrgPage(3, -1);
UpdateWorkRam();
}
void Serialize(Serializer& s) override
{
BaseMapper::Serialize(s);
SV(_audio);
SV(_command);
SV(_workRamValue);
SV(_irqEnabled);
SV(_irqCounterEnabled);
SV(_irqCounter);
if(!s.IsSaving()) {
UpdateWorkRam();
}
}
void ProcessCpuClock() override
{
BaseProcessCpuClock();
if(_irqCounterEnabled) {
_irqCounter--;
if(_irqCounter == 0xFFFF) {
if(_irqEnabled) {
_console->GetCpu()->SetIrqSource(IRQSource::External);
}
}
}
_audio->Clock();
}
void UpdateWorkRam()
{
if(_workRamValue & 0x40) {
MemoryAccessType accessType = (_workRamValue & 0x80) ? MemoryAccessType::ReadWrite : MemoryAccessType::NoAccess;
SetCpuMemoryMapping(0x6000, 0x7FFF, _workRamValue & 0x3F, HasBattery() ? PrgMemoryType::SaveRam : PrgMemoryType::WorkRam, accessType);
} else {
SetCpuMemoryMapping(0x6000, 0x7FFF, _workRamValue & 0x3F, PrgMemoryType::PrgRom);
}
}
void WriteRegister(uint16_t addr, uint8_t value) override
{
switch(addr & 0xE000) {
case 0x8000:
_command = value & 0x0F;
break;
case 0xA000:
switch(_command) {
case 0: case 1: case 2: case 3: case 4: case 5: case 6: case 7:
SelectChrPage(_command, value);
break;
case 8: {
_workRamValue = value;
UpdateWorkRam();
break;
}
case 9: case 0xA: case 0xB:
SelectPrgPage(_command - 9, value & 0x3F);
break;
case 0xC:
switch(value & 0x03) {
case 0: SetMirroringType(MirroringType::Vertical); break;
case 1: SetMirroringType(MirroringType::Horizontal); break;
case 2: SetMirroringType(MirroringType::ScreenAOnly); break;
case 3: SetMirroringType(MirroringType::ScreenBOnly); break;
}
break;
case 0xD:
_irqEnabled = (value & 0x01) == 0x01;
_irqCounterEnabled = (value & 0x80) == 0x80;
_console->GetCpu()->ClearIrqSource(IRQSource::External);
break;
case 0xE:
_irqCounter = (_irqCounter & 0xFF00) | value;
break;
case 0xF:
_irqCounter = (_irqCounter & 0xFF) | (value << 8);
break;
}
break;
case 0xC000:
case 0xE000:
_audio->WriteRegister(addr, value);
break;
}
}
};