summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMark Kettenis <kettenis@cvs.openbsd.org>2011-03-24 10:52:23 +0000
committerMark Kettenis <kettenis@cvs.openbsd.org>2011-03-24 10:52:23 +0000
commitf29d0c90edafff04236dcb7aa2d03d51267679b7 (patch)
tree80895e7fa229ed970129f50966ea20074b7ab3dd
parente9f91f62c09a0cfba8dca21498e8e8c2f45e85ea (diff)
Process interrupts in priority order. Joint work with jsing@.
ok jsing@
-rw-r--r--sys/arch/hppa/hppa/intr.c76
1 files changed, 38 insertions, 38 deletions
diff --git a/sys/arch/hppa/hppa/intr.c b/sys/arch/hppa/hppa/intr.c
index 4cf7ad594f4..a703a9dd7f8 100644
--- a/sys/arch/hppa/hppa/intr.c
+++ b/sys/arch/hppa/hppa/intr.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: intr.c,v 1.37 2011/01/01 16:33:37 kettenis Exp $ */
+/* $OpenBSD: intr.c,v 1.38 2011/03/24 10:52:22 kettenis Exp $ */
/*
* Copyright (c) 2002-2004 Michael Shalayeff
@@ -276,11 +276,10 @@ cpu_intr(void *v)
{
struct cpu_info *ci = curcpu();
struct trapframe *frame = v;
+ struct hppa_iv *iv;
+ int pri, r, s, bit;
u_long mask;
-#ifdef MULTIPROCESSOR
- int pri;
-#endif
- int s;
+ void *arg;
mtctl(0, CR_EIEM);
@@ -288,54 +287,55 @@ cpu_intr(void *v)
if (ci->ci_in_intr++)
frame->tf_flags |= TFF_INTR;
- while ((mask = ci->ci_ipending & ~imask[s])) {
- int r, bit = fls(mask) - 1;
+ /* Process higher priority interrupts first. */
+ for (pri = NIPL - 1; pri > s; pri--) {
-#ifdef MULTIPROCESSOR
- /* XXX - Ensure that IPIs run first. */
- if (mask & (1 << 30))
- bit = 30;
-#endif
+ mask = imask[pri] ^ imask[pri - 1];
+
+ while (ci->ci_ipending & mask) {
+ bit = fls(ci->ci_ipending & mask) - 1;
+ iv = &intr_table[bit];
- struct hppa_iv *iv = &intr_table[bit];
+ ci->ci_ipending &= ~(1L << bit);
- ci->ci_ipending &= ~(1L << bit);
- if (iv->flags & HPPA_IV_CALL)
- continue;
+ if (iv->flags & HPPA_IV_CALL)
+ continue;
- uvmexp.intrs++;
- if (iv->flags & HPPA_IV_SOFT)
- uvmexp.softs++;
+ uvmexp.intrs++;
+ if (iv->flags & HPPA_IV_SOFT)
+ uvmexp.softs++;
- ci->ci_cpl = iv->pri;
- mtctl(frame->tf_eiem, CR_EIEM);
+ ci->ci_cpl = iv->pri;
+ mtctl(frame->tf_eiem, CR_EIEM);
#ifdef MULTIPROCESSOR
- pri = iv->pri;
- if (pri < IPL_IPI && s < IPL_SCHED)
- __mp_lock(&kernel_lock);
+ if (pri < IPL_IPI && s < IPL_SCHED)
+ __mp_lock(&kernel_lock);
#endif
- for (r = iv->flags & HPPA_IV_SOFT;
- iv && iv->handler; iv = iv->next)
- /* no arg means pass the frame */
- if ((iv->handler)(iv->arg? iv->arg : v) == 1) {
- if (iv->cnt)
- iv->cnt->ec_count++;
- r |= 1;
+ for (r = iv->flags & HPPA_IV_SOFT;
+ iv && iv->handler; iv = iv->next) {
+ /* no arg means pass the frame */
+ arg = iv->arg ? iv->arg : v;
+ if ((iv->handler)(arg) == 1) {
+ if (iv->cnt)
+ iv->cnt->ec_count++;
+ r |= 1;
+ }
}
#if 0 /* XXX this does not work, lasi gives us double ints */
- if (!r) {
- ci->ci_cpl = 0;
- printf("stray interrupt %d\n", bit);
- }
+ if (!r) {
+ ci->ci_cpl = 0;
+ printf("stray interrupt %d\n", bit);
+ }
#endif
#ifdef MULTIPROCESSOR
- if (pri < IPL_IPI && s < IPL_SCHED)
- __mp_unlock(&kernel_lock);
+ if (pri < IPL_IPI && s < IPL_SCHED)
+ __mp_unlock(&kernel_lock);
#endif
- mtctl(0, CR_EIEM);
+ mtctl(0, CR_EIEM);
+ }
}
ci->ci_in_intr--;
ci->ci_cpl = s;