diff options
Diffstat (limited to 'sys/dev/pci/if_ix.c')
-rw-r--r-- | sys/dev/pci/if_ix.c | 100 |
1 files changed, 73 insertions, 27 deletions
diff --git a/sys/dev/pci/if_ix.c b/sys/dev/pci/if_ix.c index adbc073a92b..1ee022c88ef 100644 --- a/sys/dev/pci/if_ix.c +++ b/sys/dev/pci/if_ix.c @@ -1,4 +1,4 @@ -/* $OpenBSD: if_ix.c,v 1.124 2015/09/02 16:08:49 deraadt Exp $ */ +/* $OpenBSD: if_ix.c,v 1.125 2015/09/11 12:09:10 claudio Exp $ */ /****************************************************************************** @@ -210,6 +210,8 @@ ixgbe_attach(struct device *parent, struct device *self, void *aux) sc->osdep.os_sc = sc; sc->osdep.os_pa = *pa; + mtx_init(&sc->rx_mtx, IPL_NET); + /* Set up the timer callout */ timeout_set(&sc->timer, ixgbe_local_timer, sc); timeout_set(&sc->rx_refill, ixgbe_rxrefill, sc); @@ -862,7 +864,7 @@ ixgbe_intr(void *arg) struct tx_ring *txr = sc->tx_rings; struct ixgbe_hw *hw = &sc->hw; uint32_t reg_eicr; - int i, refill = 0; + int i, refill = 0, was_active = 0; reg_eicr = IXGBE_READ_REG(&sc->hw, IXGBE_EICR); if (reg_eicr == 0) { @@ -870,23 +872,55 @@ ixgbe_intr(void *arg) return (0); } + if (ISSET(ifp->if_flags, IFF_RUNNING)) { + ixgbe_rxeof(que); + refill = 1; + + if (ISSET(ifp->if_flags, IFF_OACTIVE)) + was_active = 1; + ixgbe_txeof(txr); + } + + if (refill) { + if (ixgbe_rxfill(que->rxr)) { + /* Advance the Rx Queue "Tail Pointer" */ + IXGBE_WRITE_REG(&sc->hw, IXGBE_RDT(que->rxr->me), + que->rxr->last_desc_filled); + } else + timeout_add(&sc->rx_refill, 1); + } + /* Link status change */ - if (reg_eicr & IXGBE_EICR_LSC) + if (reg_eicr & IXGBE_EICR_LSC) { + KERNEL_LOCK(); ixgbe_update_link_status(sc); + KERNEL_UNLOCK(); + } + /* ... more link status change */ if (hw->mac.type != ixgbe_mac_82598EB) { if (reg_eicr & IXGBE_EICR_GPI_SDP2) { /* Clear the interrupt */ IXGBE_WRITE_REG(hw, IXGBE_EICR, IXGBE_EICR_GPI_SDP2); + KERNEL_LOCK(); ixgbe_handle_mod(sc); + KERNEL_UNLOCK(); } else if ((hw->phy.media_type != ixgbe_media_type_copper) && (reg_eicr & IXGBE_EICR_GPI_SDP1)) { /* Clear the interrupt */ IXGBE_WRITE_REG(hw, IXGBE_EICR, IXGBE_EICR_GPI_SDP1); + KERNEL_LOCK(); ixgbe_handle_msf(sc); + KERNEL_UNLOCK(); } } + if (was_active) { + KERNEL_LOCK(); + ixgbe_start(ifp); + KERNEL_UNLOCK(); + } + /* Check for fan failure */ if ((hw->device_id == IXGBE_DEV_ID_82598AT) && (reg_eicr & IXGBE_EICR_GPI_SDP1)) { @@ -903,24 +937,6 @@ ixgbe_intr(void *arg) IXGBE_WRITE_REG(hw, IXGBE_EICR, IXGBE_EICR_TS); } - if (ifp->if_flags & IFF_RUNNING) { - ixgbe_rxeof(que); - ixgbe_txeof(txr); - refill = 1; - } - - if (refill) { - if (ixgbe_rxfill(que->rxr)) { - /* Advance the Rx Queue "Tail Pointer" */ - IXGBE_WRITE_REG(&sc->hw, IXGBE_RDT(que->rxr->me), - que->rxr->last_desc_filled); - } else - timeout_add(&sc->rx_refill, 1); - } - - if (ifp->if_flags & IFF_RUNNING && !IFQ_IS_EMPTY(&ifp->if_snd)) - ixgbe_start(ifp); - for (i = 0; i < sc->num_queues; i++, que++) ixgbe_enable_queue(sc, que->msix); @@ -1449,7 +1465,7 @@ ixgbe_allocate_legacy(struct ix_softc *sc) #endif intrstr = pci_intr_string(pc, ih); - sc->tag = pci_intr_establish(pc, ih, IPL_NET, + sc->tag = pci_intr_establish(pc, ih, IPL_NET | IPL_MPSAFE, ixgbe_intr, sc, sc->dev.dv_xname); if (sc->tag == NULL) { printf(": couldn't establish interrupt"); @@ -2333,19 +2349,31 @@ ixgbe_txeof(struct tx_ring *txr) struct ixgbe_tx_buf *tx_buffer; struct ixgbe_legacy_tx_desc *tx_desc, *eop_desc; + if (!ISSET(ifp->if_flags, IFF_RUNNING)) + return FALSE; + if (txr->tx_avail == sc->num_tx_desc) { txr->queue_status = IXGBE_QUEUE_IDLE; return FALSE; } + KERNEL_LOCK(); + processed = 0; first = txr->next_to_clean; + /* was the txt queue cleaned up in the meantime */ + if (txr->tx_buffers == NULL) { + KERNEL_UNLOCK(); + return FALSE; + } tx_buffer = &txr->tx_buffers[first]; /* For cleanup we just use legacy struct */ tx_desc = (struct ixgbe_legacy_tx_desc *)&txr->tx_base[first]; last = tx_buffer->eop_index; - if (last == -1) + if (last == -1) { + KERNEL_UNLOCK(); return FALSE; + } eop_desc = (struct ixgbe_legacy_tx_desc *)&txr->tx_base[last]; /* @@ -2422,6 +2450,7 @@ ixgbe_txeof(struct tx_ring *txr) if (txr->tx_avail == sc->num_tx_desc) { ifp->if_timer = 0; txr->watchdog_timer = 0; + KERNEL_UNLOCK(); return FALSE; } /* Some were cleaned, so reset timer */ @@ -2431,6 +2460,8 @@ ixgbe_txeof(struct tx_ring *txr) } } + KERNEL_UNLOCK(); + return TRUE; } @@ -2580,6 +2611,8 @@ ixgbe_rxfill(struct rx_ring *rxr) u_int slots; int i; + mtx_enter(&sc->rx_mtx); + i = rxr->last_desc_filled; for (slots = if_rxr_get(&rxr->rx_ring, sc->num_rx_desc); slots > 0; slots--) { @@ -2595,6 +2628,8 @@ ixgbe_rxfill(struct rx_ring *rxr) if_rxr_put(&rxr->rx_ring, slots); + mtx_leave(&sc->rx_mtx); + return (post); } @@ -2759,10 +2794,15 @@ ixgbe_initialize_receive_units(struct ix_softc *sc) void ixgbe_free_receive_structures(struct ix_softc *sc) { - struct rx_ring *rxr = sc->rx_rings; + struct rx_ring *rxr; int i; - for (i = 0; i < sc->num_queues; i++, rxr++) + mtx_enter(&sc->rx_mtx); + for (i = 0, rxr = sc->rx_rings; i < sc->num_queues; i++, rxr++) + if_rxr_init(&rxr->rx_ring, 0, 0); + mtx_leave(&sc->rx_mtx); + + for (i = 0, rxr = sc->rx_rings; i < sc->num_queues; i++, rxr++) ixgbe_free_receive_buffers(rxr); } @@ -2814,6 +2854,7 @@ ixgbe_rxeof(struct ix_queue *que) struct rx_ring *rxr = que->rxr; struct ifnet *ifp = &sc->arpcom.ac_if; struct mbuf_list ml = MBUF_LIST_INITIALIZER(); + struct mbuf_list free_ml = MBUF_LIST_INITIALIZER(); struct mbuf *mp, *sendmp; uint8_t eop = 0; uint16_t len, vtag; @@ -2826,6 +2867,7 @@ ixgbe_rxeof(struct ix_queue *que) if (!ISSET(ifp->if_flags, IFF_RUNNING)) return FALSE; + mtx_enter(&sc->rx_mtx); i = rxr->next_to_check; while (if_rxr_inuse(&rxr->rx_ring) > 0) { bus_dmamap_sync(rxr->rxdma.dma_tag, rxr->rxdma.dma_map, @@ -2860,11 +2902,11 @@ ixgbe_rxeof(struct ix_queue *que) sc->dropped_pkts++; if (rxbuf->fmp) { - m_freem(rxbuf->fmp); + ml_enqueue(&free_ml, rxbuf->fmp); rxbuf->fmp = NULL; } - m_freem(mp); + ml_enqueue(&free_ml, mp); rxbuf->buf = NULL; goto next_desc; } @@ -2942,6 +2984,10 @@ next_desc: i = 0; } rxr->next_to_check = i; + mtx_leave(&sc->rx_mtx); + + while ((mp = ml_dequeue(&free_ml))) + m_freem(mp); if_input(ifp, &ml); |