#pragma once #include "stdafx.h" #include "MMC3.h" class MMC3_126 : public MMC3 { private: uint8_t _exRegs[4]; void SelectPRGPage(uint16_t slot, uint16_t page, PrgMemoryType memoryType = PrgMemoryType::PrgRom) override { uint16_t reg = _exRegs[0]; page &= ((~reg >> 2) & 0x10) | 0x0F; page |= (reg & (0x06 | (reg & 0x40) >> 6)) << 4 | (reg & 0x10) << 3; if(!(_exRegs[3] & 0x03)) { MMC3::SelectPRGPage(slot, page, memoryType); } else if((_prgMode << 1) == slot) { if((_exRegs[3] & 0x03) == 0x03) { MMC3::SelectPRGPage(0, page, memoryType); MMC3::SelectPRGPage(1, page + 1, memoryType); MMC3::SelectPRGPage(2, page + 2, memoryType); MMC3::SelectPRGPage(3, page + 3, memoryType); } else { MMC3::SelectPRGPage(0, page, memoryType); MMC3::SelectPRGPage(1, page + 1, memoryType); MMC3::SelectPRGPage(2, page, memoryType); MMC3::SelectPRGPage(3, page + 1, memoryType); } } } void SelectCHRPage(uint16_t slot, uint16_t page, ChrMemoryType memoryType) override { if(!(_exRegs[3] & 0x10)) { MMC3::SelectCHRPage(slot, GetChrOuterBank() | (page & ((_exRegs[0] & 0x80) - 1))); } } uint16_t GetChrOuterBank() { uint16_t reg = _exRegs[0]; return (~reg << 0 & 0x0080 & _exRegs[2]) | (reg << 4 & 0x0080 & reg) | (reg << 3 & 0x0100) | (reg << 5 & 0x0200); } void InitMapper() override { MMC3::InitMapper(); memset(_exRegs, 0, sizeof(_exRegs)); AddRegisterRange(0x6000, 0x8000, MemoryOperation::Write); } void StreamState(bool saving) override { MMC3::StreamState(saving); Stream(_exRegs[0], _exRegs[1], _exRegs[2], _exRegs[3]); } void WriteRegister(uint16_t addr, uint8_t value) override { if(addr < 0x8000) { addr &= 0x03; if(addr == 0x01 || addr == 0x02 || ((addr == 0x00 || addr == 0x03) && !(_exRegs[3] & 0x80))) { if(_exRegs[addr] != value) { _exRegs[addr] = value; if(_exRegs[3] & 0x10) { uint16_t page = GetChrOuterBank() | ((_exRegs[2] & 0x0F) << 3); for(int i = 0; i < 8; i++) { MMC3::SelectCHRPage(i, page + i); } } else { MMC3::UpdateChrMapping(); } MMC3::UpdatePrgMapping(); } } } else { MMC3::WriteRegister(addr, value); } } };