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

90 lines
No EOL
2.2 KiB
C++

#pragma once
#include "pch.h"
#include "NES/BaseMapper.h"
#include "NES/NesConsole.h"
#include "NES/NesCpu.h"
class Kaiser7017 : public BaseMapper
{
uint8_t _prgReg = 0;
MirroringType _mirroring = {};
uint16_t _irqCounter = 0;
bool _irqEnabled = false;
protected:
uint16_t GetPrgPageSize() override { return 0x4000; }
uint16_t GetChrPageSize() override { return 0x2000; }
bool AllowRegisterRead() override { return true; }
uint16_t RegisterStartAddress() override { return 0x4020; }
uint16_t RegisterEndAddress() override { return 0x5FFF; }
bool EnableCpuClockHook() override { return true; }
void InitMapper() override
{
RemoveRegisterRange(0x4020, 0x5FFF, MemoryOperation::Read);
AddRegisterRange(0x4030, 0x4030, MemoryOperation::Read);
SelectChrPage(0, 0);
_prgReg = 0;
_mirroring = MirroringType::Vertical;
_irqCounter = 0;
_irqEnabled = false;
UpdateState();
}
void Serialize(Serializer& s) override
{
BaseMapper::Serialize(s);
SV(_prgReg);
SV(_mirroring);
SV(_irqCounter);
SV(_irqEnabled);
}
void UpdateState()
{
SelectPrgPage(0, _prgReg);
SelectPrgPage(1, 2);
SetMirroringType(_mirroring);
}
void ProcessCpuClock() override
{
BaseProcessCpuClock();
if(_irqEnabled && _irqCounter) {
_irqCounter--;
if(_irqCounter == 0) {
_irqEnabled = false;
_console->GetCpu()->SetIrqSource(IRQSource::External);
}
}
}
uint8_t ReadRegister(uint16_t addr) override
{
bool irqPending = _console->GetCpu()->HasIrqSource(IRQSource::External);
_console->GetCpu()->ClearIrqSource(IRQSource::External);
return irqPending ? 0x01 : 0x00;
}
void WriteRegister(uint16_t addr, uint8_t value) override
{
if((addr & 0xFF00) == 0x4A00) {
_prgReg = ((addr >> 2) & 0x03) | ((addr >> 4) & 0x04);
} else if((addr & 0xFF00) == 0x5100) {
UpdateState();
} else if(addr == 0x4020) {
_console->GetCpu()->ClearIrqSource(IRQSource::External);
_irqCounter = (_irqCounter & 0xFF00) | value;
} else if(addr == 0x4021) {
_console->GetCpu()->ClearIrqSource(IRQSource::External);
_irqCounter = (_irqCounter & 0xFF) | (value << 8);
_irqEnabled = true;
} else if(addr == 0x4025) {
_mirroring = ((value >> 3) & 0x01) ? MirroringType::Horizontal : MirroringType::Vertical;
}
}
};