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

72 lines
No EOL
1.8 KiB
C++

#pragma once
#include "pch.h"
#include "NES/BaseMapper.h"
#include "NES/Mappers/Konami/VrcIrq.h"
class Waixing252 : public BaseMapper
{
private:
uint8_t _chrRegs[8] = {};
unique_ptr<VrcIrq> _irq;
protected:
uint16_t GetPrgPageSize() override { return 0x2000; }
uint16_t GetChrPageSize() override { return 0x400; }
bool EnableCpuClockHook() override { return true; }
void InitMapper() override
{
_irq.reset(new VrcIrq(_console));
memset(_chrRegs, 0, sizeof(_chrRegs));
SelectPrgPage(2, -2);
SelectPrgPage(3, -1);
}
void Serialize(Serializer& s) override
{
BaseMapper::Serialize(s);
SV(_irq);
SVArray(_chrRegs, 8);
if(!s.IsSaving()) {
UpdateState();
}
}
void ProcessCpuClock() override
{
BaseProcessCpuClock();
_irq->ProcessCpuClock();
}
void UpdateState()
{
for(int i = 0; i < 8; i++) {
//CHR needs to be writeable (according to Nestopia's source, and this does remove visual glitches from the game)
SetPpuMemoryMapping(0x400 * i, 0x400 * i + 0x3FF, _chrRegs[i], ChrMemoryType::Default, MemoryAccessType::ReadWrite);
}
}
void WriteRegister(uint16_t addr, uint8_t value) override
{
if(addr <= 0x8FFF) {
SelectPrgPage(0, value);
} else if(addr >= 0xA000 && addr <= 0xAFFF) {
SelectPrgPage(1, value);
} else if(addr >= 0xB000 && addr <= 0xEFFF) {
uint8_t shift = addr & 0x4;
uint8_t bank = (((addr - 0xB000) >> 1 & 0x1800) | (addr << 7 & 0x0400)) / 0x400;
_chrRegs[bank] = ((_chrRegs[bank] & (0xF0 >> shift)) | ((value & 0x0F) << shift));
UpdateState();
} else {
switch(addr & 0xF00C) {
case 0xF000: _irq->SetReloadValueNibble(value, false); break;
case 0xF004: _irq->SetReloadValueNibble(value, true); break;
case 0xF008: _irq->SetControlValue(value); break;
case 0xF00C: _irq->AcknowledgeIrq(); break;
}
}
}
};