#include #include #include #include #include static uint64_t uptime = 0, goal = ~0; static Proc *scheduled = NULL; uint64_t uptime_ms(void) { return uptime; } static void update_goal(void) { goal = scheduled ? scheduled->waits4timer.goal : ~(uint64_t)0; } void pit_irq(void) { // TODO inefficient - getting here executes a lot of code which could just be a few lines of asm uptime++; if (uptime < goal) return; Proc *p = scheduled; assert(p); scheduled = p->waits4timer.next; proc_setstate(p, PS_RUNNING); update_goal(); } void timer_schedule(Proc *p, uint64_t time) { proc_setstate(p, PS_WAITS4TIMER); p->waits4timer.goal = time; Proc **slot = &scheduled; while (*slot && (*slot)->waits4timer.goal <= time) { assert((*slot)->state == PS_WAITS4TIMER); slot = &(*slot)->waits4timer.next; } p->waits4timer.next = *slot; *slot = p; update_goal(); } void timer_deschedule(Proc *p) { assert(p->state == PS_WAITS4TIMER); Proc **slot = &scheduled; while (*slot && *slot != p) slot = &(*slot)->waits4timer.next; assert(*slot); *slot = p->waits4timer.next; proc_setstate(p, PS_RUNNING); update_goal(); } void timer_init(void) { irq_fn[IRQ_PIT] = pit_irq; }