summaryrefslogtreecommitdiff
path: root/sys/arch/sgi
diff options
context:
space:
mode:
authorMiod Vallat <miod@cvs.openbsd.org>2009-10-22 22:08:55 +0000
committerMiod Vallat <miod@cvs.openbsd.org>2009-10-22 22:08:55 +0000
commit96629079d3f1a0da94f72c0045f030432bddc51d (patch)
tree91639a4e0d7c5dc462c850a9bed7dc553484768d /sys/arch/sgi
parent94815ddee781a683d8d1d9258ec9c8959763443f (diff)
Completely overhaul interrupt handling on sgi. Cpu state now only stores a
logical IPL level, and per-platform (IP27/IP30/IP32) code will from the necessary hardware mask registers. This allows the use of more than one interrupt mask register. Also, the generic (platform independent) interrupt code shrinks a lot, and the actual interrupt handler chains and masking information is now per-platform private data. Interrupt dispatching is generated from a template; more routines will be added to the template to reduce platform-specific changes and share as much code as possible. Tested on IP27, IP30, IP32 and IP35.
Diffstat (limited to 'sys/arch/sgi')
-rw-r--r--sys/arch/sgi/include/intr.h105
-rw-r--r--sys/arch/sgi/include/mutex.h6
-rw-r--r--sys/arch/sgi/localbus/crimebus.h4
-rw-r--r--sys/arch/sgi/localbus/macebus.c280
-rw-r--r--sys/arch/sgi/sgi/genassym.cf4
-rw-r--r--sys/arch/sgi/sgi/intr_template.c125
-rw-r--r--sys/arch/sgi/sgi/ip27_machdep.c330
-rw-r--r--sys/arch/sgi/sgi/ip30_machdep.c13
-rw-r--r--sys/arch/sgi/sgi/mutex.c11
-rw-r--r--sys/arch/sgi/xbow/hub.h56
-rw-r--r--sys/arch/sgi/xbow/xbridge.c11
-rw-r--r--sys/arch/sgi/xbow/xheart.c272
-rw-r--r--sys/arch/sgi/xbow/xheartreg.h62
13 files changed, 712 insertions, 567 deletions
diff --git a/sys/arch/sgi/include/intr.h b/sys/arch/sgi/include/intr.h
index 33cbd8aec22..1829fc7dd22 100644
--- a/sys/arch/sgi/include/intr.h
+++ b/sys/arch/sgi/include/intr.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: intr.h,v 1.33 2009/10/22 20:39:17 miod Exp $ */
+/* $OpenBSD: intr.h,v 1.34 2009/10/22 22:08:54 miod Exp $ */
/*
* Copyright (c) 2001-2004 Opsycon AB (www.opsycon.se / www.opsycon.com)
@@ -30,14 +30,19 @@
#define _MACHINE_INTR_H_
/*
- * The interrupt mask cpl is a mask which is used with an external
- * HW mask register.
- * The CPU mask is never changed from the value it gets when interrupt
- * dispatchers are registered.
+ * The interrupt level ipl is a logical level; per-platform interrupt
+ * code will turn it into the appropriate hardware interrupt masks
+ * values.
*
- * Clock interrupts are always allowed to happen but will not be serviced
+ * Interrupt sources on the CPU are kept enabled regardless of the
+ * current ipl value; individual hardware sources interrupting while
+ * logically masked are masked on the fly, remembered as pending, and
+ * unmasked at the first splx() opportunity.
+ *
+ * An exception to this rule is the clock interrupt. Clock interrupts
+ * are always allowed to happen, but will (of course!) not be serviced
* if logically masked. The reason for this is that clocks usually sit on
- * INT5 and cannot be easily masked if external HW masking is used.
+ * INT5 and cannot be easily masked if external hardware masking is used.
*/
/* Interrupt priority `levels'; not mutually exclusive. */
@@ -58,12 +63,9 @@
#define IST_EDGE 2 /* edge-triggered */
#define IST_LEVEL 3 /* level-triggered */
-#define SINTBIT(q) (31 - (q))
+#define SINTBIT(q) (q)
#define SINTMASK(q) (1 << SINTBIT(q))
-#define SPL_CLOCK SINTBIT(SI_NQUEUES)
-#define SPL_CLOCKMASK SINTMASK(SI_NQUEUES)
-
/* Soft interrupt masks. */
#define IPL_SOFT 0
@@ -76,8 +78,6 @@
#define SI_SOFTNET 2 /* for IPL_SOFTNET */
#define SI_SOFTTTY 3 /* for IPL_SOFTTTY */
-#define SINT_ALLMASK (SINTMASK(SI_SOFT) | SINTMASK(SI_SOFTCLOCK) | \
- SINTMASK(SI_SOFTNET) | SINTMASK(SI_SOFTTTY))
#define SI_NQUEUES 4
#ifndef _LOCORE
@@ -110,45 +110,32 @@ extern struct soft_intrhand *softnet_intrhand;
#define setsoftnet() softintr_schedule(softnet_intrhand)
-#define splsoft() splraise(imask[IPL_SOFTINT])
-#define splbio() splraise(imask[IPL_BIO])
-#define splnet() splraise(imask[IPL_NET])
-#define spltty() splraise(imask[IPL_TTY])
-#define splaudio() splraise(imask[IPL_AUDIO])
-#define splclock() splraise(imask[IPL_CLOCK])
-#define splvm() splraise(imask[IPL_VM])
-#define splsoftclock() splraise(SINTMASK(SI_SOFTCLOCK) | \
- SINTMASK(SI_SOFT))
-#define splsoftnet() splraise(SINTMASK(SI_SOFTNET) | \
- SINTMASK(SI_SOFTCLOCK) | \
- SINTMASK(SI_SOFT))
-#define splstatclock() splhigh()
-#define splsched() splhigh()
-#define spllock() splhigh()
-#define splhigh() splraise(-1)
-#define spl0() spllower(0)
+#define splsoft() splraise(IPL_SOFTINT)
+#define splbio() splraise(IPL_BIO)
+#define splnet() splraise(IPL_NET)
+#define spltty() splraise(IPL_TTY)
+#define splaudio() splraise(IPL_AUDIO)
+#define splclock() splraise(IPL_CLOCK)
+#define splvm() splraise(IPL_VM)
+#define splhigh() splraise(IPL_HIGH)
+
+#define splsoftclock() splsoft()
+#define splsoftnet() splsoft()
+#define splstatclock() splhigh()
+
+#define splsched() splhigh()
+#define spllock() splhigh()
+#define spl0() spllower(0)
void splinit(void);
#define splassert(X)
#define splsoftassert(X)
-/*
- * Schedule prioritys for base interrupts (CPU)
- */
-#define INTPRI_CLOCK 1
-#define INTPRI_MACEIO 2 /* O2 I/O interrupt */
-#define INTPRI_XBOWMUX 2 /* Origin 200/2000 I/O interrupt */
-#define INTPRI_MACEAUX 3
-
-#define INTMASKSIZE 32
-
-extern uint32_t imask[NIPLS];
-
/* Inlines */
static __inline void register_splx_handler(void (*)(int));
-typedef void (int_f) (int);
+typedef void (int_f)(int);
extern int_f *splx_hand;
static __inline void
@@ -169,32 +156,32 @@ int spllower(int);
#include <sys/evcount.h>
struct intrhand {
- struct intrhand *ih_next;
- int (*ih_fun)(void *);
- void *ih_arg;
- int ih_level;
- int ih_irq;
- void *frame;
- struct evcount ih_count;
+ struct intrhand *ih_next;
+ int (*ih_fun)(void *);
+ void *ih_arg;
+ int ih_level;
+ int ih_irq;
+ void *frame;
+ struct evcount ih_count;
};
-extern struct intrhand *intrhand[INTMASKSIZE];
-
/*
* Low level interrupt dispatcher registration data.
*/
-#define NLOWINT 16 /* Number of low level registrations possible */
-struct trap_frame;
+/* Schedule priorities for base interrupts (CPU) */
+#define INTPRI_CLOCK 0
+/* other values are system-specific */
+
+#define NLOWINT 16 /* Number of low level registrations possible */
extern uint32_t idle_mask;
-extern int last_low_int;
-void set_intr(int, uint32_t, uint32_t(*)(uint32_t, struct trap_frame *));
+struct trap_frame;
+void set_intr(int, uint32_t, uint32_t(*)(uint32_t, struct trap_frame *));
-void hw_setintrmask(uint32_t);
-u_int32_t updateimask(uint32_t);
-void dosoftint(uint32_t);
+uint32_t updateimask(uint32_t);
+void dosoftint(void);
#endif /* _LOCORE */
diff --git a/sys/arch/sgi/include/mutex.h b/sys/arch/sgi/include/mutex.h
index 858d27c6a8f..df505d80ba8 100644
--- a/sys/arch/sgi/include/mutex.h
+++ b/sys/arch/sgi/include/mutex.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: mutex.h,v 1.3 2007/05/14 17:32:15 miod Exp $ */
+/* $OpenBSD: mutex.h,v 1.4 2009/10/22 22:08:54 miod Exp $ */
/*
* Copyright (c) 2004 Artur Grabowski <art@openbsd.org>
@@ -34,7 +34,7 @@
struct mutex {
int mtx_lock;
int mtx_wantipl;
- int mtx_oldcpl;
+ int mtx_oldipl;
};
void mtx_init(struct mutex *, int);
@@ -56,6 +56,6 @@ void mtx_init(struct mutex *, int);
#define MUTEX_ASSERT_UNLOCKED(mtx) do { } while (0)
#endif
-#define MUTEX_OLDIPL(mtx) (mtx)->mtx_oldcpl
+#define MUTEX_OLDIPL(mtx) (mtx)->mtx_oldipl
#endif
diff --git a/sys/arch/sgi/localbus/crimebus.h b/sys/arch/sgi/localbus/crimebus.h
index 902669355da..db7c6bf19bc 100644
--- a/sys/arch/sgi/localbus/crimebus.h
+++ b/sys/arch/sgi/localbus/crimebus.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: crimebus.h,v 1.7 2007/10/31 13:59:53 jsing Exp $ */
+/* $OpenBSD: crimebus.h,v 1.8 2009/10/22 22:08:54 miod Exp $ */
/*
* Copyright (c) 2003-2004 Opsycon AB (www.opsycon.se).
@@ -84,6 +84,8 @@
#define CRIME_INT_SOFT_2 0x40000000 /* ??? */
#define CRIME_INT_VICE 0x80000000 /* Video Image Compression Engine */
+#define CRIME_NINTS 32
+
/*
* Watchdog?
diff --git a/sys/arch/sgi/localbus/macebus.c b/sys/arch/sgi/localbus/macebus.c
index 39ff5fdc706..98e91dbddcd 100644
--- a/sys/arch/sgi/localbus/macebus.c
+++ b/sys/arch/sgi/localbus/macebus.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: macebus.c,v 1.50 2009/10/22 20:59:24 miod Exp $ */
+/* $OpenBSD: macebus.c,v 1.51 2009/10/22 22:08:54 miod Exp $ */
/*
* Copyright (c) 2000-2004 Opsycon AB (www.opsycon.se)
@@ -57,6 +57,7 @@ void macebus_intr_makemasks(void);
void macebus_splx(int);
uint32_t macebus_iointr(uint32_t, struct trap_frame *);
uint32_t macebus_aux(uint32_t, struct trap_frame *);
+void mace_setintrmask(int);
u_int8_t mace_read_1(bus_space_tag_t, bus_space_handle_t, bus_size_t);
u_int16_t mace_read_2(bus_space_tag_t, bus_space_handle_t, bus_size_t);
@@ -157,6 +158,20 @@ struct machine_bus_dma_tag mace_bus_dma_tag = {
};
/*
+ * CRIME/MACE interrupt handling declarations: 32 CRIME sources, 32 MACE
+ * sources (unmanaged); 1 level.
+ * We define another level for periodic tasks as well.
+ */
+
+struct intrhand *mace_intrhand[CRIME_NINTS];
+
+#define INTPRI_MACEIO (INTPRI_CLOCK + 1)
+#define INTPRI_MACEAUX (INTPRI_MACEIO + 1)
+
+uint64_t mace_intem;
+uint64_t mace_imask[NIPLS];
+
+/*
* Match bus only to targets which have this bus.
*/
int
@@ -237,12 +252,12 @@ macebusattach(struct device *parent, struct device *self, void *aux)
* Map and setup MACE ISA control registers.
*/
if (bus_space_map(&macebus_tag, MACE_ISA_OFFS, 0x400, 0, &mace_h)) {
- printf("%s: can't map MACE ISA control registers\n",
+ printf("%s: can't map MACE control registers\n",
self->dv_xname);
return;
}
- /* Turn on all interrupts except for MACE compare/timer. */
+ /* Turn on all MACE interrupts except for MACE compare/timer. */
bus_space_write_8(&macebus_tag, mace_h, MACE_ISA_INT_MASK,
0xffffffff & ~MACE_ISA_INT_TIMER);
bus_space_write_8(&macebus_tag, mace_h, MACE_ISA_INT_STAT, 0);
@@ -430,93 +445,56 @@ macebus_device_to_pa(bus_addr_t addr)
* Macebus interrupt handler driver.
*/
-uint64_t mace_intem = 0x0;
-static uint32_t intrtype[INTMASKSIZE];
-static uint32_t intrmask[INTMASKSIZE];
-static uint32_t intrlevel[INTMASKSIZE];
-
-static int fakeintr(void *);
-static int fakeintr(void *a) {return 0;}
-
/*
* Establish an interrupt handler called from the dispatcher.
* The interrupt function established should return zero if there was nothing
* to serve (no int) and non-zero when an interrupt was serviced.
+ *
* Interrupts are numbered from 1 and up where 1 maps to HW int 0.
+ * XXX There is no reason to keep this... except for hardcoded interrupts
+ * XXX in kernel configuration files...
*/
void *
macebus_intr_establish(void *icp, u_long irq, int type, int level,
int (*ih_fun)(void *), void *ih_arg, const char *ih_what)
{
struct intrhand **p, *q, *ih;
- static struct intrhand fakehand = {NULL, fakeintr};
- int edge;
- extern int cold;
- static int initialized = 0;
-
- if (!initialized) {
- /*INIT CODE HERE*/
- initialized = 1;
- }
+ int s;
- if (irq > SPL_CLOCK || irq < 1) {
+#ifdef DIAGNOSTIC
+ if (irq > CRIME_NINTS || irq < 1)
panic("intr_establish: illegal irq %d", irq);
- }
+#endif
+
irq -= 1; /* Adjust for 1 being first (0 is no int) */
- /* No point in sleeping unless someone can free memory. */
- ih = malloc(sizeof *ih, M_DEVBUF, cold ? M_NOWAIT : M_WAITOK);
+ ih = malloc(sizeof *ih, M_DEVBUF, M_NOWAIT);
if (ih == NULL)
- panic("intr_establish: can't malloc handler info");
+ return NULL;
- if (type == IST_NONE || type == IST_PULSE)
- panic("intr_establish: bogus type");
-
- switch (intrtype[irq]) {
- case IST_EDGE:
- case IST_LEVEL:
- if (type == intrtype[irq])
- break;
- }
+ ih->ih_next = NULL;
+ ih->ih_fun = ih_fun;
+ ih->ih_arg = ih_arg;
+ ih->ih_level = level;
+ ih->ih_irq = irq + 1;
+ evcount_attach(&ih->ih_count, ih_what, (void *)&ih->ih_irq,
+ &evcount_intr);
- switch (type) {
- case IST_EDGE:
- edge |= 1 << irq;
- break;
- case IST_LEVEL:
- edge &= ~(1 << irq);
- break;
- }
+ s = splhigh();
/*
* Figure out where to put the handler.
* This is O(N^2), but we want to preserve the order, and N is
* generally small.
*/
- for (p = &intrhand[irq]; (q = *p) != NULL; p = &q->ih_next)
+ for (p = &mace_intrhand[irq]; (q = *p) != NULL; p = &q->ih_next)
;
+ *p = ih;
- /*
- * Actually install a fake handler momentarily, since we might be doing
- * this with interrupts enabled and don't want the real routine called
- * until masking is set up.
- */
- fakehand.ih_level = level;
- *p = &fakehand;
-
+ mace_intem |= 1UL << irq;
macebus_intr_makemasks();
- /*
- * Poke the real handler in now.
- */
- ih->ih_fun = ih_fun;
- ih->ih_arg = ih_arg;
- ih->ih_next = NULL;
- ih->ih_level = level;
- ih->ih_irq = irq + 1;
- evcount_attach(&ih->ih_count, ih_what, (void *)&ih->ih_irq,
- &evcount_intr);
- *p = ih;
+ splx(s); /* causes hw mask update */
return (ih);
}
@@ -536,165 +514,85 @@ macebus_intr_makemasks(void)
{
int irq, level;
struct intrhand *q;
+ uint intrlevel[CRIME_NINTS];
/* First, figure out which levels each IRQ uses. */
- for (irq = 0; irq < INTMASKSIZE; irq++) {
- int levels = 0;
- for (q = intrhand[irq]; q; q = q->ih_next)
+ for (irq = 0; irq < CRIME_NINTS; irq++) {
+ uint levels = 0;
+ for (q = mace_intrhand[irq]; q; q = q->ih_next)
levels |= 1 << q->ih_level;
intrlevel[irq] = levels;
}
/* Then figure out which IRQs use each level. */
- for (level = IPL_NONE; level < NIPLS; level++) {
- int irqs = 0;
- for (irq = 0; irq < INTMASKSIZE; irq++)
+ for (level = IPL_NONE; level < IPL_HIGH; level++) {
+ uint64_t irqs = 0;
+ for (irq = 0; irq < CRIME_NINTS; irq++)
if (intrlevel[irq] & (1 << level))
- irqs |= 1 << irq;
- if (level != IPL_NONE)
- irqs |= SINT_ALLMASK;
- imask[level] = irqs;
+ irqs |= 1UL << irq;
+ mace_imask[level] = irqs;
}
/*
* There are tty, network and disk drivers that use free() at interrupt
- * time, so imp > (tty | net | bio).
+ * time, so vm > (tty | net | bio).
*
* Enforce a hierarchy that gives slow devices a better chance at not
* dropping data.
*/
- imask[IPL_NET] |= imask[IPL_BIO];
- imask[IPL_TTY] |= imask[IPL_NET];
- imask[IPL_VM] |= imask[IPL_TTY];
- imask[IPL_CLOCK] |= imask[IPL_VM] | SPL_CLOCKMASK;
+ mace_imask[IPL_NET] |= mace_imask[IPL_BIO];
+ mace_imask[IPL_TTY] |= mace_imask[IPL_NET];
+ mace_imask[IPL_VM] |= mace_imask[IPL_TTY];
+ mace_imask[IPL_CLOCK] |= mace_imask[IPL_VM];
/*
* These are pseudo-levels.
*/
- imask[IPL_NONE] = 0;
- imask[IPL_HIGH] = -1;
-
- /* And eventually calculate the complete masks. */
- for (irq = 0; irq < INTMASKSIZE; irq++) {
- int irqs = 1 << irq;
- for (q = intrhand[irq]; q; q = q->ih_next)
- irqs |= imask[q->ih_level];
- intrmask[irq] = irqs | SINT_ALLMASK;
- }
-
- /* Lastly, determine which IRQs are actually in use. */
- irq = 0;
- for (level = 0; level < INTMASKSIZE; level++) {
- if (intrhand[level]) {
- irq |= 1 << level;
- }
- }
- mace_intem = irq & 0x0000ffff;
- hw_setintrmask(0);
+ mace_imask[IPL_NONE] = 0;
+ mace_imask[IPL_HIGH] = -1UL;
}
void
-macebus_splx(int newcpl)
+macebus_splx(int newipl)
{
struct cpu_info *ci = curcpu();
- /* Update masks to new cpl. Order highly important! */
- __asm__ (" .set noreorder\n");
- ci->ci_cpl = newcpl;
- __asm__ (" sync\n .set reorder\n");
- hw_setintrmask(newcpl);
+ /* Update masks to new ipl. Order highly important! */
+ __asm__ (".set noreorder\n");
+ ci->ci_ipl = newipl;
+ __asm__ ("sync\n\t.set reorder\n");
+ mace_setintrmask(newipl);
/* If we still have softints pending trigger processing. */
- if (ci->ci_softpending & ~newcpl)
+ if (ci->ci_softpending != 0 && newipl < IPL_SOFTINT)
setsoftintr0();
}
/*
- * Process interrupts. The parameter pending has non-masked interrupts.
+ * Crime interrupt handler.
*/
-uint32_t
-macebus_iointr(uint32_t hwpend, struct trap_frame *cf)
-{
- struct intrhand *ih;
- uint32_t caught, vm;
- int v;
- uint32_t pending;
- u_int64_t intstat, isastat, mask;
-#ifdef DIAGNOSTIC
- static int spurious = 0;
-#endif
- struct cpu_info *ci = curcpu();
-
- intstat = bus_space_read_8(&crimebus_tag, crime_h, CRIME_INT_STAT);
- intstat &= 0xffff;
-
- isastat = bus_space_read_8(&macebus_tag, mace_h, MACE_ISA_INT_STAT);
- caught = 0;
-
- /* Mask off masked interrupts and save them as pending. */
- if (intstat & cf->cpl) {
- mask = bus_space_read_8(&crimebus_tag, crime_h, CRIME_INT_MASK);
- bus_space_write_8(&crimebus_tag, crime_h, CRIME_INT_MASK, mask);
- caught++;
- }
-
- /* Scan all unmasked. Scan the first 16 for now. */
- pending = intstat & ~cf->cpl;
-
- for (v = 0, vm = 1; pending != 0 && v < 16 ; v++, vm <<= 1) {
- if (pending & vm) {
- ih = intrhand[v];
-
- while (ih) {
- ih->frame = cf;
- if ((*ih->ih_fun)(ih->ih_arg)) {
- caught |= vm;
- ih->ih_count.ec_count++;
- }
- ih = ih->ih_next;
- }
- }
- }
-
- if (caught) {
-#ifdef DIAGNOSTIC
- spurious = 0;
-#endif
- return CR_INT_0;
- }
-
-#ifdef DIAGNOSTIC
- if (pending != 0) {
- intstat = bus_space_read_8(&crimebus_tag, crime_h,
- CRIME_INT_STAT) &
- bus_space_read_8(&crimebus_tag, crime_h, CRIME_INT_MASK);
- isastat = bus_space_read_8(&macebus_tag, mace_h,
- MACE_ISA_INT_STAT) &
- bus_space_read_8(&macebus_tag, mace_h, MACE_ISA_INT_MASK);
-
- if (intstat != 0 || isastat != 0) {
- printf("stray interrupt, mace mask %lx stat %lx\n"
- "crime mask %lx stat %lx hard %lx "
- "(pending %lx caught %lx)\n",
- bus_space_read_8(&macebus_tag, mace_h,
- MACE_ISA_INT_MASK),
- bus_space_read_8(&macebus_tag, mace_h,
- MACE_ISA_INT_STAT),
- bus_space_read_8(&crimebus_tag, crime_h,
- CRIME_INT_MASK),
- bus_space_read_8(&crimebus_tag, crime_h,
- CRIME_INT_STAT),
- bus_space_read_8(&crimebus_tag, crime_h,
- CRIME_INT_HARD),
- pending, caught);
- if (++spurious >= 10)
- panic("too many stray interrupts");
- }
- }
-#endif
-
- return 0; /* Not found here. */
-}
+#define INTR_FUNCTIONNAME macebus_iointr
+#define INTR_LOCAL_DECLS
+#define INTR_GETMASKS \
+do { \
+ isr = bus_space_read_8(&crimebus_tag, crime_h, CRIME_INT_STAT); \
+ imr = bus_space_read_8(&crimebus_tag, crime_h, CRIME_INT_MASK); \
+ bit = 63; \
+} while (0)
+#define INTR_MASKPENDING \
+ bus_space_write_8(&crimebus_tag, crime_h, CRIME_INT_MASK, imr & ~isr)
+#define INTR_IMASK(ipl) mace_imask[ipl]
+#define INTR_HANDLER(bit) mace_intrhand[bit]
+#define INTR_SPURIOUS(bit) \
+do { \
+ /* XXX +1 because of -1 in intr_establish() */ \
+ if (bit != 4) \
+ printf("spurious crime interrupt %d\n", bit + 1); \
+} while (0)
+#define INTR_MASKRESTORE \
+ bus_space_write_8(&crimebus_tag, crime_h, CRIME_INT_MASK, imr)
+
+#include <sgi/sgi/intr_template.c>
/*
* Macebus auxilary functions run each clock interrupt.
@@ -720,12 +618,12 @@ macebus_aux(uint32_t hwpend, struct trap_frame *cf)
}
bus_space_write_8(&macebus_tag, mace_h, MACE_ISA_MISC_REG, mask);
- return 0; /* Real clock int handler registers. */
+ return 0; /* Real clock int handler will claim the interrupt. */
}
void
-hw_setintrmask(uint32_t m)
+mace_setintrmask(int level)
{
*(volatile uint64_t *)(PHYS_TO_XKPHYS(CRIMEBUS_BASE, CCA_NC) +
- CRIME_INT_MASK) = mace_intem & ~((uint64_t)m);
+ CRIME_INT_MASK) = mace_intem & ~mace_imask[level];
}
diff --git a/sys/arch/sgi/sgi/genassym.cf b/sys/arch/sgi/sgi/genassym.cf
index d619bccd0a6..5833c78c70b 100644
--- a/sys/arch/sgi/sgi/genassym.cf
+++ b/sys/arch/sgi/sgi/genassym.cf
@@ -1,4 +1,4 @@
-# $OpenBSD: genassym.cf,v 1.16 2009/10/22 20:59:24 miod Exp $
+# $OpenBSD: genassym.cf,v 1.17 2009/10/22 22:08:54 miod Exp $
#
# Copyright (c) 1997 Per Fogelstrom / Opsycon AB
#
@@ -58,7 +58,7 @@ member pcb_segtab
struct cpu_info
member ci_curproc
member ci_curprocpaddr
-member ci_cpl
+member ci_ipl
export VM_MIN_KERNEL_ADDRESS
export SIGFPE
diff --git a/sys/arch/sgi/sgi/intr_template.c b/sys/arch/sgi/sgi/intr_template.c
new file mode 100644
index 00000000000..30ca5ceb517
--- /dev/null
+++ b/sys/arch/sgi/sgi/intr_template.c
@@ -0,0 +1,125 @@
+/* $OpenBSD: intr_template.c,v 1.1 2009/10/22 22:08:54 miod Exp $ */
+
+/*
+ * Copyright (c) 2009 Miodrag Vallat.
+ *
+ * Permission to use, copy, modify, and distribute this software for any
+ * purpose with or without fee is hereby granted, provided that the above
+ * copyright notice and this permission notice appear in all copies.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+ * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+ * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+ * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+ */
+
+/*
+ * Common interrupt dispatcher bowels.
+ *
+ * This file is not a standalone file; to use it, define the following
+ * macros and #include <sgi/sgi/intr_template.c>:
+ *
+ * INTR_FUNCTIONNAME interrupt handler function name
+ * INTR_GETMASKS logic to get `imr', `isr', and initialize `bit'
+ * INTR_HANDLER(bit) logic to access intrhand array head for `bit'
+ * INTR_IMASK(ipl) logic to access imask array for `ipl'
+ * INTR_LOCAL_DECLS local declarations (may be empty)
+ * INTR_MASKPENDING logic to mask `isr'
+ * INTR_MASKRESTORE logic to reset `imr'
+ * INTR_SPURIOUS(bit) print a spurious interrupt message for `bit'
+ */
+
+uint32_t
+INTR_FUNCTIONNAME(uint32_t hwpend, struct trap_frame *frame)
+{
+ struct cpu_info *ci = curcpu();
+ uint64_t imr, isr, mask;
+ int ipl;
+ int bit;
+ struct intrhand *ih;
+ int rc;
+ INTR_LOCAL_DECLS
+
+ INTR_GETMASKS;
+
+ isr &= imr;
+ if (isr == 0)
+ return 0; /* not for us */
+
+ /*
+ * Mask all pending interrupts.
+ */
+ INTR_MASKPENDING;
+
+ /*
+ * If interrupts are spl-masked, mask them and wait for splx()
+ * to reenable them when necessary.
+ */
+ if ((mask = isr & INTR_IMASK(frame->ipl)) != 0) {
+ isr &= ~mask;
+ imr &= ~mask;
+ }
+
+ /*
+ * Now process allowed interrupts.
+ */
+ if (isr != 0) {
+ int lvl, bitno;
+ uint64_t tmpisr;
+
+ __asm__ (".set noreorder\n");
+ ipl = ci->ci_ipl;
+ __asm__ ("sync\n\t.set reorder\n");
+
+ /* Service higher level interrupts first */
+ for (lvl = IPL_HIGH - 1; lvl != IPL_NONE; lvl--) {
+ tmpisr = isr & (INTR_IMASK(lvl) ^ INTR_IMASK(lvl - 1));
+ if (tmpisr == 0)
+ continue;
+ for (bitno = bit, mask = 1UL << bitno; tmpisr != 0;
+ bitno--, mask >>= 1) {
+ if ((tmpisr & mask) == 0)
+ continue;
+
+ rc = 0;
+ for (ih = INTR_HANDLER(bitno); ih != NULL;
+ ih = ih->ih_next) {
+ splraise(ih->ih_level);
+ ih->frame = frame;
+ if ((*ih->ih_fun)(ih->ih_arg) != 0) {
+ rc = 1;
+ ih->ih_count.ec_count++;
+ }
+ __asm__ (".set noreorder\n");
+ ci->ci_ipl = ipl;
+ __asm__ ("sync\n\t.set reorder\n");
+ }
+ if (rc == 0)
+ INTR_SPURIOUS(bitno);
+
+ isr ^= mask;
+ if ((tmpisr ^= mask) == 0)
+ break;
+ }
+ }
+
+ /*
+ * Reenable interrupts which have been serviced.
+ */
+ INTR_MASKRESTORE;
+ }
+
+ return hwpend;
+}
+
+#undef INTR_FUNCTIONNAME
+#undef INTR_GETMASKS
+#undef INTR_HANDLER
+#undef INTR_IMASK
+#undef INTR_LOCAL_DECLS
+#undef INTR_MASKPENDING
+#undef INTR_MASKRESTORE
+#undef INTR_SPURIOUS
diff --git a/sys/arch/sgi/sgi/ip27_machdep.c b/sys/arch/sgi/sgi/ip27_machdep.c
index 77d547a8758..1c4bfc70bb8 100644
--- a/sys/arch/sgi/sgi/ip27_machdep.c
+++ b/sys/arch/sgi/sgi/ip27_machdep.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: ip27_machdep.c,v 1.27 2009/10/22 20:59:24 miod Exp $ */
+/* $OpenBSD: ip27_machdep.c,v 1.28 2009/10/22 22:08:54 miod Exp $ */
/*
* Copyright (c) 2008, 2009 Miodrag Vallat.
@@ -70,14 +70,38 @@ int ip27_hub_intr_register(int, int, int *);
int ip27_hub_intr_establish(int (*)(void *), void *, int, int,
const char *);
void ip27_hub_intr_disestablish(int);
-uint32_t ip27_hub_intr_handler(uint32_t, struct trap_frame *);
+uint32_t hubpi_intr0(uint32_t, struct trap_frame *);
+uint32_t hubpi_intr1(uint32_t, struct trap_frame *);
void ip27_hub_intr_makemasks(void);
+void ip27_hub_setintrmask(int);
void ip27_hub_splx(int);
void ip27_attach_node(struct device *, int16_t);
int ip27_print(void *, const char *);
void ip27_nmi(void *);
+/*
+ * IP27 interrupt handling declarations: 128 hw sources, plus timers and
+ * hub error sources; 5 levels.
+ */
+
+struct intrhand *hubpi_intrhand0[HUBPI_NINTS];
+struct intrhand *hubpi_intrhand1[HUBPI_NINTS];
+
+#ifdef notyet
+#define INTPRI_XBOW_HUB (INTPRI_CLOCK + 1) /* HUB errors */
+#define INTPRI_XBOW_TIMER (INTPRI_XBOW_HUB + 1) /* prof timer */
+#define INTPRI_XBOW_CLOCK (INTPRI_XBOW_TIMER + 1) /* RTC */
+#define INTPRI_XBOW_HW1 (INTPRI_XBOW_CLOCK + 1) /* HW level 1 */
+#else
+#define INTPRI_XBOW_HW1 (INTPRI_CLOCK + 1) /* HW level 1 */
+#endif
+#define INTPRI_XBOW_HW0 (INTPRI_XBOW_HW1 + 1) /* HW level 0 */
+
+struct {
+ uint64_t hw[2];
+} hubpi_intem, hubpi_imask[NIPLS];
+
void
ip27_setup()
{
@@ -230,7 +254,8 @@ ip27_setup()
xbow_intr_widget_intr_establish = ip27_hub_intr_establish;
xbow_intr_widget_intr_disestablish = ip27_hub_intr_disestablish;
- set_intr(INTPRI_XBOWMUX, CR_INT_0, ip27_hub_intr_handler);
+ set_intr(INTPRI_XBOW_HW1, CR_INT_1, hubpi_intr1);
+ set_intr(INTPRI_XBOW_HW0, CR_INT_0, hubpi_intr0);
register_splx_handler(ip27_hub_splx);
/*
@@ -560,8 +585,6 @@ ip27_halt(int howto)
* Local HUB interrupt handling routines
*/
-uint64_t ip27_hub_intrmask;
-
/*
* Find a suitable interrupt bit for the given interrupt.
*/
@@ -571,18 +594,27 @@ ip27_hub_intr_register(int widget, int level, int *intrbit)
int bit;
/*
- * All interrupts will be serviced at hardware level 0,
- * so the `level' argument can be ignored.
- * On HUB, the low 7 bits of the level 0 interrupt register
- * are reserved.
+ * Try to allocate a bit on hardware level 0 first.
*/
- for (bit = SPL_CLOCK - 1; bit >= 7; bit--)
- if ((ip27_hub_intrmask & (1 << bit)) == 0)
- break;
+ for (bit = HUBPI_INTR0_WIDGET_MAX; bit >= HUBPI_INTR0_WIDGET_MIN; bit--)
+ if ((hubpi_intem.hw[0] & (1UL << bit)) == 0)
+ goto found;
- if (bit < 7)
- return EINVAL;
+#ifdef notyet
+ /*
+ * If all level 0 sources are in use, try to allocate a bit on
+ * level 1.
+ */
+ for (bit = HUBPI_INTR1_WIDGET_MAX; bit >= HUBPI_INTR1_WIDGET_MIN; bit--)
+ if ((hubpi_intem.hw[1] & (1UL << bit)) == 0) {
+ bit += HUBPI_NINTS;
+ goto found;
+ }
+#endif
+
+ return EINVAL;
+found:
*intrbit = bit;
return 0;
}
@@ -594,18 +626,27 @@ int
ip27_hub_intr_establish(int (*func)(void *), void *arg, int intrbit,
int level, const char *name)
{
- struct intrhand *ih;
+ struct intrhand *ih, **anchor;
+ int s;
#ifdef DIAGNOSTIC
- if (intrbit < 0 || intrbit >= SPL_CLOCK)
+ if (intrbit < 0 || intrbit >= HUBPI_NINTS + HUBPI_NINTS)
return EINVAL;
#endif
/*
* Widget interrupts are not supposed to be shared - the interrupt
- * mask is large enough for all widgets.
+ * mask is supposedly large enough for all interrupt sources.
+ *
+ * XXX On systems with many widgets and/or nodes, this assumption
+ * XXX will no longer stand; we'll need to implement interrupt
+ * XXX sharing at some point.
*/
- if (intrhand[intrbit] != NULL)
+ if (intrbit >= HUBPI_NINTS)
+ anchor = &hubpi_intrhand1[intrbit % HUBPI_NINTS];
+ else
+ anchor = &hubpi_intrhand0[intrbit];
+ if (*anchor != NULL)
return EEXIST;
ih = malloc(sizeof(*ih), M_DEVBUF, M_NOWAIT);
@@ -620,15 +661,15 @@ ip27_hub_intr_establish(int (*func)(void *), void *arg, int intrbit,
if (name != NULL)
evcount_attach(&ih->ih_count, name, &ih->ih_level,
&evcount_intr);
- intrhand[intrbit] = ih;
- ip27_hub_intrmask |= 1UL << intrbit;
+ s = splhigh();
+
+ *anchor = ih;
+
+ hubpi_intem.hw[intrbit / HUBPI_NINTS] |= 1UL << (intrbit % HUBPI_NINTS);
ip27_hub_intr_makemasks();
- /* XXX this assumes we run on cpu0 */
- IP27_LHUB_S(HUBPI_CPU0_IMR0,
- IP27_LHUB_L(HUBPI_CPU0_IMR0) | (1UL << intrbit));
- (void)IP27_LHUB_L(HUBPI_IR0);
+ splx(s); /* causes hw mask update */
return 0;
}
@@ -636,34 +677,35 @@ ip27_hub_intr_establish(int (*func)(void *), void *arg, int intrbit,
void
ip27_hub_intr_disestablish(int intrbit)
{
- struct intrhand *ih;
+ struct intrhand *ih, **anchor;
int s;
#ifdef DIAGNOSTIC
- if (intrbit < 0 || intrbit >= SPL_CLOCK)
+ if (intrbit < 0 || intrbit >= HUBPI_NINTS + HUBPI_NINTS)
return;
#endif
+ if (intrbit >= HUBPI_NINTS)
+ anchor = &hubpi_intrhand1[intrbit % HUBPI_NINTS];
+ else
+ anchor = &hubpi_intrhand0[intrbit];
+
s = splhigh();
- if ((ih = intrhand[intrbit]) == NULL) {
+ if ((ih = *anchor) == NULL) {
splx(s);
return;
}
- /* XXX this assumes we run on cpu0 */
- IP27_LHUB_S(HUBPI_CPU0_IMR0,
- IP27_LHUB_L(HUBPI_CPU0_IMR0) & ~(1UL << intrbit));
- (void)IP27_LHUB_L(HUBPI_IR0);
+ *anchor = NULL;
- intrhand[intrbit] = NULL;
-
- ip27_hub_intrmask &= ~(1UL << intrbit);
+ hubpi_intem.hw[intrbit / HUBPI_NINTS] &=
+ ~(1UL << (intrbit % HUBPI_NINTS));
ip27_hub_intr_makemasks();
- free(ih, M_DEVBUF);
-
splx(s);
+
+ free(ih, M_DEVBUF);
}
/*
@@ -672,27 +714,38 @@ ip27_hub_intr_disestablish(int intrbit)
void
ip27_hub_intr_makemasks()
{
- int irq, level;
+ int irq, level, i;
struct intrhand *q;
- uint32_t intrlevel[INTMASKSIZE];
+ uint intrlevel[HUBPI_NINTS + HUBPI_NINTS];
/* First, figure out which levels each IRQ uses. */
- for (irq = 0; irq < INTMASKSIZE; irq++) {
- int levels = 0;
- for (q = intrhand[irq]; q; q = q->ih_next)
+ for (irq = 0; irq < HUBPI_NINTS; irq++) {
+ uint levels = 0;
+ for (q = hubpi_intrhand0[irq]; q; q = q->ih_next)
+ levels |= 1 << q->ih_level;
+ for (q = hubpi_intrhand1[irq]; q; q = q->ih_next)
levels |= 1 << q->ih_level;
intrlevel[irq] = levels;
}
- /* Then figure out which IRQs use each level. */
- for (level = IPL_NONE; level < NIPLS; level++) {
- int irqs = 0;
- for (irq = 0; irq < INTMASKSIZE; irq++)
+ /*
+ * Then figure out which IRQs use each level.
+ * Note that we make sure never to overwrite imask[IPL_HIGH], in
+ * case an interrupt occurs during intr_disestablish() and causes
+ * an unfortunate splx() while we are here recomputing the masks.
+ */
+ for (level = IPL_NONE; level < IPL_HIGH; level++) {
+ uint64_t irqs = 0;
+ for (irq = 0; irq < HUBPI_NINTS; irq++)
if (intrlevel[irq] & (1 << level))
- irqs |= 1 << irq;
- if (level != IPL_NONE)
- irqs |= SINT_ALLMASK;
- imask[level] = irqs;
+ irqs |= 1UL << irq;
+ hubpi_imask[level].hw[0] = irqs;
+
+ irqs = 0;
+ for (irq = 0; irq < HUBPI_NINTS; irq++)
+ if (intrlevel[HUBPI_NINTS + irq] & (1 << level))
+ irqs |= 1UL << irq;
+ hubpi_imask[level].hw[1] = irqs;
}
/*
@@ -702,119 +755,106 @@ ip27_hub_intr_makemasks()
* Enforce a hierarchy that gives slow devices a better chance at not
* dropping data.
*/
- imask[IPL_NET] |= imask[IPL_BIO];
- imask[IPL_TTY] |= imask[IPL_NET];
- imask[IPL_VM] |= imask[IPL_TTY];
- imask[IPL_CLOCK] |= imask[IPL_VM] | SPL_CLOCKMASK;
-
- /*
- * These are pseudo-levels.
- */
- imask[IPL_NONE] = 0;
- imask[IPL_HIGH] = -1;
+ for (i = 0; i < 2; i++) {
+ hubpi_imask[IPL_NET].hw[i] |= hubpi_imask[IPL_BIO].hw[i];
+ hubpi_imask[IPL_TTY].hw[i] |= hubpi_imask[IPL_NET].hw[i];
+ hubpi_imask[IPL_VM].hw[i] |= hubpi_imask[IPL_TTY].hw[i];
+ hubpi_imask[IPL_CLOCK].hw[i] |= hubpi_imask[IPL_VM].hw[i];
- if(CPU_IS_PRIMARY(curcpu()))
- hw_setintrmask(0);
+ /*
+ * These are pseudo-levels.
+ */
+ hubpi_imask[IPL_NONE].hw[i] = 0;
+ hubpi_imask[IPL_HIGH].hw[i] = -1;
+ }
}
void
-ip27_hub_splx(int newcpl)
+ip27_hub_splx(int newipl)
{
struct cpu_info *ci = curcpu();
- /* Update masks to new cpl. Order highly important! */
- __asm__ (" .set noreorder\n");
- ci->ci_cpl = newcpl;
- __asm__ (" sync\n .set reorder\n");
+ /* Update masks to new ipl. Order highly important! */
+ __asm__ (".set noreorder\n");
+ ci->ci_ipl = newipl;
+ __asm__ ("sync\n\t.set reorder\n");
if (CPU_IS_PRIMARY(ci))
- hw_setintrmask(newcpl);
+ ip27_hub_setintrmask(newipl);
/* If we still have softints pending trigger processing. */
- if (ci->ci_softpending & ~newcpl)
+ if (ci->ci_softpending && newipl < IPL_SOFTINT)
setsoftintr0();
}
-uint32_t
-ip27_hub_intr_handler(uint32_t hwpend, struct trap_frame *frame)
-{
- uint64_t imr, isr;
- int icpl;
- int bit;
- uint32_t mask;
- struct intrhand *ih;
- int rc;
- struct cpu_info *ci = curcpu();
-
- /* XXX this assumes we run on cpu0 */
- isr = IP27_LHUB_L(HUBPI_IR0);
- imr = IP27_LHUB_L(HUBPI_CPU0_IMR0);
-
- isr &= imr;
- if (isr == 0)
- return 0; /* not for us */
-
- /*
- * Mask all pending interrupts.
- */
- IP27_LHUB_S(HUBPI_CPU0_IMR0, imr & ~isr);
- (void)IP27_LHUB_L(HUBPI_IR0);
-
- /*
- * If interrupts are spl-masked, mark them as pending only.
- */
- if ((mask = isr & frame->cpl) != 0) {
- isr &= ~mask;
- imr &= ~mask;
- }
-
- /*
- * Now process unmasked interrupts.
- */
- if (isr != 0) {
- __asm__ (" .set noreorder\n");
- icpl = ci->ci_cpl;
- __asm__ (" sync\n .set reorder\n");
-
- /* XXX Rework this to dispatch in decreasing levels */
- for (bit = SPL_CLOCK - 1, mask = 1 << bit; bit >= 7;
- bit--, mask >>= 1) {
- if ((isr & mask) == 0)
- continue;
-
- rc = 0;
- for (ih = intrhand[bit]; ih != NULL; ih = ih->ih_next) {
- splraise(imask[ih->ih_level]);
- ih->frame = frame;
- if ((*ih->ih_fun)(ih->ih_arg) != 0) {
- rc = 1;
- ih->ih_count.ec_count++;
- }
- }
- if (rc == 0)
- printf("spurious interrupt, source %d\n", bit);
-
- if ((isr ^= mask) == 0)
- break;
- }
-
- /*
- * Reenable interrupts which have been serviced.
- */
- IP27_LHUB_S(HUBPI_CPU0_IMR0, imr);
- (void)IP27_LHUB_L(HUBPI_IR0);
-
- __asm__ (" .set noreorder\n");
- ci->ci_cpl = icpl;
- __asm__ (" sync\n .set reorder\n");
- }
+/*
+ * Level 0 and level 1 interrupt dispatchers.
+ */
- return CR_INT_0;
-}
+#define INTR_FUNCTIONNAME hubpi_intr0
+#define INTR_LOCAL_DECLS
+#define INTR_GETMASKS \
+do { \
+ /* XXX this assumes we run on cpu0 */ \
+ isr = IP27_LHUB_L(HUBPI_IR0); \
+ imr = IP27_LHUB_L(HUBPI_CPU0_IMR0); \
+ bit = HUBPI_INTR0_WIDGET_MAX; \
+} while (0)
+#define INTR_MASKPENDING \
+do { \
+ IP27_LHUB_S(HUBPI_CPU0_IMR0, imr & ~isr); \
+ (void)IP27_LHUB_L(HUBPI_IR0); \
+} while (0)
+#define INTR_IMASK(ipl) hubpi_imask[ipl].hw[0]
+#define INTR_HANDLER(bit) hubpi_intrhand0[bit]
+#define INTR_SPURIOUS(bit) \
+do { \
+ printf("spurious interrupt, source %d\n", bit); \
+} while (0)
+#define INTR_MASKRESTORE \
+do { \
+ IP27_LHUB_S(HUBPI_CPU0_IMR0, imr); \
+ (void)IP27_LHUB_L(HUBPI_IR0); \
+} while (0)
+
+#include <sgi/sgi/intr_template.c>
+
+#define INTR_FUNCTIONNAME hubpi_intr1
+#define INTR_LOCAL_DECLS
+#define INTR_GETMASKS \
+do { \
+ /* XXX this assumes we run on cpu0 */ \
+ isr = IP27_LHUB_L(HUBPI_IR1); \
+ imr = IP27_LHUB_L(HUBPI_CPU0_IMR1); \
+ bit = HUBPI_INTR1_WIDGET_MAX; \
+} while (0)
+#define INTR_MASKPENDING \
+do { \
+ IP27_LHUB_S(HUBPI_CPU0_IMR1, imr & ~isr); \
+ (void)IP27_LHUB_L(HUBPI_IR1); \
+} while (0)
+#define INTR_IMASK(ipl) hubpi_imask[ipl].hw[1]
+#define INTR_HANDLER(bit) hubpi_intrhand1[bit]
+#define INTR_SPURIOUS(bit) \
+do { \
+ printf("spurious interrupt, source %d\n", bit + HUBPI_NINTS); \
+} while (0)
+#define INTR_MASKRESTORE \
+do { \
+ IP27_LHUB_S(HUBPI_CPU0_IMR1, imr); \
+ (void)IP27_LHUB_L(HUBPI_IR1); \
+} while (0)
+
+#include <sgi/sgi/intr_template.c>
void
-hw_setintrmask(uint32_t m)
+ip27_hub_setintrmask(int level)
{
- IP27_LHUB_S(HUBPI_CPU0_IMR0, ip27_hub_intrmask & ~((uint64_t)m));
+ /* XXX this assumes we run on cpu0 */
+ IP27_LHUB_S(HUBPI_CPU0_IMR0,
+ hubpi_intem.hw[0] & ~hubpi_imask[level].hw[0]);
(void)IP27_LHUB_L(HUBPI_IR0);
+ IP27_LHUB_S(HUBPI_CPU0_IMR1,
+ hubpi_intem.hw[1] & ~hubpi_imask[level].hw[1]);
+ (void)IP27_LHUB_L(HUBPI_IR1);
}
void
diff --git a/sys/arch/sgi/sgi/ip30_machdep.c b/sys/arch/sgi/sgi/ip30_machdep.c
index e8ad964a40f..ac05828741e 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.12 2009/10/22 20:05:28 miod Exp $ */
+/* $OpenBSD: ip30_machdep.c,v 1.13 2009/10/22 22:08:54 miod Exp $ */
/*
* Copyright (c) 2008, 2009 Miodrag Vallat.
@@ -198,14 +198,3 @@ ip30_widget_id(int16_t nasid, u_int widget, uint32_t *wid)
return 0;
}
-
-void
-hw_setintrmask(uint32_t m)
-{
- extern uint64_t heart_intem;
-
- paddr_t heart;
- heart = PHYS_TO_XKPHYS(HEART_PIU_BASE, CCA_NC);
- *(volatile uint64_t *)(heart + HEART_IMR(0)) =
- heart_intem & ~((uint64_t)m);
-}
diff --git a/sys/arch/sgi/sgi/mutex.c b/sys/arch/sgi/sgi/mutex.c
index 8e9202d3588..9b431010b5b 100644
--- a/sys/arch/sgi/sgi/mutex.c
+++ b/sys/arch/sgi/sgi/mutex.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: mutex.c,v 1.6 2009/08/13 17:06:05 miod Exp $ */
+/* $OpenBSD: mutex.c,v 1.7 2009/10/22 22:08:54 miod Exp $ */
/*
* Copyright (c) 2004 Artur Grabowski <art@openbsd.org>
@@ -43,16 +43,15 @@ void
mtx_init(struct mutex *mtx, int wantipl)
{
mtx->mtx_lock = 0;
- /* We can't access imask[] here, since MUTEX_INITIALIZER can't. */
mtx->mtx_wantipl = wantipl;
- mtx->mtx_oldcpl = IPL_NONE;
+ mtx->mtx_oldipl = IPL_NONE;
}
void
mtx_enter(struct mutex *mtx)
{
if (mtx->mtx_wantipl != IPL_NONE)
- mtx->mtx_oldcpl = splraise(imask[mtx->mtx_wantipl]);
+ mtx->mtx_oldipl = splraise(mtx->mtx_wantipl);
MUTEX_ASSERT_UNLOCKED(mtx);
mtx->mtx_lock = 1;
@@ -62,7 +61,7 @@ int
mtx_enter_try(struct mutex *mtx)
{
if (mtx->mtx_wantipl != IPL_NONE)
- mtx->mtx_oldcpl = splraise(imask[mtx->mtx_wantipl]);
+ mtx->mtx_oldipl = splraise(mtx->mtx_wantipl);
MUTEX_ASSERT_UNLOCKED(mtx);
mtx->mtx_lock = 1;
@@ -75,5 +74,5 @@ mtx_leave(struct mutex *mtx)
MUTEX_ASSERT_LOCKED(mtx);
mtx->mtx_lock = 0;
if (mtx->mtx_wantipl != IPL_NONE)
- splx(mtx->mtx_oldcpl);
+ splx(mtx->mtx_oldipl);
}
diff --git a/sys/arch/sgi/xbow/hub.h b/sys/arch/sgi/xbow/hub.h
index e80ea8dd508..63dfed9fb54 100644
--- a/sys/arch/sgi/xbow/hub.h
+++ b/sys/arch/sgi/xbow/hub.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: hub.h,v 1.5 2009/10/14 20:21:16 miod Exp $ */
+/* $OpenBSD: hub.h,v 1.6 2009/10/22 22:08:54 miod Exp $ */
/*
* Copyright (c) 2009 Miodrag Vallat.
@@ -109,6 +109,60 @@
#define HUBPI_OFFSET 0x00200000
/*
+ * ISR bit assignments.
+ */
+
+/** Level 1 interrupt */
+/* ?? MSC panic */
+#define HUBPI_ISR1_MSC_ERROR 63
+/* NI interface error */
+#define HUBPI_ISR1_NI_ERROR 62
+/* MD correctable error */
+#define HUBPI_ISR1_MD_COR_ERROR 61
+/* cpu correctable error B */
+#define HUBPI_ISR1_COR_ERROR_B 60
+/* cpu correctable error A */
+#define HUBPI_ISR1_COR_ERROR_A 59
+/* clock error */
+#define HUBPI_ISR1_CLOCK_ERROR 58
+/* IP35 NACK interrupts */
+#define HUBPI_ISR1_NACK_B 57
+#define HUBPI_ISR1_NACK_A 56
+/* IP35 LB error */
+#define HUBPI_ISR1_LB 55
+/* IP35 XB error */
+#define HUBPI_ISR1_XB 54
+/* 53-45 used by PROM */
+/* 44-43 available */
+/* 42-41 LLP errors */
+/* NI broadcast errors */
+#define HUBPI_ISR1_NI_ERROR_B 40
+#define HUBPI_ISR1_NI_ERROR_A 39
+/* 38-36 used by IP35 PROM */
+/* 35-0 available */
+
+/** Level 0 interrupt */
+/* 63-7 available */
+/* IPI interrupts */
+#define HUBPI_ISR0_IPI_B 6
+#define HUBPI_ISR0_IPI_A 5
+/* ? */
+#define HUBPI_ISR0_UART 4
+/* page migration interrupt */
+#define HUBPI_ISR0_PAGE_MIGRATION 3
+/* graphics->cpu interrupts */
+#define HUBPI_ISR0_GFX_B 2
+#define HUBPI_ISR0_GFX_A 1
+/* 0 reserved */
+
+#define HUBPI_INTR1_WIDGET_MAX 35
+#define HUBPI_INTR1_WIDGET_MIN 0
+#define HUBPI_INTR0_WIDGET_MAX 63
+#define HUBPI_INTR0_WIDGET_MIN 7
+
+#define HUBPI_NINTS 64 /* per register */
+
+/*
* HUB MD - Memory/Directory
*/
diff --git a/sys/arch/sgi/xbow/xbridge.c b/sys/arch/sgi/xbow/xbridge.c
index 1201db1f927..e454d79926b 100644
--- a/sys/arch/sgi/xbow/xbridge.c
+++ b/sys/arch/sgi/xbow/xbridge.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: xbridge.c,v 1.54 2009/10/22 19:55:45 miod Exp $ */
+/* $OpenBSD: xbridge.c,v 1.55 2009/10/22 22:08:54 miod Exp $ */
/*
* Copyright (c) 2008, 2009 Miodrag Vallat.
@@ -1072,11 +1072,18 @@ xbridge_intr_handler(void *v)
spurious = 0;
LIST_FOREACH(xih, &xi->xi_handlers, xih_nxt) {
- splraise(imask[xih->xih_level]);
+ splraise(xih->xih_level);
if ((*xih->xih_func)(xih->xih_arg) != 0) {
xih->xih_count.ec_count++;
rc = 1;
}
+ /*
+ * No need to lower spl here, as our caller will lower
+ * spl upon our return.
+ * However that splraise() is necessary so that interrupt
+ * handler code calling splx() will not cause our interrupt
+ * source to be unmasked.
+ */
}
if (rc == 0 && spurious == 0)
printf("%s: spurious irq %d\n", DEVNAME(xb), xi->xi_intrbit);
diff --git a/sys/arch/sgi/xbow/xheart.c b/sys/arch/sgi/xbow/xheart.c
index 34ba7f2b72d..ed51dfd4b1e 100644
--- a/sys/arch/sgi/xbow/xheart.c
+++ b/sys/arch/sgi/xbow/xheart.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: xheart.c,v 1.12 2009/10/22 20:59:24 miod Exp $ */
+/* $OpenBSD: xheart.c,v 1.13 2009/10/22 22:08:54 miod Exp $ */
/*
* Copyright (c) 2008 Miodrag Vallat.
@@ -43,8 +43,6 @@
struct xheart_softc {
struct device sc_dev;
struct onewire_bus sc_bus;
-
- uint64_t sc_intrmask;
};
int xheart_match(struct device *, void *, void *);
@@ -69,9 +67,30 @@ int xheart_intr_register(int, int, int *);
int xheart_intr_establish(int (*)(void *), void *, int, int, const char *);
void xheart_intr_disestablish(int);
uint32_t xheart_intr_handler(uint32_t, struct trap_frame *);
-void xheart_intr_makemasks(struct xheart_softc *);
+void xheart_intr_makemasks(void);
+void xheart_setintrmask(int);
void xheart_splx(int);
+/*
+ * HEART interrupt handling declarations: 64 sources; 5 levels.
+ */
+
+struct intrhand *xheart_intrhand[HEART_NINTS];
+
+#ifdef notyet
+#define INTPRI_HEART_4 (INTPRI_CLOCK + 1)
+#define INTPRI_HEART_3 (INTPRI_HEART_4 + 1)
+#define INTPRI_HEART_2 (INTPRI_HEART_3 + 1)
+#define INTPRI_HEART_1 (INTPRI_HEART_2 + 1)
+#define INTPRI_HEART_0 (INTPRI_HEART_1 + 1)
+#else
+#define INTPRI_HEART_2 (INTPRI_CLOCK + 1)
+#define INTPRI_HEART_0 (INTPRI_HEART_2 + 1)
+#endif
+
+uint64_t xheart_intem;
+uint64_t xheart_imask[NIPLS];
+
int
xheart_match(struct device *parent, void *match, void *aux)
{
@@ -118,7 +137,7 @@ xheart_attach(struct device *parent, struct device *self, void *aux)
xbow_intr_widget_intr_register = xheart_intr_register;
xbow_intr_widget_intr_establish = xheart_intr_establish;
xbow_intr_widget_intr_disestablish = xheart_intr_disestablish;
- sc->sc_intrmask = 0;
+ xheart_intem = 0;
/*
* Acknowledge and disable all interrupts.
@@ -131,7 +150,15 @@ xheart_attach(struct device *parent, struct device *self, void *aux)
*(volatile uint64_t*)(heart + HEART_IMR(2)) = 0UL;
*(volatile uint64_t*)(heart + HEART_IMR(3)) = 0UL;
- set_intr(INTPRI_XBOWMUX, CR_INT_0, xheart_intr_handler);
+#ifdef notyet
+ set_intr(INTPRI_HEART_4, CR_INT_4, xheart_intr_handler);
+ set_intr(INTPRI_HEART_3, CR_INT_3, xheart_intr_handler);
+#endif
+ set_intr(INTPRI_HEART_2, CR_INT_2, xheart_intr_handler);
+#ifdef notyet
+ set_intr(INTPRI_HEART_1, CR_INT_1, xheart_intr_handler);
+#endif
+ set_intr(INTPRI_HEART_0, CR_INT_0, xheart_intr_handler);
register_splx_handler(xheart_splx);
}
}
@@ -232,7 +259,6 @@ xheart_ow_pulse(struct xheart_softc *sc, int pulse, int data)
int
xheart_intr_register(int widget, int level, int *intrbit)
{
- struct xheart_softc *sc = (void *)xheart_cd.cd_devs[0];
int bit;
/*
@@ -240,12 +266,12 @@ xheart_intr_register(int widget, int level, int *intrbit)
* so the `level' argument can be ignored.
*/
for (bit = HEART_INTR_WIDGET_MAX; bit >= HEART_INTR_WIDGET_MIN; bit--)
- if ((sc->sc_intrmask & (1 << bit)) == 0)
- break;
+ if ((xheart_intem & (1UL << bit)) == 0)
+ goto found;
- if (bit < HEART_INTR_WIDGET_MIN)
- return EINVAL;
+ return EINVAL;
+found:
*intrbit = bit;
return 0;
}
@@ -257,12 +283,11 @@ int
xheart_intr_establish(int (*func)(void *), void *arg, int intrbit,
int level, const char *name)
{
- struct xheart_softc *sc = (void *)xheart_cd.cd_devs[0];
struct intrhand *ih;
- paddr_t heart;
+ int s;
#ifdef DIAGNOSTIC
- if (intrbit < HEART_INTR_MIN || intrbit > HEART_INTR_MAX)
+ if (intrbit < 0 || intrbit >= HEART_NINTS)
return EINVAL;
#endif
@@ -270,7 +295,7 @@ xheart_intr_establish(int (*func)(void *), void *arg, int intrbit,
* HEART interrupts are not supposed to be shared - the interrupt
* mask is large enough for all widgets.
*/
- if (intrhand[intrbit] != NULL)
+ if (xheart_intrhand[intrbit] != NULL)
return EEXIST;
ih = malloc(sizeof(*ih), M_DEVBUF, M_NOWAIT);
@@ -285,14 +310,15 @@ xheart_intr_establish(int (*func)(void *), void *arg, int intrbit,
if (name != NULL)
evcount_attach(&ih->ih_count, name, &ih->ih_level,
&evcount_intr);
- intrhand[intrbit] = ih;
- sc->sc_intrmask |= 1UL << intrbit;
- xheart_intr_makemasks(sc);
+ s = splhigh();
+
+ xheart_intrhand[intrbit] = ih;
+
+ xheart_intem |= 1UL << intrbit;
+ xheart_intr_makemasks();
- /* XXX this assumes we run on cpu0 */
- heart = PHYS_TO_XKPHYS(HEART_PIU_BASE, CCA_NC);
- *(volatile uint64_t *)(heart + HEART_IMR(0)) |= 1UL << intrbit;
+ splx(s); /* causes hw mask update */
return 0;
}
@@ -300,70 +326,61 @@ xheart_intr_establish(int (*func)(void *), void *arg, int intrbit,
void
xheart_intr_disestablish(int intrbit)
{
- struct xheart_softc *sc = (void *)xheart_cd.cd_devs[0];
struct intrhand *ih;
- paddr_t heart;
int s;
#ifdef DIAGNOSTIC
- if (intrbit < HEART_INTR_MIN || intrbit > HEART_INTR_MAX)
+ if (intrbit < 0 || intrbit >= HEART_NINTS)
return;
#endif
s = splhigh();
- if ((ih = intrhand[intrbit]) == NULL) {
+ if ((ih = xheart_intrhand[intrbit]) == NULL) {
splx(s);
return;
}
- /* XXX this assumes we run on cpu0 */
- heart = PHYS_TO_XKPHYS(HEART_PIU_BASE, CCA_NC);
- *(volatile uint64_t *)(heart + HEART_IMR(0)) &= ~(1UL << intrbit);
+ xheart_intrhand[intrbit] = NULL;
- intrhand[intrbit] = NULL;
+ xheart_intem &= ~(1UL << intrbit);
+ xheart_intr_makemasks();
- sc->sc_intrmask &= ~(1UL << intrbit);
- xheart_intr_makemasks(sc);
+ splx(s);
free(ih, M_DEVBUF);
-
- splx(s);
}
/*
- * Xheart interrupt handler driver.
- */
-
-uint64_t heart_intem = 0;
-
-/*
* Recompute interrupt masks.
*/
void
-xheart_intr_makemasks(struct xheart_softc *sc)
+xheart_intr_makemasks()
{
int irq, level;
struct intrhand *q;
- uint32_t intrlevel[INTMASKSIZE];
+ uint intrlevel[HEART_NINTS];
/* First, figure out which levels each IRQ uses. */
- for (irq = 0; irq < INTMASKSIZE; irq++) {
- int levels = 0;
- for (q = intrhand[irq]; q; q = q->ih_next)
+ for (irq = 0; irq < HEART_NINTS; irq++) {
+ uint levels = 0;
+ for (q = xheart_intrhand[irq]; q; q = q->ih_next)
levels |= 1 << q->ih_level;
intrlevel[irq] = levels;
}
- /* Then figure out which IRQs use each level. */
- for (level = IPL_NONE; level < NIPLS; level++) {
- int irqs = 0;
- for (irq = 0; irq < INTMASKSIZE; irq++)
+ /*
+ * Then figure out which IRQs use each level.
+ * Note that we make sure never to overwrite imask[IPL_HIGH], in
+ * case an interrupt occurs during intr_disestablish() and causes
+ * an unfortunate splx() while we are here recomputing the masks.
+ */
+ for (level = IPL_NONE; level < IPL_HIGH; level++) {
+ uint64_t irqs = 0;
+ for (irq = 0; irq < HEART_NINTS; irq++)
if (intrlevel[irq] & (1 << level))
- irqs |= 1 << irq;
- if (level != IPL_NONE)
- irqs |= SINT_ALLMASK;
- imask[level] = irqs;
+ irqs |= 1UL << irq;
+ xheart_imask[level] = irqs;
}
/*
@@ -373,106 +390,87 @@ xheart_intr_makemasks(struct xheart_softc *sc)
* Enforce a hierarchy that gives slow devices a better chance at not
* dropping data.
*/
- imask[IPL_NET] |= imask[IPL_BIO];
- imask[IPL_TTY] |= imask[IPL_NET];
- imask[IPL_VM] |= imask[IPL_TTY];
- imask[IPL_CLOCK] |= imask[IPL_VM] | SPL_CLOCKMASK;
+ xheart_imask[IPL_NET] |= xheart_imask[IPL_BIO];
+ xheart_imask[IPL_TTY] |= xheart_imask[IPL_NET];
+ xheart_imask[IPL_VM] |= xheart_imask[IPL_TTY];
+ xheart_imask[IPL_CLOCK] |= xheart_imask[IPL_VM];
/*
* These are pseudo-levels.
*/
- imask[IPL_NONE] = 0;
- imask[IPL_HIGH] = -1;
-
- heart_intem = sc->sc_intrmask;
- if(CPU_IS_PRIMARY(curcpu()))
- hw_setintrmask(0);
+ xheart_imask[IPL_NONE] = 0;
+ xheart_imask[IPL_HIGH] = -1UL;
}
void
-xheart_splx(int newcpl)
+xheart_splx(int newipl)
{
struct cpu_info *ci = curcpu();
- /* Update masks to new cpl. Order highly important! */
- __asm__ (" .set noreorder\n");
- ci->ci_cpl = newcpl;
- __asm__ (" sync\n .set reorder\n");
+ /* Update masks to new ipl. Order highly important! */
+ __asm__ (".set noreorder\n");
+ ci->ci_ipl = newipl;
+ __asm__ ("sync\n\t.set reorder\n");
if (CPU_IS_PRIMARY(ci))
- hw_setintrmask(newcpl);
+ xheart_setintrmask(newipl);
/* If we still have softints pending trigger processing. */
- if (ci->ci_softpending & ~newcpl)
+ if (ci->ci_softpending != 0 && newipl < IPL_SOFTINT)
setsoftintr0();
}
-uint32_t
-xheart_intr_handler(uint32_t hwpend, struct trap_frame *frame)
-{
- struct cpu_info *ci = curcpu();
- paddr_t heart;
- uint64_t imr, isr;
- int icpl;
- int bit;
- uint32_t mask;
- struct intrhand *ih;
- int rc;
-
- heart = PHYS_TO_XKPHYS(HEART_PIU_BASE, CCA_NC);
- isr = *(volatile uint64_t *)(heart + HEART_ISR);
- imr = *(volatile uint64_t *)(heart + HEART_IMR(0));
-
- isr &= imr;
- if (isr == 0)
- return 0; /* not for us */
-
- /*
- * Mask all pending interrupts.
- */
- *(volatile uint64_t *)(heart + HEART_IMR(0)) &= ~isr;
-
- /*
- * If interrupts are spl-masked, mark them as pending only.
- */
- if ((mask = isr & frame->cpl) != 0) {
- isr &= ~mask;
- }
-
- /*
- * Now process unmasked interrupts.
- */
- if (isr != 0) {
- __asm__ (" .set noreorder\n");
- icpl = ci->ci_cpl;
- __asm__ (" sync\n .set reorder\n");
-
- /* XXX Rework this to dispatch in decreasing levels */
- for (bit = HEART_INTR_MAX, mask = 1 << bit;
- bit >= HEART_INTR_MIN; bit--, mask >>= 1) {
- if ((isr & mask) == 0)
- continue;
-
- rc = 0;
- for (ih = intrhand[bit]; ih != NULL; ih = ih->ih_next) {
- splraise(imask[ih->ih_level]);
- ih->frame = frame;
- if ((*ih->ih_fun)(ih->ih_arg) != 0) {
- rc = 1;
- ih->ih_count.ec_count++;
- }
- }
- if (rc == 0)
- printf("spurious interrupt, source %d\n", bit);
- }
-
- /*
- * Reenable interrupts which have been serviced.
- */
- *(volatile uint64_t *)(heart + HEART_IMR(0)) |= isr;
+/*
+ * Heart interrupt handler. Can be registered at any hardware interrupt level.
+ */
- __asm__ (" .set noreorder\n");
- ci->ci_cpl = icpl;
- __asm__ (" sync\n .set reorder\n");
- }
+#define INTR_FUNCTIONNAME xheart_intr_handler
+#define INTR_LOCAL_DECLS \
+ paddr_t heart = PHYS_TO_XKPHYS(HEART_PIU_BASE, CCA_NC);
+#define INTR_GETMASKS \
+do { \
+ isr = *(volatile uint64_t *)(heart + HEART_ISR); \
+ imr = *(volatile uint64_t *)(heart + HEART_IMR(0)); \
+ switch (hwpend) { \
+ case CR_INT_0: \
+ isr &= HEART_ISR_LVL0_MASK; \
+ bit = HEART_ISR_LVL0_MAX; \
+ break; \
+ case CR_INT_1: \
+ isr &= HEART_ISR_LVL1_MASK; \
+ bit = HEART_ISR_LVL1_MAX; \
+ break; \
+ case CR_INT_2: \
+ isr &= HEART_ISR_LVL2_MASK; \
+ bit = HEART_ISR_LVL2_MAX; \
+ break; \
+ case CR_INT_3: \
+ isr &= HEART_ISR_LVL3_MASK; \
+ bit = HEART_ISR_LVL3_MAX; \
+ break; \
+ case CR_INT_4: \
+ isr &= HEART_ISR_LVL4_MASK; \
+ bit = HEART_ISR_LVL4_MAX; \
+ break; \
+ default: \
+ return 0; /* can't happen */ \
+ } \
+} while (0)
+#define INTR_MASKPENDING \
+ *(volatile uint64_t *)(heart + HEART_IMR(0)) &= ~isr
+#define INTR_IMASK(ipl) xheart_imask[ipl]
+#define INTR_HANDLER(bit) xheart_intrhand[bit]
+#define INTR_SPURIOUS(bit) \
+do { \
+ printf("spurious xheart interrupt %d\n", bit); \
+} while (0)
+#define INTR_MASKRESTORE \
+ *(volatile uint64_t *)(heart + HEART_IMR(0)) = imr
+
+#include <sgi/sgi/intr_template.c>
- return CR_INT_0;
+void
+xheart_setintrmask(int level)
+{
+ paddr_t heart = PHYS_TO_XKPHYS(HEART_PIU_BASE, CCA_NC);
+ *(volatile uint64_t *)(heart + HEART_IMR(0)) =
+ xheart_intem & ~xheart_imask[level];
}
diff --git a/sys/arch/sgi/xbow/xheartreg.h b/sys/arch/sgi/xbow/xheartreg.h
index 51110690bb5..1b3ba2968c7 100644
--- a/sys/arch/sgi/xbow/xheartreg.h
+++ b/sys/arch/sgi/xbow/xheartreg.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: xheartreg.h,v 1.2 2009/04/18 14:48:09 miod Exp $ */
+/* $OpenBSD: xheartreg.h,v 1.3 2009/10/22 22:08:54 miod Exp $ */
/*
* Copyright (c) 2008 Miodrag Vallat.
@@ -46,13 +46,59 @@
#define HEART_ISR 0x00010030
/*
- * ISR bit assignments (partial).
+ * ISR bit assignments.
*/
-#define HEART_INTR_ACFAIL 15
-#define HEART_INTR_POWER 14
-#define HEART_INTR_WIDGET_MAX 13
-#define HEART_INTR_WIDGET_MIN 0
+/** Level 4 interrupt: hardware error */
+#define HEART_ISR_LVL4_MASK 0xfff8000000000000UL
+#define HEART_ISR_LVL4_MAX 63
+/* Heart (widget 8) error */
+#define HEART_ISR_WID08_ERROR 63
+/* CPU bus error */
+#define HEART_ISR_CPU_BUSERR(c) (59 + (c))
+/* Crossbow (widget 0) error */
+#define HEART_ISR_WID00_ERROR 58
+/* Widget error */
+#define HEART_ISR_WID0F_ERROR 57
+#define HEART_ISR_WID0E_ERROR 56
+#define HEART_ISR_WID0D_ERROR 55
+#define HEART_ISR_WID0C_ERROR 54
+#define HEART_ISR_WID0B_ERROR 53
+#define HEART_ISR_WID0A_ERROR 52
+#define HEART_ISR_WID09_ERROR 51
-#define HEART_INTR_MAX 15
-#define HEART_INTR_MIN 0
+#define HEART_ISR_WID_ERROR(w) \
+ ((w) == 0 ? HEART_ISR_WID00_ERROR : \
+ (w) == 8 ? HEART_ISR_WID08_ERROR : HEART_ISR_WID09_ERROR + (w) - 9)
+
+/** Level 3 interrupt: heart counter/timer */
+#define HEART_ISR_LVL3_MASK 0x0004000000000000UL
+#define HEART_ISR_LVL3_MAX 50
+/* Crossbow clock */
+#define HEART_ISR_HEARTCLOCK 50
+
+/** Level 2 interrupt */
+#define HEART_ISR_LVL2_MASK 0x0003ffff00000000UL
+#define HEART_ISR_LVL2_MAX 49
+/* IPI */
+#define HEART_ISR_IPI(c) (46 + (c))
+/* Debugger interrupts */
+#define HEART_ISR_DBG(c) (42 + (c))
+/* Power switch */
+#define HEART_ISR_POWER 41
+/* 40-32 freely available */
+
+/** Level 1 interrupt */
+#define HEART_ISR_LVL1_MASK 0x00000000ffff0000UL
+#define HEART_ISR_LVL1_MAX 31
+/* 31-16 freely available */
+
+/** Level 0 interrupt */
+#define HEART_ISR_LVL0_MASK 0x000000000000ffffUL
+#define HEART_ISR_LVL0_MAX 15
+/* 15-3 freely available */
+
+#define HEART_INTR_WIDGET_MAX 15
+#define HEART_INTR_WIDGET_MIN 3
+
+#define HEART_NINTS 64