Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/arch/sparc/sparc remove schedintr_4m(), and go back to o...
details: https://anonhg.NetBSD.org/src/rev/9c1bd6fa9dbf
branches: trunk
changeset: 326114:9c1bd6fa9dbf
user: mrg <mrg%NetBSD.org@localhost>
date: Sun Jan 19 00:22:33 2014 +0000
description:
remove schedintr_4m(), and go back to only having the level 14
timer call statclock/schedclock. the randomness for this clock
makes the hardclock() call very jittery.
instead of this, trigger a softintr IPI at level 10 on non-primary
CPUs from the primary CPU's clock interrupt (at level 10.) this
cleans up the generic sparc timer code a little as well.
this makes the time goes backwards problem *much* less frequent,
but i still see it sometimes.
diffstat:
sys/arch/sparc/sparc/timer.c | 20 ++-----
sys/arch/sparc/sparc/timer_sun4m.c | 97 ++++++++++++++++++++++++++-----------
sys/arch/sparc/sparc/timervar.h | 5 +-
3 files changed, 75 insertions(+), 47 deletions(-)
diffs (279 lines):
diff -r d4a80e172235 -r 9c1bd6fa9dbf sys/arch/sparc/sparc/timer.c
--- a/sys/arch/sparc/sparc/timer.c Sat Jan 18 21:27:11 2014 +0000
+++ b/sys/arch/sparc/sparc/timer.c Sun Jan 19 00:22:33 2014 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: timer.c,v 1.31 2013/10/19 19:40:23 mrg Exp $ */
+/* $NetBSD: timer.c,v 1.32 2014/01/19 00:22:33 mrg Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -60,7 +60,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: timer.c,v 1.31 2013/10/19 19:40:23 mrg Exp $");
+__KERNEL_RCSID(0, "$NetBSD: timer.c,v 1.32 2014/01/19 00:22:33 mrg Exp $");
#include <sys/param.h>
#include <sys/kernel.h>
@@ -87,7 +87,7 @@
static struct counter {
volatile u_int *cntreg; /* counter register */
u_int limit; /* limit we count up to */
- u_int offset; /* accumulated offet due to wraps */
+ u_int offset; /* accumulated offset due to wraps */
u_int shift; /* scaling for valid bits */
u_int mask; /* valid bit mask */
} cntr;
@@ -117,12 +117,11 @@
u_int c, res, r;
int s;
-
s = splhigh();
res = c = *ctr->cntreg;
- res &= ~TMR_LIMIT;
+ res &= ~TMR_LIMIT;
if (c != res) {
r = ctr->limit;
@@ -155,7 +154,6 @@
timerattach(volatile int *cntreg, volatile int *limreg)
{
u_int prec = 0, t0;
- void (*sched_intr_fn)(void *);
/*
* Calibrate delay() by tweaking the magic constant
@@ -198,11 +196,11 @@
printf(": delay constant %d, frequency = %" PRIu64 " Hz\n",
timerblurb, counter_timecounter.tc_frequency);
+printf("timer: limit %u shift %u mask %x\n", cntr.limit, cntr.shift, cntr.mask);
#if defined(SUN4) || defined(SUN4C)
if (CPU_ISSUN4 || CPU_ISSUN4C) {
timer_init = timer_init_4;
- sched_intr_fn = schedintr;
level10.ih_fun = clockintr_4;
level14.ih_fun = statintr_4;
cntr.limit = tmr_ustolim(tick);
@@ -211,12 +209,6 @@
#if defined(SUN4M)
if (CPU_ISSUN4M) {
timer_init = timer_init_4m;
-#if defined(MULTIPROCESSOR)
- if (sparc_ncpus > 1)
- sched_intr_fn = schedintr_4m;
- else
-#endif
- sched_intr_fn = schedintr;
level10.ih_fun = clockintr_4m;
level14.ih_fun = statintr_4m;
cntr.limit = tmr_ustolim4m(tick);
@@ -227,7 +219,7 @@
intr_establish(14, 0, &level14, NULL, true);
/* Establish a soft interrupt at a lower level for schedclock */
- sched_cookie = sparc_softintr_establish(IPL_SCHED, sched_intr_fn, NULL);
+ sched_cookie = sparc_softintr_establish(IPL_SCHED, schedintr, NULL);
if (sched_cookie == NULL)
panic("timerattach: cannot establish schedintr");
diff -r d4a80e172235 -r 9c1bd6fa9dbf sys/arch/sparc/sparc/timer_sun4m.c
--- a/sys/arch/sparc/sparc/timer_sun4m.c Sat Jan 18 21:27:11 2014 +0000
+++ b/sys/arch/sparc/sparc/timer_sun4m.c Sun Jan 19 00:22:33 2014 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: timer_sun4m.c,v 1.29 2013/11/16 23:54:01 mrg Exp $ */
+/* $NetBSD: timer_sun4m.c,v 1.30 2014/01/19 00:22:33 mrg Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -58,7 +58,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: timer_sun4m.c,v 1.29 2013/11/16 23:54:01 mrg Exp $");
+__KERNEL_RCSID(0, "$NetBSD: timer_sun4m.c,v 1.30 2014/01/19 00:22:33 mrg Exp $");
#include <sys/param.h>
#include <sys/kernel.h>
@@ -78,6 +78,11 @@
#define counterreg4m cpuinfo.counterreg_4m
/*
+ * SMP hardclock handler.
+ */
+#define IPL_HARDCLOCK 10
+
+/*
* Set up the real-time and statistics clocks.
* Leave stathz 0 only if no alternative timer is available.
*
@@ -96,31 +101,45 @@
icr_si_bic(SINTR_T);
}
-void
-schedintr_4m(void *v)
-{
-
- kpreempt_disable();
#ifdef MULTIPROCESSOR
- /*
- * We call hardclock() here so that we make sure it is called on
- * all CPUs. This function ends up being called on sun4m systems
- * every tick.
- */
- if (!CPU_IS_PRIMARY(curcpu()))
- hardclock(v);
+/*
+ * Handle SMP hardclock() calling for this CPU.
+ */
+static void
+hardclock_ipi(void *cap)
+{
+ int s = splsched();
+
+ hardclock((struct clockframe *)cap);
+ splx(s);
+}
+#endif
- /*
- * The factor 8 is only valid for stathz==100.
- * See also clock.c
- */
- if ((++cpuinfo.ci_schedstate.spc_schedticks & 7) == 0 && schedhz != 0)
+/*
+ * Call hardclock on all CPUs.
+ */
+static void
+handle_hardclock(struct clockframe *cap)
+{
+ int s;
+#ifdef MULTIPROCESSOR
+ struct cpu_info *cpi;
+ CPU_INFO_ITERATOR n;
+
+ for (CPU_INFO_FOREACH(n, cpi)) {
+ if (cpi == cpuinfo.ci_self) {
+ KASSERT(CPU_IS_PRIMARY(cpi));
+ continue;
+ }
+
+ raise_ipi(cpi, IPL_HARDCLOCK);
+ }
#endif
- schedclock(curlwp);
- kpreempt_enable();
+ s = splsched();
+ hardclock(cap);
+ splx(s);
}
-
/*
* Level 10 (clock) interrupts from system counter.
*/
@@ -128,7 +147,6 @@
clockintr_4m(void *cap)
{
- KASSERT(CPU_IS_PRIMARY(curcpu()));
/*
* XXX this needs to be fixed in a more general way
* problem is that the kernel enables interrupts and THEN
@@ -146,7 +164,14 @@
/* read the limit register to clear the interrupt */
*((volatile int *)&timerreg4m->t_limit);
tickle_tc();
- hardclock((struct clockframe *)cap);
+
+ /*
+ * We don't have a system-clock per-cpu, and we'd like to keep
+ * the per-cpu timer for the statclock, so, send an IPI to
+ * everyone to call hardclock.
+ */
+ handle_hardclock(cap);
+
kpreempt_enable();
return (1);
}
@@ -183,23 +208,19 @@
* The factor 8 is only valid for stathz==100.
* See also clock.c
*/
-#if !defined(MULTIPROCESSOR)
if ((++cpuinfo.ci_schedstate.spc_schedticks & 7) == 0 && schedhz != 0) {
-#endif
if (CLKF_LOPRI(frame, IPL_SCHED)) {
/* No need to schedule a soft interrupt */
spllowerschedclock();
- schedintr_4m(cap);
+ schedintr(cap);
} else {
/*
* We're interrupting a thread that may have the
- * scheduler lock; run schedintr_4m() on this CPU later.
+ * scheduler lock; run schedintr() on this CPU later.
*/
raise_ipi(&cpuinfo, IPL_SCHED); /* sched_cookie->pil */
}
-#if !defined(MULTIPROCESSOR)
}
-#endif
kpreempt_enable();
return (1);
@@ -261,6 +282,22 @@
cpi->counterreg_4m = (struct counter_4m *)bh;
}
+#if defined(MULTIPROCESSOR)
+ if (sparc_ncpus > 1) {
+ /*
+ * Note that we don't actually use this cookie after checking
+ * it was establised, we call directly via raise_ipi() on
+ * IPL_HARDCLOCK.
+ */
+ void *hardclock_cookie;
+
+ hardclock_cookie = sparc_softintr_establish(IPL_HARDCLOCK,
+ hardclock_ipi, NULL);
+ if (hardclock_cookie == NULL)
+ panic("timerattach: cannot establish hardclock_intr");
+ }
+#endif
+
/* Put processor counter in "timer" mode */
timerreg4m->t_cfg = 0;
diff -r d4a80e172235 -r 9c1bd6fa9dbf sys/arch/sparc/sparc/timervar.h
--- a/sys/arch/sparc/sparc/timervar.h Sat Jan 18 21:27:11 2014 +0000
+++ b/sys/arch/sparc/sparc/timervar.h Sun Jan 19 00:22:33 2014 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: timervar.h,v 1.11 2012/07/29 00:04:05 matt Exp $ */
+/* $NetBSD: timervar.h,v 1.12 2014/01/19 00:22:33 mrg Exp $ */
/*
* Copyright (c) 1992, 1993
@@ -50,7 +50,6 @@
#endif /* SUN4 || SUN4C */
#if defined(SUN4M)
-void schedintr_4m(void *);
int clockintr_4m(void *);
int statintr_4m(void *);
void timer_init_4m(void);
@@ -66,7 +65,7 @@
/* Common timer attach routine in timer.c: */
void timerattach(volatile int *, volatile int *);
-extern void *sched_cookie; /* for schedclock() interrupts */
+extern void *sched_cookie; /* for schedclock() interrupts */
static inline u_long __attribute__((__unused__))
new_interval(void)
Home |
Main Index |
Thread Index |
Old Index