mirror of https://github.com/bsnes-emu/bsnes.git
101 lines
2.5 KiB
C++
101 lines
2.5 KiB
C++
auto SuperFX::read(uint24 addr, uint8 data) -> uint8 {
|
|
if((addr & 0xc00000) == 0x000000) { //$00-3f:0000-7fff,:8000-ffff
|
|
while(!regs.scmr.ron) {
|
|
step(6);
|
|
synchronize(cpu);
|
|
if(scheduler.synchronizing()) break;
|
|
}
|
|
return rom.read((((addr & 0x3f0000) >> 1) | (addr & 0x7fff)) & romMask);
|
|
}
|
|
|
|
if((addr & 0xe00000) == 0x400000) { //$40-5f:0000-ffff
|
|
while(!regs.scmr.ron) {
|
|
step(6);
|
|
synchronize(cpu);
|
|
if(scheduler.synchronizing()) break;
|
|
}
|
|
return rom.read(addr & romMask);
|
|
}
|
|
|
|
if((addr & 0xe00000) == 0x600000) { //$60-7f:0000-ffff
|
|
while(!regs.scmr.ran) {
|
|
step(6);
|
|
synchronize(cpu);
|
|
if(scheduler.synchronizing()) break;
|
|
}
|
|
return ram.read(addr & ramMask);
|
|
}
|
|
|
|
return data;
|
|
}
|
|
|
|
auto SuperFX::write(uint24 addr, uint8 data) -> void {
|
|
if((addr & 0xe00000) == 0x600000) { //$60-7f:0000-ffff
|
|
while(!regs.scmr.ran) {
|
|
step(6);
|
|
synchronize(cpu);
|
|
if(scheduler.synchronizing()) break;
|
|
}
|
|
return ram.write(addr & ramMask, data);
|
|
}
|
|
}
|
|
|
|
auto SuperFX::readOpcode(uint16 addr) -> uint8 {
|
|
uint16 offset = addr - regs.cbr;
|
|
if(offset < 512) {
|
|
if(cache.valid[offset >> 4] == false) {
|
|
uint dp = offset & 0xfff0;
|
|
uint sp = (regs.pbr << 16) + ((regs.cbr + dp) & 0xfff0);
|
|
for(uint n : range(16)) {
|
|
step(regs.clsr ? 5 : 6);
|
|
cache.buffer[dp++] = read(sp++);
|
|
}
|
|
cache.valid[offset >> 4] = true;
|
|
} else {
|
|
step(regs.clsr ? 1 : 2);
|
|
}
|
|
return cache.buffer[offset];
|
|
}
|
|
|
|
if(regs.pbr <= 0x5f) {
|
|
//$00-5f:0000-ffff ROM
|
|
syncROMBuffer();
|
|
step(regs.clsr ? 5 : 6);
|
|
return read(regs.pbr << 16 | addr);
|
|
} else {
|
|
//$60-7f:0000-ffff RAM
|
|
syncRAMBuffer();
|
|
step(regs.clsr ? 5 : 6);
|
|
return read(regs.pbr << 16 | addr);
|
|
}
|
|
}
|
|
|
|
auto SuperFX::peekpipe() -> uint8 {
|
|
uint8 result = regs.pipeline;
|
|
regs.pipeline = readOpcode(regs.r[15]);
|
|
regs.r[15].modified = false;
|
|
return result;
|
|
}
|
|
|
|
auto SuperFX::pipe() -> uint8 {
|
|
uint8 result = regs.pipeline;
|
|
regs.pipeline = readOpcode(++regs.r[15]);
|
|
regs.r[15].modified = false;
|
|
return result;
|
|
}
|
|
|
|
auto SuperFX::flushCache() -> void {
|
|
for(uint n : range(32)) cache.valid[n] = false;
|
|
}
|
|
|
|
auto SuperFX::readCache(uint16 addr) -> uint8 {
|
|
addr = (addr + regs.cbr) & 511;
|
|
return cache.buffer[addr];
|
|
}
|
|
|
|
auto SuperFX::writeCache(uint16 addr, uint8 data) -> void {
|
|
addr = (addr + regs.cbr) & 511;
|
|
cache.buffer[addr] = data;
|
|
if((addr & 15) == 15) cache.valid[addr >> 4] = true;
|
|
}
|