diff options
Diffstat (limited to 'sys/dev/pci/if_ix.c')
-rw-r--r-- | sys/dev/pci/if_ix.c | 3443 |
1 files changed, 3443 insertions, 0 deletions
diff --git a/sys/dev/pci/if_ix.c b/sys/dev/pci/if_ix.c new file mode 100644 index 00000000000..21318c1e697 --- /dev/null +++ b/sys/dev/pci/if_ix.c @@ -0,0 +1,3443 @@ +/* $OpenBSD: if_ix.c,v 1.1 2008/06/08 20:01:02 reyk Exp $ */ + +/****************************************************************************** + + Copyright (c) 2001-2008, Intel Corporation + All rights reserved. + + Redistribution and use in source and binary forms, with or without + modification, are permitted provided that the following conditions are met: + + 1. Redistributions of source code must retain the above copyright notice, + this list of conditions and the following disclaimer. + + 2. Redistributions in binary form must reproduce the above copyright + notice, this list of conditions and the following disclaimer in the + documentation and/or other materials provided with the distribution. + + 3. Neither the name of the Intel Corporation nor the names of its + contributors may be used to endorse or promote products derived from + this software without specific prior written permission. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" + AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE + LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR + CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + POSSIBILITY OF SUCH DAMAGE. + +******************************************************************************/ +/*$FreeBSD: src/sys/dev/ixgbe/ixgbe.c,v 1.5 2008/05/16 18:46:30 jfv Exp $*/ + +#include <dev/pci/if_ix.h> + +/********************************************************************* + * Driver version + *********************************************************************/ + +#define IXGBE_DRIVER_VERSION "1.4.4" + +/********************************************************************* + * PCI Device ID Table + * + * Used by probe to select devices to load on + *********************************************************************/ + +const struct pci_matchid ixgbe_devices[] = { + { PCI_VENDOR_INTEL, PCI_PRODUCT_INTEL_82598AF_DUAL }, + { PCI_VENDOR_INTEL, PCI_PRODUCT_INTEL_82598AF }, + { PCI_VENDOR_INTEL, PCI_PRODUCT_INTEL_82598AT_DUAL }, + { PCI_VENDOR_INTEL, PCI_PRODUCT_INTEL_82598EB_CX4 }, + { PCI_VENDOR_INTEL, PCI_PRODUCT_INTEL_82598EB_CX4_DUAL }, + { PCI_VENDOR_INTEL, PCI_PRODUCT_INTEL_82598EB_XF_LR }, + { PCI_VENDOR_INTEL, PCI_PRODUCT_INTEL_82598AT } +}; + +/********************************************************************* + * Function prototypes + *********************************************************************/ +int ixgbe_probe(struct device *, void *, void *); +void ixgbe_attach(struct device *, struct device *, void *); +int ixgbe_detach(struct device *, int); +void ixgbe_power(int, void *); +void ixgbe_shutdown(void *); +void ixgbe_start(struct ifnet *); +void ixgbe_start_locked(struct tx_ring *, struct ifnet *); +int ixgbe_ioctl(struct ifnet *, u_long, caddr_t); +void ixgbe_watchdog(struct ifnet *); +void ixgbe_init(void *); +void ixgbe_stop(void *); +void ixgbe_media_status(struct ifnet *, struct ifmediareq *); +int ixgbe_media_change(struct ifnet *); +void ixgbe_identify_hardware(struct ix_softc *); +int ixgbe_allocate_pci_resources(struct ix_softc *); +int ixgbe_allocate_legacy(struct ix_softc *); +int ixgbe_allocate_queues(struct ix_softc *); +void ixgbe_free_pci_resources(struct ix_softc *); +void ixgbe_local_timer(void *); +int ixgbe_hardware_init(struct ix_softc *); +void ixgbe_setup_interface(struct ix_softc *); + +int ixgbe_allocate_transmit_buffers(struct tx_ring *); +int ixgbe_setup_transmit_structures(struct ix_softc *); +void ixgbe_setup_transmit_ring(struct tx_ring *); +void ixgbe_initialize_transmit_units(struct ix_softc *); +void ixgbe_free_transmit_structures(struct ix_softc *); +void ixgbe_free_transmit_buffers(struct tx_ring *); + +int ixgbe_allocate_receive_buffers(struct rx_ring *); +int ixgbe_setup_receive_structures(struct ix_softc *); +int ixgbe_setup_receive_ring(struct rx_ring *); +void ixgbe_initialize_receive_units(struct ix_softc *); +void ixgbe_free_receive_structures(struct ix_softc *); +void ixgbe_free_receive_buffers(struct rx_ring *); + +void ixgbe_enable_intr(struct ix_softc *); +void ixgbe_disable_intr(struct ix_softc *); +void ixgbe_update_stats_counters(struct ix_softc *); +bool ixgbe_txeof(struct tx_ring *); +bool ixgbe_rxeof(struct rx_ring *, int); +void ixgbe_rx_checksum(struct ix_softc *, u32, struct mbuf *); +void ixgbe_set_promisc(struct ix_softc *); +void ixgbe_disable_promisc(struct ix_softc *); +void ixgbe_set_multi(struct ix_softc *); +#ifdef IX_DEBUG +void ixgbe_print_hw_stats(struct ix_softc *); +#endif +void ixgbe_update_link_status(struct ix_softc *); +int ixgbe_get_buf(struct rx_ring *, int, struct mbuf *); +int ixgbe_encap(struct tx_ring *, struct mbuf *); +#if NVLAN > 0 +void ixgbe_enable_hw_vlans(struct ix_softc * sc); +#endif +#if 0 +void ixgbe_print_debug_info(struct ix_softc *); +int ixgbe_set_flowcntl(SYSCTL_HANDLER_ARGS); +#endif +int ixgbe_dma_malloc(struct ix_softc *, bus_size_t, + struct ixgbe_dma_alloc *, int); +void ixgbe_dma_free(struct ix_softc *, struct ixgbe_dma_alloc *); +#ifdef IX_CSUM_OFFLOAD +boolean_t ixgbe_tx_ctx_setup(struct tx_ring *, struct mbuf *); +boolean_t ixgbe_tso_setup(struct tx_ring *, struct mbuf *, u32 *); +#endif +void ixgbe_set_ivar(struct ix_softc *, u16, u8); +void ixgbe_configure_ivars(struct ix_softc *); +u8 *ixgbe_mc_array_itr(struct ixgbe_hw *, u8 **, u32 *); + +/* Legacy (single vector interrupt handler */ +int ixgbe_legacy_irq(void *); + +#if 0 +/* The MSI/X Interrupt handlers */ +int ixgbe_allocate_msix(struct ix_softc *); +int ixgbe_setup_msix(struct ix_softc *); +void ixgbe_msix_tx(void *); +void ixgbe_msix_rx(void *); +void ixgbe_msix_link(void *); +#endif + +#ifndef NO_82598_A0_SUPPORT +void desc_flip(void *); +#endif + +/********************************************************************* + * OpenBSD Device Interface Entry Points + *********************************************************************/ + +struct cfdriver ix_cd = { + 0, "ix", DV_IFNET +}; + +struct cfattach ix_ca = { + sizeof(struct ix_softc), ixgbe_probe, ixgbe_attach, ixgbe_detach +}; + +/* Total number of Interfaces - need for config sanity check */ +static int ixgbe_total_ports; + +/********************************************************************* + * Device identification routine + * + * ixgbe_probe determines if the driver should be loaded on + * sc based on PCI vendor/device id of the sc. + * + * return 0 on success, positive on failure + *********************************************************************/ + +int +ixgbe_probe(struct device *parent, void *match, void *aux) +{ + INIT_DEBUGOUT("ixgbe_probe: begin"); + + return (pci_matchbyid((struct pci_attach_args *)aux, ixgbe_devices, + sizeof(ixgbe_devices)/sizeof(ixgbe_devices[0]))); +} + +/********************************************************************* + * Device initialization routine + * + * The attach entry point is called when the driver is being loaded. + * This routine identifies the type of hardware, allocates all resources + * and initializes the hardware. + * + * return 0 on success, positive on failure + *********************************************************************/ + +void +ixgbe_attach(struct device *parent, struct device *self, void *aux) +{ + struct pci_attach_args *pa = (struct pci_attach_args *)aux; + struct ix_softc *sc = (struct ix_softc *)self; + int error = 0; + u32 ctrl_ext; + + INIT_DEBUGOUT("ixgbe_attach: begin"); + + sc->osdep.os_sc = sc; + sc->osdep.os_pa = pa; + + /* Core Lock Init*/ + mtx_init(&sc->core_mtx, IPL_NET); + + /* Set up the timer callout */ + timeout_set(&sc->timer, ixgbe_local_timer, sc); + + /* Determine hardware revision */ + ixgbe_identify_hardware(sc); + + /* Indicate to RX setup to use Jumbo Clusters */ + sc->bigbufs = FALSE; + sc->num_tx_desc = DEFAULT_TXD; + sc->num_rx_desc = DEFAULT_RXD; + sc->rx_process_limit = 100; // XXX + + /* Do base PCI setup - map BAR0 */ + if (ixgbe_allocate_pci_resources(sc)) { + printf(": allocation of PCI resources failed\n"); + goto err_out; + } + + /* Allocate our TX/RX Queues */ + if (ixgbe_allocate_queues(sc)) + goto err_out; + + /* Initialize the shared code */ + sc->hw.mac.type = ixgbe_mac_82598EB; + if (ixgbe_init_ops_82598(&sc->hw) != 0) { + printf(": failed to init the 82598EB\n"); + goto err_late; + } + + /* Initialize the hardware */ + if (ixgbe_hardware_init(sc)) { + printf(": unable to initialize the hardware\n"); + goto err_late; + } + + /* XXX sc->msix > 1 && ixgbe_allocate_msix() */ + error = ixgbe_allocate_legacy(sc); + if (error) + goto err_late; + + /* Setup OS specific network interface */ + ixgbe_setup_interface(sc); + + /* Initialize statistics */ + ixgbe_update_stats_counters(sc); + + /* let hardware know driver is loaded */ + ctrl_ext = IXGBE_READ_REG(&sc->hw, IXGBE_CTRL_EXT); + ctrl_ext |= IXGBE_CTRL_EXT_DRV_LOAD; + IXGBE_WRITE_REG(&sc->hw, IXGBE_CTRL_EXT, ctrl_ext); + + sc->powerhook = powerhook_establish(ixgbe_power, sc); + sc->shutdownhook = shutdownhook_establish(ixgbe_shutdown, sc); + + printf(", address %s\n", ether_sprintf(sc->hw.mac.addr)); + + INIT_DEBUGOUT("ixgbe_attach: end"); + return; + +err_late: + ixgbe_free_transmit_structures(sc); + ixgbe_free_receive_structures(sc); +err_out: + ixgbe_free_pci_resources(sc); +} + +/********************************************************************* + * Device removal routine + * + * The detach entry point is called when the driver is being removed. + * This routine stops the sc and deallocates all the resources + * that were allocated for driver operation. + * + * return 0 on success, positive on failure + *********************************************************************/ + +int +ixgbe_detach(struct device *self, int flags) +{ + struct ix_softc *sc = (struct ix_softc *)self; + struct ifnet *ifp = &sc->arpcom.ac_if; + u32 ctrl_ext; + + INIT_DEBUGOUT("ixgbe_detach: begin"); + + ixgbe_stop(sc); + + /* let hardware know driver is unloading */ + ctrl_ext = IXGBE_READ_REG(&sc->hw, IXGBE_CTRL_EXT); + ctrl_ext &= ~IXGBE_CTRL_EXT_DRV_LOAD; + IXGBE_WRITE_REG(&sc->hw, IXGBE_CTRL_EXT, ctrl_ext); + + ether_ifdetach(ifp); + if_detach(ifp); + + timeout_del(&sc->timer); + ixgbe_free_pci_resources(sc); + + ixgbe_free_transmit_structures(sc); + ixgbe_free_receive_structures(sc); + + return (0); +} + +void +ixgbe_power(int why, void *arg) +{ + struct ix_softc *sc = (struct ix_softc *)arg; + struct ifnet *ifp; + + if (why == PWR_RESUME) { + ifp = &sc->arpcom.ac_if; + if (ifp->if_flags & IFF_UP) + ixgbe_init(sc); + } +} + +/********************************************************************* + * + * Shutdown entry point + * + **********************************************************************/ + +void +ixgbe_shutdown(void *arg) +{ + struct ix_softc *sc = (struct ix_softc *)arg; + + ixgbe_stop(sc); +} + +/********************************************************************* + * Transmit entry point + * + * ixgbe_start is called by the stack to initiate a transmit. + * The driver will remain in this routine as long as there are + * packets to transmit and transmit resources are available. + * In case resources are not available stack is notified and + * the packet is requeued. + **********************************************************************/ + +void +ixgbe_start_locked(struct tx_ring *txr, struct ifnet * ifp) +{ + struct mbuf *m_head; + struct ix_softc *sc = txr->sc; + + if ((ifp->if_flags & (IFF_RUNNING|IFF_OACTIVE)) != IFF_RUNNING) + return; + + if (!sc->link_active) + return; + + for (;;) { + IFQ_POLL(&ifp->if_snd, m_head); + + if (m_head == NULL) + break; + + if (ixgbe_encap(txr, m_head)) { + ifp->if_flags |= IFF_OACTIVE; + break; + } + + IFQ_DEQUEUE(&ifp->if_snd, m_head); + +#if NBPFILTER > 0 + if (ifp->if_bpf) + bpf_mtap(ifp->if_bpf, m_head, BPF_DIRECTION_OUT); +#endif + + /* Set timeout in case hardware has problems transmitting */ + txr->watchdog_timer = IXGBE_TX_TIMEOUT; + ifp->if_timer = IXGBE_TX_TIMEOUT; + } + return; +} + + +void +ixgbe_start(struct ifnet *ifp) +{ + struct ix_softc *sc = ifp->if_softc; + struct tx_ring *txr = sc->tx_rings; + u32 queue = 0; + +#if 0 + /* + * This is really just here for testing + * TX multiqueue, ultimately what is + * needed is the flow support in the stack + * and appropriate logic here to deal with + * it. -jfv + */ + if (sc->num_tx_queues > 1) + queue = (curcpu % sc->num_tx_queues); +#endif + + txr = &sc->tx_rings[queue]; + + if (ifp->if_flags & IFF_RUNNING) + ixgbe_start_locked(txr, ifp); + + return; +} + +/********************************************************************* + * Ioctl entry point + * + * ixgbe_ioctl is called when the user wants to configure the + * interface. + * + * return 0 on success, positive on failure + **********************************************************************/ + +int +ixgbe_ioctl(struct ifnet * ifp, u_long command, caddr_t data) +{ + int s, error = 0; + struct ifreq *ifr = (struct ifreq *) data; + struct ifaddr *ifa = (struct ifaddr *) data; + struct ix_softc *sc = ifp->if_softc; + + s = splnet(); + + if ((error = ether_ioctl(ifp, &sc->arpcom, command, data)) > 0) { + splx(s); + return (error); + } + + switch (command) { + case SIOCSIFADDR: + IOCTL_DEBUGOUT("ioctl: SIOCxIFADDR (Get/Set Interface Addr)"); + ifp->if_flags |= IFF_UP; + if (!(ifp->if_flags & IFF_RUNNING)) + ixgbe_init(sc); +#ifdef INET + if (ifa->ifa_addr->sa_family == AF_INET) + arp_ifinit(&sc->arpcom, ifa); +#endif + break; + case SIOCSIFMTU: + IOCTL_DEBUGOUT("ioctl: SIOCSIFMTU (Set Interface MTU)"); + if (ifr->ifr_mtu < ETHERMIN || ifr->ifr_mtu > ifp->if_hardmtu) + error = EINVAL; + else if (ifp->if_mtu != ifr->ifr_mtu) { + ifp->if_mtu = ifr->ifr_mtu; + sc->max_frame_size = + ifp->if_mtu + ETHER_HDR_LEN + ETHER_CRC_LEN; + ixgbe_init(sc); + } + break; + case SIOCSIFFLAGS: + IOCTL_DEBUGOUT("ioctl: SIOCSIFFLAGS (Set Interface Flags)"); + if (ifp->if_flags & IFF_UP) { + if ((ifp->if_flags & IFF_RUNNING)) { + if ((ifp->if_flags ^ sc->if_flags) & + (IFF_PROMISC | IFF_ALLMULTI)) { + ixgbe_disable_promisc(sc); + ixgbe_set_promisc(sc); + } + } else + ixgbe_init(sc); + } else + if (ifp->if_flags & IFF_RUNNING) + ixgbe_stop(sc); + sc->if_flags = ifp->if_flags; + break; + case SIOCADDMULTI: + case SIOCDELMULTI: + IOCTL_DEBUGOUT("ioctl: SIOC(ADD|DEL)MULTI"); + error = (command == SIOCADDMULTI) ? + ether_addmulti(ifr, &sc->arpcom) : + ether_delmulti(ifr, &sc->arpcom); + + if (error == ENETRESET) { + if (ifp->if_flags & IFF_RUNNING) { + ixgbe_disable_intr(sc); + ixgbe_set_multi(sc); + ixgbe_enable_intr(sc); + } + error = 0; + } + break; + case SIOCSIFMEDIA: + case SIOCGIFMEDIA: + IOCTL_DEBUGOUT("ioctl: SIOCxIFMEDIA (Get/Set Interface Media)"); + error = ifmedia_ioctl(ifp, ifr, &sc->media, command); + break; + default: + IOCTL_DEBUGOUT1("ioctl: UNKNOWN (0x%X)\n", (int)command); + error = ENOTTY; + break; + } + + splx(s); + return (error); +} + +/********************************************************************* + * Watchdog entry point + * + * This routine is called by the local timer + * to detect hardware hangs . + * + **********************************************************************/ + +void +ixgbe_watchdog(struct ifnet * ifp) +{ + struct ix_softc *sc = (struct ix_softc *)ifp->if_softc; + struct tx_ring *txr = sc->tx_rings; + struct ixgbe_hw *hw = &sc->hw; + bool tx_hang = FALSE; + int i; + + /* + * The timer is set to 5 every time ixgbe_start() queues a packet. + * Then ixgbe_txeof() keeps resetting to 5 as long as it cleans at + * least one descriptor. + * Finally, anytime all descriptors are clean the timer is + * set to 0. + */ + for (i = 0; i < sc->num_tx_queues; i++, txr++) { + if (txr->watchdog_timer == 0 || --txr->watchdog_timer) + continue; + else { + tx_hang = TRUE; + break; + } + } + if (tx_hang == FALSE) + return; + + /* + * If we are in this routine because of pause frames, then don't + * reset the hardware. + */ + if (IXGBE_READ_REG(hw, IXGBE_TFCS) & IXGBE_TFCS_TXOFF) { + for (i = 0; i < sc->num_tx_queues; i++, txr++) + txr->watchdog_timer = IXGBE_TX_TIMEOUT; + ifp->if_timer = IXGBE_TX_TIMEOUT; + return; + } + + + printf("%s: Watchdog timeout -- resetting\n", ifp->if_xname); + for (i = 0; i < sc->num_tx_queues; i++, txr++) { + printf("%s: Queue(%d) tdh = %d, hw tdt = %d\n", ifp->if_xname, i, + IXGBE_READ_REG(hw, IXGBE_TDH(i)), + IXGBE_READ_REG(hw, IXGBE_TDT(i))); + printf("%s: TX(%d) desc avail = %d, Next TX to Clean = %d\n", ifp->if_xname, + i, txr->tx_avail, txr->next_tx_to_clean); + } + ifp->if_flags &= ~IFF_RUNNING; + sc->watchdog_events++; + + ixgbe_init(sc); + return; +} + +/********************************************************************* + * Init entry point + * + * This routine is used in two ways. It is used by the stack as + * init entry point in network interface structure. It is also used + * by the driver as a hw/sw initialization routine to get to a + * consistent state. + * + * return 0 on success, positive on failure + **********************************************************************/ +#define IXGBE_MHADD_MFS_SHIFT 16 + +void +ixgbe_init(void *arg) +{ + struct ix_softc *sc = (struct ix_softc *)arg; + struct ifnet *ifp = &sc->arpcom.ac_if; + u32 txdctl, rxdctl, mhadd, gpie; + int i, s; + + INIT_DEBUGOUT("ixgbe_init: begin"); + + s = splnet(); + + ixgbe_stop(sc); + + /* Get the latest mac address, User can use a LAA */ + bcopy(sc->arpcom.ac_enaddr, sc->hw.mac.addr, + IXGBE_ETH_LENGTH_OF_ADDRESS); + ixgbe_hw(&sc->hw, set_rar, 0, sc->hw.mac.addr, 0, 1); + sc->hw.addr_ctrl.rar_used_count = 1; + + /* Initialize the hardware */ + if (ixgbe_hardware_init(sc)) { + printf("%s: Unable to initialize the hardware\n", + ifp->if_xname); + splx(s); + return; + } + +#if NVLAN > 0 + if (ifp->if_capabilities & IFCAP_VLAN_HWTAGGING) + ixgbe_enable_hw_vlans(sc); +#endif + + /* Prepare transmit descriptors and buffers */ + if (ixgbe_setup_transmit_structures(sc)) { + printf("%s: Could not setup transmit structures\n", + ifp->if_xname); + ixgbe_stop(sc); + splx(s); + return; + } + + ixgbe_initialize_transmit_units(sc); + + /* Setup Multicast table */ + ixgbe_set_multi(sc); + + /* + * If we are resetting MTU smaller than 2K + * drop to small RX buffers + */ + if (sc->max_frame_size <= MCLBYTES) + sc->bigbufs = FALSE; + + /* Prepare receive descriptors and buffers */ + if (ixgbe_setup_receive_structures(sc)) { + printf("%s: Could not setup receive structures\n", ifp->if_xname); + ixgbe_stop(sc); + splx(s); + return; + } + + /* Configure RX settings */ + ixgbe_initialize_receive_units(sc); + + gpie = IXGBE_READ_REG(&sc->hw, IXGBE_GPIE); + /* Enable Fan Failure Interrupt */ + if (sc->hw.phy.media_type == ixgbe_media_type_copper) + gpie |= IXGBE_SDP1_GPIEN; + if (sc->msix) { + /* Enable Enhanced MSIX mode */ + gpie |= IXGBE_GPIE_MSIX_MODE; + gpie |= IXGBE_GPIE_EIAME | IXGBE_GPIE_PBA_SUPPORT | + IXGBE_GPIE_OCD; + } + IXGBE_WRITE_REG(&sc->hw, IXGBE_GPIE, gpie); + + /* Set MTU size */ + if (ifp->if_mtu > ETHERMTU) { + mhadd = IXGBE_READ_REG(&sc->hw, IXGBE_MHADD); + mhadd &= ~IXGBE_MHADD_MFS_MASK; + mhadd |= sc->max_frame_size << IXGBE_MHADD_MFS_SHIFT; + IXGBE_WRITE_REG(&sc->hw, IXGBE_MHADD, mhadd); + } + + /* Now enable all the queues */ + + for (i = 0; i < sc->num_tx_queues; i++) { + txdctl = IXGBE_READ_REG(&sc->hw, IXGBE_TXDCTL(i)); + txdctl |= IXGBE_TXDCTL_ENABLE; + /* Set WTHRESH to 8, burst writeback */ + txdctl |= (8 << 16); + IXGBE_WRITE_REG(&sc->hw, IXGBE_TXDCTL(i), txdctl); + } + + for (i = 0; i < sc->num_rx_queues; i++) { + rxdctl = IXGBE_READ_REG(&sc->hw, IXGBE_RXDCTL(i)); + /* PTHRESH set to 32 */ + rxdctl |= 0x0020; + rxdctl |= IXGBE_RXDCTL_ENABLE; + IXGBE_WRITE_REG(&sc->hw, IXGBE_RXDCTL(i), rxdctl); + } + + timeout_add(&sc->timer, hz); + + /* Set up MSI/X routing */ + ixgbe_configure_ivars(sc); + + ixgbe_enable_intr(sc); + + /* Now inform the stack we're ready */ + ifp->if_flags |= IFF_RUNNING; + ifp->if_flags &= ~IFF_OACTIVE; + + splx(s); +} + +/********************************************************************* + * + * Legacy Interrupt Service routine + * + **********************************************************************/ + +int +ixgbe_legacy_irq(void *arg) +{ + struct ix_softc *sc = (struct ix_softc *)arg; + struct ifnet *ifp = &sc->arpcom.ac_if; + u32 reg_eicr; + struct tx_ring *txr = sc->tx_rings; + struct rx_ring *rxr = sc->rx_rings; + struct ixgbe_hw *hw = &sc->hw; + int claimed = 0; + + for (;;) { + reg_eicr = IXGBE_READ_REG(&sc->hw, IXGBE_EICR); + if (reg_eicr == 0) + break; + + claimed = 1; + + if (ifp->if_flags & IFF_RUNNING) { + ixgbe_rxeof(rxr, -1); + ixgbe_txeof(txr); + } + + /* Check for fan failure */ + if ((hw->phy.media_type == ixgbe_media_type_copper) && + (reg_eicr & IXGBE_EICR_GPI_SDP1)) { + printf("%s: \nCRITICAL: FAN FAILURE!! " + "REPLACE IMMEDIATELY!!\n", ifp->if_xname); + IXGBE_WRITE_REG(&sc->hw, IXGBE_EIMS, + IXGBE_EICR_GPI_SDP1); + } + + /* Link status change */ + if (reg_eicr & IXGBE_EICR_LSC) { + timeout_del(&sc->timer); + ixgbe_update_link_status(sc); + timeout_add(&sc->timer, hz); + } + } + + if (ifp->if_flags & IFF_RUNNING && !IFQ_IS_EMPTY(&ifp->if_snd)) + ixgbe_start_locked(txr, ifp); + + return (claimed); +} + +/********************************************************************* + * + * Media Ioctl callback + * + * This routine is called whenever the user queries the status of + * the interface using ifconfig. + * + **********************************************************************/ +void +ixgbe_media_status(struct ifnet * ifp, struct ifmediareq * ifmr) +{ + struct ix_softc *sc = ifp->if_softc; + + INIT_DEBUGOUT("ixgbe_media_status: begin"); + ixgbe_update_link_status(sc); + + ifmr->ifm_status = IFM_AVALID; + ifmr->ifm_active = IFM_ETHER; + + if (!sc->link_active) { + ifmr->ifm_status |= IFM_NONE; + return; + } + + ifmr->ifm_status |= IFM_ACTIVE; + + switch (sc->link_speed) { + case IXGBE_LINK_SPEED_1GB_FULL: + ifmr->ifm_active |= IFM_1000_T | IFM_FDX; + break; + case IXGBE_LINK_SPEED_10GB_FULL: + ifmr->ifm_active |= sc->optics | IFM_FDX; + break; + } +} + +/********************************************************************* + * + * Media Ioctl callback + * + * This routine is called when the user changes speed/duplex using + * media/mediopt option with ifconfig. + * + **********************************************************************/ +int +ixgbe_media_change(struct ifnet * ifp) +{ + struct ix_softc *sc = ifp->if_softc; + struct ifmedia *ifm = &sc->media; + + INIT_DEBUGOUT("ixgbe_media_change: begin"); + + if (IFM_TYPE(ifm->ifm_media) != IFM_ETHER) + return (EINVAL); + + switch (IFM_SUBTYPE(ifm->ifm_media)) { + case IFM_AUTO: + sc->hw.mac.autoneg = TRUE; + sc->hw.phy.autoneg_advertised = + IXGBE_LINK_SPEED_1GB_FULL | IXGBE_LINK_SPEED_10GB_FULL; + break; + default: + printf("%s: Only auto media type\n", ifp->if_xname); + return (EINVAL); + } + + return (0); +} + +/********************************************************************* + * + * This routine maps the mbufs to tx descriptors. + * WARNING: while this code is using an MQ style infrastructure, + * it would NOT work as is with more than 1 queue. + * + * return 0 on success, positive on failure + **********************************************************************/ + +int +ixgbe_encap(struct tx_ring *txr, struct mbuf *m_head) +{ + struct ix_softc *sc = txr->sc; + u32 olinfo_status = 0, cmd_type_len = 0; + int i, j, error; + int first, last = 0; + bus_dmamap_t map; + struct ixgbe_tx_buf *txbuf, *txbuf_mapped; + union ixgbe_adv_tx_desc *txd = NULL; +#ifdef IX_CSUM_OFFLOAD + u32 paylen = 0; +#endif + + /* Basic descriptor defines */ + cmd_type_len |= IXGBE_ADVTXD_DTYP_DATA; + cmd_type_len |= IXGBE_ADVTXD_DCMD_IFCS | IXGBE_ADVTXD_DCMD_DEXT; + +#if NVLAN > 0 + /* VLAN tagging? */ + if ((m_head->m_flags & (M_PROTO1|M_PKTHDR)) == (M_PROTO1|M_PKTHDR) && + m_head->m_pkthdr.rcvif != NULL) + cmd_type_len |= IXGBE_ADVTXD_DCMD_VLE; +#endif + + /* + * Force a cleanup if number of TX descriptors + * available is below the threshold. If it fails + * to get above, then abort transmit. + */ + if (txr->tx_avail <= IXGBE_TX_CLEANUP_THRESHOLD) { + ixgbe_txeof(txr); + /* Make sure things have improved */ + if (txr->tx_avail <= IXGBE_TX_OP_THRESHOLD) { + txr->no_tx_desc_avail++; + return (ENOBUFS); + } + } + + /* + * Important to capture the first descriptor + * used because it will contain the index of + * the one we tell the hardware to report back + */ + first = txr->next_avail_tx_desc; + txbuf = &txr->tx_buffers[first]; + txbuf_mapped = txbuf; + map = txbuf->map; + + /* + * Map the packet for DMA. + */ + error = bus_dmamap_load_mbuf(txr->txtag, map, + m_head, BUS_DMA_NOWAIT); + + if (error == ENOMEM) { + sc->no_tx_dma_setup++; + return (error); + } else if (error != 0) { + sc->no_tx_dma_setup++; + return (error); + } + + /* Make certain there are enough descriptors */ + if (map->dm_nsegs > txr->tx_avail - 2) { + txr->no_tx_desc_avail++; + error = ENOBUFS; + goto xmit_fail; + } + +#ifdef IX_CSUM_OFFLOAD + /* + * Set the appropriate offload context + * this becomes the first descriptor of + * a packet. + */ + if (ixgbe_tso_setup(txr, m_head, &paylen)) { + cmd_type_len |= IXGBE_ADVTXD_DCMD_TSE; + olinfo_status |= IXGBE_TXD_POPTS_IXSM << 8; + olinfo_status |= IXGBE_TXD_POPTS_TXSM << 8; + olinfo_status |= paylen << IXGBE_ADVTXD_PAYLEN_SHIFT; + ++sc->tso_tx; + } else if (ixgbe_tx_ctx_setup(txr, m_head)) + olinfo_status |= IXGBE_TXD_POPTS_IXSM << 8; +#endif + + i = txr->next_avail_tx_desc; + for (j = 0; j < map->dm_nsegs; j++) { + txbuf = &txr->tx_buffers[i]; + txd = &txr->tx_base[i]; + + txd->read.buffer_addr = htole64(map->dm_segs[j].ds_addr); + txd->read.cmd_type_len = htole32(txr->txd_cmd | + cmd_type_len | map->dm_segs[j].ds_len); + txd->read.olinfo_status = htole32(olinfo_status); + last = i; /* Next descriptor that will get completed */ + + if (++i == sc->num_tx_desc) + i = 0; + + txbuf->m_head = NULL; + + /* + * we have to do this inside the loop right now + * because of the hardware workaround. + */ + if (j == (map->dm_nsegs -1)) /* Last descriptor gets EOP and RS */ + txd->read.cmd_type_len |= + htole32(IXGBE_TXD_CMD_EOP | IXGBE_TXD_CMD_RS); +#ifndef NO_82598_A0_SUPPORT + if (sc->hw.revision_id == 0) + desc_flip(txd); +#endif + } + + txr->tx_avail -= map->dm_nsegs; + txr->next_avail_tx_desc = i; + + txbuf->m_head = m_head; + txbuf->map = map; + bus_dmamap_sync(txr->txtag, map, 0, map->dm_mapsize, + BUS_DMASYNC_PREWRITE); + + /* Set the index of the descriptor that will be marked done */ + txbuf = &txr->tx_buffers[first]; + + bus_dmamap_sync(txr->txdma.dma_tag, txr->txdma.dma_map, + 0, txr->txdma.dma_map->dm_mapsize, + BUS_DMASYNC_PREREAD | BUS_DMASYNC_PREWRITE); + + /* + * Advance the Transmit Descriptor Tail (Tdt), this tells the + * hardware that this frame is available to transmit. + */ + IXGBE_WRITE_REG(&sc->hw, IXGBE_TDT(txr->me), i); + ++txr->tx_packets; + return (0); + +xmit_fail: + bus_dmamap_unload(txr->txtag, txbuf->map); + return (error); + +} + +void +ixgbe_set_promisc(struct ix_softc *sc) +{ + + u_int32_t reg_rctl; + struct ifnet *ifp = &sc->arpcom.ac_if; + + reg_rctl = IXGBE_READ_REG(&sc->hw, IXGBE_FCTRL); + + if (ifp->if_flags & IFF_PROMISC) { + reg_rctl |= (IXGBE_FCTRL_UPE | IXGBE_FCTRL_MPE); + IXGBE_WRITE_REG(&sc->hw, IXGBE_FCTRL, reg_rctl); + } else if (ifp->if_flags & IFF_ALLMULTI) { + reg_rctl |= IXGBE_FCTRL_MPE; + reg_rctl &= ~IXGBE_FCTRL_UPE; + IXGBE_WRITE_REG(&sc->hw, IXGBE_FCTRL, reg_rctl); + } + return; +} + +void +ixgbe_disable_promisc(struct ix_softc * sc) +{ + u_int32_t reg_rctl; + + reg_rctl = IXGBE_READ_REG(&sc->hw, IXGBE_FCTRL); + + reg_rctl &= (~IXGBE_FCTRL_UPE); + reg_rctl &= (~IXGBE_FCTRL_MPE); + IXGBE_WRITE_REG(&sc->hw, IXGBE_FCTRL, reg_rctl); + + return; +} + + +/********************************************************************* + * Multicast Update + * + * This routine is called whenever multicast address list is updated. + * + **********************************************************************/ +#define IXGBE_RAR_ENTRIES 16 + +void +ixgbe_set_multi(struct ix_softc *sc) +{ + u32 fctrl; + u8 mta[MAX_NUM_MULTICAST_ADDRESSES * IXGBE_ETH_LENGTH_OF_ADDRESS]; + u8 *update_ptr; + struct ether_multi *enm; + struct ether_multistep step; + int mcnt = 0; + struct ifnet *ifp = &sc->arpcom.ac_if; + + IOCTL_DEBUGOUT("ixgbe_set_multi: begin"); + + fctrl = IXGBE_READ_REG(&sc->hw, IXGBE_FCTRL); + fctrl |= (IXGBE_FCTRL_UPE | IXGBE_FCTRL_MPE); + if (ifp->if_flags & IFF_PROMISC) + fctrl |= (IXGBE_FCTRL_UPE | IXGBE_FCTRL_MPE); + else if (ifp->if_flags & IFF_ALLMULTI) { + fctrl |= IXGBE_FCTRL_MPE; + fctrl &= ~IXGBE_FCTRL_UPE; + } else + fctrl &= ~(IXGBE_FCTRL_UPE | IXGBE_FCTRL_MPE); + + IXGBE_WRITE_REG(&sc->hw, IXGBE_FCTRL, fctrl); + + ETHER_FIRST_MULTI(step, &sc->arpcom, enm); + while (enm != NULL) { + if (bcmp(enm->enm_addrlo, enm->enm_addrhi, ETHER_ADDR_LEN)) { + ifp->if_flags |= IFF_ALLMULTI; + mcnt = MAX_NUM_MULTICAST_ADDRESSES; + } + if (mcnt == MAX_NUM_MULTICAST_ADDRESSES) + break; + bcopy(enm->enm_addrlo, + &mta[mcnt * IXGBE_ETH_LENGTH_OF_ADDRESS], + IXGBE_ETH_LENGTH_OF_ADDRESS); + mcnt++; + ETHER_NEXT_MULTI(step, enm); + } + + update_ptr = mta; + ixgbe_hw(&sc->hw, update_mc_addr_list, + update_ptr, mcnt, ixgbe_mc_array_itr); + + return; +} + +/* + * This is an iterator function now needed by the multicast + * shared code. It simply feeds the shared code routine the + * addresses in the array of ixgbe_set_multi() one by one. + */ +u8 * +ixgbe_mc_array_itr(struct ixgbe_hw *hw, u8 **update_ptr, u32 *vmdq) +{ + u8 *addr = *update_ptr; + u8 *newptr; + *vmdq = 0; + + newptr = addr + IXGBE_ETH_LENGTH_OF_ADDRESS; + *update_ptr = newptr; + return addr; +} + + +/********************************************************************* + * Timer routine + * + * This routine checks for link status,updates statistics, + * and runs the watchdog timer. + * + **********************************************************************/ + +void +ixgbe_local_timer(void *arg) +{ + struct ix_softc *sc = arg; +#ifdef IX_DEBUG + struct ifnet *ifp = &sc->arpcom.ac_if; +#endif + int s; + + s = splnet(); + + ixgbe_update_link_status(sc); + ixgbe_update_stats_counters(sc); + +#ifdef IX_DEBUG + if ((ifp->if_flags & (IFF_RUNNING|IFF_DEBUG)) == + (IFF_RUNNING|IFF_DEBUG)) + ixgbe_print_hw_stats(sc); +#endif + + timeout_add(&sc->timer, hz); + + splx(s); +} + +void +ixgbe_update_link_status(struct ix_softc *sc) +{ + boolean_t link_up = FALSE; + struct ifnet *ifp = &sc->arpcom.ac_if; + struct tx_ring *txr = sc->tx_rings; + int i; + + ixgbe_hw(&sc->hw, check_link, &sc->link_speed, &link_up, 0); + + switch (sc->link_speed) { + case IXGBE_LINK_SPEED_UNKNOWN: + ifp->if_baudrate = 0; + break; + case IXGBE_LINK_SPEED_100_FULL: + ifp->if_baudrate = IF_Mbps(100); + break; + case IXGBE_LINK_SPEED_1GB_FULL: + ifp->if_baudrate = IF_Gbps(1); + break; + case IXGBE_LINK_SPEED_10GB_FULL: + ifp->if_baudrate = IF_Gbps(10); + break; + } + + if (link_up){ + if (sc->link_active == FALSE) { + sc->link_active = TRUE; + ifp->if_link_state = LINK_STATE_FULL_DUPLEX; + if_link_state_change(ifp); + } + } else { /* Link down */ + if (sc->link_active == TRUE) { + ifp->if_baudrate = 0; + ifp->if_link_state = LINK_STATE_DOWN; + if_link_state_change(ifp); + sc->link_active = FALSE; + for (i = 0; i < sc->num_tx_queues; + i++, txr++) + txr->watchdog_timer = FALSE; + ifp->if_timer = 0; + } + } + + return; +} + + + +/********************************************************************* + * + * This routine disables all traffic on the sc by issuing a + * global reset on the MAC and deallocates TX/RX buffers. + * + **********************************************************************/ + +void +ixgbe_stop(void *arg) +{ + struct ix_softc *sc = arg; + struct ifnet *ifp = &sc->arpcom.ac_if; + + INIT_DEBUGOUT("ixgbe_stop: begin\n"); + ixgbe_disable_intr(sc); + + /* Tell the stack that the interface is no longer active */ + ifp->if_flags &= ~(IFF_RUNNING | IFF_OACTIVE); + + ixgbe_hw(&sc->hw, reset_hw); + sc->hw.adapter_stopped = FALSE; + ixgbe_hw(&sc->hw, stop_adapter); + timeout_del(&sc->timer); + + /* reprogram the RAR[0] in case user changed it. */ + ixgbe_hw(&sc->hw, set_rar, 0, sc->hw.mac.addr, 0, IXGBE_RAH_AV); +} + + +/********************************************************************* + * + * Determine hardware revision. + * + **********************************************************************/ +void +ixgbe_identify_hardware(struct ix_softc *sc) +{ + struct ixgbe_osdep *os = &sc->osdep; + struct pci_attach_args *pa = os->os_pa; + u_int32_t reg; + + /* Save off the information about this board */ + sc->hw.vendor_id = PCI_VENDOR(pa->pa_id); + sc->hw.device_id = PCI_PRODUCT(pa->pa_id); + + reg = pci_conf_read(pa->pa_pc, pa->pa_tag, PCI_CLASS_REG); + sc->hw.revision_id = PCI_REVISION(reg); + + reg = pci_conf_read(pa->pa_pc, pa->pa_tag, PCI_SUBSYS_ID_REG); + sc->hw.subsystem_vendor_id = PCI_VENDOR(reg); + sc->hw.subsystem_device_id = PCI_PRODUCT(reg); + + ixgbe_total_ports++; + switch (sc->hw.device_id) { + case PCI_PRODUCT_INTEL_82598AF_DUAL: + case PCI_PRODUCT_INTEL_82598EB_CX4_DUAL: + case PCI_PRODUCT_INTEL_82598AT_DUAL: + ixgbe_total_ports++; + break; + } + + switch (sc->hw.device_id) { + case PCI_PRODUCT_INTEL_82598AF_DUAL: + case PCI_PRODUCT_INTEL_82598AF: + sc->optics = IFM_10G_SR; + break; + case PCI_PRODUCT_INTEL_82598EB_CX4_DUAL: + case PCI_PRODUCT_INTEL_82598EB_CX4: + sc->optics = IFM_10G_CX4; + break; + case PCI_PRODUCT_INTEL_82598EB_XF_LR: + sc->optics = IFM_10G_LR; + break; + case PCI_PRODUCT_INTEL_82598AT_DUAL: + case PCI_PRODUCT_INTEL_82598AT: + default: + sc->optics = IFM_AUTO; + break; + } +} + +/********************************************************************* + * + * Setup the Legacy or MSI Interrupt handler + * + **********************************************************************/ +int +ixgbe_allocate_legacy(struct ix_softc *sc) +{ + struct ifnet *ifp = &sc->arpcom.ac_if; + struct ixgbe_osdep *os = &sc->osdep; + struct pci_attach_args *pa = os->os_pa; + const char *intrstr = NULL; + pci_chipset_tag_t pc = pa->pa_pc; + pci_intr_handle_t ih; + + /* Legacy RID at 0 */ + if (sc->msix == 0) + sc->rid[0] = 0; + + /* We allocate a single interrupt resource */ + if (pci_intr_map(pa, &ih)) { + printf(": couldn't map interrupt\n"); + return (ENXIO); + } + + intrstr = pci_intr_string(pc, ih); + sc->tag[0] = pci_intr_establish(pc, ih, IPL_NET, + ixgbe_legacy_irq, sc, ifp->if_xname); + if (sc->tag[0] == NULL) { + printf(": couldn't establish interrupt"); + if (intrstr != NULL) + printf(" at %s", intrstr); + printf("\n"); + return (ENXIO); + } + printf(": %s", intrstr); + + return (0); +} + +#if 0 +/********************************************************************* + * + * Setup MSIX Interrupt resources and handlers + * + **********************************************************************/ +int +ixgbe_allocate_msix(struct ix_softc *sc) +{ + struct ifnet *ifp = &sc->arpcom.ac_if; + struct tx_ring *txr = sc->tx_rings; + struct rx_ring *rxr = sc->rx_rings; + int error, vector = 0, i; + + /* TX setup: the code is here for multi tx, + there are other parts of the driver not ready for it */ + for (i = 0; i < sc->num_tx_queues; i++, vector++, txr++) { + sc->res[vector] = bus_alloc_resource_any(dev, + SYS_RES_IRQ, &sc->rid[vector], + RF_SHAREABLE | RF_ACTIVE); + if (!sc->res[vector]) { + printf("%s: Unable to allocate" + " bus resource: tx interrupt [%d]\n", ifp->if_xname, vector); + return (ENXIO); + } + /* Set the handler function */ + error = bus_setup_intr(dev, sc->res[vector], + INTR_TYPE_NET | INTR_MPSAFE, NULL, + ixgbe_msix_tx, txr, &sc->tag[vector]); + if (error) { + sc->res[vector] = NULL; + printf("%s: Failed to register TX handler"); + return (error); + } + txr->msix = vector; + txr->eims = IXGBE_IVAR_TX_QUEUE(vector); + /* Make tasklet for deferred handling - one per queue */ + txr->tx_task = ixgbe_handle_tx; + txr->tq = workq_create("ix_txq", sc->num_tx_queues + 1); + } + + /* RX setup */ + for (i = 0; i < sc->num_rx_queues; i++, vector++, rxr++) { + sc->res[vector] = bus_alloc_resource_any(dev, + SYS_RES_IRQ, &sc->rid[vector], + RF_SHAREABLE | RF_ACTIVE); + if (!sc->res[vector]) { + printf("%s: Unable to allocate" + " bus resource: rx interrupt [%d]," + "rid = %d\n", ifp->if_xname, i, sc->rid[vector]); + return (ENXIO); + } + /* Set the handler function */ + error = bus_setup_intr(dev, sc->res[vector], + INTR_TYPE_NET | INTR_MPSAFE, NULL, ixgbe_msix_rx, + rxr, &sc->tag[vector]); + if (error) { + sc->res[vector] = NULL; + printf("%s: Failed to register RX handler"); + return (error); + } + rxr->msix = vector; + rxr->eims = IXGBE_IVAR_RX_QUEUE(vector); + rxr->rx_task = ixgbe_handle_rx; + rxr->tq = workq_create("ix_rxq", sc->num_rx_queues + 1); + } + + /* Now for Link changes */ + sc->res[vector] = bus_alloc_resource_any(dev, + SYS_RES_IRQ, &sc->rid[vector], RF_SHAREABLE | RF_ACTIVE); + if (!sc->res[vector]) { + printf("%s: Unable to allocate" + " bus resource: Link interrupt [%d]\n", ifp->if_xname, sc->rid[vector]); + return (ENXIO); + } + /* Set the link handler function */ + error = bus_setup_intr(dev, sc->res[vector], + INTR_TYPE_NET | INTR_MPSAFE, NULL, ixgbe_msix_link, + sc, &sc->tag[vector]); + if (error) { + sc->res[vector] = NULL; + printf("%s: Failed to register LINK handler"); + return (error); + } + sc->linkvec = vector; + sc->link_task = ixgbe_handle_link; + + return (0); +} + +/* + * Setup Either MSI/X or MSI + */ +int +ixgbe_setup_msix(struct ix_softc *sc) +{ + struct ifnet *ifp = &sc->arpcom.ac_if; + int rid, want, queues, msgs; + + /* First try MSI/X */ + rid = PCIR_BAR(IXGBE_MSIX_BAR); + sc->msix_mem = bus_alloc_resource_any(dev, + SYS_RES_MEMORY, &rid, RF_ACTIVE); + if (!sc->msix_mem) { + /* May not be enabled */ + printf("%s: Unable to map MSIX table \n", ifp->if_xname); + goto msi; + } + + msgs = pci_msix_count(dev); + if (msgs == 0) { /* system has msix disabled */ + bus_release_resource(dev, SYS_RES_MEMORY, + PCIR_BAR(IXGBE_MSIX_BAR), sc->msix_mem); + sc->msix_mem = NULL; + goto msi; + } + + /* Figure out a reasonable auto config value */ + queues = (mp_ncpus > ((msgs-1)/2)) ? (msgs-1)/2 : mp_ncpus; + + if (ixgbe_tx_queues == 0) + ixgbe_tx_queues = queues; + if (ixgbe_rx_queues == 0) + ixgbe_rx_queues = queues; + want = ixgbe_tx_queues + ixgbe_rx_queues + 1; + if (msgs >= want) + msgs = want; + else { + printf("%s: MSIX Configuration Problem, " + "%d vectors but %d queues wanted!\n", ifp->if_xname, + msgs, want); + return (ENXIO); + } + if ((msgs) && pci_alloc_msix(dev, &msgs) == 0) { + printf("%s: Using MSIX interrupts with %d vectors\n", + ifp->if_xname, msgs); + sc->num_tx_queues = ixgbe_tx_queues; + sc->num_rx_queues = ixgbe_rx_queues; + return (msgs); + } +msi: + msgs = pci_msi_count(dev); + if (msgs == 1 && pci_alloc_msi(dev, &msgs) == 0) + printf("%s: Using MSI interrupt\n", ifp->if_xname); + return (msgs); + /* MSI is not supported yet */ + return (0); +} +#endif + +int +ixgbe_allocate_pci_resources(struct ix_softc *sc) +{ + struct ixgbe_osdep *os = &sc->osdep; + struct pci_attach_args *pa = os->os_pa; + int val, i; + + val = pci_conf_read(pa->pa_pc, pa->pa_tag, PCIR_BAR(0)); + if (PCI_MAPREG_TYPE(val) != PCI_MAPREG_TYPE_MEM && + PCI_MAPREG_TYPE(val) != PCI_MAPREG_MEM_TYPE_64BIT) { + printf(": mmba is not mem space\n"); + return (ENXIO); + } + + if (pci_mapreg_map(pa, PCIR_BAR(0), PCI_MAPREG_MEM_TYPE(val), 0, + &os->os_memt, &os->os_memh, &os->os_membase, &os->os_memsize, 0)) { + printf(": cannot find mem space\n"); + return (ENXIO); + } + sc->hw.hw_addr = (u8 *)os->os_membase; + + /* + * Init the resource arrays + */ + for (i = 0; i < IXGBE_MSGS; i++) { + sc->rid[i] = i + 1; /* MSI/X RID starts at 1 */ + sc->tag[i] = NULL; + sc->res[i] = NULL; + } + + /* Legacy defaults */ + sc->num_tx_queues = 1; + sc->num_rx_queues = 1; + + /* Now setup MSI or MSI/X */ +#if 0 + sc->msix = ixgbe_setup_msix(sc); +#endif + sc->hw.back = os; + + return (0); +} + +void +ixgbe_free_pci_resources(struct ix_softc * sc) +{ + struct ixgbe_osdep *os = &sc->osdep; + struct pci_attach_args *pa = os->os_pa; + +#if 0 + int i; + + /* + * Legacy has this set to 0, but we need + * to run this once, so reset it. + */ + if (sc->msix == 0) + sc->msix = 1; + + /* + * First release all the interrupt resources: + * notice that since these are just kept + * in an array we can do the same logic + * whether its MSIX or just legacy. + */ + for (i = 0; i < sc->msix; i++) { + if (sc->tag[i] != NULL) { + pci_intr_disestablish(pa->pa_pc, sc->tag[0]); + sc->tag[i] = NULL; + } + if (sc->res[i] != NULL) { + bus_release_resource(dev, SYS_RES_IRQ, + sc->rid[i], sc->res[i]); + } + } + + if (sc->msix) + pci_release_msi(dev); + + if (sc->msix_mem != NULL) + bus_release_resource(dev, SYS_RES_MEMORY, + PCIR_BAR(IXGBE_MSIX_BAR), sc->msix_mem); +#else + pci_intr_disestablish(pa->pa_pc, sc->tag[0]); + sc->tag[0] = NULL; +#endif + + if (os->os_membase != NULL) + bus_space_unmap(os->os_memt, os->os_memh, os->os_memsize); + os->os_membase = 0; + + return; +} + +/********************************************************************* + * + * Initialize the hardware to a configuration as specified by the + * sc structure. The controller is reset, the EEPROM is + * verified, the MAC address is set, then the shared initialization + * routines are called. + * + **********************************************************************/ +int +ixgbe_hardware_init(struct ix_softc *sc) +{ + struct ifnet *ifp = &sc->arpcom.ac_if; + u16 csum; + + csum = 0; + /* Issue a global reset */ + sc->hw.adapter_stopped = FALSE; + ixgbe_hw(&sc->hw, stop_adapter); + + /* Make sure we have a good EEPROM before we read from it */ + if (ixgbe_ee(&sc->hw, validate_checksum, &csum) < 0) { + printf("%s: The EEPROM Checksum Is Not Valid\n", ifp->if_xname); + return (EIO); + } + + /* Get Hardware Flow Control setting */ + sc->hw.fc.type = ixgbe_fc_full; + sc->hw.fc.pause_time = IXGBE_FC_PAUSE; + sc->hw.fc.low_water = IXGBE_FC_LO; + sc->hw.fc.high_water = IXGBE_FC_HI; + sc->hw.fc.send_xon = TRUE; + + if (ixgbe_hw(&sc->hw, init_hw) != 0) { + printf("%s: Hardware Initialization Failed"); + return (EIO); + } + bcopy(sc->hw.mac.addr, sc->arpcom.ac_enaddr, + IXGBE_ETH_LENGTH_OF_ADDRESS); + + return (0); +} + +/********************************************************************* + * + * Setup networking device structure and register an interface. + * + **********************************************************************/ +void +ixgbe_setup_interface(struct ix_softc *sc) +{ + struct ixgbe_hw *hw = &sc->hw; + struct ifnet *ifp = &sc->arpcom.ac_if; + INIT_DEBUGOUT("ixgbe_setup_interface: begin"); + + strlcpy(ifp->if_xname, sc->dev.dv_xname, IFNAMSIZ); + ifp->if_baudrate = IF_Gbps(10); +#if 0 + ifp->if_init = ixgbe_init; +#endif + ifp->if_softc = sc; + ifp->if_flags = IFF_BROADCAST | IFF_SIMPLEX | IFF_MULTICAST; + ifp->if_ioctl = ixgbe_ioctl; + ifp->if_start = ixgbe_start; + ifp->if_timer = 0; + ifp->if_watchdog = ixgbe_watchdog; + ifp->if_hardmtu = IXGBE_MAX_FRAME_SIZE - + ETHER_HDR_LEN - ETHER_CRC_LEN; + IFQ_SET_MAXLEN(&ifp->if_snd, sc->num_tx_desc - 1); + IFQ_SET_READY(&ifp->if_snd); + + ifp->if_capabilities |= IFCAP_VLAN_MTU; +#ifdef IX_CSUM_OFFLOAD + ifp->if_capabilities |= IFCAP_VLAN_HWTAGGING; + ifp->if_capabilities |= IFCAP_CSUM_IPv4; + ifp->if_capabilities |= IFCAP_CSUM_TCPv4 | IFCAP_CSUM_UDPv4; +#endif + + sc->max_frame_size = + ifp->if_mtu + ETHER_HDR_LEN + ETHER_CRC_LEN; + + if ((hw->device_id == PCI_PRODUCT_INTEL_82598AT) || + (hw->device_id == PCI_PRODUCT_INTEL_82598AT_DUAL)) + ixgbe_hw(hw, setup_link_speed, + IXGBE_LINK_SPEED_10GB_FULL | + IXGBE_LINK_SPEED_1GB_FULL, TRUE, TRUE); + else + ixgbe_hw(hw, setup_link_speed, + IXGBE_LINK_SPEED_10GB_FULL, + TRUE, FALSE); + + /* + * Specify the media types supported by this sc and register + * callbacks to update media and link information + */ + ifmedia_init(&sc->media, IFM_IMASK, ixgbe_media_change, + ixgbe_media_status); + ifmedia_add(&sc->media, IFM_ETHER | sc->optics | + IFM_FDX, 0, NULL); + if ((hw->device_id == PCI_PRODUCT_INTEL_82598AT) || + (hw->device_id == PCI_PRODUCT_INTEL_82598AT_DUAL)) { + ifmedia_add(&sc->media, + IFM_ETHER | IFM_1000_T | IFM_FDX, 0, NULL); + ifmedia_add(&sc->media, + IFM_ETHER | IFM_1000_T, 0, NULL); + } + ifmedia_add(&sc->media, IFM_ETHER | IFM_AUTO, 0, NULL); + ifmedia_set(&sc->media, IFM_ETHER | IFM_AUTO); + + if_attach(ifp); + ether_ifattach(ifp); + + + return; +} + +int +ixgbe_dma_malloc(struct ix_softc *sc, bus_size_t size, + struct ixgbe_dma_alloc *dma, int mapflags) +{ + struct ifnet *ifp = &sc->arpcom.ac_if; + struct ixgbe_osdep *os = &sc->osdep; + int r; + + dma->dma_tag = os->os_pa->pa_dmat; + r = bus_dmamap_create(dma->dma_tag, size, 1, size, 0, + BUS_DMA_NOWAIT, &dma->dma_map); + if (r != 0) { + printf("%s: ixgbe_dma_malloc: bus_dma_tag_create failed; " + "error %u\n", ifp->if_xname, r); + goto fail_0; + } + + r = bus_dmamem_alloc(dma->dma_tag, size, PAGE_SIZE, 0, + &dma->dma_seg, 1, &dma->dma_nseg, BUS_DMA_NOWAIT); + if (r != 0) { + printf("%s: ixgbe_dma_malloc: bus_dmamem_alloc failed; " + "error %u\n", ifp->if_xname, r); + goto fail_1; + } + r = bus_dmamem_map(dma->dma_tag, &dma->dma_seg, dma->dma_nseg, size, + &dma->dma_vaddr, BUS_DMA_NOWAIT); + if (r != 0) { + printf("%s: ixgbe_dma_malloc: bus_dmamem_map failed; " + "error %u\n", ifp->if_xname, r); + goto fail_2; + } + r = bus_dmamap_load(dma->dma_tag, dma->dma_map, + dma->dma_vaddr, size, NULL, + mapflags | BUS_DMA_NOWAIT); + if (r != 0) { + printf("%s: ixgbe_dma_malloc: bus_dmamap_load failed; " + "error %u\n", ifp->if_xname, r); + goto fail_3; + } + dma->dma_size = size; + return (0); +fail_3: + bus_dmamem_unmap(dma->dma_tag, dma->dma_vaddr, size); +fail_2: + bus_dmamem_free(dma->dma_tag, &dma->dma_seg, dma->dma_nseg); +fail_1: + bus_dmamap_destroy(dma->dma_tag, dma->dma_map); +fail_0: + dma->dma_map = NULL; + dma->dma_tag = NULL; + return (r); +} + +void +ixgbe_dma_free(struct ix_softc *sc, struct ixgbe_dma_alloc *dma) +{ + if (dma->dma_tag == NULL) + return; + + if (dma->dma_map != NULL) { + bus_dmamap_sync(dma->dma_tag, dma->dma_map, 0, + dma->dma_map->dm_mapsize, + BUS_DMASYNC_POSTREAD | BUS_DMASYNC_POSTWRITE); + bus_dmamap_unload(dma->dma_tag, dma->dma_map); + bus_dmamem_unmap(dma->dma_tag, dma->dma_vaddr, dma->dma_size); + bus_dmamem_free(dma->dma_tag, &dma->dma_seg, dma->dma_nseg); + bus_dmamap_destroy(dma->dma_tag, dma->dma_map); + } +} + + +/********************************************************************* + * + * Allocate memory for the transmit and receive rings, and then + * the descriptors associated with each, called only once at attach. + * + **********************************************************************/ +int +ixgbe_allocate_queues(struct ix_softc *sc) +{ + struct ifnet *ifp = &sc->arpcom.ac_if; + struct tx_ring *txr; + struct rx_ring *rxr; + int rsize, tsize, error = IXGBE_SUCCESS; + int txconf = 0, rxconf = 0, i; + + /* First allocate the TX ring struct memory */ + if (!(sc->tx_rings = + (struct tx_ring *) malloc(sizeof(struct tx_ring) * + sc->num_tx_queues, M_DEVBUF, M_NOWAIT | M_ZERO))) { + printf("%s: Unable to allocate TX ring memory\n", ifp->if_xname); + error = ENOMEM; + goto fail; + } + txr = sc->tx_rings; + + /* Next allocate the RX */ + if (!(sc->rx_rings = + (struct rx_ring *) malloc(sizeof(struct rx_ring) * + sc->num_rx_queues, M_DEVBUF, M_NOWAIT | M_ZERO))) { + printf("%s: Unable to allocate RX ring memory\n", ifp->if_xname); + error = ENOMEM; + goto rx_fail; + } + rxr = sc->rx_rings; + + /* For the ring itself */ + tsize = roundup2(sc->num_tx_desc * + sizeof(union ixgbe_adv_tx_desc), 4096); + + /* + * Now set up the TX queues, txconf is needed to handle the + * possibility that things fail midcourse and we need to + * undo memory gracefully + */ + for (i = 0; i < sc->num_tx_queues; i++, txconf++) { + /* Set up some basics */ + txr = &sc->tx_rings[i]; + txr->sc = sc; + txr->me = i; + + /* Initialize the TX side lock */ + mtx_init(&txr->tx_mtx, IPL_NET); + + if (ixgbe_dma_malloc(sc, tsize, + &txr->txdma, BUS_DMA_NOWAIT)) { + printf("%s: Unable to allocate TX Descriptor memory\n", + ifp->if_xname); + error = ENOMEM; + goto err_tx_desc; + } + txr->tx_base = (union ixgbe_adv_tx_desc *)txr->txdma.dma_vaddr; + bzero((void *)txr->tx_base, tsize); + + if (ixgbe_dma_malloc(sc, sizeof(u_int32_t), + &txr->txwbdma, BUS_DMA_NOWAIT)) { + printf("%s: Unable to allocate TX Write Back memory\n", + ifp->if_xname); + error = ENOMEM; + goto err_tx_desc; + } + txr->tx_hwb = (u_int32_t *)txr->txwbdma.dma_vaddr; + *txr->tx_hwb = 0; + + /* Now allocate transmit buffers for the ring */ + if (ixgbe_allocate_transmit_buffers(txr)) { + printf("%s: Critical Failure setting up transmit buffers\n", + ifp->if_xname); + error = ENOMEM; + goto err_tx_desc; + } + + } + + /* + * Next the RX queues... + */ + rsize = roundup2(sc->num_rx_desc * + sizeof(union ixgbe_adv_rx_desc), 4096); + for (i = 0; i < sc->num_rx_queues; i++, rxconf++) { + rxr = &sc->rx_rings[i]; + /* Set up some basics */ + rxr->sc = sc; + rxr->me = i; + + /* Initialize the TX side lock */ + mtx_init(&rxr->rx_mtx, IPL_NET); + + if (ixgbe_dma_malloc(sc, rsize, + &rxr->rxdma, BUS_DMA_NOWAIT)) { + printf("%s: Unable to allocate RxDescriptor memory\n", + ifp->if_xname); + error = ENOMEM; + goto err_rx_desc; + } + rxr->rx_base = (union ixgbe_adv_rx_desc *)rxr->rxdma.dma_vaddr; + bzero((void *)rxr->rx_base, rsize); + + /* Allocate receive buffers for the ring*/ + if (ixgbe_allocate_receive_buffers(rxr)) { + printf("%s: Critical Failure setting up receive buffers\n", + ifp->if_xname); + error = ENOMEM; + goto err_rx_desc; + } + } + + return (0); + +err_rx_desc: + for (rxr = sc->rx_rings; rxconf > 0; rxr++, rxconf--) + ixgbe_dma_free(sc, &rxr->rxdma); +err_tx_desc: + for (txr = sc->tx_rings; txconf > 0; txr++, txconf--) { + ixgbe_dma_free(sc, &txr->txdma); + ixgbe_dma_free(sc, &txr->txwbdma); + } + free(sc->rx_rings, M_DEVBUF); +rx_fail: + free(sc->tx_rings, M_DEVBUF); +fail: + return (error); +} + +/********************************************************************* + * + * Allocate memory for tx_buffer structures. The tx_buffer stores all + * the information needed to transmit a packet on the wire. This is + * called only once at attach, setup is done every reset. + * + **********************************************************************/ +int +ixgbe_allocate_transmit_buffers(struct tx_ring *txr) +{ + struct ix_softc *sc = txr->sc; + struct ixgbe_osdep *os = &sc->osdep; + struct ifnet *ifp = &sc->arpcom.ac_if; + struct ixgbe_tx_buf *txbuf; + int error, i; + + txr->txtag = os->os_pa->pa_dmat; + if (!(txr->tx_buffers = + (struct ixgbe_tx_buf *) malloc(sizeof(struct ixgbe_tx_buf) * + sc->num_tx_desc, M_DEVBUF, M_NOWAIT | M_ZERO))) { + printf("%s: Unable to allocate tx_buffer memory\n", ifp->if_xname); + error = ENOMEM; + goto fail; + } + + /* Create the descriptor buffer dma maps */ + txbuf = txr->tx_buffers; + for (i = 0; i < sc->num_tx_desc; i++, txbuf++) { + error = bus_dmamap_create(txr->txtag, IXGBE_TSO_SIZE, + IXGBE_MAX_SCATTER, PAGE_SIZE, 0, + BUS_DMA_NOWAIT, &txbuf->map); + if (error != 0) { + printf("%s: Unable to create TX DMA map\n", ifp->if_xname); + goto fail; + } + } + + return 0; +fail: + /* We free all, it handles case where we are in the middle */ + ixgbe_free_transmit_structures(sc); + return (error); +} + +/********************************************************************* + * + * Initialize a transmit ring. + * + **********************************************************************/ +void +ixgbe_setup_transmit_ring(struct tx_ring *txr) +{ + struct ix_softc *sc = txr->sc; + struct ixgbe_tx_buf *txbuf; + int i; + + /* Clear the old ring contents */ + bzero((void *)txr->tx_base, + (sizeof(union ixgbe_adv_tx_desc)) * sc->num_tx_desc); + /* Reset indices */ + txr->next_avail_tx_desc = 0; + txr->next_tx_to_clean = 0; + + /* Free any existing tx buffers. */ + txbuf = txr->tx_buffers; + for (i = 0; i < sc->num_tx_desc; i++, txbuf++) { + if (txbuf->m_head != NULL) { + bus_dmamap_sync(txr->txtag, txbuf->map, + 0, txbuf->map->dm_mapsize, + BUS_DMASYNC_POSTWRITE); + bus_dmamap_unload(txr->txtag, txbuf->map); + m_freem(txbuf->m_head); + } + txbuf->m_head = NULL; + } + + /* Set number of descriptors available */ + txr->tx_avail = sc->num_tx_desc; + + bus_dmamap_sync(txr->txdma.dma_tag, txr->txdma.dma_map, + 0, txr->txdma.dma_map->dm_mapsize, + BUS_DMASYNC_PREREAD | BUS_DMASYNC_PREWRITE); + +} + +/********************************************************************* + * + * Initialize all transmit rings. + * + **********************************************************************/ +int +ixgbe_setup_transmit_structures(struct ix_softc *sc) +{ + struct tx_ring *txr = sc->tx_rings; + int i; + + for (i = 0; i < sc->num_tx_queues; i++, txr++) + ixgbe_setup_transmit_ring(txr); + + return (0); +} + +/********************************************************************* + * + * Enable transmit unit. + * + **********************************************************************/ +void +ixgbe_initialize_transmit_units(struct ix_softc *sc) +{ + struct ifnet *ifp = &sc->arpcom.ac_if; + struct tx_ring *txr; + struct ixgbe_hw *hw = &sc->hw; + int i; + u64 tdba, txhwb; + u32 txctrl; + + /* Setup the Base and Length of the Tx Descriptor Ring */ + + for (i = 0; i < sc->num_tx_queues; i++) { + txr = &sc->tx_rings[i]; + + /* Setup descriptor base address */ + tdba = txr->txdma.dma_map->dm_segs[0].ds_addr; + IXGBE_WRITE_REG(hw, IXGBE_TDBAL(i), + (tdba & 0x00000000ffffffffULL)); + IXGBE_WRITE_REG(hw, IXGBE_TDBAH(i), (tdba >> 32)); + IXGBE_WRITE_REG(hw, IXGBE_TDLEN(i), + sc->num_tx_desc * sizeof(struct ixgbe_legacy_tx_desc)); + + /* Setup for Head WriteBack */ + txhwb = txr->txwbdma.dma_map->dm_segs[0].ds_addr; + txhwb |= IXGBE_TDWBAL_HEAD_WB_ENABLE; + IXGBE_WRITE_REG(hw, IXGBE_TDWBAL(i), + (txhwb & 0x00000000ffffffffULL)); + IXGBE_WRITE_REG(hw, IXGBE_TDWBAH(i), + (txhwb >> 32)); + txctrl = IXGBE_READ_REG(hw, IXGBE_DCA_TXCTRL(i)); + txctrl &= ~IXGBE_DCA_TXCTRL_TX_WB_RO_EN; + IXGBE_WRITE_REG(hw, IXGBE_DCA_TXCTRL(i), txctrl); + + /* Setup the HW Tx Head and Tail descriptor pointers */ + IXGBE_WRITE_REG(hw, IXGBE_TDH(i), 0); + IXGBE_WRITE_REG(hw, IXGBE_TDT(i), 0); + + /* Setup Transmit Descriptor Cmd Settings */ + txr->txd_cmd = IXGBE_TXD_CMD_IFCS; + + txr->watchdog_timer = 0; + } + ifp->if_timer = 0; + + return; +} + +/********************************************************************* + * + * Free all transmit rings. + * + **********************************************************************/ +void +ixgbe_free_transmit_structures(struct ix_softc *sc) +{ + struct tx_ring *txr = sc->tx_rings; + int i; + + for (i = 0; i < sc->num_tx_queues; i++, txr++) { + ixgbe_free_transmit_buffers(txr); + ixgbe_dma_free(sc, &txr->txdma); + ixgbe_dma_free(sc, &txr->txwbdma); + } + free(sc->tx_rings, M_DEVBUF); +} + +/********************************************************************* + * + * Free transmit ring related data structures. + * + **********************************************************************/ +void +ixgbe_free_transmit_buffers(struct tx_ring *txr) +{ + struct ix_softc *sc = txr->sc; + struct ixgbe_tx_buf *tx_buffer; + int i; + + INIT_DEBUGOUT("free_transmit_ring: begin"); + + if (txr->tx_buffers == NULL) + return; + + tx_buffer = txr->tx_buffers; + for (i = 0; i < sc->num_tx_desc; i++, tx_buffer++) { + if (tx_buffer->m_head != NULL) { + bus_dmamap_sync(txr->txtag, tx_buffer->map, + 0, tx_buffer->map->dm_mapsize, + BUS_DMASYNC_POSTWRITE); + bus_dmamap_unload(txr->txtag, + tx_buffer->map); + m_freem(tx_buffer->m_head); + if (tx_buffer->map != NULL) { + bus_dmamap_destroy(txr->txtag, + tx_buffer->map); + } + } else if (tx_buffer->map != NULL) { + bus_dmamap_unload(txr->txtag, + tx_buffer->map); + bus_dmamap_destroy(txr->txtag, + tx_buffer->map); + } + tx_buffer->m_head = NULL; + tx_buffer->map = NULL; + } + + if (txr->tx_buffers != NULL) + free(txr->tx_buffers, M_DEVBUF); + txr->tx_buffers = NULL; +#if 0 + if (txr->txtag != NULL) + bus_dma_tag_destroy(txr->txtag); +#endif + txr->txtag = NULL; + return; +} + +#ifdef IX_CSUM_OFFLOAD +/********************************************************************* + * + * Advanced Context Descriptor setup for VLAN or CSUM + * + **********************************************************************/ + +boolean_t +ixgbe_tx_ctx_setup(struct tx_ring *txr, struct mbuf *mp) +{ + struct ix_softc *sc = txr->sc; + struct ifnet *ifp = &sc->arpcom.ac_if; + struct ixgbe_adv_tx_context_desc *TXD; + struct ixgbe_tx_buf *tx_buffer; + u32 vlan_macip_lens = 0, type_tucmd_mlhl = 0; + struct ip *ip; + struct ip6_hdr *ip6; + int ehdrlen, ip_hlen = 0; + u16 etype; + u8 ipproto = 0; + bool offload = TRUE; + int ctxd = txr->next_avail_tx_desc; +#if NVLAN > 0 + struct ether_vlan_header *eh; + struct ifvlan *ifv = NULL; + + if ((mp->m_flags & (M_PROTO1|M_PKTHDR)) == (M_PROTO1|M_PKTHDR) && + mp->m_pkthdr.rcvif != NULL) + ifv = mp->m_pkthdr.rcvif->if_softc; +#else + struct ether_header *eh; +#endif + + if ((ifp->if_capabilities & IFCAP_CSUM_IPv4) == 0) + offload = FALSE; + + tx_buffer = &txr->tx_buffers[ctxd]; + TXD = (struct ixgbe_adv_tx_context_desc *) &txr->tx_base[ctxd]; + + /* + * In advanced descriptors the vlan tag must + * be placed into the descriptor itself. + */ +#if NVLAN > 0 + if (ifv != NULL) { + vlan_macip_lens |= + htole16(ifv->ifv_tag) << IXGBE_ADVTXD_VLAN_SHIFT; + } else +#endif + if (offload == FALSE) + return FALSE; /* No need for CTX */ + + /* + * Determine where frame payload starts. + * Jump over vlan headers if already present, + * helpful for QinQ too. + */ +#if NVLAN > 0 + eh = mtod(mp, struct ether_vlan_header *); + if (eh->evl_encap_proto == htons(ETHERTYPE_VLAN)) { + etype = ntohs(eh->evl_proto); + ehdrlen = ETHER_HDR_LEN + ETHER_VLAN_ENCAP_LEN; + } else { + etype = ntohs(eh->evl_encap_proto); + ehdrlen = ETHER_HDR_LEN; + } +#else + eh = mtod(mp, struct ether_header *); + etype = ntohs(eh->ether_type); + ehdrlen = ETHER_HDR_LEN; +#endif + + /* Set the ether header length */ + vlan_macip_lens |= ehdrlen << IXGBE_ADVTXD_MACLEN_SHIFT; + + switch (etype) { + case ETHERTYPE_IP: + ip = (struct ip *)(mp->m_data + ehdrlen); + ip_hlen = ip->ip_hl << 2; + if (mp->m_len < ehdrlen + ip_hlen) + return FALSE; /* failure */ + ipproto = ip->ip_p; + if (mp->m_pkthdr.csum_flags & M_IPV4_CSUM_OUT) + type_tucmd_mlhl |= IXGBE_ADVTXD_TUCMD_IPV4; + break; + case ETHERTYPE_IPV6: + ip6 = (struct ip6_hdr *)(mp->m_data + ehdrlen); + ip_hlen = sizeof(struct ip6_hdr); + if (mp->m_len < ehdrlen + ip_hlen) + return FALSE; /* failure */ + ipproto = ip6->ip6_nxt; + if (mp->m_pkthdr.csum_flags & M_IPV4_CSUM_OUT) + type_tucmd_mlhl |= IXGBE_ADVTXD_TUCMD_IPV6; + break; + default: + offload = FALSE; + break; + } + + vlan_macip_lens |= ip_hlen; + type_tucmd_mlhl |= IXGBE_ADVTXD_DCMD_DEXT | IXGBE_ADVTXD_DTYP_CTXT; + + switch (ipproto) { + case IPPROTO_TCP: + if (mp->m_pkthdr.csum_flags & M_TCPV4_CSUM_OUT) + type_tucmd_mlhl |= IXGBE_ADVTXD_TUCMD_L4T_TCP; + break; + case IPPROTO_UDP: + if (mp->m_pkthdr.csum_flags & M_UDPV4_CSUM_OUT) + type_tucmd_mlhl |= IXGBE_ADVTXD_TUCMD_L4T_UDP; + break; + default: + offload = FALSE; + break; + } + + /* Now copy bits into descriptor */ + TXD->vlan_macip_lens |= htole32(vlan_macip_lens); + TXD->type_tucmd_mlhl |= htole32(type_tucmd_mlhl); + TXD->seqnum_seed = htole32(0); + TXD->mss_l4len_idx = htole32(0); + +#ifndef NO_82598_A0_SUPPORT + if (sc->hw.revision_id == 0) + desc_flip(TXD); +#endif + + tx_buffer->m_head = NULL; + + /* We've consumed the first desc, adjust counters */ + if (++ctxd == sc->num_tx_desc) + ctxd = 0; + txr->next_avail_tx_desc = ctxd; + --txr->tx_avail; + + return (offload); +} + +#if 0 +/********************************************************************** + * + * Setup work for hardware segmentation offload (TSO) on + * scs using advanced tx descriptors + * + **********************************************************************/ +boolean_t +ixgbe_tso_setup(struct tx_ring *txr, struct mbuf *mp, u32 *paylen) +{ + struct ix_softc *sc = txr->sc; + struct ixgbe_adv_tx_context_desc *TXD; + struct ixgbe_tx_buf *tx_buffer; + u32 vlan_macip_lens = 0, type_tucmd_mlhl = 0; + u32 mss_l4len_idx = 0; + int ctxd, ehdrlen, hdrlen, ip_hlen, tcp_hlen; +#if NVLAN > 0 + u16 vtag = 0; + struct ether_vlan_header *eh; + + struct ifvlan *ifv = NULL; + + if ((mp->m_flags & (M_PROTO1|M_PKTHDR)) == (M_PROTO1|M_PKTHDR) && + mp->m_pkthdr.rcvif != NULL) + ifv = mp->m_pkthdr.rcvif->if_softc; +#else + struct ether_header *eh; +#endif + struct ip *ip; + struct tcphdr *th; + + if (((mp->m_pkthdr.csum_flags & CSUM_TSO) == 0) || + (mp->m_pkthdr.len <= IXGBE_TX_BUFFER_SIZE)) + return FALSE; + + /* + * Determine where frame payload starts. + * Jump over vlan headers if already present + */ +#if NVLAN > 0 + eh = mtod(mp, struct ether_vlan_header *); + if (eh->evl_encap_proto == htons(ETHERTYPE_VLAN)) + ehdrlen = ETHER_HDR_LEN + ETHER_VLAN_ENCAP_LEN; + else + ehdrlen = ETHER_HDR_LEN; +#else + eh = mtod(mp, struct ether_header *); + ehdrlen = ETHER_HDR_LEN; +#endif + + /* Ensure we have at least the IP+TCP header in the first mbuf. */ + if (mp->m_len < ehdrlen + sizeof(struct ip) + sizeof(struct tcphdr)) + return FALSE; + + ctxd = txr->next_avail_tx_desc; + tx_buffer = &txr->tx_buffers[ctxd]; + TXD = (struct ixgbe_adv_tx_context_desc *) &txr->tx_base[ctxd]; + + ip = (struct ip *)(mp->m_data + ehdrlen); + if (ip->ip_p != IPPROTO_TCP) + return FALSE; /* 0 */ + ip->ip_len = 0; + ip->ip_sum = 0; + ip_hlen = ip->ip_hl << 2; + th = (struct tcphdr *)((caddr_t)ip + ip_hlen); + th->th_sum = in_pseudo(ip->ip_src.s_addr, + ip->ip_dst.s_addr, htons(IPPROTO_TCP)); + tcp_hlen = th->th_off << 2; + hdrlen = ehdrlen + ip_hlen + tcp_hlen; + /* This is used in the transmit desc in encap */ + *paylen = mp->m_pkthdr.len - hdrlen; + +#if NVLAN > 0 + /* VLAN MACLEN IPLEN */ + if (ifv != NULL) { + vtag = htole16(mp->m_pkthdr.ether_vtag); + vlan_macip_lens |= (ifv->ifv_tag << IXGBE_ADVTXD_VLAN_SHIFT); + } +#endif + + vlan_macip_lens |= ehdrlen << IXGBE_ADVTXD_MACLEN_SHIFT; + vlan_macip_lens |= ip_hlen; + TXD->vlan_macip_lens |= htole32(vlan_macip_lens); + + /* ADV DTYPE TUCMD */ + type_tucmd_mlhl |= IXGBE_ADVTXD_DCMD_DEXT | IXGBE_ADVTXD_DTYP_CTXT; + type_tucmd_mlhl |= IXGBE_ADVTXD_TUCMD_L4T_TCP; + type_tucmd_mlhl |= IXGBE_ADVTXD_TUCMD_IPV4; + TXD->type_tucmd_mlhl |= htole32(type_tucmd_mlhl); + + + /* MSS L4LEN IDX */ + mss_l4len_idx |= (mp->m_pkthdr.tso_segsz << IXGBE_ADVTXD_MSS_SHIFT); + mss_l4len_idx |= (tcp_hlen << IXGBE_ADVTXD_L4LEN_SHIFT); + TXD->mss_l4len_idx = htole32(mss_l4len_idx); + + TXD->seqnum_seed = htole32(0); + tx_buffer->m_head = NULL; + +#ifndef NO_82598_A0_SUPPORT + if (sc->hw.revision_id == 0) + desc_flip(TXD); +#endif + + if (++ctxd == sc->num_tx_desc) + ctxd = 0; + + txr->tx_avail--; + txr->next_avail_tx_desc = ctxd; + return TRUE; +} + +#else /* For 6.2 RELEASE */ +/* This makes it easy to keep the code common */ +boolean_t +ixgbe_tso_setup(struct tx_ring *txr, struct mbuf *mp, u32 *paylen) +{ + return (FALSE); +} +#endif +#endif + +/********************************************************************** + * + * Examine each tx_buffer in the used queue. If the hardware is done + * processing the packet then free associated resources. The + * tx_buffer is put back on the free queue. + * + **********************************************************************/ +boolean_t +ixgbe_txeof(struct tx_ring *txr) +{ + struct ix_softc *sc = txr->sc; + struct ifnet *ifp = &sc->arpcom.ac_if; + u_int first, last, done, num_avail; + struct ixgbe_tx_buf *tx_buffer; + struct ixgbe_legacy_tx_desc *tx_desc; + + if (txr->tx_avail == sc->num_tx_desc) + return FALSE; + + num_avail = txr->tx_avail; + first = txr->next_tx_to_clean; + + tx_buffer = &txr->tx_buffers[first]; + + /* For cleanup we just use legacy struct */ + tx_desc = (struct ixgbe_legacy_tx_desc *)&txr->tx_base[first]; + + /* Get the HWB */ + bus_dmamap_sync(txr->txwbdma.dma_tag, txr->txwbdma.dma_map, + 0, txr->txwbdma.dma_map->dm_mapsize, + BUS_DMASYNC_POSTREAD); + done = *txr->tx_hwb; + + bus_dmamap_sync(txr->txdma.dma_tag, txr->txdma.dma_map, + 0, txr->txdma.dma_map->dm_mapsize, + BUS_DMASYNC_POSTREAD); + + while (TRUE) { + /* We clean the range til last head write back */ + while (first != done) { + tx_desc->upper.data = 0; + tx_desc->lower.data = 0; + tx_desc->buffer_addr = 0; + num_avail++; + + if (tx_buffer->m_head) { + ifp->if_opackets++; + bus_dmamap_sync(txr->txtag, + tx_buffer->map, + 0, tx_buffer->map->dm_mapsize, + BUS_DMASYNC_POSTWRITE); + bus_dmamap_unload(txr->txtag, + tx_buffer->map); + m_freem(tx_buffer->m_head); + tx_buffer->m_head = NULL; + } + + if (++first == sc->num_tx_desc) + first = 0; + + tx_buffer = &txr->tx_buffers[first]; + tx_desc = (struct ixgbe_legacy_tx_desc *) + &txr->tx_base[first]; + } + /* See if there is more work now */ + last = done; + bus_dmamap_sync(txr->txwbdma.dma_tag, txr->txwbdma.dma_map, + 0, txr->txwbdma.dma_map->dm_mapsize, + BUS_DMASYNC_POSTREAD); + done = *txr->tx_hwb; + if (last == done) + break; + } + + bus_dmamap_sync(txr->txdma.dma_tag, txr->txdma.dma_map, + 0, txr->txdma.dma_map->dm_mapsize, + BUS_DMASYNC_PREREAD | BUS_DMASYNC_PREWRITE); + + txr->next_tx_to_clean = first; + + /* + * If we have enough room, clear IFF_OACTIVE to tell the stack that + * it is OK to send packets. If there are no pending descriptors, + * clear the timeout. Otherwise, if some descriptors have been freed, + * restart the timeout. + */ + if (num_avail > IXGBE_TX_CLEANUP_THRESHOLD) { + ifp->if_flags &= ~IFF_OACTIVE; + + /* If all are clean turn off the timer */ + if (num_avail == sc->num_tx_desc) { + ifp->if_timer = 0; + txr->watchdog_timer = 0; + txr->tx_avail = num_avail; + return FALSE; + } + /* Some were cleaned, so reset timer */ + else if (num_avail != txr->tx_avail) { + ifp->if_timer = IXGBE_TX_TIMEOUT; + txr->watchdog_timer = IXGBE_TX_TIMEOUT; + } + } + + txr->tx_avail = num_avail; + + return TRUE; +} + +/********************************************************************* + * + * Get a buffer from system mbuf buffer pool. + * + **********************************************************************/ +int +ixgbe_get_buf(struct rx_ring *rxr, int i, struct mbuf *nmp) +{ + struct ix_softc *sc = rxr->sc; + struct mbuf *mp = nmp; + bus_dmamap_t map; + int error, old, s = 0; + int size = MCLBYTES; + struct ixgbe_rx_buf *rxbuf; + +#if 0 + /* Are we going to Jumbo clusters? */ + if (sc->bigbufs) { + size = MJUMPAGESIZE; + s = 1; + }; + + mp = m_getjcl(M_DONTWAIT, MT_DATA, M_PKTHDR, size); + if (mp == NULL) { + sc->mbuf_alloc_failed++; + return (ENOBUFS); + } +#endif + + if (mp == NULL) { + MGETHDR(mp, M_DONTWAIT, MT_DATA); + if (mp == NULL) { + sc->mbuf_alloc_failed++; + return (ENOBUFS); + } + MCLGET(mp, M_DONTWAIT); + if ((mp->m_flags & M_EXT) == 0) { + m_freem(mp); + sc->mbuf_cluster_failed++; + return (ENOBUFS); + } + mp->m_len = mp->m_pkthdr.len = size; + } else { + mp->m_len = mp->m_pkthdr.len = size; + mp->m_data = mp->m_ext.ext_buf; + mp->m_next = NULL; + } + + if (sc->max_frame_size <= (MCLBYTES - ETHER_ALIGN)) + m_adj(mp, ETHER_ALIGN); + + /* + * Using memory from the mbuf cluster pool, invoke the bus_dma + * machinery to arrange the memory mapping. + */ + error = bus_dmamap_load_mbuf(rxr->rxtag[s], rxr->spare_map[s], + mp, BUS_DMA_NOWAIT); + if (error) { + m_freem(mp); + return (error); + } + + /* Now check our target buffer for existing mapping */ + rxbuf = &rxr->rx_buffers[i]; + old = rxbuf->bigbuf; + if (rxbuf->m_head != NULL) + bus_dmamap_unload(rxr->rxtag[old], rxbuf->map[old]); + + map = rxbuf->map[old]; + rxbuf->map[s] = rxr->spare_map[s]; + rxr->spare_map[old] = map; + rxbuf->m_head = mp; + rxbuf->bigbuf = s; + + rxr->rx_base[i].read.pkt_addr = + htole64(rxbuf->map[s]->dm_segs[0].ds_addr); + + bus_dmamap_sync(rxr->rxtag[s], rxbuf->map[s], + 0, rxbuf->map[s]->dm_mapsize, BUS_DMASYNC_PREREAD); + +#ifndef NO_82598_A0_SUPPORT + /* A0 needs to One's Compliment descriptors */ + if (sc->hw.revision_id == 0) { + struct dhack {u32 a1; u32 a2; u32 b1; u32 b2;}; + struct dhack *d; + + d = (struct dhack *)&rxr->rx_base[i]; + d->a1 = ~(d->a1); + d->a2 = ~(d->a2); + } +#endif + + return (0); +} + +/********************************************************************* + * + * Allocate memory for rx_buffer structures. Since we use one + * rx_buffer per received packet, the maximum number of rx_buffer's + * that we'll need is equal to the number of receive descriptors + * that we've allocated. + * + **********************************************************************/ +int +ixgbe_allocate_receive_buffers(struct rx_ring *rxr) +{ + struct ix_softc *sc = rxr->sc; + struct ifnet *ifp = &sc->arpcom.ac_if; + struct ixgbe_osdep *os = &sc->osdep; + struct ixgbe_rx_buf *rxbuf; + int i, bsize, error; + + bsize = sizeof(struct ixgbe_rx_buf) * sc->num_rx_desc; + if (!(rxr->rx_buffers = + (struct ixgbe_rx_buf *) malloc(bsize, + M_DEVBUF, M_NOWAIT | M_ZERO))) { + printf("%s: Unable to allocate rx_buffer memory\n", ifp->if_xname); + error = ENOMEM; + goto fail; + } + rxr->rxtag[0] = rxr->rxtag[1] = os->os_pa->pa_dmat; + + /* Create the spare maps (used by getbuf) */ + error = bus_dmamap_create(rxr->rxtag[0], + MCLBYTES, 1, MCLBYTES, 0, BUS_DMA_NOWAIT, &rxr->spare_map[0]); + if (error) { + printf("%s: %s: bus_dmamap_create failed: %d\n", ifp->if_xname, + __func__, error); + goto fail; + } + error = bus_dmamap_create(rxr->rxtag[1], + MJUMPAGESIZE, 1, MJUMPAGESIZE, 0, BUS_DMA_NOWAIT, &rxr->spare_map[1]); + if (error) { + printf("%s: %s: bus_dmamap_create failed: %d\n", ifp->if_xname, + __func__, error); + goto fail; + } + + for (i = 0; i < sc->num_rx_desc; i++, rxbuf++) { + rxbuf = &rxr->rx_buffers[i]; + error = bus_dmamap_create(rxr->rxtag[0], MCLBYTES, 1, MCLBYTES, + 0, BUS_DMA_NOWAIT, &rxbuf->map[0]); + if (error) { + printf("%s: Unable to create Small RX DMA map\n", ifp->if_xname); + goto fail; + } + error = bus_dmamap_create(rxr->rxtag[1], MJUMPAGESIZE, 1, MJUMPAGESIZE, + 0, BUS_DMA_NOWAIT, &rxbuf->map[1]); + if (error) { + printf("%s: Unable to create Large RX DMA map\n", ifp->if_xname); + goto fail; + } + } + + return (0); + +fail: + /* Frees all, but can handle partial completion */ + ixgbe_free_receive_structures(sc); + return (error); +} + +/********************************************************************* + * + * Initialize a receive ring and its buffers. + * + **********************************************************************/ +int +ixgbe_setup_receive_ring(struct rx_ring *rxr) +{ + struct ix_softc *sc = rxr->sc; + struct ixgbe_rx_buf *rxbuf; +#if 0 + struct ifnet *ifp = &sc->arpcom.ac_if; + struct lro_ctrl *lro = &rxr->lro; +#endif + int j, rsize, s = 0, i; + + rsize = roundup2(sc->num_rx_desc * + sizeof(union ixgbe_adv_rx_desc), 4096); + /* Clear the ring contents */ + bzero((void *)rxr->rx_base, rsize); + + /* + ** Free current RX buffers: the size buffer + ** that is loaded is indicated by the buffer + ** bigbuf value. + */ + for (i = 0; i < sc->num_rx_desc; i++) { + rxbuf = &rxr->rx_buffers[i]; + s = rxbuf->bigbuf; + if (rxbuf->m_head != NULL) { + bus_dmamap_sync(rxr->rxtag[s], rxbuf->map[s], + 0, rxbuf->map[s]->dm_mapsize, + BUS_DMASYNC_POSTREAD); + bus_dmamap_unload(rxr->rxtag[s], rxbuf->map[s]); + m_freem(rxbuf->m_head); + rxbuf->m_head = NULL; + } + } + + for (j = 0; j < sc->num_rx_desc; j++) { + if (ixgbe_get_buf(rxr, j, NULL) == ENOBUFS) { + rxr->rx_buffers[j].m_head = NULL; + rxr->rx_base[j].read.pkt_addr = 0; + /* If we fail some may have change size */ + s = sc->bigbufs; + goto fail; + } + } + + /* Setup our descriptor indices */ + rxr->next_to_check = 0; + rxr->last_cleaned = 0; + + bus_dmamap_sync(rxr->rxdma.dma_tag, rxr->rxdma.dma_map, + 0, rxr->rxdma.dma_map->dm_mapsize, + BUS_DMASYNC_PREREAD | BUS_DMASYNC_PREWRITE); + +#if 0 + /* Now set up the LRO interface */ + if (ixgbe_enable_lro) { + int err = tcp_lro_init(lro); + if (err) { + printf("%s: LRO Initialization failed!\n", ifp->if_xname); + goto fail; + } + printf("%s: RX LRO Initialized\n", ifp->if_xname); + lro->ifp = &sc->arpcom.ac_if; + } +#endif + + return (0); +fail: + /* + * We need to clean up any buffers allocated so far + * 'j' is the failing index, decrement it to get the + * last success. + */ + for (--j; j < 0; j--) { + rxbuf = &rxr->rx_buffers[j]; + if (rxbuf->m_head != NULL) { + bus_dmamap_sync(rxr->rxtag[s], rxbuf->map[s], + 0, rxbuf->map[s]->dm_mapsize, + BUS_DMASYNC_POSTREAD); + bus_dmamap_unload(rxr->rxtag[s], rxbuf->map[s]); + m_freem(rxbuf->m_head); + rxbuf->m_head = NULL; + } + } + return (ENOBUFS); +} + +/********************************************************************* + * + * Initialize all receive rings. + * + **********************************************************************/ +int +ixgbe_setup_receive_structures(struct ix_softc *sc) +{ + struct rx_ring *rxr = sc->rx_rings; + int i, j, s; + + for (i = 0; i < sc->num_rx_queues; i++, rxr++) + if (ixgbe_setup_receive_ring(rxr)) + goto fail; + + return (0); +fail: + /* + * Free RX buffers allocated so far, we will only handle + * the rings that completed, the failing case will have + * cleaned up for itself. The value of 'i' will be the + * failed ring so we must pre-decrement it. + */ + rxr = sc->rx_rings; + for (--i; i > 0; i--, rxr++) { + for (j = 0; j < sc->num_rx_desc; j++) { + struct ixgbe_rx_buf *rxbuf; + rxbuf = &rxr->rx_buffers[j]; + s = rxbuf->bigbuf; + if (rxbuf->m_head != NULL) { + bus_dmamap_sync(rxr->rxtag[s], rxbuf->map[s], + 0, rxbuf->map[s]->dm_mapsize, BUS_DMASYNC_POSTREAD); + bus_dmamap_unload(rxr->rxtag[s], rxbuf->map[s]); + m_freem(rxbuf->m_head); + rxbuf->m_head = NULL; + } + } + } + + return (ENOBUFS); +} + +/********************************************************************* + * + * Enable receive unit. + * + **********************************************************************/ +void +ixgbe_initialize_receive_units(struct ix_softc *sc) +{ + struct rx_ring *rxr = sc->rx_rings; + struct ifnet *ifp = &sc->arpcom.ac_if; + u32 rxctrl, fctrl, srrctl, rxcsum; + u32 reta, mrqc, hlreg, linkvec; + u32 random[10]; + int i; + + + /* + * Make sure receives are disabled while + * setting up the descriptor ring + */ + rxctrl = IXGBE_READ_REG(&sc->hw, IXGBE_RXCTRL); + IXGBE_WRITE_REG(&sc->hw, IXGBE_RXCTRL, + rxctrl & ~IXGBE_RXCTRL_RXEN); + + /* Enable broadcasts */ + fctrl = IXGBE_READ_REG(&sc->hw, IXGBE_FCTRL); + fctrl |= IXGBE_FCTRL_BAM; + IXGBE_WRITE_REG(&sc->hw, IXGBE_FCTRL, fctrl); + + hlreg = IXGBE_READ_REG(&sc->hw, IXGBE_HLREG0); + if (ifp->if_mtu > ETHERMTU) + hlreg |= IXGBE_HLREG0_JUMBOEN; + else + hlreg &= ~IXGBE_HLREG0_JUMBOEN; + IXGBE_WRITE_REG(&sc->hw, IXGBE_HLREG0, hlreg); + + srrctl = IXGBE_READ_REG(&sc->hw, IXGBE_SRRCTL(0)); + srrctl &= ~IXGBE_SRRCTL_BSIZEHDR_MASK; + srrctl &= ~IXGBE_SRRCTL_BSIZEPKT_MASK; + if (sc->bigbufs) + srrctl |= 4096 >> IXGBE_SRRCTL_BSIZEPKT_SHIFT; + else + srrctl |= 2048 >> IXGBE_SRRCTL_BSIZEPKT_SHIFT; + srrctl |= IXGBE_SRRCTL_DESCTYPE_ADV_ONEBUF; + IXGBE_WRITE_REG(&sc->hw, IXGBE_SRRCTL(0), srrctl); + + /* Set Queue moderation rate */ + for (i = 0; i < IXGBE_MSGS; i++) + IXGBE_WRITE_REG(&sc->hw, IXGBE_EITR(i), DEFAULT_ITR); + + /* Set Link moderation lower */ + linkvec = sc->num_tx_queues + sc->num_rx_queues; + IXGBE_WRITE_REG(&sc->hw, IXGBE_EITR(linkvec), LINK_ITR); + + for (i = 0; i < sc->num_rx_queues; i++, rxr++) { + u64 rdba = rxr->rxdma.dma_map->dm_segs[0].ds_addr; + /* Setup the Base and Length of the Rx Descriptor Ring */ + IXGBE_WRITE_REG(&sc->hw, IXGBE_RDBAL(i), + (rdba & 0x00000000ffffffffULL)); + IXGBE_WRITE_REG(&sc->hw, IXGBE_RDBAH(i), (rdba >> 32)); + IXGBE_WRITE_REG(&sc->hw, IXGBE_RDLEN(i), + sc->num_rx_desc * sizeof(union ixgbe_adv_rx_desc)); + + /* Setup the HW Rx Head and Tail Descriptor Pointers */ + IXGBE_WRITE_REG(&sc->hw, IXGBE_RDH(i), 0); + IXGBE_WRITE_REG(&sc->hw, IXGBE_RDT(i), + sc->num_rx_desc - 1); + } + + rxcsum = IXGBE_READ_REG(&sc->hw, IXGBE_RXCSUM); + + if (sc->num_rx_queues > 1) { + /* set up random bits */ + arc4random_bytes(&random, sizeof(random)); + switch (sc->num_rx_queues) { + case 8: + case 4: + reta = 0x00010203; + break; + case 2: + reta = 0x00010001; + break; + default: + reta = 0x00000000; + } + + /* Set up the redirection table */ + for (i = 0; i < 32; i++) { + IXGBE_WRITE_REG(&sc->hw, IXGBE_RETA(i), reta); + if (sc->num_rx_queues > 4) { + ++i; + IXGBE_WRITE_REG(&sc->hw, + IXGBE_RETA(i), 0x04050607); + } + } + + /* Now fill our hash function seeds */ + for (i = 0; i < 10; i++) + IXGBE_WRITE_REG_ARRAY(&sc->hw, + IXGBE_RSSRK(0), i, random[i]); + + mrqc = IXGBE_MRQC_RSSEN + /* Perform hash on these packet types */ + | IXGBE_MRQC_RSS_FIELD_IPV4 + | IXGBE_MRQC_RSS_FIELD_IPV4_TCP + | IXGBE_MRQC_RSS_FIELD_IPV4_UDP + | IXGBE_MRQC_RSS_FIELD_IPV6_EX_TCP + | IXGBE_MRQC_RSS_FIELD_IPV6_EX + | IXGBE_MRQC_RSS_FIELD_IPV6 + | IXGBE_MRQC_RSS_FIELD_IPV6_TCP + | IXGBE_MRQC_RSS_FIELD_IPV6_UDP + | IXGBE_MRQC_RSS_FIELD_IPV6_EX_UDP; + IXGBE_WRITE_REG(&sc->hw, IXGBE_MRQC, mrqc); + + /* RSS and RX IPP Checksum are mutually exclusive */ + rxcsum |= IXGBE_RXCSUM_PCSD; + } + +#if 0 + if (ifp->if_capenable & IFCAP_RXCSUM) + rxcsum |= IXGBE_RXCSUM_PCSD; +#endif + + if (!(rxcsum & IXGBE_RXCSUM_PCSD)) + rxcsum |= IXGBE_RXCSUM_IPPCSE; + + IXGBE_WRITE_REG(&sc->hw, IXGBE_RXCSUM, rxcsum); + + /* Enable Receive engine */ + rxctrl |= (IXGBE_RXCTRL_RXEN | IXGBE_RXCTRL_DMBYPS); + IXGBE_WRITE_REG(&sc->hw, IXGBE_RXCTRL, rxctrl); + + return; +} + +/********************************************************************* + * + * Free all receive rings. + * + **********************************************************************/ +void +ixgbe_free_receive_structures(struct ix_softc *sc) +{ + struct rx_ring *rxr = sc->rx_rings; + int i; + + for (i = 0; i < sc->num_rx_queues; i++, rxr++) { +#if 0 + struct lro_ctrl *lro = &rxr->lro; + /* Free LRO memory */ + tcp_lro_free(lro); +#endif + ixgbe_free_receive_buffers(rxr); + /* Free the ring memory as well */ + ixgbe_dma_free(sc, &rxr->rxdma); + } + + free(sc->rx_rings, M_DEVBUF); +} + +/********************************************************************* + * + * Free receive ring data structures + * + **********************************************************************/ +void +ixgbe_free_receive_buffers(struct rx_ring *rxr) +{ + struct ix_softc *sc = NULL; + struct ixgbe_rx_buf *rxbuf = NULL; + int i, s; + + INIT_DEBUGOUT("free_receive_buffers: begin"); + sc = rxr->sc; + if (rxr->rx_buffers != NULL) { + rxbuf = &rxr->rx_buffers[0]; + for (i = 0; i < sc->num_rx_desc; i++) { + int s = rxbuf->bigbuf; + if (rxbuf->map != NULL) { + bus_dmamap_unload(rxr->rxtag[s], rxbuf->map[s]); + bus_dmamap_destroy(rxr->rxtag[s], rxbuf->map[s]); + } + if (rxbuf->m_head != NULL) { + m_freem(rxbuf->m_head); + } + rxbuf->m_head = NULL; + ++rxbuf; + } + } + if (rxr->rx_buffers != NULL) { + free(rxr->rx_buffers, M_DEVBUF); + rxr->rx_buffers = NULL; + } + for (s = 0; s < 2; s++) { + if (rxr->rxtag[s] != NULL) { +#if 0 + bus_dma_tag_destroy(rxr->rxtag[s]); +#endif + rxr->rxtag[s] = NULL; + } + } + return; +} + +/********************************************************************* + * + * This routine executes in interrupt context. It replenishes + * the mbufs in the descriptor and sends data which has been + * dma'ed into host memory to upper layer. + * + * We loop at most count times if count is > 0, or until done if + * count < 0. + * + *********************************************************************/ +bool +ixgbe_rxeof(struct rx_ring *rxr, int count) +{ + struct ix_softc *sc = rxr->sc; + struct ifnet *ifp = &sc->arpcom.ac_if; +#if 0 + struct lro_ctrl *lro = &rxr->lro; + struct lro_entry *queued; +#endif + struct mbuf *mp; + int len, i, eop = 0; + u8 accept_frame = 0; + u32 staterr; + union ixgbe_adv_rx_desc *cur; + + i = rxr->next_to_check; + cur = &rxr->rx_base[i]; + + bus_dmamap_sync(rxr->rxdma.dma_tag, rxr->rxdma.dma_map, 0, + rxr->rxdma.dma_map->dm_mapsize, BUS_DMASYNC_POSTREAD); + + staterr = cur->wb.upper.status_error; + if (!(staterr & IXGBE_RXD_STAT_DD)) + return FALSE; + + while ((staterr & IXGBE_RXD_STAT_DD) && (count != 0) && + (ifp->if_flags & IFF_RUNNING)) { + struct mbuf *m = NULL; + int s; + + mp = rxr->rx_buffers[i].m_head; + s = rxr->rx_buffers[i].bigbuf; + bus_dmamap_sync(rxr->rxtag[s], rxr->rx_buffers[i].map[s], + 0, rxr->rx_buffers[i].map[s]->dm_mapsize, + BUS_DMASYNC_POSTREAD); + bus_dmamap_unload(rxr->rxtag[s], + rxr->rx_buffers[i].map[s]); + + accept_frame = 1; + if (staterr & IXGBE_RXD_STAT_EOP) { + count--; + eop = 1; + } else { + eop = 0; + } + len = cur->wb.upper.length; + + if (staterr & IXGBE_RXDADV_ERR_FRAME_ERR_MASK) + accept_frame = 0; + + if (accept_frame) { + /* Get a fresh buffer */ + if (ixgbe_get_buf(rxr, i, NULL) != 0) { + ifp->if_iqdrops++; + goto discard; + } + + /* Assign correct length to the current fragment */ + mp->m_len = len; + + if (rxr->fmp == NULL) { + mp->m_pkthdr.len = len; + rxr->fmp = mp; /* Store the first mbuf */ + rxr->lmp = mp; + } else { + /* Chain mbuf's together */ + mp->m_flags &= ~M_PKTHDR; + rxr->lmp->m_next = mp; + rxr->lmp = rxr->lmp->m_next; + rxr->fmp->m_pkthdr.len += len; + } + + if (eop) { + rxr->fmp->m_pkthdr.rcvif = ifp; + ifp->if_ipackets++; + rxr->packet_count++; + rxr->byte_count += rxr->fmp->m_pkthdr.len; + m = rxr->fmp; + + ixgbe_rx_checksum(sc, staterr, rxr->fmp); + +#if NVLAN > 0 && defined(IX_CSUM_OFFLOAD) + if (staterr & IXGBE_RXD_STAT_VP) { + struct ether_vlan_header vh; + + if (m->m_pkthdr.len < ETHER_HDR_LEN) + goto discard; + m_copydata(m, 0, + ETHER_HDR_LEN, (caddr_t)&vh); + vh.evl_proto = vh.evl_encap_proto; + vh.evl_tag = + letoh16(cur->wb.upper.vlan); + vh.evl_encap_proto = + htons(ETHERTYPE_VLAN); + m_adj(m, ETHER_HDR_LEN); + M_PREPEND(m, sizeof(vh), M_DONTWAIT); + if (m == NULL) + goto discard; + m_copyback(m, 0, sizeof(vh), &vh); + } +#endif + rxr->fmp = NULL; + rxr->lmp = NULL; + } + } else { +discard: + ixgbe_get_buf(rxr, i, mp); + if (rxr->fmp != NULL) { + m_freem(rxr->fmp); + rxr->fmp = NULL; + rxr->lmp = NULL; + } + m = NULL; + } + + /* Zero out the receive descriptors status */ + cur->wb.upper.status_error = 0; + bus_dmamap_sync(rxr->rxdma.dma_tag, rxr->rxdma.dma_map, + 0, rxr->rxdma.dma_map->dm_mapsize, + BUS_DMASYNC_PREREAD | BUS_DMASYNC_PREWRITE); + + rxr->last_cleaned = i; /* for updating tail */ + + if (++i == sc->num_rx_desc) + i = 0; + + /* Now send up to the stack */ + if (m != NULL) { + rxr->next_to_check = i; +#ifdef notyet + /* Use LRO if possible */ + if ((!lro->lro_cnt) || (tcp_lro_rx(lro, m, 0))) { +#endif +#if NBPFILTER > 0 + if (ifp->if_bpf) + bpf_mtap(ifp->if_bpf, m, BPF_DIRECTION_IN); +#endif + ether_input_mbuf(ifp, m); + i = rxr->next_to_check; +#ifdef notyet + } +#endif + } + /* Get next descriptor */ + cur = &rxr->rx_base[i]; + staterr = cur->wb.upper.status_error; + } + rxr->next_to_check = i; + + /* Advance the IXGB's Receive Queue "Tail Pointer" */ + IXGBE_WRITE_REG(&sc->hw, IXGBE_RDT(rxr->me), rxr->last_cleaned); + +#if 0 + /* + ** Flush any outstanding LRO work + ** this may call into the stack and + ** must not hold a driver lock. + */ + while(!SLIST_EMPTY(&lro->lro_active)) { + queued = SLIST_FIRST(&lro->lro_active); + SLIST_REMOVE_HEAD(&lro->lro_active, next); + tcp_lro_flush(lro, queued); + } +#endif + + if (!(staterr & IXGBE_RXD_STAT_DD)) + return FALSE; + + return TRUE; +} + +/********************************************************************* + * + * Verify that the hardware indicated that the checksum is valid. + * Inform the stack about the status of checksum so that stack + * doesn't spend time verifying the checksum. + * + *********************************************************************/ +void +ixgbe_rx_checksum(struct ix_softc *sc, + u32 staterr, struct mbuf * mp) +{ + struct ifnet *ifp = &sc->arpcom.ac_if; + u16 status = (u16) staterr; + u8 errors = (u8) (staterr >> 24); + + /* Not offloading */ + if ((ifp->if_capabilities & IFCAP_CSUM_IPv4) == 0) { + mp->m_pkthdr.csum_flags = 0; + return; + } + + // XXX + printf("%s: status 0x%04x errors 0x%02x\n", ifp->if_xname, + status, errors); + + mp->m_pkthdr.csum_flags = 0; + if (status & IXGBE_RXD_STAT_IPCS) { + /* Did it pass? */ + if (!(errors & IXGBE_RXD_ERR_IPE)) + /* IP Checksum Good */ + mp->m_pkthdr.csum_flags = M_IPV4_CSUM_IN_OK; + } + /* Did it pass? */ + if (errors & IXGBE_RXD_ERR_TCPE) + return; + if (status & IXGBE_RXD_STAT_L4CS) + mp->m_pkthdr.csum_flags |= M_TCP_CSUM_IN_OK; + if (status & IXGBE_RXD_STAT_UDPCS) + mp->m_pkthdr.csum_flags |= M_UDP_CSUM_IN_OK; +} + +#if NVLAN > 0 +void +ixgbe_enable_hw_vlans(struct ix_softc *sc) +{ + u32 ctrl; + + ixgbe_disable_intr(sc); + ctrl = IXGBE_READ_REG(&sc->hw, IXGBE_VLNCTRL); + ctrl |= IXGBE_VLNCTRL_VME; + ctrl &= ~IXGBE_VLNCTRL_CFIEN; + IXGBE_WRITE_REG(&sc->hw, IXGBE_VLNCTRL, ctrl); + ixgbe_enable_intr(sc); +} +#endif + +void +ixgbe_enable_intr(struct ix_softc *sc) +{ + struct ixgbe_hw *hw = &sc->hw; + u32 mask = IXGBE_EIMS_ENABLE_MASK; + + /* Enable Fan Failure detection */ + if (hw->phy.media_type == ixgbe_media_type_copper) + mask |= IXGBE_EIMS_GPI_SDP1; + /* With RSS we use auto clear */ + if (sc->msix_mem) { + /* Dont autoclear Link */ + mask &= ~IXGBE_EIMS_OTHER; + mask &= ~IXGBE_EIMS_LSC; + IXGBE_WRITE_REG(&sc->hw, IXGBE_EIAC, + sc->eims_mask | mask); + } + + IXGBE_WRITE_REG(hw, IXGBE_EIMS, mask); + IXGBE_WRITE_FLUSH(hw); + + return; +} + +void +ixgbe_disable_intr(struct ix_softc *sc) +{ + if (sc->msix_mem) + IXGBE_WRITE_REG(&sc->hw, IXGBE_EIAC, 0); + IXGBE_WRITE_REG(&sc->hw, IXGBE_EIMC, ~0); + IXGBE_WRITE_FLUSH(&sc->hw); + return; +} + +u16 +ixgbe_read_pci_cfg(struct ixgbe_hw *hw, u32 reg) +{ + struct pci_attach_args *pa; + u16 value; + + pa = ((struct ixgbe_osdep *)hw->back)->os_pa; + + /* Should we do read/mask/write...? 16 vs 32 bit!!! */ + value = pci_conf_read(pa->pa_pc, pa->pa_tag, reg) & 0xffff; + + return (value); +} + +void +ixgbe_set_ivar(struct ix_softc *sc, u16 entry, u8 vector) +{ + u32 ivar, index; + + vector |= IXGBE_IVAR_ALLOC_VAL; + index = (entry >> 2) & 0x1F; + ivar = IXGBE_READ_REG(&sc->hw, IXGBE_IVAR(index)); + ivar &= ~(0xFF << (8 * (entry & 0x3))); + ivar |= (vector << (8 * (entry & 0x3))); + IXGBE_WRITE_REG(&sc->hw, IXGBE_IVAR(index), ivar); +} + +void +ixgbe_configure_ivars(struct ix_softc *sc) +{ + struct tx_ring *txr = sc->tx_rings; + struct rx_ring *rxr = sc->rx_rings; + int i; + + for (i = 0; i < sc->num_rx_queues; i++, rxr++) { + ixgbe_set_ivar(sc, IXGBE_IVAR_RX_QUEUE(i), rxr->msix); + sc->eims_mask |= rxr->eims; + } + + for (i = 0; i < sc->num_tx_queues; i++, txr++) { + ixgbe_set_ivar(sc, IXGBE_IVAR_TX_QUEUE(i), txr->msix); + sc->eims_mask |= txr->eims; + } + + /* For the Link interrupt */ + ixgbe_set_ivar(sc, IXGBE_IVAR_OTHER_CAUSES_INDEX, + sc->linkvec); + sc->eims_mask |= IXGBE_IVAR_OTHER_CAUSES_INDEX; +} + +/********************************************************************** + * + * Update the board statistics counters. + * + **********************************************************************/ +void +ixgbe_update_stats_counters(struct ix_softc *sc) +{ + struct ifnet *ifp = &sc->arpcom.ac_if;; + struct ixgbe_hw *hw = &sc->hw; + u32 missed_rx = 0, bprc, lxon, lxoff, total; + int i; + + sc->stats.crcerrs += IXGBE_READ_REG(hw, IXGBE_CRCERRS); + + for (i = 0; i < 8; i++) { + int mp; + mp = IXGBE_READ_REG(hw, IXGBE_MPC(i)); + missed_rx += mp; + sc->stats.mpc[i] += mp; + sc->stats.rnbc[i] += IXGBE_READ_REG(hw, IXGBE_RNBC(i)); + } + + /* Hardware workaround, gprc counts missed packets */ + sc->stats.gprc += IXGBE_READ_REG(hw, IXGBE_GPRC); + sc->stats.gprc -= missed_rx; + + sc->stats.gorc += IXGBE_READ_REG(hw, IXGBE_GORCH); + sc->stats.gotc += IXGBE_READ_REG(hw, IXGBE_GOTCH); + sc->stats.tor += IXGBE_READ_REG(hw, IXGBE_TORH); + + /* + * Workaround: mprc hardware is incorrectly counting + * broadcasts, so for now we subtract those. + */ + bprc = IXGBE_READ_REG(hw, IXGBE_BPRC); + sc->stats.bprc += bprc; + sc->stats.mprc += IXGBE_READ_REG(hw, IXGBE_MPRC); + sc->stats.mprc -= bprc; + + sc->stats.roc += IXGBE_READ_REG(hw, IXGBE_ROC); + sc->stats.prc64 += IXGBE_READ_REG(hw, IXGBE_PRC64); + sc->stats.prc127 += IXGBE_READ_REG(hw, IXGBE_PRC127); + sc->stats.prc255 += IXGBE_READ_REG(hw, IXGBE_PRC255); + sc->stats.prc511 += IXGBE_READ_REG(hw, IXGBE_PRC511); + sc->stats.prc1023 += IXGBE_READ_REG(hw, IXGBE_PRC1023); + sc->stats.prc1522 += IXGBE_READ_REG(hw, IXGBE_PRC1522); + sc->stats.rlec += IXGBE_READ_REG(hw, IXGBE_RLEC); + + sc->stats.lxonrxc += IXGBE_READ_REG(hw, IXGBE_LXONRXC); + sc->stats.lxoffrxc += IXGBE_READ_REG(hw, IXGBE_LXOFFRXC); + + lxon = IXGBE_READ_REG(hw, IXGBE_LXONTXC); + sc->stats.lxontxc += lxon; + lxoff = IXGBE_READ_REG(hw, IXGBE_LXOFFTXC); + sc->stats.lxofftxc += lxoff; + total = lxon + lxoff; + + sc->stats.gptc += IXGBE_READ_REG(hw, IXGBE_GPTC); + sc->stats.mptc += IXGBE_READ_REG(hw, IXGBE_MPTC); + sc->stats.ptc64 += IXGBE_READ_REG(hw, IXGBE_PTC64); + sc->stats.gptc -= total; + sc->stats.mptc -= total; + sc->stats.ptc64 -= total; + sc->stats.gotc -= total * ETHER_MIN_LEN; + + sc->stats.ruc += IXGBE_READ_REG(hw, IXGBE_RUC); + sc->stats.rfc += IXGBE_READ_REG(hw, IXGBE_RFC); + sc->stats.rjc += IXGBE_READ_REG(hw, IXGBE_RJC); + sc->stats.tpr += IXGBE_READ_REG(hw, IXGBE_TPR); + sc->stats.ptc127 += IXGBE_READ_REG(hw, IXGBE_PTC127); + sc->stats.ptc255 += IXGBE_READ_REG(hw, IXGBE_PTC255); + sc->stats.ptc511 += IXGBE_READ_REG(hw, IXGBE_PTC511); + sc->stats.ptc1023 += IXGBE_READ_REG(hw, IXGBE_PTC1023); + sc->stats.ptc1522 += IXGBE_READ_REG(hw, IXGBE_PTC1522); + sc->stats.bptc += IXGBE_READ_REG(hw, IXGBE_BPTC); + +#if 0 + /* Fill out the OS statistics structure */ + ifp->if_ipackets = sc->stats.gprc; + ifp->if_opackets = sc->stats.gptc; + ifp->if_ibytes = sc->stats.gorc; + ifp->if_obytes = sc->stats.gotc; + ifp->if_imcasts = sc->stats.mprc; +#endif + ifp->if_collisions = 0; + ifp->if_oerrors = sc->watchdog_events; + ifp->if_ierrors = missed_rx + sc->stats.crcerrs + sc->stats.rlec; +} + +#ifdef IX_DEBUG +/********************************************************************** + * + * This routine is called only when ixgbe_display_debug_stats is enabled. + * This routine provides a way to take a look at important statistics + * maintained by the driver and hardware. + * + **********************************************************************/ +void +ixgbe_print_hw_stats(struct ix_softc * sc) +{ + struct ifnet *ifp = &sc->arpcom.ac_if;; + + printf("%s: mbuf alloc failed %lu, mbuf cluster failed %lu, " + "missed pkts %llu, rx len errs %llu, crc errs %llu, " + "dropped pkts %lu, watchdog timeouts %ld, " + "XON rx %llu, XON tx %llu, XOFF rx %llu, XOFF tx %llu, " + "total pkts rx %llu, good pkts rx %llu, good pkts tx %llu, " + "tso tx %lu\n", + ifp->if_xname, + sc->mbuf_alloc_failed, + sc->mbuf_cluster_failed, + (long long)sc->stats.mpc[0], + (long long)sc->stats.roc + (long long)sc->stats.ruc, + (long long)sc->stats.crcerrs, + sc->dropped_pkts, + sc->watchdog_events, + (long long)sc->stats.lxonrxc, + (long long)sc->stats.lxontxc, + (long long)sc->stats.lxoffrxc, + (long long)sc->stats.lxofftxc, + (long long)sc->stats.tpr, + (long long)sc->stats.gprc, + (long long)sc->stats.gptc, + sc->tso_tx); +} + +/* + * Set flow control using sysctl: + * Flow control values: + * 0 - off + * 1 - rx pause + * 2 - tx pause + * 3 - full + */ +int +ixgbe_set_flowcntl(SYSCTL_HANDLER_ARGS) +{ + int error; + struct ix_softc *sc; + + error = sysctl_handle_int(oidp, &ixgbe_flow_control, 0, req); + + if (error) + return (error); + + sc = (struct ix_softc *) arg1; + switch (ixgbe_flow_control) { + case ixgbe_fc_rx_pause: + case ixgbe_fc_tx_pause: + case ixgbe_fc_full: + sc->hw.fc.type = ixgbe_flow_control; + break; + case ixgbe_fc_none: + default: + sc->hw.fc.type = ixgbe_fc_none; + } + + ixgbe_setup_fc(&sc->hw, 0); + return error; +} +#endif + +#ifndef NO_82598_A0_SUPPORT +/* + * A0 Workaround: invert descriptor for hardware + */ +void +desc_flip(void *desc) +{ + struct dhack {u32 a1; u32 a2; u32 b1; u32 b2;}; + struct dhack *d; + + d = (struct dhack *)desc; + d->a1 = ~(d->a1); + d->a2 = ~(d->a2); + d->b1 = ~(d->b1); + d->b2 = ~(d->b2); + d->b2 &= 0xFFFFFFF0; + d->b1 &= ~IXGBE_ADVTXD_DCMD_RS; +} +#endif + + + |