summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorKenji Aoyama <aoyama@cvs.openbsd.org>2012-02-28 13:40:54 +0000
committerKenji Aoyama <aoyama@cvs.openbsd.org>2012-02-28 13:40:54 +0000
commit2f5f4e7da2bc763d902d0377b978e03c2699e33a (patch)
tree61eacc508c2882fa53ce2ea02aaea2e01a6a6510
parent2dd6065e0774d3dc6c94ad67169992234c288f44 (diff)
Luna88k multi-processor support, step 3.
GENERIC.MP on luna88k is now stable. Changes are: - Make secondary processors boot sequence closer to OpenBSD/aviion. This makes boot sequence more stable. Now we have not see any assertions or panics during boot. - Implement IPI as level 1 hardware interrupt by using software interrupt register. - Add IPI support and some clean-ups in interrupt handler function.
-rw-r--r--sys/arch/luna88k/include/board.h12
-rw-r--r--sys/arch/luna88k/luna88k/machdep.c222
-rw-r--r--sys/arch/luna88k/luna88k/mainbus.c21
3 files changed, 177 insertions, 78 deletions
diff --git a/sys/arch/luna88k/include/board.h b/sys/arch/luna88k/include/board.h
index 5402fc6bdcf..f424c209644 100644
--- a/sys/arch/luna88k/include/board.h
+++ b/sys/arch/luna88k/include/board.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: board.h,v 1.10 2011/03/23 16:54:35 pirofti Exp $ */
+/* $OpenBSD: board.h,v 1.11 2012/02/28 13:40:53 aoyama Exp $ */
/*
* Mach Operating System
* Copyright (c) 1993-1991 Carnegie Mellon University
@@ -107,11 +107,11 @@
#define INT_LEVEL 8 /* # of interrupt level + 1 */
#define INT_SET_LV7 U(0x00000000) /* disable interrupts */
#define INT_SET_LV6 U(0x00000000) /* enable level 7 */
-#define INT_SET_LV5 U(0x80000000) /* enable level 7-6 */
-#define INT_SET_LV4 U(0xC0000000) /* enable level 7-5 */
-#define INT_SET_LV3 U(0xE0000000) /* enable level 7-4 */
-#define INT_SET_LV2 U(0xF0000000) /* enable level 7-3 */
-#define INT_SET_LV1 U(0xF8000000) /* enable level 7-2 */
+#define INT_SET_LV5 U(0x84000000) /* enable level 7-6 */
+#define INT_SET_LV4 U(0xC4000000) /* enable level 7-5 */
+#define INT_SET_LV3 U(0xE4000000) /* enable level 7-4 */
+#define INT_SET_LV2 U(0xF4000000) /* enable level 7-3 */
+#define INT_SET_LV1 U(0xFC000000) /* enable level 7-2 */
#define INT_SET_LV0 U(0xFC000000) /* enable interrupts */
#define INT_SLAVE_MASK U(0x84000000) /* slave can only enable 6 and 1 */
#define INT_CLOCK_MASK 0xBFFFFFFF /* mask clock */
diff --git a/sys/arch/luna88k/luna88k/machdep.c b/sys/arch/luna88k/luna88k/machdep.c
index 0e3b877f017..8c0cbeccf4b 100644
--- a/sys/arch/luna88k/luna88k/machdep.c
+++ b/sys/arch/luna88k/luna88k/machdep.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: machdep.c,v 1.80 2012/01/28 11:34:19 aoyama Exp $ */
+/* $OpenBSD: machdep.c,v 1.81 2012/02/28 13:40:53 aoyama Exp $ */
/*
* Copyright (c) 1998, 1999, 2000, 2001 Steve Murphree, Jr.
* Copyright (c) 1996 Nivas Madhur
@@ -103,30 +103,36 @@
#endif /* DDB */
void consinit(void);
+#ifdef MULTIPROCESSOR
void cpu_boot_secondary_processors(void);
+#endif
+void cpu_setup_secondary_processors(void);
void dumpconf(void);
void dumpsys(void);
int getcpuspeed(void);
+void get_fuse_rom_data(void);
+void get_nvram_data(void);
void identifycpu(void);
void luna88k_bootstrap(void);
+#ifdef MULTIPROCESSOR
+void luna88k_ipi_handler(struct trapframe *);
+#endif
+char *nvram_by_symbol(char *);
+void powerdown(void);
void savectx(struct pcb *);
void secondary_main(void);
vaddr_t secondary_pre_main(void);
void setlevel(u_int);
-
vaddr_t size_memory(void);
-void powerdown(void);
-void get_fuse_rom_data(void);
-void get_nvram_data(void);
-char *nvram_by_symbol(char *);
-void get_autoboot_device(void); /* in autoconf.c */
-int clockintr(void *); /* in clock.c */
+
+extern int clockintr(void *); /* in clock.c */
+extern void get_autoboot_device(void); /* in autoconf.c */
/*
* *int_mask_reg[CPU]
* Points to the hardware interrupt status register for each CPU.
*/
-u_int32_t *volatile int_mask_reg[] = {
+volatile u_int32_t *int_mask_reg[] = {
(u_int32_t *)INT_ST_MASK0,
(u_int32_t *)INT_ST_MASK1,
(u_int32_t *)INT_ST_MASK2,
@@ -147,9 +153,21 @@ u_int32_t int_set_val[INT_LEVEL] = {
};
/*
+ * *swi_reg[CPU]
+ * Points to the software interrupt register for each CPU.
+ */
+volatile u_int32_t *swi_reg[] = {
+ (u_int32_t *)SOFT_INT0,
+ (u_int32_t *)SOFT_INT1,
+ (u_int32_t *)SOFT_INT2,
+ (u_int32_t *)SOFT_INT3
+};
+
+/*
* *clock_reg[CPU]
+ * Points to the clock register for each CPU.
*/
-u_int32_t *volatile clock_reg[] = {
+volatile u_int32_t *clock_reg[] = {
(u_int32_t *)OBIO_CLOCK0,
(u_int32_t *)OBIO_CLOCK1,
(u_int32_t *)OBIO_CLOCK2,
@@ -199,7 +217,7 @@ extern char *esym;
#endif
int machtype = LUNA_88K; /* may be overwritten in cpu_startup() */
-int cputyp = CPU_88100; /* XXX: aoyama */
+int cputyp = CPU_88100;
int boothowto; /* XXX: should be set in boot loader and locore.S */
int bootdev; /* XXX: should be set in boot loader and locore.S */
int cpuspeed = 33; /* safe guess */
@@ -670,25 +688,31 @@ abort:
}
/*
- * Release the cpu_{hatch,boot}_mutex; secondary processors will now
- * have their chance to initialize.
+ * Release cpu_hatch_mutex to let secondary processors initialize.
*/
void
-cpu_boot_secondary_processors()
+cpu_setup_secondary_processors()
{
#ifdef MULTIPROCESSOR
hatch_pending_count = ncpusfound - 1;
#endif
__cpu_simple_unlock(&cpu_hatch_mutex);
-
#ifdef MULTIPROCESSOR
while (hatch_pending_count != 0)
delay(10000); /* 10ms */
- __cpu_simple_unlock(&cpu_boot_mutex);
#endif
}
#ifdef MULTIPROCESSOR
+/*
+ * Release cpu_boot_mutex to let secondary processors start running
+ * processes.
+ */
+void
+cpu_boot_secondary_processors()
+{
+ __cpu_simple_unlock(&cpu_boot_mutex);
+}
/*
* Secondary CPU early initialization routine.
@@ -791,60 +815,69 @@ luna88k_ext_int(struct trapframe *eframe)
{
#ifdef MULTIPROCESSOR
struct cpu_info *ci = curcpu();
- uint cpu = ci->ci_cpuid;
+ u_int cpu = ci->ci_cpuid;
#else
u_int cpu = cpu_number();
#endif
- u_int32_t cur_mask, cur_int;
- u_int level, old_spl;
+ u_int32_t cur_isr, ign_mask;
+ u_int level, cur_int_level, old_spl;
+ int unmasked = 0;
- cur_mask = *int_mask_reg[cpu];
- old_spl = luna88k_curspl[cpu];
- eframe->tf_mask = old_spl;
+ cur_isr = *int_mask_reg[cpu];
+ ign_mask = 0;
+ old_spl = eframe->tf_mask;
- cur_int = cur_mask >> 29;
+ cur_int_level = cur_isr >> 29;
- if (cur_int == 0) {
+ if (cur_int_level == 0) {
/*
- * Spurious interrupts - may be caused by debug output clearing
- * serial port interrupts.
+ * ignore level 0 interrupt, as CMU Mach do.
*/
-#ifdef DEBUG
- printf("luna88k_ext_int(): Spurious interrupts?\n");
-#endif
- flush_pipeline();
+ flush_pipeline(); /* need this? */
goto out;
}
-
+
uvmexp.intrs++;
- /*
- * We want to service all interrupts marked in the IST register
- * They are all valid because the mask would have prevented them
- * from being generated otherwise. We will service them in order of
- * priority.
+#ifdef MULTIPROCESSOR
+ /*
+ * Handle unmaskable IPIs immediately, so that we can reenable
+ * interrupts before further processing. We rely on the interrupt
+ * mask to make sure that if we get an IPI, it's really for us
+ * and no other processor.
+ *
+ * On luna88k, IPL_SOFTINT (level 1 interrupt) is used as IPI.
*/
+ while (cur_int_level == IPL_SOFTINT) {
+ luna88k_ipi_handler(eframe);
+
+ cur_isr = *int_mask_reg[cpu];
+ cur_int_level = cur_isr >> 29;
+ }
+ if (cur_int_level == 0) {
+ flush_pipeline(); /* need this? */
+ goto out;
+ }
+#endif
#ifdef MULTIPROCESSOR
if (old_spl < IPL_SCHED)
__mp_lock(&kernel_lock);
#endif
- /* XXX: This is very rough. Should be considered more. (aoyama) */
+ /*
+ * Service the highest interrupt, in order.
+ */
do {
- level = (cur_int > old_spl ? cur_int : old_spl);
+ level = (cur_int_level > old_spl ? cur_int_level : old_spl);
+ setipl(level);
-#ifdef DEBUG
- if (level > 7 || (char)level < 0) {
- panic("int level (%x) is not between 0 and 7", level);
+ if (unmasked == 0) {
+ set_psr(get_psr() & ~PSR_IND);
+ unmasked = 1;
}
-#endif
- setipl(level);
-
- set_psr(get_psr() & ~PSR_IND);
-
- switch(cur_int) {
- case CLOCK_INT_LEVEL:
+ switch (cur_int_level) {
+ case 6:
clockintr((void *)eframe);
break;
case 5:
@@ -853,31 +886,26 @@ luna88k_ext_int(struct trapframe *eframe)
#ifdef MULTIPROCESSOR
if (cpu == master_cpu) {
#endif
- isrdispatch_autovec(cur_int);
+ isrdispatch_autovec(cur_int_level);
#ifdef MULTIPROCESSOR
}
#endif
break;
default:
- printf("luna88k_ext_int(): level %d interrupt.\n", cur_int);
+ printf("%s: cpu%d level %d interrupt.\n",
+ __func__, cpu, cur_int_level);
break;
}
- cur_int = (*int_mask_reg[cpu]) >> 29;
-#ifdef MULTIPROCESSOR
- if ((cpu != master_cpu) &&
- (cur_int != CLOCK_INT_LEVEL) && (cur_int != 0)) {
- printf("cpu%d: cur_int=%d\n", cpu, cur_int);
- flush_pipeline();
- cur_int = 0;
- }
-#endif
- } while (cur_int != 0);
+ cur_isr = *int_mask_reg[cpu];
+ cur_int_level = cur_isr >> 29;
+ } while (cur_int_level != 0);
#ifdef MULTIPROCESSOR
if (old_spl < IPL_SCHED)
__mp_unlock(&kernel_lock);
#endif
+
out:
/*
* process any remaining data access exceptions before
@@ -970,10 +998,12 @@ luna88k_bootstrap()
cmmu = &cmmu8820x;
/* clear and disable all interrupts */
- *int_mask_reg[0] = 0;
- *int_mask_reg[1] = 0;
- *int_mask_reg[2] = 0;
- *int_mask_reg[3] = 0;
+ *int_mask_reg[0] = *int_mask_reg[1] =
+ *int_mask_reg[2] = *int_mask_reg[3] = 0;
+
+ /* clear software interrupts; just read registers */
+ *swi_reg[0]; *swi_reg[1];
+ *swi_reg[2]; *swi_reg[3];
uvmexp.pagesize = PAGE_SIZE;
uvm_setpagesize();
@@ -1030,7 +1060,7 @@ luna88k_bootstrap()
#ifndef MULTIPROCESSOR
/* Release the cpu_hatch_mutex */
- cpu_boot_secondary_processors();
+ cpu_setup_secondary_processors();
#endif
#ifdef DEBUG
@@ -1301,12 +1331,70 @@ raiseipl(int level)
void
m88k_send_ipi(int ipi, cpuid_t cpu)
{
- /* XXX: not yet */
+ struct cpu_info *ci = &m88k_cpus[cpu];
+
+ if (ci->ci_ipi & ipi)
+ return;
+
+ atomic_setbits_int(&ci->ci_ipi, ipi);
+ *swi_reg[cpu] = 0xffffffff;
+}
+
+/*
+ * Process inter-processor interrupts.
+ */
+
+/*
+ * Unmaskable IPIs - those are processed with interrupts disabled,
+ * and no lock held.
+ */
+void
+luna88k_ipi_handler(struct trapframe *eframe)
+{
+ struct cpu_info *ci = curcpu();
+ int cpu = ci->ci_cpuid;
+ int ipi = ci->ci_ipi & (CI_IPI_DDB | CI_IPI_NOTIFY);
+
+ /* just read; reset software interrupt */
+ *swi_reg[cpu];
+ atomic_clearbits_int(&ci->ci_ipi, ipi);
+
+ if (ipi & CI_IPI_DDB) {
+#ifdef DDB
+ /*
+ * Another processor has entered DDB. Spin on the ddb lock
+ * until it is done.
+ */
+ extern struct __mp_lock ddb_mp_lock;
+
+ __mp_lock(&ddb_mp_lock);
+ __mp_unlock(&ddb_mp_lock);
+
+ /*
+ * If ddb is hoping to us, it's our turn to enter ddb now.
+ */
+ if (ci->ci_cpuid == ddb_mp_nextcpu)
+ Debugger();
+#endif
+ }
+ if (ipi & CI_IPI_NOTIFY) {
+ /* nothing to do */
+ }
}
void
m88k_broadcast_ipi(int ipi)
{
- /* XXX: not yet */
+ struct cpu_info *us = curcpu();
+ struct cpu_info *ci;
+ CPU_INFO_ITERATOR cii;
+
+ CPU_INFO_FOREACH(cii, ci) {
+ if (ci == us)
+ continue;
+
+ if (ISSET(ci->ci_flags, CIF_ALIVE))
+ m88k_send_ipi(ipi, ci->ci_cpuid);
+ }
}
#endif
diff --git a/sys/arch/luna88k/luna88k/mainbus.c b/sys/arch/luna88k/luna88k/mainbus.c
index 7c53bc54f6b..e0957d4a62b 100644
--- a/sys/arch/luna88k/luna88k/mainbus.c
+++ b/sys/arch/luna88k/luna88k/mainbus.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: mainbus.c,v 1.8 2008/06/26 05:42:11 ray Exp $ */
+/* $OpenBSD: mainbus.c,v 1.9 2012/02/28 13:40:53 aoyama Exp $ */
/* $NetBSD: mainbus.c,v 1.2 2000/01/07 05:13:08 nisimura Exp $ */
/*-
@@ -58,9 +58,12 @@ static const struct mainbus_attach_args devs[] = {
#endif
};
-void mainbus_attach(struct device *, struct device *, void *);
-int mainbus_match(struct device *, void *, void *);
-int mainbus_print(void *, const char *);
+void mainbus_attach(struct device *, struct device *, void *);
+int mainbus_match(struct device *, void *, void *);
+int mainbus_print(void *, const char *);
+#ifdef MULTIPROCESSOR
+extern void cpu_setup_secondary_processors(void); /* in machdep.c */
+#endif
const struct cfattach mainbus_ca = {
sizeof(struct device), mainbus_match, mainbus_attach
@@ -95,10 +98,18 @@ mainbus_attach(parent, self, args)
printf(": %s\n", cpu_model);
/*
- * Display cpu/mmu details. Only for the master CPU so far.
+ * Display cpu/mmu details for the main processor.
*/
cpu_configuration_print(1);
+#ifdef MULTIPROCESSOR
+ /*
+ * Let secondary processors initialize further and print
+ * their configuration information now.
+ */
+ cpu_setup_secondary_processors();
+#endif
+
for (i = 0; i < sizeof(devs)/sizeof(devs[0]); i++)
if (devs[i].ma_machine & machtype)
config_found(self, (void *)&devs[i], mainbus_print);