ref: b07a6e33bd1d0653709e4789ee7463a6ff102f67
dir: /sys/src/9/port/alarm.c/
#include "u.h" #include "../port/lib.h" #include "mem.h" #include "dat.h" #include "fns.h" static Alarms alarms; static Rendez alarmr; void alarmkproc(void*) { Proc *rp; ulong now, when; while(waserror()) ; for(;;){ now = MACHP(0)->ticks; qlock(&alarms); for(rp = alarms.head; rp != nil; rp = rp->palarm){ if((when = rp->alarm) == 0) continue; if((long)(now - when) < 0) break; if(!canqlock(&rp->debug)) break; if(rp->alarm != 0){ static Note alarm = { "alarm", NUser, 1, }; incref(&alarm); pushnote(rp, &alarm); rp->alarm = 0; } qunlock(&rp->debug); } alarms.head = rp; qunlock(&alarms); sleep(&alarmr, return0, 0); } } /* * called every clock tick on cpu0 */ void checkalarms(void) { Proc *p; ulong now, when; p = alarms.head; if(p != nil){ now = MACHP(0)->ticks; when = p->alarm; if(when == 0 || (long)(now - when) >= 0) wakeup(&alarmr); } } ulong procalarm(ulong time) { Proc **l, *f; ulong when, old; when = MACHP(0)->ticks; old = up->alarm; if(old) { old -= when; if((long)old > 0) old = tk2ms(old); else old = 0; } if(time == 0) { up->alarm = 0; return old; } when += ms2tk(time); if(when == 0) when = 1; qlock(&alarms); l = &alarms.head; for(f = *l; f; f = f->palarm) { if(up == f){ *l = f->palarm; break; } l = &f->palarm; } l = &alarms.head; for(f = *l; f; f = f->palarm) { time = f->alarm; if(time != 0 && (long)(time - when) >= 0) break; l = &f->palarm; } up->palarm = f; *l = up; up->alarm = when; qunlock(&alarms); return old; }