summaryrefslogtreecommitdiff
path: root/sys/arch/macppc
diff options
context:
space:
mode:
authorDale Rahn <drahn@cvs.openbsd.org>2003-07-14 18:56:13 +0000
committerDale Rahn <drahn@cvs.openbsd.org>2003-07-14 18:56:13 +0000
commite7ac75feb95c3c36be55742473e6ec11e821e1e3 (patch)
tree4be6cd40c8b54a3b50e0fef03cd5e73159e7a4fa /sys/arch/macppc
parent657f19a3a9e8853e048179c1f3e98f52a299c47e (diff)
'dual' clock support for macppc, this allows the stat clock to run
at a rate independant of the HZ clock. when profiling is enabled stat clock now runs at 1000/s instead of the normal 100/s. Other improvements to the ppc clock handling and KNF/ansi.
Diffstat (limited to 'sys/arch/macppc')
-rw-r--r--sys/arch/macppc/macppc/clock.c206
-rw-r--r--sys/arch/macppc/macppc/locore.S4
-rw-r--r--sys/arch/macppc/macppc/machdep.c4
3 files changed, 154 insertions, 60 deletions
diff --git a/sys/arch/macppc/macppc/clock.c b/sys/arch/macppc/macppc/clock.c
index c49db1c3b52..f89cba9f989 100644
--- a/sys/arch/macppc/macppc/clock.c
+++ b/sys/arch/macppc/macppc/clock.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: clock.c,v 1.8 2003/07/08 23:05:20 drahn Exp $ */
+/* $OpenBSD: clock.c,v 1.9 2003/07/14 18:56:12 drahn Exp $ */
/* $NetBSD: clock.c,v 1.1 1996/09/30 16:34:40 ws Exp $ */
/*
@@ -71,8 +71,6 @@ typedef int (clock_read_t)(int *sec, int *min, int *hour, int *day,
typedef int (time_read_t)(u_int32_t *sec);
typedef int (time_write_t)(u_int32_t sec);
-int power4e_getclock(int *, int *, int *, int *, int *, int *);
-
clock_read_t *clock_read = NULL;
time_read_t *time_read = NULL;
time_write_t *time_write = NULL;
@@ -80,6 +78,14 @@ time_write_t *time_write = NULL;
static u_int32_t chiptotime(int sec, int min, int hour, int day, int mon,
int year);
+/* event tracking variables, when the next events of each time should occur */
+u_int64_t nexttimerevent, nextstatevent;
+
+/* vars for stats */
+int statint;
+u_int32_t statvar;
+u_int32_t statmin;
+
/*
* For now we let the machine run with boot time, not changing the clock
* at inittodr at all.
@@ -90,8 +96,7 @@ static u_int32_t chiptotime(int sec, int min, int hour, int day, int mon,
/* ARGSUSED */
void
-inittodr(base)
- time_t base;
+inittodr(time_t base)
{
int sec, min, hour, day, mon, year;
@@ -119,7 +124,7 @@ inittodr(base)
} else {
/* force failure */
time.tv_sec = 0;
- }
+ }
if (time.tv_sec == 0) {
printf("WARNING: unable to get date/time");
@@ -162,8 +167,7 @@ const short dayyr[12] =
{ 0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334 };
static u_int32_t
-chiptotime(sec, min, hour, day, mon, year)
- int sec, min, hour, day, mon, year;
+chiptotime(int sec, int min, int hour, int day, int mon, int year)
{
int days, yr;
@@ -192,7 +196,7 @@ chiptotime(sec, min, hour, day, mon, year)
* Similar to the above
*/
void
-resettodr()
+resettodr(void)
{
struct timeval curtime = time;
if (time_write != NULL) {
@@ -204,16 +208,16 @@ resettodr()
}
}
+volatile int tickspending, statspending;
void
-decr_intr(frame)
- struct clockframe *frame;
+decr_intr(struct clockframe *frame)
{
- int msr;
u_int64_t tb;
- int32_t tick;
+ u_int64_t nextevent;
int nticks;
- int pri;
+ int nstats;
+ int s;
/*
* Check whether we are initialized.
@@ -221,62 +225,132 @@ decr_intr(frame)
if (!ticks_per_intr)
return;
- intrcnt[PPC_CLK_IRQ]++;
/*
* Based on the actual time delay since the last decrementer reload,
* we arrange for earlier interrupt next time.
*/
+
tb = ppc_mftb();
- tick = ppc_mfdec();
- for (nticks = 0; tick < 0; nticks++)
- tick += ticks_per_intr;
+ for (nticks = 0; nexttimerevent <= tb; nticks++)
+ nexttimerevent += ticks_per_intr;
+
+ for (nstats = 0; nextstatevent <= tb; nstats++) {
+ int r;
+ do {
+ r = random() & (statvar -1);
+ } while (r == 0); /* random == 0 not allowed */
+ nextstatevent += statmin + r;
+ }
+
+ /* only count timer ticks for CLK_IRQ */
+ intrcnt[PPC_CLK_IRQ] += nticks;
+ intrcnt[PPC_STAT_IRQ] += nstats;
+
+ if (nexttimerevent < nextstatevent)
+ nextevent = nexttimerevent;
+ else
+ nextevent = nextstatevent;
+
+ /*
+ * Need to work about the near constant skew this introduces???
+ * reloading tb here could cause a missed tick.
+ */
+ ppc_mtdec(nextevent - tb);
- ppc_mtdec(tick);
/*
* lasttb is used during microtime. Set it to the virtual
* start of this tick interval.
*/
- lasttb = tb + tick - ticks_per_intr;
+ lasttb = nexttimerevent - ticks_per_intr;
- pri = splclock();
-
- if (pri & SPL_CLOCK) {
+ if (cpl & SPL_CLOCK) {
tickspending += nticks;
+ statspending += nstats;
} else {
- nticks += tickspending;
- tickspending = 0;
- /*
- * Reenable interrupts
- */
- asm volatile ("mfmsr %0; ori %0, %0, %1; mtmsr %0"
- : "=r"(msr) : "K"(PSL_EE));
-
- /*
- * Do standard timer interrupt stuff.
- * Do softclock stuff only on the last iteration.
- */
- frame->pri = pri | SINT_CLOCK;
- while (--nticks > 0)
- hardclock(frame);
- frame->pri = pri;
- hardclock(frame);
+ do {
+ nticks += tickspending;
+ nstats += statspending;
+ tickspending = 0;
+ statspending = 0;
+
+ s = splclock();
+
+ /*
+ * Reenable interrupts
+ */
+ ppc_intr_enable(1);
+
+ /*
+ * Do standard timer interrupt stuff.
+ * Do softclock stuff only on the last iteration.
+ */
+ frame->pri = s | SINT_CLOCK;
+ if (nticks > 1)
+ while (--nticks > 1)
+ hardclock(frame);
+
+ frame->pri = s;
+ if (nticks)
+ hardclock(frame);
+
+ while (nstats-- > 0)
+ statclock(frame);
+
+ splx(s);
+ (void) ppc_intr_disable();
+
+ /* if a tick has occurred while dealing with these,
+ * service it now, do not delay until the next tick.
+ */
+ nstats = 0;
+ nticks = 0;
+ } while (tickspending != 0 || statspending != 0);
}
- splx(pri);
}
void
cpu_initclocks()
{
- int s;
- s = ppc_intr_disable();
+ int intrstate;
+ int r;
+ int minint;
+ u_int64_t nextevent;
+
+ intrstate = ppc_intr_disable();
+
+ stathz = 100;
+ profhz = 1000; /* must be a multiple of stathz */
+
+ /* init secondary clock to stathz */
+ statint = ticks_per_sec / stathz;
+ statvar = 0x40000000; /* really big power of two */
+ /* find largest 2^n which is nearly smaller than statint/2 */
+ minint = statint / 2 + 100;
+ while (statvar > minint)
+ statvar >>= 1;
+
+ statmin = statint - (statvar >> 1);
+
+
lasttb = ppc_mftb();
- ppc_mtdec(ticks_per_intr);
- ppc_intr_enable(s);
+ nexttimerevent = lasttb + ticks_per_intr;
+ do {
+ r = random() & (statvar -1);
+ } while (r == 0); /* random == 0 not allowed */
+ nextstatevent = lasttb + statmin + r;
+
+ if (nexttimerevent < nextstatevent)
+ nextevent = nexttimerevent;
+ else
+ nextevent = nextstatevent;
+
+ ppc_mtdec(nextevent-lasttb);
+ ppc_intr_enable(intrstate);
}
void
-calc_delayconst()
+calc_delayconst(void)
{
int qhandle, phandle;
char name[32];
@@ -289,7 +363,7 @@ calc_delayconst()
if (OF_getprop(qhandle, "device_type", name, sizeof name) >= 0
&& !strcmp(name, "cpu")
&& OF_getprop(qhandle, "timebase-frequency",
- & ticks_per_sec, sizeof ticks_per_sec) >= 0) {
+ &ticks_per_sec, sizeof ticks_per_sec) >= 0) {
/*
* Should check for correct CPU here? XXX
*/
@@ -297,6 +371,10 @@ calc_delayconst()
ns_per_tick = 1000000000 / ticks_per_sec;
ticks_per_intr = ticks_per_sec / hz;
ppc_intr_enable(s);
+ printf("clock error tick %x tick/sec %x hz %x err %x\n",
+ ticks_per_intr, ticks_per_sec, hz,
+ ticks_per_sec - ticks_per_intr * hz);
+
break;
}
if ((phandle = OF_child(qhandle)))
@@ -307,6 +385,7 @@ calc_delayconst()
qhandle = OF_parent(qhandle);
}
}
+
if (!phandle)
panic("no cpu node");
}
@@ -315,8 +394,7 @@ calc_delayconst()
* Fill in *tvp with current time with microsecond resolution.
*/
void
-microtime(tvp)
- struct timeval *tvp;
+microtime(struct timeval *tvp)
{
u_int64_t tb;
u_int32_t ticks;
@@ -339,8 +417,7 @@ microtime(tvp)
* Wait for about n microseconds (us) (at least!).
*/
void
-delay(n)
- unsigned n;
+delay(unsigned n)
{
u_int64_t tb;
u_int32_t tbh, tbl, scratch;
@@ -348,20 +425,35 @@ delay(n)
tb = ppc_mftb();
tb += (n * 1000 + ns_per_tick - 1) / ns_per_tick;
tbh = tb >> 32;
- tbl = tb;
+ tbl = (u_int32_t)tb;
asm ("1: mftbu %0; cmplw %0,%1; blt 1b; bgt 2f;"
" mftb %0; cmplw %0,%2; blt 1b; 2:"
:: "r"(scratch), "r"(tbh), "r"(tbl));
-
- tb = ppc_mftb();
}
/*
* Nothing to do.
*/
void
-setstatclockrate(arg)
- int arg;
+setstatclockrate(int newhz)
{
- /* Do nothing */
+ int minint;
+ int intrstate;
+
+ intrstate = ppc_intr_disable();
+
+ statint = ticks_per_sec / newhz;
+ statvar = 0x40000000; /* really big power of two */
+ /* find largest 2^n which is nearly smaller than statint/2 */
+ minint = statint / 2 + 100;
+ while (statvar > minint)
+ statvar >>= 1;
+
+ statmin = statint - (statvar >> 1);
+ ppc_intr_enable(intrstate);
+
+ /*
+ * XXX this allows the next stat timer to occur then it switches
+ * to the new frequency. Rather than switching instantly.
+ */
}
diff --git a/sys/arch/macppc/macppc/locore.S b/sys/arch/macppc/macppc/locore.S
index cf3ce0f7ceb..edffd1f6fcd 100644
--- a/sys/arch/macppc/macppc/locore.S
+++ b/sys/arch/macppc/macppc/locore.S
@@ -1,4 +1,4 @@
-/* $OpenBSD: locore.S,v 1.8 2002/11/17 20:36:53 drahn Exp $ */
+/* $OpenBSD: locore.S,v 1.9 2003/07/14 18:56:12 drahn Exp $ */
/* $NetBSD: locore.S,v 1.2 1996/10/16 19:33:09 ws Exp $ */
/*
@@ -1313,6 +1313,7 @@ _C_LABEL(intrnames):
.string "irq56" "irq57" "irq58" "irq59"
.string "irq60" "irq61" "irq62" "irq63"
.string "clock"
+ .string "stat"
.space 512
_C_LABEL(eintrnames):
.align 4
@@ -1338,6 +1339,7 @@ _C_LABEL(intrcnt):
.long 0,0,0,0
.long 0,0,0,0
.long 0
+ .long 0
_C_LABEL(eintrcnt):
#ifdef DDB
diff --git a/sys/arch/macppc/macppc/machdep.c b/sys/arch/macppc/macppc/machdep.c
index 3513c6fefab..ab270e65b2a 100644
--- a/sys/arch/macppc/macppc/machdep.c
+++ b/sys/arch/macppc/macppc/machdep.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: machdep.c,v 1.53 2003/07/02 21:30:13 drahn Exp $ */
+/* $OpenBSD: machdep.c,v 1.54 2003/07/14 18:56:12 drahn Exp $ */
/* $NetBSD: machdep.c,v 1.4 1996/10/16 19:33:11 ws Exp $ */
/*
@@ -833,7 +833,7 @@ dumpsys()
printf("dumpsys: TBD\n");
}
-volatile int cpl, ipending, astpending, tickspending;
+volatile int cpl, ipending, astpending;
int imask[7];
/*