bsnes/higan/sfc/coprocessor/superfx/memory.cpp

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;
}