bsnes/higan/sfc/cpu/io.cpp

311 lines
6.8 KiB
C++

auto CPU::readAPU(uint24 addr, uint8 data) -> uint8 {
synchronize(smp);
return smp.readPort(addr.bits(0,1));
}
auto CPU::readCPU(uint24 addr, uint8 data) -> uint8 {
switch((uint16)addr) {
//WMDATA
case 0x2180: {
return bus.read(0x7e0000 | io.wramAddress++, r.mdr);
}
//JOYSER0
//7-2 = MDR
//1-0 = Joypad serial data
case 0x4016: {
uint8 v = r.mdr & 0xfc;
v |= controllerPort1.device->data();
return v;
}
//JOYSER1
case 0x4017: {
//7-5 = MDR
//4-2 = Always 1 (pins are connected to GND)
//1-0 = Joypad serial data
uint8 v = (r.mdr & 0xe0) | 0x1c;
v |= controllerPort2.device->data();
return v;
}
//RDNMI
case 0x4210: {
//7 = NMI acknowledge
//6-4 = MDR
//3-0 = CPU (5a22) version
uint8 v = (r.mdr & 0x70);
v |= (uint8)(rdnmi()) << 7;
v |= (version & 0x0f);
return v;
}
//TIMEUP
case 0x4211: {
//7 = IRQ acknowledge
//6-0 = MDR
uint8 v = (r.mdr & 0x7f);
v |= (uint8)(timeup()) << 7;
return v;
}
//HVBJOY
case 0x4212: {
//7 = VBLANK acknowledge
//6 = HBLANK acknowledge
//5-1 = MDR
//0 = JOYPAD acknowledge
uint8 v = (r.mdr & 0x3e);
if(status.autoJoypadActive) v |= 0x01;
if(hcounter() <= 2 || hcounter() >= 1096) v |= 0x40; //hblank
if(vcounter() >= ppu.vdisp()) v |= 0x80; //vblank
return v;
}
//RDIO
case 0x4213: {
return io.pio;
}
//RDDIVL
case 0x4214: {
return io.rddiv.byte(0);
}
//RDDIVH
case 0x4215: {
return io.rddiv.byte(1);
}
//RDMPYL
case 0x4216: {
return io.rdmpy.byte(0);
}
//RDMPYH
case 0x4217: {
return io.rdmpy.byte(1);
}
case 0x4218: return io.joy1.byte(0); //JOY1L
case 0x4219: return io.joy1.byte(1); //JOY1H
case 0x421a: return io.joy2.byte(0); //JOY2L
case 0x421b: return io.joy2.byte(1); //JOY2H
case 0x421c: return io.joy3.byte(0); //JOY3L
case 0x421d: return io.joy3.byte(1); //JOY3H
case 0x421e: return io.joy4.byte(0); //JOY4L
case 0x421f: return io.joy4.byte(1); //JOY4H
}
return data;
}
auto CPU::readDMA(uint24 addr, uint8 data) -> uint8 {
auto& channel = this->channel[addr.bits(4,6)];
switch(addr & 0xff0f) {
//DMAPx
case 0x4300: return (
channel.transferMode << 0
| channel.fixedTransfer << 3
| channel.reverseTransfer << 4
| channel.unused << 5
| channel.indirect << 6
| channel.direction << 7
);
//BBADx
case 0x4301: return channel.targetAddress;
//A1TxL
case 0x4302: return channel.sourceAddress >> 0;
//A1TxH
case 0x4303: return channel.sourceAddress >> 8;
//A1Bx
case 0x4304: return channel.sourceBank;
//DASxL -- union { uint16 transferSize; uint16 indirectAddress; };
case 0x4305: return channel.transferSize.byte(0);
//DASxH -- union { uint16 transferSize; uint16 indirectAddress; };
case 0x4306: return channel.transferSize.byte(1);
//DASBx
case 0x4307: return channel.indirectBank;
//A2AxL
case 0x4308: return channel.hdmaAddress.byte(0);
//A2AxH
case 0x4309: return channel.hdmaAddress.byte(1);
//NTRLx
case 0x430a: return channel.lineCounter;
//???
case 0x430b:
case 0x430f: return channel.unknown;
}
return data;
}
auto CPU::writeAPU(uint24 addr, uint8 data) -> void {
synchronize(smp);
return writePort(addr.bits(0,1), data);
}
auto CPU::writeCPU(uint24 addr, uint8 data) -> void {
switch((uint16)addr) {
//WMDATA
case 0x2180: {
return bus.write(0x7e0000 | io.wramAddress++, data);
}
case 0x2181: io.wramAddress.bits( 0, 7) = data; return; //WMADDL
case 0x2182: io.wramAddress.bits( 8,15) = data; return; //WMADDM
case 0x2183: io.wramAddress.bit (16 ) = data.bit(0); return; //WMADDH
//JOYSER0
case 0x4016: {
//bit 0 is shared between JOYSER0 and JOYSER1, therefore
//strobing $4016.d0 affects both controller port latches.
//$4017 bit 0 writes are ignored.
controllerPort1.device->latch(data.bit(0));
controllerPort2.device->latch(data.bit(0));
return;
}
//NMITIMEN
case 0x4200: {
io.autoJoypadPoll = data.bit(0);
nmitimenUpdate(data);
return;
}
//WRIO
case 0x4201: {
if(io.pio.bit(7) && !data.bit(7)) ppu.latchCounters();
io.pio = data;
return;
}
//WRMPYA
case 0x4202: io.wrmpya = data; return;
//WRMPYB
case 0x4203: {
io.rdmpy = 0;
if(alu.mpyctr || alu.divctr) return;
io.wrmpyb = data;
io.rddiv = (io.wrmpyb << 8) | io.wrmpya;
alu.mpyctr = 8; //perform multiplication over the next eight cycles
alu.shift = io.wrmpyb;
return;
}
case 0x4204: { io.wrdiva.byte(0) = data; return; } //WRDIVL
case 0x4205: { io.wrdiva.byte(1) = data; return; } //WRDIVH
//WRDIVB
case 0x4206: {
io.rdmpy = io.wrdiva;
if(alu.mpyctr || alu.divctr) return;
io.wrdivb = data;
alu.divctr = 16; //perform division over the next sixteen cycles
alu.shift = io.wrdivb << 16;
return;
}
case 0x4207: io.hirqPos.bits(0,7) = data; return; //HTIMEL
case 0x4208: io.hirqPos.bit (8 ) = data.bit(0); return; //HTIMEH
case 0x4209: io.virqPos.bits(0,7) = data; return; //VTIMEL
case 0x420a: io.virqPos.bit (8 ) = data.bit(0); return; //VTIMEH
//DMAEN
case 0x420b: {
for(auto n : range(8)) channel[n].dmaEnabled = data.bit(n);
if(data) status.dmaPending = true;
return;
}
//HDMAEN
case 0x420c: {
for(auto n : range(8)) channel[n].hdmaEnabled = data.bit(n);
return;
}
//MEMSEL
case 0x420d: {
io.romSpeed = data.bit(0) ? 6 : 8;
return;
}
}
}
auto CPU::writeDMA(uint24 addr, uint8 data) -> void {
auto& channel = this->channel[addr.bits(4,6)];
switch(addr & 0xff0f) {
//DMAPx
case 0x4300: {
channel.transferMode = data.bits(0,2);
channel.fixedTransfer = data.bit (3);
channel.reverseTransfer = data.bit (4);
channel.unused = data.bit (5);
channel.indirect = data.bit (6);
channel.direction = data.bit (7);
return;
}
//DDBADx
case 0x4301: channel.targetAddress = data; return;
//A1TxL
case 0x4302: channel.sourceAddress.byte(0) = data; return;
//A1TxH
case 0x4303: channel.sourceAddress.byte(1) = data; return;
//A1Bx
case 0x4304: channel.sourceBank = data; return;
//DASxL -- union { uint16 transferSize; uint16 indirectAddress; };
case 0x4305: channel.transferSize.byte(0) = data; return;
//DASxH -- union { uint16 transferSize; uint16 indirectAddress; };
case 0x4306: channel.transferSize.byte(1) = data; return;
//DASBx
case 0x4307: channel.indirectBank = data; return;
//A2AxL
case 0x4308: channel.hdmaAddress.byte(0) = data; return;
//A2AxH
case 0x4309: channel.hdmaAddress.byte(1) = data; return;
//NTRLx
case 0x430a: channel.lineCounter = data; return;
//???
case 0x430b:
case 0x430f: channel.unknown = data; return;
}
}