diff options
-rw-r--r-- | sys/dev/pci/pci.c | 46 | ||||
-rw-r--r-- | sys/dev/pci/pcivar.h | 10 |
2 files changed, 51 insertions, 5 deletions
diff --git a/sys/dev/pci/pci.c b/sys/dev/pci/pci.c index e75d16ea2fb..29c9e453a5f 100644 --- a/sys/dev/pci/pci.c +++ b/sys/dev/pci/pci.c @@ -1,5 +1,5 @@ -/* $OpenBSD: pci.c,v 1.5 1996/11/28 23:28:09 niklas Exp $ */ -/* $NetBSD: pci.c,v 1.24 1996/10/21 22:56:55 thorpej Exp $ */ +/* $OpenBSD: pci.c,v 1.6 1997/01/24 19:34:15 niklas Exp $ */ +/* $NetBSD: pci.c,v 1.26 1996/12/05 01:25:30 cgd Exp $ */ /* * Copyright (c) 1995, 1996 Christopher G. Demetriou. All rights reserved. @@ -56,6 +56,30 @@ struct cfdriver pci_cd = { int pciprint __P((void *, const char *)); int pcisubmatch __P((struct device *, void *, void *)); +/* + * Callback so that ISA/EISA bridges can attach their child busses + * after PCI configuration is done. + * + * This works because: + * (1) there can be at most one ISA/EISA bridge per PCI bus, and + * (2) any ISA/EISA bridges must be attached to primary PCI + * busses (i.e. bus zero). + * + * That boils down to: there can only be one of these outstanding + * at a time, it is cleared when configuring PCI bus 0 before any + * subdevices have been found, and it is run after all subdevices + * of PCI bus 0 have been found. + * + * This is needed because there are some (legacy) PCI devices which + * can show up as ISA/EISA devices as well (the prime example of which + * are VGA controllers). If you attach ISA from a PCI-ISA/EISA bridge, + * and the bridge is seen before the video board is, the board can show + * up as an ISA device, and that can (bogusly) complicate the PCI device's + * attach code, or make the PCI device not be properly attached at all. + */ +static void (*pci_isa_bridge_callback) __P((void *)); +static void *pci_isa_bridge_callback_arg; + int pcimatch(parent, match, aux) struct device *parent; @@ -102,6 +126,9 @@ pciattach(parent, self, aux) bus = pba->pba_bus; maxndevs = pci_bus_maxdevs(pc, bus); + if (bus == 0) + pci_isa_bridge_callback = NULL; + for (device = 0; device < maxndevs; device++) { pcitag_t tag; pcireg_t id, class, intr, bhlcr; @@ -158,6 +185,9 @@ pciattach(parent, self, aux) config_found_sm(self, &pa, pciprint, pcisubmatch); } } + + if (bus == 0 && pci_isa_bridge_callback != NULL) + (*pci_isa_bridge_callback)(pci_isa_bridge_callback_arg); } int @@ -293,3 +323,15 @@ pci_mem_find(pc, pcitag, reg, membasep, memsizep, cacheablep) return 0; } + +void +set_pci_isa_bridge_callback(fn, arg) + void (*fn) __P((void *)); + void *arg; +{ + + if (pci_isa_bridge_callback != NULL) + panic("set_pci_isa_bridge_callback"); + pci_isa_bridge_callback = fn; + pci_isa_bridge_callback_arg = arg; +} diff --git a/sys/dev/pci/pcivar.h b/sys/dev/pci/pcivar.h index ffdf225dc56..225907936fa 100644 --- a/sys/dev/pci/pcivar.h +++ b/sys/dev/pci/pcivar.h @@ -1,5 +1,5 @@ -/* $OpenBSD: pcivar.h,v 1.9 1996/11/28 23:28:14 niklas Exp $ */ -/* $NetBSD: pcivar.h,v 1.16 1996/10/21 22:56:57 thorpej Exp $ */ +/* $OpenBSD: pcivar.h,v 1.10 1997/01/24 19:34:16 niklas Exp $ */ +/* $NetBSD: pcivar.h,v 1.18 1996/12/01 21:02:18 leo Exp $ */ /* * Copyright (c) 1996 Christopher G. Demetriou. All rights reserved. @@ -54,12 +54,15 @@ struct pcibus_attach_args; /* * Machine-dependent definitions. */ -#if (alpha + i386 != 1) +#if (alpha + atari + i386 != 1) ERROR: COMPILING FOR UNSUPPORTED MACHINE, OR MORE THAN ONE. #endif #if alpha #include <alpha/pci/pci_machdep.h> #endif +#if atari +#include <atari/pci/pci_machdep.h> +#endif #if i386 #include <i386/pci/pci_machdep.h> #endif @@ -138,5 +141,6 @@ int pci_mem_find __P((pci_chipset_tag_t, pcitag_t, int, bus_addr_t *, * Helper functions for autoconfiguration. */ void pci_devinfo __P((pcireg_t, pcireg_t, int, char *)); +void set_pci_isa_bridge_callback __P((void (*)(void *), void *)); #endif /* _DEV_PCI_PCIVAR_H_ */ |