diff options
Diffstat (limited to 'sys')
-rw-r--r-- | sys/dev/pci/ppb.c | 110 |
1 files changed, 103 insertions, 7 deletions
diff --git a/sys/dev/pci/ppb.c b/sys/dev/pci/ppb.c index 2a9361785dc..95c60b963a8 100644 --- a/sys/dev/pci/ppb.c +++ b/sys/dev/pci/ppb.c @@ -1,4 +1,4 @@ -/* $OpenBSD: ppb.c,v 1.19 2007/05/21 22:10:45 kettenis Exp $ */ +/* $OpenBSD: ppb.c,v 1.20 2007/11/24 21:33:58 kettenis Exp $ */ /* $NetBSD: ppb.c,v 1.16 1997/06/06 23:48:05 thorpej Exp $ */ /* @@ -35,6 +35,8 @@ #include <sys/systm.h> #include <sys/kernel.h> #include <sys/device.h> +#include <sys/proc.h> +#include <sys/workq.h> #include <dev/pci/pcireg.h> #include <dev/pci/pcivar.h> @@ -46,6 +48,9 @@ struct ppb_softc { pci_chipset_tag_t sc_pc; /* our PCI chipset... */ pcitag_t sc_tag; /* ...and tag. */ pci_intr_handle_t sc_ih[4]; + struct device *sc_psc; + int sc_cap_off; + struct timeout sc_to; }; int ppbmatch(struct device *, void *, void *); @@ -59,6 +64,11 @@ struct cfdriver ppb_cd = { NULL, "ppb", DV_DULL }; +int ppb_intr(void *); +void ppb_hotplug_insert(void *, void *); +void ppb_hotplug_insert_finish(void *); +void ppb_hotplug_rescan(void *, void *); +void ppb_hotplug_remove(void *, void *); int ppbprint(void *, const char *pnp); int @@ -91,19 +101,17 @@ ppbattach(struct device *parent, struct device *self, void *aux) struct pci_attach_args *pa = aux; pci_chipset_tag_t pc = pa->pa_pc; struct pcibus_attach_args pba; - pcireg_t busdata; + pci_intr_handle_t ih; + pcireg_t busdata, reg; int pin; - printf("\n"); - sc->sc_pc = pc; sc->sc_tag = pa->pa_tag; busdata = pci_conf_read(pc, pa->pa_tag, PPB_REG_BUSINFO); if (PPB_BUSINFO_SECONDARY(busdata) == 0) { - printf("%s: not configured by system firmware\n", - self->dv_xname); + printf(": not configured by system firmware\n"); return; } @@ -125,6 +133,27 @@ ppbattach(struct device *parent, struct device *self, void *aux) pa->pa_bus, PPB_BUSINFO_PRIMARY(busdata)); #endif + /* Check for PCI Express capabilities and setup hotplug support. */ + if (pci_get_capability(pc, pa->pa_tag, PCI_CAP_PCIEXPRESS, + &sc->sc_cap_off, ®) && (reg & PCI_PCIE_XCAP_SI)) { + if (pci_intr_map(pa, &ih) == 0 && + pci_intr_establish(pc, ih, IPL_TTY, ppb_intr, sc, + self->dv_xname)) { + printf(": %s", pci_intr_string(pc, ih)); + + /* Enable hotplug interrupt. */ + reg = pci_conf_read(pc, pa->pa_tag, + sc->sc_cap_off + PCI_PCIE_SLCSR); + reg |= (PCI_PCIE_SLCSR_HPE | PCI_PCIE_SLCSR_PDE); + pci_conf_write(pc, pa->pa_tag, + sc->sc_cap_off + PCI_PCIE_SLCSR, reg); + + timeout_set(&sc->sc_to, ppb_hotplug_insert_finish, sc); + } + } + + printf("\n"); + /* * Attach the PCI bus that hangs off of it. * @@ -146,7 +175,74 @@ ppbattach(struct device *parent, struct device *self, void *aux) pba.pba_intrswiz = pa->pa_intrswiz; pba.pba_intrtag = pa->pa_intrtag; - config_found(self, &pba, ppbprint); + sc->sc_psc = config_found(self, &pba, ppbprint); +} + +int +ppb_intr(void *arg) +{ + struct ppb_softc *sc = arg; + pcireg_t reg; + + reg = pci_conf_read(sc->sc_pc, sc->sc_tag, + sc->sc_cap_off + PCI_PCIE_SLCSR); + if (reg & PCI_PCIE_SLCSR_PDC) { + if (reg & PCI_PCIE_SLCSR_PDS) + workq_add_task(NULL, 0, ppb_hotplug_insert, sc, NULL); + else + workq_add_task(NULL, 0, ppb_hotplug_remove, sc, NULL); + + /* Clear interrupts. */ + pci_conf_write(sc->sc_pc, sc->sc_tag, + sc->sc_cap_off + PCI_PCIE_SLCSR, reg); + return (1); + } + + return (0); +} + +#ifdef PCI_MACHDEP_ENUMERATE_BUS +#define pci_enumerate_bus PCI_MACHDEP_ENUMERATE_BUS +#else +extern int pci_enumerate_bus(struct pci_softc *, + int (*)(struct pci_attach_args *), struct pci_attach_args *); +#endif + +void +ppb_hotplug_insert(void *arg1, void *arg2) +{ + struct ppb_softc *sc = arg1; + + /* XXX Powerup the card. */ + + /* XXX Turn on LEDs. */ + + /* Wait a second for things to settle. */ + timeout_add(&sc->sc_to, 1 * hz); +} + +void +ppb_hotplug_insert_finish(void *arg) +{ + workq_add_task(NULL, 0, ppb_hotplug_rescan, arg, NULL); +} + +void +ppb_hotplug_rescan(void *arg1, void *arg2) +{ + struct ppb_softc *sc = arg1; + + if (sc->sc_psc) + pci_enumerate_bus((struct pci_softc *)sc->sc_psc, NULL, NULL); +} + +void +ppb_hotplug_remove(void *arg1, void *arg2) +{ + struct ppb_softc *sc = arg1; + + if (sc->sc_psc) + config_detach_children(sc->sc_psc, DETACH_FORCE); } int |