summaryrefslogtreecommitdiffstats
path: root/sys
diff options
context:
space:
mode:
authorgallatin <gallatin@FreeBSD.org>2009-06-24 21:09:56 +0000
committergallatin <gallatin@FreeBSD.org>2009-06-24 21:09:56 +0000
commitce189363a0ec4e65f27076207a88142a8554bfbb (patch)
tree5ded461e649a2bf5399abf07694d4ab18dae34d9 /sys
parent4a7daf6d41bedfcb21beeb7da44f965d90ee301e (diff)
downloadFreeBSD-src-ce189363a0ec4e65f27076207a88142a8554bfbb.zip
FreeBSD-src-ce189363a0ec4e65f27076207a88142a8554bfbb.tar.gz
Add a dying flag to prevent races at detach.
I tried re-ordering ether_ifdetach(), but this created a new race where sometimes, when under heavy receive load (>1Mpps) and running tcpdump, the machine would panic. At panic, the ithread was still in the original (not dead) if_input() path, and was accessing stale BPF data structs. By using a dying flag, I can close the interface prior to if_detach() to be certain the interface cannot send packets up in the middle of ether_ifdetach.
Diffstat (limited to 'sys')
-rw-r--r--sys/dev/mxge/if_mxge.c6
-rw-r--r--sys/dev/mxge/if_mxge_var.h1
2 files changed, 7 insertions, 0 deletions
diff --git a/sys/dev/mxge/if_mxge.c b/sys/dev/mxge/if_mxge.c
index c5d497b..16453e9 100644
--- a/sys/dev/mxge/if_mxge.c
+++ b/sys/dev/mxge/if_mxge.c
@@ -3906,6 +3906,10 @@ mxge_ioctl(struct ifnet *ifp, u_long command, caddr_t data)
case SIOCSIFFLAGS:
mtx_lock(&sc->driver_mtx);
+ if (sc->dying) {
+ mtx_unlock(&sc->driver_mtx);
+ return EINVAL;
+ }
if (ifp->if_flags & IFF_UP) {
if (!(ifp->if_drv_flags & IFF_DRV_RUNNING)) {
err = mxge_open(sc);
@@ -4590,6 +4594,7 @@ mxge_attach(device_t dev)
mxge_media_status);
mxge_set_media(sc, IFM_ETHER | IFM_AUTO);
mxge_media_probe(sc);
+ sc->dying = 0;
ether_ifattach(ifp, sc->mac_addr);
/* ether_ifattach sets mtu to ETHERMTU */
if (mxge_initial_mtu != ETHERMTU)
@@ -4637,6 +4642,7 @@ mxge_detach(device_t dev)
return EBUSY;
}
mtx_lock(&sc->driver_mtx);
+ sc->dying = 1;
if (sc->ifp->if_drv_flags & IFF_DRV_RUNNING)
mxge_close(sc);
mtx_unlock(&sc->driver_mtx);
diff --git a/sys/dev/mxge/if_mxge_var.h b/sys/dev/mxge/if_mxge_var.h
index 8d1da2e..47c39b5 100644
--- a/sys/dev/mxge/if_mxge_var.h
+++ b/sys/dev/mxge/if_mxge_var.h
@@ -266,6 +266,7 @@ struct mxge_softc {
int need_media_probe;
int num_slices;
int rx_ring_size;
+ int dying;
mxge_dma_t dmabench_dma;
struct callout co_hdl;
struct sysctl_oid *slice_sysctl_tree;
OpenPOWER on IntegriCloud