2013-10-26 08:54:06 +00:00
|
|
|
#include "smp.hpp"
|
|
|
|
|
|
|
|
#include <cstdlib>
|
|
|
|
|
|
|
|
#define SMP_CPP
|
|
|
|
namespace SuperFamicom {
|
|
|
|
|
|
|
|
#include "memory.cpp"
|
|
|
|
#include "timing.cpp"
|
|
|
|
|
|
|
|
void SMP::step(unsigned clocks) {
|
|
|
|
clock += clocks;
|
2013-10-26 18:12:50 +00:00
|
|
|
dsp.clock -= clocks * dsp_clock_step;
|
2013-10-26 08:54:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SMP::synchronize_dsp() {
|
2013-10-27 22:00:13 +00:00
|
|
|
while(dsp.clock < 0) dsp.enter();
|
2013-10-26 08:54:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SMP::enter() {
|
2013-10-27 22:00:13 +00:00
|
|
|
while(sample_buffer < sample_buffer_end) {
|
|
|
|
clock -= (sample_buffer_end - sample_buffer) * 24 * 16;
|
|
|
|
while(status.clock_speed != 2 && clock < 0) op_step();
|
|
|
|
if(status.clock_speed == 2) step(-clock);
|
2013-10-26 08:54:06 +00:00
|
|
|
synchronize_dsp();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void SMP::render(int16_t * buffer, unsigned count) {
|
2013-10-27 22:00:13 +00:00
|
|
|
while (count > 4096) {
|
|
|
|
sample_buffer = buffer;
|
|
|
|
sample_buffer_end = buffer + 4096;
|
|
|
|
buffer += 4096;
|
|
|
|
count -= 4096;
|
|
|
|
enter();
|
|
|
|
}
|
2013-10-26 08:54:06 +00:00
|
|
|
sample_buffer = buffer;
|
|
|
|
sample_buffer_end = buffer + count;
|
|
|
|
enter();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SMP::skip(unsigned count) {
|
|
|
|
while (count > 4096) {
|
|
|
|
sample_buffer = 0;
|
2013-10-26 17:30:36 +00:00
|
|
|
sample_buffer_end = ((const int16_t *)0) + 4096;
|
2013-10-26 08:54:06 +00:00
|
|
|
count -= 4096;
|
|
|
|
enter();
|
|
|
|
}
|
|
|
|
sample_buffer = 0;
|
2013-10-26 17:30:36 +00:00
|
|
|
sample_buffer_end = ((const int16_t *)0) + count;
|
2013-10-26 08:54:06 +00:00
|
|
|
enter();
|
|
|
|
}
|
|
|
|
|
2013-10-27 22:00:13 +00:00
|
|
|
bool SMP::sample(int16_t left, int16_t right) {
|
|
|
|
if ( sample_buffer_end - sample_buffer < 2 ) return false;
|
2013-10-26 17:30:36 +00:00
|
|
|
if ( sample_buffer > ((const int16_t *)0) + 4096 ) {
|
2013-10-27 22:00:13 +00:00
|
|
|
*sample_buffer++ = left;
|
|
|
|
*sample_buffer++ = right;
|
2013-10-26 08:54:06 +00:00
|
|
|
}
|
2013-10-27 22:00:13 +00:00
|
|
|
else {
|
2013-10-26 08:54:06 +00:00
|
|
|
sample_buffer += 2;
|
|
|
|
}
|
2013-10-27 22:00:13 +00:00
|
|
|
return true;
|
2013-10-26 08:54:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SMP::power() {
|
|
|
|
//targets not initialized/changed upon reset
|
|
|
|
timer0.target = 0;
|
|
|
|
timer1.target = 0;
|
|
|
|
timer2.target = 0;
|
|
|
|
|
|
|
|
dsp.power();
|
|
|
|
|
|
|
|
reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SMP::reset() {
|
|
|
|
regs.pc = 0xffc0;
|
|
|
|
regs.a = 0x00;
|
|
|
|
regs.x = 0x00;
|
|
|
|
regs.y = 0x00;
|
|
|
|
regs.s = 0xef;
|
|
|
|
regs.p = 0x02;
|
|
|
|
|
|
|
|
for(auto& n : apuram) n = rand();
|
|
|
|
apuram[0x00f4] = 0x00;
|
|
|
|
apuram[0x00f5] = 0x00;
|
|
|
|
apuram[0x00f6] = 0x00;
|
|
|
|
apuram[0x00f7] = 0x00;
|
|
|
|
|
|
|
|
status.clock_counter = 0;
|
|
|
|
status.dsp_counter = 0;
|
|
|
|
status.timer_step = 3;
|
|
|
|
|
|
|
|
//$00f0
|
|
|
|
status.clock_speed = 0;
|
|
|
|
status.timer_speed = 0;
|
|
|
|
status.timers_enable = true;
|
|
|
|
status.ram_disable = false;
|
|
|
|
status.ram_writable = true;
|
|
|
|
status.timers_disable = false;
|
|
|
|
|
|
|
|
//$00f1
|
|
|
|
status.iplrom_enable = true;
|
|
|
|
|
|
|
|
//$00f2
|
|
|
|
status.dsp_addr = 0x00;
|
|
|
|
|
|
|
|
//$00f8,$00f9
|
|
|
|
status.ram00f8 = 0x00;
|
|
|
|
status.ram00f9 = 0x00;
|
|
|
|
|
|
|
|
timer0.stage0_ticks = 0;
|
|
|
|
timer1.stage0_ticks = 0;
|
|
|
|
timer2.stage0_ticks = 0;
|
|
|
|
|
|
|
|
timer0.stage1_ticks = 0;
|
|
|
|
timer1.stage1_ticks = 0;
|
|
|
|
timer2.stage1_ticks = 0;
|
|
|
|
|
|
|
|
timer0.stage2_ticks = 0;
|
|
|
|
timer1.stage2_ticks = 0;
|
|
|
|
timer2.stage2_ticks = 0;
|
|
|
|
|
|
|
|
timer0.stage3_ticks = 0;
|
|
|
|
timer1.stage3_ticks = 0;
|
|
|
|
timer2.stage3_ticks = 0;
|
|
|
|
|
|
|
|
timer0.current_line = 0;
|
|
|
|
timer1.current_line = 0;
|
|
|
|
timer2.current_line = 0;
|
|
|
|
|
|
|
|
timer0.enable = false;
|
|
|
|
timer1.enable = false;
|
|
|
|
timer2.enable = false;
|
|
|
|
|
|
|
|
dsp.reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
SMP::SMP() : dsp( *this ), timer0( *this ), timer1( *this ), timer2( *this ), clock( 0 ) {
|
|
|
|
for(auto& byte : iplrom) byte = 0;
|
2013-10-29 04:55:13 +00:00
|
|
|
set_sfm_queue(0, 0, 0);
|
2013-10-26 18:12:50 +00:00
|
|
|
set_tempo(1.0);
|
2013-10-26 08:54:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SMP::~SMP() {
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|