diff options
author | Mike Belopuhov <mikeb@cvs.openbsd.org> | 2012-12-10 10:38:57 +0000 |
---|---|---|
committer | Mike Belopuhov <mikeb@cvs.openbsd.org> | 2012-12-10 10:38:57 +0000 |
commit | b22a9a65ea0b3ddf62bc34748a7516266438fa95 (patch) | |
tree | 3161796479a44b1787a14e410e2edeb76c582344 /sys/dev | |
parent | 3c5c51272495548adeece566befea44fb6300748 (diff) |
Under some circumstances (currently only reproducible with IPsec)
bnx can be left w/o clusters on the receive ring and will stall.
To prevent that schedule a timeout if refill fails. Bug was
reported by jj@, fix tested by me, ok dlg
Diffstat (limited to 'sys/dev')
-rw-r--r-- | sys/dev/pci/if_bnx.c | 28 | ||||
-rw-r--r-- | sys/dev/pci/if_bnxreg.h | 3 |
2 files changed, 26 insertions, 5 deletions
diff --git a/sys/dev/pci/if_bnx.c b/sys/dev/pci/if_bnx.c index 0807d013996..9c7b5094a36 100644 --- a/sys/dev/pci/if_bnx.c +++ b/sys/dev/pci/if_bnx.c @@ -1,4 +1,4 @@ -/* $OpenBSD: if_bnx.c,v 1.98 2012/12/05 23:20:20 deraadt Exp $ */ +/* $OpenBSD: if_bnx.c,v 1.99 2012/12/10 10:38:56 mikeb Exp $ */ /*- * Copyright (c) 2006 Broadcom Corporation @@ -358,11 +358,12 @@ int bnx_get_buf(struct bnx_softc *, u_int16_t *, u_int16_t *, u_int32_t *); int bnx_init_tx_chain(struct bnx_softc *); void bnx_init_tx_context(struct bnx_softc *); -void bnx_fill_rx_chain(struct bnx_softc *); +int bnx_fill_rx_chain(struct bnx_softc *); void bnx_init_rx_context(struct bnx_softc *); int bnx_init_rx_chain(struct bnx_softc *); void bnx_free_rx_chain(struct bnx_softc *); void bnx_free_tx_chain(struct bnx_softc *); +void bnx_rxrefill(void *); int bnx_tx_encap(struct bnx_softc *, struct mbuf *); void bnx_start(struct ifnet *); @@ -933,6 +934,7 @@ bnx_attachhook(void *xsc) ether_ifattach(ifp); timeout_set(&sc->bnx_timeout, bnx_tick, sc); + timeout_set(&sc->bnx_rxrefill, bnx_rxrefill, sc); /* Print some important debugging info. */ DBRUN(BNX_INFO, bnx_dump_driver_state(sc)); @@ -3271,6 +3273,7 @@ bnx_stop(struct bnx_softc *sc) DBPRINT(sc, BNX_VERBOSE_RESET, "Entering %s()\n", __FUNCTION__); timeout_del(&sc->bnx_timeout); + timeout_del(&sc->bnx_rxrefill); ifp->if_flags &= ~(IFF_RUNNING | IFF_OACTIVE); @@ -4026,11 +4029,12 @@ bnx_init_rx_context(struct bnx_softc *sc) /* Returns: */ /* Nothing */ /****************************************************************************/ -void +int bnx_fill_rx_chain(struct bnx_softc *sc) { u_int16_t prod, chain_prod; u_int32_t prod_bseq; + int ndesc = 0; #ifdef BNX_DEBUG int rx_mbuf_alloc_before, free_rx_bd_before; #endif @@ -4053,6 +4057,7 @@ bnx_fill_rx_chain(struct bnx_softc *sc) break; } prod = NEXT_RX_BD(prod); + ndesc++; } #if 0 @@ -4071,6 +4076,8 @@ bnx_fill_rx_chain(struct bnx_softc *sc) REG_WR(sc, MB_RX_CID_ADDR + BNX_L2CTX_HOST_BSEQ, sc->rx_prod_bseq); DBPRINT(sc, BNX_EXCESSIVE_RECV, "Exiting %s()\n", __FUNCTION__); + + return (ndesc); } /****************************************************************************/ @@ -4188,6 +4195,18 @@ bnx_free_rx_chain(struct bnx_softc *sc) DBPRINT(sc, BNX_VERBOSE_RESET, "Exiting %s()\n", __FUNCTION__); } +void +bnx_rxrefill(void *xsc) +{ + struct bnx_softc *sc = xsc; + int s; + + s = splnet(); + if (!bnx_fill_rx_chain(sc)) + timeout_add(&sc->bnx_rxrefill, 1); + splx(s); +} + /****************************************************************************/ /* Set media options. */ /* */ @@ -4553,7 +4572,8 @@ bnx_rx_int_next_rx: /* No new packets to process. Refill the RX chain and exit. */ sc->rx_cons = sw_cons; - bnx_fill_rx_chain(sc); + if (!bnx_fill_rx_chain(sc)) + timeout_add(&sc->bnx_rxrefill, 1); for (i = 0; i < RX_PAGES; i++) bus_dmamap_sync(sc->bnx_dmatag, diff --git a/sys/dev/pci/if_bnxreg.h b/sys/dev/pci/if_bnxreg.h index ce73c95693a..1dd2bcc6659 100644 --- a/sys/dev/pci/if_bnxreg.h +++ b/sys/dev/pci/if_bnxreg.h @@ -1,4 +1,4 @@ -/* $OpenBSD: if_bnxreg.h,v 1.38 2012/07/05 13:50:15 phessler Exp $ */ +/* $OpenBSD: if_bnxreg.h,v 1.39 2012/12/10 10:38:56 mikeb Exp $ */ /*- * Copyright (c) 2006 Broadcom Corporation @@ -4869,6 +4869,7 @@ struct bnx_softc { int bnx_link; struct timeout bnx_timeout; + struct timeout bnx_rxrefill; /* Frame size and mbuf allocation size for RX frames. */ u_int32_t max_frame_size; |