summaryrefslogtreecommitdiff
path: root/sys/dev
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev')
-rw-r--r--sys/dev/pci/if_oce.c37
1 files changed, 25 insertions, 12 deletions
diff --git a/sys/dev/pci/if_oce.c b/sys/dev/pci/if_oce.c
index 1dffad18b8b..87caaf3497e 100644
--- a/sys/dev/pci/if_oce.c
+++ b/sys/dev/pci/if_oce.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: if_oce.c,v 1.8 2012/08/09 12:43:46 mikeb Exp $ */
+/* $OpenBSD: if_oce.c,v 1.9 2012/08/09 18:41:45 mikeb Exp $ */
/*
* Copyright (c) 2012 Mike Belopuhov
@@ -117,6 +117,7 @@ int oce_get_buf(struct oce_rq *rq);
int oce_alloc_rx_bufs(struct oce_rq *rq);
void oce_refill_rx(void *arg);
+void oce_watchdog(struct ifnet *ifp);
void oce_start(struct ifnet *ifp);
int oce_encap(struct oce_softc *sc, struct mbuf **mpp, int wq_index);
void oce_txeof(struct oce_wq *wq, uint32_t wqe_idx, uint32_t status);
@@ -766,10 +767,12 @@ oce_txeof(struct oce_wq *wq, uint32_t wqe_idx, uint32_t status)
if (ifp->if_flags & IFF_OACTIVE) {
if (wq->ring->num_used < (wq->ring->num_items / 2)) {
- ifp->if_flags &= ~(IFF_OACTIVE);
+ ifp->if_flags &= ~IFF_OACTIVE;
oce_start(ifp);
}
}
+ if (wq->ring->num_used == 0)
+ ifp->if_timer = 0;
}
#if OCE_TSO
@@ -844,27 +847,33 @@ oce_tso_setup(struct oce_softc *sc, struct mbuf **mpp)
#endif
void
+oce_watchdog(struct ifnet *ifp)
+{
+ printf("%s: watchdog timeout -- resetting\n", ifp->if_xname);
+
+ oce_init(ifp->if_softc);
+
+ ifp->if_oerrors++;
+}
+
+void
oce_start(struct ifnet *ifp)
{
struct oce_softc *sc = ifp->if_softc;
struct mbuf *m;
- int rc = 0;
- int def_q = 0; /* Defualt tx queue is 0 */
+ int pkts = 0;
if ((ifp->if_flags & (IFF_RUNNING | IFF_OACTIVE)) != IFF_RUNNING)
return;
- do {
+ for (;;) {
IFQ_POLL(&ifp->if_snd, m);
if (m == NULL)
break;
- rc = oce_encap(sc, &m, def_q);
- if (rc) {
- if (m != NULL) {
- sc->wq[def_q]->tx_stats.tx_stops++;
+ if (oce_encap(sc, &m, 0)) {
+ if (m)
ifp->if_flags |= IFF_OACTIVE;
- }
break;
}
@@ -874,9 +883,12 @@ oce_start(struct ifnet *ifp)
if (ifp->if_bpf)
bpf_mtap_ether(ifp->if_bpf, m, BPF_DIRECTION_OUT);
#endif
- } while (TRUE);
+ pkts++;
+ }
- return;
+ /* Set a timeout in case the chip goes out to lunch */
+ if (pkts)
+ ifp->if_timer = 5;
}
/* Handle the Completion Queue for transmit */
@@ -1354,6 +1366,7 @@ oce_attach_ifp(struct oce_softc *sc)
ifp->if_flags = IFF_BROADCAST | IFF_SIMPLEX | IFF_MULTICAST;
ifp->if_ioctl = oce_ioctl;
ifp->if_start = oce_start;
+ ifp->if_watchdog = oce_watchdog;
ifp->if_hardmtu = OCE_MAX_MTU;
ifp->if_softc = sc;
IFQ_SET_MAXLEN(&ifp->if_snd, sc->tx_ring_size - 1);