mirror of https://github.com/bsnes-emu/bsnes.git
195 lines
3.4 KiB
C++
195 lines
3.4 KiB
C++
#include <sfc/sfc.hpp>
|
|
|
|
namespace SuperFamicom {
|
|
|
|
#include "memory.cpp"
|
|
#include "time.cpp"
|
|
#include "serialization.cpp"
|
|
EpsonRTC epsonrtc;
|
|
|
|
auto EpsonRTC::Enter() -> void {
|
|
while(true) scheduler.synchronize(), epsonrtc.main();
|
|
}
|
|
|
|
auto EpsonRTC::main() -> void {
|
|
if(wait) { if(--wait == 0) ready = 1; }
|
|
|
|
clocks++;
|
|
if((clocks & ~0x00ff) == 0) roundSeconds(); //125 microseconds
|
|
if((clocks & ~0x3fff) == 0) duty(); //1/128th second
|
|
if((clocks & ~0x7fff) == 0) irq(0); //1/64th second
|
|
if(clocks == 0) { //1 second
|
|
seconds++;
|
|
irq(1);
|
|
if(seconds % 60 == 0) irq(2); //1 minute
|
|
if(seconds % 1440 == 0) irq(3), seconds = 0; //1 hour
|
|
tick();
|
|
}
|
|
|
|
step(1);
|
|
synchronize(cpu);
|
|
}
|
|
|
|
auto EpsonRTC::initialize() -> void {
|
|
secondlo = 0;
|
|
secondhi = 0;
|
|
batteryfailure = 1;
|
|
|
|
minutelo = 0;
|
|
minutehi = 0;
|
|
resync = 0;
|
|
|
|
hourlo = 0;
|
|
hourhi = 0;
|
|
meridian = 0;
|
|
|
|
daylo = 0;
|
|
dayhi = 0;
|
|
dayram = 0;
|
|
|
|
monthlo = 0;
|
|
monthhi = 0;
|
|
monthram = 0;
|
|
|
|
yearlo = 0;
|
|
yearhi = 0;
|
|
|
|
weekday = 0;
|
|
|
|
hold = 0;
|
|
calendar = 0;
|
|
irqflag = 0;
|
|
roundseconds = 0;
|
|
|
|
irqmask = 0;
|
|
irqduty = 0;
|
|
irqperiod = 0;
|
|
|
|
pause = 0;
|
|
stop = 0;
|
|
atime = 0;
|
|
test = 0;
|
|
}
|
|
|
|
auto EpsonRTC::power() -> void {
|
|
create(EpsonRTC::Enter, 32'768 * 64);
|
|
|
|
clocks = 0;
|
|
seconds = 0;
|
|
|
|
chipselect = 0;
|
|
state = State::Mode;
|
|
offset = 0;
|
|
wait = 0;
|
|
ready = 0;
|
|
holdtick = 0;
|
|
}
|
|
|
|
auto EpsonRTC::synchronize(uint64 timestamp) -> void {
|
|
time_t systime = timestamp;
|
|
tm* timeinfo = localtime(&systime);
|
|
|
|
uint second = min(59, timeinfo->tm_sec);
|
|
secondlo = second % 10;
|
|
secondhi = second / 10;
|
|
|
|
uint minute = timeinfo->tm_min;
|
|
minutelo = minute % 10;
|
|
minutehi = minute / 10;
|
|
|
|
uint hour = timeinfo->tm_hour;
|
|
if(atime) {
|
|
hourlo = hour % 10;
|
|
hourhi = hour / 10;
|
|
} else {
|
|
meridian = hour >= 12;
|
|
hour %= 12;
|
|
if(hour == 0) hour = 12;
|
|
hourlo = hour % 10;
|
|
hourhi = hour / 10;
|
|
}
|
|
|
|
uint day = timeinfo->tm_mday;
|
|
daylo = day % 10;
|
|
dayhi = day / 10;
|
|
|
|
uint month = 1 + timeinfo->tm_mon;
|
|
monthlo = month % 10;
|
|
monthhi = month / 10;
|
|
|
|
uint year = timeinfo->tm_year % 100;
|
|
yearlo = year % 10;
|
|
yearhi = year / 10;
|
|
|
|
weekday = timeinfo->tm_wday;
|
|
|
|
resync = true; //alert program that time has changed
|
|
}
|
|
|
|
auto EpsonRTC::read(uint24 addr, uint8 data) -> uint8 {
|
|
cpu.synchronize(*this);
|
|
addr &= 3;
|
|
|
|
if(addr == 0) {
|
|
return chipselect;
|
|
}
|
|
|
|
if(addr == 1) {
|
|
if(chipselect != 1) return 0;
|
|
if(ready == 0) return 0;
|
|
if(state == State::Write) return mdr;
|
|
if(state != State::Read) return 0;
|
|
ready = 0;
|
|
wait = 8;
|
|
return rtcRead(offset++);
|
|
}
|
|
|
|
if(addr == 2) {
|
|
return ready << 7;
|
|
}
|
|
|
|
return data;
|
|
}
|
|
|
|
auto EpsonRTC::write(uint24 addr, uint8 data) -> void {
|
|
cpu.synchronize(*this);
|
|
addr &= 3, data &= 15;
|
|
|
|
if(addr == 0) {
|
|
chipselect = data;
|
|
if(chipselect != 1) rtcReset();
|
|
ready = 1;
|
|
}
|
|
|
|
if(addr == 1) {
|
|
if(chipselect != 1) return;
|
|
if(ready == 0) return;
|
|
|
|
if(state == State::Mode) {
|
|
if(data != 0x03 && data != 0x0c) return;
|
|
state = State::Seek;
|
|
ready = 0;
|
|
wait = 8;
|
|
mdr = data;
|
|
}
|
|
|
|
else if(state == State::Seek) {
|
|
if(mdr == 0x03) state = State::Write;
|
|
if(mdr == 0x0c) state = State::Read;
|
|
offset = data;
|
|
ready = 0;
|
|
wait = 8;
|
|
mdr = data;
|
|
}
|
|
|
|
else if(state == State::Write) {
|
|
rtcWrite(offset++, data);
|
|
ready = 0;
|
|
wait = 8;
|
|
mdr = data;
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|