summaryrefslogtreecommitdiff
path: root/sys/dev/pci
diff options
context:
space:
mode:
authorJonathan Gray <jsg@cvs.openbsd.org>2009-06-02 17:27:40 +0000
committerJonathan Gray <jsg@cvs.openbsd.org>2009-06-02 17:27:40 +0000
commit480e6d488fd326ab2dabdb02127a6b300d07795c (patch)
tree98ff9fa954b4c9ca2eb0241215a63ac47deeba58 /sys/dev/pci
parent8a695611f6f56d2cb8adf2bff424e36a79473c36 (diff)
make rl at pci detachable; untested.
Diffstat (limited to 'sys/dev/pci')
-rw-r--r--sys/dev/pci/if_rl_pci.c44
1 files changed, 36 insertions, 8 deletions
diff --git a/sys/dev/pci/if_rl_pci.c b/sys/dev/pci/if_rl_pci.c
index c78cd121767..69aa2d6a2c9 100644
--- a/sys/dev/pci/if_rl_pci.c
+++ b/sys/dev/pci/if_rl_pci.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: if_rl_pci.c,v 1.16 2009/06/02 04:03:39 jsg Exp $ */
+/* $OpenBSD: if_rl_pci.c,v 1.17 2009/06/02 17:27:39 jsg Exp $ */
/*
* Copyright (c) 1997, 1998
@@ -84,9 +84,16 @@
int rl_pci_match(struct device *, void *, void *);
void rl_pci_attach(struct device *, struct device *, void *);
+int rl_pci_detach(struct device *, int);
+
+struct rl_pci_softc {
+ struct rl_softc psc_softc;
+ pci_chipset_tag_t psc_pc;
+ bus_size_t psc_mapsize;
+};
struct cfattach rl_pci_ca = {
- sizeof(struct rl_softc), rl_pci_match, rl_pci_attach,
+ sizeof(struct rl_pci_softc), rl_pci_match, rl_pci_attach, rl_pci_detach
};
const struct pci_matchid rl_pci_devices[] = {
@@ -128,12 +135,12 @@ rl_pci_attach(parent, self, aux)
struct device *parent, *self;
void *aux;
{
- struct rl_softc *sc = (struct rl_softc *)self;
+ struct rl_pci_softc *psc = (void *)self;
+ struct rl_softc *sc = &psc->psc_softc;
struct pci_attach_args *pa = aux;
pci_chipset_tag_t pc = pa->pa_pc;
pci_intr_handle_t ih;
const char *intrstr = NULL;
- bus_size_t size;
/*
* Map control/status registers.
@@ -141,13 +148,13 @@ rl_pci_attach(parent, self, aux)
#ifdef RL_USEIOSPACE
if (pci_mapreg_map(pa, RL_PCI_LOIO, PCI_MAPREG_TYPE_IO, 0,
- &sc->rl_btag, &sc->rl_bhandle, NULL, &size, 0)) {
+ &sc->rl_btag, &sc->rl_bhandle, NULL, &psc->psc_mapsize, 0)) {
printf(": can't map i/o space\n");
return;
}
#else
if (pci_mapreg_map(pa, RL_PCI_LOMEM, PCI_MAPREG_TYPE_MEM, 0,
- &sc->rl_btag, &sc->rl_bhandle, NULL, &size, 0)){
+ &sc->rl_btag, &sc->rl_bhandle, NULL, &psc->psc_mapsize, 0)){
printf(": can't map mem space\n");
return;
}
@@ -158,10 +165,11 @@ rl_pci_attach(parent, self, aux)
*/
if (pci_intr_map(pa, &ih)) {
printf(": couldn't map interrupt\n");
- bus_space_unmap(sc->rl_btag, sc->rl_bhandle, size);
+ bus_space_unmap(sc->rl_btag, sc->rl_bhandle, psc->psc_mapsize);
return;
}
+ psc->psc_pc = pa->pa_pc;
intrstr = pci_intr_string(pc, ih);
sc->sc_ih = pci_intr_establish(pc, ih, IPL_NET, rl_intr, sc,
self->dv_xname);
@@ -170,7 +178,7 @@ rl_pci_attach(parent, self, aux)
if (intrstr != NULL)
printf(" at %s", intrstr);
printf("\n");
- bus_space_unmap(sc->rl_btag, sc->rl_bhandle, size);
+ bus_space_unmap(sc->rl_btag, sc->rl_bhandle, psc->psc_mapsize);
return;
}
printf(": %s", intrstr);
@@ -179,3 +187,23 @@ rl_pci_attach(parent, self, aux)
rl_attach(sc);
}
+
+int
+rl_pci_detach(struct device *self, int flags)
+{
+ struct rl_pci_softc *psc = (void *)self;
+ struct rl_softc *sc = &psc->psc_softc;
+ int rv;
+
+ rv = rl_detach(sc);
+ if (rv)
+ return (rv);
+
+ if (sc->sc_ih != NULL)
+ pci_intr_disestablish(psc->psc_pc, sc->sc_ih);
+
+ bus_space_unmap(sc->rl_btag, sc->rl_bhandle, psc->psc_mapsize);
+
+ return (0);
+}
+