ref: dd16eef8365cb0b9e4805c23291aadcaa596dd87
dir: /os/sa1110/clock.c/
#include "u.h" #include "../port/lib.h" #include "mem.h" #include "dat.h" #include "fns.h" #include "io.h" #include "ureg.h" static ulong timer_incr[4] = { 0, 0, 0, -1 }; typedef struct Clock0link Clock0link; typedef struct Clock0link { void (*clock)(void); Clock0link* link; } Clock0link; static Clock0link *clock0link; static Lock clock0lock; static void (*prof_fcn)(Ureg *, int); Timer* addclock0link(void (*clock)(void), int) { Clock0link *lp; if((lp = malloc(sizeof(Clock0link))) == 0){ print("addclock0link: too many links\n"); return nil; } ilock(&clock0lock); lp->clock = clock; lp->link = clock0link; clock0link = lp; iunlock(&clock0lock); return nil; } static void profintr(Ureg *ur, void*) { OstmrReg *ost = OSTMRREG; int t; if((ost->osmr[3] - ost->oscr) < 2*CLOCKFREQ) { /* less than 2 seconds before reset, say something */ setpanic(); clockpoll(); dumpregs(ur); panic("Watchdog timer will expire"); } /* advance the profile clock tick */ ost->osmr[2] += timer_incr[2]; ost->ossr = (1 << 2); /* Clear the SR */ t = 1; while((ost->osmr[2] - ost->oscr) > 0x80000000) { ost->osmr[2] += timer_incr[2]; t++; } if(prof_fcn) prof_fcn(ur, t); } static void clockintr(Ureg*, void*) { Clock0link *lp; int losttick = 0; OstmrReg *ost = OSTMRREG; m->ticks++; ost->osmr[3] = ost->oscr + timer_incr[3]; /* restart the watchdog */ ost->osmr[0] += timer_incr[0]; /* advance the clock tick */ ost->ossr = (1<<0); /* Clear the SR */ while((ost->osmr[0] - ost->oscr) >= 0x80000000) { ost->osmr[0] += timer_incr[0]; losttick++; m->ticks++; } checkalarms(); if(canlock(&clock0lock)){ for(lp = clock0link; lp; lp = lp->link) if(lp->clock) lp->clock(); unlock(&clock0lock); } /* round robin time slice is done by trap.c and proc.c */ } void timerenable( int timer, int Hz, void (*f)(Ureg *, void*), void* a) { OstmrReg *ost = OSTMRREG; char name[KNAMELEN]; if(timer < 0 || timer > 3) return; timer_incr[timer] = CLOCKFREQ/Hz; /* set up freq */ ost->osmr[timer] = ost->oscr+timer_incr[timer]; snprint(name, sizeof(name), "timer%d", timer); intrenable(OSTimerbit(timer), f, a, BusCPU, name); ost->ossr = (1 << timer); /* clear any pending interrupt */ ost->oier |= (1 << timer); /* enable interrupt */ } void timerdisable( int timer ) { OstmrReg *ost = OSTMRREG; if(timer < 0 || timer > 3) return; ost->osmr[timer] = 0; /* clear freq */ ost->oier &= ~(1 << timer); /* disable interrupt */ } void installprof(void (*pf)(Ureg *, int)) { int s; s = splfhi(); prof_fcn = pf; timerenable( 2, HZ+1, profintr, 0); timer_incr[2] = timer_incr[0]+63; /* fine tuning */ splx(s); } void clockinit(void) { OstmrReg *ost = OSTMRREG; m->ticks = 0; /* Set up timer registers */ ost->ossr = 0xf; /* clear all four OSSR trigger bits */ ost->oier = 0; ost->osmr[0] = 0; ost->osmr[1] = 0; ost->osmr[2] = 0; // ost->osmr[3] = 0; timerenable( 0, HZ, clockintr, 0); timer_incr[3] = CLOCKFREQ*10; /* 10 second watchdog */ timer_setwatchdog(timer_incr[3]); timerenable( 2, 1, profintr, 0); /* watch the watchdog */ } void clockpoll(void) { OstmrReg *ost = OSTMRREG; ost->osmr[3] = ost->oscr + timer_incr[3]; /* restart the watchdog */ } void clockcheck(void) { OstmrReg *ost = OSTMRREG; if((ost->osmr[3] - ost->oscr) < CLOCKFREQ) { setpanic(); clockpoll(); dumpstack(); panic("Watchdog timer will expire"); } } // macros for fixed-point math ulong _mularsv(ulong m0, ulong m1, ulong a, ulong s); /* truncated: */ #define FXDPTDIV(a,b,n) ((ulong)(((uvlong)(a) << (n)) / (b))) #define MAXMUL(a,n) ((ulong)((((uvlong)1<<(n))-1)/(a))) #define MULDIV(x,a,b,n) (((x)*FXDPTDIV(a,b,n)) >> (n)) #define MULDIV64(x,a,b,n) ((ulong)_mularsv(x, FXDPTDIV(a,b,n), 0, (n))) /* rounded: */ #define FXDPTDIVR(a,b,n) ((ulong)((((uvlong)(a) << (n))+((b)/2)) / (b))) #define MAXMULR(a,n) ((ulong)((((uvlong)1<<(n))-1)/(a))) #define MULDIVR(x,a,b,n) (((x)*FXDPTDIVR(a,b,n)+(1<<((n)-1))) >> (n)) #define MULDIVR64(x,a,b,n) ((ulong)_mularsv(x, FXDPTDIVR(a,b,n), 1<<((n)-1), (n))) // these routines are all limited to a maximum of 1165 seconds, // due to the wrap-around of the OSTIMER ulong timer_start(void) { return OSTMRREG->oscr; } ulong timer_ticks(ulong t0) { return OSTMRREG->oscr - t0; } int timer_devwait(ulong *adr, ulong mask, ulong val, int ost) { int i; ulong t0 = timer_start(); while((*adr & mask) != val) if(timer_ticks(t0) > ost) return ((*adr & mask) == val) ? 0 : -1; else for(i = 0; i < 10; i++); /* don't pound OSCR too hard! (why not?) */ return 0; } void timer_setwatchdog(int t) { OstmrReg *ost = OSTMRREG; ost->osmr[3] = ost->oscr + t; if(t) { ost->ossr = (1<<3); ost->oier |= (1<<3); ost->ower = 1; } else ost->oier &= ~(1<<3); } void timer_delay(int t) { ulong t0 = timer_start(); while(timer_ticks(t0) < t) ; } ulong us2tmr(int us) { return MULDIV64(us, CLOCKFREQ, 1000000, 24); } int tmr2us(ulong t) { return MULDIV64(t, 1000000, CLOCKFREQ, 24); } void microdelay(int us) { ulong t0 = timer_start(); ulong t = us2tmr(us); while(timer_ticks(t0) <= t) ; } ulong ms2tmr(int ms) { return MULDIV64(ms, CLOCKFREQ, 1000, 20); } int tmr2ms(ulong t) { return MULDIV64(t, 1000, CLOCKFREQ, 32); } void delay(int ms) { ulong t0 = timer_start(); ulong t = ms2tmr(ms); while(timer_ticks(t0) <= t) clockpoll(); } /* * for devbench.c */ vlong archrdtsc(void) { return OSTMRREG->oscr; } ulong archrdtsc32(void) { return OSTMRREG->oscr; } /* * for devkprof.c */ long archkprofmicrosecondspertick(void) { return MS2HZ*1000; } void archkprofenable(int) { /* TO DO */ } uvlong fastticks(uvlong *hz) { if(hz) *hz = HZ; return m->ticks; }