summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authormav <mav@FreeBSD.org>2009-05-02 12:20:43 +0000
committermav <mav@FreeBSD.org>2009-05-02 12:20:43 +0000
commitbaf55d1d06ec21028f521e2a0594294a49bc3bce (patch)
treef18d346c3109119beca341888aa2edfe43eac759
parenta5dcb177014ca524da48b6cd1ecdbde4e6b42789 (diff)
downloadFreeBSD-src-baf55d1d06ec21028f521e2a0594294a49bc3bce.zip
FreeBSD-src-baf55d1d06ec21028f521e2a0594294a49bc3bce.tar.gz
Add support for using i8254 and rtc timers as event sources for amd64 SMP
system. Redistribute hard-/stat-/profclock events to other CPUs using IPIs.
-rw-r--r--sys/amd64/amd64/mp_machdep.c10
-rw-r--r--sys/amd64/include/apicvar.h5
-rw-r--r--sys/amd64/include/clock.h6
-rw-r--r--sys/amd64/isa/clock.c51
4 files changed, 69 insertions, 3 deletions
diff --git a/sys/amd64/amd64/mp_machdep.c b/sys/amd64/amd64/mp_machdep.c
index ffda046..7a93023 100644
--- a/sys/amd64/amd64/mp_machdep.c
+++ b/sys/amd64/amd64/mp_machdep.c
@@ -57,6 +57,7 @@ __FBSDID("$FreeBSD$");
#include <vm/vm_extern.h>
#include <machine/apicreg.h>
+#include <machine/clock.h>
#include <machine/cputypes.h>
#include <machine/cpufunc.h>
#include <machine/md_var.h>
@@ -1124,6 +1125,15 @@ ipi_bitmap_handler(struct trapframe frame)
sched_preempt(curthread);
/* Nothing to do for AST */
+
+ if (ipi_bitmap & (1 << IPI_HARDCLOCK))
+ hardclockintr(&frame);
+
+ if (ipi_bitmap & (1 << IPI_STATCLOCK))
+ statclockintr(&frame);
+
+ if (ipi_bitmap & (1 << IPI_PROFCLOCK))
+ profclockintr(&frame);
}
/*
diff --git a/sys/amd64/include/apicvar.h b/sys/amd64/include/apicvar.h
index 3cc028d..84ba3b8 100644
--- a/sys/amd64/include/apicvar.h
+++ b/sys/amd64/include/apicvar.h
@@ -126,7 +126,10 @@
/* IPIs handled by IPI_BITMAPED_VECTOR (XXX ups is there a better place?) */
#define IPI_AST 0 /* Generate software trap. */
#define IPI_PREEMPT 1
-#define IPI_BITMAP_LAST IPI_PREEMPT
+#define IPI_HARDCLOCK 2
+#define IPI_STATCLOCK 3
+#define IPI_PROFCLOCK 4
+#define IPI_BITMAP_LAST IPI_PROFCLOCK
#define IPI_IS_BITMAPED(x) ((x) <= IPI_BITMAP_LAST)
#define IPI_STOP (APIC_IPI_INTS + 7) /* Stop CPU until restarted. */
diff --git a/sys/amd64/include/clock.h b/sys/amd64/include/clock.h
index f8745f9c..155fdd1 100644
--- a/sys/amd64/include/clock.h
+++ b/sys/amd64/include/clock.h
@@ -24,6 +24,12 @@ extern int tsc_is_invariant;
void i8254_init(void);
+struct trapframe;
+
+int hardclockintr(struct trapframe *frame);
+int statclockintr(struct trapframe *frame);
+int profclockintr(struct trapframe *frame);
+
/*
* Driver to clock driver interface.
*/
diff --git a/sys/amd64/isa/clock.c b/sys/amd64/isa/clock.c
index 4d365c9..b526025 100644
--- a/sys/amd64/isa/clock.c
+++ b/sys/amd64/isa/clock.c
@@ -53,6 +53,7 @@ __FBSDID("$FreeBSD$");
#include <sys/kernel.h>
#include <sys/module.h>
#include <sys/sched.h>
+#include <sys/smp.h>
#include <sys/sysctl.h>
#include <machine/clock.h>
@@ -62,6 +63,7 @@ __FBSDID("$FreeBSD$");
#include <machine/apicvar.h>
#include <machine/ppireg.h>
#include <machine/timerreg.h>
+#include <machine/smp.h>
#include <isa/rtc.h>
#ifdef DEV_ISA
@@ -112,6 +114,35 @@ static struct timecounter i8254_timecounter = {
0 /* quality */
};
+int
+hardclockintr(struct trapframe *frame)
+{
+
+ if (PCPU_GET(cpuid) == 0)
+ hardclock(TRAPF_USERMODE(frame), TRAPF_PC(frame));
+ else
+ hardclock_cpu(TRAPF_USERMODE(frame));
+ return (FILTER_HANDLED);
+}
+
+int
+statclockintr(struct trapframe *frame)
+{
+
+ if (profprocs != 0)
+ profclock(TRAPF_USERMODE(frame), TRAPF_PC(frame));
+ statclock(TRAPF_USERMODE(frame));
+ return (FILTER_HANDLED);
+}
+
+int
+profclockintr(struct trapframe *frame)
+{
+
+ profclock(TRAPF_USERMODE(frame), TRAPF_PC(frame));
+ return (FILTER_HANDLED);
+}
+
static int
clkintr(struct trapframe *frame)
{
@@ -128,7 +159,14 @@ clkintr(struct trapframe *frame)
mtx_unlock_spin(&clock_lock);
}
KASSERT(!using_lapic_timer, ("clk interrupt enabled with lapic timer"));
- hardclock(TRAPF_USERMODE(frame), TRAPF_PC(frame));
+#ifdef SMP
+ if (smp_started)
+ ipi_all_but_self(IPI_HARDCLOCK);
+#endif
+ if (PCPU_GET(cpuid) == 0)
+ hardclock(TRAPF_USERMODE(frame), TRAPF_PC(frame));
+ else
+ hardclock_cpu(TRAPF_USERMODE(frame));
return (FILTER_HANDLED);
}
@@ -209,10 +247,19 @@ rtcintr(struct trapframe *frame)
if (profprocs != 0) {
if (--pscnt == 0)
pscnt = psdiv;
+#ifdef SMP
+ if (pscnt != psdiv && smp_started)
+ ipi_all_but_self(IPI_PROFCLOCK);
+#endif
profclock(TRAPF_USERMODE(frame), TRAPF_PC(frame));
}
- if (pscnt == psdiv)
+ if (pscnt == psdiv) {
+#ifdef SMP
+ if (smp_started)
+ ipi_all_but_self(IPI_STATCLOCK);
+#endif
statclock(TRAPF_USERMODE(frame));
+ }
}
return(flag ? FILTER_HANDLED : FILTER_STRAY);
}
OpenPOWER on IntegriCloud