Mesen2/Core/NES/Mappers/Taito/TaitoX1017.h
2022-09-08 21:29:52 -04:00

117 lines
No EOL
3 KiB
C++

#pragma once
#include "pch.h"
#include "NES/BaseMapper.h"
class TaitoX1017 : public BaseMapper
{
private:
uint8_t _chrMode = 0;
uint8_t _chrRegs[6] = {};
uint8_t _ramPermission[3] = {};
void UpdateRamAccess()
{
SetCpuMemoryMapping(0x6000, 0x63FF, 0, PrgMemoryType::SaveRam, _ramPermission[0] == 0xCA ? MemoryAccessType::ReadWrite : MemoryAccessType::NoAccess);
SetCpuMemoryMapping(0x6400, 0x67FF, 1, PrgMemoryType::SaveRam, _ramPermission[0] == 0xCA ? MemoryAccessType::ReadWrite : MemoryAccessType::NoAccess);
SetCpuMemoryMapping(0x6800, 0x6BFF, 2, PrgMemoryType::SaveRam, _ramPermission[1] == 0x69 ? MemoryAccessType::ReadWrite : MemoryAccessType::NoAccess);
SetCpuMemoryMapping(0x6C00, 0x6FFF, 3, PrgMemoryType::SaveRam, _ramPermission[1] == 0x69 ? MemoryAccessType::ReadWrite : MemoryAccessType::NoAccess);
SetCpuMemoryMapping(0x7000, 0x73FF, 4, PrgMemoryType::SaveRam, _ramPermission[2] == 0x84 ? MemoryAccessType::ReadWrite : MemoryAccessType::NoAccess);
}
protected:
uint16_t GetPrgPageSize() override { return 0x2000; }
uint16_t GetChrPageSize() override { return 0x0400; }
uint16_t RegisterStartAddress() override { return 0x7EF0; }
uint16_t RegisterEndAddress() override { return 0x7EFF; }
uint32_t GetSaveRamSize() override { return 0x1400; }
uint32_t GetSaveRamPageSize() override { return 0x400; }
void InitMapper() override
{
_chrMode = 0;
memset(_ramPermission, 0, sizeof(_ramPermission));
memset(_chrRegs, 0, sizeof(_chrRegs));
SelectPrgPage(3, -1);
UpdateRamAccess();
}
void UpdateChrBanking()
{
if(_chrMode == 0) {
//Regs 0 & 1 ignore the LSB
SelectChrPage2x(0, _chrRegs[0] & 0xFE);
SelectChrPage2x(1, _chrRegs[1] & 0xFE);
SelectChrPage(4, _chrRegs[2]);
SelectChrPage(5, _chrRegs[3]);
SelectChrPage(6, _chrRegs[4]);
SelectChrPage(7, _chrRegs[5]);
} else {
SelectChrPage(0, _chrRegs[2]);
SelectChrPage(1, _chrRegs[3]);
SelectChrPage(2, _chrRegs[4]);
SelectChrPage(3, _chrRegs[5]);
//Regs 0 & 1 ignore the LSB
SelectChrPage2x(2, _chrRegs[0] & 0xFE);
SelectChrPage2x(3, _chrRegs[1] & 0xFE);
}
}
void WriteRegister(uint16_t addr, uint8_t value) override
{
switch(addr) {
case 0x7EF0:
case 0x7EF1:
case 0x7EF2:
case 0x7EF3:
case 0x7EF4:
case 0x7EF5:
_chrRegs[(addr & 0xF)] = value;
UpdateChrBanking();
break;
case 0x7EF6:
SetMirroringType((value & 0x01) == 0x01 ? MirroringType::Vertical : MirroringType::Horizontal);
_chrMode = (value & 0x02) >> 1;
UpdateChrBanking();
break;
case 0x7EF7:
case 0x7EF8:
case 0x7EF9:
_ramPermission[(addr & 0xF) - 7] = value;
UpdateRamAccess();
break;
case 0x7EFA:
SelectPrgPage(0, value >> 2);
break;
case 0x7EFB:
SelectPrgPage(1, value >> 2);
break;
case 0x7EFC:
SelectPrgPage(2, value >> 2);
break;
}
}
void Serialize(Serializer& s) override
{
BaseMapper::Serialize(s);
SVArray(_ramPermission, 3);
SVArray(_chrRegs, 6);
SV(_chrMode);
if(!s.IsSaving()) {
UpdateRamAccess();
}
}
};