diff options
author | Miod Vallat <miod@cvs.openbsd.org> | 2003-06-23 09:23:32 +0000 |
---|---|---|
committer | Miod Vallat <miod@cvs.openbsd.org> | 2003-06-23 09:23:32 +0000 |
commit | 052ac3aaacdadc455e62ce4a3555bc41f56817af (patch) | |
tree | af62ca660b7e53e02f0c9dbc67b7cf543e3c260e | |
parent | 8af88431cda86370c50efa58de70466188f5fe5c (diff) |
Handle SBus cards with an "interrupts" property but no "intr" property.
Adapted from NetBSD in a quite ugly way, but it works...
ok deraadt@
-rw-r--r-- | sys/arch/sparc/sparc/autoconf.c | 39 |
1 files changed, 36 insertions, 3 deletions
diff --git a/sys/arch/sparc/sparc/autoconf.c b/sys/arch/sparc/sparc/autoconf.c index cdad86e3e42..87b04ec2073 100644 --- a/sys/arch/sparc/sparc/autoconf.c +++ b/sys/arch/sparc/sparc/autoconf.c @@ -1,4 +1,4 @@ -/* $OpenBSD: autoconf.c,v 1.58 2003/06/02 23:27:55 millert Exp $ */ +/* $OpenBSD: autoconf.c,v 1.59 2003/06/23 09:23:31 miod Exp $ */ /* $NetBSD: autoconf.c,v 1.73 1997/07/29 09:41:53 fair Exp $ */ /* @@ -137,6 +137,14 @@ LIST_HEAD(, mountroot_hook) mrh_list; static struct device fakerdrootdev = { DV_DISK, {}, NULL, 0, "rd0", NULL }; #endif +/* Translate SBus interrupt level to processor IPL */ +int intr_sbus2ipl_4c[] = { + 0, 1, 2, 3, 5, 7, 8, 9 +}; +int intr_sbus2ipl_4m[] = { + 0, 2, 3, 5, 7, 9, 11, 13 +}; + /* * Most configuration on the SPARC is done by matching OPENPROM Forth * device names with our internal names. @@ -981,7 +989,7 @@ romprop(rp, cp, node) const char *cp; register int node; { - register int len; + int len, n; union { char regbuf[256]; struct rom_reg rr[RA_MAXREG]; } u; static const char pl[] = "property length"; @@ -1020,6 +1028,32 @@ romprop(rp, cp, node) len = getprop(node, "intr", (void *)&rp->ra_intr, sizeof rp->ra_intr); if (len == -1) len = 0; + + /* + * Some SBus cards only provide an "interrupts" properly, listing + * SBus levels. But since obio devices will usually also provide + * both properties, only check for "interrupts" last. + */ + if (len == 0) { + u_int32_t *interrupts; + len = getproplen(node, "interrupts"); + if (len > 0 && + (interrupts = malloc(len, M_TEMP, M_NOWAIT)) != NULL) { + /* Build rom_intr structures from the list */ + getprop(node, "interrupts", interrupts, len); + len /= sizeof(u_int32_t); + for (n = 0; n < len; n++) { + rp->ra_intr[n].int_pri = CPU_ISSUN4M ? + intr_sbus2ipl_4m[interrupts[n]] : + intr_sbus2ipl_4c[interrupts[n]]; + rp->ra_intr[n].int_vec = 0; + }; + len *= sizeof(struct rom_intr); + free(interrupts, M_TEMP); + } else + len = 0; + } + if (len & 7) { printf("%s \"intr\" %s = %d (need multiple of 8)\n", cp, pl, len); @@ -1037,7 +1071,6 @@ romprop(rp, cp, node) if (CPU_ISSUN4M) { /* What's in these high bits anyway? */ rp->ra_intr[len].int_pri &= 0xf; - /* Look at "interrupts" property too? */ } #endif |