2002-09-27 06:24:06 +04:00
|
|
|
/* $NetBSD: isa.c,v 1.111 2002/09/27 02:24:30 thorpej Exp $ */
|
1994-10-27 07:14:23 +03:00
|
|
|
|
1993-03-21 12:45:37 +03:00
|
|
|
/*-
|
2002-01-08 00:46:56 +03:00
|
|
|
* Copyright (c) 1998, 2001 The NetBSD Foundation, Inc.
|
1998-08-15 07:51:30 +04:00
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* This code is derived from software contributed to The NetBSD Foundation
|
2002-01-08 00:46:56 +03:00
|
|
|
* by Charles M. Hannum; by Jason R. Thorpe of Wasabi Systems, Inc.
|
1993-03-21 12:45:37 +03:00
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
* 3. All advertising materials mentioning features or use of this software
|
|
|
|
* must display the following acknowledgement:
|
1998-08-15 07:51:30 +04:00
|
|
|
* This product includes software developed by the NetBSD
|
|
|
|
* Foundation, Inc. and its contributors.
|
|
|
|
* 4. Neither the name of The NetBSD Foundation nor the names of its
|
|
|
|
* contributors may be used to endorse or promote products derived
|
|
|
|
* from this software without specific prior written permission.
|
1993-03-21 12:45:37 +03:00
|
|
|
*
|
1998-08-15 07:51:30 +04:00
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
|
|
|
|
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
|
|
|
|
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
|
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
|
|
|
|
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
1993-03-21 12:45:37 +03:00
|
|
|
*/
|
|
|
|
|
2001-11-13 11:01:09 +03:00
|
|
|
#include <sys/cdefs.h>
|
2002-09-27 06:24:06 +04:00
|
|
|
__KERNEL_RCSID(0, "$NetBSD: isa.c,v 1.111 2002/09/27 02:24:30 thorpej Exp $");
|
2001-11-13 11:01:09 +03:00
|
|
|
|
1993-12-20 12:05:17 +03:00
|
|
|
#include <sys/param.h>
|
|
|
|
#include <sys/systm.h>
|
1994-03-06 20:37:56 +03:00
|
|
|
#include <sys/kernel.h>
|
1993-12-20 12:05:17 +03:00
|
|
|
#include <sys/malloc.h>
|
1994-03-29 08:34:18 +04:00
|
|
|
#include <sys/device.h>
|
1993-12-20 12:05:17 +03:00
|
|
|
|
1996-05-14 04:31:04 +04:00
|
|
|
#include <machine/intr.h>
|
|
|
|
|
1995-04-17 16:06:30 +04:00
|
|
|
#include <dev/isa/isareg.h>
|
|
|
|
#include <dev/isa/isavar.h>
|
1997-06-07 03:43:45 +04:00
|
|
|
#include <dev/isa/isadmareg.h>
|
1993-05-28 13:10:52 +04:00
|
|
|
|
1998-06-11 12:29:33 +04:00
|
|
|
#include "isadma.h"
|
|
|
|
|
1998-07-30 22:03:34 +04:00
|
|
|
#include "isapnp.h"
|
2001-06-19 16:45:23 +04:00
|
|
|
#if NISAPNP > 0
|
1998-07-30 22:03:34 +04:00
|
|
|
#include <dev/isapnp/isapnpreg.h>
|
|
|
|
#include <dev/isapnp/isapnpvar.h>
|
|
|
|
#endif
|
|
|
|
|
2002-01-08 00:46:56 +03:00
|
|
|
int isamatch(struct device *, struct cfdata *, void *);
|
|
|
|
void isaattach(struct device *, struct device *, void *);
|
|
|
|
int isaprint(void *, const char *);
|
1996-02-28 04:43:45 +03:00
|
|
|
|
1996-03-17 03:43:52 +03:00
|
|
|
struct cfattach isa_ca = {
|
|
|
|
sizeof(struct isa_softc), isamatch, isaattach
|
|
|
|
};
|
|
|
|
|
2002-01-08 00:46:56 +03:00
|
|
|
void isa_attach_knowndevs(struct isa_softc *);
|
|
|
|
void isa_free_knowndevs(struct isa_softc *);
|
|
|
|
|
|
|
|
int isasubmatch(struct device *, struct cfdata *, void *);
|
|
|
|
int isasearch(struct device *, struct cfdata *, void *);
|
1996-12-05 04:25:23 +03:00
|
|
|
|
1996-02-28 04:43:45 +03:00
|
|
|
int
|
2002-01-08 00:46:56 +03:00
|
|
|
isamatch(struct device *parent, struct cfdata *cf, void *aux)
|
1996-02-28 04:43:45 +03:00
|
|
|
{
|
|
|
|
struct isabus_attach_args *iba = aux;
|
|
|
|
|
2002-09-27 06:24:06 +04:00
|
|
|
if (strcmp(iba->iba_busname, cf->cf_name))
|
1996-02-28 04:43:45 +03:00
|
|
|
return (0);
|
|
|
|
|
|
|
|
/* XXX check other indicators */
|
|
|
|
|
|
|
|
return (1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2002-01-08 00:46:56 +03:00
|
|
|
isaattach(struct device *parent, struct device *self, void *aux)
|
1996-02-28 04:43:45 +03:00
|
|
|
{
|
|
|
|
struct isa_softc *sc = (struct isa_softc *)self;
|
|
|
|
struct isabus_attach_args *iba = aux;
|
|
|
|
|
2002-01-08 00:46:56 +03:00
|
|
|
TAILQ_INIT(&sc->sc_knowndevs);
|
|
|
|
sc->sc_dynamicdevs = 0;
|
|
|
|
|
1996-04-12 02:25:44 +04:00
|
|
|
isa_attach_hook(parent, self, iba);
|
1996-10-13 05:37:04 +04:00
|
|
|
printf("\n");
|
1996-02-28 04:43:45 +03:00
|
|
|
|
2002-01-08 00:46:56 +03:00
|
|
|
/* XXX Add code to fetch known-devices. */
|
|
|
|
|
1996-10-22 02:34:38 +04:00
|
|
|
sc->sc_iot = iba->iba_iot;
|
|
|
|
sc->sc_memt = iba->iba_memt;
|
1997-06-07 03:43:45 +04:00
|
|
|
sc->sc_dmat = iba->iba_dmat;
|
1996-04-12 02:25:44 +04:00
|
|
|
sc->sc_ic = iba->iba_ic;
|
1996-03-08 23:36:21 +03:00
|
|
|
|
1998-07-30 22:03:34 +04:00
|
|
|
#if NISAPNP > 0
|
|
|
|
/*
|
|
|
|
* Reset isapnp cards that the bios configured for us
|
|
|
|
*/
|
|
|
|
isapnp_isa_attach_hook(sc);
|
|
|
|
#endif
|
|
|
|
|
1998-06-11 12:29:33 +04:00
|
|
|
#if NISADMA > 0
|
1997-06-07 03:43:45 +04:00
|
|
|
/*
|
1998-06-09 04:00:21 +04:00
|
|
|
* Initialize our DMA state.
|
1997-06-07 03:43:45 +04:00
|
|
|
*/
|
1998-06-09 04:00:21 +04:00
|
|
|
isa_dmainit(sc->sc_ic, sc->sc_iot, sc->sc_dmat, self);
|
1998-06-11 12:29:33 +04:00
|
|
|
#endif
|
1997-06-07 03:43:45 +04:00
|
|
|
|
2002-01-08 00:46:56 +03:00
|
|
|
/* Attach all direct-config children. */
|
|
|
|
isa_attach_knowndevs(sc);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If we don't support dynamic hello/goodbye of devices,
|
|
|
|
* then free the knowndevs info now.
|
|
|
|
*/
|
|
|
|
if (sc->sc_dynamicdevs == 0)
|
|
|
|
isa_free_knowndevs(sc);
|
|
|
|
|
|
|
|
/* Attach all indrect-config children. */
|
1996-12-05 04:25:23 +03:00
|
|
|
config_search(isasearch, self, NULL);
|
1996-02-28 04:43:45 +03:00
|
|
|
}
|
|
|
|
|
2002-01-08 00:46:56 +03:00
|
|
|
void
|
|
|
|
isa_attach_knowndevs(struct isa_softc *sc)
|
|
|
|
{
|
|
|
|
struct isa_attach_args ia;
|
|
|
|
struct isa_knowndev *ik;
|
|
|
|
|
|
|
|
if (TAILQ_EMPTY(&sc->sc_knowndevs))
|
|
|
|
return;
|
|
|
|
|
|
|
|
TAILQ_FOREACH(ik, &sc->sc_knowndevs, ik_list) {
|
|
|
|
if (ik->ik_claimed != NULL)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
ia.ia_iot = sc->sc_iot;
|
|
|
|
ia.ia_memt = sc->sc_memt;
|
|
|
|
ia.ia_dmat = sc->sc_dmat;
|
|
|
|
ia.ia_ic = sc->sc_ic;
|
|
|
|
|
|
|
|
ia.ia_pnpname = ik->ik_pnpname;
|
|
|
|
ia.ia_pnpcompatnames = ik->ik_pnpcompatnames;
|
|
|
|
|
|
|
|
ia.ia_io = ik->ik_io;
|
|
|
|
ia.ia_nio = ik->ik_nio;
|
|
|
|
|
|
|
|
ia.ia_iomem = ik->ik_iomem;
|
|
|
|
ia.ia_niomem = ik->ik_niomem;
|
|
|
|
|
|
|
|
ia.ia_irq = ik->ik_irq;
|
|
|
|
ia.ia_nirq = ik->ik_nirq;
|
|
|
|
|
|
|
|
ia.ia_drq = ik->ik_drq;
|
|
|
|
ia.ia_ndrq = ik->ik_ndrq;
|
|
|
|
|
|
|
|
ia.ia_aux = NULL;
|
|
|
|
|
|
|
|
ik->ik_claimed = config_found_sm(&sc->sc_dev, &ia,
|
|
|
|
isaprint, isasubmatch);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
isa_free_knowndevs(struct isa_softc *sc)
|
|
|
|
{
|
|
|
|
struct isa_knowndev *ik;
|
|
|
|
struct isa_pnpname *ipn;
|
|
|
|
|
|
|
|
#define FREEIT(x) if (x != NULL) free(x, M_DEVBUF)
|
|
|
|
|
|
|
|
while ((ik = TAILQ_FIRST(&sc->sc_knowndevs)) != NULL) {
|
|
|
|
TAILQ_REMOVE(&sc->sc_knowndevs, ik, ik_list);
|
|
|
|
FREEIT(ik->ik_pnpname);
|
|
|
|
while ((ipn = ik->ik_pnpcompatnames) != NULL) {
|
|
|
|
ik->ik_pnpcompatnames = ipn->ipn_next;
|
|
|
|
free(ipn->ipn_name, M_DEVBUF);
|
|
|
|
free(ipn, M_DEVBUF);
|
|
|
|
}
|
|
|
|
FREEIT(ik->ik_io);
|
|
|
|
FREEIT(ik->ik_iomem);
|
|
|
|
FREEIT(ik->ik_irq);
|
|
|
|
FREEIT(ik->ik_drq);
|
|
|
|
free(ik, M_DEVBUF);
|
|
|
|
}
|
|
|
|
|
|
|
|
#undef FREEIT
|
|
|
|
}
|
|
|
|
|
1994-03-29 08:34:18 +04:00
|
|
|
int
|
2002-01-08 00:46:56 +03:00
|
|
|
isasubmatch(struct device *parent, struct cfdata *cf, void *aux)
|
1994-03-29 08:34:18 +04:00
|
|
|
{
|
|
|
|
struct isa_attach_args *ia = aux;
|
2002-01-08 00:46:56 +03:00
|
|
|
int i;
|
|
|
|
|
|
|
|
if (ia->ia_nio == 0) {
|
|
|
|
if (cf->cf_iobase != ISACF_PORT_DEFAULT)
|
|
|
|
return (0);
|
|
|
|
} else {
|
|
|
|
if (cf->cf_iobase != ISACF_PORT_DEFAULT &&
|
|
|
|
cf->cf_iobase != ia->ia_io[0].ir_addr)
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ia->ia_niomem == 0) {
|
|
|
|
if (cf->cf_maddr != ISACF_IOMEM_DEFAULT)
|
|
|
|
return (0);
|
|
|
|
} else {
|
|
|
|
if (cf->cf_maddr != ISACF_IOMEM_DEFAULT &&
|
|
|
|
cf->cf_maddr != ia->ia_iomem[0].ir_addr)
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ia->ia_nirq == 0) {
|
|
|
|
if (cf->cf_irq != ISACF_IRQ_DEFAULT)
|
|
|
|
return (0);
|
|
|
|
} else {
|
|
|
|
if (cf->cf_irq != ISACF_IRQ_DEFAULT &&
|
|
|
|
cf->cf_irq != ia->ia_irq[0].ir_irq)
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ia->ia_ndrq == 0) {
|
|
|
|
if (cf->cf_drq != ISACF_DRQ_DEFAULT)
|
|
|
|
return (0);
|
|
|
|
if (cf->cf_drq2 != ISACF_DRQ_DEFAULT)
|
|
|
|
return (0);
|
|
|
|
} else {
|
|
|
|
for (i = 0; i < 2; i++) {
|
|
|
|
if (i == ia->ia_ndrq)
|
|
|
|
break;
|
|
|
|
if (cf->cf_loc[ISACF_DRQ + i] != ISACF_DRQ_DEFAULT &&
|
|
|
|
cf->cf_loc[ISACF_DRQ + i] != ia->ia_drq[i].ir_drq)
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
for (; i < 2; i++) {
|
|
|
|
if (cf->cf_loc[ISACF_DRQ + i] != ISACF_DRQ_DEFAULT)
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ((*cf->cf_attach->ca_match)(parent, cf, aux));
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
isaprint(void *aux, const char *isa)
|
|
|
|
{
|
|
|
|
struct isa_attach_args *ia = aux;
|
|
|
|
const char *sep;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This block of code only fires if we have a direct-config'd
|
|
|
|
* device for which there is no driver match.
|
|
|
|
*/
|
|
|
|
if (isa != NULL) {
|
|
|
|
struct isa_pnpname *ipn;
|
|
|
|
|
|
|
|
if (ia->ia_pnpname != NULL)
|
|
|
|
printf("%s", ia->ia_pnpname);
|
|
|
|
if ((ipn = ia->ia_pnpcompatnames) != NULL) {
|
|
|
|
printf(" ("); /* ) */
|
|
|
|
for (sep = ""; ipn != NULL;
|
|
|
|
ipn = ipn->ipn_next, sep = " ") {
|
|
|
|
printf("%s%s", sep, ipn->ipn_name);
|
|
|
|
}
|
|
|
|
/* ( */ printf(")");
|
|
|
|
}
|
|
|
|
printf(" at %s", isa);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ia->ia_nio) {
|
|
|
|
sep = "";
|
|
|
|
printf(" port ");
|
|
|
|
for (i = 0; i < ia->ia_nio; i++) {
|
|
|
|
if (ia->ia_io[i].ir_size == 0)
|
|
|
|
continue;
|
|
|
|
printf("%s0x%x", sep, ia->ia_io[i].ir_addr);
|
|
|
|
if (ia->ia_io[i].ir_size > 1)
|
|
|
|
printf("-0x%x", ia->ia_io[i].ir_addr +
|
|
|
|
ia->ia_io[i].ir_size - 1);
|
|
|
|
sep = ",";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ia->ia_niomem) {
|
|
|
|
sep = "";
|
|
|
|
printf(" iomem ");
|
|
|
|
for (i = 0; i < ia->ia_niomem; i++) {
|
|
|
|
if (ia->ia_iomem[i].ir_size == 0)
|
|
|
|
continue;
|
|
|
|
printf("%s0x%x", sep, ia->ia_iomem[i].ir_addr);
|
|
|
|
if (ia->ia_iomem[i].ir_size > 1)
|
|
|
|
printf("-0x%x", ia->ia_iomem[i].ir_addr +
|
|
|
|
ia->ia_iomem[i].ir_size - 1);
|
|
|
|
sep = ",";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ia->ia_nirq) {
|
|
|
|
sep = "";
|
|
|
|
printf(" irq ");
|
|
|
|
for (i = 0; i < ia->ia_nirq; i++) {
|
|
|
|
if (ia->ia_irq[i].ir_irq == ISACF_IRQ_DEFAULT)
|
|
|
|
continue;
|
|
|
|
printf("%s%d", sep, ia->ia_irq[i].ir_irq);
|
|
|
|
sep = ",";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ia->ia_ndrq) {
|
|
|
|
sep = "";
|
|
|
|
printf(" drq ");
|
|
|
|
for (i = 0; i < ia->ia_ndrq; i++) {
|
|
|
|
if (ia->ia_drq[i].ir_drq == ISACF_DRQ_DEFAULT)
|
|
|
|
continue;
|
|
|
|
printf("%s%d", sep, ia->ia_drq[i].ir_drq);
|
|
|
|
sep = ",";
|
|
|
|
}
|
|
|
|
}
|
1994-03-29 08:34:18 +04:00
|
|
|
|
1995-01-16 13:36:26 +03:00
|
|
|
return (UNCONF);
|
1994-03-29 08:34:18 +04:00
|
|
|
}
|
|
|
|
|
1997-01-26 06:49:28 +03:00
|
|
|
int
|
2002-01-08 00:46:56 +03:00
|
|
|
isasearch(struct device *parent, struct cfdata *cf, void *aux)
|
1997-01-26 06:49:28 +03:00
|
|
|
{
|
2002-01-08 00:46:56 +03:00
|
|
|
struct isa_io res_io[1];
|
|
|
|
struct isa_iomem res_mem[1];
|
|
|
|
struct isa_irq res_irq[1];
|
|
|
|
struct isa_drq res_drq[2];
|
1997-01-26 06:49:28 +03:00
|
|
|
struct isa_softc *sc = (struct isa_softc *)parent;
|
|
|
|
struct isa_attach_args ia;
|
|
|
|
int tryagain;
|
|
|
|
|
|
|
|
do {
|
2002-01-08 00:46:56 +03:00
|
|
|
ia.ia_pnpname = NULL;
|
|
|
|
ia.ia_pnpcompatnames = NULL;
|
|
|
|
|
|
|
|
res_io[0].ir_addr = cf->cf_loc[ISACF_PORT];
|
|
|
|
res_io[0].ir_size = 0;
|
|
|
|
|
|
|
|
res_mem[0].ir_addr = cf->cf_loc[ISACF_IOMEM];
|
|
|
|
res_mem[0].ir_size = cf->cf_loc[ISACF_IOSIZ];
|
|
|
|
|
|
|
|
res_irq[0].ir_irq =
|
|
|
|
cf->cf_loc[ISACF_IRQ] == 2 ? 9 : cf->cf_loc[ISACF_IRQ];
|
|
|
|
|
|
|
|
res_drq[0].ir_drq = cf->cf_loc[ISACF_DRQ];
|
|
|
|
res_drq[1].ir_drq = cf->cf_loc[ISACF_DRQ2];
|
|
|
|
|
1997-01-26 06:49:28 +03:00
|
|
|
ia.ia_iot = sc->sc_iot;
|
|
|
|
ia.ia_memt = sc->sc_memt;
|
1997-06-07 03:43:45 +04:00
|
|
|
ia.ia_dmat = sc->sc_dmat;
|
1997-01-26 06:49:28 +03:00
|
|
|
ia.ia_ic = sc->sc_ic;
|
2002-01-08 00:46:56 +03:00
|
|
|
|
|
|
|
ia.ia_io = res_io;
|
|
|
|
ia.ia_nio = 1;
|
|
|
|
|
|
|
|
ia.ia_iomem = res_mem;
|
|
|
|
ia.ia_niomem = 1;
|
|
|
|
|
|
|
|
ia.ia_irq = res_irq;
|
|
|
|
ia.ia_nirq = 1;
|
|
|
|
|
|
|
|
ia.ia_drq = res_drq;
|
|
|
|
ia.ia_ndrq = 2;
|
1997-01-26 06:49:28 +03:00
|
|
|
|
|
|
|
tryagain = 0;
|
|
|
|
if ((*cf->cf_attach->ca_match)(parent, cf, &ia) > 0) {
|
|
|
|
config_attach(parent, cf, &ia, isaprint);
|
|
|
|
tryagain = (cf->cf_fstate == FSTATE_STAR);
|
|
|
|
}
|
|
|
|
} while (tryagain);
|
|
|
|
|
1996-12-05 04:25:23 +03:00
|
|
|
return (0);
|
1994-11-04 09:40:11 +03:00
|
|
|
}
|
|
|
|
|
1995-04-17 16:06:30 +04:00
|
|
|
char *
|
2002-01-08 00:46:56 +03:00
|
|
|
isa_intr_typename(int type)
|
1995-04-17 16:06:30 +04:00
|
|
|
{
|
|
|
|
|
|
|
|
switch (type) {
|
1995-12-24 05:29:35 +03:00
|
|
|
case IST_NONE :
|
1995-04-17 16:06:30 +04:00
|
|
|
return ("none");
|
1995-12-24 05:29:35 +03:00
|
|
|
case IST_PULSE:
|
1995-04-17 16:06:30 +04:00
|
|
|
return ("pulsed");
|
1995-12-24 05:29:35 +03:00
|
|
|
case IST_EDGE:
|
1995-04-17 16:06:30 +04:00
|
|
|
return ("edge-triggered");
|
1995-12-24 05:29:35 +03:00
|
|
|
case IST_LEVEL:
|
1995-04-17 16:06:30 +04:00
|
|
|
return ("level-triggered");
|
|
|
|
default:
|
|
|
|
panic("isa_intr_typename: invalid type %d", type);
|
|
|
|
}
|
|
|
|
}
|