170 lines
2.6 KiB
C++
Executable File
170 lines
2.6 KiB
C++
Executable File
|
|
#include "types.h"
|
|
#include "sh4_interrupts.h"
|
|
#include "sh4_core.h"
|
|
#include "sh4_sched.h"
|
|
#include "oslib/oslib.h"
|
|
|
|
|
|
//sh4 scheduler
|
|
|
|
/*
|
|
|
|
register handler
|
|
request callback at time
|
|
|
|
single fire events only
|
|
|
|
sh4_sched_register(id)
|
|
sh4_sched_request(id, in_cycles)
|
|
|
|
sh4_sched_now()
|
|
|
|
*/
|
|
u64 sh4_sched_ffb;
|
|
u32 sh4_sched_intr;
|
|
|
|
|
|
vector<sched_list> list;
|
|
|
|
int sh4_sched_next_id=-1;
|
|
|
|
u32 sh4_sched_remaining(int id, u32 reference)
|
|
{
|
|
if (list[id].end != -1)
|
|
{
|
|
return list[id].end - reference;
|
|
}
|
|
else
|
|
{
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
u32 sh4_sched_remaining(int id)
|
|
{
|
|
return sh4_sched_remaining(id, sh4_sched_now());
|
|
}
|
|
|
|
void sh4_sched_ffts()
|
|
{
|
|
u32 diff=-1;
|
|
int slot=-1;
|
|
|
|
for (size_t i=0;i<list.size();i++)
|
|
{
|
|
if (sh4_sched_remaining(i)<diff)
|
|
{
|
|
slot=i;
|
|
diff=sh4_sched_remaining(i);
|
|
}
|
|
}
|
|
|
|
sh4_sched_ffb-=Sh4cntx.sh4_sched_next;
|
|
|
|
sh4_sched_next_id=slot;
|
|
if (slot!=-1)
|
|
{
|
|
Sh4cntx.sh4_sched_next=diff;
|
|
}
|
|
else
|
|
{
|
|
Sh4cntx.sh4_sched_next=SH4_MAIN_CLOCK;
|
|
}
|
|
|
|
sh4_sched_ffb+=Sh4cntx.sh4_sched_next;
|
|
}
|
|
|
|
int sh4_sched_register(int tag, sh4_sched_callback* ssc)
|
|
{
|
|
sched_list t={ssc,tag,-1,-1};
|
|
|
|
list.push_back(t);
|
|
|
|
return list.size()-1;
|
|
}
|
|
|
|
/*
|
|
Return current cycle count, in 32 bits (wraps after 21 dreamcast seconds)
|
|
*/
|
|
u32 sh4_sched_now()
|
|
{
|
|
return sh4_sched_ffb-Sh4cntx.sh4_sched_next;
|
|
}
|
|
|
|
/*
|
|
Return current cycle count, in 64 bits (effectivelly never wraps)
|
|
*/
|
|
u64 sh4_sched_now64()
|
|
{
|
|
return sh4_sched_ffb-Sh4cntx.sh4_sched_next;
|
|
}
|
|
void sh4_sched_request(int id, int cycles)
|
|
{
|
|
verify(cycles== -1 || (cycles >= 0 && cycles <= SH4_MAIN_CLOCK));
|
|
|
|
list[id].start=sh4_sched_now();
|
|
|
|
if (cycles == -1) {
|
|
list[id].end = -1;
|
|
}
|
|
else
|
|
{
|
|
list[id].end = list[id].start + cycles;
|
|
if (list[id].end == -1)
|
|
list[id].end++;
|
|
}
|
|
|
|
sh4_sched_ffts();
|
|
}
|
|
|
|
int sh4_sched_elapsed(int id)
|
|
{
|
|
if (list[id].end!=-1)
|
|
{
|
|
int rv=sh4_sched_now()-list[id].start;
|
|
list[id].start=sh4_sched_now();
|
|
return rv;
|
|
}
|
|
else
|
|
return -1;
|
|
}
|
|
|
|
void handle_cb(int id)
|
|
{
|
|
int remain=list[id].end-list[id].start;
|
|
int elapsd=sh4_sched_elapsed(id);
|
|
int jitter=elapsd-remain;
|
|
|
|
list[id].end=-1;
|
|
int re_sch=list[id].cb(list[id].tag,remain,jitter);
|
|
|
|
if (re_sch>0) sh4_sched_request(id,re_sch-jitter);
|
|
}
|
|
|
|
void sh4_sched_tick(int cycles)
|
|
{
|
|
/*
|
|
Sh4cntx.sh4_sched_time+=cycles;
|
|
Sh4cntx.sh4_sched_next-=cycles;
|
|
*/
|
|
|
|
if (Sh4cntx.sh4_sched_next<0)
|
|
{
|
|
u32 fztime=sh4_sched_now()-cycles;
|
|
sh4_sched_intr++;
|
|
if (sh4_sched_next_id!=-1)
|
|
{
|
|
for (int i=0;i<list.size();i++)
|
|
{
|
|
int remaining = sh4_sched_remaining(i, fztime);
|
|
verify(remaining >= 0 || remaining == -1);
|
|
if (remaining >= 0 && remaining <= (u32)cycles) {
|
|
handle_cb(i);
|
|
}
|
|
}
|
|
}
|
|
sh4_sched_ffts();
|
|
}
|
|
}
|