#include namespace SuperFamicom { #include "bus.cpp" #include "dma.cpp" #include "memory.cpp" #include "io.cpp" #include "serialization.cpp" SA1 sa1; auto SA1::Enter() -> void { while(true) scheduler.synchronize(), sa1.main(); } auto SA1::main() -> void { if(r.wai) return instructionWait(); if(r.stp) return instructionStop(); if(mmio.sa1_rdyb || mmio.sa1_resb) { //SA-1 co-processor is asleep tick(); synchronize(cpu); return; } if(status.interruptPending) { status.interruptPending = false; interrupt(); return; } instruction(); } //override R65816::interrupt() to support SA-1 vector location IO registers auto SA1::interrupt() -> void { read(r.pc); idle(); if(!r.e) push(r.pc >> 16); push(r.pc >> 8); push(r.pc >> 0); push(r.e ? r.p & ~0x10 : r.p); r.p.i = 1; r.p.d = 0; r.pc = r.vector; //PC bank set to 0x00 } auto SA1::lastCycle() -> void { if(mmio.sa1_nmi && !mmio.sa1_nmicl) { status.interruptPending = true; r.vector = mmio.cnv; mmio.sa1_nmifl = true; mmio.sa1_nmicl = 1; r.wai = false; } else if(!r.p.i) { if(mmio.timer_irqen && !mmio.timer_irqcl) { status.interruptPending = true; r.vector = mmio.civ; mmio.timer_irqfl = true; r.wai = false; } else if(mmio.dma_irqen && !mmio.dma_irqcl) { status.interruptPending = true; r.vector = mmio.civ; mmio.dma_irqfl = true; r.wai = false; } else if(mmio.sa1_irq && !mmio.sa1_irqcl) { status.interruptPending = true; r.vector = mmio.civ; mmio.sa1_irqfl = true; r.wai = false; } } } auto SA1::interruptPending() const -> bool { return status.interruptPending; } auto SA1::synchronizing() const -> bool { return scheduler.synchronizing(); } auto SA1::tick() -> void { step(2); if(++status.counter == 0) synchronize(cpu); //adjust counters: //note that internally, status counters are in clocks; //whereas MMIO register counters are in dots (4 clocks = 1 dot) if(mmio.hvselb == 0) { //HV timer status.hcounter += 2; if(status.hcounter >= 1364) { status.hcounter = 0; if(++status.vcounter >= status.scanlines) status.vcounter = 0; } } else { //linear timer status.hcounter += 2; status.vcounter += (status.hcounter >> 11); status.hcounter &= 0x07ff; status.vcounter &= 0x01ff; } //test counters for timer IRQ switch((mmio.ven << 1) + (mmio.hen << 0)) { case 0: break; case 1: if(status.hcounter == (mmio.hcnt << 2)) triggerIRQ(); break; case 2: if(status.vcounter == mmio.vcnt && status.hcounter == 0) triggerIRQ(); break; case 3: if(status.vcounter == mmio.vcnt && status.hcounter == (mmio.hcnt << 2)) triggerIRQ(); break; } } auto SA1::triggerIRQ() -> void { mmio.timer_irqfl = true; if(mmio.timer_irqen) mmio.timer_irqcl = 0; } auto SA1::unload() -> void { rom.reset(); iram.reset(); bwram.reset(); } auto SA1::power() -> void { WDC65816::power(); create(SA1::Enter, system.cpuFrequency()); rom.writeProtect(true); bwram.writeProtect(false); iram.writeProtect(false); cpubwram.dma = false; for(auto addr : range(iram.size())) { iram.write(addr, 0x00); } status.counter = 0; status.interruptPending = false; status.scanlines = Region::PAL() ? 312 : 262; status.vcounter = 0; status.hcounter = 0; dma.line = 0; //$2200 CCNT mmio.sa1_irq = false; mmio.sa1_rdyb = false; mmio.sa1_resb = true; mmio.sa1_nmi = false; mmio.smeg = 0; //$2201 SIE mmio.cpu_irqen = false; mmio.chdma_irqen = false; //$2202 SIC mmio.cpu_irqcl = false; mmio.chdma_irqcl = false; //$2203,$2204 CRV mmio.crv = 0x0000; //$2205,$2206 CNV mmio.cnv = 0x0000; //$2207,$2208 CIV mmio.civ = 0x0000; //$2209 SCNT mmio.cpu_irq = false; mmio.cpu_ivsw = false; mmio.cpu_nvsw = false; mmio.cmeg = 0; //$220a CIE mmio.sa1_irqen = false; mmio.timer_irqen = false; mmio.dma_irqen = false; mmio.sa1_nmien = false; //$220b CIC mmio.sa1_irqcl = false; mmio.timer_irqcl = false; mmio.dma_irqcl = false; mmio.sa1_nmicl = false; //$220c,$220d SNV mmio.snv = 0x0000; //$220e,$220f SIV mmio.siv = 0x0000; //$2210 mmio.hvselb = false; mmio.ven = false; mmio.hen = false; //$2212,$2213 HCNT mmio.hcnt = 0x0000; //$2214,$2215 VCNT mmio.vcnt = 0x0000; //$2220-2223 CXB, DXB, EXB, FXB mmio.cbmode = 0; mmio.dbmode = 0; mmio.ebmode = 0; mmio.fbmode = 0; mmio.cb = 0x00; mmio.db = 0x01; mmio.eb = 0x02; mmio.fb = 0x03; //$2224 BMAPS mmio.sbm = 0x00; //$2225 BMAP mmio.sw46 = false; mmio.cbm = 0x00; //$2226 SWBE mmio.swen = false; //$2227 CWBE mmio.cwen = false; //$2228 BWPA mmio.bwp = 0x0f; //$2229 SIWP mmio.siwp = 0x00; //$222a CIWP mmio.ciwp = 0x00; //$2230 DCNT mmio.dmaen = false; mmio.dprio = false; mmio.cden = false; mmio.cdsel = false; mmio.dd = 0; mmio.sd = 0; //$2231 CDMA mmio.chdend = false; mmio.dmasize = 0; mmio.dmacb = 0; //$2232-$2234 SDA mmio.dsa = 0x000000; //$2235-$2237 DDA mmio.dda = 0x000000; //$2238,$2239 DTC mmio.dtc = 0x0000; //$223f BBF mmio.bbf = 0; //$2240-$224f BRF for(auto& n : mmio.brf) n = 0x00; //$2250 MCNT mmio.acm = 0; mmio.md = 0; //$2251,$2252 MA mmio.ma = 0x0000; //$2253,$2254 MB mmio.mb = 0x0000; //$2258 VBD mmio.hl = false; mmio.vb = 16; //$2259-$225b mmio.va = 0x000000; mmio.vbit = 0; //$2300 SFR mmio.cpu_irqfl = false; mmio.chdma_irqfl = false; //$2301 CFR mmio.sa1_irqfl = false; mmio.timer_irqfl = false; mmio.dma_irqfl = false; mmio.sa1_nmifl = false; //$2302,$2303 HCR mmio.hcr = 0x0000; //$2304,$2305 VCR mmio.vcr = 0x0000; //$2306-$230a MR mmio.mr = 0; //$230b mmio.overflow = false; } }