summaryrefslogtreecommitdiffstats
path: root/sys/dev/sk
diff options
context:
space:
mode:
authornjl <njl@FreeBSD.org>2003-04-17 20:32:06 +0000
committernjl <njl@FreeBSD.org>2003-04-17 20:32:06 +0000
commitfe07eb9dfb67dec3b9867ee376a889a70b355179 (patch)
tree3cf87a7e7fce73f14cb753a6554ec38575d14502 /sys/dev/sk
parent9aba79052e570949849c1985ff95ff264ff272ec (diff)
downloadFreeBSD-src-fe07eb9dfb67dec3b9867ee376a889a70b355179.zip
FreeBSD-src-fe07eb9dfb67dec3b9867ee376a889a70b355179.tar.gz
Revise attach/detach resource cleanup
- Unconditionally call *_stop() if device is in the tree. This is to prevent callouts from happening after the device is gone. Checks for bus_child_present() should be added in the future to keep from touching potentially non-existent hardware in *_detach(). Found by iedowse@. - Always check for and free miibus children, even if the device is not in the tree since some failure cases could have gotten here. - Call ether_ifdetach() in the irq setup failure case - ti(4), xl(4): move ifmedia_init() calls to the beginning of attach so that ifmedia_removeall() can be unconditionally called on detach. There is no way to detect whether ifmedia has been initialized without using a separate variable (as tl(4) does). - Add comments to indicate assumptions of code path
Diffstat (limited to 'sys/dev/sk')
-rw-r--r--sys/dev/sk/if_sk.c18
1 files changed, 14 insertions, 4 deletions
diff --git a/sys/dev/sk/if_sk.c b/sys/dev/sk/if_sk.c
index 1093f34..49424bb 100644
--- a/sys/dev/sk/if_sk.c
+++ b/sys/dev/sk/if_sk.c
@@ -1178,6 +1178,7 @@ sk_attach_xmac(dev)
if (mii_phy_probe(dev, &sc_if->sk_miibus,
sk_ifmedia_upd, sk_ifmedia_sts)) {
printf("skc%d: no PHY found!\n", sc_if->sk_unit);
+ ether_ifdetach(ifp);
error = ENXIO;
goto fail_xmac;
}
@@ -1335,6 +1336,7 @@ sk_attach(dev)
bus_generic_attach(dev);
+ /* Hook interrupt last to avoid having to lock softc */
error = bus_setup_intr(dev, sc->sk_irq, INTR_TYPE_NET,
sk_intr, sc, &sc->sk_intrhand);
@@ -1350,6 +1352,13 @@ fail:
return(error);
}
+/*
+ * Shutdown hardware and free up resources. This can be called any
+ * time after the mutex has been initialized. It is called in both
+ * the error case in attach and the normal detach case so it needs
+ * to be careful about only freeing resources that have actually been
+ * allocated.
+ */
static int
sk_detach_xmac(dev)
device_t dev;
@@ -1365,13 +1374,14 @@ sk_detach_xmac(dev)
SK_IF_LOCK(sc_if);
ifp = &sc_if->arpcom.ac_if;
+ /* These should only be active if attach_xmac succeeded */
if (device_is_alive(dev)) {
- if (bus_child_present(dev))
- sk_stop(sc_if);
+ sk_stop(sc_if);
ether_ifdetach(ifp);
- device_delete_child(dev, sc_if->sk_miibus);
- bus_generic_detach(dev);
}
+ if (sc_if->sk_miibus)
+ device_delete_child(dev, sc_if->sk_miibus);
+ bus_generic_detach(dev);
if (sc_if->sk_cdata.sk_jumbo_buf)
contigfree(sc_if->sk_cdata.sk_jumbo_buf, SK_JMEM, M_DEVBUF);
if (sc_if->sk_rdata) {
OpenPOWER on IntegriCloud