summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorTakuya ASADA <syuu@cvs.openbsd.org>2009-11-22 18:33:49 +0000
committerTakuya ASADA <syuu@cvs.openbsd.org>2009-11-22 18:33:49 +0000
commit7214e153c6803bef90d67532be57bf02fe908165 (patch)
tree4405dc2cbed4f5f4596e25a7f06f8de7e9808199
parentd141025c60b43583bd546a8b8ae007ea2cf1bd6c (diff)
SMP support on MIPS clock.
ok miod@
-rw-r--r--sys/arch/mips64/include/cpu.h8
-rw-r--r--sys/arch/mips64/mips64/clock.c74
-rw-r--r--sys/arch/sgi/sgi/ip30_machdep.c15
3 files changed, 65 insertions, 32 deletions
diff --git a/sys/arch/mips64/include/cpu.h b/sys/arch/mips64/include/cpu.h
index 64063fd30d4..d8745b50de0 100644
--- a/sys/arch/mips64/include/cpu.h
+++ b/sys/arch/mips64/include/cpu.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: cpu.h,v 1.43 2009/11/19 20:16:26 miod Exp $ */
+/* $OpenBSD: cpu.h,v 1.44 2009/11/22 18:33:48 syuu Exp $ */
/*-
* Copyright (c) 1992, 1993
@@ -378,6 +378,10 @@ struct cpu_info {
uint32_t ci_randseed; /* per cpu random seed */
int ci_ipl; /* software IPL */
uint32_t ci_softpending; /* pending soft interrupts */
+ int ci_clock_started;
+ u_int32_t ci_cpu_counter_last;
+ u_int32_t ci_cpu_counter_interval;
+ u_int32_t ci_pendingticks;
#ifdef MULTIPROCESSOR
u_long ci_flags; /* flags; see below */
#endif
@@ -417,6 +421,8 @@ void cpu_boot_secondary_processors(void);
#define cpu_unidle(ci)
#endif
+void cpu_startclock(struct cpu_info *);
+
#include <machine/frame.h>
#endif /* _LOCORE */
diff --git a/sys/arch/mips64/mips64/clock.c b/sys/arch/mips64/mips64/clock.c
index 68583c03b1a..f4677a2db3b 100644
--- a/sys/arch/mips64/mips64/clock.c
+++ b/sys/arch/mips64/mips64/clock.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: clock.c,v 1.27 2009/11/21 23:28:14 syuu Exp $ */
+/* $OpenBSD: clock.c,v 1.28 2009/11/22 18:33:48 syuu Exp $ */
/*
* Copyright (c) 2001-2004 Opsycon AB (www.opsycon.se / www.opsycon.com)
@@ -54,11 +54,6 @@ struct cfattach clock_ca = {
uint32_t clock_int5(uint32_t, struct trap_frame *);
-int clock_started;
-u_int32_t cpu_counter_last;
-u_int32_t cpu_counter_interval;
-u_int32_t pendingticks;
-
u_int cp0_get_timecount(struct timecounter *);
struct timecounter cp0_timecounter = {
@@ -123,13 +118,14 @@ uint32_t
clock_int5(uint32_t mask, struct trap_frame *tf)
{
u_int32_t clkdiff;
+ struct cpu_info *ci = curcpu();
/*
* If we got an interrupt before we got ready to process it,
* retrigger it as far as possible. cpu_initclocks() will
* take care of retriggering it correctly.
*/
- if (clock_started == 0) {
+ if (ci->ci_clock_started == 0) {
cp0_set_compare(cp0_get_count() - 1);
return CR_INT_5;
}
@@ -137,38 +133,44 @@ clock_int5(uint32_t mask, struct trap_frame *tf)
/*
* Count how many ticks have passed since the last clock interrupt...
*/
- clkdiff = cp0_get_count() - cpu_counter_last;
- while (clkdiff >= cpu_counter_interval) {
- cpu_counter_last += cpu_counter_interval;
- clkdiff = cp0_get_count() - cpu_counter_last;
- pendingticks++;
+ clkdiff = cp0_get_count() - ci->ci_cpu_counter_last;
+ while (clkdiff >= ci->ci_cpu_counter_interval) {
+ ci->ci_cpu_counter_last += ci->ci_cpu_counter_interval;
+ clkdiff = cp0_get_count() - ci->ci_cpu_counter_last;
+ ci->ci_pendingticks++;
}
- pendingticks++;
- cpu_counter_last += cpu_counter_interval;
+ ci->ci_pendingticks++;
+ ci->ci_cpu_counter_last += ci->ci_cpu_counter_interval;
/*
* Set up next tick, and check if it has just been hit; in this
* case count it and schedule one tick ahead.
*/
- cp0_set_compare(cpu_counter_last);
- clkdiff = cp0_get_count() - cpu_counter_last;
+ cp0_set_compare(ci->ci_cpu_counter_last);
+ clkdiff = cp0_get_count() - ci->ci_cpu_counter_last;
if ((int)clkdiff >= 0) {
- cpu_counter_last += cpu_counter_interval;
- pendingticks++;
- cp0_set_compare(cpu_counter_last);
+ ci->ci_cpu_counter_last += ci->ci_cpu_counter_interval;
+ ci->ci_pendingticks++;
+ cp0_set_compare(ci->ci_cpu_counter_last);
}
/*
* Process clock interrupt unless it is currently masked.
*/
if (tf->ipl < IPL_CLOCK) {
- KERNEL_LOCK();
- while (pendingticks) {
+#ifdef MULTIPROCESSOR
+ if (ci->ci_ipl < IPL_SCHED)
+ __mp_lock(&kernel_lock);
+#endif
+ while (ci->ci_pendingticks) {
clk_count.ec_count++;
hardclock(tf);
- pendingticks--;
+ ci->ci_pendingticks--;
}
- KERNEL_UNLOCK();
+#ifdef MULTIPROCESSOR
+ if (ci->ci_ipl < IPL_SCHED)
+ __mp_unlock(&kernel_lock);
+#endif
}
return CR_INT_5; /* Clock is always on 5 */
@@ -209,7 +211,7 @@ cpu_initclocks()
struct tod_time ct;
u_int first_cp0, second_cp0, cycles_per_sec;
int first_sec;
- int s;
+ struct cpu_info *ci = curcpu();
hz = 100;
profhz = 100;
@@ -243,13 +245,29 @@ cpu_initclocks()
cp0_timecounter.tc_frequency = sys_config.cpu[0].clock / 2;
tc_init(&cp0_timecounter);
+ cpu_startclock(ci);
+}
+
+void
+cpu_startclock(struct cpu_info *ci)
+{
+ int s;
+
+ if (!CPU_IS_PRIMARY(ci)) {
+ s = splhigh();
+ microuptime(&ci->ci_schedstate.spc_runtime);
+ splx(s);
+
+ /* try to avoid getting clock interrupts early */
+ cp0_set_compare(cp0_get_count() - 1);
+ }
/* Start the clock. */
s = splclock();
- cpu_counter_interval = cp0_timecounter.tc_frequency / hz;
- cpu_counter_last = cp0_get_count() + cpu_counter_interval;
- cp0_set_compare(cpu_counter_last);
- clock_started++;
+ ci->ci_cpu_counter_interval = cp0_timecounter.tc_frequency / hz;
+ ci->ci_cpu_counter_last = cp0_get_count() + ci->ci_cpu_counter_interval;
+ cp0_set_compare(ci->ci_cpu_counter_last);
+ ci->ci_clock_started++;
splx(s);
}
diff --git a/sys/arch/sgi/sgi/ip30_machdep.c b/sys/arch/sgi/sgi/ip30_machdep.c
index 0b4d45f20e8..fcf6a3f75fe 100644
--- a/sys/arch/sgi/sgi/ip30_machdep.c
+++ b/sys/arch/sgi/sgi/ip30_machdep.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: ip30_machdep.c,v 1.19 2009/11/21 20:55:44 miod Exp $ */
+/* $OpenBSD: ip30_machdep.c,v 1.20 2009/11/22 18:33:48 syuu Exp $ */
/*
* Copyright (c) 2008, 2009 Miodrag Vallat.
@@ -403,7 +403,16 @@ hw_cpu_hatch(struct cpu_info *ci)
cpuset_add(&cpus_running, ci);
- for (;;)
- ;
+ cpu_startclock(ci);
+
+ spl0();
+ (void)updateimask(0);
+#ifdef notyet
+ SCHED_LOCK(s);
+ cpu_switchto(NULL, sched_chooseproc());
+#else
+ for(;;)
+ ;
+#endif
}
#endif