package com.grapeshot.halfnes.mappers;
import com.grapeshot.halfnes.*;
public class Mapper182 extends Mapper {
//Pirate MMC3 clone with scrambled registers
private int whichbank = 0;
private boolean prgconfig = false;
private boolean chrconfig = false;
private int irqctrreload = 0;
private int irqctr = 0;
private boolean irqenable = false;
private boolean irqreload = false;
private int bank6 = 0;
private int[] chrreg = {0, 0, 0, 0, 0, 0};
private boolean interrupted = false;
@Override
public void loadrom() throws BadMapperException {
//needs to be in every mapper. Fill with initial cfg
super.loadrom();
for (int i = 1; i <= 32; ++i) {
prg_map[32 - i] = prgsize - (1024 * i);
}
for (int i = 0; i < 8; ++i) {
chr_map[i] = 0;
}
setbank6();
//cpuram.setPrgRAMEnable(false);
}
@Override
public final void cartWrite(int addr, int data) {
if (addr < 0x8000 || addr > 0xffff) {
super.cartWrite(addr, data);
return;
}
//bankswitches here
//different register for even/odd writes
if (((addr & (utils.BIT0)) != 0)) {
//odd registers
if ((addr >= 0x8000) && (addr <= 0x9fff)) {
//mirroring setup
setmirroring(((data & (utils.BIT0)) != 0) ? MirrorType.H_MIRROR : MirrorType.V_MIRROR);
} else if ((addr >= 0xA000) && (addr <= 0xBFFF)) {
//prg ram write protect
//cpuram.setPrgRAMEnable(!utils.getbit(data, 7));
} else if ((addr >= 0xC000) && (addr <= 0xDFFF)) {
//any value here reloads irq counter _@ end of scanline_
irqreload = true;
irqctrreload = data;
} else if ((addr >= 0xE000) && (addr <= 0xFFFF)) {
//any value here enables interrupts
irqenable = true;
}
} else {
//even registers
if ((addr >= 0xA000) && (addr <= 0xBFFF)) {
//bank select
whichbank = data & 7;
prgconfig = ((data & (utils.BIT4)) != 0);
//if bit is false, 8000-9fff swappable and c000-dfff fixed to 2nd to last bank
//if bit is true, c000-dfff swappable and 8000-9fff fixed to 2nd to last bank
chrconfig = ((data & (utils.BIT5)) != 0);
//if false: 2 2k banks @ 0000-0fff, 4 1k banks in 1000-1fff
//if true: 4 1k banks @ 0000-0fff, 2 2k banks @ 1000-1fff
setupchr();
setbank6(); //OOPS FORGOT THIS I GUESS
} else if ((addr >= 0xC000) && (addr <= 0xDFFF)) {
//bank select
switch (whichbank) {
case 0:
chrreg[0] = data;
setupchr();
break;
case 1:
chrreg[3] = data;
setupchr();
break;
case 2:
chrreg[1] = data;
setupchr();
break;
case 3:
chrreg[5] = data;
setupchr();
break;
case 4:
bank6 = data;
setbank6();
break;
case 5:
//bank 5 always swappable, always in same place
for (int i = 0; i < 8; ++i) {
prg_map[i + 8] = (1024 * (i + (data * 8))) % prgsize;
}
break;
case 6:
chrreg[2] = data;
setupchr();
break;
case 7:
chrreg[4] = data;
setupchr();
break;
}
} else if ((addr >= 0xE000) && (addr <= 0xFFFF)) {
//any value here disables IRQ and acknowledges
if (interrupted) {
--cpu.interrupt;
}
interrupted = false;
irqenable = false;
irqctr = irqctrreload;
}
}
}
private void setupchr() {
if (chrconfig) {
setppubank(1, 0, chrreg[2]);
setppubank(1, 1, chrreg[3]);
setppubank(1, 2, chrreg[4]);
setppubank(1, 3, chrreg[5]);
//Lowest bit of bank number IS IGNORED for the 2k banks
setppubank(2, 4, (chrreg[0] >> 1) << 1);
setppubank(2, 6, (chrreg[1] >> 1) << 1);
} else {
setppubank(1, 4, chrreg[2]);
setppubank(1, 5, chrreg[3]);
setppubank(1, 6, chrreg[4]);
setppubank(1, 7, chrreg[5]);
setppubank(2, 0, (chrreg[0] >> 1) << 1);
setppubank(2, 2, (chrreg[1] >> 1) << 1);
}
}
private void setbank6() {
if (!prgconfig) {
//map c000-dfff to last bank, 8000-9fff to selected bank
for (int i = 0; i < 8; ++i) {
prg_map[i] = (1024 * (i + (bank6 * 8))) % prgsize;
prg_map[i + 16] = ((prgsize - 16384) + 1024 * i);
}
} else {
//map 8000-9fff to last bank, c000 to dfff to selected bank
for (int i = 0; i < 8; ++i) {
prg_map[i] = ((prgsize - 16384) + 1024 * i);
prg_map[i + 16] = (1024 * (i + (bank6 * 8))) % prgsize;
}
}
}
@Override
public void notifyscanline(int scanline) {
//Scanline counter
if (scanline > 239 && scanline != 261) {
//clocked on LAST line of vblank and all lines of frame. Not on 240.
return;
}
if (!ppu.mmc3CounterClocking()) {
return;
}
if (irqreload) {
irqreload = false;
irqctr = irqctrreload;
}
if (irqctr-- <= 0) {
if (irqctrreload == 0) {
return;
//irqs stop being generated if reload set to zero
}
if (irqenable && !interrupted) {
++cpu.interrupt;
interrupted = true;
}
irqctr = irqctrreload;
}
}
private void setppubank(int banksize, int bankpos, int banknum) {
// System.err.println(banksize + ", " + bankpos + ", "+ banknum);
for (int i = 0; i < banksize; ++i) {
chr_map[i + bankpos] = (1024 * ((banknum) + i)) % chrsize;
}
}
}