summaryrefslogtreecommitdiff
path: root/sys/arch/sparc64
diff options
context:
space:
mode:
authorJason Wright <jason@cvs.openbsd.org>2001-11-28 05:32:12 +0000
committerJason Wright <jason@cvs.openbsd.org>2001-11-28 05:32:12 +0000
commitb8c114e45206188bcb3d1e1c3f6ba8cf28a617b4 (patch)
tree5d2c79b64871d053f8355bdaf13ae358461bf8dc /sys/arch/sparc64
parentf2be4e06d06b3eac5bd5dad6f671aed37acea885 (diff)
rewrite large chunks of intr_establish to make it more clear what's going on.
Also, while here, fix a work around (hack) for the problem of more than 3 devices sharing a particular level (make a copy of the interrupt handler rather than modifying the one added to the pil table). This allows qec+qe (and probably qfe) to get through autoconf without hanging.
Diffstat (limited to 'sys/arch/sparc64')
-rw-r--r--sys/arch/sparc64/sparc64/intr.c75
1 files changed, 41 insertions, 34 deletions
diff --git a/sys/arch/sparc64/sparc64/intr.c b/sys/arch/sparc64/sparc64/intr.c
index 349d59b997b..5f188196ed5 100644
--- a/sys/arch/sparc64/sparc64/intr.c
+++ b/sys/arch/sparc64/sparc64/intr.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: intr.c,v 1.5 2001/09/21 02:02:22 art Exp $ */
+/* $OpenBSD: intr.c,v 1.6 2001/11/28 05:32:11 jason Exp $ */
/* $NetBSD: intr.c,v 1.39 2001/07/19 23:38:11 eeh Exp $ */
/*
@@ -247,8 +247,9 @@ intr_establish(level, ih)
ih->ih_pending = 0; /* XXXX caller should have done this before */
ih->ih_next = NULL;
for (p = &intrhand[level]; (q = *p) != NULL; p = &q->ih_next)
- ;
+ continue;
*p = ih;
+
/*
* Store in fast lookup table
*/
@@ -259,44 +260,50 @@ intr_establish(level, ih)
Debugger();
}
#endif
- if (ih->ih_number < MAXINTNUM && ih->ih_number >= 0) {
- if ((q = intrlev[ih->ih_number])) {
- struct intrhand *nih;
- /*
- * Interrupt is already there. We need to create a
- * new interrupt handler and interpose it.
- */
+
+ if (ih->ih_number <= 0 || ih->ih_number > MAXINTNUM)
+ panic("intr_establish: bad intr number %x", ih->ih_number);
+
+ q = intrlev[ih->ih_number];
+ if (q == NULL) {
+ /* No interrupt already there, just put handler in place. */
+ intrlev[ih->ih_number] = ih;
+ } else {
+ struct intrhand *nih;
+
+ /*
+ * Interrupt is already there. We need to create a
+ * new interrupt handler and interpose it.
+ */
#ifdef DEBUG
- printf("intr_establish: intr reused %x\n",
- ih->ih_number);
+ printf("intr_establish: intr reused %x\n", ih->ih_number);
#endif
- if (q->ih_fun != intr_list_handler) {
- nih = (struct intrhand *)
- malloc(sizeof(struct intrhand),
- M_DEVBUF, M_NOWAIT);
- /* Point the old IH at the new handler */
- *nih = *q;
- q->ih_fun = intr_list_handler;
- q->ih_arg = (void *)nih;
- nih->ih_next = NULL;
- }
- /* Add the ih to the head of the list */
- ih->ih_next = (struct intrhand *)q->ih_arg;
- q->ih_arg = (void *)ih;
+ if (q->ih_fun != intr_list_handler) {
+ nih = (struct intrhand *)malloc(sizeof(struct intrhand),
+ M_DEVBUF, M_NOWAIT);
+ /* Point the old IH at the new handler */
+ *nih = *q;
+ q->ih_fun = intr_list_handler;
+ q->ih_arg = (void *)nih;
+ nih->ih_next = NULL;
}
- else
- intrlev[ih->ih_number] = ih;
+ nih = (struct intrhand *)malloc(sizeof(struct intrhand),
+ M_DEVBUF, M_NOWAIT);
+ *nih = *ih;
+ /* Add the ih to the head of the list */
+ nih->ih_next = (struct intrhand *)q->ih_arg;
+ q->ih_arg = (void *)nih;
+ }
+
#ifdef NOT_DEBUG
- printf("\nintr_establish: vector %x pil %x mapintr %p "
- "clrintr %p fun %p arg %p\n",
- ih->ih_number, ih->ih_pil, (void *)ih->ih_map,
- (void *)ih->ih_clr, (void *)ih->ih_fun,
- (void *)ih->ih_arg);
- /*Debugger();*/
+ printf("\nintr_establish: vector %x pil %x mapintr %p "
+ "clrintr %p fun %p arg %p\n",
+ ih->ih_number, ih->ih_pil, (void *)ih->ih_map,
+ (void *)ih->ih_clr, (void *)ih->ih_fun,
+ (void *)ih->ih_arg);
#endif
- } else
- panic("intr_establish: bad intr number %x", ih->ih_number);
+
splx(s);
}