179251f5eSSepherosa Ziehau /* 279251f5eSSepherosa Ziehau * Copyright (c) 2001-2013, Intel Corporation 379251f5eSSepherosa Ziehau * All rights reserved. 479251f5eSSepherosa Ziehau * 579251f5eSSepherosa Ziehau * Redistribution and use in source and binary forms, with or without 679251f5eSSepherosa Ziehau * modification, are permitted provided that the following conditions are met: 779251f5eSSepherosa Ziehau * 879251f5eSSepherosa Ziehau * 1. Redistributions of source code must retain the above copyright notice, 979251f5eSSepherosa Ziehau * this list of conditions and the following disclaimer. 1079251f5eSSepherosa Ziehau * 1179251f5eSSepherosa Ziehau * 2. Redistributions in binary form must reproduce the above copyright 1279251f5eSSepherosa Ziehau * notice, this list of conditions and the following disclaimer in the 1379251f5eSSepherosa Ziehau * documentation and/or other materials provided with the distribution. 1479251f5eSSepherosa Ziehau * 1579251f5eSSepherosa Ziehau * 3. Neither the name of the Intel Corporation nor the names of its 1679251f5eSSepherosa Ziehau * contributors may be used to endorse or promote products derived from 1779251f5eSSepherosa Ziehau * this software without specific prior written permission. 1879251f5eSSepherosa Ziehau * 1979251f5eSSepherosa Ziehau * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" 2079251f5eSSepherosa Ziehau * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 2179251f5eSSepherosa Ziehau * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 2279251f5eSSepherosa Ziehau * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE 2379251f5eSSepherosa Ziehau * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 2479251f5eSSepherosa Ziehau * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 2579251f5eSSepherosa Ziehau * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 2679251f5eSSepherosa Ziehau * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 2779251f5eSSepherosa Ziehau * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 2879251f5eSSepherosa Ziehau * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 2979251f5eSSepherosa Ziehau * POSSIBILITY OF SUCH DAMAGE. 3079251f5eSSepherosa Ziehau */ 3179251f5eSSepherosa Ziehau 3279251f5eSSepherosa Ziehau #include "opt_ix.h" 3379251f5eSSepherosa Ziehau 3479251f5eSSepherosa Ziehau #include <sys/param.h> 3579251f5eSSepherosa Ziehau #include <sys/bus.h> 3679251f5eSSepherosa Ziehau #include <sys/endian.h> 3779251f5eSSepherosa Ziehau #include <sys/interrupt.h> 3879251f5eSSepherosa Ziehau #include <sys/kernel.h> 3979251f5eSSepherosa Ziehau #include <sys/malloc.h> 4079251f5eSSepherosa Ziehau #include <sys/mbuf.h> 4179251f5eSSepherosa Ziehau #include <sys/proc.h> 4279251f5eSSepherosa Ziehau #include <sys/rman.h> 4379251f5eSSepherosa Ziehau #include <sys/serialize.h> 4479251f5eSSepherosa Ziehau #include <sys/serialize2.h> 4579251f5eSSepherosa Ziehau #include <sys/socket.h> 4679251f5eSSepherosa Ziehau #include <sys/sockio.h> 4779251f5eSSepherosa Ziehau #include <sys/sysctl.h> 4879251f5eSSepherosa Ziehau #include <sys/systm.h> 4979251f5eSSepherosa Ziehau 5079251f5eSSepherosa Ziehau #include <net/bpf.h> 5179251f5eSSepherosa Ziehau #include <net/ethernet.h> 5279251f5eSSepherosa Ziehau #include <net/if.h> 5379251f5eSSepherosa Ziehau #include <net/if_arp.h> 5479251f5eSSepherosa Ziehau #include <net/if_dl.h> 5579251f5eSSepherosa Ziehau #include <net/if_media.h> 5679251f5eSSepherosa Ziehau #include <net/ifq_var.h> 5779251f5eSSepherosa Ziehau #include <net/toeplitz.h> 5879251f5eSSepherosa Ziehau #include <net/toeplitz2.h> 5979251f5eSSepherosa Ziehau #include <net/vlan/if_vlan_var.h> 6079251f5eSSepherosa Ziehau #include <net/vlan/if_vlan_ether.h> 6179251f5eSSepherosa Ziehau #include <net/if_poll.h> 6279251f5eSSepherosa Ziehau 6379251f5eSSepherosa Ziehau #include <netinet/in_systm.h> 6479251f5eSSepherosa Ziehau #include <netinet/in.h> 6579251f5eSSepherosa Ziehau #include <netinet/ip.h> 6679251f5eSSepherosa Ziehau 6779251f5eSSepherosa Ziehau #include <bus/pci/pcivar.h> 6879251f5eSSepherosa Ziehau #include <bus/pci/pcireg.h> 6979251f5eSSepherosa Ziehau 7079251f5eSSepherosa Ziehau #include <dev/netif/ix/ixgbe_api.h> 7179251f5eSSepherosa Ziehau #include <dev/netif/ix/if_ix.h> 7279251f5eSSepherosa Ziehau 7379251f5eSSepherosa Ziehau #ifdef IX_RSS_DEBUG 7479251f5eSSepherosa Ziehau #define IX_RSS_DPRINTF(sc, lvl, fmt, ...) \ 7579251f5eSSepherosa Ziehau do { \ 7679251f5eSSepherosa Ziehau if (sc->rss_debug >= lvl) \ 7779251f5eSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, fmt, __VA_ARGS__); \ 7879251f5eSSepherosa Ziehau } while (0) 7979251f5eSSepherosa Ziehau #else /* !IX_RSS_DEBUG */ 8079251f5eSSepherosa Ziehau #define IX_RSS_DPRINTF(sc, lvl, fmt, ...) ((void)0) 8179251f5eSSepherosa Ziehau #endif /* IX_RSS_DEBUG */ 8279251f5eSSepherosa Ziehau 8379251f5eSSepherosa Ziehau #define IX_NAME "Intel(R) PRO/10GbE " 8479251f5eSSepherosa Ziehau #define IX_DEVICE(id) \ 8579251f5eSSepherosa Ziehau { IXGBE_VENDOR_ID, IXGBE_DEV_ID_##id, IX_NAME #id } 8679251f5eSSepherosa Ziehau #define IX_DEVICE_NULL { 0, 0, NULL } 8779251f5eSSepherosa Ziehau 8879251f5eSSepherosa Ziehau static struct ix_device { 8979251f5eSSepherosa Ziehau uint16_t vid; 9079251f5eSSepherosa Ziehau uint16_t did; 9179251f5eSSepherosa Ziehau const char *desc; 9279251f5eSSepherosa Ziehau } ix_devices[] = { 9379251f5eSSepherosa Ziehau IX_DEVICE(82598AF_DUAL_PORT), 9479251f5eSSepherosa Ziehau IX_DEVICE(82598AF_SINGLE_PORT), 9579251f5eSSepherosa Ziehau IX_DEVICE(82598EB_CX4), 9679251f5eSSepherosa Ziehau IX_DEVICE(82598AT), 9779251f5eSSepherosa Ziehau IX_DEVICE(82598AT2), 9879251f5eSSepherosa Ziehau IX_DEVICE(82598), 9979251f5eSSepherosa Ziehau IX_DEVICE(82598_DA_DUAL_PORT), 10079251f5eSSepherosa Ziehau IX_DEVICE(82598_CX4_DUAL_PORT), 10179251f5eSSepherosa Ziehau IX_DEVICE(82598EB_XF_LR), 10279251f5eSSepherosa Ziehau IX_DEVICE(82598_SR_DUAL_PORT_EM), 10379251f5eSSepherosa Ziehau IX_DEVICE(82598EB_SFP_LOM), 10479251f5eSSepherosa Ziehau IX_DEVICE(82599_KX4), 10579251f5eSSepherosa Ziehau IX_DEVICE(82599_KX4_MEZZ), 10679251f5eSSepherosa Ziehau IX_DEVICE(82599_SFP), 10779251f5eSSepherosa Ziehau IX_DEVICE(82599_XAUI_LOM), 10879251f5eSSepherosa Ziehau IX_DEVICE(82599_CX4), 10979251f5eSSepherosa Ziehau IX_DEVICE(82599_T3_LOM), 11079251f5eSSepherosa Ziehau IX_DEVICE(82599_COMBO_BACKPLANE), 11179251f5eSSepherosa Ziehau IX_DEVICE(82599_BACKPLANE_FCOE), 11279251f5eSSepherosa Ziehau IX_DEVICE(82599_SFP_SF2), 11379251f5eSSepherosa Ziehau IX_DEVICE(82599_SFP_FCOE), 11479251f5eSSepherosa Ziehau IX_DEVICE(82599EN_SFP), 11579251f5eSSepherosa Ziehau IX_DEVICE(82599_SFP_SF_QP), 11679251f5eSSepherosa Ziehau IX_DEVICE(X540T), 11779251f5eSSepherosa Ziehau 11879251f5eSSepherosa Ziehau /* required last entry */ 11979251f5eSSepherosa Ziehau IX_DEVICE_NULL 12079251f5eSSepherosa Ziehau }; 12179251f5eSSepherosa Ziehau 12279251f5eSSepherosa Ziehau static int ix_probe(device_t); 12379251f5eSSepherosa Ziehau static int ix_attach(device_t); 12479251f5eSSepherosa Ziehau static int ix_detach(device_t); 12579251f5eSSepherosa Ziehau static int ix_shutdown(device_t); 12679251f5eSSepherosa Ziehau 12779251f5eSSepherosa Ziehau static void ix_serialize(struct ifnet *, enum ifnet_serialize); 12879251f5eSSepherosa Ziehau static void ix_deserialize(struct ifnet *, enum ifnet_serialize); 12979251f5eSSepherosa Ziehau static int ix_tryserialize(struct ifnet *, enum ifnet_serialize); 13079251f5eSSepherosa Ziehau #ifdef INVARIANTS 13179251f5eSSepherosa Ziehau static void ix_serialize_assert(struct ifnet *, enum ifnet_serialize, 13279251f5eSSepherosa Ziehau boolean_t); 13379251f5eSSepherosa Ziehau #endif 13479251f5eSSepherosa Ziehau static void ix_start(struct ifnet *, struct ifaltq_subque *); 13579251f5eSSepherosa Ziehau static void ix_watchdog(struct ifaltq_subque *); 13679251f5eSSepherosa Ziehau static int ix_ioctl(struct ifnet *, u_long, caddr_t, struct ucred *); 13779251f5eSSepherosa Ziehau static void ix_init(void *); 13879251f5eSSepherosa Ziehau static void ix_stop(struct ix_softc *); 13979251f5eSSepherosa Ziehau static void ix_media_status(struct ifnet *, struct ifmediareq *); 14079251f5eSSepherosa Ziehau static int ix_media_change(struct ifnet *); 14179251f5eSSepherosa Ziehau static void ix_timer(void *); 14279251f5eSSepherosa Ziehau 14379251f5eSSepherosa Ziehau static void ix_add_sysctl(struct ix_softc *); 144*189a0ff3SSepherosa Ziehau static void ix_add_intr_rate_sysctl(struct ix_softc *, int, 145*189a0ff3SSepherosa Ziehau const char *, int (*)(SYSCTL_HANDLER_ARGS), const char *); 14679251f5eSSepherosa Ziehau static int ix_sysctl_tx_wreg_nsegs(SYSCTL_HANDLER_ARGS); 14779251f5eSSepherosa Ziehau static int ix_sysctl_rx_wreg_nsegs(SYSCTL_HANDLER_ARGS); 14879251f5eSSepherosa Ziehau static int ix_sysctl_txd(SYSCTL_HANDLER_ARGS); 14979251f5eSSepherosa Ziehau static int ix_sysctl_rxd(SYSCTL_HANDLER_ARGS); 15079251f5eSSepherosa Ziehau static int ix_sysctl_tx_intr_nsegs(SYSCTL_HANDLER_ARGS); 151*189a0ff3SSepherosa Ziehau static int ix_sysctl_intr_rate(SYSCTL_HANDLER_ARGS, int); 152*189a0ff3SSepherosa Ziehau static int ix_sysctl_rxtx_intr_rate(SYSCTL_HANDLER_ARGS); 153*189a0ff3SSepherosa Ziehau static int ix_sysctl_rx_intr_rate(SYSCTL_HANDLER_ARGS); 154*189a0ff3SSepherosa Ziehau static int ix_sysctl_tx_intr_rate(SYSCTL_HANDLER_ARGS); 155*189a0ff3SSepherosa Ziehau static int ix_sysctl_sts_intr_rate(SYSCTL_HANDLER_ARGS); 15679251f5eSSepherosa Ziehau static int ix_sysctl_flowctrl(SYSCTL_HANDLER_ARGS); 15779251f5eSSepherosa Ziehau #ifdef foo 15879251f5eSSepherosa Ziehau static int ix_sysctl_advspeed(SYSCTL_HANDLER_ARGS); 15979251f5eSSepherosa Ziehau #endif 16079251f5eSSepherosa Ziehau #if 0 16179251f5eSSepherosa Ziehau static void ix_add_hw_stats(struct ix_softc *); 16279251f5eSSepherosa Ziehau #endif 16379251f5eSSepherosa Ziehau 16479251f5eSSepherosa Ziehau static void ix_slot_info(struct ix_softc *); 16579251f5eSSepherosa Ziehau static int ix_alloc_rings(struct ix_softc *); 16679251f5eSSepherosa Ziehau static void ix_free_rings(struct ix_softc *); 16779251f5eSSepherosa Ziehau static void ix_setup_ifp(struct ix_softc *); 16879251f5eSSepherosa Ziehau static void ix_setup_serialize(struct ix_softc *); 16979251f5eSSepherosa Ziehau static void ix_set_ring_inuse(struct ix_softc *, boolean_t); 17079251f5eSSepherosa Ziehau static void ix_set_timer_cpuid(struct ix_softc *, boolean_t); 17179251f5eSSepherosa Ziehau static void ix_update_stats(struct ix_softc *); 17279251f5eSSepherosa Ziehau 17379251f5eSSepherosa Ziehau static void ix_set_promisc(struct ix_softc *); 17479251f5eSSepherosa Ziehau static void ix_set_multi(struct ix_softc *); 17579251f5eSSepherosa Ziehau static void ix_set_vlan(struct ix_softc *); 17679251f5eSSepherosa Ziehau static uint8_t *ix_mc_array_itr(struct ixgbe_hw *, uint8_t **, uint32_t *); 17779251f5eSSepherosa Ziehau 17879251f5eSSepherosa Ziehau static int ix_get_txring_inuse(const struct ix_softc *, boolean_t); 17979251f5eSSepherosa Ziehau static void ix_init_tx_ring(struct ix_tx_ring *); 18079251f5eSSepherosa Ziehau static void ix_free_tx_ring(struct ix_tx_ring *); 18179251f5eSSepherosa Ziehau static int ix_create_tx_ring(struct ix_tx_ring *); 18279251f5eSSepherosa Ziehau static void ix_destroy_tx_ring(struct ix_tx_ring *, int); 18379251f5eSSepherosa Ziehau static void ix_init_tx_unit(struct ix_softc *); 18479251f5eSSepherosa Ziehau static int ix_encap(struct ix_tx_ring *, struct mbuf **, 18579251f5eSSepherosa Ziehau uint16_t *, int *); 18679251f5eSSepherosa Ziehau static int ix_tx_ctx_setup(struct ix_tx_ring *, 18779251f5eSSepherosa Ziehau const struct mbuf *, uint32_t *, uint32_t *); 18879251f5eSSepherosa Ziehau static int ix_tso_ctx_setup(struct ix_tx_ring *, 18979251f5eSSepherosa Ziehau const struct mbuf *, uint32_t *, uint32_t *); 190*189a0ff3SSepherosa Ziehau static void ix_txeof(struct ix_tx_ring *, int); 19179251f5eSSepherosa Ziehau 19279251f5eSSepherosa Ziehau static int ix_get_rxring_inuse(const struct ix_softc *, boolean_t); 19379251f5eSSepherosa Ziehau static int ix_init_rx_ring(struct ix_rx_ring *); 19479251f5eSSepherosa Ziehau static void ix_free_rx_ring(struct ix_rx_ring *); 19579251f5eSSepherosa Ziehau static int ix_create_rx_ring(struct ix_rx_ring *); 19679251f5eSSepherosa Ziehau static void ix_destroy_rx_ring(struct ix_rx_ring *, int); 19779251f5eSSepherosa Ziehau static void ix_init_rx_unit(struct ix_softc *); 19879251f5eSSepherosa Ziehau #if 0 19979251f5eSSepherosa Ziehau static void ix_setup_hw_rsc(struct ix_rx_ring *); 20079251f5eSSepherosa Ziehau #endif 20179251f5eSSepherosa Ziehau static int ix_newbuf(struct ix_rx_ring *, int, boolean_t); 20279251f5eSSepherosa Ziehau static void ix_rxeof(struct ix_rx_ring *); 20379251f5eSSepherosa Ziehau static void ix_rx_discard(struct ix_rx_ring *, int, boolean_t); 20479251f5eSSepherosa Ziehau static void ix_enable_rx_drop(struct ix_softc *); 20579251f5eSSepherosa Ziehau static void ix_disable_rx_drop(struct ix_softc *); 20679251f5eSSepherosa Ziehau 207*189a0ff3SSepherosa Ziehau static void ix_alloc_msix(struct ix_softc *); 208*189a0ff3SSepherosa Ziehau static void ix_free_msix(struct ix_softc *, boolean_t); 209*189a0ff3SSepherosa Ziehau static void ix_conf_rx_msix(struct ix_softc *, int, int *, int); 210*189a0ff3SSepherosa Ziehau static void ix_conf_tx_msix(struct ix_softc *, int, int *, int); 211*189a0ff3SSepherosa Ziehau static void ix_setup_msix_eims(const struct ix_softc *, int, 212*189a0ff3SSepherosa Ziehau uint32_t *, uint32_t *); 21379251f5eSSepherosa Ziehau static int ix_alloc_intr(struct ix_softc *); 21479251f5eSSepherosa Ziehau static void ix_free_intr(struct ix_softc *); 21579251f5eSSepherosa Ziehau static int ix_setup_intr(struct ix_softc *); 21679251f5eSSepherosa Ziehau static void ix_teardown_intr(struct ix_softc *, int); 21779251f5eSSepherosa Ziehau static void ix_enable_intr(struct ix_softc *); 21879251f5eSSepherosa Ziehau static void ix_disable_intr(struct ix_softc *); 21979251f5eSSepherosa Ziehau static void ix_set_ivar(struct ix_softc *, uint8_t, uint8_t, int8_t); 22079251f5eSSepherosa Ziehau static void ix_set_eitr(struct ix_softc *, int, int); 221*189a0ff3SSepherosa Ziehau static void ix_intr_status(struct ix_softc *, uint32_t); 22279251f5eSSepherosa Ziehau static void ix_intr(void *); 223*189a0ff3SSepherosa Ziehau static void ix_msix_rxtx(void *); 224*189a0ff3SSepherosa Ziehau static void ix_msix_rx(void *); 225*189a0ff3SSepherosa Ziehau static void ix_msix_tx(void *); 226*189a0ff3SSepherosa Ziehau static void ix_msix_status(void *); 22779251f5eSSepherosa Ziehau 22879251f5eSSepherosa Ziehau static void ix_config_link(struct ix_softc *); 22979251f5eSSepherosa Ziehau static boolean_t ix_sfp_probe(struct ix_softc *); 23079251f5eSSepherosa Ziehau static boolean_t ix_is_sfp(const struct ixgbe_hw *); 23179251f5eSSepherosa Ziehau static void ix_setup_optics(struct ix_softc *); 23279251f5eSSepherosa Ziehau static void ix_update_link_status(struct ix_softc *); 23379251f5eSSepherosa Ziehau static void ix_handle_link(struct ix_softc *); 23479251f5eSSepherosa Ziehau static void ix_handle_mod(struct ix_softc *); 23579251f5eSSepherosa Ziehau static void ix_handle_msf(struct ix_softc *); 23679251f5eSSepherosa Ziehau 23779251f5eSSepherosa Ziehau /* XXX Shared code structure requires this for the moment */ 23879251f5eSSepherosa Ziehau extern void ixgbe_stop_mac_link_on_d3_82599(struct ixgbe_hw *); 23979251f5eSSepherosa Ziehau 24079251f5eSSepherosa Ziehau static device_method_t ix_methods[] = { 24179251f5eSSepherosa Ziehau /* Device interface */ 24279251f5eSSepherosa Ziehau DEVMETHOD(device_probe, ix_probe), 24379251f5eSSepherosa Ziehau DEVMETHOD(device_attach, ix_attach), 24479251f5eSSepherosa Ziehau DEVMETHOD(device_detach, ix_detach), 24579251f5eSSepherosa Ziehau DEVMETHOD(device_shutdown, ix_shutdown), 24679251f5eSSepherosa Ziehau DEVMETHOD_END 24779251f5eSSepherosa Ziehau }; 24879251f5eSSepherosa Ziehau 24979251f5eSSepherosa Ziehau static driver_t ix_driver = { 25079251f5eSSepherosa Ziehau "ix", 25179251f5eSSepherosa Ziehau ix_methods, 25279251f5eSSepherosa Ziehau sizeof(struct ix_softc) 25379251f5eSSepherosa Ziehau }; 25479251f5eSSepherosa Ziehau 25579251f5eSSepherosa Ziehau static devclass_t ix_devclass; 25679251f5eSSepherosa Ziehau 25779251f5eSSepherosa Ziehau DECLARE_DUMMY_MODULE(if_ix); 25879251f5eSSepherosa Ziehau DRIVER_MODULE(if_ix, pci, ix_driver, ix_devclass, NULL, NULL); 25979251f5eSSepherosa Ziehau 26079251f5eSSepherosa Ziehau static int ix_msi_enable = 1; 261*189a0ff3SSepherosa Ziehau static int ix_msix_enable = 1; 262*189a0ff3SSepherosa Ziehau static int ix_msix_agg_rxtx = 1; 26379251f5eSSepherosa Ziehau static int ix_rxr = 0; 264*189a0ff3SSepherosa Ziehau static int ix_txr = 0; 26579251f5eSSepherosa Ziehau static int ix_txd = IX_PERF_TXD; 26679251f5eSSepherosa Ziehau static int ix_rxd = IX_PERF_RXD; 26779251f5eSSepherosa Ziehau static int ix_unsupported_sfp = 0; 26879251f5eSSepherosa Ziehau 26979251f5eSSepherosa Ziehau TUNABLE_INT("hw.ix.msi.enable", &ix_msi_enable); 270*189a0ff3SSepherosa Ziehau TUNABLE_INT("hw.ix.msix.enable", &ix_msix_enable); 271*189a0ff3SSepherosa Ziehau TUNABLE_INT("hw.ix.msix.agg_rxtx", &ix_msix_agg_rxtx); 27279251f5eSSepherosa Ziehau TUNABLE_INT("hw.ix.rxr", &ix_rxr); 273*189a0ff3SSepherosa Ziehau TUNABLE_INT("hw.ix.txr", &ix_txr); 27479251f5eSSepherosa Ziehau TUNABLE_INT("hw.ix.txd", &ix_txd); 27579251f5eSSepherosa Ziehau TUNABLE_INT("hw.ix.rxd", &ix_rxd); 27679251f5eSSepherosa Ziehau TUNABLE_INT("hw.ix.unsupported_sfp", &ix_unsupported_sfp); 27779251f5eSSepherosa Ziehau 27879251f5eSSepherosa Ziehau /* 27979251f5eSSepherosa Ziehau * Smart speed setting, default to on. This only works 28079251f5eSSepherosa Ziehau * as a compile option right now as its during attach, 28179251f5eSSepherosa Ziehau * set this to 'ixgbe_smart_speed_off' to disable. 28279251f5eSSepherosa Ziehau */ 28379251f5eSSepherosa Ziehau static const enum ixgbe_smart_speed ix_smart_speed = 28479251f5eSSepherosa Ziehau ixgbe_smart_speed_on; 28579251f5eSSepherosa Ziehau 28679251f5eSSepherosa Ziehau static int 28779251f5eSSepherosa Ziehau ix_probe(device_t dev) 28879251f5eSSepherosa Ziehau { 28979251f5eSSepherosa Ziehau const struct ix_device *d; 29079251f5eSSepherosa Ziehau uint16_t vid, did; 29179251f5eSSepherosa Ziehau 29279251f5eSSepherosa Ziehau vid = pci_get_vendor(dev); 29379251f5eSSepherosa Ziehau did = pci_get_device(dev); 29479251f5eSSepherosa Ziehau 29579251f5eSSepherosa Ziehau for (d = ix_devices; d->desc != NULL; ++d) { 29679251f5eSSepherosa Ziehau if (vid == d->vid && did == d->did) { 29779251f5eSSepherosa Ziehau device_set_desc(dev, d->desc); 29879251f5eSSepherosa Ziehau return 0; 29979251f5eSSepherosa Ziehau } 30079251f5eSSepherosa Ziehau } 30179251f5eSSepherosa Ziehau return ENXIO; 30279251f5eSSepherosa Ziehau } 30379251f5eSSepherosa Ziehau 30479251f5eSSepherosa Ziehau static int 30579251f5eSSepherosa Ziehau ix_attach(device_t dev) 30679251f5eSSepherosa Ziehau { 30779251f5eSSepherosa Ziehau struct ix_softc *sc = device_get_softc(dev); 30879251f5eSSepherosa Ziehau struct ixgbe_hw *hw; 309*189a0ff3SSepherosa Ziehau int error, ring_cnt_max; 31079251f5eSSepherosa Ziehau uint16_t csum; 31179251f5eSSepherosa Ziehau uint32_t ctrl_ext; 31279251f5eSSepherosa Ziehau 31379251f5eSSepherosa Ziehau sc->dev = sc->osdep.dev = dev; 31479251f5eSSepherosa Ziehau hw = &sc->hw; 31579251f5eSSepherosa Ziehau 31679251f5eSSepherosa Ziehau if_initname(&sc->arpcom.ac_if, device_get_name(dev), 31779251f5eSSepherosa Ziehau device_get_unit(dev)); 31879251f5eSSepherosa Ziehau ifmedia_init(&sc->media, IFM_IMASK, 31979251f5eSSepherosa Ziehau ix_media_change, ix_media_status); 32079251f5eSSepherosa Ziehau 32179251f5eSSepherosa Ziehau /* Save frame size */ 32279251f5eSSepherosa Ziehau sc->max_frame_size = ETHERMTU + ETHER_HDR_LEN + ETHER_CRC_LEN; 32379251f5eSSepherosa Ziehau 32479251f5eSSepherosa Ziehau callout_init_mp(&sc->timer); 32579251f5eSSepherosa Ziehau lwkt_serialize_init(&sc->main_serialize); 32679251f5eSSepherosa Ziehau 32779251f5eSSepherosa Ziehau /* 32879251f5eSSepherosa Ziehau * Save off the information about this board 32979251f5eSSepherosa Ziehau */ 33079251f5eSSepherosa Ziehau hw->vendor_id = pci_get_vendor(dev); 33179251f5eSSepherosa Ziehau hw->device_id = pci_get_device(dev); 33279251f5eSSepherosa Ziehau hw->revision_id = pci_read_config(dev, PCIR_REVID, 1); 33379251f5eSSepherosa Ziehau hw->subsystem_vendor_id = pci_read_config(dev, PCIR_SUBVEND_0, 2); 33479251f5eSSepherosa Ziehau hw->subsystem_device_id = pci_read_config(dev, PCIR_SUBDEV_0, 2); 33579251f5eSSepherosa Ziehau 33679251f5eSSepherosa Ziehau ixgbe_set_mac_type(hw); 33779251f5eSSepherosa Ziehau 33879251f5eSSepherosa Ziehau /* Pick up the 82599 and VF settings */ 33979251f5eSSepherosa Ziehau if (hw->mac.type != ixgbe_mac_82598EB) 34079251f5eSSepherosa Ziehau hw->phy.smart_speed = ix_smart_speed; 34179251f5eSSepherosa Ziehau 34279251f5eSSepherosa Ziehau /* Enable bus mastering */ 34379251f5eSSepherosa Ziehau pci_enable_busmaster(dev); 34479251f5eSSepherosa Ziehau 34579251f5eSSepherosa Ziehau /* 34679251f5eSSepherosa Ziehau * Allocate IO memory 34779251f5eSSepherosa Ziehau */ 34879251f5eSSepherosa Ziehau sc->mem_rid = PCIR_BAR(0); 34979251f5eSSepherosa Ziehau sc->mem_res = bus_alloc_resource_any(dev, SYS_RES_MEMORY, 35079251f5eSSepherosa Ziehau &sc->mem_rid, RF_ACTIVE); 35179251f5eSSepherosa Ziehau if (sc->mem_res == NULL) { 35279251f5eSSepherosa Ziehau device_printf(dev, "Unable to allocate bus resource: memory\n"); 35379251f5eSSepherosa Ziehau error = ENXIO; 35479251f5eSSepherosa Ziehau goto failed; 35579251f5eSSepherosa Ziehau } 35679251f5eSSepherosa Ziehau 35779251f5eSSepherosa Ziehau sc->osdep.mem_bus_space_tag = rman_get_bustag(sc->mem_res); 35879251f5eSSepherosa Ziehau sc->osdep.mem_bus_space_handle = rman_get_bushandle(sc->mem_res); 35979251f5eSSepherosa Ziehau 36079251f5eSSepherosa Ziehau sc->hw.hw_addr = (uint8_t *)&sc->osdep.mem_bus_space_handle; 36179251f5eSSepherosa Ziehau sc->hw.back = &sc->osdep; 36279251f5eSSepherosa Ziehau 36379251f5eSSepherosa Ziehau /* 36479251f5eSSepherosa Ziehau * Configure total supported RX/TX ring count 36579251f5eSSepherosa Ziehau */ 36679251f5eSSepherosa Ziehau sc->rx_ring_cnt = device_getenv_int(dev, "rxr", ix_rxr); 36779251f5eSSepherosa Ziehau sc->rx_ring_cnt = if_ring_count2(sc->rx_ring_cnt, IX_MAX_RXRING); 36879251f5eSSepherosa Ziehau sc->rx_ring_inuse = sc->rx_ring_cnt; 36979251f5eSSepherosa Ziehau 370*189a0ff3SSepherosa Ziehau switch (hw->mac.type) { 371*189a0ff3SSepherosa Ziehau case ixgbe_mac_82598EB: 372*189a0ff3SSepherosa Ziehau ring_cnt_max = IX_MAX_TXRING_82598; 373*189a0ff3SSepherosa Ziehau break; 374*189a0ff3SSepherosa Ziehau 375*189a0ff3SSepherosa Ziehau case ixgbe_mac_82599EB: 376*189a0ff3SSepherosa Ziehau ring_cnt_max = IX_MAX_TXRING_82599; 377*189a0ff3SSepherosa Ziehau break; 378*189a0ff3SSepherosa Ziehau 379*189a0ff3SSepherosa Ziehau case ixgbe_mac_X540: 380*189a0ff3SSepherosa Ziehau ring_cnt_max = IX_MAX_TXRING_X540; 381*189a0ff3SSepherosa Ziehau break; 382*189a0ff3SSepherosa Ziehau 383*189a0ff3SSepherosa Ziehau default: 384*189a0ff3SSepherosa Ziehau ring_cnt_max = 1; 385*189a0ff3SSepherosa Ziehau break; 386*189a0ff3SSepherosa Ziehau } 387*189a0ff3SSepherosa Ziehau sc->tx_ring_cnt = device_getenv_int(dev, "txr", ix_txr); 388*189a0ff3SSepherosa Ziehau sc->tx_ring_cnt = if_ring_count2(sc->tx_ring_cnt, ring_cnt_max); 38979251f5eSSepherosa Ziehau sc->tx_ring_inuse = sc->tx_ring_cnt; 39079251f5eSSepherosa Ziehau 39179251f5eSSepherosa Ziehau /* Allocate TX/RX rings */ 39279251f5eSSepherosa Ziehau error = ix_alloc_rings(sc); 39379251f5eSSepherosa Ziehau if (error) 39479251f5eSSepherosa Ziehau goto failed; 39579251f5eSSepherosa Ziehau 39679251f5eSSepherosa Ziehau /* Allocate interrupt */ 39779251f5eSSepherosa Ziehau error = ix_alloc_intr(sc); 39879251f5eSSepherosa Ziehau if (error) 39979251f5eSSepherosa Ziehau goto failed; 40079251f5eSSepherosa Ziehau 40179251f5eSSepherosa Ziehau /* Setup serializes */ 40279251f5eSSepherosa Ziehau ix_setup_serialize(sc); 40379251f5eSSepherosa Ziehau 40479251f5eSSepherosa Ziehau /* Allocate multicast array memory. */ 40579251f5eSSepherosa Ziehau sc->mta = kmalloc(IXGBE_ETH_LENGTH_OF_ADDRESS * IX_MAX_MCASTADDR, 40679251f5eSSepherosa Ziehau M_DEVBUF, M_WAITOK); 40779251f5eSSepherosa Ziehau 40879251f5eSSepherosa Ziehau /* Initialize the shared code */ 40979251f5eSSepherosa Ziehau hw->allow_unsupported_sfp = ix_unsupported_sfp; 41079251f5eSSepherosa Ziehau error = ixgbe_init_shared_code(hw); 41179251f5eSSepherosa Ziehau if (error == IXGBE_ERR_SFP_NOT_PRESENT) { 41279251f5eSSepherosa Ziehau /* 41379251f5eSSepherosa Ziehau * No optics in this port; ask timer routine 41479251f5eSSepherosa Ziehau * to probe for later insertion. 41579251f5eSSepherosa Ziehau */ 41679251f5eSSepherosa Ziehau sc->sfp_probe = TRUE; 41779251f5eSSepherosa Ziehau error = 0; 41879251f5eSSepherosa Ziehau } else if (error == IXGBE_ERR_SFP_NOT_SUPPORTED) { 41979251f5eSSepherosa Ziehau device_printf(dev, "Unsupported SFP+ module detected!\n"); 42079251f5eSSepherosa Ziehau error = EIO; 42179251f5eSSepherosa Ziehau goto failed; 42279251f5eSSepherosa Ziehau } else if (error) { 42379251f5eSSepherosa Ziehau device_printf(dev, "Unable to initialize the shared code\n"); 42479251f5eSSepherosa Ziehau error = EIO; 42579251f5eSSepherosa Ziehau goto failed; 42679251f5eSSepherosa Ziehau } 42779251f5eSSepherosa Ziehau 42879251f5eSSepherosa Ziehau /* Make sure we have a good EEPROM before we read from it */ 42979251f5eSSepherosa Ziehau if (ixgbe_validate_eeprom_checksum(&sc->hw, &csum) < 0) { 43079251f5eSSepherosa Ziehau device_printf(dev, "The EEPROM Checksum Is Not Valid\n"); 43179251f5eSSepherosa Ziehau error = EIO; 43279251f5eSSepherosa Ziehau goto failed; 43379251f5eSSepherosa Ziehau } 43479251f5eSSepherosa Ziehau 43579251f5eSSepherosa Ziehau error = ixgbe_init_hw(hw); 43679251f5eSSepherosa Ziehau if (error == IXGBE_ERR_EEPROM_VERSION) { 43779251f5eSSepherosa Ziehau device_printf(dev, "Pre-production device detected\n"); 43879251f5eSSepherosa Ziehau } else if (error == IXGBE_ERR_SFP_NOT_SUPPORTED) { 43979251f5eSSepherosa Ziehau device_printf(dev, "Unsupported SFP+ Module\n"); 44079251f5eSSepherosa Ziehau error = EIO; 44179251f5eSSepherosa Ziehau goto failed; 44279251f5eSSepherosa Ziehau } else if (error == IXGBE_ERR_SFP_NOT_PRESENT) { 44379251f5eSSepherosa Ziehau device_printf(dev, "No SFP+ Module found\n"); 44479251f5eSSepherosa Ziehau } 44579251f5eSSepherosa Ziehau 44679251f5eSSepherosa Ziehau /* Detect and set physical type */ 44779251f5eSSepherosa Ziehau ix_setup_optics(sc); 44879251f5eSSepherosa Ziehau 44979251f5eSSepherosa Ziehau /* Setup OS specific network interface */ 45079251f5eSSepherosa Ziehau ix_setup_ifp(sc); 45179251f5eSSepherosa Ziehau 45279251f5eSSepherosa Ziehau /* Add sysctl tree */ 45379251f5eSSepherosa Ziehau ix_add_sysctl(sc); 45479251f5eSSepherosa Ziehau 45579251f5eSSepherosa Ziehau error = ix_setup_intr(sc); 45679251f5eSSepherosa Ziehau if (error) { 45779251f5eSSepherosa Ziehau ether_ifdetach(&sc->arpcom.ac_if); 45879251f5eSSepherosa Ziehau goto failed; 45979251f5eSSepherosa Ziehau } 46079251f5eSSepherosa Ziehau 46179251f5eSSepherosa Ziehau /* Initialize statistics */ 46279251f5eSSepherosa Ziehau ix_update_stats(sc); 46379251f5eSSepherosa Ziehau 46479251f5eSSepherosa Ziehau /* 46579251f5eSSepherosa Ziehau * Check PCIE slot type/speed/width 46679251f5eSSepherosa Ziehau */ 46779251f5eSSepherosa Ziehau ix_slot_info(sc); 46879251f5eSSepherosa Ziehau 46979251f5eSSepherosa Ziehau /* Set an initial default flow control value */ 47079251f5eSSepherosa Ziehau sc->fc = ixgbe_fc_full; 47179251f5eSSepherosa Ziehau 47279251f5eSSepherosa Ziehau /* Let hardware know driver is loaded */ 47379251f5eSSepherosa Ziehau ctrl_ext = IXGBE_READ_REG(hw, IXGBE_CTRL_EXT); 47479251f5eSSepherosa Ziehau ctrl_ext |= IXGBE_CTRL_EXT_DRV_LOAD; 47579251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_CTRL_EXT, ctrl_ext); 47679251f5eSSepherosa Ziehau 47779251f5eSSepherosa Ziehau return 0; 47879251f5eSSepherosa Ziehau failed: 47979251f5eSSepherosa Ziehau ix_detach(dev); 48079251f5eSSepherosa Ziehau return error; 48179251f5eSSepherosa Ziehau } 48279251f5eSSepherosa Ziehau 48379251f5eSSepherosa Ziehau static int 48479251f5eSSepherosa Ziehau ix_detach(device_t dev) 48579251f5eSSepherosa Ziehau { 48679251f5eSSepherosa Ziehau struct ix_softc *sc = device_get_softc(dev); 48779251f5eSSepherosa Ziehau 48879251f5eSSepherosa Ziehau if (device_is_attached(dev)) { 48979251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 49079251f5eSSepherosa Ziehau uint32_t ctrl_ext; 49179251f5eSSepherosa Ziehau 49279251f5eSSepherosa Ziehau ifnet_serialize_all(ifp); 49379251f5eSSepherosa Ziehau 49479251f5eSSepherosa Ziehau ix_stop(sc); 49579251f5eSSepherosa Ziehau ix_teardown_intr(sc, sc->intr_cnt); 49679251f5eSSepherosa Ziehau 49779251f5eSSepherosa Ziehau ifnet_deserialize_all(ifp); 49879251f5eSSepherosa Ziehau 49979251f5eSSepherosa Ziehau callout_terminate(&sc->timer); 50079251f5eSSepherosa Ziehau ether_ifdetach(ifp); 50179251f5eSSepherosa Ziehau 50279251f5eSSepherosa Ziehau /* Let hardware know driver is unloading */ 50379251f5eSSepherosa Ziehau ctrl_ext = IXGBE_READ_REG(&sc->hw, IXGBE_CTRL_EXT); 50479251f5eSSepherosa Ziehau ctrl_ext &= ~IXGBE_CTRL_EXT_DRV_LOAD; 50579251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_CTRL_EXT, ctrl_ext); 50679251f5eSSepherosa Ziehau } 50779251f5eSSepherosa Ziehau 50879251f5eSSepherosa Ziehau ifmedia_removeall(&sc->media); 50979251f5eSSepherosa Ziehau bus_generic_detach(dev); 51079251f5eSSepherosa Ziehau 51179251f5eSSepherosa Ziehau if (sc->sysctl_tree != NULL) 51279251f5eSSepherosa Ziehau sysctl_ctx_free(&sc->sysctl_ctx); 51379251f5eSSepherosa Ziehau 51479251f5eSSepherosa Ziehau ix_free_intr(sc); 51579251f5eSSepherosa Ziehau 516*189a0ff3SSepherosa Ziehau if (sc->msix_mem_res != NULL) { 517*189a0ff3SSepherosa Ziehau bus_release_resource(dev, SYS_RES_MEMORY, sc->msix_mem_rid, 518*189a0ff3SSepherosa Ziehau sc->msix_mem_res); 519*189a0ff3SSepherosa Ziehau } 52079251f5eSSepherosa Ziehau if (sc->mem_res != NULL) { 52179251f5eSSepherosa Ziehau bus_release_resource(dev, SYS_RES_MEMORY, sc->mem_rid, 52279251f5eSSepherosa Ziehau sc->mem_res); 52379251f5eSSepherosa Ziehau } 52479251f5eSSepherosa Ziehau 52579251f5eSSepherosa Ziehau ix_free_rings(sc); 52679251f5eSSepherosa Ziehau 52779251f5eSSepherosa Ziehau if (sc->mta != NULL) 52879251f5eSSepherosa Ziehau kfree(sc->mta, M_DEVBUF); 52979251f5eSSepherosa Ziehau if (sc->serializes != NULL) 53079251f5eSSepherosa Ziehau kfree(sc->serializes, M_DEVBUF); 53179251f5eSSepherosa Ziehau 53279251f5eSSepherosa Ziehau return 0; 53379251f5eSSepherosa Ziehau } 53479251f5eSSepherosa Ziehau 53579251f5eSSepherosa Ziehau static int 53679251f5eSSepherosa Ziehau ix_shutdown(device_t dev) 53779251f5eSSepherosa Ziehau { 53879251f5eSSepherosa Ziehau struct ix_softc *sc = device_get_softc(dev); 53979251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 54079251f5eSSepherosa Ziehau 54179251f5eSSepherosa Ziehau ifnet_serialize_all(ifp); 54279251f5eSSepherosa Ziehau ix_stop(sc); 54379251f5eSSepherosa Ziehau ifnet_deserialize_all(ifp); 54479251f5eSSepherosa Ziehau 54579251f5eSSepherosa Ziehau return 0; 54679251f5eSSepherosa Ziehau } 54779251f5eSSepherosa Ziehau 54879251f5eSSepherosa Ziehau static void 54979251f5eSSepherosa Ziehau ix_start(struct ifnet *ifp, struct ifaltq_subque *ifsq) 55079251f5eSSepherosa Ziehau { 55179251f5eSSepherosa Ziehau struct ix_softc *sc = ifp->if_softc; 55279251f5eSSepherosa Ziehau struct ix_tx_ring *txr = ifsq_get_priv(ifsq); 55379251f5eSSepherosa Ziehau int idx = -1; 55479251f5eSSepherosa Ziehau uint16_t nsegs; 55579251f5eSSepherosa Ziehau 55679251f5eSSepherosa Ziehau KKASSERT(txr->tx_ifsq == ifsq); 55779251f5eSSepherosa Ziehau ASSERT_SERIALIZED(&txr->tx_serialize); 55879251f5eSSepherosa Ziehau 55979251f5eSSepherosa Ziehau if ((ifp->if_flags & IFF_RUNNING) == 0 || ifsq_is_oactive(ifsq)) 56079251f5eSSepherosa Ziehau return; 56179251f5eSSepherosa Ziehau 56279251f5eSSepherosa Ziehau if (!sc->link_active) { 56379251f5eSSepherosa Ziehau ifsq_purge(ifsq); 56479251f5eSSepherosa Ziehau return; 56579251f5eSSepherosa Ziehau } 56679251f5eSSepherosa Ziehau 56779251f5eSSepherosa Ziehau while (!ifsq_is_empty(ifsq)) { 56879251f5eSSepherosa Ziehau struct mbuf *m_head; 56979251f5eSSepherosa Ziehau 57079251f5eSSepherosa Ziehau if (txr->tx_avail <= IX_MAX_SCATTER + IX_TX_RESERVED) { 57179251f5eSSepherosa Ziehau ifsq_set_oactive(ifsq); 57279251f5eSSepherosa Ziehau txr->tx_watchdog.wd_timer = 5; 57379251f5eSSepherosa Ziehau break; 57479251f5eSSepherosa Ziehau } 57579251f5eSSepherosa Ziehau 57679251f5eSSepherosa Ziehau m_head = ifsq_dequeue(ifsq); 57779251f5eSSepherosa Ziehau if (m_head == NULL) 57879251f5eSSepherosa Ziehau break; 57979251f5eSSepherosa Ziehau 58079251f5eSSepherosa Ziehau if (ix_encap(txr, &m_head, &nsegs, &idx)) { 58179251f5eSSepherosa Ziehau IFNET_STAT_INC(ifp, oerrors, 1); 58279251f5eSSepherosa Ziehau continue; 58379251f5eSSepherosa Ziehau } 58479251f5eSSepherosa Ziehau 58579251f5eSSepherosa Ziehau if (nsegs >= txr->tx_wreg_nsegs) { 58679251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_TDT(txr->tx_idx), idx); 58779251f5eSSepherosa Ziehau nsegs = 0; 58879251f5eSSepherosa Ziehau idx = -1; 58979251f5eSSepherosa Ziehau } 59079251f5eSSepherosa Ziehau 59179251f5eSSepherosa Ziehau ETHER_BPF_MTAP(ifp, m_head); 59279251f5eSSepherosa Ziehau } 59379251f5eSSepherosa Ziehau if (idx >= 0) 59479251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_TDT(txr->tx_idx), idx); 59579251f5eSSepherosa Ziehau } 59679251f5eSSepherosa Ziehau 59779251f5eSSepherosa Ziehau static int 59879251f5eSSepherosa Ziehau ix_ioctl(struct ifnet *ifp, u_long command, caddr_t data, struct ucred *cr) 59979251f5eSSepherosa Ziehau { 60079251f5eSSepherosa Ziehau struct ix_softc *sc = ifp->if_softc; 60179251f5eSSepherosa Ziehau struct ifreq *ifr = (struct ifreq *) data; 60279251f5eSSepherosa Ziehau int error = 0, mask, reinit; 60379251f5eSSepherosa Ziehau 60479251f5eSSepherosa Ziehau ASSERT_IFNET_SERIALIZED_ALL(ifp); 60579251f5eSSepherosa Ziehau 60679251f5eSSepherosa Ziehau switch (command) { 60779251f5eSSepherosa Ziehau case SIOCSIFMTU: 60879251f5eSSepherosa Ziehau if (ifr->ifr_mtu > IX_MAX_FRAME_SIZE - ETHER_HDR_LEN) { 60979251f5eSSepherosa Ziehau error = EINVAL; 61079251f5eSSepherosa Ziehau } else { 61179251f5eSSepherosa Ziehau ifp->if_mtu = ifr->ifr_mtu; 61279251f5eSSepherosa Ziehau sc->max_frame_size = 61379251f5eSSepherosa Ziehau ifp->if_mtu + ETHER_HDR_LEN + ETHER_CRC_LEN; 61479251f5eSSepherosa Ziehau ix_init(sc); 61579251f5eSSepherosa Ziehau } 61679251f5eSSepherosa Ziehau break; 61779251f5eSSepherosa Ziehau 61879251f5eSSepherosa Ziehau case SIOCSIFFLAGS: 61979251f5eSSepherosa Ziehau if (ifp->if_flags & IFF_UP) { 62079251f5eSSepherosa Ziehau if (ifp->if_flags & IFF_RUNNING) { 62179251f5eSSepherosa Ziehau if ((ifp->if_flags ^ sc->if_flags) & 62279251f5eSSepherosa Ziehau (IFF_PROMISC | IFF_ALLMULTI)) 62379251f5eSSepherosa Ziehau ix_set_promisc(sc); 62479251f5eSSepherosa Ziehau } else { 62579251f5eSSepherosa Ziehau ix_init(sc); 62679251f5eSSepherosa Ziehau } 62779251f5eSSepherosa Ziehau } else if (ifp->if_flags & IFF_RUNNING) { 62879251f5eSSepherosa Ziehau ix_stop(sc); 62979251f5eSSepherosa Ziehau } 63079251f5eSSepherosa Ziehau sc->if_flags = ifp->if_flags; 63179251f5eSSepherosa Ziehau break; 63279251f5eSSepherosa Ziehau 63379251f5eSSepherosa Ziehau case SIOCADDMULTI: 63479251f5eSSepherosa Ziehau case SIOCDELMULTI: 63579251f5eSSepherosa Ziehau if (ifp->if_flags & IFF_RUNNING) { 63679251f5eSSepherosa Ziehau ix_disable_intr(sc); 63779251f5eSSepherosa Ziehau ix_set_multi(sc); 63879251f5eSSepherosa Ziehau ix_enable_intr(sc); 63979251f5eSSepherosa Ziehau } 64079251f5eSSepherosa Ziehau break; 64179251f5eSSepherosa Ziehau 64279251f5eSSepherosa Ziehau case SIOCSIFMEDIA: 64379251f5eSSepherosa Ziehau case SIOCGIFMEDIA: 64479251f5eSSepherosa Ziehau error = ifmedia_ioctl(ifp, ifr, &sc->media, command); 64579251f5eSSepherosa Ziehau break; 64679251f5eSSepherosa Ziehau 64779251f5eSSepherosa Ziehau case SIOCSIFCAP: 64879251f5eSSepherosa Ziehau reinit = 0; 64979251f5eSSepherosa Ziehau mask = ifr->ifr_reqcap ^ ifp->if_capenable; 65079251f5eSSepherosa Ziehau if (mask & IFCAP_RXCSUM) { 65179251f5eSSepherosa Ziehau ifp->if_capenable ^= IFCAP_RXCSUM; 65279251f5eSSepherosa Ziehau reinit = 1; 65379251f5eSSepherosa Ziehau } 65479251f5eSSepherosa Ziehau if (mask & IFCAP_VLAN_HWTAGGING) { 65579251f5eSSepherosa Ziehau ifp->if_capenable ^= IFCAP_VLAN_HWTAGGING; 65679251f5eSSepherosa Ziehau reinit = 1; 65779251f5eSSepherosa Ziehau } 65879251f5eSSepherosa Ziehau if (mask & IFCAP_TXCSUM) { 65979251f5eSSepherosa Ziehau ifp->if_capenable ^= IFCAP_TXCSUM; 66079251f5eSSepherosa Ziehau if (ifp->if_capenable & IFCAP_TXCSUM) 66179251f5eSSepherosa Ziehau ifp->if_hwassist |= CSUM_OFFLOAD; 66279251f5eSSepherosa Ziehau else 66379251f5eSSepherosa Ziehau ifp->if_hwassist &= ~CSUM_OFFLOAD; 66479251f5eSSepherosa Ziehau } 66579251f5eSSepherosa Ziehau if (mask & IFCAP_TSO) { 66679251f5eSSepherosa Ziehau ifp->if_capenable ^= IFCAP_TSO; 66779251f5eSSepherosa Ziehau if (ifp->if_capenable & IFCAP_TSO) 66879251f5eSSepherosa Ziehau ifp->if_hwassist |= CSUM_TSO; 66979251f5eSSepherosa Ziehau else 67079251f5eSSepherosa Ziehau ifp->if_hwassist &= ~CSUM_TSO; 67179251f5eSSepherosa Ziehau } 67279251f5eSSepherosa Ziehau if (mask & IFCAP_RSS) 67379251f5eSSepherosa Ziehau ifp->if_capenable ^= IFCAP_RSS; 67479251f5eSSepherosa Ziehau if (reinit && (ifp->if_flags & IFF_RUNNING)) 67579251f5eSSepherosa Ziehau ix_init(sc); 67679251f5eSSepherosa Ziehau break; 67779251f5eSSepherosa Ziehau 67879251f5eSSepherosa Ziehau #if 0 67979251f5eSSepherosa Ziehau case SIOCGI2C: 68079251f5eSSepherosa Ziehau { 68179251f5eSSepherosa Ziehau struct ixgbe_i2c_req i2c; 68279251f5eSSepherosa Ziehau error = copyin(ifr->ifr_data, &i2c, sizeof(i2c)); 68379251f5eSSepherosa Ziehau if (error) 68479251f5eSSepherosa Ziehau break; 68579251f5eSSepherosa Ziehau if ((i2c.dev_addr != 0xA0) || (i2c.dev_addr != 0xA2)){ 68679251f5eSSepherosa Ziehau error = EINVAL; 68779251f5eSSepherosa Ziehau break; 68879251f5eSSepherosa Ziehau } 68979251f5eSSepherosa Ziehau hw->phy.ops.read_i2c_byte(hw, i2c.offset, 69079251f5eSSepherosa Ziehau i2c.dev_addr, i2c.data); 69179251f5eSSepherosa Ziehau error = copyout(&i2c, ifr->ifr_data, sizeof(i2c)); 69279251f5eSSepherosa Ziehau break; 69379251f5eSSepherosa Ziehau } 69479251f5eSSepherosa Ziehau #endif 69579251f5eSSepherosa Ziehau 69679251f5eSSepherosa Ziehau default: 69779251f5eSSepherosa Ziehau error = ether_ioctl(ifp, command, data); 69879251f5eSSepherosa Ziehau break; 69979251f5eSSepherosa Ziehau } 70079251f5eSSepherosa Ziehau return error; 70179251f5eSSepherosa Ziehau } 70279251f5eSSepherosa Ziehau 70379251f5eSSepherosa Ziehau #define IXGBE_MHADD_MFS_SHIFT 16 70479251f5eSSepherosa Ziehau 70579251f5eSSepherosa Ziehau static void 70679251f5eSSepherosa Ziehau ix_init(void *xsc) 70779251f5eSSepherosa Ziehau { 70879251f5eSSepherosa Ziehau struct ix_softc *sc = xsc; 70979251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 71079251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 71179251f5eSSepherosa Ziehau uint32_t rxpb, frame, size, tmp; 71279251f5eSSepherosa Ziehau uint32_t gpie, rxctrl; 71379251f5eSSepherosa Ziehau int i, error; 71479251f5eSSepherosa Ziehau 71579251f5eSSepherosa Ziehau ASSERT_IFNET_SERIALIZED_ALL(ifp); 71679251f5eSSepherosa Ziehau 71779251f5eSSepherosa Ziehau ix_stop(sc); 71879251f5eSSepherosa Ziehau 71979251f5eSSepherosa Ziehau /* Configure # of used RX/TX rings */ 72079251f5eSSepherosa Ziehau ix_set_ring_inuse(sc, FALSE); 72179251f5eSSepherosa Ziehau ifq_set_subq_mask(&ifp->if_snd, sc->tx_ring_inuse - 1); 72279251f5eSSepherosa Ziehau 72379251f5eSSepherosa Ziehau /* Get the latest mac address, User can use a LAA */ 72479251f5eSSepherosa Ziehau bcopy(IF_LLADDR(ifp), hw->mac.addr, IXGBE_ETH_LENGTH_OF_ADDRESS); 72579251f5eSSepherosa Ziehau ixgbe_set_rar(hw, 0, hw->mac.addr, 0, 1); 72679251f5eSSepherosa Ziehau hw->addr_ctrl.rar_used_count = 1; 72779251f5eSSepherosa Ziehau 72879251f5eSSepherosa Ziehau /* Prepare transmit descriptors and buffers */ 72979251f5eSSepherosa Ziehau for (i = 0; i < sc->tx_ring_inuse; ++i) 73079251f5eSSepherosa Ziehau ix_init_tx_ring(&sc->tx_rings[i]); 73179251f5eSSepherosa Ziehau 73279251f5eSSepherosa Ziehau ixgbe_init_hw(hw); 73379251f5eSSepherosa Ziehau ix_init_tx_unit(sc); 73479251f5eSSepherosa Ziehau 73579251f5eSSepherosa Ziehau /* Setup Multicast table */ 73679251f5eSSepherosa Ziehau ix_set_multi(sc); 73779251f5eSSepherosa Ziehau 73879251f5eSSepherosa Ziehau /* Prepare receive descriptors and buffers */ 73979251f5eSSepherosa Ziehau for (i = 0; i < sc->rx_ring_inuse; ++i) { 74079251f5eSSepherosa Ziehau error = ix_init_rx_ring(&sc->rx_rings[i]); 74179251f5eSSepherosa Ziehau if (error) { 74279251f5eSSepherosa Ziehau if_printf(ifp, "Could not initialize RX ring%d\n", i); 74379251f5eSSepherosa Ziehau ix_stop(sc); 74479251f5eSSepherosa Ziehau return; 74579251f5eSSepherosa Ziehau } 74679251f5eSSepherosa Ziehau } 74779251f5eSSepherosa Ziehau 74879251f5eSSepherosa Ziehau /* Configure RX settings */ 74979251f5eSSepherosa Ziehau ix_init_rx_unit(sc); 75079251f5eSSepherosa Ziehau 75179251f5eSSepherosa Ziehau gpie = IXGBE_READ_REG(hw, IXGBE_GPIE); 75279251f5eSSepherosa Ziehau 75379251f5eSSepherosa Ziehau /* Enable Fan Failure Interrupt */ 75479251f5eSSepherosa Ziehau gpie |= IXGBE_SDP1_GPIEN; 75579251f5eSSepherosa Ziehau 75679251f5eSSepherosa Ziehau /* Add for Module detection */ 75779251f5eSSepherosa Ziehau if (hw->mac.type == ixgbe_mac_82599EB) 75879251f5eSSepherosa Ziehau gpie |= IXGBE_SDP2_GPIEN; 75979251f5eSSepherosa Ziehau 76079251f5eSSepherosa Ziehau /* Thermal Failure Detection */ 76179251f5eSSepherosa Ziehau if (hw->mac.type == ixgbe_mac_X540) 76279251f5eSSepherosa Ziehau gpie |= IXGBE_SDP0_GPIEN; 76379251f5eSSepherosa Ziehau 76479251f5eSSepherosa Ziehau if (sc->intr_type == PCI_INTR_TYPE_MSIX) { 76579251f5eSSepherosa Ziehau /* Enable Enhanced MSIX mode */ 76679251f5eSSepherosa Ziehau gpie |= IXGBE_GPIE_MSIX_MODE; 76779251f5eSSepherosa Ziehau gpie |= IXGBE_GPIE_EIAME | IXGBE_GPIE_PBA_SUPPORT | 76879251f5eSSepherosa Ziehau IXGBE_GPIE_OCD; 76979251f5eSSepherosa Ziehau } 77079251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_GPIE, gpie); 77179251f5eSSepherosa Ziehau 77279251f5eSSepherosa Ziehau /* Set MTU size */ 77379251f5eSSepherosa Ziehau if (ifp->if_mtu > ETHERMTU) { 77479251f5eSSepherosa Ziehau uint32_t mhadd; 77579251f5eSSepherosa Ziehau 77679251f5eSSepherosa Ziehau mhadd = IXGBE_READ_REG(hw, IXGBE_MHADD); 77779251f5eSSepherosa Ziehau mhadd &= ~IXGBE_MHADD_MFS_MASK; 77879251f5eSSepherosa Ziehau mhadd |= sc->max_frame_size << IXGBE_MHADD_MFS_SHIFT; 77979251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_MHADD, mhadd); 78079251f5eSSepherosa Ziehau } 78179251f5eSSepherosa Ziehau 78279251f5eSSepherosa Ziehau /* 78379251f5eSSepherosa Ziehau * Enable TX rings 78479251f5eSSepherosa Ziehau */ 78579251f5eSSepherosa Ziehau for (i = 0; i < sc->tx_ring_inuse; ++i) { 78679251f5eSSepherosa Ziehau uint32_t txdctl; 78779251f5eSSepherosa Ziehau 78879251f5eSSepherosa Ziehau txdctl = IXGBE_READ_REG(hw, IXGBE_TXDCTL(i)); 78979251f5eSSepherosa Ziehau txdctl |= IXGBE_TXDCTL_ENABLE; 79079251f5eSSepherosa Ziehau 79179251f5eSSepherosa Ziehau /* 79279251f5eSSepherosa Ziehau * Set WTHRESH to 0, since TX head write-back is used 79379251f5eSSepherosa Ziehau */ 79479251f5eSSepherosa Ziehau txdctl &= ~(0x7f << 16); 79579251f5eSSepherosa Ziehau 79679251f5eSSepherosa Ziehau /* 79779251f5eSSepherosa Ziehau * When the internal queue falls below PTHRESH (32), 79879251f5eSSepherosa Ziehau * start prefetching as long as there are at least 79979251f5eSSepherosa Ziehau * HTHRESH (1) buffers ready. The values are taken 80079251f5eSSepherosa Ziehau * from the Intel linux driver 3.8.21. 80179251f5eSSepherosa Ziehau * Prefetching enables tx line rate even with 1 queue. 80279251f5eSSepherosa Ziehau */ 80379251f5eSSepherosa Ziehau txdctl |= (32 << 0) | (1 << 8); 80479251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_TXDCTL(i), txdctl); 80579251f5eSSepherosa Ziehau } 80679251f5eSSepherosa Ziehau 80779251f5eSSepherosa Ziehau /* 80879251f5eSSepherosa Ziehau * Enable RX rings 80979251f5eSSepherosa Ziehau */ 81079251f5eSSepherosa Ziehau for (i = 0; i < sc->rx_ring_inuse; ++i) { 81179251f5eSSepherosa Ziehau uint32_t rxdctl; 81279251f5eSSepherosa Ziehau int k; 81379251f5eSSepherosa Ziehau 81479251f5eSSepherosa Ziehau rxdctl = IXGBE_READ_REG(hw, IXGBE_RXDCTL(i)); 81579251f5eSSepherosa Ziehau if (hw->mac.type == ixgbe_mac_82598EB) { 81679251f5eSSepherosa Ziehau /* 81779251f5eSSepherosa Ziehau * PTHRESH = 21 81879251f5eSSepherosa Ziehau * HTHRESH = 4 81979251f5eSSepherosa Ziehau * WTHRESH = 8 82079251f5eSSepherosa Ziehau */ 82179251f5eSSepherosa Ziehau rxdctl &= ~0x3FFFFF; 82279251f5eSSepherosa Ziehau rxdctl |= 0x080420; 82379251f5eSSepherosa Ziehau } 82479251f5eSSepherosa Ziehau rxdctl |= IXGBE_RXDCTL_ENABLE; 82579251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RXDCTL(i), rxdctl); 82679251f5eSSepherosa Ziehau for (k = 0; k < 10; ++k) { 82779251f5eSSepherosa Ziehau if (IXGBE_READ_REG(hw, IXGBE_RXDCTL(i)) & 82879251f5eSSepherosa Ziehau IXGBE_RXDCTL_ENABLE) 82979251f5eSSepherosa Ziehau break; 83079251f5eSSepherosa Ziehau else 83179251f5eSSepherosa Ziehau msec_delay(1); 83279251f5eSSepherosa Ziehau } 83379251f5eSSepherosa Ziehau wmb(); 83479251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RDT(i), 83579251f5eSSepherosa Ziehau sc->rx_rings[0].rx_ndesc - 1); 83679251f5eSSepherosa Ziehau } 83779251f5eSSepherosa Ziehau 83879251f5eSSepherosa Ziehau /* Set up VLAN support and filter */ 83979251f5eSSepherosa Ziehau ix_set_vlan(sc); 84079251f5eSSepherosa Ziehau 84179251f5eSSepherosa Ziehau /* Enable Receive engine */ 84279251f5eSSepherosa Ziehau rxctrl = IXGBE_READ_REG(hw, IXGBE_RXCTRL); 84379251f5eSSepherosa Ziehau if (hw->mac.type == ixgbe_mac_82598EB) 84479251f5eSSepherosa Ziehau rxctrl |= IXGBE_RXCTRL_DMBYPS; 84579251f5eSSepherosa Ziehau rxctrl |= IXGBE_RXCTRL_RXEN; 84679251f5eSSepherosa Ziehau ixgbe_enable_rx_dma(hw, rxctrl); 84779251f5eSSepherosa Ziehau 848*189a0ff3SSepherosa Ziehau for (i = 0; i < sc->tx_ring_inuse; ++i) { 849*189a0ff3SSepherosa Ziehau const struct ix_tx_ring *txr = &sc->tx_rings[i]; 850*189a0ff3SSepherosa Ziehau 851*189a0ff3SSepherosa Ziehau if (txr->tx_intr_vec >= 0) { 852*189a0ff3SSepherosa Ziehau ix_set_ivar(sc, i, txr->tx_intr_vec, 1); 853*189a0ff3SSepherosa Ziehau } else { 854*189a0ff3SSepherosa Ziehau /* 855*189a0ff3SSepherosa Ziehau * Unconfigured TX interrupt vector could only 856*189a0ff3SSepherosa Ziehau * happen for MSI-X. 857*189a0ff3SSepherosa Ziehau */ 858*189a0ff3SSepherosa Ziehau KASSERT(sc->intr_type == PCI_INTR_TYPE_MSIX, 859*189a0ff3SSepherosa Ziehau ("TX intr vector is not set")); 860*189a0ff3SSepherosa Ziehau KASSERT(i < sc->rx_ring_inuse, 861*189a0ff3SSepherosa Ziehau ("invalid TX ring %d, no piggyback RX ring", i)); 862*189a0ff3SSepherosa Ziehau KASSERT(sc->rx_rings[i].rx_txr == txr, 863*189a0ff3SSepherosa Ziehau ("RX ring %d piggybacked TX ring mismatch", i)); 864*189a0ff3SSepherosa Ziehau if (bootverbose) 865*189a0ff3SSepherosa Ziehau if_printf(ifp, "IVAR skips TX ring %d\n", i); 866*189a0ff3SSepherosa Ziehau } 867*189a0ff3SSepherosa Ziehau } 868*189a0ff3SSepherosa Ziehau for (i = 0; i < sc->rx_ring_inuse; ++i) { 869*189a0ff3SSepherosa Ziehau const struct ix_rx_ring *rxr = &sc->rx_rings[i]; 870*189a0ff3SSepherosa Ziehau 871*189a0ff3SSepherosa Ziehau KKASSERT(rxr->rx_intr_vec >= 0); 872*189a0ff3SSepherosa Ziehau ix_set_ivar(sc, i, rxr->rx_intr_vec, 0); 873*189a0ff3SSepherosa Ziehau if (rxr->rx_txr != NULL) { 874*189a0ff3SSepherosa Ziehau /* 875*189a0ff3SSepherosa Ziehau * Piggyback the TX ring interrupt onto the RX 876*189a0ff3SSepherosa Ziehau * ring interrupt vector. 877*189a0ff3SSepherosa Ziehau */ 878*189a0ff3SSepherosa Ziehau KASSERT(rxr->rx_txr->tx_intr_vec < 0, 879*189a0ff3SSepherosa Ziehau ("piggybacked TX ring configured intr vector")); 880*189a0ff3SSepherosa Ziehau KASSERT(rxr->rx_txr->tx_idx == i, 881*189a0ff3SSepherosa Ziehau ("RX ring %d piggybacked TX ring %u", 882*189a0ff3SSepherosa Ziehau i, rxr->rx_txr->tx_idx)); 883*189a0ff3SSepherosa Ziehau ix_set_ivar(sc, i, rxr->rx_intr_vec, 1); 884*189a0ff3SSepherosa Ziehau if (bootverbose) { 885*189a0ff3SSepherosa Ziehau if_printf(ifp, "IVAR RX ring %d piggybacks " 886*189a0ff3SSepherosa Ziehau "TX ring %u\n", i, rxr->rx_txr->tx_idx); 887*189a0ff3SSepherosa Ziehau } 888*189a0ff3SSepherosa Ziehau } 889*189a0ff3SSepherosa Ziehau } 89079251f5eSSepherosa Ziehau if (sc->intr_type == PCI_INTR_TYPE_MSIX) { 891*189a0ff3SSepherosa Ziehau /* Set up status MSI-X vector; it is using fixed entry 1 */ 892*189a0ff3SSepherosa Ziehau ix_set_ivar(sc, 1, sc->sts_msix_vec, -1); 893*189a0ff3SSepherosa Ziehau 894*189a0ff3SSepherosa Ziehau /* Set up auto-mask for TX and RX rings */ 895*189a0ff3SSepherosa Ziehau if (hw->mac.type == ixgbe_mac_82598EB) { 896*189a0ff3SSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_EIAM, IXGBE_EIMS_RTX_QUEUE); 897*189a0ff3SSepherosa Ziehau } else { 89879251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_EIAM_EX(0), 0xFFFFFFFF); 89979251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_EIAM_EX(1), 0xFFFFFFFF); 90079251f5eSSepherosa Ziehau } 90179251f5eSSepherosa Ziehau } else { 902*189a0ff3SSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_EIAM, IXGBE_EIMS_RTX_QUEUE); 90379251f5eSSepherosa Ziehau } 904*189a0ff3SSepherosa Ziehau for (i = 0; i < sc->intr_cnt; ++i) 905*189a0ff3SSepherosa Ziehau ix_set_eitr(sc, i, sc->intr_data[i].intr_rate); 90679251f5eSSepherosa Ziehau 90779251f5eSSepherosa Ziehau /* 90879251f5eSSepherosa Ziehau * Check on any SFP devices that need to be kick-started 90979251f5eSSepherosa Ziehau */ 91079251f5eSSepherosa Ziehau if (hw->phy.type == ixgbe_phy_none) { 91179251f5eSSepherosa Ziehau error = hw->phy.ops.identify(hw); 91279251f5eSSepherosa Ziehau if (error == IXGBE_ERR_SFP_NOT_SUPPORTED) { 91379251f5eSSepherosa Ziehau if_printf(ifp, 91479251f5eSSepherosa Ziehau "Unsupported SFP+ module type was detected.\n"); 91579251f5eSSepherosa Ziehau /* XXX stop */ 91679251f5eSSepherosa Ziehau return; 91779251f5eSSepherosa Ziehau } 91879251f5eSSepherosa Ziehau } 91979251f5eSSepherosa Ziehau 92079251f5eSSepherosa Ziehau /* Config/Enable Link */ 92179251f5eSSepherosa Ziehau ix_config_link(sc); 92279251f5eSSepherosa Ziehau 92379251f5eSSepherosa Ziehau /* 92479251f5eSSepherosa Ziehau * Hardware Packet Buffer & Flow Control setup 92579251f5eSSepherosa Ziehau */ 92679251f5eSSepherosa Ziehau frame = sc->max_frame_size; 92779251f5eSSepherosa Ziehau 92879251f5eSSepherosa Ziehau /* Calculate High Water */ 92979251f5eSSepherosa Ziehau if (hw->mac.type == ixgbe_mac_X540) 93079251f5eSSepherosa Ziehau tmp = IXGBE_DV_X540(frame, frame); 93179251f5eSSepherosa Ziehau else 93279251f5eSSepherosa Ziehau tmp = IXGBE_DV(frame, frame); 93379251f5eSSepherosa Ziehau size = IXGBE_BT2KB(tmp); 93479251f5eSSepherosa Ziehau rxpb = IXGBE_READ_REG(hw, IXGBE_RXPBSIZE(0)) >> 10; 93579251f5eSSepherosa Ziehau hw->fc.high_water[0] = rxpb - size; 93679251f5eSSepherosa Ziehau 93779251f5eSSepherosa Ziehau /* Now calculate Low Water */ 93879251f5eSSepherosa Ziehau if (hw->mac.type == ixgbe_mac_X540) 93979251f5eSSepherosa Ziehau tmp = IXGBE_LOW_DV_X540(frame); 94079251f5eSSepherosa Ziehau else 94179251f5eSSepherosa Ziehau tmp = IXGBE_LOW_DV(frame); 94279251f5eSSepherosa Ziehau hw->fc.low_water[0] = IXGBE_BT2KB(tmp); 94379251f5eSSepherosa Ziehau 94479251f5eSSepherosa Ziehau hw->fc.requested_mode = sc->fc; 94579251f5eSSepherosa Ziehau hw->fc.pause_time = IX_FC_PAUSE; 94679251f5eSSepherosa Ziehau hw->fc.send_xon = TRUE; 94779251f5eSSepherosa Ziehau 94879251f5eSSepherosa Ziehau /* Initialize the FC settings */ 94979251f5eSSepherosa Ziehau ixgbe_start_hw(hw); 95079251f5eSSepherosa Ziehau 95179251f5eSSepherosa Ziehau /* And now turn on interrupts */ 95279251f5eSSepherosa Ziehau ix_enable_intr(sc); 95379251f5eSSepherosa Ziehau 95479251f5eSSepherosa Ziehau ifp->if_flags |= IFF_RUNNING; 95579251f5eSSepherosa Ziehau for (i = 0; i < sc->tx_ring_inuse; ++i) { 95679251f5eSSepherosa Ziehau ifsq_clr_oactive(sc->tx_rings[i].tx_ifsq); 95779251f5eSSepherosa Ziehau ifsq_watchdog_start(&sc->tx_rings[i].tx_watchdog); 95879251f5eSSepherosa Ziehau } 95979251f5eSSepherosa Ziehau 96079251f5eSSepherosa Ziehau ix_set_timer_cpuid(sc, FALSE); 96179251f5eSSepherosa Ziehau callout_reset_bycpu(&sc->timer, hz, ix_timer, sc, sc->timer_cpuid); 96279251f5eSSepherosa Ziehau } 96379251f5eSSepherosa Ziehau 96479251f5eSSepherosa Ziehau static void 96579251f5eSSepherosa Ziehau ix_intr(void *xsc) 96679251f5eSSepherosa Ziehau { 96779251f5eSSepherosa Ziehau struct ix_softc *sc = xsc; 96879251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 96979251f5eSSepherosa Ziehau uint32_t eicr; 97079251f5eSSepherosa Ziehau 97179251f5eSSepherosa Ziehau ASSERT_SERIALIZED(&sc->main_serialize); 97279251f5eSSepherosa Ziehau 97379251f5eSSepherosa Ziehau eicr = IXGBE_READ_REG(hw, IXGBE_EICR); 97479251f5eSSepherosa Ziehau if (eicr == 0) { 97579251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_EIMS, sc->intr_mask); 97679251f5eSSepherosa Ziehau return; 97779251f5eSSepherosa Ziehau } 97879251f5eSSepherosa Ziehau 97979251f5eSSepherosa Ziehau if (eicr & IX_RX0_INTR_MASK) { 98079251f5eSSepherosa Ziehau struct ix_rx_ring *rxr = &sc->rx_rings[0]; 98179251f5eSSepherosa Ziehau 98279251f5eSSepherosa Ziehau lwkt_serialize_enter(&rxr->rx_serialize); 98379251f5eSSepherosa Ziehau ix_rxeof(rxr); 98479251f5eSSepherosa Ziehau lwkt_serialize_exit(&rxr->rx_serialize); 98579251f5eSSepherosa Ziehau } 98679251f5eSSepherosa Ziehau if (eicr & IX_RX1_INTR_MASK) { 98779251f5eSSepherosa Ziehau struct ix_rx_ring *rxr; 98879251f5eSSepherosa Ziehau 98979251f5eSSepherosa Ziehau KKASSERT(sc->rx_ring_inuse == IX_MIN_RXRING_RSS); 99079251f5eSSepherosa Ziehau rxr = &sc->rx_rings[1]; 99179251f5eSSepherosa Ziehau 99279251f5eSSepherosa Ziehau lwkt_serialize_enter(&rxr->rx_serialize); 99379251f5eSSepherosa Ziehau ix_rxeof(rxr); 99479251f5eSSepherosa Ziehau lwkt_serialize_exit(&rxr->rx_serialize); 99579251f5eSSepherosa Ziehau } 99679251f5eSSepherosa Ziehau 99779251f5eSSepherosa Ziehau if (eicr & IX_TX_INTR_MASK) { 99879251f5eSSepherosa Ziehau struct ix_tx_ring *txr = &sc->tx_rings[0]; 99979251f5eSSepherosa Ziehau 100079251f5eSSepherosa Ziehau lwkt_serialize_enter(&txr->tx_serialize); 1001*189a0ff3SSepherosa Ziehau ix_txeof(txr, *(txr->tx_hdr)); 100279251f5eSSepherosa Ziehau if (!ifsq_is_empty(txr->tx_ifsq)) 100379251f5eSSepherosa Ziehau ifsq_devstart(txr->tx_ifsq); 100479251f5eSSepherosa Ziehau lwkt_serialize_exit(&txr->tx_serialize); 100579251f5eSSepherosa Ziehau } 100679251f5eSSepherosa Ziehau 1007*189a0ff3SSepherosa Ziehau if (__predict_false(eicr & IX_EICR_STATUS)) 1008*189a0ff3SSepherosa Ziehau ix_intr_status(sc, eicr); 100979251f5eSSepherosa Ziehau 101079251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_EIMS, sc->intr_mask); 101179251f5eSSepherosa Ziehau } 101279251f5eSSepherosa Ziehau 101379251f5eSSepherosa Ziehau static void 101479251f5eSSepherosa Ziehau ix_media_status(struct ifnet *ifp, struct ifmediareq *ifmr) 101579251f5eSSepherosa Ziehau { 101679251f5eSSepherosa Ziehau struct ix_softc *sc = ifp->if_softc; 101779251f5eSSepherosa Ziehau 101879251f5eSSepherosa Ziehau ix_update_link_status(sc); 101979251f5eSSepherosa Ziehau 102079251f5eSSepherosa Ziehau ifmr->ifm_status = IFM_AVALID; 102179251f5eSSepherosa Ziehau ifmr->ifm_active = IFM_ETHER; 102279251f5eSSepherosa Ziehau 102379251f5eSSepherosa Ziehau if (!sc->link_active) 102479251f5eSSepherosa Ziehau return; 102579251f5eSSepherosa Ziehau 102679251f5eSSepherosa Ziehau ifmr->ifm_status |= IFM_ACTIVE; 102779251f5eSSepherosa Ziehau 102879251f5eSSepherosa Ziehau switch (sc->link_speed) { 102979251f5eSSepherosa Ziehau case IXGBE_LINK_SPEED_100_FULL: 103079251f5eSSepherosa Ziehau ifmr->ifm_active |= IFM_100_TX | IFM_FDX; 103179251f5eSSepherosa Ziehau break; 103279251f5eSSepherosa Ziehau case IXGBE_LINK_SPEED_1GB_FULL: 103379251f5eSSepherosa Ziehau ifmr->ifm_active |= IFM_1000_SX | IFM_FDX; 103479251f5eSSepherosa Ziehau break; 103579251f5eSSepherosa Ziehau case IXGBE_LINK_SPEED_10GB_FULL: 103679251f5eSSepherosa Ziehau ifmr->ifm_active |= sc->optics | IFM_FDX; 103779251f5eSSepherosa Ziehau break; 103879251f5eSSepherosa Ziehau } 103979251f5eSSepherosa Ziehau } 104079251f5eSSepherosa Ziehau 104179251f5eSSepherosa Ziehau static int 104279251f5eSSepherosa Ziehau ix_media_change(struct ifnet *ifp) 104379251f5eSSepherosa Ziehau { 104479251f5eSSepherosa Ziehau struct ix_softc *sc = ifp->if_softc; 104579251f5eSSepherosa Ziehau struct ifmedia *ifm = &sc->media; 104679251f5eSSepherosa Ziehau 104779251f5eSSepherosa Ziehau if (IFM_TYPE(ifm->ifm_media) != IFM_ETHER) 104879251f5eSSepherosa Ziehau return EINVAL; 104979251f5eSSepherosa Ziehau 105079251f5eSSepherosa Ziehau switch (IFM_SUBTYPE(ifm->ifm_media)) { 105179251f5eSSepherosa Ziehau case IFM_AUTO: 105279251f5eSSepherosa Ziehau sc->hw.phy.autoneg_advertised = 105379251f5eSSepherosa Ziehau IXGBE_LINK_SPEED_100_FULL | 105479251f5eSSepherosa Ziehau IXGBE_LINK_SPEED_1GB_FULL | 105579251f5eSSepherosa Ziehau IXGBE_LINK_SPEED_10GB_FULL; 105679251f5eSSepherosa Ziehau break; 105779251f5eSSepherosa Ziehau default: 105879251f5eSSepherosa Ziehau if_printf(ifp, "Only auto media type\n"); 105979251f5eSSepherosa Ziehau return EINVAL; 106079251f5eSSepherosa Ziehau } 106179251f5eSSepherosa Ziehau return 0; 106279251f5eSSepherosa Ziehau } 106379251f5eSSepherosa Ziehau 106479251f5eSSepherosa Ziehau static __inline int 106579251f5eSSepherosa Ziehau ix_tso_pullup(struct mbuf **mp) 106679251f5eSSepherosa Ziehau { 106779251f5eSSepherosa Ziehau int hoff, iphlen, thoff; 106879251f5eSSepherosa Ziehau struct mbuf *m; 106979251f5eSSepherosa Ziehau 107079251f5eSSepherosa Ziehau m = *mp; 107179251f5eSSepherosa Ziehau KASSERT(M_WRITABLE(m), ("TSO mbuf not writable")); 107279251f5eSSepherosa Ziehau 107379251f5eSSepherosa Ziehau iphlen = m->m_pkthdr.csum_iphlen; 107479251f5eSSepherosa Ziehau thoff = m->m_pkthdr.csum_thlen; 107579251f5eSSepherosa Ziehau hoff = m->m_pkthdr.csum_lhlen; 107679251f5eSSepherosa Ziehau 107779251f5eSSepherosa Ziehau KASSERT(iphlen > 0, ("invalid ip hlen")); 107879251f5eSSepherosa Ziehau KASSERT(thoff > 0, ("invalid tcp hlen")); 107979251f5eSSepherosa Ziehau KASSERT(hoff > 0, ("invalid ether hlen")); 108079251f5eSSepherosa Ziehau 108179251f5eSSepherosa Ziehau if (__predict_false(m->m_len < hoff + iphlen + thoff)) { 108279251f5eSSepherosa Ziehau m = m_pullup(m, hoff + iphlen + thoff); 108379251f5eSSepherosa Ziehau if (m == NULL) { 108479251f5eSSepherosa Ziehau *mp = NULL; 108579251f5eSSepherosa Ziehau return ENOBUFS; 108679251f5eSSepherosa Ziehau } 108779251f5eSSepherosa Ziehau *mp = m; 108879251f5eSSepherosa Ziehau } 108979251f5eSSepherosa Ziehau return 0; 109079251f5eSSepherosa Ziehau } 109179251f5eSSepherosa Ziehau 109279251f5eSSepherosa Ziehau static int 109379251f5eSSepherosa Ziehau ix_encap(struct ix_tx_ring *txr, struct mbuf **m_headp, 109479251f5eSSepherosa Ziehau uint16_t *segs_used, int *idx) 109579251f5eSSepherosa Ziehau { 109679251f5eSSepherosa Ziehau uint32_t olinfo_status = 0, cmd_type_len, cmd_rs = 0; 109779251f5eSSepherosa Ziehau int i, j, error, nsegs, first, maxsegs; 109879251f5eSSepherosa Ziehau struct mbuf *m_head = *m_headp; 109979251f5eSSepherosa Ziehau bus_dma_segment_t segs[IX_MAX_SCATTER]; 110079251f5eSSepherosa Ziehau bus_dmamap_t map; 110179251f5eSSepherosa Ziehau struct ix_tx_buf *txbuf; 110279251f5eSSepherosa Ziehau union ixgbe_adv_tx_desc *txd = NULL; 110379251f5eSSepherosa Ziehau 110479251f5eSSepherosa Ziehau if (m_head->m_pkthdr.csum_flags & CSUM_TSO) { 110579251f5eSSepherosa Ziehau error = ix_tso_pullup(m_headp); 110679251f5eSSepherosa Ziehau if (__predict_false(error)) 110779251f5eSSepherosa Ziehau return error; 110879251f5eSSepherosa Ziehau m_head = *m_headp; 110979251f5eSSepherosa Ziehau } 111079251f5eSSepherosa Ziehau 111179251f5eSSepherosa Ziehau /* Basic descriptor defines */ 111279251f5eSSepherosa Ziehau cmd_type_len = (IXGBE_ADVTXD_DTYP_DATA | 111379251f5eSSepherosa Ziehau IXGBE_ADVTXD_DCMD_IFCS | IXGBE_ADVTXD_DCMD_DEXT); 111479251f5eSSepherosa Ziehau 111579251f5eSSepherosa Ziehau if (m_head->m_flags & M_VLANTAG) 111679251f5eSSepherosa Ziehau cmd_type_len |= IXGBE_ADVTXD_DCMD_VLE; 111779251f5eSSepherosa Ziehau 111879251f5eSSepherosa Ziehau /* 111979251f5eSSepherosa Ziehau * Important to capture the first descriptor 112079251f5eSSepherosa Ziehau * used because it will contain the index of 112179251f5eSSepherosa Ziehau * the one we tell the hardware to report back 112279251f5eSSepherosa Ziehau */ 112379251f5eSSepherosa Ziehau first = txr->tx_next_avail; 112479251f5eSSepherosa Ziehau txbuf = &txr->tx_buf[first]; 112579251f5eSSepherosa Ziehau map = txbuf->map; 112679251f5eSSepherosa Ziehau 112779251f5eSSepherosa Ziehau /* 112879251f5eSSepherosa Ziehau * Map the packet for DMA. 112979251f5eSSepherosa Ziehau */ 113079251f5eSSepherosa Ziehau maxsegs = txr->tx_avail - IX_TX_RESERVED; 113179251f5eSSepherosa Ziehau if (maxsegs > IX_MAX_SCATTER) 113279251f5eSSepherosa Ziehau maxsegs = IX_MAX_SCATTER; 113379251f5eSSepherosa Ziehau 113479251f5eSSepherosa Ziehau error = bus_dmamap_load_mbuf_defrag(txr->tx_tag, map, m_headp, 113579251f5eSSepherosa Ziehau segs, maxsegs, &nsegs, BUS_DMA_NOWAIT); 113679251f5eSSepherosa Ziehau if (__predict_false(error)) { 113779251f5eSSepherosa Ziehau m_freem(*m_headp); 113879251f5eSSepherosa Ziehau *m_headp = NULL; 113979251f5eSSepherosa Ziehau return error; 114079251f5eSSepherosa Ziehau } 114179251f5eSSepherosa Ziehau bus_dmamap_sync(txr->tx_tag, map, BUS_DMASYNC_PREWRITE); 114279251f5eSSepherosa Ziehau 114379251f5eSSepherosa Ziehau m_head = *m_headp; 114479251f5eSSepherosa Ziehau 114579251f5eSSepherosa Ziehau /* 114679251f5eSSepherosa Ziehau * Set up the appropriate offload context if requested, 114779251f5eSSepherosa Ziehau * this may consume one TX descriptor. 114879251f5eSSepherosa Ziehau */ 114979251f5eSSepherosa Ziehau if (ix_tx_ctx_setup(txr, m_head, &cmd_type_len, &olinfo_status)) { 115079251f5eSSepherosa Ziehau (*segs_used)++; 115179251f5eSSepherosa Ziehau txr->tx_nsegs++; 115279251f5eSSepherosa Ziehau } 115379251f5eSSepherosa Ziehau 115479251f5eSSepherosa Ziehau *segs_used += nsegs; 115579251f5eSSepherosa Ziehau txr->tx_nsegs += nsegs; 115679251f5eSSepherosa Ziehau if (txr->tx_nsegs >= txr->tx_intr_nsegs) { 115779251f5eSSepherosa Ziehau /* 115879251f5eSSepherosa Ziehau * Report Status (RS) is turned on every intr_nsegs 115979251f5eSSepherosa Ziehau * descriptors (roughly). 116079251f5eSSepherosa Ziehau */ 116179251f5eSSepherosa Ziehau txr->tx_nsegs = 0; 116279251f5eSSepherosa Ziehau cmd_rs = IXGBE_TXD_CMD_RS; 116379251f5eSSepherosa Ziehau } 116479251f5eSSepherosa Ziehau 116579251f5eSSepherosa Ziehau i = txr->tx_next_avail; 116679251f5eSSepherosa Ziehau for (j = 0; j < nsegs; j++) { 116779251f5eSSepherosa Ziehau bus_size_t seglen; 116879251f5eSSepherosa Ziehau bus_addr_t segaddr; 116979251f5eSSepherosa Ziehau 117079251f5eSSepherosa Ziehau txbuf = &txr->tx_buf[i]; 117179251f5eSSepherosa Ziehau txd = &txr->tx_base[i]; 117279251f5eSSepherosa Ziehau seglen = segs[j].ds_len; 117379251f5eSSepherosa Ziehau segaddr = htole64(segs[j].ds_addr); 117479251f5eSSepherosa Ziehau 117579251f5eSSepherosa Ziehau txd->read.buffer_addr = segaddr; 117679251f5eSSepherosa Ziehau txd->read.cmd_type_len = htole32(IXGBE_TXD_CMD_IFCS | 117779251f5eSSepherosa Ziehau cmd_type_len |seglen); 117879251f5eSSepherosa Ziehau txd->read.olinfo_status = htole32(olinfo_status); 117979251f5eSSepherosa Ziehau 118079251f5eSSepherosa Ziehau if (++i == txr->tx_ndesc) 118179251f5eSSepherosa Ziehau i = 0; 118279251f5eSSepherosa Ziehau } 118379251f5eSSepherosa Ziehau txd->read.cmd_type_len |= htole32(IXGBE_TXD_CMD_EOP | cmd_rs); 118479251f5eSSepherosa Ziehau 118579251f5eSSepherosa Ziehau txr->tx_avail -= nsegs; 118679251f5eSSepherosa Ziehau txr->tx_next_avail = i; 118779251f5eSSepherosa Ziehau 118879251f5eSSepherosa Ziehau txbuf->m_head = m_head; 118979251f5eSSepherosa Ziehau txr->tx_buf[first].map = txbuf->map; 119079251f5eSSepherosa Ziehau txbuf->map = map; 119179251f5eSSepherosa Ziehau 119279251f5eSSepherosa Ziehau /* 119379251f5eSSepherosa Ziehau * Defer TDT updating, until enough descrptors are setup 119479251f5eSSepherosa Ziehau */ 119579251f5eSSepherosa Ziehau *idx = i; 119679251f5eSSepherosa Ziehau 119779251f5eSSepherosa Ziehau return 0; 119879251f5eSSepherosa Ziehau } 119979251f5eSSepherosa Ziehau 120079251f5eSSepherosa Ziehau static void 120179251f5eSSepherosa Ziehau ix_set_promisc(struct ix_softc *sc) 120279251f5eSSepherosa Ziehau { 120379251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 120479251f5eSSepherosa Ziehau uint32_t reg_rctl; 120579251f5eSSepherosa Ziehau int mcnt = 0; 120679251f5eSSepherosa Ziehau 120779251f5eSSepherosa Ziehau reg_rctl = IXGBE_READ_REG(&sc->hw, IXGBE_FCTRL); 120879251f5eSSepherosa Ziehau reg_rctl &= ~IXGBE_FCTRL_UPE; 120979251f5eSSepherosa Ziehau if (ifp->if_flags & IFF_ALLMULTI) { 121079251f5eSSepherosa Ziehau mcnt = IX_MAX_MCASTADDR; 121179251f5eSSepherosa Ziehau } else { 121279251f5eSSepherosa Ziehau struct ifmultiaddr *ifma; 121379251f5eSSepherosa Ziehau 121479251f5eSSepherosa Ziehau TAILQ_FOREACH(ifma, &ifp->if_multiaddrs, ifma_link) { 121579251f5eSSepherosa Ziehau if (ifma->ifma_addr->sa_family != AF_LINK) 121679251f5eSSepherosa Ziehau continue; 121779251f5eSSepherosa Ziehau if (mcnt == IX_MAX_MCASTADDR) 121879251f5eSSepherosa Ziehau break; 121979251f5eSSepherosa Ziehau mcnt++; 122079251f5eSSepherosa Ziehau } 122179251f5eSSepherosa Ziehau } 122279251f5eSSepherosa Ziehau if (mcnt < IX_MAX_MCASTADDR) 122379251f5eSSepherosa Ziehau reg_rctl &= ~IXGBE_FCTRL_MPE; 122479251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_FCTRL, reg_rctl); 122579251f5eSSepherosa Ziehau 122679251f5eSSepherosa Ziehau if (ifp->if_flags & IFF_PROMISC) { 122779251f5eSSepherosa Ziehau reg_rctl |= IXGBE_FCTRL_UPE | IXGBE_FCTRL_MPE; 122879251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_FCTRL, reg_rctl); 122979251f5eSSepherosa Ziehau } else if (ifp->if_flags & IFF_ALLMULTI) { 123079251f5eSSepherosa Ziehau reg_rctl |= IXGBE_FCTRL_MPE; 123179251f5eSSepherosa Ziehau reg_rctl &= ~IXGBE_FCTRL_UPE; 123279251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_FCTRL, reg_rctl); 123379251f5eSSepherosa Ziehau } 123479251f5eSSepherosa Ziehau } 123579251f5eSSepherosa Ziehau 123679251f5eSSepherosa Ziehau static void 123779251f5eSSepherosa Ziehau ix_set_multi(struct ix_softc *sc) 123879251f5eSSepherosa Ziehau { 123979251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 124079251f5eSSepherosa Ziehau struct ifmultiaddr *ifma; 124179251f5eSSepherosa Ziehau uint32_t fctrl; 124279251f5eSSepherosa Ziehau uint8_t *mta; 124379251f5eSSepherosa Ziehau int mcnt = 0; 124479251f5eSSepherosa Ziehau 124579251f5eSSepherosa Ziehau mta = sc->mta; 124679251f5eSSepherosa Ziehau bzero(mta, IXGBE_ETH_LENGTH_OF_ADDRESS * IX_MAX_MCASTADDR); 124779251f5eSSepherosa Ziehau 124879251f5eSSepherosa Ziehau TAILQ_FOREACH(ifma, &ifp->if_multiaddrs, ifma_link) { 124979251f5eSSepherosa Ziehau if (ifma->ifma_addr->sa_family != AF_LINK) 125079251f5eSSepherosa Ziehau continue; 125179251f5eSSepherosa Ziehau if (mcnt == IX_MAX_MCASTADDR) 125279251f5eSSepherosa Ziehau break; 125379251f5eSSepherosa Ziehau bcopy(LLADDR((struct sockaddr_dl *)ifma->ifma_addr), 125479251f5eSSepherosa Ziehau &mta[mcnt * IXGBE_ETH_LENGTH_OF_ADDRESS], 125579251f5eSSepherosa Ziehau IXGBE_ETH_LENGTH_OF_ADDRESS); 125679251f5eSSepherosa Ziehau mcnt++; 125779251f5eSSepherosa Ziehau } 125879251f5eSSepherosa Ziehau 125979251f5eSSepherosa Ziehau fctrl = IXGBE_READ_REG(&sc->hw, IXGBE_FCTRL); 126079251f5eSSepherosa Ziehau fctrl |= (IXGBE_FCTRL_UPE | IXGBE_FCTRL_MPE); 126179251f5eSSepherosa Ziehau if (ifp->if_flags & IFF_PROMISC) { 126279251f5eSSepherosa Ziehau fctrl |= IXGBE_FCTRL_UPE | IXGBE_FCTRL_MPE; 126379251f5eSSepherosa Ziehau } else if (mcnt >= IX_MAX_MCASTADDR || (ifp->if_flags & IFF_ALLMULTI)) { 126479251f5eSSepherosa Ziehau fctrl |= IXGBE_FCTRL_MPE; 126579251f5eSSepherosa Ziehau fctrl &= ~IXGBE_FCTRL_UPE; 126679251f5eSSepherosa Ziehau } else { 126779251f5eSSepherosa Ziehau fctrl &= ~(IXGBE_FCTRL_UPE | IXGBE_FCTRL_MPE); 126879251f5eSSepherosa Ziehau } 126979251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_FCTRL, fctrl); 127079251f5eSSepherosa Ziehau 127179251f5eSSepherosa Ziehau if (mcnt < IX_MAX_MCASTADDR) { 127279251f5eSSepherosa Ziehau ixgbe_update_mc_addr_list(&sc->hw, 127379251f5eSSepherosa Ziehau mta, mcnt, ix_mc_array_itr, TRUE); 127479251f5eSSepherosa Ziehau } 127579251f5eSSepherosa Ziehau } 127679251f5eSSepherosa Ziehau 127779251f5eSSepherosa Ziehau /* 127879251f5eSSepherosa Ziehau * This is an iterator function now needed by the multicast 127979251f5eSSepherosa Ziehau * shared code. It simply feeds the shared code routine the 128079251f5eSSepherosa Ziehau * addresses in the array of ix_set_multi() one by one. 128179251f5eSSepherosa Ziehau */ 128279251f5eSSepherosa Ziehau static uint8_t * 128379251f5eSSepherosa Ziehau ix_mc_array_itr(struct ixgbe_hw *hw, uint8_t **update_ptr, uint32_t *vmdq) 128479251f5eSSepherosa Ziehau { 128579251f5eSSepherosa Ziehau uint8_t *addr = *update_ptr; 128679251f5eSSepherosa Ziehau uint8_t *newptr; 128779251f5eSSepherosa Ziehau *vmdq = 0; 128879251f5eSSepherosa Ziehau 128979251f5eSSepherosa Ziehau newptr = addr + IXGBE_ETH_LENGTH_OF_ADDRESS; 129079251f5eSSepherosa Ziehau *update_ptr = newptr; 129179251f5eSSepherosa Ziehau return addr; 129279251f5eSSepherosa Ziehau } 129379251f5eSSepherosa Ziehau 129479251f5eSSepherosa Ziehau static void 129579251f5eSSepherosa Ziehau ix_timer(void *arg) 129679251f5eSSepherosa Ziehau { 129779251f5eSSepherosa Ziehau struct ix_softc *sc = arg; 129879251f5eSSepherosa Ziehau 129979251f5eSSepherosa Ziehau lwkt_serialize_enter(&sc->main_serialize); 130079251f5eSSepherosa Ziehau 130179251f5eSSepherosa Ziehau if ((sc->arpcom.ac_if.if_flags & IFF_RUNNING) == 0) { 130279251f5eSSepherosa Ziehau lwkt_serialize_exit(&sc->main_serialize); 130379251f5eSSepherosa Ziehau return; 130479251f5eSSepherosa Ziehau } 130579251f5eSSepherosa Ziehau 130679251f5eSSepherosa Ziehau /* Check for pluggable optics */ 130779251f5eSSepherosa Ziehau if (sc->sfp_probe) { 130879251f5eSSepherosa Ziehau if (!ix_sfp_probe(sc)) 130979251f5eSSepherosa Ziehau goto done; /* Nothing to do */ 131079251f5eSSepherosa Ziehau } 131179251f5eSSepherosa Ziehau 131279251f5eSSepherosa Ziehau ix_update_link_status(sc); 131379251f5eSSepherosa Ziehau ix_update_stats(sc); 131479251f5eSSepherosa Ziehau 131579251f5eSSepherosa Ziehau done: 131679251f5eSSepherosa Ziehau callout_reset_bycpu(&sc->timer, hz, ix_timer, sc, sc->timer_cpuid); 131779251f5eSSepherosa Ziehau lwkt_serialize_exit(&sc->main_serialize); 131879251f5eSSepherosa Ziehau } 131979251f5eSSepherosa Ziehau 132079251f5eSSepherosa Ziehau static void 132179251f5eSSepherosa Ziehau ix_update_link_status(struct ix_softc *sc) 132279251f5eSSepherosa Ziehau { 132379251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 132479251f5eSSepherosa Ziehau 132579251f5eSSepherosa Ziehau if (sc->link_up) { 132679251f5eSSepherosa Ziehau if (sc->link_active == FALSE) { 132779251f5eSSepherosa Ziehau if (bootverbose) { 132879251f5eSSepherosa Ziehau if_printf(ifp, "Link is up %d Gbps %s\n", 132979251f5eSSepherosa Ziehau sc->link_speed == 128 ? 10 : 1, 133079251f5eSSepherosa Ziehau "Full Duplex"); 133179251f5eSSepherosa Ziehau } 133279251f5eSSepherosa Ziehau sc->link_active = TRUE; 133379251f5eSSepherosa Ziehau 133479251f5eSSepherosa Ziehau /* Update any Flow Control changes */ 133579251f5eSSepherosa Ziehau ixgbe_fc_enable(&sc->hw); 133679251f5eSSepherosa Ziehau 133779251f5eSSepherosa Ziehau ifp->if_link_state = LINK_STATE_UP; 133879251f5eSSepherosa Ziehau if_link_state_change(ifp); 133979251f5eSSepherosa Ziehau } 134079251f5eSSepherosa Ziehau } else { /* Link down */ 134179251f5eSSepherosa Ziehau if (sc->link_active == TRUE) { 134279251f5eSSepherosa Ziehau if (bootverbose) 134379251f5eSSepherosa Ziehau if_printf(ifp, "Link is Down\n"); 134479251f5eSSepherosa Ziehau ifp->if_link_state = LINK_STATE_DOWN; 134579251f5eSSepherosa Ziehau if_link_state_change(ifp); 134679251f5eSSepherosa Ziehau 134779251f5eSSepherosa Ziehau sc->link_active = FALSE; 134879251f5eSSepherosa Ziehau } 134979251f5eSSepherosa Ziehau } 135079251f5eSSepherosa Ziehau } 135179251f5eSSepherosa Ziehau 135279251f5eSSepherosa Ziehau static void 135379251f5eSSepherosa Ziehau ix_stop(struct ix_softc *sc) 135479251f5eSSepherosa Ziehau { 135579251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 135679251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 135779251f5eSSepherosa Ziehau int i; 135879251f5eSSepherosa Ziehau 135979251f5eSSepherosa Ziehau ASSERT_IFNET_SERIALIZED_ALL(ifp); 136079251f5eSSepherosa Ziehau 136179251f5eSSepherosa Ziehau ix_disable_intr(sc); 136279251f5eSSepherosa Ziehau callout_stop(&sc->timer); 136379251f5eSSepherosa Ziehau 136479251f5eSSepherosa Ziehau ifp->if_flags &= ~IFF_RUNNING; 136579251f5eSSepherosa Ziehau for (i = 0; i < sc->tx_ring_cnt; ++i) { 136679251f5eSSepherosa Ziehau ifsq_clr_oactive(sc->tx_rings[i].tx_ifsq); 136779251f5eSSepherosa Ziehau ifsq_watchdog_stop(&sc->tx_rings[i].tx_watchdog); 136879251f5eSSepherosa Ziehau } 136979251f5eSSepherosa Ziehau 137079251f5eSSepherosa Ziehau ixgbe_reset_hw(hw); 137179251f5eSSepherosa Ziehau hw->adapter_stopped = FALSE; 137279251f5eSSepherosa Ziehau ixgbe_stop_adapter(hw); 137379251f5eSSepherosa Ziehau if (hw->mac.type == ixgbe_mac_82599EB) 137479251f5eSSepherosa Ziehau ixgbe_stop_mac_link_on_d3_82599(hw); 137579251f5eSSepherosa Ziehau /* Turn off the laser - noop with no optics */ 137679251f5eSSepherosa Ziehau ixgbe_disable_tx_laser(hw); 137779251f5eSSepherosa Ziehau 137879251f5eSSepherosa Ziehau /* Update the stack */ 137979251f5eSSepherosa Ziehau sc->link_up = FALSE; 138079251f5eSSepherosa Ziehau ix_update_link_status(sc); 138179251f5eSSepherosa Ziehau 138279251f5eSSepherosa Ziehau /* Reprogram the RAR[0] in case user changed it. */ 138379251f5eSSepherosa Ziehau ixgbe_set_rar(hw, 0, hw->mac.addr, 0, IXGBE_RAH_AV); 138479251f5eSSepherosa Ziehau 138579251f5eSSepherosa Ziehau for (i = 0; i < sc->tx_ring_cnt; ++i) 138679251f5eSSepherosa Ziehau ix_free_tx_ring(&sc->tx_rings[i]); 138779251f5eSSepherosa Ziehau 138879251f5eSSepherosa Ziehau for (i = 0; i < sc->rx_ring_cnt; ++i) 138979251f5eSSepherosa Ziehau ix_free_rx_ring(&sc->rx_rings[i]); 139079251f5eSSepherosa Ziehau } 139179251f5eSSepherosa Ziehau 139279251f5eSSepherosa Ziehau static void 139379251f5eSSepherosa Ziehau ix_setup_optics(struct ix_softc *sc) 139479251f5eSSepherosa Ziehau { 139579251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 139679251f5eSSepherosa Ziehau int layer; 139779251f5eSSepherosa Ziehau 139879251f5eSSepherosa Ziehau layer = ixgbe_get_supported_physical_layer(hw); 139979251f5eSSepherosa Ziehau 140079251f5eSSepherosa Ziehau if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_T) { 140179251f5eSSepherosa Ziehau sc->optics = IFM_10G_T; 140279251f5eSSepherosa Ziehau return; 140379251f5eSSepherosa Ziehau } 140479251f5eSSepherosa Ziehau 140579251f5eSSepherosa Ziehau if (layer & IXGBE_PHYSICAL_LAYER_1000BASE_T) { 140679251f5eSSepherosa Ziehau sc->optics = IFM_1000_T; 140779251f5eSSepherosa Ziehau return; 140879251f5eSSepherosa Ziehau } 140979251f5eSSepherosa Ziehau 141079251f5eSSepherosa Ziehau if (layer & IXGBE_PHYSICAL_LAYER_1000BASE_SX) { 141179251f5eSSepherosa Ziehau sc->optics = IFM_1000_SX; 141279251f5eSSepherosa Ziehau return; 141379251f5eSSepherosa Ziehau } 141479251f5eSSepherosa Ziehau 141579251f5eSSepherosa Ziehau if (layer & (IXGBE_PHYSICAL_LAYER_10GBASE_LR | 141679251f5eSSepherosa Ziehau IXGBE_PHYSICAL_LAYER_10GBASE_LRM)) { 141779251f5eSSepherosa Ziehau sc->optics = IFM_10G_LR; 141879251f5eSSepherosa Ziehau return; 141979251f5eSSepherosa Ziehau } 142079251f5eSSepherosa Ziehau 142179251f5eSSepherosa Ziehau if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_SR) { 142279251f5eSSepherosa Ziehau sc->optics = IFM_10G_SR; 142379251f5eSSepherosa Ziehau return; 142479251f5eSSepherosa Ziehau } 142579251f5eSSepherosa Ziehau 142679251f5eSSepherosa Ziehau if (layer & IXGBE_PHYSICAL_LAYER_SFP_PLUS_CU) { 142779251f5eSSepherosa Ziehau sc->optics = IFM_10G_TWINAX; 142879251f5eSSepherosa Ziehau return; 142979251f5eSSepherosa Ziehau } 143079251f5eSSepherosa Ziehau 143179251f5eSSepherosa Ziehau if (layer & (IXGBE_PHYSICAL_LAYER_10GBASE_KX4 | 143279251f5eSSepherosa Ziehau IXGBE_PHYSICAL_LAYER_10GBASE_CX4)) { 143379251f5eSSepherosa Ziehau sc->optics = IFM_10G_CX4; 143479251f5eSSepherosa Ziehau return; 143579251f5eSSepherosa Ziehau } 143679251f5eSSepherosa Ziehau 143779251f5eSSepherosa Ziehau /* If we get here just set the default */ 143879251f5eSSepherosa Ziehau sc->optics = IFM_ETHER | IFM_AUTO; 143979251f5eSSepherosa Ziehau } 144079251f5eSSepherosa Ziehau 144179251f5eSSepherosa Ziehau static void 144279251f5eSSepherosa Ziehau ix_setup_ifp(struct ix_softc *sc) 144379251f5eSSepherosa Ziehau { 144479251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 144579251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 144679251f5eSSepherosa Ziehau int i; 144779251f5eSSepherosa Ziehau 144879251f5eSSepherosa Ziehau ifp->if_baudrate = IF_Gbps(10UL); 144979251f5eSSepherosa Ziehau 145079251f5eSSepherosa Ziehau ifp->if_softc = sc; 145179251f5eSSepherosa Ziehau ifp->if_flags = IFF_BROADCAST | IFF_SIMPLEX | IFF_MULTICAST; 145279251f5eSSepherosa Ziehau ifp->if_init = ix_init; 145379251f5eSSepherosa Ziehau ifp->if_ioctl = ix_ioctl; 145479251f5eSSepherosa Ziehau ifp->if_start = ix_start; 145579251f5eSSepherosa Ziehau ifp->if_serialize = ix_serialize; 145679251f5eSSepherosa Ziehau ifp->if_deserialize = ix_deserialize; 145779251f5eSSepherosa Ziehau ifp->if_tryserialize = ix_tryserialize; 145879251f5eSSepherosa Ziehau #ifdef INVARIANTS 145979251f5eSSepherosa Ziehau ifp->if_serialize_assert = ix_serialize_assert; 146079251f5eSSepherosa Ziehau #endif 146179251f5eSSepherosa Ziehau 1462*189a0ff3SSepherosa Ziehau /* Increase TSO burst length */ 1463*189a0ff3SSepherosa Ziehau ifp->if_tsolen = (8 * ETHERMTU); 1464*189a0ff3SSepherosa Ziehau 146579251f5eSSepherosa Ziehau ifq_set_maxlen(&ifp->if_snd, sc->tx_rings[0].tx_ndesc - 2); 146679251f5eSSepherosa Ziehau ifq_set_ready(&ifp->if_snd); 146779251f5eSSepherosa Ziehau ifq_set_subq_cnt(&ifp->if_snd, sc->tx_ring_cnt); 146879251f5eSSepherosa Ziehau 146979251f5eSSepherosa Ziehau ifp->if_mapsubq = ifq_mapsubq_mask; 147079251f5eSSepherosa Ziehau ifq_set_subq_mask(&ifp->if_snd, 0); 147179251f5eSSepherosa Ziehau 147279251f5eSSepherosa Ziehau ether_ifattach(ifp, hw->mac.addr, NULL); 147379251f5eSSepherosa Ziehau 147479251f5eSSepherosa Ziehau ifp->if_capabilities = 147579251f5eSSepherosa Ziehau IFCAP_HWCSUM | IFCAP_TSO | IFCAP_VLAN_HWTAGGING | IFCAP_VLAN_MTU; 147679251f5eSSepherosa Ziehau if (IX_ENABLE_HWRSS(sc)) 147779251f5eSSepherosa Ziehau ifp->if_capabilities |= IFCAP_RSS; 147879251f5eSSepherosa Ziehau ifp->if_capenable = ifp->if_capabilities; 147979251f5eSSepherosa Ziehau ifp->if_hwassist = CSUM_OFFLOAD | CSUM_TSO; 148079251f5eSSepherosa Ziehau 148179251f5eSSepherosa Ziehau /* 148279251f5eSSepherosa Ziehau * Tell the upper layer(s) we support long frames. 148379251f5eSSepherosa Ziehau */ 148479251f5eSSepherosa Ziehau ifp->if_data.ifi_hdrlen = sizeof(struct ether_vlan_header); 148579251f5eSSepherosa Ziehau 148679251f5eSSepherosa Ziehau /* Setup TX rings and subqueues */ 148779251f5eSSepherosa Ziehau for (i = 0; i < sc->tx_ring_cnt; ++i) { 148879251f5eSSepherosa Ziehau struct ifaltq_subque *ifsq = ifq_get_subq(&ifp->if_snd, i); 148979251f5eSSepherosa Ziehau struct ix_tx_ring *txr = &sc->tx_rings[i]; 149079251f5eSSepherosa Ziehau 149179251f5eSSepherosa Ziehau ifsq_set_cpuid(ifsq, txr->tx_intr_cpuid); 149279251f5eSSepherosa Ziehau ifsq_set_priv(ifsq, txr); 149379251f5eSSepherosa Ziehau ifsq_set_hw_serialize(ifsq, &txr->tx_serialize); 149479251f5eSSepherosa Ziehau txr->tx_ifsq = ifsq; 149579251f5eSSepherosa Ziehau 149679251f5eSSepherosa Ziehau ifsq_watchdog_init(&txr->tx_watchdog, ifsq, ix_watchdog); 149779251f5eSSepherosa Ziehau } 149879251f5eSSepherosa Ziehau 149979251f5eSSepherosa Ziehau /* 150079251f5eSSepherosa Ziehau * Specify the media types supported by this adapter and register 150179251f5eSSepherosa Ziehau * callbacks to update media and link information 150279251f5eSSepherosa Ziehau */ 150379251f5eSSepherosa Ziehau ifmedia_add(&sc->media, IFM_ETHER | sc->optics, 0, NULL); 150479251f5eSSepherosa Ziehau ifmedia_set(&sc->media, IFM_ETHER | sc->optics); 150579251f5eSSepherosa Ziehau if (hw->device_id == IXGBE_DEV_ID_82598AT) { 150679251f5eSSepherosa Ziehau ifmedia_add(&sc->media, 150779251f5eSSepherosa Ziehau IFM_ETHER | IFM_1000_T | IFM_FDX, 0, NULL); 150879251f5eSSepherosa Ziehau ifmedia_add(&sc->media, IFM_ETHER | IFM_1000_T, 0, NULL); 150979251f5eSSepherosa Ziehau } 151079251f5eSSepherosa Ziehau ifmedia_add(&sc->media, IFM_ETHER | IFM_AUTO, 0, NULL); 151179251f5eSSepherosa Ziehau ifmedia_set(&sc->media, IFM_ETHER | IFM_AUTO); 151279251f5eSSepherosa Ziehau } 151379251f5eSSepherosa Ziehau 151479251f5eSSepherosa Ziehau static boolean_t 151579251f5eSSepherosa Ziehau ix_is_sfp(const struct ixgbe_hw *hw) 151679251f5eSSepherosa Ziehau { 151779251f5eSSepherosa Ziehau switch (hw->phy.type) { 151879251f5eSSepherosa Ziehau case ixgbe_phy_sfp_avago: 151979251f5eSSepherosa Ziehau case ixgbe_phy_sfp_ftl: 152079251f5eSSepherosa Ziehau case ixgbe_phy_sfp_intel: 152179251f5eSSepherosa Ziehau case ixgbe_phy_sfp_unknown: 152279251f5eSSepherosa Ziehau case ixgbe_phy_sfp_passive_tyco: 152379251f5eSSepherosa Ziehau case ixgbe_phy_sfp_passive_unknown: 152479251f5eSSepherosa Ziehau return TRUE; 152579251f5eSSepherosa Ziehau default: 152679251f5eSSepherosa Ziehau return FALSE; 152779251f5eSSepherosa Ziehau } 152879251f5eSSepherosa Ziehau } 152979251f5eSSepherosa Ziehau 153079251f5eSSepherosa Ziehau static void 153179251f5eSSepherosa Ziehau ix_config_link(struct ix_softc *sc) 153279251f5eSSepherosa Ziehau { 153379251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 153479251f5eSSepherosa Ziehau boolean_t sfp; 153579251f5eSSepherosa Ziehau 153679251f5eSSepherosa Ziehau sfp = ix_is_sfp(hw); 153779251f5eSSepherosa Ziehau if (sfp) { 153879251f5eSSepherosa Ziehau if (hw->phy.multispeed_fiber) { 153979251f5eSSepherosa Ziehau hw->mac.ops.setup_sfp(hw); 154079251f5eSSepherosa Ziehau ixgbe_enable_tx_laser(hw); 154179251f5eSSepherosa Ziehau ix_handle_msf(sc); 154279251f5eSSepherosa Ziehau } else { 154379251f5eSSepherosa Ziehau ix_handle_mod(sc); 154479251f5eSSepherosa Ziehau } 154579251f5eSSepherosa Ziehau } else { 154679251f5eSSepherosa Ziehau uint32_t autoneg, err = 0; 154779251f5eSSepherosa Ziehau 154879251f5eSSepherosa Ziehau if (hw->mac.ops.check_link != NULL) { 154979251f5eSSepherosa Ziehau err = ixgbe_check_link(hw, &sc->link_speed, 155079251f5eSSepherosa Ziehau &sc->link_up, FALSE); 155179251f5eSSepherosa Ziehau if (err) 155279251f5eSSepherosa Ziehau return; 155379251f5eSSepherosa Ziehau } 155479251f5eSSepherosa Ziehau 155579251f5eSSepherosa Ziehau autoneg = hw->phy.autoneg_advertised; 155679251f5eSSepherosa Ziehau if (!autoneg && hw->mac.ops.get_link_capabilities != NULL) { 155779251f5eSSepherosa Ziehau bool negotiate; 155879251f5eSSepherosa Ziehau 155979251f5eSSepherosa Ziehau err = hw->mac.ops.get_link_capabilities(hw, 156079251f5eSSepherosa Ziehau &autoneg, &negotiate); 156179251f5eSSepherosa Ziehau if (err) 156279251f5eSSepherosa Ziehau return; 156379251f5eSSepherosa Ziehau } 156479251f5eSSepherosa Ziehau 156579251f5eSSepherosa Ziehau if (hw->mac.ops.setup_link != NULL) { 156679251f5eSSepherosa Ziehau err = hw->mac.ops.setup_link(hw, 156779251f5eSSepherosa Ziehau autoneg, sc->link_up); 156879251f5eSSepherosa Ziehau if (err) 156979251f5eSSepherosa Ziehau return; 157079251f5eSSepherosa Ziehau } 157179251f5eSSepherosa Ziehau } 157279251f5eSSepherosa Ziehau } 157379251f5eSSepherosa Ziehau 157479251f5eSSepherosa Ziehau static int 157579251f5eSSepherosa Ziehau ix_alloc_rings(struct ix_softc *sc) 157679251f5eSSepherosa Ziehau { 157779251f5eSSepherosa Ziehau int error, i; 157879251f5eSSepherosa Ziehau 157979251f5eSSepherosa Ziehau /* 158079251f5eSSepherosa Ziehau * Create top level busdma tag 158179251f5eSSepherosa Ziehau */ 158279251f5eSSepherosa Ziehau error = bus_dma_tag_create(NULL, 1, 0, 158379251f5eSSepherosa Ziehau BUS_SPACE_MAXADDR, BUS_SPACE_MAXADDR, NULL, NULL, 158479251f5eSSepherosa Ziehau BUS_SPACE_MAXSIZE_32BIT, 0, BUS_SPACE_MAXSIZE_32BIT, 0, 158579251f5eSSepherosa Ziehau &sc->parent_tag); 158679251f5eSSepherosa Ziehau if (error) { 158779251f5eSSepherosa Ziehau device_printf(sc->dev, "could not create top level DMA tag\n"); 158879251f5eSSepherosa Ziehau return error; 158979251f5eSSepherosa Ziehau } 159079251f5eSSepherosa Ziehau 159179251f5eSSepherosa Ziehau /* 159279251f5eSSepherosa Ziehau * Allocate TX descriptor rings and buffers 159379251f5eSSepherosa Ziehau */ 159479251f5eSSepherosa Ziehau sc->tx_rings = kmalloc_cachealign( 159579251f5eSSepherosa Ziehau sizeof(struct ix_tx_ring) * sc->tx_ring_cnt, 159679251f5eSSepherosa Ziehau M_DEVBUF, M_WAITOK | M_ZERO); 159779251f5eSSepherosa Ziehau for (i = 0; i < sc->tx_ring_cnt; ++i) { 159879251f5eSSepherosa Ziehau struct ix_tx_ring *txr = &sc->tx_rings[i]; 159979251f5eSSepherosa Ziehau 160079251f5eSSepherosa Ziehau txr->tx_sc = sc; 160179251f5eSSepherosa Ziehau txr->tx_idx = i; 1602*189a0ff3SSepherosa Ziehau txr->tx_intr_vec = -1; 160379251f5eSSepherosa Ziehau lwkt_serialize_init(&txr->tx_serialize); 160479251f5eSSepherosa Ziehau 160579251f5eSSepherosa Ziehau error = ix_create_tx_ring(txr); 160679251f5eSSepherosa Ziehau if (error) 160779251f5eSSepherosa Ziehau return error; 160879251f5eSSepherosa Ziehau } 160979251f5eSSepherosa Ziehau 161079251f5eSSepherosa Ziehau /* 161179251f5eSSepherosa Ziehau * Allocate RX descriptor rings and buffers 161279251f5eSSepherosa Ziehau */ 161379251f5eSSepherosa Ziehau sc->rx_rings = kmalloc_cachealign( 161479251f5eSSepherosa Ziehau sizeof(struct ix_rx_ring) * sc->rx_ring_cnt, 161579251f5eSSepherosa Ziehau M_DEVBUF, M_WAITOK | M_ZERO); 161679251f5eSSepherosa Ziehau for (i = 0; i < sc->rx_ring_cnt; ++i) { 161779251f5eSSepherosa Ziehau struct ix_rx_ring *rxr = &sc->rx_rings[i]; 161879251f5eSSepherosa Ziehau 161979251f5eSSepherosa Ziehau rxr->rx_sc = sc; 162079251f5eSSepherosa Ziehau rxr->rx_idx = i; 1621*189a0ff3SSepherosa Ziehau rxr->rx_intr_vec = -1; 162279251f5eSSepherosa Ziehau lwkt_serialize_init(&rxr->rx_serialize); 162379251f5eSSepherosa Ziehau 162479251f5eSSepherosa Ziehau error = ix_create_rx_ring(rxr); 162579251f5eSSepherosa Ziehau if (error) 162679251f5eSSepherosa Ziehau return error; 162779251f5eSSepherosa Ziehau } 162879251f5eSSepherosa Ziehau 162979251f5eSSepherosa Ziehau return 0; 163079251f5eSSepherosa Ziehau } 163179251f5eSSepherosa Ziehau 163279251f5eSSepherosa Ziehau static int 163379251f5eSSepherosa Ziehau ix_create_tx_ring(struct ix_tx_ring *txr) 163479251f5eSSepherosa Ziehau { 163579251f5eSSepherosa Ziehau int error, i, tsize, ntxd; 163679251f5eSSepherosa Ziehau 163779251f5eSSepherosa Ziehau /* 163879251f5eSSepherosa Ziehau * Validate number of transmit descriptors. It must not exceed 163979251f5eSSepherosa Ziehau * hardware maximum, and must be multiple of IX_DBA_ALIGN. 164079251f5eSSepherosa Ziehau */ 164179251f5eSSepherosa Ziehau ntxd = device_getenv_int(txr->tx_sc->dev, "txd", ix_txd); 164279251f5eSSepherosa Ziehau if (((ntxd * sizeof(union ixgbe_adv_tx_desc)) % IX_DBA_ALIGN) != 0 || 164379251f5eSSepherosa Ziehau ntxd < IX_MIN_TXD || ntxd > IX_MAX_TXD) { 164479251f5eSSepherosa Ziehau device_printf(txr->tx_sc->dev, 164579251f5eSSepherosa Ziehau "Using %d TX descriptors instead of %d!\n", 164679251f5eSSepherosa Ziehau IX_DEF_TXD, ntxd); 164779251f5eSSepherosa Ziehau txr->tx_ndesc = IX_DEF_TXD; 164879251f5eSSepherosa Ziehau } else { 164979251f5eSSepherosa Ziehau txr->tx_ndesc = ntxd; 165079251f5eSSepherosa Ziehau } 165179251f5eSSepherosa Ziehau 165279251f5eSSepherosa Ziehau /* 165379251f5eSSepherosa Ziehau * Allocate TX head write-back buffer 165479251f5eSSepherosa Ziehau */ 165579251f5eSSepherosa Ziehau txr->tx_hdr = bus_dmamem_coherent_any(txr->tx_sc->parent_tag, 165679251f5eSSepherosa Ziehau __VM_CACHELINE_SIZE, __VM_CACHELINE_SIZE, BUS_DMA_WAITOK, 165779251f5eSSepherosa Ziehau &txr->tx_hdr_dtag, &txr->tx_hdr_map, &txr->tx_hdr_paddr); 165879251f5eSSepherosa Ziehau if (txr->tx_hdr == NULL) { 165979251f5eSSepherosa Ziehau device_printf(txr->tx_sc->dev, 166079251f5eSSepherosa Ziehau "Unable to allocate TX head write-back buffer\n"); 166179251f5eSSepherosa Ziehau return ENOMEM; 166279251f5eSSepherosa Ziehau } 166379251f5eSSepherosa Ziehau 166479251f5eSSepherosa Ziehau /* 166579251f5eSSepherosa Ziehau * Allocate TX descriptor ring 166679251f5eSSepherosa Ziehau */ 166779251f5eSSepherosa Ziehau tsize = roundup2(txr->tx_ndesc * sizeof(union ixgbe_adv_tx_desc), 166879251f5eSSepherosa Ziehau IX_DBA_ALIGN); 166979251f5eSSepherosa Ziehau txr->tx_base = bus_dmamem_coherent_any(txr->tx_sc->parent_tag, 167079251f5eSSepherosa Ziehau IX_DBA_ALIGN, tsize, BUS_DMA_WAITOK | BUS_DMA_ZERO, 167179251f5eSSepherosa Ziehau &txr->tx_base_dtag, &txr->tx_base_map, &txr->tx_base_paddr); 167279251f5eSSepherosa Ziehau if (txr->tx_base == NULL) { 167379251f5eSSepherosa Ziehau device_printf(txr->tx_sc->dev, 167479251f5eSSepherosa Ziehau "Unable to allocate TX Descriptor memory\n"); 167579251f5eSSepherosa Ziehau return ENOMEM; 167679251f5eSSepherosa Ziehau } 167779251f5eSSepherosa Ziehau 167879251f5eSSepherosa Ziehau tsize = __VM_CACHELINE_ALIGN(sizeof(struct ix_tx_buf) * txr->tx_ndesc); 167979251f5eSSepherosa Ziehau txr->tx_buf = kmalloc_cachealign(tsize, M_DEVBUF, M_WAITOK | M_ZERO); 168079251f5eSSepherosa Ziehau 168179251f5eSSepherosa Ziehau /* 168279251f5eSSepherosa Ziehau * Create DMA tag for TX buffers 168379251f5eSSepherosa Ziehau */ 168479251f5eSSepherosa Ziehau error = bus_dma_tag_create(txr->tx_sc->parent_tag, 168579251f5eSSepherosa Ziehau 1, 0, /* alignment, bounds */ 168679251f5eSSepherosa Ziehau BUS_SPACE_MAXADDR, /* lowaddr */ 168779251f5eSSepherosa Ziehau BUS_SPACE_MAXADDR, /* highaddr */ 168879251f5eSSepherosa Ziehau NULL, NULL, /* filter, filterarg */ 168979251f5eSSepherosa Ziehau IX_TSO_SIZE, /* maxsize */ 169079251f5eSSepherosa Ziehau IX_MAX_SCATTER, /* nsegments */ 169179251f5eSSepherosa Ziehau PAGE_SIZE, /* maxsegsize */ 169279251f5eSSepherosa Ziehau BUS_DMA_WAITOK | BUS_DMA_ALLOCNOW | 169379251f5eSSepherosa Ziehau BUS_DMA_ONEBPAGE, /* flags */ 169479251f5eSSepherosa Ziehau &txr->tx_tag); 169579251f5eSSepherosa Ziehau if (error) { 169679251f5eSSepherosa Ziehau device_printf(txr->tx_sc->dev, 169779251f5eSSepherosa Ziehau "Unable to allocate TX DMA tag\n"); 169879251f5eSSepherosa Ziehau kfree(txr->tx_buf, M_DEVBUF); 169979251f5eSSepherosa Ziehau txr->tx_buf = NULL; 170079251f5eSSepherosa Ziehau return error; 170179251f5eSSepherosa Ziehau } 170279251f5eSSepherosa Ziehau 170379251f5eSSepherosa Ziehau /* 170479251f5eSSepherosa Ziehau * Create DMA maps for TX buffers 170579251f5eSSepherosa Ziehau */ 170679251f5eSSepherosa Ziehau for (i = 0; i < txr->tx_ndesc; ++i) { 170779251f5eSSepherosa Ziehau struct ix_tx_buf *txbuf = &txr->tx_buf[i]; 170879251f5eSSepherosa Ziehau 170979251f5eSSepherosa Ziehau error = bus_dmamap_create(txr->tx_tag, 171079251f5eSSepherosa Ziehau BUS_DMA_WAITOK | BUS_DMA_ONEBPAGE, &txbuf->map); 171179251f5eSSepherosa Ziehau if (error) { 171279251f5eSSepherosa Ziehau device_printf(txr->tx_sc->dev, 171379251f5eSSepherosa Ziehau "Unable to create TX DMA map\n"); 171479251f5eSSepherosa Ziehau ix_destroy_tx_ring(txr, i); 171579251f5eSSepherosa Ziehau return error; 171679251f5eSSepherosa Ziehau } 171779251f5eSSepherosa Ziehau } 171879251f5eSSepherosa Ziehau 171979251f5eSSepherosa Ziehau /* 172079251f5eSSepherosa Ziehau * Initialize various watermark 172179251f5eSSepherosa Ziehau */ 172279251f5eSSepherosa Ziehau txr->tx_wreg_nsegs = IX_DEF_TXWREG_NSEGS; 172379251f5eSSepherosa Ziehau txr->tx_intr_nsegs = txr->tx_ndesc / 16; 172479251f5eSSepherosa Ziehau 172579251f5eSSepherosa Ziehau return 0; 172679251f5eSSepherosa Ziehau } 172779251f5eSSepherosa Ziehau 172879251f5eSSepherosa Ziehau static void 172979251f5eSSepherosa Ziehau ix_destroy_tx_ring(struct ix_tx_ring *txr, int ndesc) 173079251f5eSSepherosa Ziehau { 173179251f5eSSepherosa Ziehau int i; 173279251f5eSSepherosa Ziehau 173379251f5eSSepherosa Ziehau if (txr->tx_hdr != NULL) { 173479251f5eSSepherosa Ziehau bus_dmamap_unload(txr->tx_hdr_dtag, txr->tx_hdr_map); 173579251f5eSSepherosa Ziehau bus_dmamem_free(txr->tx_hdr_dtag, 173679251f5eSSepherosa Ziehau __DEVOLATILE(void *, txr->tx_hdr), txr->tx_hdr_map); 173779251f5eSSepherosa Ziehau bus_dma_tag_destroy(txr->tx_hdr_dtag); 173879251f5eSSepherosa Ziehau txr->tx_hdr = NULL; 173979251f5eSSepherosa Ziehau } 174079251f5eSSepherosa Ziehau 174179251f5eSSepherosa Ziehau if (txr->tx_base != NULL) { 174279251f5eSSepherosa Ziehau bus_dmamap_unload(txr->tx_base_dtag, txr->tx_base_map); 174379251f5eSSepherosa Ziehau bus_dmamem_free(txr->tx_base_dtag, txr->tx_base, 174479251f5eSSepherosa Ziehau txr->tx_base_map); 174579251f5eSSepherosa Ziehau bus_dma_tag_destroy(txr->tx_base_dtag); 174679251f5eSSepherosa Ziehau txr->tx_base = NULL; 174779251f5eSSepherosa Ziehau } 174879251f5eSSepherosa Ziehau 174979251f5eSSepherosa Ziehau if (txr->tx_buf == NULL) 175079251f5eSSepherosa Ziehau return; 175179251f5eSSepherosa Ziehau 175279251f5eSSepherosa Ziehau for (i = 0; i < ndesc; ++i) { 175379251f5eSSepherosa Ziehau struct ix_tx_buf *txbuf = &txr->tx_buf[i]; 175479251f5eSSepherosa Ziehau 175579251f5eSSepherosa Ziehau KKASSERT(txbuf->m_head == NULL); 175679251f5eSSepherosa Ziehau bus_dmamap_destroy(txr->tx_tag, txbuf->map); 175779251f5eSSepherosa Ziehau } 175879251f5eSSepherosa Ziehau bus_dma_tag_destroy(txr->tx_tag); 175979251f5eSSepherosa Ziehau 176079251f5eSSepherosa Ziehau kfree(txr->tx_buf, M_DEVBUF); 176179251f5eSSepherosa Ziehau txr->tx_buf = NULL; 176279251f5eSSepherosa Ziehau } 176379251f5eSSepherosa Ziehau 176479251f5eSSepherosa Ziehau static void 176579251f5eSSepherosa Ziehau ix_init_tx_ring(struct ix_tx_ring *txr) 176679251f5eSSepherosa Ziehau { 176779251f5eSSepherosa Ziehau /* Clear the old ring contents */ 176879251f5eSSepherosa Ziehau bzero(txr->tx_base, sizeof(union ixgbe_adv_tx_desc) * txr->tx_ndesc); 176979251f5eSSepherosa Ziehau 177079251f5eSSepherosa Ziehau /* Clear TX head write-back buffer */ 177179251f5eSSepherosa Ziehau *(txr->tx_hdr) = 0; 177279251f5eSSepherosa Ziehau 177379251f5eSSepherosa Ziehau /* Reset indices */ 177479251f5eSSepherosa Ziehau txr->tx_next_avail = 0; 177579251f5eSSepherosa Ziehau txr->tx_next_clean = 0; 177679251f5eSSepherosa Ziehau txr->tx_nsegs = 0; 177779251f5eSSepherosa Ziehau 177879251f5eSSepherosa Ziehau /* Set number of descriptors available */ 177979251f5eSSepherosa Ziehau txr->tx_avail = txr->tx_ndesc; 178079251f5eSSepherosa Ziehau } 178179251f5eSSepherosa Ziehau 178279251f5eSSepherosa Ziehau static void 178379251f5eSSepherosa Ziehau ix_init_tx_unit(struct ix_softc *sc) 178479251f5eSSepherosa Ziehau { 178579251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 178679251f5eSSepherosa Ziehau int i; 178779251f5eSSepherosa Ziehau 178879251f5eSSepherosa Ziehau /* 178979251f5eSSepherosa Ziehau * Setup the Base and Length of the Tx Descriptor Ring 179079251f5eSSepherosa Ziehau */ 179179251f5eSSepherosa Ziehau for (i = 0; i < sc->tx_ring_inuse; ++i) { 179279251f5eSSepherosa Ziehau struct ix_tx_ring *txr = &sc->tx_rings[i]; 179379251f5eSSepherosa Ziehau uint64_t tdba = txr->tx_base_paddr; 179479251f5eSSepherosa Ziehau uint64_t hdr_paddr = txr->tx_hdr_paddr; 179579251f5eSSepherosa Ziehau uint32_t txctrl; 179679251f5eSSepherosa Ziehau 179779251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_TDBAL(i), (uint32_t)tdba); 179879251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_TDBAH(i), (uint32_t)(tdba >> 32)); 179979251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_TDLEN(i), 180079251f5eSSepherosa Ziehau txr->tx_ndesc * sizeof(union ixgbe_adv_tx_desc)); 180179251f5eSSepherosa Ziehau 180279251f5eSSepherosa Ziehau /* Setup the HW Tx Head and Tail descriptor pointers */ 180379251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_TDH(i), 0); 180479251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_TDT(i), 0); 180579251f5eSSepherosa Ziehau 180679251f5eSSepherosa Ziehau /* Disable TX head write-back relax ordering */ 180779251f5eSSepherosa Ziehau switch (hw->mac.type) { 180879251f5eSSepherosa Ziehau case ixgbe_mac_82598EB: 180979251f5eSSepherosa Ziehau txctrl = IXGBE_READ_REG(hw, IXGBE_DCA_TXCTRL(i)); 181079251f5eSSepherosa Ziehau break; 181179251f5eSSepherosa Ziehau case ixgbe_mac_82599EB: 181279251f5eSSepherosa Ziehau case ixgbe_mac_X540: 181379251f5eSSepherosa Ziehau default: 181479251f5eSSepherosa Ziehau txctrl = IXGBE_READ_REG(hw, IXGBE_DCA_TXCTRL_82599(i)); 181579251f5eSSepherosa Ziehau break; 181679251f5eSSepherosa Ziehau } 181779251f5eSSepherosa Ziehau txctrl &= ~IXGBE_DCA_TXCTRL_DESC_WRO_EN; 181879251f5eSSepherosa Ziehau switch (hw->mac.type) { 181979251f5eSSepherosa Ziehau case ixgbe_mac_82598EB: 182079251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_DCA_TXCTRL(i), txctrl); 182179251f5eSSepherosa Ziehau break; 182279251f5eSSepherosa Ziehau case ixgbe_mac_82599EB: 182379251f5eSSepherosa Ziehau case ixgbe_mac_X540: 182479251f5eSSepherosa Ziehau default: 182579251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_DCA_TXCTRL_82599(i), txctrl); 182679251f5eSSepherosa Ziehau break; 182779251f5eSSepherosa Ziehau } 182879251f5eSSepherosa Ziehau 182979251f5eSSepherosa Ziehau /* Enable TX head write-back */ 183079251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_TDWBAH(i), 183179251f5eSSepherosa Ziehau (uint32_t)(hdr_paddr >> 32)); 183279251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_TDWBAL(i), 183379251f5eSSepherosa Ziehau ((uint32_t)hdr_paddr) | IXGBE_TDWBAL_HEAD_WB_ENABLE); 183479251f5eSSepherosa Ziehau } 183579251f5eSSepherosa Ziehau 183679251f5eSSepherosa Ziehau if (hw->mac.type != ixgbe_mac_82598EB) { 183779251f5eSSepherosa Ziehau uint32_t dmatxctl, rttdcs; 183879251f5eSSepherosa Ziehau 183979251f5eSSepherosa Ziehau dmatxctl = IXGBE_READ_REG(hw, IXGBE_DMATXCTL); 184079251f5eSSepherosa Ziehau dmatxctl |= IXGBE_DMATXCTL_TE; 184179251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_DMATXCTL, dmatxctl); 184279251f5eSSepherosa Ziehau 184379251f5eSSepherosa Ziehau /* Disable arbiter to set MTQC */ 184479251f5eSSepherosa Ziehau rttdcs = IXGBE_READ_REG(hw, IXGBE_RTTDCS); 184579251f5eSSepherosa Ziehau rttdcs |= IXGBE_RTTDCS_ARBDIS; 184679251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RTTDCS, rttdcs); 184779251f5eSSepherosa Ziehau 184879251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_MTQC, IXGBE_MTQC_64Q_1PB); 184979251f5eSSepherosa Ziehau 185079251f5eSSepherosa Ziehau /* Reenable aribter */ 185179251f5eSSepherosa Ziehau rttdcs &= ~IXGBE_RTTDCS_ARBDIS; 185279251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RTTDCS, rttdcs); 185379251f5eSSepherosa Ziehau } 185479251f5eSSepherosa Ziehau } 185579251f5eSSepherosa Ziehau 185679251f5eSSepherosa Ziehau static int 185779251f5eSSepherosa Ziehau ix_tx_ctx_setup(struct ix_tx_ring *txr, const struct mbuf *mp, 185879251f5eSSepherosa Ziehau uint32_t *cmd_type_len, uint32_t *olinfo_status) 185979251f5eSSepherosa Ziehau { 186079251f5eSSepherosa Ziehau struct ixgbe_adv_tx_context_desc *TXD; 186179251f5eSSepherosa Ziehau uint32_t vlan_macip_lens = 0, type_tucmd_mlhl = 0; 186279251f5eSSepherosa Ziehau int ehdrlen, ip_hlen = 0, ctxd; 186379251f5eSSepherosa Ziehau boolean_t offload = TRUE; 186479251f5eSSepherosa Ziehau 186579251f5eSSepherosa Ziehau /* First check if TSO is to be used */ 186679251f5eSSepherosa Ziehau if (mp->m_pkthdr.csum_flags & CSUM_TSO) { 186779251f5eSSepherosa Ziehau return ix_tso_ctx_setup(txr, mp, 186879251f5eSSepherosa Ziehau cmd_type_len, olinfo_status); 186979251f5eSSepherosa Ziehau } 187079251f5eSSepherosa Ziehau 187179251f5eSSepherosa Ziehau if ((mp->m_pkthdr.csum_flags & CSUM_OFFLOAD) == 0) 187279251f5eSSepherosa Ziehau offload = FALSE; 187379251f5eSSepherosa Ziehau 187479251f5eSSepherosa Ziehau /* Indicate the whole packet as payload when not doing TSO */ 187579251f5eSSepherosa Ziehau *olinfo_status |= mp->m_pkthdr.len << IXGBE_ADVTXD_PAYLEN_SHIFT; 187679251f5eSSepherosa Ziehau 187779251f5eSSepherosa Ziehau /* 187879251f5eSSepherosa Ziehau * In advanced descriptors the vlan tag must be placed into the 187979251f5eSSepherosa Ziehau * context descriptor. Hence we need to make one even if not 188079251f5eSSepherosa Ziehau * doing checksum offloads. 188179251f5eSSepherosa Ziehau */ 188279251f5eSSepherosa Ziehau if (mp->m_flags & M_VLANTAG) { 188379251f5eSSepherosa Ziehau vlan_macip_lens |= htole16(mp->m_pkthdr.ether_vlantag) << 188479251f5eSSepherosa Ziehau IXGBE_ADVTXD_VLAN_SHIFT; 188579251f5eSSepherosa Ziehau } else if (!offload) { 188679251f5eSSepherosa Ziehau /* No TX descriptor is consumed */ 188779251f5eSSepherosa Ziehau return 0; 188879251f5eSSepherosa Ziehau } 188979251f5eSSepherosa Ziehau 189079251f5eSSepherosa Ziehau /* Set the ether header length */ 189179251f5eSSepherosa Ziehau ehdrlen = mp->m_pkthdr.csum_lhlen; 189279251f5eSSepherosa Ziehau KASSERT(ehdrlen > 0, ("invalid ether hlen")); 189379251f5eSSepherosa Ziehau vlan_macip_lens |= ehdrlen << IXGBE_ADVTXD_MACLEN_SHIFT; 189479251f5eSSepherosa Ziehau 189579251f5eSSepherosa Ziehau if (mp->m_pkthdr.csum_flags & CSUM_IP) { 189679251f5eSSepherosa Ziehau *olinfo_status |= IXGBE_TXD_POPTS_IXSM << 8; 189779251f5eSSepherosa Ziehau type_tucmd_mlhl |= IXGBE_ADVTXD_TUCMD_IPV4; 189879251f5eSSepherosa Ziehau ip_hlen = mp->m_pkthdr.csum_iphlen; 189979251f5eSSepherosa Ziehau KASSERT(ip_hlen > 0, ("invalid ip hlen")); 190079251f5eSSepherosa Ziehau } 190179251f5eSSepherosa Ziehau vlan_macip_lens |= ip_hlen; 190279251f5eSSepherosa Ziehau 190379251f5eSSepherosa Ziehau type_tucmd_mlhl |= IXGBE_ADVTXD_DCMD_DEXT | IXGBE_ADVTXD_DTYP_CTXT; 190479251f5eSSepherosa Ziehau if (mp->m_pkthdr.csum_flags & CSUM_TCP) 190579251f5eSSepherosa Ziehau type_tucmd_mlhl |= IXGBE_ADVTXD_TUCMD_L4T_TCP; 190679251f5eSSepherosa Ziehau else if (mp->m_pkthdr.csum_flags & CSUM_UDP) 190779251f5eSSepherosa Ziehau type_tucmd_mlhl |= IXGBE_ADVTXD_TUCMD_L4T_UDP; 190879251f5eSSepherosa Ziehau 190979251f5eSSepherosa Ziehau if (mp->m_pkthdr.csum_flags & (CSUM_TCP | CSUM_UDP)) 191079251f5eSSepherosa Ziehau *olinfo_status |= IXGBE_TXD_POPTS_TXSM << 8; 191179251f5eSSepherosa Ziehau 191279251f5eSSepherosa Ziehau /* Now ready a context descriptor */ 191379251f5eSSepherosa Ziehau ctxd = txr->tx_next_avail; 191479251f5eSSepherosa Ziehau TXD = (struct ixgbe_adv_tx_context_desc *)&txr->tx_base[ctxd]; 191579251f5eSSepherosa Ziehau 191679251f5eSSepherosa Ziehau /* Now copy bits into descriptor */ 191779251f5eSSepherosa Ziehau TXD->vlan_macip_lens = htole32(vlan_macip_lens); 191879251f5eSSepherosa Ziehau TXD->type_tucmd_mlhl = htole32(type_tucmd_mlhl); 191979251f5eSSepherosa Ziehau TXD->seqnum_seed = htole32(0); 192079251f5eSSepherosa Ziehau TXD->mss_l4len_idx = htole32(0); 192179251f5eSSepherosa Ziehau 192279251f5eSSepherosa Ziehau /* We've consumed the first desc, adjust counters */ 192379251f5eSSepherosa Ziehau if (++ctxd == txr->tx_ndesc) 192479251f5eSSepherosa Ziehau ctxd = 0; 192579251f5eSSepherosa Ziehau txr->tx_next_avail = ctxd; 192679251f5eSSepherosa Ziehau --txr->tx_avail; 192779251f5eSSepherosa Ziehau 192879251f5eSSepherosa Ziehau /* One TX descriptor is consumed */ 192979251f5eSSepherosa Ziehau return 1; 193079251f5eSSepherosa Ziehau } 193179251f5eSSepherosa Ziehau 193279251f5eSSepherosa Ziehau static int 193379251f5eSSepherosa Ziehau ix_tso_ctx_setup(struct ix_tx_ring *txr, const struct mbuf *mp, 193479251f5eSSepherosa Ziehau uint32_t *cmd_type_len, uint32_t *olinfo_status) 193579251f5eSSepherosa Ziehau { 193679251f5eSSepherosa Ziehau struct ixgbe_adv_tx_context_desc *TXD; 193779251f5eSSepherosa Ziehau uint32_t vlan_macip_lens = 0, type_tucmd_mlhl = 0; 193879251f5eSSepherosa Ziehau uint32_t mss_l4len_idx = 0, paylen; 193979251f5eSSepherosa Ziehau int ctxd, ehdrlen, ip_hlen, tcp_hlen; 194079251f5eSSepherosa Ziehau 194179251f5eSSepherosa Ziehau ehdrlen = mp->m_pkthdr.csum_lhlen; 194279251f5eSSepherosa Ziehau KASSERT(ehdrlen > 0, ("invalid ether hlen")); 194379251f5eSSepherosa Ziehau 194479251f5eSSepherosa Ziehau ip_hlen = mp->m_pkthdr.csum_iphlen; 194579251f5eSSepherosa Ziehau KASSERT(ip_hlen > 0, ("invalid ip hlen")); 194679251f5eSSepherosa Ziehau 194779251f5eSSepherosa Ziehau tcp_hlen = mp->m_pkthdr.csum_thlen; 194879251f5eSSepherosa Ziehau KASSERT(tcp_hlen > 0, ("invalid tcp hlen")); 194979251f5eSSepherosa Ziehau 195079251f5eSSepherosa Ziehau ctxd = txr->tx_next_avail; 195179251f5eSSepherosa Ziehau TXD = (struct ixgbe_adv_tx_context_desc *) &txr->tx_base[ctxd]; 195279251f5eSSepherosa Ziehau 195379251f5eSSepherosa Ziehau if (mp->m_flags & M_VLANTAG) { 195479251f5eSSepherosa Ziehau vlan_macip_lens |= htole16(mp->m_pkthdr.ether_vlantag) << 195579251f5eSSepherosa Ziehau IXGBE_ADVTXD_VLAN_SHIFT; 195679251f5eSSepherosa Ziehau } 195779251f5eSSepherosa Ziehau vlan_macip_lens |= ehdrlen << IXGBE_ADVTXD_MACLEN_SHIFT; 195879251f5eSSepherosa Ziehau vlan_macip_lens |= ip_hlen; 195979251f5eSSepherosa Ziehau TXD->vlan_macip_lens = htole32(vlan_macip_lens); 196079251f5eSSepherosa Ziehau 196179251f5eSSepherosa Ziehau /* ADV DTYPE TUCMD */ 196279251f5eSSepherosa Ziehau type_tucmd_mlhl |= IXGBE_ADVTXD_TUCMD_IPV4; 196379251f5eSSepherosa Ziehau type_tucmd_mlhl |= IXGBE_ADVTXD_DCMD_DEXT | IXGBE_ADVTXD_DTYP_CTXT; 196479251f5eSSepherosa Ziehau type_tucmd_mlhl |= IXGBE_ADVTXD_TUCMD_L4T_TCP; 196579251f5eSSepherosa Ziehau TXD->type_tucmd_mlhl = htole32(type_tucmd_mlhl); 196679251f5eSSepherosa Ziehau 196779251f5eSSepherosa Ziehau /* MSS L4LEN IDX */ 196879251f5eSSepherosa Ziehau mss_l4len_idx |= (mp->m_pkthdr.tso_segsz << IXGBE_ADVTXD_MSS_SHIFT); 196979251f5eSSepherosa Ziehau mss_l4len_idx |= (tcp_hlen << IXGBE_ADVTXD_L4LEN_SHIFT); 197079251f5eSSepherosa Ziehau TXD->mss_l4len_idx = htole32(mss_l4len_idx); 197179251f5eSSepherosa Ziehau 197279251f5eSSepherosa Ziehau TXD->seqnum_seed = htole32(0); 197379251f5eSSepherosa Ziehau 197479251f5eSSepherosa Ziehau if (++ctxd == txr->tx_ndesc) 197579251f5eSSepherosa Ziehau ctxd = 0; 197679251f5eSSepherosa Ziehau 197779251f5eSSepherosa Ziehau txr->tx_avail--; 197879251f5eSSepherosa Ziehau txr->tx_next_avail = ctxd; 197979251f5eSSepherosa Ziehau 198079251f5eSSepherosa Ziehau *cmd_type_len |= IXGBE_ADVTXD_DCMD_TSE; 198179251f5eSSepherosa Ziehau 198279251f5eSSepherosa Ziehau /* This is used in the transmit desc in encap */ 198379251f5eSSepherosa Ziehau paylen = mp->m_pkthdr.len - ehdrlen - ip_hlen - tcp_hlen; 198479251f5eSSepherosa Ziehau 198579251f5eSSepherosa Ziehau *olinfo_status |= IXGBE_TXD_POPTS_IXSM << 8; 198679251f5eSSepherosa Ziehau *olinfo_status |= IXGBE_TXD_POPTS_TXSM << 8; 198779251f5eSSepherosa Ziehau *olinfo_status |= paylen << IXGBE_ADVTXD_PAYLEN_SHIFT; 198879251f5eSSepherosa Ziehau 198979251f5eSSepherosa Ziehau /* One TX descriptor is consumed */ 199079251f5eSSepherosa Ziehau return 1; 199179251f5eSSepherosa Ziehau } 199279251f5eSSepherosa Ziehau 199379251f5eSSepherosa Ziehau static void 1994*189a0ff3SSepherosa Ziehau ix_txeof(struct ix_tx_ring *txr, int hdr) 199579251f5eSSepherosa Ziehau { 199679251f5eSSepherosa Ziehau struct ifnet *ifp = &txr->tx_sc->arpcom.ac_if; 1997*189a0ff3SSepherosa Ziehau int first, avail; 199879251f5eSSepherosa Ziehau 199979251f5eSSepherosa Ziehau if (txr->tx_avail == txr->tx_ndesc) 200079251f5eSSepherosa Ziehau return; 200179251f5eSSepherosa Ziehau 200279251f5eSSepherosa Ziehau first = txr->tx_next_clean; 200379251f5eSSepherosa Ziehau if (first == hdr) 200479251f5eSSepherosa Ziehau return; 200579251f5eSSepherosa Ziehau 200679251f5eSSepherosa Ziehau avail = txr->tx_avail; 200779251f5eSSepherosa Ziehau while (first != hdr) { 200879251f5eSSepherosa Ziehau struct ix_tx_buf *txbuf = &txr->tx_buf[first]; 200979251f5eSSepherosa Ziehau 201079251f5eSSepherosa Ziehau ++avail; 201179251f5eSSepherosa Ziehau if (txbuf->m_head) { 201279251f5eSSepherosa Ziehau bus_dmamap_unload(txr->tx_tag, txbuf->map); 201379251f5eSSepherosa Ziehau m_freem(txbuf->m_head); 201479251f5eSSepherosa Ziehau txbuf->m_head = NULL; 201579251f5eSSepherosa Ziehau IFNET_STAT_INC(ifp, opackets, 1); 201679251f5eSSepherosa Ziehau } 201779251f5eSSepherosa Ziehau if (++first == txr->tx_ndesc) 201879251f5eSSepherosa Ziehau first = 0; 201979251f5eSSepherosa Ziehau } 202079251f5eSSepherosa Ziehau txr->tx_next_clean = first; 202179251f5eSSepherosa Ziehau txr->tx_avail = avail; 202279251f5eSSepherosa Ziehau 202379251f5eSSepherosa Ziehau if (txr->tx_avail > IX_MAX_SCATTER + IX_TX_RESERVED) { 202479251f5eSSepherosa Ziehau ifsq_clr_oactive(txr->tx_ifsq); 202579251f5eSSepherosa Ziehau txr->tx_watchdog.wd_timer = 0; 202679251f5eSSepherosa Ziehau } 202779251f5eSSepherosa Ziehau } 202879251f5eSSepherosa Ziehau 202979251f5eSSepherosa Ziehau static int 203079251f5eSSepherosa Ziehau ix_create_rx_ring(struct ix_rx_ring *rxr) 203179251f5eSSepherosa Ziehau { 203279251f5eSSepherosa Ziehau int i, rsize, error, nrxd; 203379251f5eSSepherosa Ziehau 203479251f5eSSepherosa Ziehau /* 203579251f5eSSepherosa Ziehau * Validate number of receive descriptors. It must not exceed 203679251f5eSSepherosa Ziehau * hardware maximum, and must be multiple of IX_DBA_ALIGN. 203779251f5eSSepherosa Ziehau */ 203879251f5eSSepherosa Ziehau nrxd = device_getenv_int(rxr->rx_sc->dev, "rxd", ix_rxd); 203979251f5eSSepherosa Ziehau if (((nrxd * sizeof(union ixgbe_adv_rx_desc)) % IX_DBA_ALIGN) != 0 || 204079251f5eSSepherosa Ziehau nrxd < IX_MIN_RXD || nrxd > IX_MAX_RXD) { 204179251f5eSSepherosa Ziehau device_printf(rxr->rx_sc->dev, 204279251f5eSSepherosa Ziehau "Using %d RX descriptors instead of %d!\n", 204379251f5eSSepherosa Ziehau IX_DEF_RXD, nrxd); 204479251f5eSSepherosa Ziehau rxr->rx_ndesc = IX_DEF_RXD; 204579251f5eSSepherosa Ziehau } else { 204679251f5eSSepherosa Ziehau rxr->rx_ndesc = nrxd; 204779251f5eSSepherosa Ziehau } 204879251f5eSSepherosa Ziehau 204979251f5eSSepherosa Ziehau /* 205079251f5eSSepherosa Ziehau * Allocate RX descriptor ring 205179251f5eSSepherosa Ziehau */ 205279251f5eSSepherosa Ziehau rsize = roundup2(rxr->rx_ndesc * sizeof(union ixgbe_adv_rx_desc), 205379251f5eSSepherosa Ziehau IX_DBA_ALIGN); 205479251f5eSSepherosa Ziehau rxr->rx_base = bus_dmamem_coherent_any(rxr->rx_sc->parent_tag, 205579251f5eSSepherosa Ziehau IX_DBA_ALIGN, rsize, BUS_DMA_WAITOK | BUS_DMA_ZERO, 205679251f5eSSepherosa Ziehau &rxr->rx_base_dtag, &rxr->rx_base_map, &rxr->rx_base_paddr); 205779251f5eSSepherosa Ziehau if (rxr->rx_base == NULL) { 205879251f5eSSepherosa Ziehau device_printf(rxr->rx_sc->dev, 205979251f5eSSepherosa Ziehau "Unable to allocate TX Descriptor memory\n"); 206079251f5eSSepherosa Ziehau return ENOMEM; 206179251f5eSSepherosa Ziehau } 206279251f5eSSepherosa Ziehau 206379251f5eSSepherosa Ziehau rsize = __VM_CACHELINE_ALIGN(sizeof(struct ix_rx_buf) * rxr->rx_ndesc); 206479251f5eSSepherosa Ziehau rxr->rx_buf = kmalloc_cachealign(rsize, M_DEVBUF, M_WAITOK | M_ZERO); 206579251f5eSSepherosa Ziehau 206679251f5eSSepherosa Ziehau /* 206779251f5eSSepherosa Ziehau * Create DMA tag for RX buffers 206879251f5eSSepherosa Ziehau */ 206979251f5eSSepherosa Ziehau error = bus_dma_tag_create(rxr->rx_sc->parent_tag, 207079251f5eSSepherosa Ziehau 1, 0, /* alignment, bounds */ 207179251f5eSSepherosa Ziehau BUS_SPACE_MAXADDR, /* lowaddr */ 207279251f5eSSepherosa Ziehau BUS_SPACE_MAXADDR, /* highaddr */ 207379251f5eSSepherosa Ziehau NULL, NULL, /* filter, filterarg */ 207479251f5eSSepherosa Ziehau PAGE_SIZE, /* maxsize */ 207579251f5eSSepherosa Ziehau 1, /* nsegments */ 207679251f5eSSepherosa Ziehau PAGE_SIZE, /* maxsegsize */ 207779251f5eSSepherosa Ziehau BUS_DMA_WAITOK | BUS_DMA_ALLOCNOW, /* flags */ 207879251f5eSSepherosa Ziehau &rxr->rx_tag); 207979251f5eSSepherosa Ziehau if (error) { 208079251f5eSSepherosa Ziehau device_printf(rxr->rx_sc->dev, 208179251f5eSSepherosa Ziehau "Unable to create RX DMA tag\n"); 208279251f5eSSepherosa Ziehau kfree(rxr->rx_buf, M_DEVBUF); 208379251f5eSSepherosa Ziehau rxr->rx_buf = NULL; 208479251f5eSSepherosa Ziehau return error; 208579251f5eSSepherosa Ziehau } 208679251f5eSSepherosa Ziehau 208779251f5eSSepherosa Ziehau /* 208879251f5eSSepherosa Ziehau * Create spare DMA map for RX buffers 208979251f5eSSepherosa Ziehau */ 209079251f5eSSepherosa Ziehau error = bus_dmamap_create(rxr->rx_tag, BUS_DMA_WAITOK, 209179251f5eSSepherosa Ziehau &rxr->rx_sparemap); 209279251f5eSSepherosa Ziehau if (error) { 209379251f5eSSepherosa Ziehau device_printf(rxr->rx_sc->dev, 209479251f5eSSepherosa Ziehau "Unable to create spare RX DMA map\n"); 209579251f5eSSepherosa Ziehau bus_dma_tag_destroy(rxr->rx_tag); 209679251f5eSSepherosa Ziehau kfree(rxr->rx_buf, M_DEVBUF); 209779251f5eSSepherosa Ziehau rxr->rx_buf = NULL; 209879251f5eSSepherosa Ziehau return error; 209979251f5eSSepherosa Ziehau } 210079251f5eSSepherosa Ziehau 210179251f5eSSepherosa Ziehau /* 210279251f5eSSepherosa Ziehau * Create DMA maps for RX buffers 210379251f5eSSepherosa Ziehau */ 210479251f5eSSepherosa Ziehau for (i = 0; i < rxr->rx_ndesc; ++i) { 210579251f5eSSepherosa Ziehau struct ix_rx_buf *rxbuf = &rxr->rx_buf[i]; 210679251f5eSSepherosa Ziehau 210779251f5eSSepherosa Ziehau error = bus_dmamap_create(rxr->rx_tag, 210879251f5eSSepherosa Ziehau BUS_DMA_WAITOK, &rxbuf->map); 210979251f5eSSepherosa Ziehau if (error) { 211079251f5eSSepherosa Ziehau device_printf(rxr->rx_sc->dev, 211179251f5eSSepherosa Ziehau "Unable to create RX dma map\n"); 211279251f5eSSepherosa Ziehau ix_destroy_rx_ring(rxr, i); 211379251f5eSSepherosa Ziehau return error; 211479251f5eSSepherosa Ziehau } 211579251f5eSSepherosa Ziehau } 211679251f5eSSepherosa Ziehau 211779251f5eSSepherosa Ziehau /* 211879251f5eSSepherosa Ziehau * Initialize various watermark 211979251f5eSSepherosa Ziehau */ 212079251f5eSSepherosa Ziehau rxr->rx_wreg_nsegs = IX_DEF_RXWREG_NSEGS; 212179251f5eSSepherosa Ziehau 212279251f5eSSepherosa Ziehau return 0; 212379251f5eSSepherosa Ziehau } 212479251f5eSSepherosa Ziehau 212579251f5eSSepherosa Ziehau static void 212679251f5eSSepherosa Ziehau ix_destroy_rx_ring(struct ix_rx_ring *rxr, int ndesc) 212779251f5eSSepherosa Ziehau { 212879251f5eSSepherosa Ziehau int i; 212979251f5eSSepherosa Ziehau 213079251f5eSSepherosa Ziehau if (rxr->rx_base != NULL) { 213179251f5eSSepherosa Ziehau bus_dmamap_unload(rxr->rx_base_dtag, rxr->rx_base_map); 213279251f5eSSepherosa Ziehau bus_dmamem_free(rxr->rx_base_dtag, rxr->rx_base, 213379251f5eSSepherosa Ziehau rxr->rx_base_map); 213479251f5eSSepherosa Ziehau bus_dma_tag_destroy(rxr->rx_base_dtag); 213579251f5eSSepherosa Ziehau rxr->rx_base = NULL; 213679251f5eSSepherosa Ziehau } 213779251f5eSSepherosa Ziehau 213879251f5eSSepherosa Ziehau if (rxr->rx_buf == NULL) 213979251f5eSSepherosa Ziehau return; 214079251f5eSSepherosa Ziehau 214179251f5eSSepherosa Ziehau for (i = 0; i < ndesc; ++i) { 214279251f5eSSepherosa Ziehau struct ix_rx_buf *rxbuf = &rxr->rx_buf[i]; 214379251f5eSSepherosa Ziehau 214479251f5eSSepherosa Ziehau KKASSERT(rxbuf->m_head == NULL); 214579251f5eSSepherosa Ziehau bus_dmamap_destroy(rxr->rx_tag, rxbuf->map); 214679251f5eSSepherosa Ziehau } 214779251f5eSSepherosa Ziehau bus_dmamap_destroy(rxr->rx_tag, rxr->rx_sparemap); 214879251f5eSSepherosa Ziehau bus_dma_tag_destroy(rxr->rx_tag); 214979251f5eSSepherosa Ziehau 215079251f5eSSepherosa Ziehau kfree(rxr->rx_buf, M_DEVBUF); 215179251f5eSSepherosa Ziehau rxr->rx_buf = NULL; 215279251f5eSSepherosa Ziehau } 215379251f5eSSepherosa Ziehau 215479251f5eSSepherosa Ziehau /* 215579251f5eSSepherosa Ziehau ** Used to detect a descriptor that has 215679251f5eSSepherosa Ziehau ** been merged by Hardware RSC. 215779251f5eSSepherosa Ziehau */ 215879251f5eSSepherosa Ziehau static __inline uint32_t 215979251f5eSSepherosa Ziehau ix_rsc_count(union ixgbe_adv_rx_desc *rx) 216079251f5eSSepherosa Ziehau { 216179251f5eSSepherosa Ziehau return (le32toh(rx->wb.lower.lo_dword.data) & 216279251f5eSSepherosa Ziehau IXGBE_RXDADV_RSCCNT_MASK) >> IXGBE_RXDADV_RSCCNT_SHIFT; 216379251f5eSSepherosa Ziehau } 216479251f5eSSepherosa Ziehau 216579251f5eSSepherosa Ziehau #if 0 216679251f5eSSepherosa Ziehau /********************************************************************* 216779251f5eSSepherosa Ziehau * 216879251f5eSSepherosa Ziehau * Initialize Hardware RSC (LRO) feature on 82599 216979251f5eSSepherosa Ziehau * for an RX ring, this is toggled by the LRO capability 217079251f5eSSepherosa Ziehau * even though it is transparent to the stack. 217179251f5eSSepherosa Ziehau * 217279251f5eSSepherosa Ziehau * NOTE: since this HW feature only works with IPV4 and 217379251f5eSSepherosa Ziehau * our testing has shown soft LRO to be as effective 217479251f5eSSepherosa Ziehau * I have decided to disable this by default. 217579251f5eSSepherosa Ziehau * 217679251f5eSSepherosa Ziehau **********************************************************************/ 217779251f5eSSepherosa Ziehau static void 217879251f5eSSepherosa Ziehau ix_setup_hw_rsc(struct ix_rx_ring *rxr) 217979251f5eSSepherosa Ziehau { 218079251f5eSSepherosa Ziehau struct ix_softc *sc = rxr->rx_sc; 218179251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 218279251f5eSSepherosa Ziehau uint32_t rscctrl, rdrxctl; 218379251f5eSSepherosa Ziehau 218479251f5eSSepherosa Ziehau #if 0 218579251f5eSSepherosa Ziehau /* If turning LRO/RSC off we need to disable it */ 218679251f5eSSepherosa Ziehau if ((sc->arpcom.ac_if.if_capenable & IFCAP_LRO) == 0) { 218779251f5eSSepherosa Ziehau rscctrl = IXGBE_READ_REG(hw, IXGBE_RSCCTL(rxr->me)); 218879251f5eSSepherosa Ziehau rscctrl &= ~IXGBE_RSCCTL_RSCEN; 218979251f5eSSepherosa Ziehau return; 219079251f5eSSepherosa Ziehau } 219179251f5eSSepherosa Ziehau #endif 219279251f5eSSepherosa Ziehau 219379251f5eSSepherosa Ziehau rdrxctl = IXGBE_READ_REG(hw, IXGBE_RDRXCTL); 219479251f5eSSepherosa Ziehau rdrxctl &= ~IXGBE_RDRXCTL_RSCFRSTSIZE; 219579251f5eSSepherosa Ziehau rdrxctl |= IXGBE_RDRXCTL_CRCSTRIP; 219679251f5eSSepherosa Ziehau rdrxctl |= IXGBE_RDRXCTL_RSCACKC; 219779251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RDRXCTL, rdrxctl); 219879251f5eSSepherosa Ziehau 219979251f5eSSepherosa Ziehau rscctrl = IXGBE_READ_REG(hw, IXGBE_RSCCTL(rxr->me)); 220079251f5eSSepherosa Ziehau rscctrl |= IXGBE_RSCCTL_RSCEN; 220179251f5eSSepherosa Ziehau /* 220279251f5eSSepherosa Ziehau ** Limit the total number of descriptors that 220379251f5eSSepherosa Ziehau ** can be combined, so it does not exceed 64K 220479251f5eSSepherosa Ziehau */ 220579251f5eSSepherosa Ziehau if (rxr->mbuf_sz == MCLBYTES) 220679251f5eSSepherosa Ziehau rscctrl |= IXGBE_RSCCTL_MAXDESC_16; 220779251f5eSSepherosa Ziehau else if (rxr->mbuf_sz == MJUMPAGESIZE) 220879251f5eSSepherosa Ziehau rscctrl |= IXGBE_RSCCTL_MAXDESC_8; 220979251f5eSSepherosa Ziehau else if (rxr->mbuf_sz == MJUM9BYTES) 221079251f5eSSepherosa Ziehau rscctrl |= IXGBE_RSCCTL_MAXDESC_4; 221179251f5eSSepherosa Ziehau else /* Using 16K cluster */ 221279251f5eSSepherosa Ziehau rscctrl |= IXGBE_RSCCTL_MAXDESC_1; 221379251f5eSSepherosa Ziehau 221479251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RSCCTL(rxr->me), rscctrl); 221579251f5eSSepherosa Ziehau 221679251f5eSSepherosa Ziehau /* Enable TCP header recognition */ 221779251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_PSRTYPE(0), 221879251f5eSSepherosa Ziehau (IXGBE_READ_REG(hw, IXGBE_PSRTYPE(0)) | 221979251f5eSSepherosa Ziehau IXGBE_PSRTYPE_TCPHDR)); 222079251f5eSSepherosa Ziehau 222179251f5eSSepherosa Ziehau /* Disable RSC for ACK packets */ 222279251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RSCDBU, 222379251f5eSSepherosa Ziehau (IXGBE_RSCDBU_RSCACKDIS | IXGBE_READ_REG(hw, IXGBE_RSCDBU))); 222479251f5eSSepherosa Ziehau 222579251f5eSSepherosa Ziehau rxr->hw_rsc = TRUE; 222679251f5eSSepherosa Ziehau } 222779251f5eSSepherosa Ziehau #endif 222879251f5eSSepherosa Ziehau 222979251f5eSSepherosa Ziehau static int 223079251f5eSSepherosa Ziehau ix_init_rx_ring(struct ix_rx_ring *rxr) 223179251f5eSSepherosa Ziehau { 223279251f5eSSepherosa Ziehau int i; 223379251f5eSSepherosa Ziehau 223479251f5eSSepherosa Ziehau /* Clear the ring contents */ 223579251f5eSSepherosa Ziehau bzero(rxr->rx_base, rxr->rx_ndesc * sizeof(union ixgbe_adv_rx_desc)); 223679251f5eSSepherosa Ziehau 223779251f5eSSepherosa Ziehau /* XXX we need JUMPAGESIZE for RSC too */ 223879251f5eSSepherosa Ziehau if (rxr->rx_sc->max_frame_size <= MCLBYTES) 223979251f5eSSepherosa Ziehau rxr->rx_mbuf_sz = MCLBYTES; 224079251f5eSSepherosa Ziehau else 224179251f5eSSepherosa Ziehau rxr->rx_mbuf_sz = MJUMPAGESIZE; 224279251f5eSSepherosa Ziehau 224379251f5eSSepherosa Ziehau /* Now replenish the mbufs */ 224479251f5eSSepherosa Ziehau for (i = 0; i < rxr->rx_ndesc; ++i) { 224579251f5eSSepherosa Ziehau int error; 224679251f5eSSepherosa Ziehau 224779251f5eSSepherosa Ziehau error = ix_newbuf(rxr, i, TRUE); 224879251f5eSSepherosa Ziehau if (error) 224979251f5eSSepherosa Ziehau return error; 225079251f5eSSepherosa Ziehau } 225179251f5eSSepherosa Ziehau 225279251f5eSSepherosa Ziehau /* Setup our descriptor indices */ 225379251f5eSSepherosa Ziehau rxr->rx_next_check = 0; 225479251f5eSSepherosa Ziehau rxr->rx_flags &= ~IX_RXRING_FLAG_DISC; 225579251f5eSSepherosa Ziehau 225679251f5eSSepherosa Ziehau #if 0 225779251f5eSSepherosa Ziehau /* 225879251f5eSSepherosa Ziehau ** Now set up the LRO interface: 225979251f5eSSepherosa Ziehau */ 226079251f5eSSepherosa Ziehau if (ixgbe_rsc_enable) 226179251f5eSSepherosa Ziehau ix_setup_hw_rsc(rxr); 226279251f5eSSepherosa Ziehau #endif 226379251f5eSSepherosa Ziehau 226479251f5eSSepherosa Ziehau return 0; 226579251f5eSSepherosa Ziehau } 226679251f5eSSepherosa Ziehau 226779251f5eSSepherosa Ziehau #define IXGBE_SRRCTL_BSIZEHDRSIZE_SHIFT 2 226879251f5eSSepherosa Ziehau 226979251f5eSSepherosa Ziehau #define BSIZEPKT_ROUNDUP ((1<<IXGBE_SRRCTL_BSIZEPKT_SHIFT)-1) 227079251f5eSSepherosa Ziehau 227179251f5eSSepherosa Ziehau static void 227279251f5eSSepherosa Ziehau ix_init_rx_unit(struct ix_softc *sc) 227379251f5eSSepherosa Ziehau { 227479251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 227579251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 227679251f5eSSepherosa Ziehau uint32_t bufsz, rxctrl, fctrl, rxcsum, hlreg; 227779251f5eSSepherosa Ziehau int i; 227879251f5eSSepherosa Ziehau 227979251f5eSSepherosa Ziehau /* 228079251f5eSSepherosa Ziehau * Make sure receives are disabled while setting up the descriptor ring 228179251f5eSSepherosa Ziehau */ 228279251f5eSSepherosa Ziehau rxctrl = IXGBE_READ_REG(hw, IXGBE_RXCTRL); 228379251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RXCTRL, rxctrl & ~IXGBE_RXCTRL_RXEN); 228479251f5eSSepherosa Ziehau 228579251f5eSSepherosa Ziehau /* Enable broadcasts */ 228679251f5eSSepherosa Ziehau fctrl = IXGBE_READ_REG(hw, IXGBE_FCTRL); 228779251f5eSSepherosa Ziehau fctrl |= IXGBE_FCTRL_BAM; 228879251f5eSSepherosa Ziehau fctrl |= IXGBE_FCTRL_DPF; 228979251f5eSSepherosa Ziehau fctrl |= IXGBE_FCTRL_PMCF; 229079251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_FCTRL, fctrl); 229179251f5eSSepherosa Ziehau 229279251f5eSSepherosa Ziehau /* Set for Jumbo Frames? */ 229379251f5eSSepherosa Ziehau hlreg = IXGBE_READ_REG(hw, IXGBE_HLREG0); 229479251f5eSSepherosa Ziehau if (ifp->if_mtu > ETHERMTU) 229579251f5eSSepherosa Ziehau hlreg |= IXGBE_HLREG0_JUMBOEN; 229679251f5eSSepherosa Ziehau else 229779251f5eSSepherosa Ziehau hlreg &= ~IXGBE_HLREG0_JUMBOEN; 229879251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_HLREG0, hlreg); 229979251f5eSSepherosa Ziehau 230079251f5eSSepherosa Ziehau KKASSERT(sc->rx_rings[0].rx_mbuf_sz >= MCLBYTES); 230179251f5eSSepherosa Ziehau bufsz = (sc->rx_rings[0].rx_mbuf_sz + BSIZEPKT_ROUNDUP) >> 230279251f5eSSepherosa Ziehau IXGBE_SRRCTL_BSIZEPKT_SHIFT; 230379251f5eSSepherosa Ziehau 230479251f5eSSepherosa Ziehau for (i = 0; i < sc->rx_ring_inuse; ++i) { 230579251f5eSSepherosa Ziehau struct ix_rx_ring *rxr = &sc->rx_rings[i]; 230679251f5eSSepherosa Ziehau uint64_t rdba = rxr->rx_base_paddr; 230779251f5eSSepherosa Ziehau uint32_t srrctl; 230879251f5eSSepherosa Ziehau 230979251f5eSSepherosa Ziehau /* Setup the Base and Length of the Rx Descriptor Ring */ 231079251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RDBAL(i), (uint32_t)rdba); 231179251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RDBAH(i), (uint32_t)(rdba >> 32)); 231279251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RDLEN(i), 231379251f5eSSepherosa Ziehau rxr->rx_ndesc * sizeof(union ixgbe_adv_rx_desc)); 231479251f5eSSepherosa Ziehau 231579251f5eSSepherosa Ziehau /* 231679251f5eSSepherosa Ziehau * Set up the SRRCTL register 231779251f5eSSepherosa Ziehau */ 231879251f5eSSepherosa Ziehau srrctl = IXGBE_READ_REG(hw, IXGBE_SRRCTL(i)); 231979251f5eSSepherosa Ziehau 232079251f5eSSepherosa Ziehau srrctl &= ~IXGBE_SRRCTL_BSIZEHDR_MASK; 232179251f5eSSepherosa Ziehau srrctl &= ~IXGBE_SRRCTL_BSIZEPKT_MASK; 232279251f5eSSepherosa Ziehau srrctl |= bufsz; 232379251f5eSSepherosa Ziehau srrctl |= IXGBE_SRRCTL_DESCTYPE_ADV_ONEBUF; 232479251f5eSSepherosa Ziehau if (sc->rx_ring_inuse > 1) { 232579251f5eSSepherosa Ziehau /* See the commend near ix_enable_rx_drop() */ 232679251f5eSSepherosa Ziehau switch (sc->fc) { 232779251f5eSSepherosa Ziehau case ixgbe_fc_rx_pause: 232879251f5eSSepherosa Ziehau case ixgbe_fc_tx_pause: 232979251f5eSSepherosa Ziehau case ixgbe_fc_full: 233079251f5eSSepherosa Ziehau srrctl &= ~IXGBE_SRRCTL_DROP_EN; 233179251f5eSSepherosa Ziehau if (i == 0 && bootverbose) { 233279251f5eSSepherosa Ziehau if_printf(ifp, "flow control %d, " 233379251f5eSSepherosa Ziehau "disable RX drop\n", sc->fc); 233479251f5eSSepherosa Ziehau } 233579251f5eSSepherosa Ziehau break; 233679251f5eSSepherosa Ziehau 233779251f5eSSepherosa Ziehau case ixgbe_fc_none: 233879251f5eSSepherosa Ziehau srrctl |= IXGBE_SRRCTL_DROP_EN; 233979251f5eSSepherosa Ziehau if (i == 0 && bootverbose) { 234079251f5eSSepherosa Ziehau if_printf(ifp, "flow control %d, " 234179251f5eSSepherosa Ziehau "enable RX drop\n", sc->fc); 234279251f5eSSepherosa Ziehau } 234379251f5eSSepherosa Ziehau break; 234479251f5eSSepherosa Ziehau 234579251f5eSSepherosa Ziehau default: 234679251f5eSSepherosa Ziehau break; 234779251f5eSSepherosa Ziehau } 234879251f5eSSepherosa Ziehau } 234979251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_SRRCTL(i), srrctl); 235079251f5eSSepherosa Ziehau 235179251f5eSSepherosa Ziehau /* Setup the HW Rx Head and Tail Descriptor Pointers */ 235279251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RDH(i), 0); 235379251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RDT(i), 0); 235479251f5eSSepherosa Ziehau } 235579251f5eSSepherosa Ziehau 235679251f5eSSepherosa Ziehau if (sc->hw.mac.type != ixgbe_mac_82598EB) 235779251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_PSRTYPE(0), 0); 235879251f5eSSepherosa Ziehau 235979251f5eSSepherosa Ziehau rxcsum = IXGBE_READ_REG(hw, IXGBE_RXCSUM); 236079251f5eSSepherosa Ziehau 236179251f5eSSepherosa Ziehau /* 236279251f5eSSepherosa Ziehau * Setup RSS 236379251f5eSSepherosa Ziehau */ 236479251f5eSSepherosa Ziehau if (IX_ENABLE_HWRSS(sc)) { 236579251f5eSSepherosa Ziehau uint8_t key[IX_NRSSRK * IX_RSSRK_SIZE]; 236679251f5eSSepherosa Ziehau int j, r; 236779251f5eSSepherosa Ziehau 236879251f5eSSepherosa Ziehau /* 236979251f5eSSepherosa Ziehau * NOTE: 237079251f5eSSepherosa Ziehau * When we reach here, RSS has already been disabled 237179251f5eSSepherosa Ziehau * in ix_stop(), so we could safely configure RSS key 237279251f5eSSepherosa Ziehau * and redirect table. 237379251f5eSSepherosa Ziehau */ 237479251f5eSSepherosa Ziehau 237579251f5eSSepherosa Ziehau /* 237679251f5eSSepherosa Ziehau * Configure RSS key 237779251f5eSSepherosa Ziehau */ 237879251f5eSSepherosa Ziehau toeplitz_get_key(key, sizeof(key)); 237979251f5eSSepherosa Ziehau for (i = 0; i < IX_NRSSRK; ++i) { 238079251f5eSSepherosa Ziehau uint32_t rssrk; 238179251f5eSSepherosa Ziehau 238279251f5eSSepherosa Ziehau rssrk = IX_RSSRK_VAL(key, i); 238379251f5eSSepherosa Ziehau IX_RSS_DPRINTF(sc, 1, "rssrk%d 0x%08x\n", 238479251f5eSSepherosa Ziehau i, rssrk); 238579251f5eSSepherosa Ziehau 238679251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RSSRK(i), rssrk); 238779251f5eSSepherosa Ziehau } 238879251f5eSSepherosa Ziehau 238979251f5eSSepherosa Ziehau /* 239079251f5eSSepherosa Ziehau * Configure RSS redirect table in following fashion: 239179251f5eSSepherosa Ziehau * (hash & ring_cnt_mask) == rdr_table[(hash & rdr_table_mask)] 239279251f5eSSepherosa Ziehau */ 239379251f5eSSepherosa Ziehau r = 0; 239479251f5eSSepherosa Ziehau for (j = 0; j < IX_NRETA; ++j) { 239579251f5eSSepherosa Ziehau uint32_t reta = 0; 239679251f5eSSepherosa Ziehau 239779251f5eSSepherosa Ziehau for (i = 0; i < IX_RETA_SIZE; ++i) { 239879251f5eSSepherosa Ziehau uint32_t q; 239979251f5eSSepherosa Ziehau 240079251f5eSSepherosa Ziehau q = r % sc->rx_ring_inuse; 240179251f5eSSepherosa Ziehau reta |= q << (8 * i); 240279251f5eSSepherosa Ziehau ++r; 240379251f5eSSepherosa Ziehau } 240479251f5eSSepherosa Ziehau IX_RSS_DPRINTF(sc, 1, "reta 0x%08x\n", reta); 240579251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RETA(j), reta); 240679251f5eSSepherosa Ziehau } 240779251f5eSSepherosa Ziehau 240879251f5eSSepherosa Ziehau /* 240979251f5eSSepherosa Ziehau * Enable multiple receive queues. 241079251f5eSSepherosa Ziehau * Enable IPv4 RSS standard hash functions. 241179251f5eSSepherosa Ziehau */ 241279251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_MRQC, 241379251f5eSSepherosa Ziehau IXGBE_MRQC_RSSEN | 241479251f5eSSepherosa Ziehau IXGBE_MRQC_RSS_FIELD_IPV4 | 241579251f5eSSepherosa Ziehau IXGBE_MRQC_RSS_FIELD_IPV4_TCP); 241679251f5eSSepherosa Ziehau 241779251f5eSSepherosa Ziehau /* 241879251f5eSSepherosa Ziehau * NOTE: 241979251f5eSSepherosa Ziehau * PCSD must be enabled to enable multiple 242079251f5eSSepherosa Ziehau * receive queues. 242179251f5eSSepherosa Ziehau */ 242279251f5eSSepherosa Ziehau rxcsum |= IXGBE_RXCSUM_PCSD; 242379251f5eSSepherosa Ziehau } 242479251f5eSSepherosa Ziehau 242579251f5eSSepherosa Ziehau if (ifp->if_capenable & IFCAP_RXCSUM) 242679251f5eSSepherosa Ziehau rxcsum |= IXGBE_RXCSUM_PCSD; 242779251f5eSSepherosa Ziehau 242879251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RXCSUM, rxcsum); 242979251f5eSSepherosa Ziehau } 243079251f5eSSepherosa Ziehau 243179251f5eSSepherosa Ziehau static __inline void 243279251f5eSSepherosa Ziehau ix_rx_refresh(struct ix_rx_ring *rxr, int i) 243379251f5eSSepherosa Ziehau { 243479251f5eSSepherosa Ziehau if (--i < 0) 243579251f5eSSepherosa Ziehau i = rxr->rx_ndesc - 1; 243679251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&rxr->rx_sc->hw, IXGBE_RDT(rxr->rx_idx), i); 243779251f5eSSepherosa Ziehau } 243879251f5eSSepherosa Ziehau 243979251f5eSSepherosa Ziehau static __inline void 244079251f5eSSepherosa Ziehau ix_rxcsum(uint32_t staterr, struct mbuf *mp, uint32_t ptype) 244179251f5eSSepherosa Ziehau { 244279251f5eSSepherosa Ziehau if ((ptype & 244379251f5eSSepherosa Ziehau (IXGBE_RXDADV_PKTTYPE_IPV4 | IXGBE_RXDADV_PKTTYPE_IPV4_EX)) == 0) { 244479251f5eSSepherosa Ziehau /* Not IPv4 */ 244579251f5eSSepherosa Ziehau return; 244679251f5eSSepherosa Ziehau } 244779251f5eSSepherosa Ziehau 244879251f5eSSepherosa Ziehau if ((staterr & (IXGBE_RXD_STAT_IPCS | IXGBE_RXDADV_ERR_IPE)) == 244979251f5eSSepherosa Ziehau IXGBE_RXD_STAT_IPCS) 245079251f5eSSepherosa Ziehau mp->m_pkthdr.csum_flags |= CSUM_IP_CHECKED | CSUM_IP_VALID; 245179251f5eSSepherosa Ziehau 245279251f5eSSepherosa Ziehau if ((ptype & 245379251f5eSSepherosa Ziehau (IXGBE_RXDADV_PKTTYPE_TCP | IXGBE_RXDADV_PKTTYPE_UDP)) == 0) { 245479251f5eSSepherosa Ziehau /* 245579251f5eSSepherosa Ziehau * - Neither TCP nor UDP 245679251f5eSSepherosa Ziehau * - IPv4 fragment 245779251f5eSSepherosa Ziehau */ 245879251f5eSSepherosa Ziehau return; 245979251f5eSSepherosa Ziehau } 246079251f5eSSepherosa Ziehau 246179251f5eSSepherosa Ziehau if ((staterr & (IXGBE_RXD_STAT_L4CS | IXGBE_RXDADV_ERR_TCPE)) == 246279251f5eSSepherosa Ziehau IXGBE_RXD_STAT_L4CS) { 246379251f5eSSepherosa Ziehau mp->m_pkthdr.csum_flags |= CSUM_DATA_VALID | CSUM_PSEUDO_HDR | 246479251f5eSSepherosa Ziehau CSUM_FRAG_NOT_CHECKED; 246579251f5eSSepherosa Ziehau mp->m_pkthdr.csum_data = htons(0xffff); 246679251f5eSSepherosa Ziehau } 246779251f5eSSepherosa Ziehau } 246879251f5eSSepherosa Ziehau 246979251f5eSSepherosa Ziehau static __inline struct pktinfo * 247079251f5eSSepherosa Ziehau ix_rssinfo(struct mbuf *m, struct pktinfo *pi, 247179251f5eSSepherosa Ziehau uint32_t hash, uint32_t hashtype, uint32_t ptype) 247279251f5eSSepherosa Ziehau { 247379251f5eSSepherosa Ziehau switch (hashtype) { 247479251f5eSSepherosa Ziehau case IXGBE_RXDADV_RSSTYPE_IPV4_TCP: 247579251f5eSSepherosa Ziehau pi->pi_netisr = NETISR_IP; 247679251f5eSSepherosa Ziehau pi->pi_flags = 0; 247779251f5eSSepherosa Ziehau pi->pi_l3proto = IPPROTO_TCP; 247879251f5eSSepherosa Ziehau break; 247979251f5eSSepherosa Ziehau 248079251f5eSSepherosa Ziehau case IXGBE_RXDADV_RSSTYPE_IPV4: 248179251f5eSSepherosa Ziehau if ((ptype & IXGBE_RXDADV_PKTTYPE_UDP) == 0) { 248279251f5eSSepherosa Ziehau /* Not UDP or is fragment */ 248379251f5eSSepherosa Ziehau return NULL; 248479251f5eSSepherosa Ziehau } 248579251f5eSSepherosa Ziehau pi->pi_netisr = NETISR_IP; 248679251f5eSSepherosa Ziehau pi->pi_flags = 0; 248779251f5eSSepherosa Ziehau pi->pi_l3proto = IPPROTO_UDP; 248879251f5eSSepherosa Ziehau break; 248979251f5eSSepherosa Ziehau 249079251f5eSSepherosa Ziehau default: 249179251f5eSSepherosa Ziehau return NULL; 249279251f5eSSepherosa Ziehau } 249379251f5eSSepherosa Ziehau 249479251f5eSSepherosa Ziehau m->m_flags |= M_HASH; 249579251f5eSSepherosa Ziehau m->m_pkthdr.hash = toeplitz_hash(hash); 249679251f5eSSepherosa Ziehau return pi; 249779251f5eSSepherosa Ziehau } 249879251f5eSSepherosa Ziehau 249979251f5eSSepherosa Ziehau static __inline void 250079251f5eSSepherosa Ziehau ix_setup_rxdesc(union ixgbe_adv_rx_desc *rxd, const struct ix_rx_buf *rxbuf) 250179251f5eSSepherosa Ziehau { 250279251f5eSSepherosa Ziehau rxd->read.pkt_addr = htole64(rxbuf->paddr); 250379251f5eSSepherosa Ziehau rxd->wb.upper.status_error = 0; 250479251f5eSSepherosa Ziehau } 250579251f5eSSepherosa Ziehau 250679251f5eSSepherosa Ziehau static void 250779251f5eSSepherosa Ziehau ix_rx_discard(struct ix_rx_ring *rxr, int i, boolean_t eop) 250879251f5eSSepherosa Ziehau { 250979251f5eSSepherosa Ziehau struct ix_rx_buf *rxbuf = &rxr->rx_buf[i]; 251079251f5eSSepherosa Ziehau 251179251f5eSSepherosa Ziehau /* 251279251f5eSSepherosa Ziehau * XXX discard may not be correct 251379251f5eSSepherosa Ziehau */ 251479251f5eSSepherosa Ziehau if (eop) { 251579251f5eSSepherosa Ziehau IFNET_STAT_INC(&rxr->rx_sc->arpcom.ac_if, ierrors, 1); 251679251f5eSSepherosa Ziehau rxr->rx_flags &= ~IX_RXRING_FLAG_DISC; 251779251f5eSSepherosa Ziehau } else { 251879251f5eSSepherosa Ziehau rxr->rx_flags |= IX_RXRING_FLAG_DISC; 251979251f5eSSepherosa Ziehau } 252079251f5eSSepherosa Ziehau if (rxbuf->fmp != NULL) { 252179251f5eSSepherosa Ziehau m_freem(rxbuf->fmp); 252279251f5eSSepherosa Ziehau rxbuf->fmp = NULL; 252379251f5eSSepherosa Ziehau rxbuf->lmp = NULL; 252479251f5eSSepherosa Ziehau } 252579251f5eSSepherosa Ziehau ix_setup_rxdesc(&rxr->rx_base[i], rxbuf); 252679251f5eSSepherosa Ziehau } 252779251f5eSSepherosa Ziehau 252879251f5eSSepherosa Ziehau static void 252979251f5eSSepherosa Ziehau ix_rxeof(struct ix_rx_ring *rxr) 253079251f5eSSepherosa Ziehau { 253179251f5eSSepherosa Ziehau struct ifnet *ifp = &rxr->rx_sc->arpcom.ac_if; 253279251f5eSSepherosa Ziehau int i, nsegs = 0; 253379251f5eSSepherosa Ziehau 253479251f5eSSepherosa Ziehau i = rxr->rx_next_check; 253579251f5eSSepherosa Ziehau for (;;) { 253679251f5eSSepherosa Ziehau struct ix_rx_buf *rxbuf, *nbuf = NULL; 253779251f5eSSepherosa Ziehau union ixgbe_adv_rx_desc *cur; 253879251f5eSSepherosa Ziehau struct mbuf *sendmp = NULL, *mp; 253979251f5eSSepherosa Ziehau struct pktinfo *pi = NULL, pi0; 254079251f5eSSepherosa Ziehau uint32_t rsc = 0, ptype, staterr, hash, hashtype; 254179251f5eSSepherosa Ziehau uint16_t len; 254279251f5eSSepherosa Ziehau boolean_t eop; 254379251f5eSSepherosa Ziehau 254479251f5eSSepherosa Ziehau cur = &rxr->rx_base[i]; 254579251f5eSSepherosa Ziehau staterr = le32toh(cur->wb.upper.status_error); 254679251f5eSSepherosa Ziehau 254779251f5eSSepherosa Ziehau if ((staterr & IXGBE_RXD_STAT_DD) == 0) 254879251f5eSSepherosa Ziehau break; 254979251f5eSSepherosa Ziehau ++nsegs; 255079251f5eSSepherosa Ziehau 255179251f5eSSepherosa Ziehau rxbuf = &rxr->rx_buf[i]; 255279251f5eSSepherosa Ziehau mp = rxbuf->m_head; 255379251f5eSSepherosa Ziehau 255479251f5eSSepherosa Ziehau len = le16toh(cur->wb.upper.length); 255579251f5eSSepherosa Ziehau ptype = le32toh(cur->wb.lower.lo_dword.data) & 255679251f5eSSepherosa Ziehau IXGBE_RXDADV_PKTTYPE_MASK; 255779251f5eSSepherosa Ziehau hash = le32toh(cur->wb.lower.hi_dword.rss); 255879251f5eSSepherosa Ziehau hashtype = le32toh(cur->wb.lower.lo_dword.data) & 255979251f5eSSepherosa Ziehau IXGBE_RXDADV_RSSTYPE_MASK; 256079251f5eSSepherosa Ziehau eop = ((staterr & IXGBE_RXD_STAT_EOP) != 0); 256179251f5eSSepherosa Ziehau 256279251f5eSSepherosa Ziehau /* 256379251f5eSSepherosa Ziehau * Make sure bad packets are discarded 256479251f5eSSepherosa Ziehau */ 256579251f5eSSepherosa Ziehau if ((staterr & IXGBE_RXDADV_ERR_FRAME_ERR_MASK) || 256679251f5eSSepherosa Ziehau (rxr->rx_flags & IX_RXRING_FLAG_DISC)) { 256779251f5eSSepherosa Ziehau ix_rx_discard(rxr, i, eop); 256879251f5eSSepherosa Ziehau goto next_desc; 256979251f5eSSepherosa Ziehau } 257079251f5eSSepherosa Ziehau 257179251f5eSSepherosa Ziehau bus_dmamap_sync(rxr->rx_tag, rxbuf->map, BUS_DMASYNC_POSTREAD); 257279251f5eSSepherosa Ziehau if (ix_newbuf(rxr, i, FALSE) != 0) { 257379251f5eSSepherosa Ziehau ix_rx_discard(rxr, i, eop); 257479251f5eSSepherosa Ziehau goto next_desc; 257579251f5eSSepherosa Ziehau } 257679251f5eSSepherosa Ziehau 257779251f5eSSepherosa Ziehau /* 257879251f5eSSepherosa Ziehau * On 82599 which supports a hardware LRO, packets 257979251f5eSSepherosa Ziehau * need not be fragmented across sequential descriptors, 258079251f5eSSepherosa Ziehau * rather the next descriptor is indicated in bits 258179251f5eSSepherosa Ziehau * of the descriptor. This also means that we might 258279251f5eSSepherosa Ziehau * proceses more than one packet at a time, something 258379251f5eSSepherosa Ziehau * that has never been true before, it required 258479251f5eSSepherosa Ziehau * eliminating global chain pointers in favor of what 258579251f5eSSepherosa Ziehau * we are doing here. 258679251f5eSSepherosa Ziehau */ 258779251f5eSSepherosa Ziehau if (!eop) { 258879251f5eSSepherosa Ziehau int nextp; 258979251f5eSSepherosa Ziehau 259079251f5eSSepherosa Ziehau /* 259179251f5eSSepherosa Ziehau * Figure out the next descriptor 259279251f5eSSepherosa Ziehau * of this frame. 259379251f5eSSepherosa Ziehau */ 259479251f5eSSepherosa Ziehau if (rxr->rx_flags & IX_RXRING_FLAG_LRO) 259579251f5eSSepherosa Ziehau rsc = ix_rsc_count(cur); 259679251f5eSSepherosa Ziehau if (rsc) { /* Get hardware index */ 259779251f5eSSepherosa Ziehau nextp = ((staterr & 259879251f5eSSepherosa Ziehau IXGBE_RXDADV_NEXTP_MASK) >> 259979251f5eSSepherosa Ziehau IXGBE_RXDADV_NEXTP_SHIFT); 260079251f5eSSepherosa Ziehau } else { /* Just sequential */ 260179251f5eSSepherosa Ziehau nextp = i + 1; 260279251f5eSSepherosa Ziehau if (nextp == rxr->rx_ndesc) 260379251f5eSSepherosa Ziehau nextp = 0; 260479251f5eSSepherosa Ziehau } 260579251f5eSSepherosa Ziehau nbuf = &rxr->rx_buf[nextp]; 260679251f5eSSepherosa Ziehau prefetch(nbuf); 260779251f5eSSepherosa Ziehau } 260879251f5eSSepherosa Ziehau mp->m_len = len; 260979251f5eSSepherosa Ziehau 261079251f5eSSepherosa Ziehau /* 261179251f5eSSepherosa Ziehau * Rather than using the fmp/lmp global pointers 261279251f5eSSepherosa Ziehau * we now keep the head of a packet chain in the 261379251f5eSSepherosa Ziehau * buffer struct and pass this along from one 261479251f5eSSepherosa Ziehau * descriptor to the next, until we get EOP. 261579251f5eSSepherosa Ziehau */ 261679251f5eSSepherosa Ziehau if (rxbuf->fmp == NULL) { 261779251f5eSSepherosa Ziehau mp->m_pkthdr.len = len; 261879251f5eSSepherosa Ziehau rxbuf->fmp = mp; 261979251f5eSSepherosa Ziehau rxbuf->lmp = mp; 262079251f5eSSepherosa Ziehau } else { 262179251f5eSSepherosa Ziehau rxbuf->fmp->m_pkthdr.len += len; 262279251f5eSSepherosa Ziehau rxbuf->lmp->m_next = mp; 262379251f5eSSepherosa Ziehau rxbuf->lmp = mp; 262479251f5eSSepherosa Ziehau } 262579251f5eSSepherosa Ziehau 262679251f5eSSepherosa Ziehau if (nbuf != NULL) { 262779251f5eSSepherosa Ziehau /* 262879251f5eSSepherosa Ziehau * Not the last fragment of this frame, 262979251f5eSSepherosa Ziehau * pass this fragment list on 263079251f5eSSepherosa Ziehau */ 263179251f5eSSepherosa Ziehau nbuf->fmp = rxbuf->fmp; 263279251f5eSSepherosa Ziehau nbuf->lmp = rxbuf->lmp; 263379251f5eSSepherosa Ziehau } else { 263479251f5eSSepherosa Ziehau /* 263579251f5eSSepherosa Ziehau * Send this frame 263679251f5eSSepherosa Ziehau */ 263779251f5eSSepherosa Ziehau sendmp = rxbuf->fmp; 263879251f5eSSepherosa Ziehau 263979251f5eSSepherosa Ziehau sendmp->m_pkthdr.rcvif = ifp; 264079251f5eSSepherosa Ziehau IFNET_STAT_INC(ifp, ipackets, 1); 264179251f5eSSepherosa Ziehau #ifdef IX_RSS_DEBUG 264279251f5eSSepherosa Ziehau rxr->rx_pkts++; 264379251f5eSSepherosa Ziehau #endif 264479251f5eSSepherosa Ziehau 264579251f5eSSepherosa Ziehau /* Process vlan info */ 264679251f5eSSepherosa Ziehau if (staterr & IXGBE_RXD_STAT_VP) { 264779251f5eSSepherosa Ziehau sendmp->m_pkthdr.ether_vlantag = 264879251f5eSSepherosa Ziehau le16toh(cur->wb.upper.vlan); 264979251f5eSSepherosa Ziehau sendmp->m_flags |= M_VLANTAG; 265079251f5eSSepherosa Ziehau } 265179251f5eSSepherosa Ziehau if (ifp->if_capenable & IFCAP_RXCSUM) 265279251f5eSSepherosa Ziehau ix_rxcsum(staterr, sendmp, ptype); 265379251f5eSSepherosa Ziehau if (ifp->if_capenable & IFCAP_RSS) { 265479251f5eSSepherosa Ziehau pi = ix_rssinfo(sendmp, &pi0, 265579251f5eSSepherosa Ziehau hash, hashtype, ptype); 265679251f5eSSepherosa Ziehau } 265779251f5eSSepherosa Ziehau } 265879251f5eSSepherosa Ziehau rxbuf->fmp = NULL; 265979251f5eSSepherosa Ziehau rxbuf->lmp = NULL; 266079251f5eSSepherosa Ziehau next_desc: 266179251f5eSSepherosa Ziehau /* Advance our pointers to the next descriptor. */ 266279251f5eSSepherosa Ziehau if (++i == rxr->rx_ndesc) 266379251f5eSSepherosa Ziehau i = 0; 266479251f5eSSepherosa Ziehau 266579251f5eSSepherosa Ziehau if (sendmp != NULL) 266679251f5eSSepherosa Ziehau ether_input_pkt(ifp, sendmp, pi); 266779251f5eSSepherosa Ziehau 266879251f5eSSepherosa Ziehau if (nsegs >= rxr->rx_wreg_nsegs) { 266979251f5eSSepherosa Ziehau ix_rx_refresh(rxr, i); 267079251f5eSSepherosa Ziehau nsegs = 0; 267179251f5eSSepherosa Ziehau } 267279251f5eSSepherosa Ziehau } 267379251f5eSSepherosa Ziehau rxr->rx_next_check = i; 267479251f5eSSepherosa Ziehau 267579251f5eSSepherosa Ziehau if (nsegs > 0) 267679251f5eSSepherosa Ziehau ix_rx_refresh(rxr, i); 267779251f5eSSepherosa Ziehau } 267879251f5eSSepherosa Ziehau 267979251f5eSSepherosa Ziehau static void 268079251f5eSSepherosa Ziehau ix_set_vlan(struct ix_softc *sc) 268179251f5eSSepherosa Ziehau { 268279251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 268379251f5eSSepherosa Ziehau uint32_t ctrl; 268479251f5eSSepherosa Ziehau 268579251f5eSSepherosa Ziehau if (hw->mac.type == ixgbe_mac_82598EB) { 268679251f5eSSepherosa Ziehau ctrl = IXGBE_READ_REG(hw, IXGBE_VLNCTRL); 268779251f5eSSepherosa Ziehau ctrl |= IXGBE_VLNCTRL_VME; 268879251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_VLNCTRL, ctrl); 268979251f5eSSepherosa Ziehau } else { 269079251f5eSSepherosa Ziehau int i; 269179251f5eSSepherosa Ziehau 269279251f5eSSepherosa Ziehau /* 269379251f5eSSepherosa Ziehau * On 82599 and later chips the VLAN enable is 269479251f5eSSepherosa Ziehau * per queue in RXDCTL 269579251f5eSSepherosa Ziehau */ 269679251f5eSSepherosa Ziehau for (i = 0; i < sc->rx_ring_inuse; ++i) { 269779251f5eSSepherosa Ziehau ctrl = IXGBE_READ_REG(hw, IXGBE_RXDCTL(i)); 269879251f5eSSepherosa Ziehau ctrl |= IXGBE_RXDCTL_VME; 269979251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RXDCTL(i), ctrl); 270079251f5eSSepherosa Ziehau } 270179251f5eSSepherosa Ziehau } 270279251f5eSSepherosa Ziehau } 270379251f5eSSepherosa Ziehau 270479251f5eSSepherosa Ziehau static void 270579251f5eSSepherosa Ziehau ix_enable_intr(struct ix_softc *sc) 270679251f5eSSepherosa Ziehau { 270779251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 2708*189a0ff3SSepherosa Ziehau uint32_t fwsm; 270979251f5eSSepherosa Ziehau int i; 271079251f5eSSepherosa Ziehau 271179251f5eSSepherosa Ziehau for (i = 0; i < sc->intr_cnt; ++i) 271279251f5eSSepherosa Ziehau lwkt_serialize_handler_enable(sc->intr_data[i].intr_serialize); 271379251f5eSSepherosa Ziehau 2714*189a0ff3SSepherosa Ziehau sc->intr_mask = (IXGBE_EIMS_ENABLE_MASK & ~IXGBE_EIMS_RTX_QUEUE); 271579251f5eSSepherosa Ziehau 271679251f5eSSepherosa Ziehau /* Enable Fan Failure detection */ 271779251f5eSSepherosa Ziehau if (hw->device_id == IXGBE_DEV_ID_82598AT) 2718*189a0ff3SSepherosa Ziehau sc->intr_mask |= IXGBE_EIMS_GPI_SDP1; 271979251f5eSSepherosa Ziehau 272079251f5eSSepherosa Ziehau switch (sc->hw.mac.type) { 272179251f5eSSepherosa Ziehau case ixgbe_mac_82599EB: 2722*189a0ff3SSepherosa Ziehau sc->intr_mask |= IXGBE_EIMS_ECC; 2723*189a0ff3SSepherosa Ziehau sc->intr_mask |= IXGBE_EIMS_GPI_SDP0; 2724*189a0ff3SSepherosa Ziehau sc->intr_mask |= IXGBE_EIMS_GPI_SDP1; 2725*189a0ff3SSepherosa Ziehau sc->intr_mask |= IXGBE_EIMS_GPI_SDP2; 272679251f5eSSepherosa Ziehau break; 2727*189a0ff3SSepherosa Ziehau 272879251f5eSSepherosa Ziehau case ixgbe_mac_X540: 2729*189a0ff3SSepherosa Ziehau sc->intr_mask |= IXGBE_EIMS_ECC; 273079251f5eSSepherosa Ziehau /* Detect if Thermal Sensor is enabled */ 273179251f5eSSepherosa Ziehau fwsm = IXGBE_READ_REG(hw, IXGBE_FWSM); 273279251f5eSSepherosa Ziehau if (fwsm & IXGBE_FWSM_TS_ENABLED) 2733*189a0ff3SSepherosa Ziehau sc->intr_mask |= IXGBE_EIMS_TS; 273479251f5eSSepherosa Ziehau /* FALL THROUGH */ 273579251f5eSSepherosa Ziehau default: 273679251f5eSSepherosa Ziehau break; 273779251f5eSSepherosa Ziehau } 273879251f5eSSepherosa Ziehau 2739*189a0ff3SSepherosa Ziehau /* With MSI-X we use auto clear for RX and TX rings */ 274079251f5eSSepherosa Ziehau if (sc->intr_type == PCI_INTR_TYPE_MSIX) { 2741*189a0ff3SSepherosa Ziehau /* 2742*189a0ff3SSepherosa Ziehau * There are no EIAC1/EIAC2 for newer chips; the related 2743*189a0ff3SSepherosa Ziehau * bits for TX and RX rings > 16 are always auto clear. 2744*189a0ff3SSepherosa Ziehau * 2745*189a0ff3SSepherosa Ziehau * XXX which bits? There are _no_ documented EICR1 and 2746*189a0ff3SSepherosa Ziehau * EICR2 at all; only EICR. 2747*189a0ff3SSepherosa Ziehau */ 2748*189a0ff3SSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_EIAC, IXGBE_EIMS_RTX_QUEUE); 274979251f5eSSepherosa Ziehau } else { 2750*189a0ff3SSepherosa Ziehau sc->intr_mask |= IX_TX_INTR_MASK | IX_RX0_INTR_MASK; 275179251f5eSSepherosa Ziehau 275279251f5eSSepherosa Ziehau KKASSERT(sc->rx_ring_inuse <= IX_MIN_RXRING_RSS); 275379251f5eSSepherosa Ziehau if (sc->rx_ring_inuse == IX_MIN_RXRING_RSS) 275479251f5eSSepherosa Ziehau sc->intr_mask |= IX_RX1_INTR_MASK; 275579251f5eSSepherosa Ziehau } 275679251f5eSSepherosa Ziehau 275779251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_EIMS, sc->intr_mask); 2758*189a0ff3SSepherosa Ziehau 2759*189a0ff3SSepherosa Ziehau /* 2760*189a0ff3SSepherosa Ziehau * Enable RX and TX rings for MSI-X 2761*189a0ff3SSepherosa Ziehau */ 2762*189a0ff3SSepherosa Ziehau if (sc->intr_type == PCI_INTR_TYPE_MSIX) { 2763*189a0ff3SSepherosa Ziehau for (i = 0; i < sc->tx_ring_inuse; ++i) { 2764*189a0ff3SSepherosa Ziehau const struct ix_tx_ring *txr = &sc->tx_rings[i]; 2765*189a0ff3SSepherosa Ziehau 2766*189a0ff3SSepherosa Ziehau if (txr->tx_intr_vec >= 0) { 2767*189a0ff3SSepherosa Ziehau IXGBE_WRITE_REG(hw, txr->tx_eims, 2768*189a0ff3SSepherosa Ziehau txr->tx_eims_val); 2769*189a0ff3SSepherosa Ziehau } 2770*189a0ff3SSepherosa Ziehau } 2771*189a0ff3SSepherosa Ziehau for (i = 0; i < sc->rx_ring_inuse; ++i) { 2772*189a0ff3SSepherosa Ziehau const struct ix_rx_ring *rxr = &sc->rx_rings[i]; 2773*189a0ff3SSepherosa Ziehau 2774*189a0ff3SSepherosa Ziehau KKASSERT(rxr->rx_intr_vec >= 0); 2775*189a0ff3SSepherosa Ziehau IXGBE_WRITE_REG(hw, rxr->rx_eims, rxr->rx_eims_val); 2776*189a0ff3SSepherosa Ziehau } 2777*189a0ff3SSepherosa Ziehau } 277879251f5eSSepherosa Ziehau 277979251f5eSSepherosa Ziehau IXGBE_WRITE_FLUSH(hw); 278079251f5eSSepherosa Ziehau } 278179251f5eSSepherosa Ziehau 278279251f5eSSepherosa Ziehau static void 278379251f5eSSepherosa Ziehau ix_disable_intr(struct ix_softc *sc) 278479251f5eSSepherosa Ziehau { 278579251f5eSSepherosa Ziehau int i; 278679251f5eSSepherosa Ziehau 2787*189a0ff3SSepherosa Ziehau if (sc->intr_type == PCI_INTR_TYPE_MSIX) 278879251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_EIAC, 0); 2789*189a0ff3SSepherosa Ziehau 279079251f5eSSepherosa Ziehau if (sc->hw.mac.type == ixgbe_mac_82598EB) { 279179251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_EIMC, ~0); 279279251f5eSSepherosa Ziehau } else { 279379251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_EIMC, 0xFFFF0000); 279479251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_EIMC_EX(0), ~0); 279579251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_EIMC_EX(1), ~0); 279679251f5eSSepherosa Ziehau } 279779251f5eSSepherosa Ziehau IXGBE_WRITE_FLUSH(&sc->hw); 279879251f5eSSepherosa Ziehau 279979251f5eSSepherosa Ziehau for (i = 0; i < sc->intr_cnt; ++i) 280079251f5eSSepherosa Ziehau lwkt_serialize_handler_disable(sc->intr_data[i].intr_serialize); 280179251f5eSSepherosa Ziehau } 280279251f5eSSepherosa Ziehau 280379251f5eSSepherosa Ziehau uint16_t 280479251f5eSSepherosa Ziehau ixgbe_read_pci_cfg(struct ixgbe_hw *hw, uint32_t reg) 280579251f5eSSepherosa Ziehau { 280679251f5eSSepherosa Ziehau return pci_read_config(((struct ixgbe_osdep *)hw->back)->dev, 280779251f5eSSepherosa Ziehau reg, 2); 280879251f5eSSepherosa Ziehau } 280979251f5eSSepherosa Ziehau 281079251f5eSSepherosa Ziehau void 281179251f5eSSepherosa Ziehau ixgbe_write_pci_cfg(struct ixgbe_hw *hw, uint32_t reg, uint16_t value) 281279251f5eSSepherosa Ziehau { 281379251f5eSSepherosa Ziehau pci_write_config(((struct ixgbe_osdep *)hw->back)->dev, 281479251f5eSSepherosa Ziehau reg, value, 2); 281579251f5eSSepherosa Ziehau } 281679251f5eSSepherosa Ziehau 281779251f5eSSepherosa Ziehau static void 281879251f5eSSepherosa Ziehau ix_slot_info(struct ix_softc *sc) 281979251f5eSSepherosa Ziehau { 282079251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 282179251f5eSSepherosa Ziehau device_t dev = sc->dev; 282279251f5eSSepherosa Ziehau struct ixgbe_mac_info *mac = &hw->mac; 282379251f5eSSepherosa Ziehau uint16_t link; 282479251f5eSSepherosa Ziehau uint32_t offset; 282579251f5eSSepherosa Ziehau 282679251f5eSSepherosa Ziehau /* For most devices simply call the shared code routine */ 282779251f5eSSepherosa Ziehau if (hw->device_id != IXGBE_DEV_ID_82599_SFP_SF_QP) { 282879251f5eSSepherosa Ziehau ixgbe_get_bus_info(hw); 282979251f5eSSepherosa Ziehau goto display; 283079251f5eSSepherosa Ziehau } 283179251f5eSSepherosa Ziehau 283279251f5eSSepherosa Ziehau /* 283379251f5eSSepherosa Ziehau * For the Quad port adapter we need to parse back 283479251f5eSSepherosa Ziehau * up the PCI tree to find the speed of the expansion 283579251f5eSSepherosa Ziehau * slot into which this adapter is plugged. A bit more work. 283679251f5eSSepherosa Ziehau */ 283779251f5eSSepherosa Ziehau dev = device_get_parent(device_get_parent(dev)); 283879251f5eSSepherosa Ziehau #ifdef IXGBE_DEBUG 283979251f5eSSepherosa Ziehau device_printf(dev, "parent pcib = %x,%x,%x\n", 284079251f5eSSepherosa Ziehau pci_get_bus(dev), pci_get_slot(dev), pci_get_function(dev)); 284179251f5eSSepherosa Ziehau #endif 284279251f5eSSepherosa Ziehau dev = device_get_parent(device_get_parent(dev)); 284379251f5eSSepherosa Ziehau #ifdef IXGBE_DEBUG 284479251f5eSSepherosa Ziehau device_printf(dev, "slot pcib = %x,%x,%x\n", 284579251f5eSSepherosa Ziehau pci_get_bus(dev), pci_get_slot(dev), pci_get_function(dev)); 284679251f5eSSepherosa Ziehau #endif 284779251f5eSSepherosa Ziehau /* Now get the PCI Express Capabilities offset */ 284879251f5eSSepherosa Ziehau offset = pci_get_pciecap_ptr(dev); 284979251f5eSSepherosa Ziehau /* ...and read the Link Status Register */ 285079251f5eSSepherosa Ziehau link = pci_read_config(dev, offset + PCIER_LINKSTAT, 2); 285179251f5eSSepherosa Ziehau switch (link & IXGBE_PCI_LINK_WIDTH) { 285279251f5eSSepherosa Ziehau case IXGBE_PCI_LINK_WIDTH_1: 285379251f5eSSepherosa Ziehau hw->bus.width = ixgbe_bus_width_pcie_x1; 285479251f5eSSepherosa Ziehau break; 285579251f5eSSepherosa Ziehau case IXGBE_PCI_LINK_WIDTH_2: 285679251f5eSSepherosa Ziehau hw->bus.width = ixgbe_bus_width_pcie_x2; 285779251f5eSSepherosa Ziehau break; 285879251f5eSSepherosa Ziehau case IXGBE_PCI_LINK_WIDTH_4: 285979251f5eSSepherosa Ziehau hw->bus.width = ixgbe_bus_width_pcie_x4; 286079251f5eSSepherosa Ziehau break; 286179251f5eSSepherosa Ziehau case IXGBE_PCI_LINK_WIDTH_8: 286279251f5eSSepherosa Ziehau hw->bus.width = ixgbe_bus_width_pcie_x8; 286379251f5eSSepherosa Ziehau break; 286479251f5eSSepherosa Ziehau default: 286579251f5eSSepherosa Ziehau hw->bus.width = ixgbe_bus_width_unknown; 286679251f5eSSepherosa Ziehau break; 286779251f5eSSepherosa Ziehau } 286879251f5eSSepherosa Ziehau 286979251f5eSSepherosa Ziehau switch (link & IXGBE_PCI_LINK_SPEED) { 287079251f5eSSepherosa Ziehau case IXGBE_PCI_LINK_SPEED_2500: 287179251f5eSSepherosa Ziehau hw->bus.speed = ixgbe_bus_speed_2500; 287279251f5eSSepherosa Ziehau break; 287379251f5eSSepherosa Ziehau case IXGBE_PCI_LINK_SPEED_5000: 287479251f5eSSepherosa Ziehau hw->bus.speed = ixgbe_bus_speed_5000; 287579251f5eSSepherosa Ziehau break; 287679251f5eSSepherosa Ziehau case IXGBE_PCI_LINK_SPEED_8000: 287779251f5eSSepherosa Ziehau hw->bus.speed = ixgbe_bus_speed_8000; 287879251f5eSSepherosa Ziehau break; 287979251f5eSSepherosa Ziehau default: 288079251f5eSSepherosa Ziehau hw->bus.speed = ixgbe_bus_speed_unknown; 288179251f5eSSepherosa Ziehau break; 288279251f5eSSepherosa Ziehau } 288379251f5eSSepherosa Ziehau 288479251f5eSSepherosa Ziehau mac->ops.set_lan_id(hw); 288579251f5eSSepherosa Ziehau 288679251f5eSSepherosa Ziehau display: 288779251f5eSSepherosa Ziehau device_printf(dev, "PCI Express Bus: Speed %s %s\n", 288879251f5eSSepherosa Ziehau hw->bus.speed == ixgbe_bus_speed_8000 ? "8.0GT/s" : 288979251f5eSSepherosa Ziehau hw->bus.speed == ixgbe_bus_speed_5000 ? "5.0GT/s" : 289079251f5eSSepherosa Ziehau hw->bus.speed == ixgbe_bus_speed_2500 ? "2.5GT/s" : "Unknown", 289179251f5eSSepherosa Ziehau hw->bus.width == ixgbe_bus_width_pcie_x8 ? "Width x8" : 289279251f5eSSepherosa Ziehau hw->bus.width == ixgbe_bus_width_pcie_x4 ? "Width x4" : 289379251f5eSSepherosa Ziehau hw->bus.width == ixgbe_bus_width_pcie_x1 ? "Width x1" : "Unknown"); 289479251f5eSSepherosa Ziehau 289579251f5eSSepherosa Ziehau if (hw->device_id != IXGBE_DEV_ID_82599_SFP_SF_QP && 289679251f5eSSepherosa Ziehau hw->bus.width <= ixgbe_bus_width_pcie_x4 && 289779251f5eSSepherosa Ziehau hw->bus.speed == ixgbe_bus_speed_2500) { 289879251f5eSSepherosa Ziehau device_printf(dev, "For optimal performance a x8 " 289979251f5eSSepherosa Ziehau "PCIE, or x4 PCIE Gen2 slot is required.\n"); 290079251f5eSSepherosa Ziehau } else if (hw->device_id == IXGBE_DEV_ID_82599_SFP_SF_QP && 290179251f5eSSepherosa Ziehau hw->bus.width <= ixgbe_bus_width_pcie_x8 && 290279251f5eSSepherosa Ziehau hw->bus.speed < ixgbe_bus_speed_8000) { 290379251f5eSSepherosa Ziehau device_printf(dev, "For optimal performance a x8 " 290479251f5eSSepherosa Ziehau "PCIE Gen3 slot is required.\n"); 290579251f5eSSepherosa Ziehau } 290679251f5eSSepherosa Ziehau } 290779251f5eSSepherosa Ziehau 290879251f5eSSepherosa Ziehau /* 290979251f5eSSepherosa Ziehau * TODO comment is incorrect 291079251f5eSSepherosa Ziehau * 291179251f5eSSepherosa Ziehau * Setup the correct IVAR register for a particular MSIX interrupt 291279251f5eSSepherosa Ziehau * - entry is the register array entry 291379251f5eSSepherosa Ziehau * - vector is the MSIX vector for this queue 291479251f5eSSepherosa Ziehau * - type is RX/TX/MISC 291579251f5eSSepherosa Ziehau */ 291679251f5eSSepherosa Ziehau static void 291779251f5eSSepherosa Ziehau ix_set_ivar(struct ix_softc *sc, uint8_t entry, uint8_t vector, 291879251f5eSSepherosa Ziehau int8_t type) 291979251f5eSSepherosa Ziehau { 292079251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 292179251f5eSSepherosa Ziehau uint32_t ivar, index; 292279251f5eSSepherosa Ziehau 292379251f5eSSepherosa Ziehau vector |= IXGBE_IVAR_ALLOC_VAL; 292479251f5eSSepherosa Ziehau 292579251f5eSSepherosa Ziehau switch (hw->mac.type) { 292679251f5eSSepherosa Ziehau case ixgbe_mac_82598EB: 292779251f5eSSepherosa Ziehau if (type == -1) 292879251f5eSSepherosa Ziehau entry = IXGBE_IVAR_OTHER_CAUSES_INDEX; 292979251f5eSSepherosa Ziehau else 293079251f5eSSepherosa Ziehau entry += (type * 64); 293179251f5eSSepherosa Ziehau index = (entry >> 2) & 0x1F; 293279251f5eSSepherosa Ziehau ivar = IXGBE_READ_REG(hw, IXGBE_IVAR(index)); 293379251f5eSSepherosa Ziehau ivar &= ~(0xFF << (8 * (entry & 0x3))); 293479251f5eSSepherosa Ziehau ivar |= (vector << (8 * (entry & 0x3))); 293579251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_IVAR(index), ivar); 293679251f5eSSepherosa Ziehau break; 293779251f5eSSepherosa Ziehau 293879251f5eSSepherosa Ziehau case ixgbe_mac_82599EB: 293979251f5eSSepherosa Ziehau case ixgbe_mac_X540: 294079251f5eSSepherosa Ziehau if (type == -1) { /* MISC IVAR */ 294179251f5eSSepherosa Ziehau index = (entry & 1) * 8; 294279251f5eSSepherosa Ziehau ivar = IXGBE_READ_REG(hw, IXGBE_IVAR_MISC); 294379251f5eSSepherosa Ziehau ivar &= ~(0xFF << index); 294479251f5eSSepherosa Ziehau ivar |= (vector << index); 294579251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_IVAR_MISC, ivar); 294679251f5eSSepherosa Ziehau } else { /* RX/TX IVARS */ 294779251f5eSSepherosa Ziehau index = (16 * (entry & 1)) + (8 * type); 294879251f5eSSepherosa Ziehau ivar = IXGBE_READ_REG(hw, IXGBE_IVAR(entry >> 1)); 294979251f5eSSepherosa Ziehau ivar &= ~(0xFF << index); 295079251f5eSSepherosa Ziehau ivar |= (vector << index); 295179251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_IVAR(entry >> 1), ivar); 295279251f5eSSepherosa Ziehau } 295379251f5eSSepherosa Ziehau 295479251f5eSSepherosa Ziehau default: 295579251f5eSSepherosa Ziehau break; 295679251f5eSSepherosa Ziehau } 295779251f5eSSepherosa Ziehau } 295879251f5eSSepherosa Ziehau 295979251f5eSSepherosa Ziehau static boolean_t 296079251f5eSSepherosa Ziehau ix_sfp_probe(struct ix_softc *sc) 296179251f5eSSepherosa Ziehau { 296279251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 296379251f5eSSepherosa Ziehau 296479251f5eSSepherosa Ziehau if (hw->phy.type == ixgbe_phy_nl && 296579251f5eSSepherosa Ziehau hw->phy.sfp_type == ixgbe_sfp_type_not_present) { 296679251f5eSSepherosa Ziehau int32_t ret; 296779251f5eSSepherosa Ziehau 296879251f5eSSepherosa Ziehau ret = hw->phy.ops.identify_sfp(hw); 296979251f5eSSepherosa Ziehau if (ret) 297079251f5eSSepherosa Ziehau return FALSE; 297179251f5eSSepherosa Ziehau 297279251f5eSSepherosa Ziehau ret = hw->phy.ops.reset(hw); 297379251f5eSSepherosa Ziehau if (ret == IXGBE_ERR_SFP_NOT_SUPPORTED) { 297479251f5eSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, 297579251f5eSSepherosa Ziehau "Unsupported SFP+ module detected! " 297679251f5eSSepherosa Ziehau "Reload driver with supported module.\n"); 297779251f5eSSepherosa Ziehau sc->sfp_probe = FALSE; 297879251f5eSSepherosa Ziehau return FALSE; 297979251f5eSSepherosa Ziehau } 298079251f5eSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, "SFP+ module detected!\n"); 298179251f5eSSepherosa Ziehau 298279251f5eSSepherosa Ziehau /* We now have supported optics */ 298379251f5eSSepherosa Ziehau sc->sfp_probe = FALSE; 298479251f5eSSepherosa Ziehau /* Set the optics type so system reports correctly */ 298579251f5eSSepherosa Ziehau ix_setup_optics(sc); 298679251f5eSSepherosa Ziehau 298779251f5eSSepherosa Ziehau return TRUE; 298879251f5eSSepherosa Ziehau } 298979251f5eSSepherosa Ziehau return FALSE; 299079251f5eSSepherosa Ziehau } 299179251f5eSSepherosa Ziehau 299279251f5eSSepherosa Ziehau static void 299379251f5eSSepherosa Ziehau ix_handle_link(struct ix_softc *sc) 299479251f5eSSepherosa Ziehau { 299579251f5eSSepherosa Ziehau ixgbe_check_link(&sc->hw, &sc->link_speed, &sc->link_up, 0); 299679251f5eSSepherosa Ziehau ix_update_link_status(sc); 299779251f5eSSepherosa Ziehau } 299879251f5eSSepherosa Ziehau 299979251f5eSSepherosa Ziehau /* 300079251f5eSSepherosa Ziehau * Handling SFP module 300179251f5eSSepherosa Ziehau */ 300279251f5eSSepherosa Ziehau static void 300379251f5eSSepherosa Ziehau ix_handle_mod(struct ix_softc *sc) 300479251f5eSSepherosa Ziehau { 300579251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 300679251f5eSSepherosa Ziehau uint32_t err; 300779251f5eSSepherosa Ziehau 300879251f5eSSepherosa Ziehau err = hw->phy.ops.identify_sfp(hw); 300979251f5eSSepherosa Ziehau if (err == IXGBE_ERR_SFP_NOT_SUPPORTED) { 301079251f5eSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, 301179251f5eSSepherosa Ziehau "Unsupported SFP+ module type was detected.\n"); 301279251f5eSSepherosa Ziehau return; 301379251f5eSSepherosa Ziehau } 301479251f5eSSepherosa Ziehau err = hw->mac.ops.setup_sfp(hw); 301579251f5eSSepherosa Ziehau if (err == IXGBE_ERR_SFP_NOT_SUPPORTED) { 301679251f5eSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, 301779251f5eSSepherosa Ziehau "Setup failure - unsupported SFP+ module type.\n"); 301879251f5eSSepherosa Ziehau return; 301979251f5eSSepherosa Ziehau } 302079251f5eSSepherosa Ziehau ix_handle_msf(sc); 302179251f5eSSepherosa Ziehau } 302279251f5eSSepherosa Ziehau 302379251f5eSSepherosa Ziehau /* 302479251f5eSSepherosa Ziehau * Handling MSF (multispeed fiber) 302579251f5eSSepherosa Ziehau */ 302679251f5eSSepherosa Ziehau static void 302779251f5eSSepherosa Ziehau ix_handle_msf(struct ix_softc *sc) 302879251f5eSSepherosa Ziehau { 302979251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 303079251f5eSSepherosa Ziehau uint32_t autoneg; 303179251f5eSSepherosa Ziehau 303279251f5eSSepherosa Ziehau autoneg = hw->phy.autoneg_advertised; 303379251f5eSSepherosa Ziehau if (!autoneg && hw->mac.ops.get_link_capabilities != NULL) { 303479251f5eSSepherosa Ziehau bool negotiate; 303579251f5eSSepherosa Ziehau 303679251f5eSSepherosa Ziehau hw->mac.ops.get_link_capabilities(hw, &autoneg, &negotiate); 303779251f5eSSepherosa Ziehau } 303879251f5eSSepherosa Ziehau if (hw->mac.ops.setup_link != NULL) 303979251f5eSSepherosa Ziehau hw->mac.ops.setup_link(hw, autoneg, TRUE); 304079251f5eSSepherosa Ziehau } 304179251f5eSSepherosa Ziehau 304279251f5eSSepherosa Ziehau static void 304379251f5eSSepherosa Ziehau ix_update_stats(struct ix_softc *sc) 304479251f5eSSepherosa Ziehau { 304579251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 304679251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 304779251f5eSSepherosa Ziehau uint32_t missed_rx = 0, bprc, lxon, lxoff, total; 304879251f5eSSepherosa Ziehau uint64_t total_missed_rx = 0; 304979251f5eSSepherosa Ziehau int i; 305079251f5eSSepherosa Ziehau 305179251f5eSSepherosa Ziehau sc->stats.crcerrs += IXGBE_READ_REG(hw, IXGBE_CRCERRS); 305279251f5eSSepherosa Ziehau sc->stats.illerrc += IXGBE_READ_REG(hw, IXGBE_ILLERRC); 305379251f5eSSepherosa Ziehau sc->stats.errbc += IXGBE_READ_REG(hw, IXGBE_ERRBC); 305479251f5eSSepherosa Ziehau sc->stats.mspdc += IXGBE_READ_REG(hw, IXGBE_MSPDC); 305579251f5eSSepherosa Ziehau 305679251f5eSSepherosa Ziehau /* 305779251f5eSSepherosa Ziehau * Note: These are for the 8 possible traffic classes, which 305879251f5eSSepherosa Ziehau * in current implementation is unused, therefore only 0 should 305979251f5eSSepherosa Ziehau * read real data. 306079251f5eSSepherosa Ziehau */ 306179251f5eSSepherosa Ziehau for (i = 0; i < 8; i++) { 306279251f5eSSepherosa Ziehau uint32_t mp; 306379251f5eSSepherosa Ziehau 306479251f5eSSepherosa Ziehau mp = IXGBE_READ_REG(hw, IXGBE_MPC(i)); 306579251f5eSSepherosa Ziehau /* missed_rx tallies misses for the gprc workaround */ 306679251f5eSSepherosa Ziehau missed_rx += mp; 306779251f5eSSepherosa Ziehau /* global total per queue */ 306879251f5eSSepherosa Ziehau sc->stats.mpc[i] += mp; 306979251f5eSSepherosa Ziehau 307079251f5eSSepherosa Ziehau /* Running comprehensive total for stats display */ 307179251f5eSSepherosa Ziehau total_missed_rx += sc->stats.mpc[i]; 307279251f5eSSepherosa Ziehau 307379251f5eSSepherosa Ziehau if (hw->mac.type == ixgbe_mac_82598EB) { 307479251f5eSSepherosa Ziehau sc->stats.rnbc[i] += IXGBE_READ_REG(hw, IXGBE_RNBC(i)); 307579251f5eSSepherosa Ziehau sc->stats.qbtc[i] += IXGBE_READ_REG(hw, IXGBE_QBTC(i)); 307679251f5eSSepherosa Ziehau sc->stats.qbrc[i] += IXGBE_READ_REG(hw, IXGBE_QBRC(i)); 307779251f5eSSepherosa Ziehau sc->stats.pxonrxc[i] += 307879251f5eSSepherosa Ziehau IXGBE_READ_REG(hw, IXGBE_PXONRXC(i)); 307979251f5eSSepherosa Ziehau } else { 308079251f5eSSepherosa Ziehau sc->stats.pxonrxc[i] += 308179251f5eSSepherosa Ziehau IXGBE_READ_REG(hw, IXGBE_PXONRXCNT(i)); 308279251f5eSSepherosa Ziehau } 308379251f5eSSepherosa Ziehau sc->stats.pxontxc[i] += IXGBE_READ_REG(hw, IXGBE_PXONTXC(i)); 308479251f5eSSepherosa Ziehau sc->stats.pxofftxc[i] += IXGBE_READ_REG(hw, IXGBE_PXOFFTXC(i)); 308579251f5eSSepherosa Ziehau sc->stats.pxoffrxc[i] += IXGBE_READ_REG(hw, IXGBE_PXOFFRXC(i)); 308679251f5eSSepherosa Ziehau sc->stats.pxon2offc[i] += 308779251f5eSSepherosa Ziehau IXGBE_READ_REG(hw, IXGBE_PXON2OFFCNT(i)); 308879251f5eSSepherosa Ziehau } 308979251f5eSSepherosa Ziehau for (i = 0; i < 16; i++) { 309079251f5eSSepherosa Ziehau sc->stats.qprc[i] += IXGBE_READ_REG(hw, IXGBE_QPRC(i)); 309179251f5eSSepherosa Ziehau sc->stats.qptc[i] += IXGBE_READ_REG(hw, IXGBE_QPTC(i)); 309279251f5eSSepherosa Ziehau sc->stats.qprdc[i] += IXGBE_READ_REG(hw, IXGBE_QPRDC(i)); 309379251f5eSSepherosa Ziehau } 309479251f5eSSepherosa Ziehau sc->stats.mlfc += IXGBE_READ_REG(hw, IXGBE_MLFC); 309579251f5eSSepherosa Ziehau sc->stats.mrfc += IXGBE_READ_REG(hw, IXGBE_MRFC); 309679251f5eSSepherosa Ziehau sc->stats.rlec += IXGBE_READ_REG(hw, IXGBE_RLEC); 309779251f5eSSepherosa Ziehau 309879251f5eSSepherosa Ziehau /* Hardware workaround, gprc counts missed packets */ 309979251f5eSSepherosa Ziehau sc->stats.gprc += IXGBE_READ_REG(hw, IXGBE_GPRC); 310079251f5eSSepherosa Ziehau sc->stats.gprc -= missed_rx; 310179251f5eSSepherosa Ziehau 310279251f5eSSepherosa Ziehau if (hw->mac.type != ixgbe_mac_82598EB) { 310379251f5eSSepherosa Ziehau sc->stats.gorc += IXGBE_READ_REG(hw, IXGBE_GORCL) + 310479251f5eSSepherosa Ziehau ((uint64_t)IXGBE_READ_REG(hw, IXGBE_GORCH) << 32); 310579251f5eSSepherosa Ziehau sc->stats.gotc += IXGBE_READ_REG(hw, IXGBE_GOTCL) + 310679251f5eSSepherosa Ziehau ((uint64_t)IXGBE_READ_REG(hw, IXGBE_GOTCH) << 32); 310779251f5eSSepherosa Ziehau sc->stats.tor += IXGBE_READ_REG(hw, IXGBE_TORL) + 310879251f5eSSepherosa Ziehau ((uint64_t)IXGBE_READ_REG(hw, IXGBE_TORH) << 32); 310979251f5eSSepherosa Ziehau sc->stats.lxonrxc += IXGBE_READ_REG(hw, IXGBE_LXONRXCNT); 311079251f5eSSepherosa Ziehau sc->stats.lxoffrxc += IXGBE_READ_REG(hw, IXGBE_LXOFFRXCNT); 311179251f5eSSepherosa Ziehau } else { 311279251f5eSSepherosa Ziehau sc->stats.lxonrxc += IXGBE_READ_REG(hw, IXGBE_LXONRXC); 311379251f5eSSepherosa Ziehau sc->stats.lxoffrxc += IXGBE_READ_REG(hw, IXGBE_LXOFFRXC); 311479251f5eSSepherosa Ziehau /* 82598 only has a counter in the high register */ 311579251f5eSSepherosa Ziehau sc->stats.gorc += IXGBE_READ_REG(hw, IXGBE_GORCH); 311679251f5eSSepherosa Ziehau sc->stats.gotc += IXGBE_READ_REG(hw, IXGBE_GOTCH); 311779251f5eSSepherosa Ziehau sc->stats.tor += IXGBE_READ_REG(hw, IXGBE_TORH); 311879251f5eSSepherosa Ziehau } 311979251f5eSSepherosa Ziehau 312079251f5eSSepherosa Ziehau /* 312179251f5eSSepherosa Ziehau * Workaround: mprc hardware is incorrectly counting 312279251f5eSSepherosa Ziehau * broadcasts, so for now we subtract those. 312379251f5eSSepherosa Ziehau */ 312479251f5eSSepherosa Ziehau bprc = IXGBE_READ_REG(hw, IXGBE_BPRC); 312579251f5eSSepherosa Ziehau sc->stats.bprc += bprc; 312679251f5eSSepherosa Ziehau sc->stats.mprc += IXGBE_READ_REG(hw, IXGBE_MPRC); 312779251f5eSSepherosa Ziehau if (hw->mac.type == ixgbe_mac_82598EB) 312879251f5eSSepherosa Ziehau sc->stats.mprc -= bprc; 312979251f5eSSepherosa Ziehau 313079251f5eSSepherosa Ziehau sc->stats.prc64 += IXGBE_READ_REG(hw, IXGBE_PRC64); 313179251f5eSSepherosa Ziehau sc->stats.prc127 += IXGBE_READ_REG(hw, IXGBE_PRC127); 313279251f5eSSepherosa Ziehau sc->stats.prc255 += IXGBE_READ_REG(hw, IXGBE_PRC255); 313379251f5eSSepherosa Ziehau sc->stats.prc511 += IXGBE_READ_REG(hw, IXGBE_PRC511); 313479251f5eSSepherosa Ziehau sc->stats.prc1023 += IXGBE_READ_REG(hw, IXGBE_PRC1023); 313579251f5eSSepherosa Ziehau sc->stats.prc1522 += IXGBE_READ_REG(hw, IXGBE_PRC1522); 313679251f5eSSepherosa Ziehau 313779251f5eSSepherosa Ziehau lxon = IXGBE_READ_REG(hw, IXGBE_LXONTXC); 313879251f5eSSepherosa Ziehau sc->stats.lxontxc += lxon; 313979251f5eSSepherosa Ziehau lxoff = IXGBE_READ_REG(hw, IXGBE_LXOFFTXC); 314079251f5eSSepherosa Ziehau sc->stats.lxofftxc += lxoff; 314179251f5eSSepherosa Ziehau total = lxon + lxoff; 314279251f5eSSepherosa Ziehau 314379251f5eSSepherosa Ziehau sc->stats.gptc += IXGBE_READ_REG(hw, IXGBE_GPTC); 314479251f5eSSepherosa Ziehau sc->stats.mptc += IXGBE_READ_REG(hw, IXGBE_MPTC); 314579251f5eSSepherosa Ziehau sc->stats.ptc64 += IXGBE_READ_REG(hw, IXGBE_PTC64); 314679251f5eSSepherosa Ziehau sc->stats.gptc -= total; 314779251f5eSSepherosa Ziehau sc->stats.mptc -= total; 314879251f5eSSepherosa Ziehau sc->stats.ptc64 -= total; 314979251f5eSSepherosa Ziehau sc->stats.gotc -= total * ETHER_MIN_LEN; 315079251f5eSSepherosa Ziehau 315179251f5eSSepherosa Ziehau sc->stats.ruc += IXGBE_READ_REG(hw, IXGBE_RUC); 315279251f5eSSepherosa Ziehau sc->stats.rfc += IXGBE_READ_REG(hw, IXGBE_RFC); 315379251f5eSSepherosa Ziehau sc->stats.roc += IXGBE_READ_REG(hw, IXGBE_ROC); 315479251f5eSSepherosa Ziehau sc->stats.rjc += IXGBE_READ_REG(hw, IXGBE_RJC); 315579251f5eSSepherosa Ziehau sc->stats.mngprc += IXGBE_READ_REG(hw, IXGBE_MNGPRC); 315679251f5eSSepherosa Ziehau sc->stats.mngpdc += IXGBE_READ_REG(hw, IXGBE_MNGPDC); 315779251f5eSSepherosa Ziehau sc->stats.mngptc += IXGBE_READ_REG(hw, IXGBE_MNGPTC); 315879251f5eSSepherosa Ziehau sc->stats.tpr += IXGBE_READ_REG(hw, IXGBE_TPR); 315979251f5eSSepherosa Ziehau sc->stats.tpt += IXGBE_READ_REG(hw, IXGBE_TPT); 316079251f5eSSepherosa Ziehau sc->stats.ptc127 += IXGBE_READ_REG(hw, IXGBE_PTC127); 316179251f5eSSepherosa Ziehau sc->stats.ptc255 += IXGBE_READ_REG(hw, IXGBE_PTC255); 316279251f5eSSepherosa Ziehau sc->stats.ptc511 += IXGBE_READ_REG(hw, IXGBE_PTC511); 316379251f5eSSepherosa Ziehau sc->stats.ptc1023 += IXGBE_READ_REG(hw, IXGBE_PTC1023); 316479251f5eSSepherosa Ziehau sc->stats.ptc1522 += IXGBE_READ_REG(hw, IXGBE_PTC1522); 316579251f5eSSepherosa Ziehau sc->stats.bptc += IXGBE_READ_REG(hw, IXGBE_BPTC); 316679251f5eSSepherosa Ziehau sc->stats.xec += IXGBE_READ_REG(hw, IXGBE_XEC); 316779251f5eSSepherosa Ziehau sc->stats.fccrc += IXGBE_READ_REG(hw, IXGBE_FCCRC); 316879251f5eSSepherosa Ziehau sc->stats.fclast += IXGBE_READ_REG(hw, IXGBE_FCLAST); 316979251f5eSSepherosa Ziehau /* Only read FCOE on 82599 */ 317079251f5eSSepherosa Ziehau if (hw->mac.type != ixgbe_mac_82598EB) { 317179251f5eSSepherosa Ziehau sc->stats.fcoerpdc += IXGBE_READ_REG(hw, IXGBE_FCOERPDC); 317279251f5eSSepherosa Ziehau sc->stats.fcoeprc += IXGBE_READ_REG(hw, IXGBE_FCOEPRC); 317379251f5eSSepherosa Ziehau sc->stats.fcoeptc += IXGBE_READ_REG(hw, IXGBE_FCOEPTC); 317479251f5eSSepherosa Ziehau sc->stats.fcoedwrc += IXGBE_READ_REG(hw, IXGBE_FCOEDWRC); 317579251f5eSSepherosa Ziehau sc->stats.fcoedwtc += IXGBE_READ_REG(hw, IXGBE_FCOEDWTC); 317679251f5eSSepherosa Ziehau } 317779251f5eSSepherosa Ziehau 317879251f5eSSepherosa Ziehau /* Rx Errors */ 317979251f5eSSepherosa Ziehau IFNET_STAT_SET(ifp, iqdrops, total_missed_rx); 318079251f5eSSepherosa Ziehau IFNET_STAT_SET(ifp, ierrors, sc->stats.crcerrs + sc->stats.rlec); 318179251f5eSSepherosa Ziehau } 318279251f5eSSepherosa Ziehau 318379251f5eSSepherosa Ziehau #if 0 318479251f5eSSepherosa Ziehau /* 318579251f5eSSepherosa Ziehau * Add sysctl variables, one per statistic, to the system. 318679251f5eSSepherosa Ziehau */ 318779251f5eSSepherosa Ziehau static void 318879251f5eSSepherosa Ziehau ix_add_hw_stats(struct ix_softc *sc) 318979251f5eSSepherosa Ziehau { 319079251f5eSSepherosa Ziehau 319179251f5eSSepherosa Ziehau device_t dev = sc->dev; 319279251f5eSSepherosa Ziehau 319379251f5eSSepherosa Ziehau struct ix_tx_ring *txr = sc->tx_rings; 319479251f5eSSepherosa Ziehau struct ix_rx_ring *rxr = sc->rx_rings; 319579251f5eSSepherosa Ziehau 319679251f5eSSepherosa Ziehau struct sysctl_ctx_list *ctx = device_get_sysctl_ctx(dev); 319779251f5eSSepherosa Ziehau struct sysctl_oid *tree = device_get_sysctl_tree(dev); 319879251f5eSSepherosa Ziehau struct sysctl_oid_list *child = SYSCTL_CHILDREN(tree); 319979251f5eSSepherosa Ziehau struct ixgbe_hw_stats *stats = &sc->stats; 320079251f5eSSepherosa Ziehau 320179251f5eSSepherosa Ziehau struct sysctl_oid *stat_node, *queue_node; 320279251f5eSSepherosa Ziehau struct sysctl_oid_list *stat_list, *queue_list; 320379251f5eSSepherosa Ziehau 320479251f5eSSepherosa Ziehau #define QUEUE_NAME_LEN 32 320579251f5eSSepherosa Ziehau char namebuf[QUEUE_NAME_LEN]; 320679251f5eSSepherosa Ziehau 320779251f5eSSepherosa Ziehau /* MAC stats get the own sub node */ 320879251f5eSSepherosa Ziehau 320979251f5eSSepherosa Ziehau stat_node = SYSCTL_ADD_NODE(ctx, child, OID_AUTO, "mac_stats", 321079251f5eSSepherosa Ziehau CTLFLAG_RD, NULL, "MAC Statistics"); 321179251f5eSSepherosa Ziehau stat_list = SYSCTL_CHILDREN(stat_node); 321279251f5eSSepherosa Ziehau 321379251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "crc_errs", 321479251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->crcerrs, 321579251f5eSSepherosa Ziehau "CRC Errors"); 321679251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "ill_errs", 321779251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->illerrc, 321879251f5eSSepherosa Ziehau "Illegal Byte Errors"); 321979251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "byte_errs", 322079251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->errbc, 322179251f5eSSepherosa Ziehau "Byte Errors"); 322279251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "short_discards", 322379251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->mspdc, 322479251f5eSSepherosa Ziehau "MAC Short Packets Discarded"); 322579251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "local_faults", 322679251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->mlfc, 322779251f5eSSepherosa Ziehau "MAC Local Faults"); 322879251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "remote_faults", 322979251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->mrfc, 323079251f5eSSepherosa Ziehau "MAC Remote Faults"); 323179251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rec_len_errs", 323279251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->rlec, 323379251f5eSSepherosa Ziehau "Receive Length Errors"); 323479251f5eSSepherosa Ziehau 323579251f5eSSepherosa Ziehau /* Flow Control stats */ 323679251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "xon_txd", 323779251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->lxontxc, 323879251f5eSSepherosa Ziehau "Link XON Transmitted"); 323979251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "xon_recvd", 324079251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->lxonrxc, 324179251f5eSSepherosa Ziehau "Link XON Received"); 324279251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "xoff_txd", 324379251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->lxofftxc, 324479251f5eSSepherosa Ziehau "Link XOFF Transmitted"); 324579251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "xoff_recvd", 324679251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->lxoffrxc, 324779251f5eSSepherosa Ziehau "Link XOFF Received"); 324879251f5eSSepherosa Ziehau 324979251f5eSSepherosa Ziehau /* Packet Reception Stats */ 325079251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "total_octets_rcvd", 325179251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->tor, 325279251f5eSSepherosa Ziehau "Total Octets Received"); 325379251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "good_octets_rcvd", 325479251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->gorc, 325579251f5eSSepherosa Ziehau "Good Octets Received"); 325679251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "total_pkts_rcvd", 325779251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->tpr, 325879251f5eSSepherosa Ziehau "Total Packets Received"); 325979251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "good_pkts_rcvd", 326079251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->gprc, 326179251f5eSSepherosa Ziehau "Good Packets Received"); 326279251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "mcast_pkts_rcvd", 326379251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->mprc, 326479251f5eSSepherosa Ziehau "Multicast Packets Received"); 326579251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "bcast_pkts_rcvd", 326679251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->bprc, 326779251f5eSSepherosa Ziehau "Broadcast Packets Received"); 326879251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_frames_64", 326979251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->prc64, 327079251f5eSSepherosa Ziehau "64 byte frames received "); 327179251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_frames_65_127", 327279251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->prc127, 327379251f5eSSepherosa Ziehau "65-127 byte frames received"); 327479251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_frames_128_255", 327579251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->prc255, 327679251f5eSSepherosa Ziehau "128-255 byte frames received"); 327779251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_frames_256_511", 327879251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->prc511, 327979251f5eSSepherosa Ziehau "256-511 byte frames received"); 328079251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_frames_512_1023", 328179251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->prc1023, 328279251f5eSSepherosa Ziehau "512-1023 byte frames received"); 328379251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_frames_1024_1522", 328479251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->prc1522, 328579251f5eSSepherosa Ziehau "1023-1522 byte frames received"); 328679251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "recv_undersized", 328779251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->ruc, 328879251f5eSSepherosa Ziehau "Receive Undersized"); 328979251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "recv_fragmented", 329079251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->rfc, 329179251f5eSSepherosa Ziehau "Fragmented Packets Received "); 329279251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "recv_oversized", 329379251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->roc, 329479251f5eSSepherosa Ziehau "Oversized Packets Received"); 329579251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "recv_jabberd", 329679251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->rjc, 329779251f5eSSepherosa Ziehau "Received Jabber"); 329879251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "management_pkts_rcvd", 329979251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->mngprc, 330079251f5eSSepherosa Ziehau "Management Packets Received"); 330179251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "management_pkts_drpd", 330279251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->mngptc, 330379251f5eSSepherosa Ziehau "Management Packets Dropped"); 330479251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "checksum_errs", 330579251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->xec, 330679251f5eSSepherosa Ziehau "Checksum Errors"); 330779251f5eSSepherosa Ziehau 330879251f5eSSepherosa Ziehau /* Packet Transmission Stats */ 330979251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "good_octets_txd", 331079251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->gotc, 331179251f5eSSepherosa Ziehau "Good Octets Transmitted"); 331279251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "total_pkts_txd", 331379251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->tpt, 331479251f5eSSepherosa Ziehau "Total Packets Transmitted"); 331579251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "good_pkts_txd", 331679251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->gptc, 331779251f5eSSepherosa Ziehau "Good Packets Transmitted"); 331879251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "bcast_pkts_txd", 331979251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->bptc, 332079251f5eSSepherosa Ziehau "Broadcast Packets Transmitted"); 332179251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "mcast_pkts_txd", 332279251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->mptc, 332379251f5eSSepherosa Ziehau "Multicast Packets Transmitted"); 332479251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "management_pkts_txd", 332579251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->mngptc, 332679251f5eSSepherosa Ziehau "Management Packets Transmitted"); 332779251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "tx_frames_64", 332879251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->ptc64, 332979251f5eSSepherosa Ziehau "64 byte frames transmitted "); 333079251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "tx_frames_65_127", 333179251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->ptc127, 333279251f5eSSepherosa Ziehau "65-127 byte frames transmitted"); 333379251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "tx_frames_128_255", 333479251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->ptc255, 333579251f5eSSepherosa Ziehau "128-255 byte frames transmitted"); 333679251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "tx_frames_256_511", 333779251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->ptc511, 333879251f5eSSepherosa Ziehau "256-511 byte frames transmitted"); 333979251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "tx_frames_512_1023", 334079251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->ptc1023, 334179251f5eSSepherosa Ziehau "512-1023 byte frames transmitted"); 334279251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "tx_frames_1024_1522", 334379251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->ptc1522, 334479251f5eSSepherosa Ziehau "1024-1522 byte frames transmitted"); 334579251f5eSSepherosa Ziehau } 334679251f5eSSepherosa Ziehau #endif 334779251f5eSSepherosa Ziehau 334879251f5eSSepherosa Ziehau /* 334979251f5eSSepherosa Ziehau * Enable the hardware to drop packets when the buffer is full. 335079251f5eSSepherosa Ziehau * This is useful when multiple RX rings are used, so that no 335179251f5eSSepherosa Ziehau * single RX ring being full stalls the entire RX engine. We 335279251f5eSSepherosa Ziehau * only enable this when multiple RX rings are used and when 335379251f5eSSepherosa Ziehau * flow control is disabled. 335479251f5eSSepherosa Ziehau */ 335579251f5eSSepherosa Ziehau static void 335679251f5eSSepherosa Ziehau ix_enable_rx_drop(struct ix_softc *sc) 335779251f5eSSepherosa Ziehau { 335879251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 335979251f5eSSepherosa Ziehau int i; 336079251f5eSSepherosa Ziehau 336179251f5eSSepherosa Ziehau if (bootverbose) { 336279251f5eSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, 336379251f5eSSepherosa Ziehau "flow control %d, enable RX drop\n", sc->fc); 336479251f5eSSepherosa Ziehau } 336579251f5eSSepherosa Ziehau 336679251f5eSSepherosa Ziehau for (i = 0; i < sc->rx_ring_inuse; ++i) { 336779251f5eSSepherosa Ziehau uint32_t srrctl = IXGBE_READ_REG(hw, IXGBE_SRRCTL(i)); 336879251f5eSSepherosa Ziehau 336979251f5eSSepherosa Ziehau srrctl |= IXGBE_SRRCTL_DROP_EN; 337079251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_SRRCTL(i), srrctl); 337179251f5eSSepherosa Ziehau } 337279251f5eSSepherosa Ziehau } 337379251f5eSSepherosa Ziehau 337479251f5eSSepherosa Ziehau static void 337579251f5eSSepherosa Ziehau ix_disable_rx_drop(struct ix_softc *sc) 337679251f5eSSepherosa Ziehau { 337779251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 337879251f5eSSepherosa Ziehau int i; 337979251f5eSSepherosa Ziehau 338079251f5eSSepherosa Ziehau if (bootverbose) { 338179251f5eSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, 338279251f5eSSepherosa Ziehau "flow control %d, disable RX drop\n", sc->fc); 338379251f5eSSepherosa Ziehau } 338479251f5eSSepherosa Ziehau 338579251f5eSSepherosa Ziehau for (i = 0; i < sc->rx_ring_inuse; ++i) { 338679251f5eSSepherosa Ziehau uint32_t srrctl = IXGBE_READ_REG(hw, IXGBE_SRRCTL(i)); 338779251f5eSSepherosa Ziehau 338879251f5eSSepherosa Ziehau srrctl &= ~IXGBE_SRRCTL_DROP_EN; 338979251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_SRRCTL(i), srrctl); 339079251f5eSSepherosa Ziehau } 339179251f5eSSepherosa Ziehau } 339279251f5eSSepherosa Ziehau 339379251f5eSSepherosa Ziehau static int 339479251f5eSSepherosa Ziehau ix_sysctl_flowctrl(SYSCTL_HANDLER_ARGS) 339579251f5eSSepherosa Ziehau { 339679251f5eSSepherosa Ziehau struct ix_softc *sc = (struct ix_softc *)arg1; 339779251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 339879251f5eSSepherosa Ziehau int error, fc; 339979251f5eSSepherosa Ziehau 340079251f5eSSepherosa Ziehau fc = sc->fc; 340179251f5eSSepherosa Ziehau error = sysctl_handle_int(oidp, &fc, 0, req); 340279251f5eSSepherosa Ziehau if (error || req->newptr == NULL) 340379251f5eSSepherosa Ziehau return error; 340479251f5eSSepherosa Ziehau 340579251f5eSSepherosa Ziehau switch (fc) { 340679251f5eSSepherosa Ziehau case ixgbe_fc_rx_pause: 340779251f5eSSepherosa Ziehau case ixgbe_fc_tx_pause: 340879251f5eSSepherosa Ziehau case ixgbe_fc_full: 340979251f5eSSepherosa Ziehau case ixgbe_fc_none: 341079251f5eSSepherosa Ziehau break; 341179251f5eSSepherosa Ziehau default: 341279251f5eSSepherosa Ziehau return EINVAL; 341379251f5eSSepherosa Ziehau } 341479251f5eSSepherosa Ziehau 341579251f5eSSepherosa Ziehau ifnet_serialize_all(ifp); 341679251f5eSSepherosa Ziehau 341779251f5eSSepherosa Ziehau /* Don't bother if it's not changed */ 341879251f5eSSepherosa Ziehau if (sc->fc == fc) 341979251f5eSSepherosa Ziehau goto done; 342079251f5eSSepherosa Ziehau sc->fc = fc; 342179251f5eSSepherosa Ziehau 342279251f5eSSepherosa Ziehau /* Don't do anything, if the interface is not up yet */ 342379251f5eSSepherosa Ziehau if ((ifp->if_flags & IFF_RUNNING) == 0) 342479251f5eSSepherosa Ziehau goto done; 342579251f5eSSepherosa Ziehau 342679251f5eSSepherosa Ziehau if (sc->rx_ring_inuse > 1) { 342779251f5eSSepherosa Ziehau switch (sc->fc) { 342879251f5eSSepherosa Ziehau case ixgbe_fc_rx_pause: 342979251f5eSSepherosa Ziehau case ixgbe_fc_tx_pause: 343079251f5eSSepherosa Ziehau case ixgbe_fc_full: 343179251f5eSSepherosa Ziehau ix_disable_rx_drop(sc); 343279251f5eSSepherosa Ziehau break; 343379251f5eSSepherosa Ziehau 343479251f5eSSepherosa Ziehau case ixgbe_fc_none: 343579251f5eSSepherosa Ziehau ix_enable_rx_drop(sc); 343679251f5eSSepherosa Ziehau break; 343779251f5eSSepherosa Ziehau 343879251f5eSSepherosa Ziehau default: 343979251f5eSSepherosa Ziehau panic("leading fc check mismatch"); 344079251f5eSSepherosa Ziehau } 344179251f5eSSepherosa Ziehau } 344279251f5eSSepherosa Ziehau 344379251f5eSSepherosa Ziehau sc->hw.fc.requested_mode = sc->fc; 344479251f5eSSepherosa Ziehau /* Don't autoneg if forcing a value */ 344579251f5eSSepherosa Ziehau sc->hw.fc.disable_fc_autoneg = TRUE; 344679251f5eSSepherosa Ziehau ixgbe_fc_enable(&sc->hw); 344779251f5eSSepherosa Ziehau 344879251f5eSSepherosa Ziehau done: 344979251f5eSSepherosa Ziehau ifnet_deserialize_all(ifp); 345079251f5eSSepherosa Ziehau return error; 345179251f5eSSepherosa Ziehau } 345279251f5eSSepherosa Ziehau 345379251f5eSSepherosa Ziehau #ifdef foo 345479251f5eSSepherosa Ziehau /* XXX not working properly w/ 82599 connected w/ DAC */ 345579251f5eSSepherosa Ziehau /* XXX only work after the interface is up */ 345679251f5eSSepherosa Ziehau static int 345779251f5eSSepherosa Ziehau ix_sysctl_advspeed(SYSCTL_HANDLER_ARGS) 345879251f5eSSepherosa Ziehau { 345979251f5eSSepherosa Ziehau struct ix_softc *sc = (struct ix_softc *)arg1; 346079251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 346179251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 346279251f5eSSepherosa Ziehau ixgbe_link_speed speed; 346379251f5eSSepherosa Ziehau int error, advspeed; 346479251f5eSSepherosa Ziehau 346579251f5eSSepherosa Ziehau advspeed = sc->advspeed; 346679251f5eSSepherosa Ziehau error = sysctl_handle_int(oidp, &advspeed, 0, req); 346779251f5eSSepherosa Ziehau if (error || req->newptr == NULL) 346879251f5eSSepherosa Ziehau return error; 346979251f5eSSepherosa Ziehau 347079251f5eSSepherosa Ziehau if (!(hw->phy.media_type == ixgbe_media_type_copper || 347179251f5eSSepherosa Ziehau hw->phy.multispeed_fiber)) 347279251f5eSSepherosa Ziehau return EOPNOTSUPP; 347379251f5eSSepherosa Ziehau if (hw->mac.ops.setup_link == NULL) 347479251f5eSSepherosa Ziehau return EOPNOTSUPP; 347579251f5eSSepherosa Ziehau 347679251f5eSSepherosa Ziehau switch (advspeed) { 347779251f5eSSepherosa Ziehau case 0: /* auto */ 347879251f5eSSepherosa Ziehau speed = IXGBE_LINK_SPEED_UNKNOWN; 347979251f5eSSepherosa Ziehau break; 348079251f5eSSepherosa Ziehau 348179251f5eSSepherosa Ziehau case 1: /* 1Gb */ 348279251f5eSSepherosa Ziehau speed = IXGBE_LINK_SPEED_1GB_FULL; 348379251f5eSSepherosa Ziehau break; 348479251f5eSSepherosa Ziehau 348579251f5eSSepherosa Ziehau case 2: /* 100Mb */ 348679251f5eSSepherosa Ziehau speed = IXGBE_LINK_SPEED_100_FULL; 348779251f5eSSepherosa Ziehau break; 348879251f5eSSepherosa Ziehau 348979251f5eSSepherosa Ziehau case 3: /* 1Gb/10Gb */ 349079251f5eSSepherosa Ziehau speed = IXGBE_LINK_SPEED_1GB_FULL | 349179251f5eSSepherosa Ziehau IXGBE_LINK_SPEED_10GB_FULL; 349279251f5eSSepherosa Ziehau break; 349379251f5eSSepherosa Ziehau 349479251f5eSSepherosa Ziehau default: 349579251f5eSSepherosa Ziehau return EINVAL; 349679251f5eSSepherosa Ziehau } 349779251f5eSSepherosa Ziehau 349879251f5eSSepherosa Ziehau ifnet_serialize_all(ifp); 349979251f5eSSepherosa Ziehau 350079251f5eSSepherosa Ziehau if (sc->advspeed == advspeed) /* no change */ 350179251f5eSSepherosa Ziehau goto done; 350279251f5eSSepherosa Ziehau 350379251f5eSSepherosa Ziehau if ((speed & IXGBE_LINK_SPEED_100_FULL) && 350479251f5eSSepherosa Ziehau hw->mac.type != ixgbe_mac_X540) { 350579251f5eSSepherosa Ziehau error = EOPNOTSUPP; 350679251f5eSSepherosa Ziehau goto done; 350779251f5eSSepherosa Ziehau } 350879251f5eSSepherosa Ziehau 350979251f5eSSepherosa Ziehau sc->advspeed = advspeed; 351079251f5eSSepherosa Ziehau 351179251f5eSSepherosa Ziehau if ((ifp->if_flags & IFF_RUNNING) == 0) 351279251f5eSSepherosa Ziehau goto done; 351379251f5eSSepherosa Ziehau 351479251f5eSSepherosa Ziehau if (speed == IXGBE_LINK_SPEED_UNKNOWN) { 351579251f5eSSepherosa Ziehau ix_config_link(sc); 351679251f5eSSepherosa Ziehau } else { 351779251f5eSSepherosa Ziehau hw->mac.autotry_restart = TRUE; 351879251f5eSSepherosa Ziehau hw->mac.ops.setup_link(hw, speed, sc->link_up); 351979251f5eSSepherosa Ziehau } 352079251f5eSSepherosa Ziehau 352179251f5eSSepherosa Ziehau done: 352279251f5eSSepherosa Ziehau ifnet_deserialize_all(ifp); 352379251f5eSSepherosa Ziehau return error; 352479251f5eSSepherosa Ziehau } 352579251f5eSSepherosa Ziehau #endif 352679251f5eSSepherosa Ziehau 352779251f5eSSepherosa Ziehau static void 352879251f5eSSepherosa Ziehau ix_setup_serialize(struct ix_softc *sc) 352979251f5eSSepherosa Ziehau { 353079251f5eSSepherosa Ziehau int i = 0, j; 353179251f5eSSepherosa Ziehau 353279251f5eSSepherosa Ziehau /* Main + RX + TX */ 353379251f5eSSepherosa Ziehau sc->nserialize = 1 + sc->rx_ring_cnt + sc->tx_ring_cnt; 353479251f5eSSepherosa Ziehau sc->serializes = 353579251f5eSSepherosa Ziehau kmalloc(sc->nserialize * sizeof(struct lwkt_serialize *), 353679251f5eSSepherosa Ziehau M_DEVBUF, M_WAITOK | M_ZERO); 353779251f5eSSepherosa Ziehau 353879251f5eSSepherosa Ziehau /* 353979251f5eSSepherosa Ziehau * Setup serializes 354079251f5eSSepherosa Ziehau * 354179251f5eSSepherosa Ziehau * NOTE: Order is critical 354279251f5eSSepherosa Ziehau */ 354379251f5eSSepherosa Ziehau 354479251f5eSSepherosa Ziehau KKASSERT(i < sc->nserialize); 354579251f5eSSepherosa Ziehau sc->serializes[i++] = &sc->main_serialize; 354679251f5eSSepherosa Ziehau 354779251f5eSSepherosa Ziehau for (j = 0; j < sc->rx_ring_cnt; ++j) { 354879251f5eSSepherosa Ziehau KKASSERT(i < sc->nserialize); 354979251f5eSSepherosa Ziehau sc->serializes[i++] = &sc->rx_rings[j].rx_serialize; 355079251f5eSSepherosa Ziehau } 355179251f5eSSepherosa Ziehau 355279251f5eSSepherosa Ziehau for (j = 0; j < sc->tx_ring_cnt; ++j) { 355379251f5eSSepherosa Ziehau KKASSERT(i < sc->nserialize); 355479251f5eSSepherosa Ziehau sc->serializes[i++] = &sc->tx_rings[j].tx_serialize; 355579251f5eSSepherosa Ziehau } 355679251f5eSSepherosa Ziehau 355779251f5eSSepherosa Ziehau KKASSERT(i == sc->nserialize); 355879251f5eSSepherosa Ziehau } 355979251f5eSSepherosa Ziehau 356079251f5eSSepherosa Ziehau static int 356179251f5eSSepherosa Ziehau ix_alloc_intr(struct ix_softc *sc) 356279251f5eSSepherosa Ziehau { 356379251f5eSSepherosa Ziehau struct ix_intr_data *intr; 356479251f5eSSepherosa Ziehau u_int intr_flags; 3565*189a0ff3SSepherosa Ziehau 3566*189a0ff3SSepherosa Ziehau ix_alloc_msix(sc); 3567*189a0ff3SSepherosa Ziehau if (sc->intr_type == PCI_INTR_TYPE_MSIX) { 3568*189a0ff3SSepherosa Ziehau ix_set_ring_inuse(sc, FALSE); 3569*189a0ff3SSepherosa Ziehau return 0; 3570*189a0ff3SSepherosa Ziehau } 357179251f5eSSepherosa Ziehau 357279251f5eSSepherosa Ziehau if (sc->intr_data != NULL) 357379251f5eSSepherosa Ziehau kfree(sc->intr_data, M_DEVBUF); 357479251f5eSSepherosa Ziehau 357579251f5eSSepherosa Ziehau sc->intr_cnt = 1; 357679251f5eSSepherosa Ziehau sc->intr_data = kmalloc(sizeof(struct ix_intr_data), M_DEVBUF, 357779251f5eSSepherosa Ziehau M_WAITOK | M_ZERO); 357879251f5eSSepherosa Ziehau intr = &sc->intr_data[0]; 357979251f5eSSepherosa Ziehau 358079251f5eSSepherosa Ziehau /* 358179251f5eSSepherosa Ziehau * Allocate MSI/legacy interrupt resource 358279251f5eSSepherosa Ziehau */ 358379251f5eSSepherosa Ziehau sc->intr_type = pci_alloc_1intr(sc->dev, ix_msi_enable, 358479251f5eSSepherosa Ziehau &intr->intr_rid, &intr_flags); 358579251f5eSSepherosa Ziehau 358679251f5eSSepherosa Ziehau intr->intr_res = bus_alloc_resource_any(sc->dev, SYS_RES_IRQ, 358779251f5eSSepherosa Ziehau &intr->intr_rid, intr_flags); 358879251f5eSSepherosa Ziehau if (intr->intr_res == NULL) { 358979251f5eSSepherosa Ziehau device_printf(sc->dev, "Unable to allocate bus resource: " 359079251f5eSSepherosa Ziehau "interrupt\n"); 359179251f5eSSepherosa Ziehau return ENXIO; 359279251f5eSSepherosa Ziehau } 359379251f5eSSepherosa Ziehau 359479251f5eSSepherosa Ziehau intr->intr_serialize = &sc->main_serialize; 359579251f5eSSepherosa Ziehau intr->intr_cpuid = rman_get_cpuid(intr->intr_res); 359679251f5eSSepherosa Ziehau intr->intr_func = ix_intr; 359779251f5eSSepherosa Ziehau intr->intr_funcarg = sc; 359879251f5eSSepherosa Ziehau intr->intr_rate = IX_INTR_RATE; 359979251f5eSSepherosa Ziehau intr->intr_use = IX_INTR_USE_RXTX; 360079251f5eSSepherosa Ziehau 3601*189a0ff3SSepherosa Ziehau sc->tx_rings[0].tx_intr_cpuid = intr->intr_cpuid; 3602*189a0ff3SSepherosa Ziehau sc->tx_rings[0].tx_intr_vec = IX_TX_INTR_VEC; 360379251f5eSSepherosa Ziehau 3604*189a0ff3SSepherosa Ziehau sc->rx_rings[0].rx_intr_vec = IX_RX0_INTR_VEC; 360579251f5eSSepherosa Ziehau 360679251f5eSSepherosa Ziehau ix_set_ring_inuse(sc, FALSE); 360779251f5eSSepherosa Ziehau 360879251f5eSSepherosa Ziehau KKASSERT(sc->rx_ring_inuse <= IX_MIN_RXRING_RSS); 360979251f5eSSepherosa Ziehau if (sc->rx_ring_inuse == IX_MIN_RXRING_RSS) 361079251f5eSSepherosa Ziehau sc->rx_rings[1].rx_intr_vec = IX_RX1_INTR_VEC; 361179251f5eSSepherosa Ziehau 361279251f5eSSepherosa Ziehau return 0; 361379251f5eSSepherosa Ziehau } 361479251f5eSSepherosa Ziehau 361579251f5eSSepherosa Ziehau static void 361679251f5eSSepherosa Ziehau ix_free_intr(struct ix_softc *sc) 361779251f5eSSepherosa Ziehau { 361879251f5eSSepherosa Ziehau if (sc->intr_data == NULL) 361979251f5eSSepherosa Ziehau return; 362079251f5eSSepherosa Ziehau 362179251f5eSSepherosa Ziehau if (sc->intr_type != PCI_INTR_TYPE_MSIX) { 362279251f5eSSepherosa Ziehau struct ix_intr_data *intr = &sc->intr_data[0]; 362379251f5eSSepherosa Ziehau 362479251f5eSSepherosa Ziehau KKASSERT(sc->intr_cnt == 1); 362579251f5eSSepherosa Ziehau if (intr->intr_res != NULL) { 362679251f5eSSepherosa Ziehau bus_release_resource(sc->dev, SYS_RES_IRQ, 362779251f5eSSepherosa Ziehau intr->intr_rid, intr->intr_res); 362879251f5eSSepherosa Ziehau } 362979251f5eSSepherosa Ziehau if (sc->intr_type == PCI_INTR_TYPE_MSI) 363079251f5eSSepherosa Ziehau pci_release_msi(sc->dev); 3631*189a0ff3SSepherosa Ziehau 363279251f5eSSepherosa Ziehau kfree(sc->intr_data, M_DEVBUF); 3633*189a0ff3SSepherosa Ziehau } else { 3634*189a0ff3SSepherosa Ziehau ix_free_msix(sc, TRUE); 3635*189a0ff3SSepherosa Ziehau } 363679251f5eSSepherosa Ziehau } 363779251f5eSSepherosa Ziehau 363879251f5eSSepherosa Ziehau static void 363979251f5eSSepherosa Ziehau ix_set_ring_inuse(struct ix_softc *sc, boolean_t polling) 364079251f5eSSepherosa Ziehau { 364179251f5eSSepherosa Ziehau sc->rx_ring_inuse = ix_get_rxring_inuse(sc, polling); 364279251f5eSSepherosa Ziehau sc->tx_ring_inuse = ix_get_txring_inuse(sc, polling); 364379251f5eSSepherosa Ziehau if (bootverbose) { 364479251f5eSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, 364579251f5eSSepherosa Ziehau "RX rings %d/%d, TX rings %d/%d\n", 364679251f5eSSepherosa Ziehau sc->rx_ring_inuse, sc->rx_ring_cnt, 364779251f5eSSepherosa Ziehau sc->tx_ring_inuse, sc->tx_ring_cnt); 364879251f5eSSepherosa Ziehau } 364979251f5eSSepherosa Ziehau } 365079251f5eSSepherosa Ziehau 365179251f5eSSepherosa Ziehau static int 365279251f5eSSepherosa Ziehau ix_get_rxring_inuse(const struct ix_softc *sc, boolean_t polling) 365379251f5eSSepherosa Ziehau { 365479251f5eSSepherosa Ziehau if (!IX_ENABLE_HWRSS(sc)) 365579251f5eSSepherosa Ziehau return 1; 365679251f5eSSepherosa Ziehau 365779251f5eSSepherosa Ziehau if (polling) 365879251f5eSSepherosa Ziehau return sc->rx_ring_cnt; 365979251f5eSSepherosa Ziehau else if (sc->intr_type != PCI_INTR_TYPE_MSIX) 366079251f5eSSepherosa Ziehau return IX_MIN_RXRING_RSS; 366179251f5eSSepherosa Ziehau else 3662*189a0ff3SSepherosa Ziehau return sc->rx_ring_msix; 366379251f5eSSepherosa Ziehau } 366479251f5eSSepherosa Ziehau 366579251f5eSSepherosa Ziehau static int 366679251f5eSSepherosa Ziehau ix_get_txring_inuse(const struct ix_softc *sc, boolean_t polling) 366779251f5eSSepherosa Ziehau { 366879251f5eSSepherosa Ziehau if (!IX_ENABLE_HWTSS(sc)) 366979251f5eSSepherosa Ziehau return 1; 367079251f5eSSepherosa Ziehau 367179251f5eSSepherosa Ziehau if (polling) 367279251f5eSSepherosa Ziehau return sc->tx_ring_cnt; 367379251f5eSSepherosa Ziehau else if (sc->intr_type != PCI_INTR_TYPE_MSIX) 367479251f5eSSepherosa Ziehau return 1; 367579251f5eSSepherosa Ziehau else 3676*189a0ff3SSepherosa Ziehau return sc->tx_ring_msix; 367779251f5eSSepherosa Ziehau } 367879251f5eSSepherosa Ziehau 367979251f5eSSepherosa Ziehau static int 368079251f5eSSepherosa Ziehau ix_setup_intr(struct ix_softc *sc) 368179251f5eSSepherosa Ziehau { 368279251f5eSSepherosa Ziehau int i; 368379251f5eSSepherosa Ziehau 368479251f5eSSepherosa Ziehau for (i = 0; i < sc->intr_cnt; ++i) { 368579251f5eSSepherosa Ziehau struct ix_intr_data *intr = &sc->intr_data[i]; 368679251f5eSSepherosa Ziehau int error; 368779251f5eSSepherosa Ziehau 368879251f5eSSepherosa Ziehau error = bus_setup_intr_descr(sc->dev, intr->intr_res, 368979251f5eSSepherosa Ziehau INTR_MPSAFE, intr->intr_func, intr->intr_funcarg, 369079251f5eSSepherosa Ziehau &intr->intr_hand, intr->intr_serialize, intr->intr_desc); 369179251f5eSSepherosa Ziehau if (error) { 369279251f5eSSepherosa Ziehau device_printf(sc->dev, "can't setup %dth intr\n", i); 369379251f5eSSepherosa Ziehau ix_teardown_intr(sc, i); 369479251f5eSSepherosa Ziehau return error; 369579251f5eSSepherosa Ziehau } 369679251f5eSSepherosa Ziehau } 369779251f5eSSepherosa Ziehau return 0; 369879251f5eSSepherosa Ziehau } 369979251f5eSSepherosa Ziehau 370079251f5eSSepherosa Ziehau static void 370179251f5eSSepherosa Ziehau ix_teardown_intr(struct ix_softc *sc, int intr_cnt) 370279251f5eSSepherosa Ziehau { 370379251f5eSSepherosa Ziehau int i; 370479251f5eSSepherosa Ziehau 370579251f5eSSepherosa Ziehau if (sc->intr_data == NULL) 370679251f5eSSepherosa Ziehau return; 370779251f5eSSepherosa Ziehau 370879251f5eSSepherosa Ziehau for (i = 0; i < intr_cnt; ++i) { 370979251f5eSSepherosa Ziehau struct ix_intr_data *intr = &sc->intr_data[i]; 371079251f5eSSepherosa Ziehau 371179251f5eSSepherosa Ziehau bus_teardown_intr(sc->dev, intr->intr_res, intr->intr_hand); 371279251f5eSSepherosa Ziehau } 371379251f5eSSepherosa Ziehau } 371479251f5eSSepherosa Ziehau 371579251f5eSSepherosa Ziehau static void 371679251f5eSSepherosa Ziehau ix_serialize(struct ifnet *ifp, enum ifnet_serialize slz) 371779251f5eSSepherosa Ziehau { 371879251f5eSSepherosa Ziehau struct ix_softc *sc = ifp->if_softc; 371979251f5eSSepherosa Ziehau 372079251f5eSSepherosa Ziehau ifnet_serialize_array_enter(sc->serializes, sc->nserialize, slz); 372179251f5eSSepherosa Ziehau } 372279251f5eSSepherosa Ziehau 372379251f5eSSepherosa Ziehau static void 372479251f5eSSepherosa Ziehau ix_deserialize(struct ifnet *ifp, enum ifnet_serialize slz) 372579251f5eSSepherosa Ziehau { 372679251f5eSSepherosa Ziehau struct ix_softc *sc = ifp->if_softc; 372779251f5eSSepherosa Ziehau 372879251f5eSSepherosa Ziehau ifnet_serialize_array_exit(sc->serializes, sc->nserialize, slz); 372979251f5eSSepherosa Ziehau } 373079251f5eSSepherosa Ziehau 373179251f5eSSepherosa Ziehau static int 373279251f5eSSepherosa Ziehau ix_tryserialize(struct ifnet *ifp, enum ifnet_serialize slz) 373379251f5eSSepherosa Ziehau { 373479251f5eSSepherosa Ziehau struct ix_softc *sc = ifp->if_softc; 373579251f5eSSepherosa Ziehau 373679251f5eSSepherosa Ziehau return ifnet_serialize_array_try(sc->serializes, sc->nserialize, slz); 373779251f5eSSepherosa Ziehau } 373879251f5eSSepherosa Ziehau 373979251f5eSSepherosa Ziehau #ifdef INVARIANTS 374079251f5eSSepherosa Ziehau 374179251f5eSSepherosa Ziehau static void 374279251f5eSSepherosa Ziehau ix_serialize_assert(struct ifnet *ifp, enum ifnet_serialize slz, 374379251f5eSSepherosa Ziehau boolean_t serialized) 374479251f5eSSepherosa Ziehau { 374579251f5eSSepherosa Ziehau struct ix_softc *sc = ifp->if_softc; 374679251f5eSSepherosa Ziehau 374779251f5eSSepherosa Ziehau ifnet_serialize_array_assert(sc->serializes, sc->nserialize, slz, 374879251f5eSSepherosa Ziehau serialized); 374979251f5eSSepherosa Ziehau } 375079251f5eSSepherosa Ziehau 375179251f5eSSepherosa Ziehau #endif /* INVARIANTS */ 375279251f5eSSepherosa Ziehau 375379251f5eSSepherosa Ziehau static void 375479251f5eSSepherosa Ziehau ix_free_rings(struct ix_softc *sc) 375579251f5eSSepherosa Ziehau { 375679251f5eSSepherosa Ziehau int i; 375779251f5eSSepherosa Ziehau 375879251f5eSSepherosa Ziehau if (sc->tx_rings != NULL) { 375979251f5eSSepherosa Ziehau for (i = 0; i < sc->tx_ring_cnt; ++i) { 376079251f5eSSepherosa Ziehau struct ix_tx_ring *txr = &sc->tx_rings[i]; 376179251f5eSSepherosa Ziehau 376279251f5eSSepherosa Ziehau ix_destroy_tx_ring(txr, txr->tx_ndesc); 376379251f5eSSepherosa Ziehau } 376479251f5eSSepherosa Ziehau kfree(sc->tx_rings, M_DEVBUF); 376579251f5eSSepherosa Ziehau } 376679251f5eSSepherosa Ziehau 376779251f5eSSepherosa Ziehau if (sc->rx_rings != NULL) { 376879251f5eSSepherosa Ziehau for (i =0; i < sc->rx_ring_cnt; ++i) { 376979251f5eSSepherosa Ziehau struct ix_rx_ring *rxr = &sc->rx_rings[i]; 377079251f5eSSepherosa Ziehau 377179251f5eSSepherosa Ziehau ix_destroy_rx_ring(rxr, rxr->rx_ndesc); 377279251f5eSSepherosa Ziehau } 377379251f5eSSepherosa Ziehau kfree(sc->rx_rings, M_DEVBUF); 377479251f5eSSepherosa Ziehau } 377579251f5eSSepherosa Ziehau 377679251f5eSSepherosa Ziehau if (sc->parent_tag != NULL) 377779251f5eSSepherosa Ziehau bus_dma_tag_destroy(sc->parent_tag); 377879251f5eSSepherosa Ziehau } 377979251f5eSSepherosa Ziehau 378079251f5eSSepherosa Ziehau static void 378179251f5eSSepherosa Ziehau ix_watchdog(struct ifaltq_subque *ifsq) 378279251f5eSSepherosa Ziehau { 378379251f5eSSepherosa Ziehau struct ix_tx_ring *txr = ifsq_get_priv(ifsq); 378479251f5eSSepherosa Ziehau struct ifnet *ifp = ifsq_get_ifp(ifsq); 378579251f5eSSepherosa Ziehau struct ix_softc *sc = ifp->if_softc; 378679251f5eSSepherosa Ziehau int i; 378779251f5eSSepherosa Ziehau 378879251f5eSSepherosa Ziehau KKASSERT(txr->tx_ifsq == ifsq); 378979251f5eSSepherosa Ziehau ASSERT_IFNET_SERIALIZED_ALL(ifp); 379079251f5eSSepherosa Ziehau 379179251f5eSSepherosa Ziehau /* 379279251f5eSSepherosa Ziehau * If the interface has been paused then don't do the watchdog check 379379251f5eSSepherosa Ziehau */ 379479251f5eSSepherosa Ziehau if (IXGBE_READ_REG(&sc->hw, IXGBE_TFCS) & IXGBE_TFCS_TXOFF) { 379579251f5eSSepherosa Ziehau txr->tx_watchdog.wd_timer = 5; 379679251f5eSSepherosa Ziehau return; 379779251f5eSSepherosa Ziehau } 379879251f5eSSepherosa Ziehau 379979251f5eSSepherosa Ziehau if_printf(ifp, "Watchdog timeout -- resetting\n"); 380079251f5eSSepherosa Ziehau if_printf(ifp, "Queue(%d) tdh = %d, hw tdt = %d\n", txr->tx_idx, 380179251f5eSSepherosa Ziehau IXGBE_READ_REG(&sc->hw, IXGBE_TDH(txr->tx_idx)), 380279251f5eSSepherosa Ziehau IXGBE_READ_REG(&sc->hw, IXGBE_TDT(txr->tx_idx))); 380379251f5eSSepherosa Ziehau if_printf(ifp, "TX(%d) desc avail = %d, next TX to Clean = %d\n", 380479251f5eSSepherosa Ziehau txr->tx_idx, txr->tx_avail, txr->tx_next_clean); 380579251f5eSSepherosa Ziehau 380679251f5eSSepherosa Ziehau ix_init(sc); 380779251f5eSSepherosa Ziehau for (i = 0; i < sc->tx_ring_inuse; ++i) 380879251f5eSSepherosa Ziehau ifsq_devstart_sched(sc->tx_rings[i].tx_ifsq); 380979251f5eSSepherosa Ziehau } 381079251f5eSSepherosa Ziehau 381179251f5eSSepherosa Ziehau static void 381279251f5eSSepherosa Ziehau ix_free_tx_ring(struct ix_tx_ring *txr) 381379251f5eSSepherosa Ziehau { 381479251f5eSSepherosa Ziehau int i; 381579251f5eSSepherosa Ziehau 381679251f5eSSepherosa Ziehau for (i = 0; i < txr->tx_ndesc; ++i) { 381779251f5eSSepherosa Ziehau struct ix_tx_buf *txbuf = &txr->tx_buf[i]; 381879251f5eSSepherosa Ziehau 381979251f5eSSepherosa Ziehau if (txbuf->m_head != NULL) { 382079251f5eSSepherosa Ziehau bus_dmamap_unload(txr->tx_tag, txbuf->map); 382179251f5eSSepherosa Ziehau m_freem(txbuf->m_head); 382279251f5eSSepherosa Ziehau txbuf->m_head = NULL; 382379251f5eSSepherosa Ziehau } 382479251f5eSSepherosa Ziehau } 382579251f5eSSepherosa Ziehau } 382679251f5eSSepherosa Ziehau 382779251f5eSSepherosa Ziehau static void 382879251f5eSSepherosa Ziehau ix_free_rx_ring(struct ix_rx_ring *rxr) 382979251f5eSSepherosa Ziehau { 383079251f5eSSepherosa Ziehau int i; 383179251f5eSSepherosa Ziehau 383279251f5eSSepherosa Ziehau for (i = 0; i < rxr->rx_ndesc; ++i) { 383379251f5eSSepherosa Ziehau struct ix_rx_buf *rxbuf = &rxr->rx_buf[i]; 383479251f5eSSepherosa Ziehau 383579251f5eSSepherosa Ziehau if (rxbuf->fmp != NULL) { 383679251f5eSSepherosa Ziehau m_freem(rxbuf->fmp); 383779251f5eSSepherosa Ziehau rxbuf->fmp = NULL; 383879251f5eSSepherosa Ziehau rxbuf->lmp = NULL; 383979251f5eSSepherosa Ziehau } else { 384079251f5eSSepherosa Ziehau KKASSERT(rxbuf->lmp == NULL); 384179251f5eSSepherosa Ziehau } 384279251f5eSSepherosa Ziehau if (rxbuf->m_head != NULL) { 384379251f5eSSepherosa Ziehau bus_dmamap_unload(rxr->rx_tag, rxbuf->map); 384479251f5eSSepherosa Ziehau m_freem(rxbuf->m_head); 384579251f5eSSepherosa Ziehau rxbuf->m_head = NULL; 384679251f5eSSepherosa Ziehau } 384779251f5eSSepherosa Ziehau } 384879251f5eSSepherosa Ziehau } 384979251f5eSSepherosa Ziehau 385079251f5eSSepherosa Ziehau static int 385179251f5eSSepherosa Ziehau ix_newbuf(struct ix_rx_ring *rxr, int i, boolean_t wait) 385279251f5eSSepherosa Ziehau { 385379251f5eSSepherosa Ziehau struct mbuf *m; 385479251f5eSSepherosa Ziehau bus_dma_segment_t seg; 385579251f5eSSepherosa Ziehau bus_dmamap_t map; 385679251f5eSSepherosa Ziehau struct ix_rx_buf *rxbuf; 385779251f5eSSepherosa Ziehau int flags, error, nseg; 385879251f5eSSepherosa Ziehau 385979251f5eSSepherosa Ziehau flags = MB_DONTWAIT; 386079251f5eSSepherosa Ziehau if (__predict_false(wait)) 386179251f5eSSepherosa Ziehau flags = MB_WAIT; 386279251f5eSSepherosa Ziehau 386379251f5eSSepherosa Ziehau m = m_getjcl(flags, MT_DATA, M_PKTHDR, rxr->rx_mbuf_sz); 386479251f5eSSepherosa Ziehau if (m == NULL) { 386579251f5eSSepherosa Ziehau if (wait) { 386679251f5eSSepherosa Ziehau if_printf(&rxr->rx_sc->arpcom.ac_if, 386779251f5eSSepherosa Ziehau "Unable to allocate RX mbuf\n"); 386879251f5eSSepherosa Ziehau } 386979251f5eSSepherosa Ziehau return ENOBUFS; 387079251f5eSSepherosa Ziehau } 387179251f5eSSepherosa Ziehau m->m_len = m->m_pkthdr.len = rxr->rx_mbuf_sz; 387279251f5eSSepherosa Ziehau 387379251f5eSSepherosa Ziehau error = bus_dmamap_load_mbuf_segment(rxr->rx_tag, 387479251f5eSSepherosa Ziehau rxr->rx_sparemap, m, &seg, 1, &nseg, BUS_DMA_NOWAIT); 387579251f5eSSepherosa Ziehau if (error) { 387679251f5eSSepherosa Ziehau m_freem(m); 387779251f5eSSepherosa Ziehau if (wait) { 387879251f5eSSepherosa Ziehau if_printf(&rxr->rx_sc->arpcom.ac_if, 387979251f5eSSepherosa Ziehau "Unable to load RX mbuf\n"); 388079251f5eSSepherosa Ziehau } 388179251f5eSSepherosa Ziehau return error; 388279251f5eSSepherosa Ziehau } 388379251f5eSSepherosa Ziehau 388479251f5eSSepherosa Ziehau rxbuf = &rxr->rx_buf[i]; 388579251f5eSSepherosa Ziehau if (rxbuf->m_head != NULL) 388679251f5eSSepherosa Ziehau bus_dmamap_unload(rxr->rx_tag, rxbuf->map); 388779251f5eSSepherosa Ziehau 388879251f5eSSepherosa Ziehau map = rxbuf->map; 388979251f5eSSepherosa Ziehau rxbuf->map = rxr->rx_sparemap; 389079251f5eSSepherosa Ziehau rxr->rx_sparemap = map; 389179251f5eSSepherosa Ziehau 389279251f5eSSepherosa Ziehau rxbuf->m_head = m; 389379251f5eSSepherosa Ziehau rxbuf->paddr = seg.ds_addr; 389479251f5eSSepherosa Ziehau 389579251f5eSSepherosa Ziehau ix_setup_rxdesc(&rxr->rx_base[i], rxbuf); 389679251f5eSSepherosa Ziehau return 0; 389779251f5eSSepherosa Ziehau } 389879251f5eSSepherosa Ziehau 389979251f5eSSepherosa Ziehau static void 390079251f5eSSepherosa Ziehau ix_add_sysctl(struct ix_softc *sc) 390179251f5eSSepherosa Ziehau { 390279251f5eSSepherosa Ziehau #ifdef IX_RSS_DEBUG 390379251f5eSSepherosa Ziehau char node[32]; 390479251f5eSSepherosa Ziehau #endif 390579251f5eSSepherosa Ziehau 390679251f5eSSepherosa Ziehau sysctl_ctx_init(&sc->sysctl_ctx); 390779251f5eSSepherosa Ziehau sc->sysctl_tree = SYSCTL_ADD_NODE(&sc->sysctl_ctx, 390879251f5eSSepherosa Ziehau SYSCTL_STATIC_CHILDREN(_hw), OID_AUTO, 390979251f5eSSepherosa Ziehau device_get_nameunit(sc->dev), CTLFLAG_RD, 0, ""); 391079251f5eSSepherosa Ziehau if (sc->sysctl_tree == NULL) { 391179251f5eSSepherosa Ziehau device_printf(sc->dev, "can't add sysctl node\n"); 391279251f5eSSepherosa Ziehau return; 391379251f5eSSepherosa Ziehau } 391479251f5eSSepherosa Ziehau 391579251f5eSSepherosa Ziehau SYSCTL_ADD_INT(&sc->sysctl_ctx, SYSCTL_CHILDREN(sc->sysctl_tree), 391679251f5eSSepherosa Ziehau OID_AUTO, "rxr", CTLFLAG_RD, &sc->rx_ring_cnt, 0, "# of RX rings"); 391779251f5eSSepherosa Ziehau SYSCTL_ADD_INT(&sc->sysctl_ctx, SYSCTL_CHILDREN(sc->sysctl_tree), 391879251f5eSSepherosa Ziehau OID_AUTO, "rxr_inuse", CTLFLAG_RD, &sc->rx_ring_inuse, 0, 391979251f5eSSepherosa Ziehau "# of RX rings used"); 392079251f5eSSepherosa Ziehau SYSCTL_ADD_INT(&sc->sysctl_ctx, SYSCTL_CHILDREN(sc->sysctl_tree), 392179251f5eSSepherosa Ziehau OID_AUTO, "txr", CTLFLAG_RD, &sc->tx_ring_cnt, 0, "# of TX rings"); 392279251f5eSSepherosa Ziehau SYSCTL_ADD_INT(&sc->sysctl_ctx, SYSCTL_CHILDREN(sc->sysctl_tree), 392379251f5eSSepherosa Ziehau OID_AUTO, "txr_inuse", CTLFLAG_RD, &sc->tx_ring_inuse, 0, 392479251f5eSSepherosa Ziehau "# of TX rings used"); 392579251f5eSSepherosa Ziehau SYSCTL_ADD_PROC(&sc->sysctl_ctx, SYSCTL_CHILDREN(sc->sysctl_tree), 392679251f5eSSepherosa Ziehau OID_AUTO, "rxd", CTLTYPE_INT | CTLFLAG_RD, 392779251f5eSSepherosa Ziehau sc, 0, ix_sysctl_rxd, "I", 392879251f5eSSepherosa Ziehau "# of RX descs"); 392979251f5eSSepherosa Ziehau SYSCTL_ADD_PROC(&sc->sysctl_ctx, SYSCTL_CHILDREN(sc->sysctl_tree), 393079251f5eSSepherosa Ziehau OID_AUTO, "txd", CTLTYPE_INT | CTLFLAG_RD, 393179251f5eSSepherosa Ziehau sc, 0, ix_sysctl_txd, "I", 393279251f5eSSepherosa Ziehau "# of TX descs"); 393379251f5eSSepherosa Ziehau SYSCTL_ADD_PROC(&sc->sysctl_ctx, SYSCTL_CHILDREN(sc->sysctl_tree), 393479251f5eSSepherosa Ziehau OID_AUTO, "tx_wreg_nsegs", CTLTYPE_INT | CTLFLAG_RW, 393579251f5eSSepherosa Ziehau sc, 0, ix_sysctl_tx_wreg_nsegs, "I", 393679251f5eSSepherosa Ziehau "# of segments sent before write to hardware register"); 393779251f5eSSepherosa Ziehau SYSCTL_ADD_PROC(&sc->sysctl_ctx, SYSCTL_CHILDREN(sc->sysctl_tree), 393879251f5eSSepherosa Ziehau OID_AUTO, "rx_wreg_nsegs", CTLTYPE_INT | CTLFLAG_RW, 393979251f5eSSepherosa Ziehau sc, 0, ix_sysctl_rx_wreg_nsegs, "I", 394079251f5eSSepherosa Ziehau "# of received segments sent before write to hardware register"); 394179251f5eSSepherosa Ziehau SYSCTL_ADD_PROC(&sc->sysctl_ctx, SYSCTL_CHILDREN(sc->sysctl_tree), 394279251f5eSSepherosa Ziehau OID_AUTO, "tx_intr_nsegs", CTLTYPE_INT | CTLFLAG_RW, 394379251f5eSSepherosa Ziehau sc, 0, ix_sysctl_tx_intr_nsegs, "I", 394479251f5eSSepherosa Ziehau "# of segments per TX interrupt"); 394579251f5eSSepherosa Ziehau 3946*189a0ff3SSepherosa Ziehau #define IX_ADD_INTR_RATE_SYSCTL(sc, use, name) \ 3947*189a0ff3SSepherosa Ziehau do { \ 3948*189a0ff3SSepherosa Ziehau ix_add_intr_rate_sysctl(sc, IX_INTR_USE_##use, #name, \ 3949*189a0ff3SSepherosa Ziehau ix_sysctl_##name, #use " interrupt rate"); \ 3950*189a0ff3SSepherosa Ziehau } while (0) 3951*189a0ff3SSepherosa Ziehau 3952*189a0ff3SSepherosa Ziehau IX_ADD_INTR_RATE_SYSCTL(sc, RXTX, rxtx_intr_rate); 3953*189a0ff3SSepherosa Ziehau IX_ADD_INTR_RATE_SYSCTL(sc, RX, rx_intr_rate); 3954*189a0ff3SSepherosa Ziehau IX_ADD_INTR_RATE_SYSCTL(sc, TX, tx_intr_rate); 3955*189a0ff3SSepherosa Ziehau IX_ADD_INTR_RATE_SYSCTL(sc, STATUS, sts_intr_rate); 3956*189a0ff3SSepherosa Ziehau 3957*189a0ff3SSepherosa Ziehau #undef IX_ADD_INTR_RATE_SYSCTL 395879251f5eSSepherosa Ziehau 395979251f5eSSepherosa Ziehau #ifdef IX_RSS_DEBUG 396079251f5eSSepherosa Ziehau SYSCTL_ADD_INT(&sc->sysctl_ctx, SYSCTL_CHILDREN(sc->sysctl_tree), 396179251f5eSSepherosa Ziehau OID_AUTO, "rss_debug", CTLFLAG_RW, &sc->rss_debug, 0, 396279251f5eSSepherosa Ziehau "RSS debug level"); 396379251f5eSSepherosa Ziehau for (i = 0; i < sc->rx_ring_cnt; ++i) { 396479251f5eSSepherosa Ziehau ksnprintf(node, sizeof(node), "rx%d_pkt", i); 396579251f5eSSepherosa Ziehau SYSCTL_ADD_ULONG(&sc->sysctl_ctx, 396679251f5eSSepherosa Ziehau SYSCTL_CHILDREN(sc->sysctl_tree), OID_AUTO, node, 396779251f5eSSepherosa Ziehau CTLFLAG_RW, &sc->rx_rings[i].rx_pkts, "RXed packets"); 396879251f5eSSepherosa Ziehau } 396979251f5eSSepherosa Ziehau #endif 397079251f5eSSepherosa Ziehau 397179251f5eSSepherosa Ziehau SYSCTL_ADD_PROC(&sc->sysctl_ctx, SYSCTL_CHILDREN(sc->sysctl_tree), 397279251f5eSSepherosa Ziehau OID_AUTO, "flowctrl", CTLTYPE_INT | CTLFLAG_RW, 397379251f5eSSepherosa Ziehau sc, 0, ix_sysctl_flowctrl, "I", 397479251f5eSSepherosa Ziehau "flow control, 0 - off, 1 - rx pause, 2 - tx pause, 3 - full"); 397579251f5eSSepherosa Ziehau 397679251f5eSSepherosa Ziehau #ifdef foo 397779251f5eSSepherosa Ziehau /* 397879251f5eSSepherosa Ziehau * Allow a kind of speed control by forcing the autoneg 397979251f5eSSepherosa Ziehau * advertised speed list to only a certain value, this 398079251f5eSSepherosa Ziehau * supports 1G on 82599 devices, and 100Mb on X540. 398179251f5eSSepherosa Ziehau */ 398279251f5eSSepherosa Ziehau SYSCTL_ADD_PROC(&sc->sysctl_ctx, SYSCTL_CHILDREN(sc->sysctl_tree), 398379251f5eSSepherosa Ziehau OID_AUTO, "advspeed", CTLTYPE_INT | CTLFLAG_RW, 398479251f5eSSepherosa Ziehau sc, 0, ix_sysctl_advspeed, "I", 398579251f5eSSepherosa Ziehau "advertised link speed, " 398679251f5eSSepherosa Ziehau "0 - auto, 1 - 1Gb, 2 - 100Mb, 3 - 1Gb/10Gb"); 398779251f5eSSepherosa Ziehau #endif 398879251f5eSSepherosa Ziehau 398979251f5eSSepherosa Ziehau #if 0 399079251f5eSSepherosa Ziehau ix_add_hw_stats(sc); 399179251f5eSSepherosa Ziehau #endif 399279251f5eSSepherosa Ziehau 399379251f5eSSepherosa Ziehau } 399479251f5eSSepherosa Ziehau 399579251f5eSSepherosa Ziehau static int 399679251f5eSSepherosa Ziehau ix_sysctl_tx_wreg_nsegs(SYSCTL_HANDLER_ARGS) 399779251f5eSSepherosa Ziehau { 399879251f5eSSepherosa Ziehau struct ix_softc *sc = (void *)arg1; 399979251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 400079251f5eSSepherosa Ziehau int error, nsegs, i; 400179251f5eSSepherosa Ziehau 400279251f5eSSepherosa Ziehau nsegs = sc->tx_rings[0].tx_wreg_nsegs; 400379251f5eSSepherosa Ziehau error = sysctl_handle_int(oidp, &nsegs, 0, req); 400479251f5eSSepherosa Ziehau if (error || req->newptr == NULL) 400579251f5eSSepherosa Ziehau return error; 400679251f5eSSepherosa Ziehau if (nsegs < 0) 400779251f5eSSepherosa Ziehau return EINVAL; 400879251f5eSSepherosa Ziehau 400979251f5eSSepherosa Ziehau ifnet_serialize_all(ifp); 401079251f5eSSepherosa Ziehau for (i = 0; i < sc->tx_ring_cnt; ++i) 401179251f5eSSepherosa Ziehau sc->tx_rings[i].tx_wreg_nsegs = nsegs; 401279251f5eSSepherosa Ziehau ifnet_deserialize_all(ifp); 401379251f5eSSepherosa Ziehau 401479251f5eSSepherosa Ziehau return 0; 401579251f5eSSepherosa Ziehau } 401679251f5eSSepherosa Ziehau 401779251f5eSSepherosa Ziehau static int 401879251f5eSSepherosa Ziehau ix_sysctl_rx_wreg_nsegs(SYSCTL_HANDLER_ARGS) 401979251f5eSSepherosa Ziehau { 402079251f5eSSepherosa Ziehau struct ix_softc *sc = (void *)arg1; 402179251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 402279251f5eSSepherosa Ziehau int error, nsegs, i; 402379251f5eSSepherosa Ziehau 402479251f5eSSepherosa Ziehau nsegs = sc->rx_rings[0].rx_wreg_nsegs; 402579251f5eSSepherosa Ziehau error = sysctl_handle_int(oidp, &nsegs, 0, req); 402679251f5eSSepherosa Ziehau if (error || req->newptr == NULL) 402779251f5eSSepherosa Ziehau return error; 402879251f5eSSepherosa Ziehau if (nsegs < 0) 402979251f5eSSepherosa Ziehau return EINVAL; 403079251f5eSSepherosa Ziehau 403179251f5eSSepherosa Ziehau ifnet_serialize_all(ifp); 403279251f5eSSepherosa Ziehau for (i = 0; i < sc->rx_ring_cnt; ++i) 403379251f5eSSepherosa Ziehau sc->rx_rings[i].rx_wreg_nsegs =nsegs; 403479251f5eSSepherosa Ziehau ifnet_deserialize_all(ifp); 403579251f5eSSepherosa Ziehau 403679251f5eSSepherosa Ziehau return 0; 403779251f5eSSepherosa Ziehau } 403879251f5eSSepherosa Ziehau 403979251f5eSSepherosa Ziehau static int 404079251f5eSSepherosa Ziehau ix_sysctl_txd(SYSCTL_HANDLER_ARGS) 404179251f5eSSepherosa Ziehau { 404279251f5eSSepherosa Ziehau struct ix_softc *sc = (void *)arg1; 404379251f5eSSepherosa Ziehau int txd; 404479251f5eSSepherosa Ziehau 404579251f5eSSepherosa Ziehau txd = sc->tx_rings[0].tx_ndesc; 404679251f5eSSepherosa Ziehau return sysctl_handle_int(oidp, &txd, 0, req); 404779251f5eSSepherosa Ziehau } 404879251f5eSSepherosa Ziehau 404979251f5eSSepherosa Ziehau static int 405079251f5eSSepherosa Ziehau ix_sysctl_rxd(SYSCTL_HANDLER_ARGS) 405179251f5eSSepherosa Ziehau { 405279251f5eSSepherosa Ziehau struct ix_softc *sc = (void *)arg1; 405379251f5eSSepherosa Ziehau int rxd; 405479251f5eSSepherosa Ziehau 405579251f5eSSepherosa Ziehau rxd = sc->rx_rings[0].rx_ndesc; 405679251f5eSSepherosa Ziehau return sysctl_handle_int(oidp, &rxd, 0, req); 405779251f5eSSepherosa Ziehau } 405879251f5eSSepherosa Ziehau 405979251f5eSSepherosa Ziehau static int 406079251f5eSSepherosa Ziehau ix_sysctl_tx_intr_nsegs(SYSCTL_HANDLER_ARGS) 406179251f5eSSepherosa Ziehau { 406279251f5eSSepherosa Ziehau struct ix_softc *sc = (void *)arg1; 406379251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 406479251f5eSSepherosa Ziehau struct ix_tx_ring *txr = &sc->tx_rings[0]; 406579251f5eSSepherosa Ziehau int error, nsegs; 406679251f5eSSepherosa Ziehau 406779251f5eSSepherosa Ziehau nsegs = txr->tx_intr_nsegs; 406879251f5eSSepherosa Ziehau error = sysctl_handle_int(oidp, &nsegs, 0, req); 406979251f5eSSepherosa Ziehau if (error || req->newptr == NULL) 407079251f5eSSepherosa Ziehau return error; 407179251f5eSSepherosa Ziehau if (nsegs < 0) 407279251f5eSSepherosa Ziehau return EINVAL; 407379251f5eSSepherosa Ziehau 407479251f5eSSepherosa Ziehau ifnet_serialize_all(ifp); 407579251f5eSSepherosa Ziehau 407679251f5eSSepherosa Ziehau if (nsegs >= txr->tx_ndesc - IX_MAX_SCATTER - IX_TX_RESERVED) { 407779251f5eSSepherosa Ziehau error = EINVAL; 407879251f5eSSepherosa Ziehau } else { 407979251f5eSSepherosa Ziehau int i; 408079251f5eSSepherosa Ziehau 408179251f5eSSepherosa Ziehau error = 0; 408279251f5eSSepherosa Ziehau for (i = 0; i < sc->tx_ring_cnt; ++i) 408379251f5eSSepherosa Ziehau sc->tx_rings[i].tx_intr_nsegs = nsegs; 408479251f5eSSepherosa Ziehau } 408579251f5eSSepherosa Ziehau 408679251f5eSSepherosa Ziehau ifnet_deserialize_all(ifp); 408779251f5eSSepherosa Ziehau 408879251f5eSSepherosa Ziehau return error; 408979251f5eSSepherosa Ziehau } 409079251f5eSSepherosa Ziehau 409179251f5eSSepherosa Ziehau static void 409279251f5eSSepherosa Ziehau ix_set_eitr(struct ix_softc *sc, int idx, int rate) 409379251f5eSSepherosa Ziehau { 409479251f5eSSepherosa Ziehau uint32_t eitr, eitr_intvl; 409579251f5eSSepherosa Ziehau 409679251f5eSSepherosa Ziehau eitr = IXGBE_READ_REG(&sc->hw, IXGBE_EITR(idx)); 409779251f5eSSepherosa Ziehau eitr_intvl = 1000000000 / 256 / rate; 409879251f5eSSepherosa Ziehau 409979251f5eSSepherosa Ziehau if (sc->hw.mac.type == ixgbe_mac_82598EB) { 410079251f5eSSepherosa Ziehau eitr &= ~IX_EITR_INTVL_MASK_82598; 410179251f5eSSepherosa Ziehau if (eitr_intvl == 0) 410279251f5eSSepherosa Ziehau eitr_intvl = 1; 410379251f5eSSepherosa Ziehau else if (eitr_intvl > IX_EITR_INTVL_MASK_82598) 410479251f5eSSepherosa Ziehau eitr_intvl = IX_EITR_INTVL_MASK_82598; 410579251f5eSSepherosa Ziehau } else { 410679251f5eSSepherosa Ziehau eitr &= ~IX_EITR_INTVL_MASK; 410779251f5eSSepherosa Ziehau 410879251f5eSSepherosa Ziehau eitr_intvl &= ~IX_EITR_INTVL_RSVD_MASK; 410979251f5eSSepherosa Ziehau if (eitr_intvl == 0) 411079251f5eSSepherosa Ziehau eitr_intvl = IX_EITR_INTVL_MIN; 411179251f5eSSepherosa Ziehau else if (eitr_intvl > IX_EITR_INTVL_MAX) 411279251f5eSSepherosa Ziehau eitr_intvl = IX_EITR_INTVL_MAX; 411379251f5eSSepherosa Ziehau } 411479251f5eSSepherosa Ziehau eitr |= eitr_intvl; 411579251f5eSSepherosa Ziehau 411679251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_EITR(idx), eitr); 411779251f5eSSepherosa Ziehau } 411879251f5eSSepherosa Ziehau 411979251f5eSSepherosa Ziehau static int 4120*189a0ff3SSepherosa Ziehau ix_sysctl_rxtx_intr_rate(SYSCTL_HANDLER_ARGS) 4121*189a0ff3SSepherosa Ziehau { 4122*189a0ff3SSepherosa Ziehau return ix_sysctl_intr_rate(oidp, arg1, arg2, req, IX_INTR_USE_RXTX); 4123*189a0ff3SSepherosa Ziehau } 4124*189a0ff3SSepherosa Ziehau 4125*189a0ff3SSepherosa Ziehau static int 4126*189a0ff3SSepherosa Ziehau ix_sysctl_rx_intr_rate(SYSCTL_HANDLER_ARGS) 4127*189a0ff3SSepherosa Ziehau { 4128*189a0ff3SSepherosa Ziehau return ix_sysctl_intr_rate(oidp, arg1, arg2, req, IX_INTR_USE_RX); 4129*189a0ff3SSepherosa Ziehau } 4130*189a0ff3SSepherosa Ziehau 4131*189a0ff3SSepherosa Ziehau static int 4132*189a0ff3SSepherosa Ziehau ix_sysctl_tx_intr_rate(SYSCTL_HANDLER_ARGS) 4133*189a0ff3SSepherosa Ziehau { 4134*189a0ff3SSepherosa Ziehau return ix_sysctl_intr_rate(oidp, arg1, arg2, req, IX_INTR_USE_TX); 4135*189a0ff3SSepherosa Ziehau } 4136*189a0ff3SSepherosa Ziehau 4137*189a0ff3SSepherosa Ziehau static int 4138*189a0ff3SSepherosa Ziehau ix_sysctl_sts_intr_rate(SYSCTL_HANDLER_ARGS) 4139*189a0ff3SSepherosa Ziehau { 4140*189a0ff3SSepherosa Ziehau return ix_sysctl_intr_rate(oidp, arg1, arg2, req, IX_INTR_USE_STATUS); 4141*189a0ff3SSepherosa Ziehau } 4142*189a0ff3SSepherosa Ziehau 4143*189a0ff3SSepherosa Ziehau static int 4144*189a0ff3SSepherosa Ziehau ix_sysctl_intr_rate(SYSCTL_HANDLER_ARGS, int use) 414579251f5eSSepherosa Ziehau { 414679251f5eSSepherosa Ziehau struct ix_softc *sc = (void *)arg1; 414779251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 414879251f5eSSepherosa Ziehau int error, rate, i; 414979251f5eSSepherosa Ziehau 415079251f5eSSepherosa Ziehau rate = 0; 415179251f5eSSepherosa Ziehau for (i = 0; i < sc->intr_cnt; ++i) { 4152*189a0ff3SSepherosa Ziehau if (sc->intr_data[i].intr_use == use) { 415379251f5eSSepherosa Ziehau rate = sc->intr_data[i].intr_rate; 415479251f5eSSepherosa Ziehau break; 415579251f5eSSepherosa Ziehau } 415679251f5eSSepherosa Ziehau } 415779251f5eSSepherosa Ziehau 415879251f5eSSepherosa Ziehau error = sysctl_handle_int(oidp, &rate, 0, req); 415979251f5eSSepherosa Ziehau if (error || req->newptr == NULL) 416079251f5eSSepherosa Ziehau return error; 416179251f5eSSepherosa Ziehau if (rate <= 0) 416279251f5eSSepherosa Ziehau return EINVAL; 416379251f5eSSepherosa Ziehau 416479251f5eSSepherosa Ziehau ifnet_serialize_all(ifp); 416579251f5eSSepherosa Ziehau 416679251f5eSSepherosa Ziehau for (i = 0; i < sc->intr_cnt; ++i) { 4167*189a0ff3SSepherosa Ziehau if (sc->intr_data[i].intr_use == use) { 416879251f5eSSepherosa Ziehau sc->intr_data[i].intr_rate = rate; 416979251f5eSSepherosa Ziehau if (ifp->if_flags & IFF_RUNNING) 417079251f5eSSepherosa Ziehau ix_set_eitr(sc, i, rate); 417179251f5eSSepherosa Ziehau } 417279251f5eSSepherosa Ziehau } 417379251f5eSSepherosa Ziehau 417479251f5eSSepherosa Ziehau ifnet_deserialize_all(ifp); 417579251f5eSSepherosa Ziehau 417679251f5eSSepherosa Ziehau return error; 417779251f5eSSepherosa Ziehau } 417879251f5eSSepherosa Ziehau 417979251f5eSSepherosa Ziehau static void 4180*189a0ff3SSepherosa Ziehau ix_add_intr_rate_sysctl(struct ix_softc *sc, int use, 4181*189a0ff3SSepherosa Ziehau const char *name, int (*handler)(SYSCTL_HANDLER_ARGS), const char *desc) 4182*189a0ff3SSepherosa Ziehau { 4183*189a0ff3SSepherosa Ziehau int i; 4184*189a0ff3SSepherosa Ziehau 4185*189a0ff3SSepherosa Ziehau for (i = 0; i < sc->intr_cnt; ++i) { 4186*189a0ff3SSepherosa Ziehau if (sc->intr_data[i].intr_use == use) { 4187*189a0ff3SSepherosa Ziehau SYSCTL_ADD_PROC(&sc->sysctl_ctx, 4188*189a0ff3SSepherosa Ziehau SYSCTL_CHILDREN(sc->sysctl_tree), 4189*189a0ff3SSepherosa Ziehau OID_AUTO, name, CTLTYPE_INT | CTLFLAG_RW, 4190*189a0ff3SSepherosa Ziehau sc, 0, handler, "I", desc); 4191*189a0ff3SSepherosa Ziehau break; 4192*189a0ff3SSepherosa Ziehau } 4193*189a0ff3SSepherosa Ziehau } 4194*189a0ff3SSepherosa Ziehau } 4195*189a0ff3SSepherosa Ziehau 4196*189a0ff3SSepherosa Ziehau static void 419779251f5eSSepherosa Ziehau ix_set_timer_cpuid(struct ix_softc *sc, boolean_t polling) 419879251f5eSSepherosa Ziehau { 419979251f5eSSepherosa Ziehau if (polling || sc->intr_type == PCI_INTR_TYPE_MSIX) 420079251f5eSSepherosa Ziehau sc->timer_cpuid = 0; /* XXX fixed */ 420179251f5eSSepherosa Ziehau else 420279251f5eSSepherosa Ziehau sc->timer_cpuid = rman_get_cpuid(sc->intr_data[0].intr_res); 420379251f5eSSepherosa Ziehau } 4204*189a0ff3SSepherosa Ziehau 4205*189a0ff3SSepherosa Ziehau static void 4206*189a0ff3SSepherosa Ziehau ix_alloc_msix(struct ix_softc *sc) 4207*189a0ff3SSepherosa Ziehau { 4208*189a0ff3SSepherosa Ziehau int msix_enable, msix_cnt, msix_cnt2, alloc_cnt; 4209*189a0ff3SSepherosa Ziehau struct ix_intr_data *intr; 4210*189a0ff3SSepherosa Ziehau int i, x, error; 4211*189a0ff3SSepherosa Ziehau int offset, offset_def, agg_rxtx, ring_max; 4212*189a0ff3SSepherosa Ziehau boolean_t aggregate, setup = FALSE; 4213*189a0ff3SSepherosa Ziehau 4214*189a0ff3SSepherosa Ziehau msix_enable = ix_msix_enable; 4215*189a0ff3SSepherosa Ziehau /* 4216*189a0ff3SSepherosa Ziehau * Don't enable MSI-X on 82598 by default, see: 4217*189a0ff3SSepherosa Ziehau * 82598 specification update errata #38 4218*189a0ff3SSepherosa Ziehau */ 4219*189a0ff3SSepherosa Ziehau if (sc->hw.mac.type == ixgbe_mac_82598EB) 4220*189a0ff3SSepherosa Ziehau msix_enable = 0; 4221*189a0ff3SSepherosa Ziehau msix_enable = device_getenv_int(sc->dev, "msix.enable", msix_enable); 4222*189a0ff3SSepherosa Ziehau if (!msix_enable) 4223*189a0ff3SSepherosa Ziehau return; 4224*189a0ff3SSepherosa Ziehau 4225*189a0ff3SSepherosa Ziehau msix_cnt = pci_msix_count(sc->dev); 4226*189a0ff3SSepherosa Ziehau #ifdef IX_MSIX_DEBUG 4227*189a0ff3SSepherosa Ziehau msix_cnt = device_getenv_int(sc->dev, "msix.count", msix_cnt); 4228*189a0ff3SSepherosa Ziehau #endif 4229*189a0ff3SSepherosa Ziehau if (msix_cnt <= 1) { 4230*189a0ff3SSepherosa Ziehau /* One MSI-X model does not make sense */ 4231*189a0ff3SSepherosa Ziehau return; 4232*189a0ff3SSepherosa Ziehau } 4233*189a0ff3SSepherosa Ziehau 4234*189a0ff3SSepherosa Ziehau i = 0; 4235*189a0ff3SSepherosa Ziehau while ((1 << (i + 1)) <= msix_cnt) 4236*189a0ff3SSepherosa Ziehau ++i; 4237*189a0ff3SSepherosa Ziehau msix_cnt2 = 1 << i; 4238*189a0ff3SSepherosa Ziehau 4239*189a0ff3SSepherosa Ziehau if (bootverbose) { 4240*189a0ff3SSepherosa Ziehau device_printf(sc->dev, "MSI-X count %d/%d\n", 4241*189a0ff3SSepherosa Ziehau msix_cnt2, msix_cnt); 4242*189a0ff3SSepherosa Ziehau } 4243*189a0ff3SSepherosa Ziehau 4244*189a0ff3SSepherosa Ziehau KKASSERT(msix_cnt >= msix_cnt2); 4245*189a0ff3SSepherosa Ziehau if (msix_cnt == msix_cnt2) { 4246*189a0ff3SSepherosa Ziehau /* We need at least one MSI-X for link status */ 4247*189a0ff3SSepherosa Ziehau msix_cnt2 >>= 1; 4248*189a0ff3SSepherosa Ziehau if (msix_cnt2 <= 1) { 4249*189a0ff3SSepherosa Ziehau /* One MSI-X for RX/TX does not make sense */ 4250*189a0ff3SSepherosa Ziehau device_printf(sc->dev, "not enough MSI-X for TX/RX, " 4251*189a0ff3SSepherosa Ziehau "MSI-X count %d/%d\n", msix_cnt2, msix_cnt); 4252*189a0ff3SSepherosa Ziehau return; 4253*189a0ff3SSepherosa Ziehau } 4254*189a0ff3SSepherosa Ziehau KKASSERT(msix_cnt > msix_cnt2); 4255*189a0ff3SSepherosa Ziehau 4256*189a0ff3SSepherosa Ziehau if (bootverbose) { 4257*189a0ff3SSepherosa Ziehau device_printf(sc->dev, "MSI-X count eq fixup %d/%d\n", 4258*189a0ff3SSepherosa Ziehau msix_cnt2, msix_cnt); 4259*189a0ff3SSepherosa Ziehau } 4260*189a0ff3SSepherosa Ziehau } 4261*189a0ff3SSepherosa Ziehau 4262*189a0ff3SSepherosa Ziehau /* 4263*189a0ff3SSepherosa Ziehau * Make sure that we don't break interrupt related registers 4264*189a0ff3SSepherosa Ziehau * (EIMS, etc) limitation. 4265*189a0ff3SSepherosa Ziehau * 4266*189a0ff3SSepherosa Ziehau * NOTE: msix_cnt > msix_cnt2, when we reach here 4267*189a0ff3SSepherosa Ziehau */ 4268*189a0ff3SSepherosa Ziehau if (sc->hw.mac.type == ixgbe_mac_82598EB) { 4269*189a0ff3SSepherosa Ziehau if (msix_cnt2 > IX_MAX_MSIX_82598) 4270*189a0ff3SSepherosa Ziehau msix_cnt2 = IX_MAX_MSIX_82598; 4271*189a0ff3SSepherosa Ziehau } else { 4272*189a0ff3SSepherosa Ziehau if (msix_cnt2 > IX_MAX_MSIX) 4273*189a0ff3SSepherosa Ziehau msix_cnt2 = IX_MAX_MSIX; 4274*189a0ff3SSepherosa Ziehau } 4275*189a0ff3SSepherosa Ziehau msix_cnt = msix_cnt2 + 1; /* +1 for status */ 4276*189a0ff3SSepherosa Ziehau 4277*189a0ff3SSepherosa Ziehau if (bootverbose) { 4278*189a0ff3SSepherosa Ziehau device_printf(sc->dev, "MSI-X count max fixup %d/%d\n", 4279*189a0ff3SSepherosa Ziehau msix_cnt2, msix_cnt); 4280*189a0ff3SSepherosa Ziehau } 4281*189a0ff3SSepherosa Ziehau 4282*189a0ff3SSepherosa Ziehau sc->rx_ring_msix = sc->rx_ring_cnt; 4283*189a0ff3SSepherosa Ziehau if (sc->rx_ring_msix > msix_cnt2) 4284*189a0ff3SSepherosa Ziehau sc->rx_ring_msix = msix_cnt2; 4285*189a0ff3SSepherosa Ziehau 4286*189a0ff3SSepherosa Ziehau sc->tx_ring_msix = sc->tx_ring_cnt; 4287*189a0ff3SSepherosa Ziehau if (sc->tx_ring_msix > msix_cnt2) 4288*189a0ff3SSepherosa Ziehau sc->tx_ring_msix = msix_cnt2; 4289*189a0ff3SSepherosa Ziehau 4290*189a0ff3SSepherosa Ziehau ring_max = sc->rx_ring_msix; 4291*189a0ff3SSepherosa Ziehau if (ring_max < sc->tx_ring_msix) 4292*189a0ff3SSepherosa Ziehau ring_max = sc->tx_ring_msix; 4293*189a0ff3SSepherosa Ziehau 4294*189a0ff3SSepherosa Ziehau /* Allow user to force independent RX/TX MSI-X handling */ 4295*189a0ff3SSepherosa Ziehau agg_rxtx = device_getenv_int(sc->dev, "msix.agg_rxtx", 4296*189a0ff3SSepherosa Ziehau ix_msix_agg_rxtx); 4297*189a0ff3SSepherosa Ziehau 4298*189a0ff3SSepherosa Ziehau if (!agg_rxtx && msix_cnt >= sc->tx_ring_msix + sc->rx_ring_msix + 1) { 4299*189a0ff3SSepherosa Ziehau /* 4300*189a0ff3SSepherosa Ziehau * Independent TX/RX MSI-X 4301*189a0ff3SSepherosa Ziehau */ 4302*189a0ff3SSepherosa Ziehau aggregate = FALSE; 4303*189a0ff3SSepherosa Ziehau if (bootverbose) 4304*189a0ff3SSepherosa Ziehau device_printf(sc->dev, "independent TX/RX MSI-X\n"); 4305*189a0ff3SSepherosa Ziehau alloc_cnt = sc->tx_ring_msix + sc->rx_ring_msix; 4306*189a0ff3SSepherosa Ziehau } else { 4307*189a0ff3SSepherosa Ziehau /* 4308*189a0ff3SSepherosa Ziehau * Aggregate TX/RX MSI-X 4309*189a0ff3SSepherosa Ziehau */ 4310*189a0ff3SSepherosa Ziehau aggregate = TRUE; 4311*189a0ff3SSepherosa Ziehau if (bootverbose) 4312*189a0ff3SSepherosa Ziehau device_printf(sc->dev, "aggregate TX/RX MSI-X\n"); 4313*189a0ff3SSepherosa Ziehau alloc_cnt = msix_cnt2; 4314*189a0ff3SSepherosa Ziehau if (alloc_cnt > ring_max) 4315*189a0ff3SSepherosa Ziehau alloc_cnt = ring_max; 4316*189a0ff3SSepherosa Ziehau KKASSERT(alloc_cnt >= sc->rx_ring_msix && 4317*189a0ff3SSepherosa Ziehau alloc_cnt >= sc->tx_ring_msix); 4318*189a0ff3SSepherosa Ziehau } 4319*189a0ff3SSepherosa Ziehau ++alloc_cnt; /* For status */ 4320*189a0ff3SSepherosa Ziehau 4321*189a0ff3SSepherosa Ziehau if (bootverbose) { 4322*189a0ff3SSepherosa Ziehau device_printf(sc->dev, "MSI-X alloc %d, " 4323*189a0ff3SSepherosa Ziehau "RX ring %d, TX ring %d\n", alloc_cnt, 4324*189a0ff3SSepherosa Ziehau sc->rx_ring_msix, sc->tx_ring_msix); 4325*189a0ff3SSepherosa Ziehau } 4326*189a0ff3SSepherosa Ziehau 4327*189a0ff3SSepherosa Ziehau sc->msix_mem_rid = PCIR_BAR(IX_MSIX_BAR_82598); 4328*189a0ff3SSepherosa Ziehau sc->msix_mem_res = bus_alloc_resource_any(sc->dev, SYS_RES_MEMORY, 4329*189a0ff3SSepherosa Ziehau &sc->msix_mem_rid, RF_ACTIVE); 4330*189a0ff3SSepherosa Ziehau if (sc->msix_mem_res == NULL) { 4331*189a0ff3SSepherosa Ziehau sc->msix_mem_rid = PCIR_BAR(IX_MSIX_BAR_82599); 4332*189a0ff3SSepherosa Ziehau sc->msix_mem_res = bus_alloc_resource_any(sc->dev, 4333*189a0ff3SSepherosa Ziehau SYS_RES_MEMORY, &sc->msix_mem_rid, RF_ACTIVE); 4334*189a0ff3SSepherosa Ziehau if (sc->msix_mem_res == NULL) { 4335*189a0ff3SSepherosa Ziehau device_printf(sc->dev, "Unable to map MSI-X table\n"); 4336*189a0ff3SSepherosa Ziehau return; 4337*189a0ff3SSepherosa Ziehau } 4338*189a0ff3SSepherosa Ziehau } 4339*189a0ff3SSepherosa Ziehau 4340*189a0ff3SSepherosa Ziehau sc->intr_cnt = alloc_cnt; 4341*189a0ff3SSepherosa Ziehau sc->intr_data = kmalloc(sizeof(struct ix_intr_data) * sc->intr_cnt, 4342*189a0ff3SSepherosa Ziehau M_DEVBUF, M_WAITOK | M_ZERO); 4343*189a0ff3SSepherosa Ziehau for (x = 0; x < sc->intr_cnt; ++x) { 4344*189a0ff3SSepherosa Ziehau intr = &sc->intr_data[x]; 4345*189a0ff3SSepherosa Ziehau intr->intr_rid = -1; 4346*189a0ff3SSepherosa Ziehau intr->intr_rate = IX_INTR_RATE; 4347*189a0ff3SSepherosa Ziehau } 4348*189a0ff3SSepherosa Ziehau 4349*189a0ff3SSepherosa Ziehau x = 0; 4350*189a0ff3SSepherosa Ziehau if (!aggregate) { 4351*189a0ff3SSepherosa Ziehau /* 4352*189a0ff3SSepherosa Ziehau * RX rings 4353*189a0ff3SSepherosa Ziehau */ 4354*189a0ff3SSepherosa Ziehau if (sc->rx_ring_msix == ncpus2) { 4355*189a0ff3SSepherosa Ziehau offset = 0; 4356*189a0ff3SSepherosa Ziehau } else { 4357*189a0ff3SSepherosa Ziehau offset_def = (sc->rx_ring_msix * 4358*189a0ff3SSepherosa Ziehau device_get_unit(sc->dev)) % ncpus2; 4359*189a0ff3SSepherosa Ziehau 4360*189a0ff3SSepherosa Ziehau offset = device_getenv_int(sc->dev, 4361*189a0ff3SSepherosa Ziehau "msix.rxoff", offset_def); 4362*189a0ff3SSepherosa Ziehau if (offset >= ncpus2 || 4363*189a0ff3SSepherosa Ziehau offset % sc->rx_ring_msix != 0) { 4364*189a0ff3SSepherosa Ziehau device_printf(sc->dev, 4365*189a0ff3SSepherosa Ziehau "invalid msix.rxoff %d, use %d\n", 4366*189a0ff3SSepherosa Ziehau offset, offset_def); 4367*189a0ff3SSepherosa Ziehau offset = offset_def; 4368*189a0ff3SSepherosa Ziehau } 4369*189a0ff3SSepherosa Ziehau } 4370*189a0ff3SSepherosa Ziehau ix_conf_rx_msix(sc, 0, &x, offset); 4371*189a0ff3SSepherosa Ziehau 4372*189a0ff3SSepherosa Ziehau /* 4373*189a0ff3SSepherosa Ziehau * TX rings 4374*189a0ff3SSepherosa Ziehau */ 4375*189a0ff3SSepherosa Ziehau if (sc->tx_ring_msix == ncpus2) { 4376*189a0ff3SSepherosa Ziehau offset = 0; 4377*189a0ff3SSepherosa Ziehau } else { 4378*189a0ff3SSepherosa Ziehau offset_def = (sc->tx_ring_msix * 4379*189a0ff3SSepherosa Ziehau device_get_unit(sc->dev)) % ncpus2; 4380*189a0ff3SSepherosa Ziehau 4381*189a0ff3SSepherosa Ziehau offset = device_getenv_int(sc->dev, 4382*189a0ff3SSepherosa Ziehau "msix.txoff", offset_def); 4383*189a0ff3SSepherosa Ziehau if (offset >= ncpus2 || 4384*189a0ff3SSepherosa Ziehau offset % sc->tx_ring_msix != 0) { 4385*189a0ff3SSepherosa Ziehau device_printf(sc->dev, 4386*189a0ff3SSepherosa Ziehau "invalid msix.txoff %d, use %d\n", 4387*189a0ff3SSepherosa Ziehau offset, offset_def); 4388*189a0ff3SSepherosa Ziehau offset = offset_def; 4389*189a0ff3SSepherosa Ziehau } 4390*189a0ff3SSepherosa Ziehau } 4391*189a0ff3SSepherosa Ziehau ix_conf_tx_msix(sc, 0, &x, offset); 4392*189a0ff3SSepherosa Ziehau } else { 4393*189a0ff3SSepherosa Ziehau int ring_agg; 4394*189a0ff3SSepherosa Ziehau 4395*189a0ff3SSepherosa Ziehau ring_agg = sc->rx_ring_msix; 4396*189a0ff3SSepherosa Ziehau if (ring_agg > sc->tx_ring_msix) 4397*189a0ff3SSepherosa Ziehau ring_agg = sc->tx_ring_msix; 4398*189a0ff3SSepherosa Ziehau 4399*189a0ff3SSepherosa Ziehau if (ring_max == ncpus2) { 4400*189a0ff3SSepherosa Ziehau offset = 0; 4401*189a0ff3SSepherosa Ziehau } else { 4402*189a0ff3SSepherosa Ziehau offset_def = (ring_max * device_get_unit(sc->dev)) % 4403*189a0ff3SSepherosa Ziehau ncpus2; 4404*189a0ff3SSepherosa Ziehau 4405*189a0ff3SSepherosa Ziehau offset = device_getenv_int(sc->dev, "msix.off", 4406*189a0ff3SSepherosa Ziehau offset_def); 4407*189a0ff3SSepherosa Ziehau if (offset >= ncpus2 || offset % ring_max != 0) { 4408*189a0ff3SSepherosa Ziehau device_printf(sc->dev, 4409*189a0ff3SSepherosa Ziehau "invalid msix.off %d, use %d\n", 4410*189a0ff3SSepherosa Ziehau offset, offset_def); 4411*189a0ff3SSepherosa Ziehau offset = offset_def; 4412*189a0ff3SSepherosa Ziehau } 4413*189a0ff3SSepherosa Ziehau } 4414*189a0ff3SSepherosa Ziehau 4415*189a0ff3SSepherosa Ziehau for (i = 0; i < ring_agg; ++i) { 4416*189a0ff3SSepherosa Ziehau struct ix_tx_ring *txr = &sc->tx_rings[i]; 4417*189a0ff3SSepherosa Ziehau struct ix_rx_ring *rxr = &sc->rx_rings[i]; 4418*189a0ff3SSepherosa Ziehau 4419*189a0ff3SSepherosa Ziehau KKASSERT(x < sc->intr_cnt); 4420*189a0ff3SSepherosa Ziehau rxr->rx_intr_vec = x; 4421*189a0ff3SSepherosa Ziehau ix_setup_msix_eims(sc, x, 4422*189a0ff3SSepherosa Ziehau &rxr->rx_eims, &rxr->rx_eims_val); 4423*189a0ff3SSepherosa Ziehau rxr->rx_txr = txr; 4424*189a0ff3SSepherosa Ziehau /* NOTE: Leave TX ring's intr_vec negative */ 4425*189a0ff3SSepherosa Ziehau 4426*189a0ff3SSepherosa Ziehau intr = &sc->intr_data[x++]; 4427*189a0ff3SSepherosa Ziehau 4428*189a0ff3SSepherosa Ziehau intr->intr_serialize = &rxr->rx_serialize; 4429*189a0ff3SSepherosa Ziehau intr->intr_func = ix_msix_rxtx; 4430*189a0ff3SSepherosa Ziehau intr->intr_funcarg = rxr; 4431*189a0ff3SSepherosa Ziehau intr->intr_use = IX_INTR_USE_RXTX; 4432*189a0ff3SSepherosa Ziehau 4433*189a0ff3SSepherosa Ziehau intr->intr_cpuid = i + offset; 4434*189a0ff3SSepherosa Ziehau KKASSERT(intr->intr_cpuid < ncpus2); 4435*189a0ff3SSepherosa Ziehau txr->tx_intr_cpuid = intr->intr_cpuid; 4436*189a0ff3SSepherosa Ziehau 4437*189a0ff3SSepherosa Ziehau ksnprintf(intr->intr_desc0, sizeof(intr->intr_desc0), 4438*189a0ff3SSepherosa Ziehau "%s rxtx%d", device_get_nameunit(sc->dev), i); 4439*189a0ff3SSepherosa Ziehau intr->intr_desc = intr->intr_desc0; 4440*189a0ff3SSepherosa Ziehau } 4441*189a0ff3SSepherosa Ziehau 4442*189a0ff3SSepherosa Ziehau if (ring_agg != ring_max) { 4443*189a0ff3SSepherosa Ziehau if (ring_max == sc->tx_ring_msix) 4444*189a0ff3SSepherosa Ziehau ix_conf_tx_msix(sc, i, &x, offset); 4445*189a0ff3SSepherosa Ziehau else 4446*189a0ff3SSepherosa Ziehau ix_conf_rx_msix(sc, i, &x, offset); 4447*189a0ff3SSepherosa Ziehau } 4448*189a0ff3SSepherosa Ziehau } 4449*189a0ff3SSepherosa Ziehau 4450*189a0ff3SSepherosa Ziehau /* 4451*189a0ff3SSepherosa Ziehau * Status MSI-X 4452*189a0ff3SSepherosa Ziehau */ 4453*189a0ff3SSepherosa Ziehau KKASSERT(x < sc->intr_cnt); 4454*189a0ff3SSepherosa Ziehau sc->sts_msix_vec = x; 4455*189a0ff3SSepherosa Ziehau 4456*189a0ff3SSepherosa Ziehau intr = &sc->intr_data[x++]; 4457*189a0ff3SSepherosa Ziehau 4458*189a0ff3SSepherosa Ziehau intr->intr_serialize = &sc->main_serialize; 4459*189a0ff3SSepherosa Ziehau intr->intr_func = ix_msix_status; 4460*189a0ff3SSepherosa Ziehau intr->intr_funcarg = sc; 4461*189a0ff3SSepherosa Ziehau intr->intr_cpuid = 0; 4462*189a0ff3SSepherosa Ziehau intr->intr_use = IX_INTR_USE_STATUS; 4463*189a0ff3SSepherosa Ziehau 4464*189a0ff3SSepherosa Ziehau ksnprintf(intr->intr_desc0, sizeof(intr->intr_desc0), "%s sts", 4465*189a0ff3SSepherosa Ziehau device_get_nameunit(sc->dev)); 4466*189a0ff3SSepherosa Ziehau intr->intr_desc = intr->intr_desc0; 4467*189a0ff3SSepherosa Ziehau 4468*189a0ff3SSepherosa Ziehau KKASSERT(x == sc->intr_cnt); 4469*189a0ff3SSepherosa Ziehau 4470*189a0ff3SSepherosa Ziehau error = pci_setup_msix(sc->dev); 4471*189a0ff3SSepherosa Ziehau if (error) { 4472*189a0ff3SSepherosa Ziehau device_printf(sc->dev, "Setup MSI-X failed\n"); 4473*189a0ff3SSepherosa Ziehau goto back; 4474*189a0ff3SSepherosa Ziehau } 4475*189a0ff3SSepherosa Ziehau setup = TRUE; 4476*189a0ff3SSepherosa Ziehau 4477*189a0ff3SSepherosa Ziehau for (i = 0; i < sc->intr_cnt; ++i) { 4478*189a0ff3SSepherosa Ziehau intr = &sc->intr_data[i]; 4479*189a0ff3SSepherosa Ziehau 4480*189a0ff3SSepherosa Ziehau error = pci_alloc_msix_vector(sc->dev, i, &intr->intr_rid, 4481*189a0ff3SSepherosa Ziehau intr->intr_cpuid); 4482*189a0ff3SSepherosa Ziehau if (error) { 4483*189a0ff3SSepherosa Ziehau device_printf(sc->dev, 4484*189a0ff3SSepherosa Ziehau "Unable to allocate MSI-X %d on cpu%d\n", i, 4485*189a0ff3SSepherosa Ziehau intr->intr_cpuid); 4486*189a0ff3SSepherosa Ziehau goto back; 4487*189a0ff3SSepherosa Ziehau } 4488*189a0ff3SSepherosa Ziehau 4489*189a0ff3SSepherosa Ziehau intr->intr_res = bus_alloc_resource_any(sc->dev, SYS_RES_IRQ, 4490*189a0ff3SSepherosa Ziehau &intr->intr_rid, RF_ACTIVE); 4491*189a0ff3SSepherosa Ziehau if (intr->intr_res == NULL) { 4492*189a0ff3SSepherosa Ziehau device_printf(sc->dev, 4493*189a0ff3SSepherosa Ziehau "Unable to allocate MSI-X %d resource\n", i); 4494*189a0ff3SSepherosa Ziehau error = ENOMEM; 4495*189a0ff3SSepherosa Ziehau goto back; 4496*189a0ff3SSepherosa Ziehau } 4497*189a0ff3SSepherosa Ziehau } 4498*189a0ff3SSepherosa Ziehau 4499*189a0ff3SSepherosa Ziehau pci_enable_msix(sc->dev); 4500*189a0ff3SSepherosa Ziehau sc->intr_type = PCI_INTR_TYPE_MSIX; 4501*189a0ff3SSepherosa Ziehau back: 4502*189a0ff3SSepherosa Ziehau if (error) 4503*189a0ff3SSepherosa Ziehau ix_free_msix(sc, setup); 4504*189a0ff3SSepherosa Ziehau } 4505*189a0ff3SSepherosa Ziehau 4506*189a0ff3SSepherosa Ziehau static void 4507*189a0ff3SSepherosa Ziehau ix_free_msix(struct ix_softc *sc, boolean_t setup) 4508*189a0ff3SSepherosa Ziehau { 4509*189a0ff3SSepherosa Ziehau int i; 4510*189a0ff3SSepherosa Ziehau 4511*189a0ff3SSepherosa Ziehau KKASSERT(sc->intr_cnt > 1); 4512*189a0ff3SSepherosa Ziehau 4513*189a0ff3SSepherosa Ziehau for (i = 0; i < sc->intr_cnt; ++i) { 4514*189a0ff3SSepherosa Ziehau struct ix_intr_data *intr = &sc->intr_data[i]; 4515*189a0ff3SSepherosa Ziehau 4516*189a0ff3SSepherosa Ziehau if (intr->intr_res != NULL) { 4517*189a0ff3SSepherosa Ziehau bus_release_resource(sc->dev, SYS_RES_IRQ, 4518*189a0ff3SSepherosa Ziehau intr->intr_rid, intr->intr_res); 4519*189a0ff3SSepherosa Ziehau } 4520*189a0ff3SSepherosa Ziehau if (intr->intr_rid >= 0) 4521*189a0ff3SSepherosa Ziehau pci_release_msix_vector(sc->dev, intr->intr_rid); 4522*189a0ff3SSepherosa Ziehau } 4523*189a0ff3SSepherosa Ziehau if (setup) 4524*189a0ff3SSepherosa Ziehau pci_teardown_msix(sc->dev); 4525*189a0ff3SSepherosa Ziehau 4526*189a0ff3SSepherosa Ziehau sc->intr_cnt = 0; 4527*189a0ff3SSepherosa Ziehau kfree(sc->intr_data, M_DEVBUF); 4528*189a0ff3SSepherosa Ziehau sc->intr_data = NULL; 4529*189a0ff3SSepherosa Ziehau } 4530*189a0ff3SSepherosa Ziehau 4531*189a0ff3SSepherosa Ziehau static void 4532*189a0ff3SSepherosa Ziehau ix_conf_rx_msix(struct ix_softc *sc, int i, int *x0, int offset) 4533*189a0ff3SSepherosa Ziehau { 4534*189a0ff3SSepherosa Ziehau int x = *x0; 4535*189a0ff3SSepherosa Ziehau 4536*189a0ff3SSepherosa Ziehau for (; i < sc->rx_ring_msix; ++i) { 4537*189a0ff3SSepherosa Ziehau struct ix_rx_ring *rxr = &sc->rx_rings[i]; 4538*189a0ff3SSepherosa Ziehau struct ix_intr_data *intr; 4539*189a0ff3SSepherosa Ziehau 4540*189a0ff3SSepherosa Ziehau KKASSERT(x < sc->intr_cnt); 4541*189a0ff3SSepherosa Ziehau rxr->rx_intr_vec = x; 4542*189a0ff3SSepherosa Ziehau ix_setup_msix_eims(sc, x, &rxr->rx_eims, &rxr->rx_eims_val); 4543*189a0ff3SSepherosa Ziehau 4544*189a0ff3SSepherosa Ziehau intr = &sc->intr_data[x++]; 4545*189a0ff3SSepherosa Ziehau 4546*189a0ff3SSepherosa Ziehau intr->intr_serialize = &rxr->rx_serialize; 4547*189a0ff3SSepherosa Ziehau intr->intr_func = ix_msix_rx; 4548*189a0ff3SSepherosa Ziehau intr->intr_funcarg = rxr; 4549*189a0ff3SSepherosa Ziehau intr->intr_rate = IX_MSIX_RX_RATE; 4550*189a0ff3SSepherosa Ziehau intr->intr_use = IX_INTR_USE_RX; 4551*189a0ff3SSepherosa Ziehau 4552*189a0ff3SSepherosa Ziehau intr->intr_cpuid = i + offset; 4553*189a0ff3SSepherosa Ziehau KKASSERT(intr->intr_cpuid < ncpus2); 4554*189a0ff3SSepherosa Ziehau 4555*189a0ff3SSepherosa Ziehau ksnprintf(intr->intr_desc0, sizeof(intr->intr_desc0), "%s rx%d", 4556*189a0ff3SSepherosa Ziehau device_get_nameunit(sc->dev), i); 4557*189a0ff3SSepherosa Ziehau intr->intr_desc = intr->intr_desc0; 4558*189a0ff3SSepherosa Ziehau } 4559*189a0ff3SSepherosa Ziehau *x0 = x; 4560*189a0ff3SSepherosa Ziehau } 4561*189a0ff3SSepherosa Ziehau 4562*189a0ff3SSepherosa Ziehau static void 4563*189a0ff3SSepherosa Ziehau ix_conf_tx_msix(struct ix_softc *sc, int i, int *x0, int offset) 4564*189a0ff3SSepherosa Ziehau { 4565*189a0ff3SSepherosa Ziehau int x = *x0; 4566*189a0ff3SSepherosa Ziehau 4567*189a0ff3SSepherosa Ziehau for (; i < sc->tx_ring_msix; ++i) { 4568*189a0ff3SSepherosa Ziehau struct ix_tx_ring *txr = &sc->tx_rings[i]; 4569*189a0ff3SSepherosa Ziehau struct ix_intr_data *intr; 4570*189a0ff3SSepherosa Ziehau 4571*189a0ff3SSepherosa Ziehau KKASSERT(x < sc->intr_cnt); 4572*189a0ff3SSepherosa Ziehau txr->tx_intr_vec = x; 4573*189a0ff3SSepherosa Ziehau ix_setup_msix_eims(sc, x, &txr->tx_eims, &txr->tx_eims_val); 4574*189a0ff3SSepherosa Ziehau 4575*189a0ff3SSepherosa Ziehau intr = &sc->intr_data[x++]; 4576*189a0ff3SSepherosa Ziehau 4577*189a0ff3SSepherosa Ziehau intr->intr_serialize = &txr->tx_serialize; 4578*189a0ff3SSepherosa Ziehau intr->intr_func = ix_msix_tx; 4579*189a0ff3SSepherosa Ziehau intr->intr_funcarg = txr; 4580*189a0ff3SSepherosa Ziehau intr->intr_rate = IX_MSIX_TX_RATE; 4581*189a0ff3SSepherosa Ziehau intr->intr_use = IX_INTR_USE_TX; 4582*189a0ff3SSepherosa Ziehau 4583*189a0ff3SSepherosa Ziehau intr->intr_cpuid = i + offset; 4584*189a0ff3SSepherosa Ziehau KKASSERT(intr->intr_cpuid < ncpus2); 4585*189a0ff3SSepherosa Ziehau txr->tx_intr_cpuid = intr->intr_cpuid; 4586*189a0ff3SSepherosa Ziehau 4587*189a0ff3SSepherosa Ziehau ksnprintf(intr->intr_desc0, sizeof(intr->intr_desc0), "%s tx%d", 4588*189a0ff3SSepherosa Ziehau device_get_nameunit(sc->dev), i); 4589*189a0ff3SSepherosa Ziehau intr->intr_desc = intr->intr_desc0; 4590*189a0ff3SSepherosa Ziehau } 4591*189a0ff3SSepherosa Ziehau *x0 = x; 4592*189a0ff3SSepherosa Ziehau } 4593*189a0ff3SSepherosa Ziehau 4594*189a0ff3SSepherosa Ziehau static void 4595*189a0ff3SSepherosa Ziehau ix_msix_rx(void *xrxr) 4596*189a0ff3SSepherosa Ziehau { 4597*189a0ff3SSepherosa Ziehau struct ix_rx_ring *rxr = xrxr; 4598*189a0ff3SSepherosa Ziehau 4599*189a0ff3SSepherosa Ziehau ASSERT_SERIALIZED(&rxr->rx_serialize); 4600*189a0ff3SSepherosa Ziehau 4601*189a0ff3SSepherosa Ziehau ix_rxeof(rxr); 4602*189a0ff3SSepherosa Ziehau IXGBE_WRITE_REG(&rxr->rx_sc->hw, rxr->rx_eims, rxr->rx_eims_val); 4603*189a0ff3SSepherosa Ziehau } 4604*189a0ff3SSepherosa Ziehau 4605*189a0ff3SSepherosa Ziehau static void 4606*189a0ff3SSepherosa Ziehau ix_msix_tx(void *xtxr) 4607*189a0ff3SSepherosa Ziehau { 4608*189a0ff3SSepherosa Ziehau struct ix_tx_ring *txr = xtxr; 4609*189a0ff3SSepherosa Ziehau 4610*189a0ff3SSepherosa Ziehau ASSERT_SERIALIZED(&txr->tx_serialize); 4611*189a0ff3SSepherosa Ziehau 4612*189a0ff3SSepherosa Ziehau ix_txeof(txr, *(txr->tx_hdr)); 4613*189a0ff3SSepherosa Ziehau if (!ifsq_is_empty(txr->tx_ifsq)) 4614*189a0ff3SSepherosa Ziehau ifsq_devstart(txr->tx_ifsq); 4615*189a0ff3SSepherosa Ziehau IXGBE_WRITE_REG(&txr->tx_sc->hw, txr->tx_eims, txr->tx_eims_val); 4616*189a0ff3SSepherosa Ziehau } 4617*189a0ff3SSepherosa Ziehau 4618*189a0ff3SSepherosa Ziehau static void 4619*189a0ff3SSepherosa Ziehau ix_msix_rxtx(void *xrxr) 4620*189a0ff3SSepherosa Ziehau { 4621*189a0ff3SSepherosa Ziehau struct ix_rx_ring *rxr = xrxr; 4622*189a0ff3SSepherosa Ziehau struct ix_tx_ring *txr; 4623*189a0ff3SSepherosa Ziehau int hdr; 4624*189a0ff3SSepherosa Ziehau 4625*189a0ff3SSepherosa Ziehau ASSERT_SERIALIZED(&rxr->rx_serialize); 4626*189a0ff3SSepherosa Ziehau 4627*189a0ff3SSepherosa Ziehau ix_rxeof(rxr); 4628*189a0ff3SSepherosa Ziehau 4629*189a0ff3SSepherosa Ziehau /* 4630*189a0ff3SSepherosa Ziehau * NOTE: 4631*189a0ff3SSepherosa Ziehau * Since tx_next_clean is only changed by ix_txeof(), 4632*189a0ff3SSepherosa Ziehau * which is called only in interrupt handler, the 4633*189a0ff3SSepherosa Ziehau * check w/o holding tx serializer is MPSAFE. 4634*189a0ff3SSepherosa Ziehau */ 4635*189a0ff3SSepherosa Ziehau txr = rxr->rx_txr; 4636*189a0ff3SSepherosa Ziehau hdr = *(txr->tx_hdr); 4637*189a0ff3SSepherosa Ziehau if (hdr != txr->tx_next_clean) { 4638*189a0ff3SSepherosa Ziehau lwkt_serialize_enter(&txr->tx_serialize); 4639*189a0ff3SSepherosa Ziehau ix_txeof(txr, hdr); 4640*189a0ff3SSepherosa Ziehau if (!ifsq_is_empty(txr->tx_ifsq)) 4641*189a0ff3SSepherosa Ziehau ifsq_devstart(txr->tx_ifsq); 4642*189a0ff3SSepherosa Ziehau lwkt_serialize_exit(&txr->tx_serialize); 4643*189a0ff3SSepherosa Ziehau } 4644*189a0ff3SSepherosa Ziehau 4645*189a0ff3SSepherosa Ziehau IXGBE_WRITE_REG(&rxr->rx_sc->hw, rxr->rx_eims, rxr->rx_eims_val); 4646*189a0ff3SSepherosa Ziehau } 4647*189a0ff3SSepherosa Ziehau 4648*189a0ff3SSepherosa Ziehau static void 4649*189a0ff3SSepherosa Ziehau ix_intr_status(struct ix_softc *sc, uint32_t eicr) 4650*189a0ff3SSepherosa Ziehau { 4651*189a0ff3SSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 4652*189a0ff3SSepherosa Ziehau 4653*189a0ff3SSepherosa Ziehau /* Link status change */ 4654*189a0ff3SSepherosa Ziehau if (eicr & IXGBE_EICR_LSC) 4655*189a0ff3SSepherosa Ziehau ix_handle_link(sc); 4656*189a0ff3SSepherosa Ziehau 4657*189a0ff3SSepherosa Ziehau if (hw->mac.type != ixgbe_mac_82598EB) { 4658*189a0ff3SSepherosa Ziehau if (eicr & IXGBE_EICR_ECC) 4659*189a0ff3SSepherosa Ziehau if_printf(&sc->arpcom.ac_if, "ECC ERROR!! Reboot!!\n"); 4660*189a0ff3SSepherosa Ziehau else if (eicr & IXGBE_EICR_GPI_SDP1) 4661*189a0ff3SSepherosa Ziehau ix_handle_msf(sc); 4662*189a0ff3SSepherosa Ziehau else if (eicr & IXGBE_EICR_GPI_SDP2) 4663*189a0ff3SSepherosa Ziehau ix_handle_mod(sc); 4664*189a0ff3SSepherosa Ziehau } 4665*189a0ff3SSepherosa Ziehau 4666*189a0ff3SSepherosa Ziehau /* Check for fan failure */ 4667*189a0ff3SSepherosa Ziehau if (hw->device_id == IXGBE_DEV_ID_82598AT && 4668*189a0ff3SSepherosa Ziehau (eicr & IXGBE_EICR_GPI_SDP1)) 4669*189a0ff3SSepherosa Ziehau if_printf(&sc->arpcom.ac_if, "FAN FAILURE!! Replace!!\n"); 4670*189a0ff3SSepherosa Ziehau 4671*189a0ff3SSepherosa Ziehau /* Check for over temp condition */ 4672*189a0ff3SSepherosa Ziehau if (hw->mac.type == ixgbe_mac_X540 && (eicr & IXGBE_EICR_TS)) { 4673*189a0ff3SSepherosa Ziehau if_printf(&sc->arpcom.ac_if, "OVER TEMP!! " 4674*189a0ff3SSepherosa Ziehau "PHY IS SHUT DOWN!! Reboot\n"); 4675*189a0ff3SSepherosa Ziehau } 4676*189a0ff3SSepherosa Ziehau } 4677*189a0ff3SSepherosa Ziehau 4678*189a0ff3SSepherosa Ziehau static void 4679*189a0ff3SSepherosa Ziehau ix_msix_status(void *xsc) 4680*189a0ff3SSepherosa Ziehau { 4681*189a0ff3SSepherosa Ziehau struct ix_softc *sc = xsc; 4682*189a0ff3SSepherosa Ziehau uint32_t eicr; 4683*189a0ff3SSepherosa Ziehau 4684*189a0ff3SSepherosa Ziehau ASSERT_SERIALIZED(&sc->main_serialize); 4685*189a0ff3SSepherosa Ziehau 4686*189a0ff3SSepherosa Ziehau eicr = IXGBE_READ_REG(&sc->hw, IXGBE_EICR); 4687*189a0ff3SSepherosa Ziehau ix_intr_status(sc, eicr); 4688*189a0ff3SSepherosa Ziehau 4689*189a0ff3SSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_EIMS, sc->intr_mask); 4690*189a0ff3SSepherosa Ziehau } 4691*189a0ff3SSepherosa Ziehau 4692*189a0ff3SSepherosa Ziehau static void 4693*189a0ff3SSepherosa Ziehau ix_setup_msix_eims(const struct ix_softc *sc, int x, 4694*189a0ff3SSepherosa Ziehau uint32_t *eims, uint32_t *eims_val) 4695*189a0ff3SSepherosa Ziehau { 4696*189a0ff3SSepherosa Ziehau if (x < 32) { 4697*189a0ff3SSepherosa Ziehau if (sc->hw.mac.type == ixgbe_mac_82598EB) { 4698*189a0ff3SSepherosa Ziehau KASSERT(x < IX_MAX_MSIX_82598, 4699*189a0ff3SSepherosa Ziehau ("%s: invalid vector %d for 82598", 4700*189a0ff3SSepherosa Ziehau device_get_nameunit(sc->dev), x)); 4701*189a0ff3SSepherosa Ziehau *eims = IXGBE_EIMS; 4702*189a0ff3SSepherosa Ziehau } else { 4703*189a0ff3SSepherosa Ziehau *eims = IXGBE_EIMS_EX(0); 4704*189a0ff3SSepherosa Ziehau } 4705*189a0ff3SSepherosa Ziehau *eims_val = 1 << x; 4706*189a0ff3SSepherosa Ziehau } else { 4707*189a0ff3SSepherosa Ziehau KASSERT(x < IX_MAX_MSIX, ("%s: invalid vector %d", 4708*189a0ff3SSepherosa Ziehau device_get_nameunit(sc->dev), x)); 4709*189a0ff3SSepherosa Ziehau KASSERT(sc->hw.mac.type != ixgbe_mac_82598EB, 4710*189a0ff3SSepherosa Ziehau ("%s: invalid vector %d for 82598", 4711*189a0ff3SSepherosa Ziehau device_get_nameunit(sc->dev), x)); 4712*189a0ff3SSepherosa Ziehau *eims = IXGBE_EIMS_EX(1); 4713*189a0ff3SSepherosa Ziehau *eims_val = 1 << (x - 32); 4714*189a0ff3SSepherosa Ziehau } 4715*189a0ff3SSepherosa Ziehau } 4716