auto CPU::sleep() -> void { prefetchStep(1); } auto CPU::get(uint mode, uint32 addr) -> uint32 { uint clocks = _wait(mode, addr); uint word = pipeline.fetch.instruction; if(addr >= 0x1000'0000) { prefetchStep(clocks); } else if(addr & 0x0800'0000) { if(mode & Prefetch && wait.prefetch) { prefetchSync(addr); word = prefetchRead(); if(mode & Word) word |= prefetchRead() << 16; } else { if(!context.dmaActive) prefetchWait(); step(clocks - 1); word = cartridge.read(mode, addr); step(1); } } else { prefetchStep(clocks - 1); if(addr < 0x0200'0000) word = bios.read(mode, addr); else if(addr < 0x0300'0000) word = readEWRAM(mode, addr); else if(addr < 0x0400'0000) word = readIWRAM(mode, addr); else if(addr >= 0x0700'0000) word = ppu.readOAM(mode, addr); else if(addr >= 0x0600'0000) word = ppu.readVRAM(mode, addr); else if(addr >= 0x0500'0000) word = ppu.readPRAM(mode, addr); else if((addr & 0xffff'fc00) == 0x0400'0000) word = bus.io[addr & 0x3ff]->readIO(mode, addr); else if((addr & 0xff00'ffff) == 0x0400'0800) word = ((IO*)this)->readIO(mode, 0x0400'0800 | (addr & 3)); prefetchStep(1); } return word; } auto CPU::set(uint mode, uint32 addr, uint32 word) -> void { uint clocks = _wait(mode, addr); if(addr >= 0x1000'0000) { prefetchStep(clocks); } else if(addr & 0x0800'0000) { if(!context.dmaActive) prefetchWait(); step(clocks); cartridge.write(mode, addr, word); } else { prefetchStep(clocks); if(addr < 0x0200'0000); else if(addr < 0x0300'0000) writeEWRAM(mode, addr, word); else if(addr < 0x0400'0000) writeIWRAM(mode, addr, word); else if(addr >= 0x0700'0000) ppu.writeOAM(mode, addr, word); else if(addr >= 0x0600'0000) ppu.writeVRAM(mode, addr, word); else if(addr >= 0x0500'0000) ppu.writePRAM(mode, addr, word); else if((addr & 0xffff'fc00) == 0x0400'0000) bus.io[addr & 0x3ff]->writeIO(mode, addr, word); else if((addr & 0xff00'ffff) == 0x0400'0800) ((IO*)this)->writeIO(mode, 0x0400'0800 | (addr & 3), word); } } auto CPU::_wait(uint mode, uint32 addr) -> uint { if(addr >= 0x1000'0000) return 1; //unmapped if(addr < 0x0200'0000) return 1; if(addr < 0x0300'0000) return (16 - memory.ewramWait) * (mode & Word ? 2 : 1); if(addr < 0x0500'0000) return 1; if(addr < 0x0700'0000) return mode & Word ? 2 : 1; if(addr < 0x0800'0000) return 1; static uint timings[] = {5, 4, 3, 9}; uint n = timings[wait.nwait[addr >> 25 & 3]]; uint s = wait.swait[addr >> 25 & 3]; switch(addr & 0x0e00'0000) { case 0x0800'0000: s = s ? 2 : 3; break; case 0x0a00'0000: s = s ? 2 : 5; break; case 0x0c00'0000: s = s ? 2 : 9; break; case 0x0e00'0000: s = n; break; } bool sequential = (mode & Sequential); if((addr & 0x1fffe) == 0) sequential = false; //N cycle on 16-bit ROM crossing 128KB page boundary (RAM S==N) uint clocks = sequential ? s : n; if(mode & Word) clocks += s; //16-bit bus requires two transfers for words return clocks; }