2011-06-12 06:25:22 +00:00
|
|
|
#define CYCLE_ACCURATE
|
2011-10-18 18:10:06 +00:00
|
|
|
#define PSEUDO_CYCLE
|
2011-06-12 06:25:22 +00:00
|
|
|
|
|
|
|
#include <snes/snes.hpp>
|
|
|
|
|
|
|
|
#define SMP_CPP
|
|
|
|
namespace SNES {
|
|
|
|
|
2017-10-05 17:22:02 +00:00
|
|
|
// TODO: reactivate once APU debugger works again
|
|
|
|
#if 0 // DEBUGGER
|
2011-06-12 06:25:22 +00:00
|
|
|
#include "debugger/debugger.cpp"
|
|
|
|
#include "debugger/disassembler.cpp"
|
|
|
|
SMPDebugger smp;
|
|
|
|
#else
|
|
|
|
SMP smp;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "algorithms.cpp"
|
|
|
|
#include "core.cpp"
|
|
|
|
#include "iplrom.cpp"
|
|
|
|
#include "memory.cpp"
|
|
|
|
#include "timing.cpp"
|
|
|
|
|
|
|
|
void SMP::synchronize_cpu() {
|
2011-06-23 10:24:13 +00:00
|
|
|
#ifndef SNES9X
|
2011-06-12 06:25:22 +00:00
|
|
|
if(CPU::Threaded == true) {
|
|
|
|
//if(clock >= 0 && scheduler.sync != Scheduler::SynchronizeMode::All) co_switch(cpu.thread);
|
|
|
|
} else {
|
|
|
|
while(clock >= 0) cpu.enter();
|
|
|
|
}
|
2011-06-22 11:03:29 +00:00
|
|
|
#endif
|
2011-06-12 06:25:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SMP::synchronize_dsp() {
|
2011-06-23 11:14:14 +00:00
|
|
|
#ifndef SNES9X
|
2011-06-12 06:25:22 +00:00
|
|
|
if(DSP::Threaded == true) {
|
|
|
|
//if(dsp.clock < 0 && scheduler.sync != Scheduler::SynchronizeMode::All) co_switch(dsp.thread);
|
|
|
|
} else {
|
|
|
|
while(dsp.clock < 0) dsp.enter();
|
|
|
|
}
|
2011-06-23 11:14:14 +00:00
|
|
|
#endif
|
2011-06-12 06:25:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SMP::enter() {
|
|
|
|
while(clock < 0) op_step();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SMP::power() {
|
2011-06-23 10:24:13 +00:00
|
|
|
#ifndef SNES9X
|
2011-06-12 06:25:22 +00:00
|
|
|
Processor::frequency = system.apu_frequency();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
Processor::clock = 0;
|
|
|
|
|
|
|
|
timer0.target = 0;
|
|
|
|
timer1.target = 0;
|
|
|
|
timer2.target = 0;
|
|
|
|
|
|
|
|
for(unsigned n = 0; n < 256; n++) {
|
|
|
|
cycle_table_dsp[n] = (cycle_count_table[n] * 24);
|
|
|
|
cycle_table_cpu[n] = (cycle_count_table[n] * 24) * cpu.frequency;
|
|
|
|
}
|
|
|
|
|
|
|
|
cycle_step_cpu = 24 * cpu.frequency;
|
|
|
|
|
|
|
|
reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SMP::reset() {
|
|
|
|
for(unsigned n = 0x0000; n <= 0xffff; n++) apuram[n] = 0x00;
|
|
|
|
|
|
|
|
opcode_number = 0;
|
|
|
|
opcode_cycle = 0;
|
|
|
|
|
|
|
|
regs.pc = 0xffc0;
|
|
|
|
regs.sp = 0xef;
|
2013-05-18 20:10:50 +00:00
|
|
|
regs.B.a = 0x00;
|
2011-06-12 06:25:22 +00:00
|
|
|
regs.x = 0x00;
|
2013-05-18 20:10:50 +00:00
|
|
|
regs.B.y = 0x00;
|
2011-06-12 06:25:22 +00:00
|
|
|
regs.p = 0x02;
|
|
|
|
|
|
|
|
//$00f1
|
|
|
|
status.iplrom_enable = true;
|
|
|
|
|
|
|
|
//$00f2
|
|
|
|
status.dsp_addr = 0x00;
|
|
|
|
|
|
|
|
//$00f8,$00f9
|
|
|
|
status.ram00f8 = 0x00;
|
|
|
|
status.ram00f9 = 0x00;
|
|
|
|
|
|
|
|
//timers
|
|
|
|
timer0.enable = timer1.enable = timer2.enable = false;
|
|
|
|
timer0.stage1_ticks = timer1.stage1_ticks = timer2.stage1_ticks = 0;
|
|
|
|
timer0.stage2_ticks = timer1.stage2_ticks = timer2.stage2_ticks = 0;
|
|
|
|
timer0.stage3_ticks = timer1.stage3_ticks = timer2.stage3_ticks = 0;
|
|
|
|
}
|
|
|
|
|
2011-06-23 10:24:13 +00:00
|
|
|
#ifndef SNES9X
|
2011-06-12 06:25:22 +00:00
|
|
|
void SMP::serialize(serializer &s) {
|
|
|
|
Processor::serialize(s);
|
|
|
|
|
|
|
|
s.array(apuram, 64 * 1024);
|
|
|
|
|
|
|
|
s.integer(opcode_number);
|
|
|
|
s.integer(opcode_cycle);
|
|
|
|
|
|
|
|
s.integer(regs.pc);
|
|
|
|
s.integer(regs.sp);
|
|
|
|
s.integer(regs.a);
|
|
|
|
s.integer(regs.x);
|
|
|
|
s.integer(regs.y);
|
|
|
|
|
|
|
|
s.integer(regs.p.n);
|
|
|
|
s.integer(regs.p.v);
|
|
|
|
s.integer(regs.p.p);
|
|
|
|
s.integer(regs.p.b);
|
|
|
|
s.integer(regs.p.h);
|
|
|
|
s.integer(regs.p.i);
|
|
|
|
s.integer(regs.p.z);
|
|
|
|
s.integer(regs.p.c);
|
|
|
|
|
|
|
|
s.integer(status.iplrom_enable);
|
|
|
|
|
|
|
|
s.integer(status.dsp_addr);
|
|
|
|
|
|
|
|
s.integer(status.ram00f8);
|
|
|
|
s.integer(status.ram00f9);
|
|
|
|
|
|
|
|
s.integer(timer0.enable);
|
|
|
|
s.integer(timer0.target);
|
|
|
|
s.integer(timer0.stage1_ticks);
|
|
|
|
s.integer(timer0.stage2_ticks);
|
|
|
|
s.integer(timer0.stage3_ticks);
|
|
|
|
|
|
|
|
s.integer(timer1.enable);
|
|
|
|
s.integer(timer1.target);
|
|
|
|
s.integer(timer1.stage1_ticks);
|
|
|
|
s.integer(timer1.stage2_ticks);
|
|
|
|
s.integer(timer1.stage3_ticks);
|
|
|
|
|
|
|
|
s.integer(timer2.enable);
|
|
|
|
s.integer(timer2.target);
|
|
|
|
|
|
|
|
s.integer(timer2.stage1_ticks);
|
|
|
|
s.integer(timer2.stage2_ticks);
|
|
|
|
s.integer(timer2.stage3_ticks);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
SMP::SMP() {
|
|
|
|
apuram = new uint8[64 * 1024];
|
|
|
|
}
|
|
|
|
|
|
|
|
SMP::~SMP() {
|
2017-10-13 20:24:30 +00:00
|
|
|
delete[] apuram;
|
2011-06-12 06:25:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|