mirror of https://github.com/bsnes-emu/bsnes.git
311 lines
6.8 KiB
C++
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;
|
|
|
|
}
|
|
}
|