179251f5eSSepherosa Ziehau /* 263d483cdSSepherosa Ziehau * Copyright (c) 2001-2014, 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 324a648aefSSepherosa Ziehau #include "opt_ifpoll.h" 3379251f5eSSepherosa Ziehau #include "opt_ix.h" 3479251f5eSSepherosa Ziehau 3579251f5eSSepherosa Ziehau #include <sys/param.h> 3679251f5eSSepherosa Ziehau #include <sys/bus.h> 3779251f5eSSepherosa Ziehau #include <sys/endian.h> 3879251f5eSSepherosa Ziehau #include <sys/interrupt.h> 3979251f5eSSepherosa Ziehau #include <sys/kernel.h> 4079251f5eSSepherosa Ziehau #include <sys/malloc.h> 4179251f5eSSepherosa Ziehau #include <sys/mbuf.h> 4279251f5eSSepherosa Ziehau #include <sys/proc.h> 4379251f5eSSepherosa Ziehau #include <sys/rman.h> 4479251f5eSSepherosa Ziehau #include <sys/serialize.h> 4579251f5eSSepherosa Ziehau #include <sys/serialize2.h> 4679251f5eSSepherosa Ziehau #include <sys/socket.h> 4779251f5eSSepherosa Ziehau #include <sys/sockio.h> 4879251f5eSSepherosa Ziehau #include <sys/sysctl.h> 4979251f5eSSepherosa Ziehau #include <sys/systm.h> 508d0afa86SSepherosa Ziehau #include <sys/taskqueue.h> 5179251f5eSSepherosa Ziehau 5279251f5eSSepherosa Ziehau #include <net/bpf.h> 5379251f5eSSepherosa Ziehau #include <net/ethernet.h> 5479251f5eSSepherosa Ziehau #include <net/if.h> 5579251f5eSSepherosa Ziehau #include <net/if_arp.h> 5679251f5eSSepherosa Ziehau #include <net/if_dl.h> 5779251f5eSSepherosa Ziehau #include <net/if_media.h> 5879251f5eSSepherosa Ziehau #include <net/ifq_var.h> 59afc5d5f3SSepherosa Ziehau #include <net/if_ringmap.h> 6079251f5eSSepherosa Ziehau #include <net/toeplitz.h> 6179251f5eSSepherosa Ziehau #include <net/toeplitz2.h> 6279251f5eSSepherosa Ziehau #include <net/vlan/if_vlan_var.h> 6379251f5eSSepherosa Ziehau #include <net/vlan/if_vlan_ether.h> 6479251f5eSSepherosa Ziehau #include <net/if_poll.h> 6579251f5eSSepherosa Ziehau 6679251f5eSSepherosa Ziehau #include <netinet/in_systm.h> 6779251f5eSSepherosa Ziehau #include <netinet/in.h> 6879251f5eSSepherosa Ziehau #include <netinet/ip.h> 6979251f5eSSepherosa Ziehau 7079251f5eSSepherosa Ziehau #include <bus/pci/pcivar.h> 7179251f5eSSepherosa Ziehau #include <bus/pci/pcireg.h> 7279251f5eSSepherosa Ziehau 7379251f5eSSepherosa Ziehau #include <dev/netif/ix/ixgbe_api.h> 7479251f5eSSepherosa Ziehau #include <dev/netif/ix/if_ix.h> 7579251f5eSSepherosa Ziehau 7663d483cdSSepherosa Ziehau #define IX_IFM_DEFAULT (IFM_ETHER | IFM_AUTO) 7763d483cdSSepherosa Ziehau 7879251f5eSSepherosa Ziehau #ifdef IX_RSS_DEBUG 7979251f5eSSepherosa Ziehau #define IX_RSS_DPRINTF(sc, lvl, fmt, ...) \ 8079251f5eSSepherosa Ziehau do { \ 8179251f5eSSepherosa Ziehau if (sc->rss_debug >= lvl) \ 8279251f5eSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, fmt, __VA_ARGS__); \ 8379251f5eSSepherosa Ziehau } while (0) 8479251f5eSSepherosa Ziehau #else /* !IX_RSS_DEBUG */ 8579251f5eSSepherosa Ziehau #define IX_RSS_DPRINTF(sc, lvl, fmt, ...) ((void)0) 8679251f5eSSepherosa Ziehau #endif /* IX_RSS_DEBUG */ 8779251f5eSSepherosa Ziehau 8879251f5eSSepherosa Ziehau #define IX_NAME "Intel(R) PRO/10GbE " 8979251f5eSSepherosa Ziehau #define IX_DEVICE(id) \ 9063d483cdSSepherosa Ziehau { IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_##id, IX_NAME #id } 9179251f5eSSepherosa Ziehau #define IX_DEVICE_NULL { 0, 0, NULL } 9279251f5eSSepherosa Ziehau 9379251f5eSSepherosa Ziehau static struct ix_device { 9479251f5eSSepherosa Ziehau uint16_t vid; 9579251f5eSSepherosa Ziehau uint16_t did; 9679251f5eSSepherosa Ziehau const char *desc; 9779251f5eSSepherosa Ziehau } ix_devices[] = { 9879251f5eSSepherosa Ziehau IX_DEVICE(82598AF_DUAL_PORT), 9979251f5eSSepherosa Ziehau IX_DEVICE(82598AF_SINGLE_PORT), 10079251f5eSSepherosa Ziehau IX_DEVICE(82598EB_CX4), 10179251f5eSSepherosa Ziehau IX_DEVICE(82598AT), 10279251f5eSSepherosa Ziehau IX_DEVICE(82598AT2), 10379251f5eSSepherosa Ziehau IX_DEVICE(82598), 10479251f5eSSepherosa Ziehau IX_DEVICE(82598_DA_DUAL_PORT), 10579251f5eSSepherosa Ziehau IX_DEVICE(82598_CX4_DUAL_PORT), 10679251f5eSSepherosa Ziehau IX_DEVICE(82598EB_XF_LR), 10779251f5eSSepherosa Ziehau IX_DEVICE(82598_SR_DUAL_PORT_EM), 10879251f5eSSepherosa Ziehau IX_DEVICE(82598EB_SFP_LOM), 10979251f5eSSepherosa Ziehau IX_DEVICE(82599_KX4), 11079251f5eSSepherosa Ziehau IX_DEVICE(82599_KX4_MEZZ), 11179251f5eSSepherosa Ziehau IX_DEVICE(82599_SFP), 11279251f5eSSepherosa Ziehau IX_DEVICE(82599_XAUI_LOM), 11379251f5eSSepherosa Ziehau IX_DEVICE(82599_CX4), 11479251f5eSSepherosa Ziehau IX_DEVICE(82599_T3_LOM), 11579251f5eSSepherosa Ziehau IX_DEVICE(82599_COMBO_BACKPLANE), 11679251f5eSSepherosa Ziehau IX_DEVICE(82599_BACKPLANE_FCOE), 11779251f5eSSepherosa Ziehau IX_DEVICE(82599_SFP_SF2), 11879251f5eSSepherosa Ziehau IX_DEVICE(82599_SFP_FCOE), 11979251f5eSSepherosa Ziehau IX_DEVICE(82599EN_SFP), 12079251f5eSSepherosa Ziehau IX_DEVICE(82599_SFP_SF_QP), 12163d483cdSSepherosa Ziehau IX_DEVICE(82599_QSFP_SF_QP), 12279251f5eSSepherosa Ziehau IX_DEVICE(X540T), 12363d483cdSSepherosa Ziehau IX_DEVICE(X540T1), 12463d483cdSSepherosa Ziehau IX_DEVICE(X550T), 12563d483cdSSepherosa Ziehau IX_DEVICE(X550EM_X_KR), 12663d483cdSSepherosa Ziehau IX_DEVICE(X550EM_X_KX4), 12763d483cdSSepherosa Ziehau IX_DEVICE(X550EM_X_10G_T), 12879251f5eSSepherosa Ziehau 12979251f5eSSepherosa Ziehau /* required last entry */ 13079251f5eSSepherosa Ziehau IX_DEVICE_NULL 13179251f5eSSepherosa Ziehau }; 13279251f5eSSepherosa Ziehau 13379251f5eSSepherosa Ziehau static int ix_probe(device_t); 13479251f5eSSepherosa Ziehau static int ix_attach(device_t); 13579251f5eSSepherosa Ziehau static int ix_detach(device_t); 13679251f5eSSepherosa Ziehau static int ix_shutdown(device_t); 13779251f5eSSepherosa Ziehau 13879251f5eSSepherosa Ziehau static void ix_serialize(struct ifnet *, enum ifnet_serialize); 13979251f5eSSepherosa Ziehau static void ix_deserialize(struct ifnet *, enum ifnet_serialize); 14079251f5eSSepherosa Ziehau static int ix_tryserialize(struct ifnet *, enum ifnet_serialize); 14179251f5eSSepherosa Ziehau #ifdef INVARIANTS 14279251f5eSSepherosa Ziehau static void ix_serialize_assert(struct ifnet *, enum ifnet_serialize, 14379251f5eSSepherosa Ziehau boolean_t); 14479251f5eSSepherosa Ziehau #endif 14579251f5eSSepherosa Ziehau static void ix_start(struct ifnet *, struct ifaltq_subque *); 14679251f5eSSepherosa Ziehau static void ix_watchdog(struct ifaltq_subque *); 14779251f5eSSepherosa Ziehau static int ix_ioctl(struct ifnet *, u_long, caddr_t, struct ucred *); 14879251f5eSSepherosa Ziehau static void ix_init(void *); 14979251f5eSSepherosa Ziehau static void ix_stop(struct ix_softc *); 15079251f5eSSepherosa Ziehau static void ix_media_status(struct ifnet *, struct ifmediareq *); 15179251f5eSSepherosa Ziehau static int ix_media_change(struct ifnet *); 15279251f5eSSepherosa Ziehau static void ix_timer(void *); 1534a648aefSSepherosa Ziehau #ifdef IFPOLL_ENABLE 1544a648aefSSepherosa Ziehau static void ix_npoll(struct ifnet *, struct ifpoll_info *); 1554a648aefSSepherosa Ziehau static void ix_npoll_rx(struct ifnet *, void *, int); 1568d0afa86SSepherosa Ziehau static void ix_npoll_rx_direct(struct ifnet *, void *, int); 1574a648aefSSepherosa Ziehau static void ix_npoll_tx(struct ifnet *, void *, int); 1584a648aefSSepherosa Ziehau static void ix_npoll_status(struct ifnet *); 1594a648aefSSepherosa Ziehau #endif 16079251f5eSSepherosa Ziehau 16179251f5eSSepherosa Ziehau static void ix_add_sysctl(struct ix_softc *); 162189a0ff3SSepherosa Ziehau static void ix_add_intr_rate_sysctl(struct ix_softc *, int, 163189a0ff3SSepherosa Ziehau const char *, int (*)(SYSCTL_HANDLER_ARGS), const char *); 16479251f5eSSepherosa Ziehau static int ix_sysctl_tx_wreg_nsegs(SYSCTL_HANDLER_ARGS); 16582db96e9SSepherosa Ziehau static int ix_sysctl_tx_nmbuf(SYSCTL_HANDLER_ARGS); 16679251f5eSSepherosa Ziehau static int ix_sysctl_rx_wreg_nsegs(SYSCTL_HANDLER_ARGS); 16779251f5eSSepherosa Ziehau static int ix_sysctl_txd(SYSCTL_HANDLER_ARGS); 16879251f5eSSepherosa Ziehau static int ix_sysctl_rxd(SYSCTL_HANDLER_ARGS); 16979251f5eSSepherosa Ziehau static int ix_sysctl_tx_intr_nsegs(SYSCTL_HANDLER_ARGS); 170189a0ff3SSepherosa Ziehau static int ix_sysctl_intr_rate(SYSCTL_HANDLER_ARGS, int); 171189a0ff3SSepherosa Ziehau static int ix_sysctl_rxtx_intr_rate(SYSCTL_HANDLER_ARGS); 172189a0ff3SSepherosa Ziehau static int ix_sysctl_rx_intr_rate(SYSCTL_HANDLER_ARGS); 173189a0ff3SSepherosa Ziehau static int ix_sysctl_tx_intr_rate(SYSCTL_HANDLER_ARGS); 174189a0ff3SSepherosa Ziehau static int ix_sysctl_sts_intr_rate(SYSCTL_HANDLER_ARGS); 17579251f5eSSepherosa Ziehau #if 0 17679251f5eSSepherosa Ziehau static void ix_add_hw_stats(struct ix_softc *); 17779251f5eSSepherosa Ziehau #endif 17879251f5eSSepherosa Ziehau 1798d0afa86SSepherosa Ziehau static void ix_watchdog_reset(struct ix_softc *); 1808d0afa86SSepherosa Ziehau static void ix_watchdog_task(void *, int); 1818d0afa86SSepherosa Ziehau static void ix_sync_netisr(struct ix_softc *, int); 18279251f5eSSepherosa Ziehau static void ix_slot_info(struct ix_softc *); 18379251f5eSSepherosa Ziehau static int ix_alloc_rings(struct ix_softc *); 18479251f5eSSepherosa Ziehau static void ix_free_rings(struct ix_softc *); 18579251f5eSSepherosa Ziehau static void ix_setup_ifp(struct ix_softc *); 18679251f5eSSepherosa Ziehau static void ix_setup_serialize(struct ix_softc *); 18779251f5eSSepherosa Ziehau static void ix_set_ring_inuse(struct ix_softc *, boolean_t); 18879251f5eSSepherosa Ziehau static void ix_set_timer_cpuid(struct ix_softc *, boolean_t); 18979251f5eSSepherosa Ziehau static void ix_update_stats(struct ix_softc *); 19079251f5eSSepherosa Ziehau 19179251f5eSSepherosa Ziehau static void ix_set_promisc(struct ix_softc *); 19279251f5eSSepherosa Ziehau static void ix_set_multi(struct ix_softc *); 19379251f5eSSepherosa Ziehau static void ix_set_vlan(struct ix_softc *); 19479251f5eSSepherosa Ziehau static uint8_t *ix_mc_array_itr(struct ixgbe_hw *, uint8_t **, uint32_t *); 195060fa21cSSepherosa Ziehau static enum ixgbe_fc_mode ix_ifmedia2fc(int); 196060fa21cSSepherosa Ziehau static const char *ix_ifmedia2str(int); 197060fa21cSSepherosa Ziehau static const char *ix_fc2str(enum ixgbe_fc_mode); 19879251f5eSSepherosa Ziehau 1993c37d13bSSepherosa Ziehau static void ix_get_txring_cnt(const struct ix_softc *, int *, int *); 20079251f5eSSepherosa Ziehau static int ix_get_txring_inuse(const struct ix_softc *, boolean_t); 20179251f5eSSepherosa Ziehau static void ix_init_tx_ring(struct ix_tx_ring *); 20279251f5eSSepherosa Ziehau static void ix_free_tx_ring(struct ix_tx_ring *); 20379251f5eSSepherosa Ziehau static int ix_create_tx_ring(struct ix_tx_ring *); 20479251f5eSSepherosa Ziehau static void ix_destroy_tx_ring(struct ix_tx_ring *, int); 20579251f5eSSepherosa Ziehau static void ix_init_tx_unit(struct ix_softc *); 20679251f5eSSepherosa Ziehau static int ix_encap(struct ix_tx_ring *, struct mbuf **, 20779251f5eSSepherosa Ziehau uint16_t *, int *); 20879251f5eSSepherosa Ziehau static int ix_tx_ctx_setup(struct ix_tx_ring *, 20979251f5eSSepherosa Ziehau const struct mbuf *, uint32_t *, uint32_t *); 21079251f5eSSepherosa Ziehau static int ix_tso_ctx_setup(struct ix_tx_ring *, 21179251f5eSSepherosa Ziehau const struct mbuf *, uint32_t *, uint32_t *); 212189a0ff3SSepherosa Ziehau static void ix_txeof(struct ix_tx_ring *, int); 21382db96e9SSepherosa Ziehau static void ix_txgc(struct ix_tx_ring *); 21482db96e9SSepherosa Ziehau static void ix_txgc_timer(void *); 21579251f5eSSepherosa Ziehau 2163c37d13bSSepherosa Ziehau static void ix_get_rxring_cnt(const struct ix_softc *, int *, int *); 21779251f5eSSepherosa Ziehau static int ix_get_rxring_inuse(const struct ix_softc *, boolean_t); 21879251f5eSSepherosa Ziehau static int ix_init_rx_ring(struct ix_rx_ring *); 21979251f5eSSepherosa Ziehau static void ix_free_rx_ring(struct ix_rx_ring *); 22079251f5eSSepherosa Ziehau static int ix_create_rx_ring(struct ix_rx_ring *); 22179251f5eSSepherosa Ziehau static void ix_destroy_rx_ring(struct ix_rx_ring *, int); 2223c37d13bSSepherosa Ziehau static void ix_init_rx_unit(struct ix_softc *, boolean_t); 22379251f5eSSepherosa Ziehau #if 0 22479251f5eSSepherosa Ziehau static void ix_setup_hw_rsc(struct ix_rx_ring *); 22579251f5eSSepherosa Ziehau #endif 22679251f5eSSepherosa Ziehau static int ix_newbuf(struct ix_rx_ring *, int, boolean_t); 2274a648aefSSepherosa Ziehau static void ix_rxeof(struct ix_rx_ring *, int); 22879251f5eSSepherosa Ziehau static void ix_rx_discard(struct ix_rx_ring *, int, boolean_t); 22979251f5eSSepherosa Ziehau static void ix_enable_rx_drop(struct ix_softc *); 23079251f5eSSepherosa Ziehau static void ix_disable_rx_drop(struct ix_softc *); 23179251f5eSSepherosa Ziehau 232189a0ff3SSepherosa Ziehau static void ix_alloc_msix(struct ix_softc *); 233189a0ff3SSepherosa Ziehau static void ix_free_msix(struct ix_softc *, boolean_t); 234189a0ff3SSepherosa Ziehau static void ix_setup_msix_eims(const struct ix_softc *, int, 235189a0ff3SSepherosa Ziehau uint32_t *, uint32_t *); 23679251f5eSSepherosa Ziehau static int ix_alloc_intr(struct ix_softc *); 23779251f5eSSepherosa Ziehau static void ix_free_intr(struct ix_softc *); 23879251f5eSSepherosa Ziehau static int ix_setup_intr(struct ix_softc *); 23979251f5eSSepherosa Ziehau static void ix_teardown_intr(struct ix_softc *, int); 24079251f5eSSepherosa Ziehau static void ix_enable_intr(struct ix_softc *); 24179251f5eSSepherosa Ziehau static void ix_disable_intr(struct ix_softc *); 24279251f5eSSepherosa Ziehau static void ix_set_ivar(struct ix_softc *, uint8_t, uint8_t, int8_t); 24379251f5eSSepherosa Ziehau static void ix_set_eitr(struct ix_softc *, int, int); 244189a0ff3SSepherosa Ziehau static void ix_intr_status(struct ix_softc *, uint32_t); 24579251f5eSSepherosa Ziehau static void ix_intr(void *); 246189a0ff3SSepherosa Ziehau static void ix_msix_rxtx(void *); 247189a0ff3SSepherosa Ziehau static void ix_msix_rx(void *); 248189a0ff3SSepherosa Ziehau static void ix_msix_tx(void *); 249189a0ff3SSepherosa Ziehau static void ix_msix_status(void *); 25079251f5eSSepherosa Ziehau 25179251f5eSSepherosa Ziehau static void ix_config_link(struct ix_softc *); 25279251f5eSSepherosa Ziehau static boolean_t ix_sfp_probe(struct ix_softc *); 25379251f5eSSepherosa Ziehau static boolean_t ix_is_sfp(const struct ixgbe_hw *); 25479251f5eSSepherosa Ziehau static void ix_update_link_status(struct ix_softc *); 25579251f5eSSepherosa Ziehau static void ix_handle_link(struct ix_softc *); 25679251f5eSSepherosa Ziehau static void ix_handle_mod(struct ix_softc *); 25779251f5eSSepherosa Ziehau static void ix_handle_msf(struct ix_softc *); 25863d483cdSSepherosa Ziehau static void ix_handle_phy(struct ix_softc *); 25963d483cdSSepherosa Ziehau static int ix_powerdown(struct ix_softc *); 26063d483cdSSepherosa Ziehau static void ix_config_flowctrl(struct ix_softc *); 26163d483cdSSepherosa Ziehau static void ix_config_dmac(struct ix_softc *); 26263d483cdSSepherosa Ziehau static void ix_init_media(struct ix_softc *); 26379251f5eSSepherosa Ziehau 26463d483cdSSepherosa Ziehau /* XXX Missing shared code prototype */ 26579251f5eSSepherosa Ziehau extern void ixgbe_stop_mac_link_on_d3_82599(struct ixgbe_hw *); 26679251f5eSSepherosa Ziehau 26779251f5eSSepherosa Ziehau static device_method_t ix_methods[] = { 26879251f5eSSepherosa Ziehau /* Device interface */ 26979251f5eSSepherosa Ziehau DEVMETHOD(device_probe, ix_probe), 27079251f5eSSepherosa Ziehau DEVMETHOD(device_attach, ix_attach), 27179251f5eSSepherosa Ziehau DEVMETHOD(device_detach, ix_detach), 27279251f5eSSepherosa Ziehau DEVMETHOD(device_shutdown, ix_shutdown), 27379251f5eSSepherosa Ziehau DEVMETHOD_END 27479251f5eSSepherosa Ziehau }; 27579251f5eSSepherosa Ziehau 27679251f5eSSepherosa Ziehau static driver_t ix_driver = { 27779251f5eSSepherosa Ziehau "ix", 27879251f5eSSepherosa Ziehau ix_methods, 27979251f5eSSepherosa Ziehau sizeof(struct ix_softc) 28079251f5eSSepherosa Ziehau }; 28179251f5eSSepherosa Ziehau 28279251f5eSSepherosa Ziehau static devclass_t ix_devclass; 28379251f5eSSepherosa Ziehau 28479251f5eSSepherosa Ziehau DECLARE_DUMMY_MODULE(if_ix); 28579251f5eSSepherosa Ziehau DRIVER_MODULE(if_ix, pci, ix_driver, ix_devclass, NULL, NULL); 28679251f5eSSepherosa Ziehau 28779251f5eSSepherosa Ziehau static int ix_msi_enable = 1; 288189a0ff3SSepherosa Ziehau static int ix_msix_enable = 1; 28979251f5eSSepherosa Ziehau static int ix_rxr = 0; 290189a0ff3SSepherosa Ziehau static int ix_txr = 0; 29179251f5eSSepherosa Ziehau static int ix_txd = IX_PERF_TXD; 29279251f5eSSepherosa Ziehau static int ix_rxd = IX_PERF_RXD; 29379251f5eSSepherosa Ziehau static int ix_unsupported_sfp = 0; 2948d0afa86SSepherosa Ziehau static int ix_direct_input = 1; 29579251f5eSSepherosa Ziehau 296c2c5f4f3SSepherosa Ziehau static char ix_flowctrl[IFM_ETH_FC_STRLEN] = IFM_ETH_FC_NONE; 297060fa21cSSepherosa Ziehau 29879251f5eSSepherosa Ziehau TUNABLE_INT("hw.ix.msi.enable", &ix_msi_enable); 299189a0ff3SSepherosa Ziehau TUNABLE_INT("hw.ix.msix.enable", &ix_msix_enable); 30079251f5eSSepherosa Ziehau TUNABLE_INT("hw.ix.rxr", &ix_rxr); 301189a0ff3SSepherosa Ziehau TUNABLE_INT("hw.ix.txr", &ix_txr); 30279251f5eSSepherosa Ziehau TUNABLE_INT("hw.ix.txd", &ix_txd); 30379251f5eSSepherosa Ziehau TUNABLE_INT("hw.ix.rxd", &ix_rxd); 30479251f5eSSepherosa Ziehau TUNABLE_INT("hw.ix.unsupported_sfp", &ix_unsupported_sfp); 305060fa21cSSepherosa Ziehau TUNABLE_STR("hw.ix.flow_ctrl", ix_flowctrl, sizeof(ix_flowctrl)); 3068d0afa86SSepherosa Ziehau TUNABLE_INT("hw.ix.direct_input", &ix_direct_input); 30779251f5eSSepherosa Ziehau 30879251f5eSSepherosa Ziehau /* 30979251f5eSSepherosa Ziehau * Smart speed setting, default to on. This only works 31079251f5eSSepherosa Ziehau * as a compile option right now as its during attach, 31179251f5eSSepherosa Ziehau * set this to 'ixgbe_smart_speed_off' to disable. 31279251f5eSSepherosa Ziehau */ 31379251f5eSSepherosa Ziehau static const enum ixgbe_smart_speed ix_smart_speed = 31479251f5eSSepherosa Ziehau ixgbe_smart_speed_on; 31579251f5eSSepherosa Ziehau 31682db96e9SSepherosa Ziehau static __inline void 31782db96e9SSepherosa Ziehau ix_try_txgc(struct ix_tx_ring *txr, int8_t dec) 31882db96e9SSepherosa Ziehau { 31982db96e9SSepherosa Ziehau 32082db96e9SSepherosa Ziehau if (txr->tx_running > 0) { 32182db96e9SSepherosa Ziehau txr->tx_running -= dec; 32282db96e9SSepherosa Ziehau if (txr->tx_running <= 0 && txr->tx_nmbuf && 32382db96e9SSepherosa Ziehau txr->tx_avail < txr->tx_ndesc && 32482db96e9SSepherosa Ziehau txr->tx_avail + txr->tx_intr_nsegs > txr->tx_ndesc) 32582db96e9SSepherosa Ziehau ix_txgc(txr); 32682db96e9SSepherosa Ziehau } 32782db96e9SSepherosa Ziehau } 32882db96e9SSepherosa Ziehau 32982db96e9SSepherosa Ziehau static void 33082db96e9SSepherosa Ziehau ix_txgc_timer(void *xtxr) 33182db96e9SSepherosa Ziehau { 33282db96e9SSepherosa Ziehau struct ix_tx_ring *txr = xtxr; 33382db96e9SSepherosa Ziehau struct ifnet *ifp = &txr->tx_sc->arpcom.ac_if; 33482db96e9SSepherosa Ziehau 33582db96e9SSepherosa Ziehau if ((ifp->if_flags & (IFF_RUNNING | IFF_UP | IFF_NPOLLING)) != 33682db96e9SSepherosa Ziehau (IFF_RUNNING | IFF_UP)) 33782db96e9SSepherosa Ziehau return; 33882db96e9SSepherosa Ziehau 33982db96e9SSepherosa Ziehau if (!lwkt_serialize_try(&txr->tx_serialize)) 34082db96e9SSepherosa Ziehau goto done; 34182db96e9SSepherosa Ziehau 34282db96e9SSepherosa Ziehau if ((ifp->if_flags & (IFF_RUNNING | IFF_UP | IFF_NPOLLING)) != 34382db96e9SSepherosa Ziehau (IFF_RUNNING | IFF_UP)) { 34482db96e9SSepherosa Ziehau lwkt_serialize_exit(&txr->tx_serialize); 34582db96e9SSepherosa Ziehau return; 34682db96e9SSepherosa Ziehau } 34782db96e9SSepherosa Ziehau ix_try_txgc(txr, IX_TX_RUNNING_DEC); 34882db96e9SSepherosa Ziehau 34982db96e9SSepherosa Ziehau lwkt_serialize_exit(&txr->tx_serialize); 35082db96e9SSepherosa Ziehau done: 35182db96e9SSepherosa Ziehau callout_reset(&txr->tx_gc_timer, 1, ix_txgc_timer, txr); 35282db96e9SSepherosa Ziehau } 35382db96e9SSepherosa Ziehau 35482db96e9SSepherosa Ziehau static __inline void 35582db96e9SSepherosa Ziehau ix_tx_intr(struct ix_tx_ring *txr, int hdr) 35682db96e9SSepherosa Ziehau { 35782db96e9SSepherosa Ziehau 35882db96e9SSepherosa Ziehau ix_txeof(txr, hdr); 35982db96e9SSepherosa Ziehau if (!ifsq_is_empty(txr->tx_ifsq)) 36082db96e9SSepherosa Ziehau ifsq_devstart(txr->tx_ifsq); 36182db96e9SSepherosa Ziehau } 36282db96e9SSepherosa Ziehau 36382db96e9SSepherosa Ziehau static __inline void 36482db96e9SSepherosa Ziehau ix_free_txbuf(struct ix_tx_ring *txr, struct ix_tx_buf *txbuf) 36582db96e9SSepherosa Ziehau { 36682db96e9SSepherosa Ziehau 36782db96e9SSepherosa Ziehau KKASSERT(txbuf->m_head != NULL); 36882db96e9SSepherosa Ziehau KKASSERT(txr->tx_nmbuf > 0); 36982db96e9SSepherosa Ziehau txr->tx_nmbuf--; 37082db96e9SSepherosa Ziehau 37182db96e9SSepherosa Ziehau bus_dmamap_unload(txr->tx_tag, txbuf->map); 37282db96e9SSepherosa Ziehau m_freem(txbuf->m_head); 37382db96e9SSepherosa Ziehau txbuf->m_head = NULL; 37482db96e9SSepherosa Ziehau } 37582db96e9SSepherosa Ziehau 37679251f5eSSepherosa Ziehau static int 37779251f5eSSepherosa Ziehau ix_probe(device_t dev) 37879251f5eSSepherosa Ziehau { 37979251f5eSSepherosa Ziehau const struct ix_device *d; 38079251f5eSSepherosa Ziehau uint16_t vid, did; 38179251f5eSSepherosa Ziehau 38279251f5eSSepherosa Ziehau vid = pci_get_vendor(dev); 38379251f5eSSepherosa Ziehau did = pci_get_device(dev); 38479251f5eSSepherosa Ziehau 38579251f5eSSepherosa Ziehau for (d = ix_devices; d->desc != NULL; ++d) { 38679251f5eSSepherosa Ziehau if (vid == d->vid && did == d->did) { 38779251f5eSSepherosa Ziehau device_set_desc(dev, d->desc); 38879251f5eSSepherosa Ziehau return 0; 38979251f5eSSepherosa Ziehau } 39079251f5eSSepherosa Ziehau } 39179251f5eSSepherosa Ziehau return ENXIO; 39279251f5eSSepherosa Ziehau } 39379251f5eSSepherosa Ziehau 3943c37d13bSSepherosa Ziehau static void 3953c37d13bSSepherosa Ziehau ix_get_rxring_cnt(const struct ix_softc *sc, int *ring_cnt, int *ring_cntmax) 3963c37d13bSSepherosa Ziehau { 3973c37d13bSSepherosa Ziehau 3983c37d13bSSepherosa Ziehau switch (sc->hw.mac.type) { 3993c37d13bSSepherosa Ziehau case ixgbe_mac_X550: 4003c37d13bSSepherosa Ziehau case ixgbe_mac_X550EM_x: 4013c37d13bSSepherosa Ziehau case ixgbe_mac_X550EM_a: 4023c37d13bSSepherosa Ziehau *ring_cntmax = IX_MAX_RXRING_X550; 4033c37d13bSSepherosa Ziehau break; 4043c37d13bSSepherosa Ziehau 4053c37d13bSSepherosa Ziehau default: 4063c37d13bSSepherosa Ziehau *ring_cntmax = IX_MAX_RXRING; 4073c37d13bSSepherosa Ziehau break; 4083c37d13bSSepherosa Ziehau } 4093c37d13bSSepherosa Ziehau *ring_cnt = device_getenv_int(sc->dev, "rxr", ix_rxr); 4103c37d13bSSepherosa Ziehau } 4113c37d13bSSepherosa Ziehau 4123c37d13bSSepherosa Ziehau static void 4133c37d13bSSepherosa Ziehau ix_get_txring_cnt(const struct ix_softc *sc, int *ring_cnt, int *ring_cntmax) 4143c37d13bSSepherosa Ziehau { 4153c37d13bSSepherosa Ziehau 4163c37d13bSSepherosa Ziehau switch (sc->hw.mac.type) { 4173c37d13bSSepherosa Ziehau case ixgbe_mac_82598EB: 4183c37d13bSSepherosa Ziehau *ring_cntmax = IX_MAX_TXRING_82598; 4193c37d13bSSepherosa Ziehau break; 4203c37d13bSSepherosa Ziehau 4213c37d13bSSepherosa Ziehau case ixgbe_mac_82599EB: 4223c37d13bSSepherosa Ziehau *ring_cntmax = IX_MAX_TXRING_82599; 4233c37d13bSSepherosa Ziehau break; 4243c37d13bSSepherosa Ziehau 4253c37d13bSSepherosa Ziehau case ixgbe_mac_X540: 4263c37d13bSSepherosa Ziehau *ring_cntmax = IX_MAX_TXRING_X540; 4273c37d13bSSepherosa Ziehau break; 4283c37d13bSSepherosa Ziehau 4293c37d13bSSepherosa Ziehau case ixgbe_mac_X550: 4303c37d13bSSepherosa Ziehau case ixgbe_mac_X550EM_x: 4313c37d13bSSepherosa Ziehau case ixgbe_mac_X550EM_a: 4323c37d13bSSepherosa Ziehau *ring_cntmax = IX_MAX_TXRING_X550; 4333c37d13bSSepherosa Ziehau break; 4343c37d13bSSepherosa Ziehau 4353c37d13bSSepherosa Ziehau default: 4363c37d13bSSepherosa Ziehau *ring_cntmax = IX_MAX_TXRING; 4373c37d13bSSepherosa Ziehau break; 4383c37d13bSSepherosa Ziehau } 4393c37d13bSSepherosa Ziehau *ring_cnt = device_getenv_int(sc->dev, "txr", ix_txr); 4403c37d13bSSepherosa Ziehau } 4413c37d13bSSepherosa Ziehau 44279251f5eSSepherosa Ziehau static int 44379251f5eSSepherosa Ziehau ix_attach(device_t dev) 44479251f5eSSepherosa Ziehau { 44579251f5eSSepherosa Ziehau struct ix_softc *sc = device_get_softc(dev); 44679251f5eSSepherosa Ziehau struct ixgbe_hw *hw; 4473c37d13bSSepherosa Ziehau int error, ring_cnt, ring_cntmax; 44879251f5eSSepherosa Ziehau uint16_t csum; 44979251f5eSSepherosa Ziehau uint32_t ctrl_ext; 450060fa21cSSepherosa Ziehau char flowctrl[IFM_ETH_FC_STRLEN]; 45179251f5eSSepherosa Ziehau 45279251f5eSSepherosa Ziehau sc->dev = sc->osdep.dev = dev; 45379251f5eSSepherosa Ziehau hw = &sc->hw; 45479251f5eSSepherosa Ziehau 45579251f5eSSepherosa Ziehau if_initname(&sc->arpcom.ac_if, device_get_name(dev), 45679251f5eSSepherosa Ziehau device_get_unit(dev)); 457060fa21cSSepherosa Ziehau ifmedia_init(&sc->media, IFM_IMASK | IFM_ETH_FCMASK, 45879251f5eSSepherosa Ziehau ix_media_change, ix_media_status); 45979251f5eSSepherosa Ziehau 46079251f5eSSepherosa Ziehau /* Save frame size */ 46179251f5eSSepherosa Ziehau sc->max_frame_size = ETHERMTU + ETHER_HDR_LEN + ETHER_CRC_LEN; 46279251f5eSSepherosa Ziehau 4638d0afa86SSepherosa Ziehau sc->direct_input = ix_direct_input; 4648d0afa86SSepherosa Ziehau TASK_INIT(&sc->wdog_task, 0, ix_watchdog_task, sc); 4658d0afa86SSepherosa Ziehau 46679251f5eSSepherosa Ziehau callout_init_mp(&sc->timer); 46779251f5eSSepherosa Ziehau lwkt_serialize_init(&sc->main_serialize); 46879251f5eSSepherosa Ziehau 46979251f5eSSepherosa Ziehau /* 47079251f5eSSepherosa Ziehau * Save off the information about this board 47179251f5eSSepherosa Ziehau */ 47279251f5eSSepherosa Ziehau hw->vendor_id = pci_get_vendor(dev); 47379251f5eSSepherosa Ziehau hw->device_id = pci_get_device(dev); 47479251f5eSSepherosa Ziehau hw->revision_id = pci_read_config(dev, PCIR_REVID, 1); 47579251f5eSSepherosa Ziehau hw->subsystem_vendor_id = pci_read_config(dev, PCIR_SUBVEND_0, 2); 47679251f5eSSepherosa Ziehau hw->subsystem_device_id = pci_read_config(dev, PCIR_SUBDEV_0, 2); 47779251f5eSSepherosa Ziehau 47879251f5eSSepherosa Ziehau ixgbe_set_mac_type(hw); 47979251f5eSSepherosa Ziehau 48063d483cdSSepherosa Ziehau /* Pick up the 82599 */ 48179251f5eSSepherosa Ziehau if (hw->mac.type != ixgbe_mac_82598EB) 48279251f5eSSepherosa Ziehau hw->phy.smart_speed = ix_smart_speed; 48379251f5eSSepherosa Ziehau 48479251f5eSSepherosa Ziehau /* Enable bus mastering */ 48579251f5eSSepherosa Ziehau pci_enable_busmaster(dev); 48679251f5eSSepherosa Ziehau 48779251f5eSSepherosa Ziehau /* 48879251f5eSSepherosa Ziehau * Allocate IO memory 48979251f5eSSepherosa Ziehau */ 49079251f5eSSepherosa Ziehau sc->mem_rid = PCIR_BAR(0); 49179251f5eSSepherosa Ziehau sc->mem_res = bus_alloc_resource_any(dev, SYS_RES_MEMORY, 49279251f5eSSepherosa Ziehau &sc->mem_rid, RF_ACTIVE); 49379251f5eSSepherosa Ziehau if (sc->mem_res == NULL) { 49479251f5eSSepherosa Ziehau device_printf(dev, "Unable to allocate bus resource: memory\n"); 49579251f5eSSepherosa Ziehau error = ENXIO; 49679251f5eSSepherosa Ziehau goto failed; 49779251f5eSSepherosa Ziehau } 49879251f5eSSepherosa Ziehau 49979251f5eSSepherosa Ziehau sc->osdep.mem_bus_space_tag = rman_get_bustag(sc->mem_res); 50079251f5eSSepherosa Ziehau sc->osdep.mem_bus_space_handle = rman_get_bushandle(sc->mem_res); 50179251f5eSSepherosa Ziehau 50279251f5eSSepherosa Ziehau sc->hw.hw_addr = (uint8_t *)&sc->osdep.mem_bus_space_handle; 50379251f5eSSepherosa Ziehau sc->hw.back = &sc->osdep; 50479251f5eSSepherosa Ziehau 50579251f5eSSepherosa Ziehau /* 50679251f5eSSepherosa Ziehau * Configure total supported RX/TX ring count 50779251f5eSSepherosa Ziehau */ 5083c37d13bSSepherosa Ziehau ix_get_rxring_cnt(sc, &ring_cnt, &ring_cntmax); 5093c37d13bSSepherosa Ziehau sc->rx_rmap = if_ringmap_alloc(dev, ring_cnt, ring_cntmax); 5103c37d13bSSepherosa Ziehau ix_get_txring_cnt(sc, &ring_cnt, &ring_cntmax); 5113c37d13bSSepherosa Ziehau sc->tx_rmap = if_ringmap_alloc(dev, ring_cnt, ring_cntmax); 5123c37d13bSSepherosa Ziehau if_ringmap_match(dev, sc->rx_rmap, sc->tx_rmap); 513737c1c7cSSepherosa Ziehau 5143c37d13bSSepherosa Ziehau sc->rx_ring_cnt = if_ringmap_count(sc->rx_rmap); 51579251f5eSSepherosa Ziehau sc->rx_ring_inuse = sc->rx_ring_cnt; 5163c37d13bSSepherosa Ziehau sc->tx_ring_cnt = if_ringmap_count(sc->tx_rmap); 51779251f5eSSepherosa Ziehau sc->tx_ring_inuse = sc->tx_ring_cnt; 51879251f5eSSepherosa Ziehau 51979251f5eSSepherosa Ziehau /* Allocate TX/RX rings */ 52079251f5eSSepherosa Ziehau error = ix_alloc_rings(sc); 52179251f5eSSepherosa Ziehau if (error) 52279251f5eSSepherosa Ziehau goto failed; 52379251f5eSSepherosa Ziehau 52479251f5eSSepherosa Ziehau /* Allocate interrupt */ 52579251f5eSSepherosa Ziehau error = ix_alloc_intr(sc); 52679251f5eSSepherosa Ziehau if (error) 52779251f5eSSepherosa Ziehau goto failed; 52879251f5eSSepherosa Ziehau 52979251f5eSSepherosa Ziehau /* Setup serializes */ 53079251f5eSSepherosa Ziehau ix_setup_serialize(sc); 53179251f5eSSepherosa Ziehau 53279251f5eSSepherosa Ziehau /* Allocate multicast array memory. */ 53379251f5eSSepherosa Ziehau sc->mta = kmalloc(IXGBE_ETH_LENGTH_OF_ADDRESS * IX_MAX_MCASTADDR, 53479251f5eSSepherosa Ziehau M_DEVBUF, M_WAITOK); 53579251f5eSSepherosa Ziehau 53679251f5eSSepherosa Ziehau /* Initialize the shared code */ 53779251f5eSSepherosa Ziehau hw->allow_unsupported_sfp = ix_unsupported_sfp; 53879251f5eSSepherosa Ziehau error = ixgbe_init_shared_code(hw); 53979251f5eSSepherosa Ziehau if (error == IXGBE_ERR_SFP_NOT_PRESENT) { 54079251f5eSSepherosa Ziehau /* 54179251f5eSSepherosa Ziehau * No optics in this port; ask timer routine 54279251f5eSSepherosa Ziehau * to probe for later insertion. 54379251f5eSSepherosa Ziehau */ 54479251f5eSSepherosa Ziehau sc->sfp_probe = TRUE; 54579251f5eSSepherosa Ziehau error = 0; 54679251f5eSSepherosa Ziehau } else if (error == IXGBE_ERR_SFP_NOT_SUPPORTED) { 54779251f5eSSepherosa Ziehau device_printf(dev, "Unsupported SFP+ module detected!\n"); 54879251f5eSSepherosa Ziehau error = EIO; 54979251f5eSSepherosa Ziehau goto failed; 55079251f5eSSepherosa Ziehau } else if (error) { 55179251f5eSSepherosa Ziehau device_printf(dev, "Unable to initialize the shared code\n"); 55279251f5eSSepherosa Ziehau error = EIO; 55379251f5eSSepherosa Ziehau goto failed; 55479251f5eSSepherosa Ziehau } 55579251f5eSSepherosa Ziehau 55679251f5eSSepherosa Ziehau /* Make sure we have a good EEPROM before we read from it */ 55779251f5eSSepherosa Ziehau if (ixgbe_validate_eeprom_checksum(&sc->hw, &csum) < 0) { 55879251f5eSSepherosa Ziehau device_printf(dev, "The EEPROM Checksum Is Not Valid\n"); 55979251f5eSSepherosa Ziehau error = EIO; 56079251f5eSSepherosa Ziehau goto failed; 56179251f5eSSepherosa Ziehau } 56279251f5eSSepherosa Ziehau 56379251f5eSSepherosa Ziehau error = ixgbe_init_hw(hw); 56479251f5eSSepherosa Ziehau if (error == IXGBE_ERR_EEPROM_VERSION) { 56579251f5eSSepherosa Ziehau device_printf(dev, "Pre-production device detected\n"); 56679251f5eSSepherosa Ziehau } else if (error == IXGBE_ERR_SFP_NOT_SUPPORTED) { 56779251f5eSSepherosa Ziehau device_printf(dev, "Unsupported SFP+ Module\n"); 56879251f5eSSepherosa Ziehau error = EIO; 56979251f5eSSepherosa Ziehau goto failed; 57079251f5eSSepherosa Ziehau } else if (error == IXGBE_ERR_SFP_NOT_PRESENT) { 57179251f5eSSepherosa Ziehau device_printf(dev, "No SFP+ Module found\n"); 57279251f5eSSepherosa Ziehau } 57379251f5eSSepherosa Ziehau 57463d483cdSSepherosa Ziehau sc->ifm_media = IX_IFM_DEFAULT; 575060fa21cSSepherosa Ziehau /* Get default flow control settings */ 576060fa21cSSepherosa Ziehau device_getenv_string(dev, "flow_ctrl", flowctrl, sizeof(flowctrl), 577060fa21cSSepherosa Ziehau ix_flowctrl); 57863d483cdSSepherosa Ziehau sc->ifm_media |= ifmedia_str2ethfc(flowctrl); 57963d483cdSSepherosa Ziehau sc->advspeed = IXGBE_LINK_SPEED_UNKNOWN; 580060fa21cSSepherosa Ziehau 58179251f5eSSepherosa Ziehau /* Setup OS specific network interface */ 58279251f5eSSepherosa Ziehau ix_setup_ifp(sc); 58379251f5eSSepherosa Ziehau 58479251f5eSSepherosa Ziehau /* Add sysctl tree */ 58579251f5eSSepherosa Ziehau ix_add_sysctl(sc); 58679251f5eSSepherosa Ziehau 58779251f5eSSepherosa Ziehau error = ix_setup_intr(sc); 58879251f5eSSepherosa Ziehau if (error) { 58979251f5eSSepherosa Ziehau ether_ifdetach(&sc->arpcom.ac_if); 59079251f5eSSepherosa Ziehau goto failed; 59179251f5eSSepherosa Ziehau } 59279251f5eSSepherosa Ziehau 59379251f5eSSepherosa Ziehau /* Initialize statistics */ 59479251f5eSSepherosa Ziehau ix_update_stats(sc); 59579251f5eSSepherosa Ziehau 59663d483cdSSepherosa Ziehau /* Check PCIE slot type/speed/width */ 59779251f5eSSepherosa Ziehau ix_slot_info(sc); 59879251f5eSSepherosa Ziehau 59963d483cdSSepherosa Ziehau /* Save initial wake up filter configuration */ 60063d483cdSSepherosa Ziehau sc->wufc = IXGBE_READ_REG(hw, IXGBE_WUFC); 60163d483cdSSepherosa Ziehau 60279251f5eSSepherosa Ziehau /* Let hardware know driver is loaded */ 60379251f5eSSepherosa Ziehau ctrl_ext = IXGBE_READ_REG(hw, IXGBE_CTRL_EXT); 60479251f5eSSepherosa Ziehau ctrl_ext |= IXGBE_CTRL_EXT_DRV_LOAD; 60579251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_CTRL_EXT, ctrl_ext); 60679251f5eSSepherosa Ziehau 60779251f5eSSepherosa Ziehau return 0; 60879251f5eSSepherosa Ziehau failed: 60979251f5eSSepherosa Ziehau ix_detach(dev); 61079251f5eSSepherosa Ziehau return error; 61179251f5eSSepherosa Ziehau } 61279251f5eSSepherosa Ziehau 61379251f5eSSepherosa Ziehau static int 61479251f5eSSepherosa Ziehau ix_detach(device_t dev) 61579251f5eSSepherosa Ziehau { 61679251f5eSSepherosa Ziehau struct ix_softc *sc = device_get_softc(dev); 61779251f5eSSepherosa Ziehau 61879251f5eSSepherosa Ziehau if (device_is_attached(dev)) { 61979251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 62079251f5eSSepherosa Ziehau uint32_t ctrl_ext; 62179251f5eSSepherosa Ziehau 6228d0afa86SSepherosa Ziehau ix_sync_netisr(sc, IFF_UP); 6238d0afa86SSepherosa Ziehau taskqueue_drain(taskqueue_thread[0], &sc->wdog_task); 6248d0afa86SSepherosa Ziehau 62579251f5eSSepherosa Ziehau ifnet_serialize_all(ifp); 62679251f5eSSepherosa Ziehau 62763d483cdSSepherosa Ziehau ix_powerdown(sc); 62879251f5eSSepherosa Ziehau ix_teardown_intr(sc, sc->intr_cnt); 62979251f5eSSepherosa Ziehau 63079251f5eSSepherosa Ziehau ifnet_deserialize_all(ifp); 63179251f5eSSepherosa Ziehau 63279251f5eSSepherosa Ziehau callout_terminate(&sc->timer); 63379251f5eSSepherosa Ziehau ether_ifdetach(ifp); 63479251f5eSSepherosa Ziehau 63579251f5eSSepherosa Ziehau /* Let hardware know driver is unloading */ 63679251f5eSSepherosa Ziehau ctrl_ext = IXGBE_READ_REG(&sc->hw, IXGBE_CTRL_EXT); 63779251f5eSSepherosa Ziehau ctrl_ext &= ~IXGBE_CTRL_EXT_DRV_LOAD; 63879251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_CTRL_EXT, ctrl_ext); 63979251f5eSSepherosa Ziehau } 64079251f5eSSepherosa Ziehau 64179251f5eSSepherosa Ziehau ifmedia_removeall(&sc->media); 64279251f5eSSepherosa Ziehau bus_generic_detach(dev); 64379251f5eSSepherosa Ziehau 64479251f5eSSepherosa Ziehau ix_free_intr(sc); 64579251f5eSSepherosa Ziehau 646189a0ff3SSepherosa Ziehau if (sc->msix_mem_res != NULL) { 647189a0ff3SSepherosa Ziehau bus_release_resource(dev, SYS_RES_MEMORY, sc->msix_mem_rid, 648189a0ff3SSepherosa Ziehau sc->msix_mem_res); 649189a0ff3SSepherosa Ziehau } 65079251f5eSSepherosa Ziehau if (sc->mem_res != NULL) { 65179251f5eSSepherosa Ziehau bus_release_resource(dev, SYS_RES_MEMORY, sc->mem_rid, 65279251f5eSSepherosa Ziehau sc->mem_res); 65379251f5eSSepherosa Ziehau } 65479251f5eSSepherosa Ziehau 65579251f5eSSepherosa Ziehau ix_free_rings(sc); 65679251f5eSSepherosa Ziehau 65779251f5eSSepherosa Ziehau if (sc->mta != NULL) 65879251f5eSSepherosa Ziehau kfree(sc->mta, M_DEVBUF); 65979251f5eSSepherosa Ziehau if (sc->serializes != NULL) 66079251f5eSSepherosa Ziehau kfree(sc->serializes, M_DEVBUF); 66179251f5eSSepherosa Ziehau 6623c37d13bSSepherosa Ziehau if (sc->rx_rmap != NULL) 6633c37d13bSSepherosa Ziehau if_ringmap_free(sc->rx_rmap); 6643c37d13bSSepherosa Ziehau if (sc->rx_rmap_intr != NULL) 6653c37d13bSSepherosa Ziehau if_ringmap_free(sc->rx_rmap_intr); 6663c37d13bSSepherosa Ziehau if (sc->tx_rmap != NULL) 6673c37d13bSSepherosa Ziehau if_ringmap_free(sc->tx_rmap); 6683c37d13bSSepherosa Ziehau if (sc->tx_rmap_intr != NULL) 6693c37d13bSSepherosa Ziehau if_ringmap_free(sc->tx_rmap_intr); 6703c37d13bSSepherosa Ziehau 67179251f5eSSepherosa Ziehau return 0; 67279251f5eSSepherosa Ziehau } 67379251f5eSSepherosa Ziehau 67479251f5eSSepherosa Ziehau static int 67579251f5eSSepherosa Ziehau ix_shutdown(device_t dev) 67679251f5eSSepherosa Ziehau { 67779251f5eSSepherosa Ziehau struct ix_softc *sc = device_get_softc(dev); 67879251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 67979251f5eSSepherosa Ziehau 6808d0afa86SSepherosa Ziehau ix_sync_netisr(sc, IFF_UP); 6818d0afa86SSepherosa Ziehau taskqueue_drain(taskqueue_thread[0], &sc->wdog_task); 6828d0afa86SSepherosa Ziehau 68379251f5eSSepherosa Ziehau ifnet_serialize_all(ifp); 68463d483cdSSepherosa Ziehau ix_powerdown(sc); 68579251f5eSSepherosa Ziehau ifnet_deserialize_all(ifp); 68679251f5eSSepherosa Ziehau 68779251f5eSSepherosa Ziehau return 0; 68879251f5eSSepherosa Ziehau } 68979251f5eSSepherosa Ziehau 69079251f5eSSepherosa Ziehau static void 69179251f5eSSepherosa Ziehau ix_start(struct ifnet *ifp, struct ifaltq_subque *ifsq) 69279251f5eSSepherosa Ziehau { 69379251f5eSSepherosa Ziehau struct ix_softc *sc = ifp->if_softc; 69479251f5eSSepherosa Ziehau struct ix_tx_ring *txr = ifsq_get_priv(ifsq); 69579251f5eSSepherosa Ziehau int idx = -1; 69679251f5eSSepherosa Ziehau uint16_t nsegs; 69779251f5eSSepherosa Ziehau 69879251f5eSSepherosa Ziehau KKASSERT(txr->tx_ifsq == ifsq); 69979251f5eSSepherosa Ziehau ASSERT_SERIALIZED(&txr->tx_serialize); 70079251f5eSSepherosa Ziehau 70179251f5eSSepherosa Ziehau if ((ifp->if_flags & IFF_RUNNING) == 0 || ifsq_is_oactive(ifsq)) 70279251f5eSSepherosa Ziehau return; 70379251f5eSSepherosa Ziehau 7044a648aefSSepherosa Ziehau if (!sc->link_active || (txr->tx_flags & IX_TXFLAG_ENABLED) == 0) { 70579251f5eSSepherosa Ziehau ifsq_purge(ifsq); 70679251f5eSSepherosa Ziehau return; 70779251f5eSSepherosa Ziehau } 70879251f5eSSepherosa Ziehau 70979251f5eSSepherosa Ziehau while (!ifsq_is_empty(ifsq)) { 71079251f5eSSepherosa Ziehau struct mbuf *m_head; 71179251f5eSSepherosa Ziehau 71279251f5eSSepherosa Ziehau if (txr->tx_avail <= IX_MAX_SCATTER + IX_TX_RESERVED) { 71379251f5eSSepherosa Ziehau ifsq_set_oactive(ifsq); 71479251f5eSSepherosa Ziehau txr->tx_watchdog.wd_timer = 5; 71579251f5eSSepherosa Ziehau break; 71679251f5eSSepherosa Ziehau } 71779251f5eSSepherosa Ziehau 71879251f5eSSepherosa Ziehau m_head = ifsq_dequeue(ifsq); 71979251f5eSSepherosa Ziehau if (m_head == NULL) 72079251f5eSSepherosa Ziehau break; 72179251f5eSSepherosa Ziehau 72279251f5eSSepherosa Ziehau if (ix_encap(txr, &m_head, &nsegs, &idx)) { 72379251f5eSSepherosa Ziehau IFNET_STAT_INC(ifp, oerrors, 1); 72479251f5eSSepherosa Ziehau continue; 72579251f5eSSepherosa Ziehau } 72679251f5eSSepherosa Ziehau 727608dda76SSepherosa Ziehau /* 728608dda76SSepherosa Ziehau * TX interrupt are aggressively aggregated, so increasing 729608dda76SSepherosa Ziehau * opackets at TX interrupt time will make the opackets 730608dda76SSepherosa Ziehau * statistics vastly inaccurate; we do the opackets increment 731608dda76SSepherosa Ziehau * now. 732608dda76SSepherosa Ziehau */ 733608dda76SSepherosa Ziehau IFNET_STAT_INC(ifp, opackets, 1); 734608dda76SSepherosa Ziehau 73579251f5eSSepherosa Ziehau if (nsegs >= txr->tx_wreg_nsegs) { 73679251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_TDT(txr->tx_idx), idx); 73779251f5eSSepherosa Ziehau nsegs = 0; 73879251f5eSSepherosa Ziehau idx = -1; 73979251f5eSSepherosa Ziehau } 74079251f5eSSepherosa Ziehau 74179251f5eSSepherosa Ziehau ETHER_BPF_MTAP(ifp, m_head); 74279251f5eSSepherosa Ziehau } 74379251f5eSSepherosa Ziehau if (idx >= 0) 74479251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_TDT(txr->tx_idx), idx); 74582db96e9SSepherosa Ziehau txr->tx_running = IX_TX_RUNNING; 74679251f5eSSepherosa Ziehau } 74779251f5eSSepherosa Ziehau 74879251f5eSSepherosa Ziehau static int 74979251f5eSSepherosa Ziehau ix_ioctl(struct ifnet *ifp, u_long command, caddr_t data, struct ucred *cr) 75079251f5eSSepherosa Ziehau { 75179251f5eSSepherosa Ziehau struct ix_softc *sc = ifp->if_softc; 75279251f5eSSepherosa Ziehau struct ifreq *ifr = (struct ifreq *) data; 75379251f5eSSepherosa Ziehau int error = 0, mask, reinit; 75479251f5eSSepherosa Ziehau 75579251f5eSSepherosa Ziehau ASSERT_IFNET_SERIALIZED_ALL(ifp); 75679251f5eSSepherosa Ziehau 75779251f5eSSepherosa Ziehau switch (command) { 75879251f5eSSepherosa Ziehau case SIOCSIFMTU: 75963d483cdSSepherosa Ziehau if (ifr->ifr_mtu > IX_MAX_MTU) { 76079251f5eSSepherosa Ziehau error = EINVAL; 76179251f5eSSepherosa Ziehau } else { 76279251f5eSSepherosa Ziehau ifp->if_mtu = ifr->ifr_mtu; 76363d483cdSSepherosa Ziehau sc->max_frame_size = ifp->if_mtu + IX_MTU_HDR; 76479251f5eSSepherosa Ziehau ix_init(sc); 76579251f5eSSepherosa Ziehau } 76679251f5eSSepherosa Ziehau break; 76779251f5eSSepherosa Ziehau 76879251f5eSSepherosa Ziehau case SIOCSIFFLAGS: 76979251f5eSSepherosa Ziehau if (ifp->if_flags & IFF_UP) { 77079251f5eSSepherosa Ziehau if (ifp->if_flags & IFF_RUNNING) { 77179251f5eSSepherosa Ziehau if ((ifp->if_flags ^ sc->if_flags) & 77279251f5eSSepherosa Ziehau (IFF_PROMISC | IFF_ALLMULTI)) 77379251f5eSSepherosa Ziehau ix_set_promisc(sc); 77479251f5eSSepherosa Ziehau } else { 77579251f5eSSepherosa Ziehau ix_init(sc); 77679251f5eSSepherosa Ziehau } 77779251f5eSSepherosa Ziehau } else if (ifp->if_flags & IFF_RUNNING) { 77879251f5eSSepherosa Ziehau ix_stop(sc); 77979251f5eSSepherosa Ziehau } 78079251f5eSSepherosa Ziehau sc->if_flags = ifp->if_flags; 78179251f5eSSepherosa Ziehau break; 78279251f5eSSepherosa Ziehau 78379251f5eSSepherosa Ziehau case SIOCADDMULTI: 78479251f5eSSepherosa Ziehau case SIOCDELMULTI: 78579251f5eSSepherosa Ziehau if (ifp->if_flags & IFF_RUNNING) { 78679251f5eSSepherosa Ziehau ix_disable_intr(sc); 78779251f5eSSepherosa Ziehau ix_set_multi(sc); 7884a648aefSSepherosa Ziehau #ifdef IFPOLL_ENABLE 7894a648aefSSepherosa Ziehau if ((ifp->if_flags & IFF_NPOLLING) == 0) 7904a648aefSSepherosa Ziehau #endif 79179251f5eSSepherosa Ziehau ix_enable_intr(sc); 79279251f5eSSepherosa Ziehau } 79379251f5eSSepherosa Ziehau break; 79479251f5eSSepherosa Ziehau 79579251f5eSSepherosa Ziehau case SIOCSIFMEDIA: 79679251f5eSSepherosa Ziehau case SIOCGIFMEDIA: 79779251f5eSSepherosa Ziehau error = ifmedia_ioctl(ifp, ifr, &sc->media, command); 79879251f5eSSepherosa Ziehau break; 79979251f5eSSepherosa Ziehau 80079251f5eSSepherosa Ziehau case SIOCSIFCAP: 80179251f5eSSepherosa Ziehau reinit = 0; 80279251f5eSSepherosa Ziehau mask = ifr->ifr_reqcap ^ ifp->if_capenable; 80379251f5eSSepherosa Ziehau if (mask & IFCAP_RXCSUM) { 80479251f5eSSepherosa Ziehau ifp->if_capenable ^= IFCAP_RXCSUM; 80579251f5eSSepherosa Ziehau reinit = 1; 80679251f5eSSepherosa Ziehau } 80779251f5eSSepherosa Ziehau if (mask & IFCAP_VLAN_HWTAGGING) { 80879251f5eSSepherosa Ziehau ifp->if_capenable ^= IFCAP_VLAN_HWTAGGING; 80979251f5eSSepherosa Ziehau reinit = 1; 81079251f5eSSepherosa Ziehau } 81179251f5eSSepherosa Ziehau if (mask & IFCAP_TXCSUM) { 81279251f5eSSepherosa Ziehau ifp->if_capenable ^= IFCAP_TXCSUM; 81379251f5eSSepherosa Ziehau if (ifp->if_capenable & IFCAP_TXCSUM) 81479251f5eSSepherosa Ziehau ifp->if_hwassist |= CSUM_OFFLOAD; 81579251f5eSSepherosa Ziehau else 81679251f5eSSepherosa Ziehau ifp->if_hwassist &= ~CSUM_OFFLOAD; 81779251f5eSSepherosa Ziehau } 81879251f5eSSepherosa Ziehau if (mask & IFCAP_TSO) { 81979251f5eSSepherosa Ziehau ifp->if_capenable ^= IFCAP_TSO; 82079251f5eSSepherosa Ziehau if (ifp->if_capenable & IFCAP_TSO) 82179251f5eSSepherosa Ziehau ifp->if_hwassist |= CSUM_TSO; 82279251f5eSSepherosa Ziehau else 82379251f5eSSepherosa Ziehau ifp->if_hwassist &= ~CSUM_TSO; 82479251f5eSSepherosa Ziehau } 82579251f5eSSepherosa Ziehau if (mask & IFCAP_RSS) 82679251f5eSSepherosa Ziehau ifp->if_capenable ^= IFCAP_RSS; 82779251f5eSSepherosa Ziehau if (reinit && (ifp->if_flags & IFF_RUNNING)) 82879251f5eSSepherosa Ziehau ix_init(sc); 82979251f5eSSepherosa Ziehau break; 83079251f5eSSepherosa Ziehau 83179251f5eSSepherosa Ziehau #if 0 83279251f5eSSepherosa Ziehau case SIOCGI2C: 83379251f5eSSepherosa Ziehau { 83479251f5eSSepherosa Ziehau struct ixgbe_i2c_req i2c; 83579251f5eSSepherosa Ziehau error = copyin(ifr->ifr_data, &i2c, sizeof(i2c)); 83679251f5eSSepherosa Ziehau if (error) 83779251f5eSSepherosa Ziehau break; 83879251f5eSSepherosa Ziehau if ((i2c.dev_addr != 0xA0) || (i2c.dev_addr != 0xA2)){ 83979251f5eSSepherosa Ziehau error = EINVAL; 84079251f5eSSepherosa Ziehau break; 84179251f5eSSepherosa Ziehau } 84279251f5eSSepherosa Ziehau hw->phy.ops.read_i2c_byte(hw, i2c.offset, 84379251f5eSSepherosa Ziehau i2c.dev_addr, i2c.data); 84479251f5eSSepherosa Ziehau error = copyout(&i2c, ifr->ifr_data, sizeof(i2c)); 84579251f5eSSepherosa Ziehau break; 84679251f5eSSepherosa Ziehau } 84779251f5eSSepherosa Ziehau #endif 84879251f5eSSepherosa Ziehau 84979251f5eSSepherosa Ziehau default: 85079251f5eSSepherosa Ziehau error = ether_ioctl(ifp, command, data); 85179251f5eSSepherosa Ziehau break; 85279251f5eSSepherosa Ziehau } 85379251f5eSSepherosa Ziehau return error; 85479251f5eSSepherosa Ziehau } 85579251f5eSSepherosa Ziehau 85679251f5eSSepherosa Ziehau #define IXGBE_MHADD_MFS_SHIFT 16 85779251f5eSSepherosa Ziehau 85879251f5eSSepherosa Ziehau static void 85979251f5eSSepherosa Ziehau ix_init(void *xsc) 86079251f5eSSepherosa Ziehau { 86179251f5eSSepherosa Ziehau struct ix_softc *sc = xsc; 86279251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 86379251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 86479251f5eSSepherosa Ziehau uint32_t gpie, rxctrl; 86579251f5eSSepherosa Ziehau int i, error; 8664a648aefSSepherosa Ziehau boolean_t polling; 86779251f5eSSepherosa Ziehau 86879251f5eSSepherosa Ziehau ASSERT_IFNET_SERIALIZED_ALL(ifp); 86979251f5eSSepherosa Ziehau 87079251f5eSSepherosa Ziehau ix_stop(sc); 87179251f5eSSepherosa Ziehau 8724a648aefSSepherosa Ziehau polling = FALSE; 8734a648aefSSepherosa Ziehau #ifdef IFPOLL_ENABLE 8744a648aefSSepherosa Ziehau if (ifp->if_flags & IFF_NPOLLING) 8754a648aefSSepherosa Ziehau polling = TRUE; 8764a648aefSSepherosa Ziehau #endif 8774a648aefSSepherosa Ziehau 87879251f5eSSepherosa Ziehau /* Configure # of used RX/TX rings */ 8794a648aefSSepherosa Ziehau ix_set_ring_inuse(sc, polling); 8803c37d13bSSepherosa Ziehau ifq_set_subq_divisor(&ifp->if_snd, sc->tx_ring_inuse); 88179251f5eSSepherosa Ziehau 88279251f5eSSepherosa Ziehau /* Get the latest mac address, User can use a LAA */ 88379251f5eSSepherosa Ziehau bcopy(IF_LLADDR(ifp), hw->mac.addr, IXGBE_ETH_LENGTH_OF_ADDRESS); 88479251f5eSSepherosa Ziehau ixgbe_set_rar(hw, 0, hw->mac.addr, 0, 1); 88579251f5eSSepherosa Ziehau hw->addr_ctrl.rar_used_count = 1; 88679251f5eSSepherosa Ziehau 88779251f5eSSepherosa Ziehau /* Prepare transmit descriptors and buffers */ 88879251f5eSSepherosa Ziehau for (i = 0; i < sc->tx_ring_inuse; ++i) 88979251f5eSSepherosa Ziehau ix_init_tx_ring(&sc->tx_rings[i]); 89079251f5eSSepherosa Ziehau 89179251f5eSSepherosa Ziehau ixgbe_init_hw(hw); 89279251f5eSSepherosa Ziehau ix_init_tx_unit(sc); 89379251f5eSSepherosa Ziehau 89479251f5eSSepherosa Ziehau /* Setup Multicast table */ 89579251f5eSSepherosa Ziehau ix_set_multi(sc); 89679251f5eSSepherosa Ziehau 89779251f5eSSepherosa Ziehau /* Prepare receive descriptors and buffers */ 89879251f5eSSepherosa Ziehau for (i = 0; i < sc->rx_ring_inuse; ++i) { 89979251f5eSSepherosa Ziehau error = ix_init_rx_ring(&sc->rx_rings[i]); 90079251f5eSSepherosa Ziehau if (error) { 90179251f5eSSepherosa Ziehau if_printf(ifp, "Could not initialize RX ring%d\n", i); 90279251f5eSSepherosa Ziehau ix_stop(sc); 90379251f5eSSepherosa Ziehau return; 90479251f5eSSepherosa Ziehau } 90579251f5eSSepherosa Ziehau } 90679251f5eSSepherosa Ziehau 90779251f5eSSepherosa Ziehau /* Configure RX settings */ 9083c37d13bSSepherosa Ziehau ix_init_rx_unit(sc, polling); 90979251f5eSSepherosa Ziehau 91079251f5eSSepherosa Ziehau gpie = IXGBE_READ_REG(hw, IXGBE_GPIE); 91179251f5eSSepherosa Ziehau 91279251f5eSSepherosa Ziehau /* Enable Fan Failure Interrupt */ 91363d483cdSSepherosa Ziehau gpie |= IXGBE_SDP1_GPIEN_BY_MAC(hw); 91479251f5eSSepherosa Ziehau 91579251f5eSSepherosa Ziehau /* Add for Module detection */ 91679251f5eSSepherosa Ziehau if (hw->mac.type == ixgbe_mac_82599EB) 91779251f5eSSepherosa Ziehau gpie |= IXGBE_SDP2_GPIEN; 91879251f5eSSepherosa Ziehau 91963d483cdSSepherosa Ziehau /* 92063d483cdSSepherosa Ziehau * Thermal Failure Detection (X540) 92163d483cdSSepherosa Ziehau * Link Detection (X552) 92263d483cdSSepherosa Ziehau */ 92363d483cdSSepherosa Ziehau if (hw->mac.type == ixgbe_mac_X540 || 92463d483cdSSepherosa Ziehau hw->device_id == IXGBE_DEV_ID_X550EM_X_SFP || 92563d483cdSSepherosa Ziehau hw->device_id == IXGBE_DEV_ID_X550EM_X_10G_T) 92663d483cdSSepherosa Ziehau gpie |= IXGBE_SDP0_GPIEN_X540; 92779251f5eSSepherosa Ziehau 92879251f5eSSepherosa Ziehau if (sc->intr_type == PCI_INTR_TYPE_MSIX) { 92979251f5eSSepherosa Ziehau /* Enable Enhanced MSIX mode */ 93079251f5eSSepherosa Ziehau gpie |= IXGBE_GPIE_MSIX_MODE; 93179251f5eSSepherosa Ziehau gpie |= IXGBE_GPIE_EIAME | IXGBE_GPIE_PBA_SUPPORT | 93279251f5eSSepherosa Ziehau IXGBE_GPIE_OCD; 93379251f5eSSepherosa Ziehau } 93479251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_GPIE, gpie); 93579251f5eSSepherosa Ziehau 93679251f5eSSepherosa Ziehau /* Set MTU size */ 93779251f5eSSepherosa Ziehau if (ifp->if_mtu > ETHERMTU) { 93879251f5eSSepherosa Ziehau uint32_t mhadd; 93979251f5eSSepherosa Ziehau 94063d483cdSSepherosa Ziehau /* aka IXGBE_MAXFRS on 82599 and newer */ 94179251f5eSSepherosa Ziehau mhadd = IXGBE_READ_REG(hw, IXGBE_MHADD); 94279251f5eSSepherosa Ziehau mhadd &= ~IXGBE_MHADD_MFS_MASK; 94379251f5eSSepherosa Ziehau mhadd |= sc->max_frame_size << IXGBE_MHADD_MFS_SHIFT; 94479251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_MHADD, mhadd); 94579251f5eSSepherosa Ziehau } 94679251f5eSSepherosa Ziehau 94779251f5eSSepherosa Ziehau /* 94879251f5eSSepherosa Ziehau * Enable TX rings 94979251f5eSSepherosa Ziehau */ 95079251f5eSSepherosa Ziehau for (i = 0; i < sc->tx_ring_inuse; ++i) { 95179251f5eSSepherosa Ziehau uint32_t txdctl; 95279251f5eSSepherosa Ziehau 95379251f5eSSepherosa Ziehau txdctl = IXGBE_READ_REG(hw, IXGBE_TXDCTL(i)); 95479251f5eSSepherosa Ziehau txdctl |= IXGBE_TXDCTL_ENABLE; 95579251f5eSSepherosa Ziehau 95679251f5eSSepherosa Ziehau /* 95779251f5eSSepherosa Ziehau * Set WTHRESH to 0, since TX head write-back is used 95879251f5eSSepherosa Ziehau */ 95979251f5eSSepherosa Ziehau txdctl &= ~(0x7f << 16); 96079251f5eSSepherosa Ziehau 96179251f5eSSepherosa Ziehau /* 96279251f5eSSepherosa Ziehau * When the internal queue falls below PTHRESH (32), 96379251f5eSSepherosa Ziehau * start prefetching as long as there are at least 96479251f5eSSepherosa Ziehau * HTHRESH (1) buffers ready. The values are taken 96579251f5eSSepherosa Ziehau * from the Intel linux driver 3.8.21. 96679251f5eSSepherosa Ziehau * Prefetching enables tx line rate even with 1 queue. 96779251f5eSSepherosa Ziehau */ 96879251f5eSSepherosa Ziehau txdctl |= (32 << 0) | (1 << 8); 96979251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_TXDCTL(i), txdctl); 97079251f5eSSepherosa Ziehau } 97179251f5eSSepherosa Ziehau 97279251f5eSSepherosa Ziehau /* 97379251f5eSSepherosa Ziehau * Enable RX rings 97479251f5eSSepherosa Ziehau */ 97579251f5eSSepherosa Ziehau for (i = 0; i < sc->rx_ring_inuse; ++i) { 97679251f5eSSepherosa Ziehau uint32_t rxdctl; 97779251f5eSSepherosa Ziehau int k; 97879251f5eSSepherosa Ziehau 97979251f5eSSepherosa Ziehau rxdctl = IXGBE_READ_REG(hw, IXGBE_RXDCTL(i)); 98079251f5eSSepherosa Ziehau if (hw->mac.type == ixgbe_mac_82598EB) { 98179251f5eSSepherosa Ziehau /* 98279251f5eSSepherosa Ziehau * PTHRESH = 21 98379251f5eSSepherosa Ziehau * HTHRESH = 4 98479251f5eSSepherosa Ziehau * WTHRESH = 8 98579251f5eSSepherosa Ziehau */ 98679251f5eSSepherosa Ziehau rxdctl &= ~0x3FFFFF; 98779251f5eSSepherosa Ziehau rxdctl |= 0x080420; 98879251f5eSSepherosa Ziehau } 98979251f5eSSepherosa Ziehau rxdctl |= IXGBE_RXDCTL_ENABLE; 99079251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RXDCTL(i), rxdctl); 99179251f5eSSepherosa Ziehau for (k = 0; k < 10; ++k) { 99279251f5eSSepherosa Ziehau if (IXGBE_READ_REG(hw, IXGBE_RXDCTL(i)) & 99379251f5eSSepherosa Ziehau IXGBE_RXDCTL_ENABLE) 99479251f5eSSepherosa Ziehau break; 99579251f5eSSepherosa Ziehau else 99679251f5eSSepherosa Ziehau msec_delay(1); 99779251f5eSSepherosa Ziehau } 99879251f5eSSepherosa Ziehau wmb(); 99979251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RDT(i), 100079251f5eSSepherosa Ziehau sc->rx_rings[0].rx_ndesc - 1); 100179251f5eSSepherosa Ziehau } 100279251f5eSSepherosa Ziehau 100379251f5eSSepherosa Ziehau /* Enable Receive engine */ 100479251f5eSSepherosa Ziehau rxctrl = IXGBE_READ_REG(hw, IXGBE_RXCTRL); 100579251f5eSSepherosa Ziehau if (hw->mac.type == ixgbe_mac_82598EB) 100679251f5eSSepherosa Ziehau rxctrl |= IXGBE_RXCTRL_DMBYPS; 100779251f5eSSepherosa Ziehau rxctrl |= IXGBE_RXCTRL_RXEN; 100879251f5eSSepherosa Ziehau ixgbe_enable_rx_dma(hw, rxctrl); 100979251f5eSSepherosa Ziehau 1010189a0ff3SSepherosa Ziehau for (i = 0; i < sc->tx_ring_inuse; ++i) { 1011189a0ff3SSepherosa Ziehau const struct ix_tx_ring *txr = &sc->tx_rings[i]; 1012189a0ff3SSepherosa Ziehau 1013189a0ff3SSepherosa Ziehau if (txr->tx_intr_vec >= 0) { 1014189a0ff3SSepherosa Ziehau ix_set_ivar(sc, i, txr->tx_intr_vec, 1); 10153c37d13bSSepherosa Ziehau } else if (!polling) { 1016189a0ff3SSepherosa Ziehau /* 1017189a0ff3SSepherosa Ziehau * Unconfigured TX interrupt vector could only 1018189a0ff3SSepherosa Ziehau * happen for MSI-X. 1019189a0ff3SSepherosa Ziehau */ 1020189a0ff3SSepherosa Ziehau KASSERT(sc->intr_type == PCI_INTR_TYPE_MSIX, 1021189a0ff3SSepherosa Ziehau ("TX intr vector is not set")); 1022189a0ff3SSepherosa Ziehau if (bootverbose) 1023189a0ff3SSepherosa Ziehau if_printf(ifp, "IVAR skips TX ring %d\n", i); 1024189a0ff3SSepherosa Ziehau } 1025189a0ff3SSepherosa Ziehau } 1026189a0ff3SSepherosa Ziehau for (i = 0; i < sc->rx_ring_inuse; ++i) { 1027189a0ff3SSepherosa Ziehau const struct ix_rx_ring *rxr = &sc->rx_rings[i]; 1028189a0ff3SSepherosa Ziehau 10293c37d13bSSepherosa Ziehau if (polling && rxr->rx_intr_vec < 0) 10303c37d13bSSepherosa Ziehau continue; 10313c37d13bSSepherosa Ziehau 1032189a0ff3SSepherosa Ziehau KKASSERT(rxr->rx_intr_vec >= 0); 1033189a0ff3SSepherosa Ziehau ix_set_ivar(sc, i, rxr->rx_intr_vec, 0); 1034189a0ff3SSepherosa Ziehau if (rxr->rx_txr != NULL) { 1035189a0ff3SSepherosa Ziehau /* 1036189a0ff3SSepherosa Ziehau * Piggyback the TX ring interrupt onto the RX 1037189a0ff3SSepherosa Ziehau * ring interrupt vector. 1038189a0ff3SSepherosa Ziehau */ 1039189a0ff3SSepherosa Ziehau KASSERT(rxr->rx_txr->tx_intr_vec < 0, 1040189a0ff3SSepherosa Ziehau ("piggybacked TX ring configured intr vector")); 10413c37d13bSSepherosa Ziehau ix_set_ivar(sc, rxr->rx_txr->tx_idx, 10423c37d13bSSepherosa Ziehau rxr->rx_intr_vec, 1); 1043189a0ff3SSepherosa Ziehau if (bootverbose) { 1044189a0ff3SSepherosa Ziehau if_printf(ifp, "IVAR RX ring %d piggybacks " 1045189a0ff3SSepherosa Ziehau "TX ring %u\n", i, rxr->rx_txr->tx_idx); 1046189a0ff3SSepherosa Ziehau } 1047189a0ff3SSepherosa Ziehau } 1048189a0ff3SSepherosa Ziehau } 104979251f5eSSepherosa Ziehau if (sc->intr_type == PCI_INTR_TYPE_MSIX) { 1050189a0ff3SSepherosa Ziehau /* Set up status MSI-X vector; it is using fixed entry 1 */ 1051189a0ff3SSepherosa Ziehau ix_set_ivar(sc, 1, sc->sts_msix_vec, -1); 1052189a0ff3SSepherosa Ziehau 1053189a0ff3SSepherosa Ziehau /* Set up auto-mask for TX and RX rings */ 1054189a0ff3SSepherosa Ziehau if (hw->mac.type == ixgbe_mac_82598EB) { 1055189a0ff3SSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_EIAM, IXGBE_EIMS_RTX_QUEUE); 1056189a0ff3SSepherosa Ziehau } else { 105779251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_EIAM_EX(0), 0xFFFFFFFF); 105879251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_EIAM_EX(1), 0xFFFFFFFF); 105979251f5eSSepherosa Ziehau } 106079251f5eSSepherosa Ziehau } else { 1061189a0ff3SSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_EIAM, IXGBE_EIMS_RTX_QUEUE); 106279251f5eSSepherosa Ziehau } 1063189a0ff3SSepherosa Ziehau for (i = 0; i < sc->intr_cnt; ++i) 1064189a0ff3SSepherosa Ziehau ix_set_eitr(sc, i, sc->intr_data[i].intr_rate); 106579251f5eSSepherosa Ziehau 106679251f5eSSepherosa Ziehau /* 106779251f5eSSepherosa Ziehau * Check on any SFP devices that need to be kick-started 106879251f5eSSepherosa Ziehau */ 106979251f5eSSepherosa Ziehau if (hw->phy.type == ixgbe_phy_none) { 107079251f5eSSepherosa Ziehau error = hw->phy.ops.identify(hw); 107179251f5eSSepherosa Ziehau if (error == IXGBE_ERR_SFP_NOT_SUPPORTED) { 107279251f5eSSepherosa Ziehau if_printf(ifp, 107379251f5eSSepherosa Ziehau "Unsupported SFP+ module type was detected.\n"); 107479251f5eSSepherosa Ziehau /* XXX stop */ 107579251f5eSSepherosa Ziehau return; 107679251f5eSSepherosa Ziehau } 107779251f5eSSepherosa Ziehau } 107879251f5eSSepherosa Ziehau 107979251f5eSSepherosa Ziehau /* Config/Enable Link */ 108079251f5eSSepherosa Ziehau ix_config_link(sc); 108179251f5eSSepherosa Ziehau 108263d483cdSSepherosa Ziehau /* Hardware Packet Buffer & Flow Control setup */ 108363d483cdSSepherosa Ziehau ix_config_flowctrl(sc); 108479251f5eSSepherosa Ziehau 108579251f5eSSepherosa Ziehau /* Initialize the FC settings */ 108679251f5eSSepherosa Ziehau ixgbe_start_hw(hw); 108779251f5eSSepherosa Ziehau 108863d483cdSSepherosa Ziehau /* Set up VLAN support and filter */ 108963d483cdSSepherosa Ziehau ix_set_vlan(sc); 109063d483cdSSepherosa Ziehau 109163d483cdSSepherosa Ziehau /* Setup DMA Coalescing */ 109263d483cdSSepherosa Ziehau ix_config_dmac(sc); 109363d483cdSSepherosa Ziehau 10944a648aefSSepherosa Ziehau /* 10954a648aefSSepherosa Ziehau * Only enable interrupts if we are not polling, make sure 10964a648aefSSepherosa Ziehau * they are off otherwise. 10974a648aefSSepherosa Ziehau */ 10984a648aefSSepherosa Ziehau if (polling) 10994a648aefSSepherosa Ziehau ix_disable_intr(sc); 11004a648aefSSepherosa Ziehau else 110179251f5eSSepherosa Ziehau ix_enable_intr(sc); 110279251f5eSSepherosa Ziehau 110379251f5eSSepherosa Ziehau ifp->if_flags |= IFF_RUNNING; 110479251f5eSSepherosa Ziehau for (i = 0; i < sc->tx_ring_inuse; ++i) { 110582db96e9SSepherosa Ziehau struct ix_tx_ring *txr = &sc->tx_rings[i]; 110682db96e9SSepherosa Ziehau 110782db96e9SSepherosa Ziehau ifsq_clr_oactive(txr->tx_ifsq); 110882db96e9SSepherosa Ziehau ifsq_watchdog_start(&txr->tx_watchdog); 110982db96e9SSepherosa Ziehau 111082db96e9SSepherosa Ziehau if (!polling) { 111182db96e9SSepherosa Ziehau callout_reset_bycpu(&txr->tx_gc_timer, 1, 111282db96e9SSepherosa Ziehau ix_txgc_timer, txr, txr->tx_intr_cpuid); 111382db96e9SSepherosa Ziehau } 111479251f5eSSepherosa Ziehau } 111579251f5eSSepherosa Ziehau 11164a648aefSSepherosa Ziehau ix_set_timer_cpuid(sc, polling); 111779251f5eSSepherosa Ziehau callout_reset_bycpu(&sc->timer, hz, ix_timer, sc, sc->timer_cpuid); 111879251f5eSSepherosa Ziehau } 111979251f5eSSepherosa Ziehau 112079251f5eSSepherosa Ziehau static void 112179251f5eSSepherosa Ziehau ix_intr(void *xsc) 112279251f5eSSepherosa Ziehau { 112379251f5eSSepherosa Ziehau struct ix_softc *sc = xsc; 112479251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 112579251f5eSSepherosa Ziehau uint32_t eicr; 112679251f5eSSepherosa Ziehau 112779251f5eSSepherosa Ziehau ASSERT_SERIALIZED(&sc->main_serialize); 112879251f5eSSepherosa Ziehau 112979251f5eSSepherosa Ziehau eicr = IXGBE_READ_REG(hw, IXGBE_EICR); 113079251f5eSSepherosa Ziehau if (eicr == 0) { 113179251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_EIMS, sc->intr_mask); 113279251f5eSSepherosa Ziehau return; 113379251f5eSSepherosa Ziehau } 113479251f5eSSepherosa Ziehau 113579251f5eSSepherosa Ziehau if (eicr & IX_RX0_INTR_MASK) { 113679251f5eSSepherosa Ziehau struct ix_rx_ring *rxr = &sc->rx_rings[0]; 113779251f5eSSepherosa Ziehau 113879251f5eSSepherosa Ziehau lwkt_serialize_enter(&rxr->rx_serialize); 11394a648aefSSepherosa Ziehau ix_rxeof(rxr, -1); 114079251f5eSSepherosa Ziehau lwkt_serialize_exit(&rxr->rx_serialize); 114179251f5eSSepherosa Ziehau } 114279251f5eSSepherosa Ziehau if (eicr & IX_RX1_INTR_MASK) { 114379251f5eSSepherosa Ziehau struct ix_rx_ring *rxr; 114479251f5eSSepherosa Ziehau 114579251f5eSSepherosa Ziehau KKASSERT(sc->rx_ring_inuse == IX_MIN_RXRING_RSS); 114679251f5eSSepherosa Ziehau rxr = &sc->rx_rings[1]; 114779251f5eSSepherosa Ziehau 114879251f5eSSepherosa Ziehau lwkt_serialize_enter(&rxr->rx_serialize); 11494a648aefSSepherosa Ziehau ix_rxeof(rxr, -1); 115079251f5eSSepherosa Ziehau lwkt_serialize_exit(&rxr->rx_serialize); 115179251f5eSSepherosa Ziehau } 115279251f5eSSepherosa Ziehau 115379251f5eSSepherosa Ziehau if (eicr & IX_TX_INTR_MASK) { 115479251f5eSSepherosa Ziehau struct ix_tx_ring *txr = &sc->tx_rings[0]; 115579251f5eSSepherosa Ziehau 115679251f5eSSepherosa Ziehau lwkt_serialize_enter(&txr->tx_serialize); 115782db96e9SSepherosa Ziehau ix_tx_intr(txr, *(txr->tx_hdr)); 115879251f5eSSepherosa Ziehau lwkt_serialize_exit(&txr->tx_serialize); 115979251f5eSSepherosa Ziehau } 116079251f5eSSepherosa Ziehau 1161189a0ff3SSepherosa Ziehau if (__predict_false(eicr & IX_EICR_STATUS)) 1162189a0ff3SSepherosa Ziehau ix_intr_status(sc, eicr); 116379251f5eSSepherosa Ziehau 116479251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_EIMS, sc->intr_mask); 116579251f5eSSepherosa Ziehau } 116679251f5eSSepherosa Ziehau 116779251f5eSSepherosa Ziehau static void 116879251f5eSSepherosa Ziehau ix_media_status(struct ifnet *ifp, struct ifmediareq *ifmr) 116979251f5eSSepherosa Ziehau { 117079251f5eSSepherosa Ziehau struct ix_softc *sc = ifp->if_softc; 117163d483cdSSepherosa Ziehau struct ifmedia *ifm = &sc->media; 117263d483cdSSepherosa Ziehau int layer; 117379251f5eSSepherosa Ziehau 117479251f5eSSepherosa Ziehau ix_update_link_status(sc); 117579251f5eSSepherosa Ziehau 117679251f5eSSepherosa Ziehau ifmr->ifm_status = IFM_AVALID; 117779251f5eSSepherosa Ziehau ifmr->ifm_active = IFM_ETHER; 117879251f5eSSepherosa Ziehau 11790d60c5c8SSepherosa Ziehau if (!sc->link_active) { 118063d483cdSSepherosa Ziehau if (IFM_SUBTYPE(ifm->ifm_media) != IFM_AUTO) 118163d483cdSSepherosa Ziehau ifmr->ifm_active |= ifm->ifm_media; 118263d483cdSSepherosa Ziehau else 11830d60c5c8SSepherosa Ziehau ifmr->ifm_active |= IFM_NONE; 118479251f5eSSepherosa Ziehau return; 11850d60c5c8SSepherosa Ziehau } 118679251f5eSSepherosa Ziehau ifmr->ifm_status |= IFM_ACTIVE; 118779251f5eSSepherosa Ziehau 118863d483cdSSepherosa Ziehau layer = ixgbe_get_supported_physical_layer(&sc->hw); 118963d483cdSSepherosa Ziehau 119063d483cdSSepherosa Ziehau if ((layer & IXGBE_PHYSICAL_LAYER_10GBASE_T) || 119163d483cdSSepherosa Ziehau (layer & IXGBE_PHYSICAL_LAYER_1000BASE_T) || 119263d483cdSSepherosa Ziehau (layer & IXGBE_PHYSICAL_LAYER_100BASE_TX)) { 119379251f5eSSepherosa Ziehau switch (sc->link_speed) { 119463d483cdSSepherosa Ziehau case IXGBE_LINK_SPEED_10GB_FULL: 119563d483cdSSepherosa Ziehau ifmr->ifm_active |= IFM_10G_T | IFM_FDX; 119663d483cdSSepherosa Ziehau break; 119763d483cdSSepherosa Ziehau case IXGBE_LINK_SPEED_1GB_FULL: 119863d483cdSSepherosa Ziehau ifmr->ifm_active |= IFM_1000_T | IFM_FDX; 119963d483cdSSepherosa Ziehau break; 120079251f5eSSepherosa Ziehau case IXGBE_LINK_SPEED_100_FULL: 120179251f5eSSepherosa Ziehau ifmr->ifm_active |= IFM_100_TX | IFM_FDX; 120279251f5eSSepherosa Ziehau break; 120363d483cdSSepherosa Ziehau } 120463d483cdSSepherosa Ziehau } else if ((layer & IXGBE_PHYSICAL_LAYER_SFP_PLUS_CU) || 120563d483cdSSepherosa Ziehau (layer & IXGBE_PHYSICAL_LAYER_SFP_ACTIVE_DA)) { 120663d483cdSSepherosa Ziehau switch (sc->link_speed) { 120763d483cdSSepherosa Ziehau case IXGBE_LINK_SPEED_10GB_FULL: 120863d483cdSSepherosa Ziehau ifmr->ifm_active |= IFM_10G_TWINAX | IFM_FDX; 120963d483cdSSepherosa Ziehau break; 121063d483cdSSepherosa Ziehau } 121163d483cdSSepherosa Ziehau } else if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_LR) { 121263d483cdSSepherosa Ziehau switch (sc->link_speed) { 121363d483cdSSepherosa Ziehau case IXGBE_LINK_SPEED_10GB_FULL: 121463d483cdSSepherosa Ziehau ifmr->ifm_active |= IFM_10G_LR | IFM_FDX; 121563d483cdSSepherosa Ziehau break; 121663d483cdSSepherosa Ziehau case IXGBE_LINK_SPEED_1GB_FULL: 121763d483cdSSepherosa Ziehau ifmr->ifm_active |= IFM_1000_LX | IFM_FDX; 121863d483cdSSepherosa Ziehau break; 121963d483cdSSepherosa Ziehau } 122063d483cdSSepherosa Ziehau } else if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_LRM) { 122163d483cdSSepherosa Ziehau switch (sc->link_speed) { 122263d483cdSSepherosa Ziehau case IXGBE_LINK_SPEED_10GB_FULL: 122363d483cdSSepherosa Ziehau ifmr->ifm_active |= IFM_10G_LRM | IFM_FDX; 122463d483cdSSepherosa Ziehau break; 122563d483cdSSepherosa Ziehau case IXGBE_LINK_SPEED_1GB_FULL: 122663d483cdSSepherosa Ziehau ifmr->ifm_active |= IFM_1000_LX | IFM_FDX; 122763d483cdSSepherosa Ziehau break; 122863d483cdSSepherosa Ziehau } 122963d483cdSSepherosa Ziehau } else if ((layer & IXGBE_PHYSICAL_LAYER_10GBASE_SR) || 123063d483cdSSepherosa Ziehau (layer & IXGBE_PHYSICAL_LAYER_1000BASE_SX)) { 123163d483cdSSepherosa Ziehau switch (sc->link_speed) { 123263d483cdSSepherosa Ziehau case IXGBE_LINK_SPEED_10GB_FULL: 123363d483cdSSepherosa Ziehau ifmr->ifm_active |= IFM_10G_SR | IFM_FDX; 123463d483cdSSepherosa Ziehau break; 123579251f5eSSepherosa Ziehau case IXGBE_LINK_SPEED_1GB_FULL: 123679251f5eSSepherosa Ziehau ifmr->ifm_active |= IFM_1000_SX | IFM_FDX; 123779251f5eSSepherosa Ziehau break; 123863d483cdSSepherosa Ziehau } 123963d483cdSSepherosa Ziehau } else if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_CX4) { 124063d483cdSSepherosa Ziehau switch (sc->link_speed) { 124179251f5eSSepherosa Ziehau case IXGBE_LINK_SPEED_10GB_FULL: 124263d483cdSSepherosa Ziehau ifmr->ifm_active |= IFM_10G_CX4 | IFM_FDX; 124379251f5eSSepherosa Ziehau break; 124463d483cdSSepherosa Ziehau } 124563d483cdSSepherosa Ziehau } else if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_KR) { 124663d483cdSSepherosa Ziehau /* 124763d483cdSSepherosa Ziehau * XXX: These need to use the proper media types once 124863d483cdSSepherosa Ziehau * they're added. 124963d483cdSSepherosa Ziehau */ 125063d483cdSSepherosa Ziehau switch (sc->link_speed) { 125163d483cdSSepherosa Ziehau case IXGBE_LINK_SPEED_10GB_FULL: 125263d483cdSSepherosa Ziehau ifmr->ifm_active |= IFM_10G_SR | IFM_FDX; 125363d483cdSSepherosa Ziehau break; 125463d483cdSSepherosa Ziehau case IXGBE_LINK_SPEED_2_5GB_FULL: 125563d483cdSSepherosa Ziehau ifmr->ifm_active |= IFM_2500_SX | IFM_FDX; 125663d483cdSSepherosa Ziehau break; 125763d483cdSSepherosa Ziehau case IXGBE_LINK_SPEED_1GB_FULL: 125863d483cdSSepherosa Ziehau ifmr->ifm_active |= IFM_1000_CX | IFM_FDX; 125963d483cdSSepherosa Ziehau break; 126063d483cdSSepherosa Ziehau } 126163d483cdSSepherosa Ziehau } else if ((layer & IXGBE_PHYSICAL_LAYER_10GBASE_KX4) || 126263d483cdSSepherosa Ziehau (layer & IXGBE_PHYSICAL_LAYER_1000BASE_KX)) { 126363d483cdSSepherosa Ziehau /* 126463d483cdSSepherosa Ziehau * XXX: These need to use the proper media types once 126563d483cdSSepherosa Ziehau * they're added. 126663d483cdSSepherosa Ziehau */ 126763d483cdSSepherosa Ziehau switch (sc->link_speed) { 126863d483cdSSepherosa Ziehau case IXGBE_LINK_SPEED_10GB_FULL: 126963d483cdSSepherosa Ziehau ifmr->ifm_active |= IFM_10G_CX4 | IFM_FDX; 127063d483cdSSepherosa Ziehau break; 127163d483cdSSepherosa Ziehau case IXGBE_LINK_SPEED_2_5GB_FULL: 127263d483cdSSepherosa Ziehau ifmr->ifm_active |= IFM_2500_SX | IFM_FDX; 127363d483cdSSepherosa Ziehau break; 127463d483cdSSepherosa Ziehau case IXGBE_LINK_SPEED_1GB_FULL: 127563d483cdSSepherosa Ziehau ifmr->ifm_active |= IFM_1000_CX | IFM_FDX; 127663d483cdSSepherosa Ziehau break; 127763d483cdSSepherosa Ziehau } 1278060fa21cSSepherosa Ziehau } 1279060fa21cSSepherosa Ziehau 128063d483cdSSepherosa Ziehau /* If nothing is recognized... */ 128163d483cdSSepherosa Ziehau if (IFM_SUBTYPE(ifmr->ifm_active) == 0) 128263d483cdSSepherosa Ziehau ifmr->ifm_active |= IFM_NONE; 128363d483cdSSepherosa Ziehau 128463d483cdSSepherosa Ziehau if (sc->ifm_media & IFM_ETH_FORCEPAUSE) 128563d483cdSSepherosa Ziehau ifmr->ifm_active |= (sc->ifm_media & IFM_ETH_FCMASK); 1286060fa21cSSepherosa Ziehau 1287060fa21cSSepherosa Ziehau switch (sc->hw.fc.current_mode) { 1288060fa21cSSepherosa Ziehau case ixgbe_fc_full: 1289060fa21cSSepherosa Ziehau ifmr->ifm_active |= IFM_ETH_RXPAUSE | IFM_ETH_TXPAUSE; 1290060fa21cSSepherosa Ziehau break; 1291060fa21cSSepherosa Ziehau case ixgbe_fc_rx_pause: 1292060fa21cSSepherosa Ziehau ifmr->ifm_active |= IFM_ETH_RXPAUSE; 1293060fa21cSSepherosa Ziehau break; 1294060fa21cSSepherosa Ziehau case ixgbe_fc_tx_pause: 1295060fa21cSSepherosa Ziehau ifmr->ifm_active |= IFM_ETH_TXPAUSE; 1296060fa21cSSepherosa Ziehau break; 1297060fa21cSSepherosa Ziehau default: 12980d60c5c8SSepherosa Ziehau break; 129979251f5eSSepherosa Ziehau } 130079251f5eSSepherosa Ziehau } 130179251f5eSSepherosa Ziehau 130279251f5eSSepherosa Ziehau static int 130379251f5eSSepherosa Ziehau ix_media_change(struct ifnet *ifp) 130479251f5eSSepherosa Ziehau { 130579251f5eSSepherosa Ziehau struct ix_softc *sc = ifp->if_softc; 130679251f5eSSepherosa Ziehau struct ifmedia *ifm = &sc->media; 130763d483cdSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 130879251f5eSSepherosa Ziehau 130979251f5eSSepherosa Ziehau if (IFM_TYPE(ifm->ifm_media) != IFM_ETHER) 131063d483cdSSepherosa Ziehau return (EINVAL); 131163d483cdSSepherosa Ziehau 131263d483cdSSepherosa Ziehau if (hw->phy.media_type == ixgbe_media_type_backplane || 131363d483cdSSepherosa Ziehau hw->mac.ops.setup_link == NULL) { 131463d483cdSSepherosa Ziehau if ((ifm->ifm_media ^ sc->ifm_media) & IFM_ETH_FCMASK) { 131563d483cdSSepherosa Ziehau /* Only flow control setting changes are allowed */ 131663d483cdSSepherosa Ziehau return (EOPNOTSUPP); 131763d483cdSSepherosa Ziehau } 131863d483cdSSepherosa Ziehau } 131979251f5eSSepherosa Ziehau 132079251f5eSSepherosa Ziehau switch (IFM_SUBTYPE(ifm->ifm_media)) { 132179251f5eSSepherosa Ziehau case IFM_AUTO: 132263d483cdSSepherosa Ziehau sc->advspeed = IXGBE_LINK_SPEED_UNKNOWN; 132379251f5eSSepherosa Ziehau break; 132463d483cdSSepherosa Ziehau 132563d483cdSSepherosa Ziehau case IFM_10G_T: 132663d483cdSSepherosa Ziehau case IFM_10G_LRM: 132763d483cdSSepherosa Ziehau case IFM_10G_SR: /* XXX also KR */ 132863d483cdSSepherosa Ziehau case IFM_10G_LR: 132963d483cdSSepherosa Ziehau case IFM_10G_CX4: /* XXX also KX4 */ 133063d483cdSSepherosa Ziehau case IFM_10G_TWINAX: 133163d483cdSSepherosa Ziehau sc->advspeed = IXGBE_LINK_SPEED_10GB_FULL; 133263d483cdSSepherosa Ziehau break; 133363d483cdSSepherosa Ziehau 133463d483cdSSepherosa Ziehau case IFM_1000_T: 133563d483cdSSepherosa Ziehau case IFM_1000_LX: 133663d483cdSSepherosa Ziehau case IFM_1000_SX: 133763d483cdSSepherosa Ziehau case IFM_1000_CX: /* XXX is KX */ 133863d483cdSSepherosa Ziehau sc->advspeed = IXGBE_LINK_SPEED_1GB_FULL; 133963d483cdSSepherosa Ziehau break; 134063d483cdSSepherosa Ziehau 134163d483cdSSepherosa Ziehau case IFM_100_TX: 134263d483cdSSepherosa Ziehau sc->advspeed = IXGBE_LINK_SPEED_100_FULL; 134363d483cdSSepherosa Ziehau break; 134463d483cdSSepherosa Ziehau 134579251f5eSSepherosa Ziehau default: 134663d483cdSSepherosa Ziehau if (bootverbose) { 134763d483cdSSepherosa Ziehau if_printf(ifp, "Invalid media type %d!\n", 134863d483cdSSepherosa Ziehau ifm->ifm_media); 134963d483cdSSepherosa Ziehau } 135079251f5eSSepherosa Ziehau return EINVAL; 135179251f5eSSepherosa Ziehau } 135263d483cdSSepherosa Ziehau sc->ifm_media = ifm->ifm_media; 1353060fa21cSSepherosa Ziehau 135463d483cdSSepherosa Ziehau #if 0 135563d483cdSSepherosa Ziehau if (hw->mac.ops.setup_link != NULL) { 135663d483cdSSepherosa Ziehau hw->mac.autotry_restart = TRUE; 135763d483cdSSepherosa Ziehau hw->mac.ops.setup_link(hw, sc->advspeed, TRUE); 135863d483cdSSepherosa Ziehau } 135963d483cdSSepherosa Ziehau #else 1360060fa21cSSepherosa Ziehau if (ifp->if_flags & IFF_RUNNING) 1361060fa21cSSepherosa Ziehau ix_init(sc); 136263d483cdSSepherosa Ziehau #endif 136379251f5eSSepherosa Ziehau return 0; 136479251f5eSSepherosa Ziehau } 136579251f5eSSepherosa Ziehau 136679251f5eSSepherosa Ziehau static __inline int 136779251f5eSSepherosa Ziehau ix_tso_pullup(struct mbuf **mp) 136879251f5eSSepherosa Ziehau { 136979251f5eSSepherosa Ziehau int hoff, iphlen, thoff; 137079251f5eSSepherosa Ziehau struct mbuf *m; 137179251f5eSSepherosa Ziehau 137279251f5eSSepherosa Ziehau m = *mp; 137379251f5eSSepherosa Ziehau KASSERT(M_WRITABLE(m), ("TSO mbuf not writable")); 137479251f5eSSepherosa Ziehau 137579251f5eSSepherosa Ziehau iphlen = m->m_pkthdr.csum_iphlen; 137679251f5eSSepherosa Ziehau thoff = m->m_pkthdr.csum_thlen; 137779251f5eSSepherosa Ziehau hoff = m->m_pkthdr.csum_lhlen; 137879251f5eSSepherosa Ziehau 137979251f5eSSepherosa Ziehau KASSERT(iphlen > 0, ("invalid ip hlen")); 138079251f5eSSepherosa Ziehau KASSERT(thoff > 0, ("invalid tcp hlen")); 138179251f5eSSepherosa Ziehau KASSERT(hoff > 0, ("invalid ether hlen")); 138279251f5eSSepherosa Ziehau 138379251f5eSSepherosa Ziehau if (__predict_false(m->m_len < hoff + iphlen + thoff)) { 138479251f5eSSepherosa Ziehau m = m_pullup(m, hoff + iphlen + thoff); 138579251f5eSSepherosa Ziehau if (m == NULL) { 138679251f5eSSepherosa Ziehau *mp = NULL; 138779251f5eSSepherosa Ziehau return ENOBUFS; 138879251f5eSSepherosa Ziehau } 138979251f5eSSepherosa Ziehau *mp = m; 139079251f5eSSepherosa Ziehau } 139179251f5eSSepherosa Ziehau return 0; 139279251f5eSSepherosa Ziehau } 139379251f5eSSepherosa Ziehau 139479251f5eSSepherosa Ziehau static int 139579251f5eSSepherosa Ziehau ix_encap(struct ix_tx_ring *txr, struct mbuf **m_headp, 139679251f5eSSepherosa Ziehau uint16_t *segs_used, int *idx) 139779251f5eSSepherosa Ziehau { 139879251f5eSSepherosa Ziehau uint32_t olinfo_status = 0, cmd_type_len, cmd_rs = 0; 139979251f5eSSepherosa Ziehau int i, j, error, nsegs, first, maxsegs; 140079251f5eSSepherosa Ziehau struct mbuf *m_head = *m_headp; 140179251f5eSSepherosa Ziehau bus_dma_segment_t segs[IX_MAX_SCATTER]; 140279251f5eSSepherosa Ziehau bus_dmamap_t map; 140379251f5eSSepherosa Ziehau struct ix_tx_buf *txbuf; 140479251f5eSSepherosa Ziehau union ixgbe_adv_tx_desc *txd = NULL; 140579251f5eSSepherosa Ziehau 140679251f5eSSepherosa Ziehau if (m_head->m_pkthdr.csum_flags & CSUM_TSO) { 140779251f5eSSepherosa Ziehau error = ix_tso_pullup(m_headp); 140879251f5eSSepherosa Ziehau if (__predict_false(error)) 140979251f5eSSepherosa Ziehau return error; 141079251f5eSSepherosa Ziehau m_head = *m_headp; 141179251f5eSSepherosa Ziehau } 141279251f5eSSepherosa Ziehau 141379251f5eSSepherosa Ziehau /* Basic descriptor defines */ 141479251f5eSSepherosa Ziehau cmd_type_len = (IXGBE_ADVTXD_DTYP_DATA | 141579251f5eSSepherosa Ziehau IXGBE_ADVTXD_DCMD_IFCS | IXGBE_ADVTXD_DCMD_DEXT); 141679251f5eSSepherosa Ziehau 141779251f5eSSepherosa Ziehau if (m_head->m_flags & M_VLANTAG) 141879251f5eSSepherosa Ziehau cmd_type_len |= IXGBE_ADVTXD_DCMD_VLE; 141979251f5eSSepherosa Ziehau 142079251f5eSSepherosa Ziehau /* 142179251f5eSSepherosa Ziehau * Important to capture the first descriptor 142279251f5eSSepherosa Ziehau * used because it will contain the index of 142379251f5eSSepherosa Ziehau * the one we tell the hardware to report back 142479251f5eSSepherosa Ziehau */ 142579251f5eSSepherosa Ziehau first = txr->tx_next_avail; 142679251f5eSSepherosa Ziehau txbuf = &txr->tx_buf[first]; 142779251f5eSSepherosa Ziehau map = txbuf->map; 142879251f5eSSepherosa Ziehau 142979251f5eSSepherosa Ziehau /* 143079251f5eSSepherosa Ziehau * Map the packet for DMA. 143179251f5eSSepherosa Ziehau */ 143279251f5eSSepherosa Ziehau maxsegs = txr->tx_avail - IX_TX_RESERVED; 143379251f5eSSepherosa Ziehau if (maxsegs > IX_MAX_SCATTER) 143479251f5eSSepherosa Ziehau maxsegs = IX_MAX_SCATTER; 143579251f5eSSepherosa Ziehau 143679251f5eSSepherosa Ziehau error = bus_dmamap_load_mbuf_defrag(txr->tx_tag, map, m_headp, 143779251f5eSSepherosa Ziehau segs, maxsegs, &nsegs, BUS_DMA_NOWAIT); 143879251f5eSSepherosa Ziehau if (__predict_false(error)) { 143979251f5eSSepherosa Ziehau m_freem(*m_headp); 144079251f5eSSepherosa Ziehau *m_headp = NULL; 144179251f5eSSepherosa Ziehau return error; 144279251f5eSSepherosa Ziehau } 144379251f5eSSepherosa Ziehau bus_dmamap_sync(txr->tx_tag, map, BUS_DMASYNC_PREWRITE); 144479251f5eSSepherosa Ziehau 144579251f5eSSepherosa Ziehau m_head = *m_headp; 144679251f5eSSepherosa Ziehau 144779251f5eSSepherosa Ziehau /* 144879251f5eSSepherosa Ziehau * Set up the appropriate offload context if requested, 144979251f5eSSepherosa Ziehau * this may consume one TX descriptor. 145079251f5eSSepherosa Ziehau */ 145179251f5eSSepherosa Ziehau if (ix_tx_ctx_setup(txr, m_head, &cmd_type_len, &olinfo_status)) { 145279251f5eSSepherosa Ziehau (*segs_used)++; 145379251f5eSSepherosa Ziehau txr->tx_nsegs++; 145479251f5eSSepherosa Ziehau } 145579251f5eSSepherosa Ziehau 145679251f5eSSepherosa Ziehau *segs_used += nsegs; 145779251f5eSSepherosa Ziehau txr->tx_nsegs += nsegs; 145879251f5eSSepherosa Ziehau if (txr->tx_nsegs >= txr->tx_intr_nsegs) { 145979251f5eSSepherosa Ziehau /* 146079251f5eSSepherosa Ziehau * Report Status (RS) is turned on every intr_nsegs 146179251f5eSSepherosa Ziehau * descriptors (roughly). 146279251f5eSSepherosa Ziehau */ 146379251f5eSSepherosa Ziehau txr->tx_nsegs = 0; 146479251f5eSSepherosa Ziehau cmd_rs = IXGBE_TXD_CMD_RS; 146579251f5eSSepherosa Ziehau } 146679251f5eSSepherosa Ziehau 146779251f5eSSepherosa Ziehau i = txr->tx_next_avail; 146879251f5eSSepherosa Ziehau for (j = 0; j < nsegs; j++) { 146979251f5eSSepherosa Ziehau bus_size_t seglen; 147079251f5eSSepherosa Ziehau bus_addr_t segaddr; 147179251f5eSSepherosa Ziehau 147279251f5eSSepherosa Ziehau txbuf = &txr->tx_buf[i]; 147379251f5eSSepherosa Ziehau txd = &txr->tx_base[i]; 147479251f5eSSepherosa Ziehau seglen = segs[j].ds_len; 147579251f5eSSepherosa Ziehau segaddr = htole64(segs[j].ds_addr); 147679251f5eSSepherosa Ziehau 147779251f5eSSepherosa Ziehau txd->read.buffer_addr = segaddr; 147879251f5eSSepherosa Ziehau txd->read.cmd_type_len = htole32(IXGBE_TXD_CMD_IFCS | 147979251f5eSSepherosa Ziehau cmd_type_len |seglen); 148079251f5eSSepherosa Ziehau txd->read.olinfo_status = htole32(olinfo_status); 148179251f5eSSepherosa Ziehau 148279251f5eSSepherosa Ziehau if (++i == txr->tx_ndesc) 148379251f5eSSepherosa Ziehau i = 0; 148479251f5eSSepherosa Ziehau } 148579251f5eSSepherosa Ziehau txd->read.cmd_type_len |= htole32(IXGBE_TXD_CMD_EOP | cmd_rs); 148679251f5eSSepherosa Ziehau 148779251f5eSSepherosa Ziehau txr->tx_avail -= nsegs; 148879251f5eSSepherosa Ziehau txr->tx_next_avail = i; 148982db96e9SSepherosa Ziehau txr->tx_nmbuf++; 149079251f5eSSepherosa Ziehau 149179251f5eSSepherosa Ziehau txbuf->m_head = m_head; 149279251f5eSSepherosa Ziehau txr->tx_buf[first].map = txbuf->map; 149379251f5eSSepherosa Ziehau txbuf->map = map; 149479251f5eSSepherosa Ziehau 149579251f5eSSepherosa Ziehau /* 149679251f5eSSepherosa Ziehau * Defer TDT updating, until enough descrptors are setup 149779251f5eSSepherosa Ziehau */ 149879251f5eSSepherosa Ziehau *idx = i; 149979251f5eSSepherosa Ziehau 150079251f5eSSepherosa Ziehau return 0; 150179251f5eSSepherosa Ziehau } 150279251f5eSSepherosa Ziehau 150379251f5eSSepherosa Ziehau static void 150479251f5eSSepherosa Ziehau ix_set_promisc(struct ix_softc *sc) 150579251f5eSSepherosa Ziehau { 150679251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 150779251f5eSSepherosa Ziehau uint32_t reg_rctl; 150879251f5eSSepherosa Ziehau int mcnt = 0; 150979251f5eSSepherosa Ziehau 151079251f5eSSepherosa Ziehau reg_rctl = IXGBE_READ_REG(&sc->hw, IXGBE_FCTRL); 151179251f5eSSepherosa Ziehau reg_rctl &= ~IXGBE_FCTRL_UPE; 151279251f5eSSepherosa Ziehau if (ifp->if_flags & IFF_ALLMULTI) { 151379251f5eSSepherosa Ziehau mcnt = IX_MAX_MCASTADDR; 151479251f5eSSepherosa Ziehau } else { 151579251f5eSSepherosa Ziehau struct ifmultiaddr *ifma; 151679251f5eSSepherosa Ziehau 151779251f5eSSepherosa Ziehau TAILQ_FOREACH(ifma, &ifp->if_multiaddrs, ifma_link) { 151879251f5eSSepherosa Ziehau if (ifma->ifma_addr->sa_family != AF_LINK) 151979251f5eSSepherosa Ziehau continue; 152079251f5eSSepherosa Ziehau if (mcnt == IX_MAX_MCASTADDR) 152179251f5eSSepherosa Ziehau break; 152279251f5eSSepherosa Ziehau mcnt++; 152379251f5eSSepherosa Ziehau } 152479251f5eSSepherosa Ziehau } 152579251f5eSSepherosa Ziehau if (mcnt < IX_MAX_MCASTADDR) 152679251f5eSSepherosa Ziehau reg_rctl &= ~IXGBE_FCTRL_MPE; 152779251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_FCTRL, reg_rctl); 152879251f5eSSepherosa Ziehau 152979251f5eSSepherosa Ziehau if (ifp->if_flags & IFF_PROMISC) { 153079251f5eSSepherosa Ziehau reg_rctl |= IXGBE_FCTRL_UPE | IXGBE_FCTRL_MPE; 153179251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_FCTRL, reg_rctl); 153279251f5eSSepherosa Ziehau } else if (ifp->if_flags & IFF_ALLMULTI) { 153379251f5eSSepherosa Ziehau reg_rctl |= IXGBE_FCTRL_MPE; 153479251f5eSSepherosa Ziehau reg_rctl &= ~IXGBE_FCTRL_UPE; 153579251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_FCTRL, reg_rctl); 153679251f5eSSepherosa Ziehau } 153779251f5eSSepherosa Ziehau } 153879251f5eSSepherosa Ziehau 153979251f5eSSepherosa Ziehau static void 154079251f5eSSepherosa Ziehau ix_set_multi(struct ix_softc *sc) 154179251f5eSSepherosa Ziehau { 154279251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 154379251f5eSSepherosa Ziehau struct ifmultiaddr *ifma; 154479251f5eSSepherosa Ziehau uint32_t fctrl; 154579251f5eSSepherosa Ziehau uint8_t *mta; 154679251f5eSSepherosa Ziehau int mcnt = 0; 154779251f5eSSepherosa Ziehau 154879251f5eSSepherosa Ziehau mta = sc->mta; 154979251f5eSSepherosa Ziehau bzero(mta, IXGBE_ETH_LENGTH_OF_ADDRESS * IX_MAX_MCASTADDR); 155079251f5eSSepherosa Ziehau 155179251f5eSSepherosa Ziehau TAILQ_FOREACH(ifma, &ifp->if_multiaddrs, ifma_link) { 155279251f5eSSepherosa Ziehau if (ifma->ifma_addr->sa_family != AF_LINK) 155379251f5eSSepherosa Ziehau continue; 155479251f5eSSepherosa Ziehau if (mcnt == IX_MAX_MCASTADDR) 155579251f5eSSepherosa Ziehau break; 155679251f5eSSepherosa Ziehau bcopy(LLADDR((struct sockaddr_dl *)ifma->ifma_addr), 155779251f5eSSepherosa Ziehau &mta[mcnt * IXGBE_ETH_LENGTH_OF_ADDRESS], 155879251f5eSSepherosa Ziehau IXGBE_ETH_LENGTH_OF_ADDRESS); 155979251f5eSSepherosa Ziehau mcnt++; 156079251f5eSSepherosa Ziehau } 156179251f5eSSepherosa Ziehau 156279251f5eSSepherosa Ziehau fctrl = IXGBE_READ_REG(&sc->hw, IXGBE_FCTRL); 156379251f5eSSepherosa Ziehau fctrl |= (IXGBE_FCTRL_UPE | IXGBE_FCTRL_MPE); 156479251f5eSSepherosa Ziehau if (ifp->if_flags & IFF_PROMISC) { 156579251f5eSSepherosa Ziehau fctrl |= IXGBE_FCTRL_UPE | IXGBE_FCTRL_MPE; 156679251f5eSSepherosa Ziehau } else if (mcnt >= IX_MAX_MCASTADDR || (ifp->if_flags & IFF_ALLMULTI)) { 156779251f5eSSepherosa Ziehau fctrl |= IXGBE_FCTRL_MPE; 156879251f5eSSepherosa Ziehau fctrl &= ~IXGBE_FCTRL_UPE; 156979251f5eSSepherosa Ziehau } else { 157079251f5eSSepherosa Ziehau fctrl &= ~(IXGBE_FCTRL_UPE | IXGBE_FCTRL_MPE); 157179251f5eSSepherosa Ziehau } 157279251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_FCTRL, fctrl); 157379251f5eSSepherosa Ziehau 157479251f5eSSepherosa Ziehau if (mcnt < IX_MAX_MCASTADDR) { 157579251f5eSSepherosa Ziehau ixgbe_update_mc_addr_list(&sc->hw, 157679251f5eSSepherosa Ziehau mta, mcnt, ix_mc_array_itr, TRUE); 157779251f5eSSepherosa Ziehau } 157879251f5eSSepherosa Ziehau } 157979251f5eSSepherosa Ziehau 158079251f5eSSepherosa Ziehau /* 158179251f5eSSepherosa Ziehau * This is an iterator function now needed by the multicast 158279251f5eSSepherosa Ziehau * shared code. It simply feeds the shared code routine the 158379251f5eSSepherosa Ziehau * addresses in the array of ix_set_multi() one by one. 158479251f5eSSepherosa Ziehau */ 158579251f5eSSepherosa Ziehau static uint8_t * 158679251f5eSSepherosa Ziehau ix_mc_array_itr(struct ixgbe_hw *hw, uint8_t **update_ptr, uint32_t *vmdq) 158779251f5eSSepherosa Ziehau { 158879251f5eSSepherosa Ziehau uint8_t *addr = *update_ptr; 158979251f5eSSepherosa Ziehau uint8_t *newptr; 159079251f5eSSepherosa Ziehau *vmdq = 0; 159179251f5eSSepherosa Ziehau 159279251f5eSSepherosa Ziehau newptr = addr + IXGBE_ETH_LENGTH_OF_ADDRESS; 159379251f5eSSepherosa Ziehau *update_ptr = newptr; 159479251f5eSSepherosa Ziehau return addr; 159579251f5eSSepherosa Ziehau } 159679251f5eSSepherosa Ziehau 159779251f5eSSepherosa Ziehau static void 159879251f5eSSepherosa Ziehau ix_timer(void *arg) 159979251f5eSSepherosa Ziehau { 160079251f5eSSepherosa Ziehau struct ix_softc *sc = arg; 160179251f5eSSepherosa Ziehau 160279251f5eSSepherosa Ziehau lwkt_serialize_enter(&sc->main_serialize); 160379251f5eSSepherosa Ziehau 160479251f5eSSepherosa Ziehau if ((sc->arpcom.ac_if.if_flags & IFF_RUNNING) == 0) { 160579251f5eSSepherosa Ziehau lwkt_serialize_exit(&sc->main_serialize); 160679251f5eSSepherosa Ziehau return; 160779251f5eSSepherosa Ziehau } 160879251f5eSSepherosa Ziehau 160979251f5eSSepherosa Ziehau /* Check for pluggable optics */ 161079251f5eSSepherosa Ziehau if (sc->sfp_probe) { 161179251f5eSSepherosa Ziehau if (!ix_sfp_probe(sc)) 161279251f5eSSepherosa Ziehau goto done; /* Nothing to do */ 161379251f5eSSepherosa Ziehau } 161479251f5eSSepherosa Ziehau 161579251f5eSSepherosa Ziehau ix_update_link_status(sc); 161679251f5eSSepherosa Ziehau ix_update_stats(sc); 161779251f5eSSepherosa Ziehau 161879251f5eSSepherosa Ziehau done: 161979251f5eSSepherosa Ziehau callout_reset_bycpu(&sc->timer, hz, ix_timer, sc, sc->timer_cpuid); 162079251f5eSSepherosa Ziehau lwkt_serialize_exit(&sc->main_serialize); 162179251f5eSSepherosa Ziehau } 162279251f5eSSepherosa Ziehau 162379251f5eSSepherosa Ziehau static void 162479251f5eSSepherosa Ziehau ix_update_link_status(struct ix_softc *sc) 162579251f5eSSepherosa Ziehau { 162679251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 162779251f5eSSepherosa Ziehau 162879251f5eSSepherosa Ziehau if (sc->link_up) { 162979251f5eSSepherosa Ziehau if (sc->link_active == FALSE) { 163079251f5eSSepherosa Ziehau if (bootverbose) { 163179251f5eSSepherosa Ziehau if_printf(ifp, "Link is up %d Gbps %s\n", 163279251f5eSSepherosa Ziehau sc->link_speed == 128 ? 10 : 1, 163379251f5eSSepherosa Ziehau "Full Duplex"); 163479251f5eSSepherosa Ziehau } 163579251f5eSSepherosa Ziehau 1636060fa21cSSepherosa Ziehau /* 1637060fa21cSSepherosa Ziehau * Update any Flow Control changes 1638060fa21cSSepherosa Ziehau */ 163979251f5eSSepherosa Ziehau ixgbe_fc_enable(&sc->hw); 1640060fa21cSSepherosa Ziehau /* MUST after ixgbe_fc_enable() */ 1641060fa21cSSepherosa Ziehau if (sc->rx_ring_inuse > 1) { 1642060fa21cSSepherosa Ziehau switch (sc->hw.fc.current_mode) { 1643060fa21cSSepherosa Ziehau case ixgbe_fc_rx_pause: 1644060fa21cSSepherosa Ziehau case ixgbe_fc_tx_pause: 1645060fa21cSSepherosa Ziehau case ixgbe_fc_full: 1646060fa21cSSepherosa Ziehau ix_disable_rx_drop(sc); 1647060fa21cSSepherosa Ziehau break; 1648060fa21cSSepherosa Ziehau 1649060fa21cSSepherosa Ziehau case ixgbe_fc_none: 1650060fa21cSSepherosa Ziehau ix_enable_rx_drop(sc); 1651060fa21cSSepherosa Ziehau break; 1652060fa21cSSepherosa Ziehau 1653060fa21cSSepherosa Ziehau default: 1654060fa21cSSepherosa Ziehau break; 1655060fa21cSSepherosa Ziehau } 1656060fa21cSSepherosa Ziehau } 1657060fa21cSSepherosa Ziehau 165863d483cdSSepherosa Ziehau /* Update DMA coalescing config */ 165963d483cdSSepherosa Ziehau ix_config_dmac(sc); 166063d483cdSSepherosa Ziehau 1661060fa21cSSepherosa Ziehau sc->link_active = TRUE; 166279251f5eSSepherosa Ziehau 166379251f5eSSepherosa Ziehau ifp->if_link_state = LINK_STATE_UP; 166479251f5eSSepherosa Ziehau if_link_state_change(ifp); 166579251f5eSSepherosa Ziehau } 166679251f5eSSepherosa Ziehau } else { /* Link down */ 166779251f5eSSepherosa Ziehau if (sc->link_active == TRUE) { 166879251f5eSSepherosa Ziehau if (bootverbose) 166979251f5eSSepherosa Ziehau if_printf(ifp, "Link is Down\n"); 167079251f5eSSepherosa Ziehau ifp->if_link_state = LINK_STATE_DOWN; 167179251f5eSSepherosa Ziehau if_link_state_change(ifp); 167279251f5eSSepherosa Ziehau 167379251f5eSSepherosa Ziehau sc->link_active = FALSE; 167479251f5eSSepherosa Ziehau } 167579251f5eSSepherosa Ziehau } 167679251f5eSSepherosa Ziehau } 167779251f5eSSepherosa Ziehau 167879251f5eSSepherosa Ziehau static void 167979251f5eSSepherosa Ziehau ix_stop(struct ix_softc *sc) 168079251f5eSSepherosa Ziehau { 168179251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 168279251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 168379251f5eSSepherosa Ziehau int i; 168479251f5eSSepherosa Ziehau 168579251f5eSSepherosa Ziehau ASSERT_IFNET_SERIALIZED_ALL(ifp); 168679251f5eSSepherosa Ziehau 168779251f5eSSepherosa Ziehau ix_disable_intr(sc); 168879251f5eSSepherosa Ziehau callout_stop(&sc->timer); 168979251f5eSSepherosa Ziehau 169079251f5eSSepherosa Ziehau ifp->if_flags &= ~IFF_RUNNING; 169179251f5eSSepherosa Ziehau for (i = 0; i < sc->tx_ring_cnt; ++i) { 16924a648aefSSepherosa Ziehau struct ix_tx_ring *txr = &sc->tx_rings[i]; 16934a648aefSSepherosa Ziehau 16944a648aefSSepherosa Ziehau ifsq_clr_oactive(txr->tx_ifsq); 16954a648aefSSepherosa Ziehau ifsq_watchdog_stop(&txr->tx_watchdog); 16964a648aefSSepherosa Ziehau txr->tx_flags &= ~IX_TXFLAG_ENABLED; 169782db96e9SSepherosa Ziehau 169882db96e9SSepherosa Ziehau txr->tx_running = 0; 169982db96e9SSepherosa Ziehau callout_stop(&txr->tx_gc_timer); 170079251f5eSSepherosa Ziehau } 170179251f5eSSepherosa Ziehau 170279251f5eSSepherosa Ziehau ixgbe_reset_hw(hw); 170379251f5eSSepherosa Ziehau hw->adapter_stopped = FALSE; 170479251f5eSSepherosa Ziehau ixgbe_stop_adapter(hw); 170579251f5eSSepherosa Ziehau if (hw->mac.type == ixgbe_mac_82599EB) 170679251f5eSSepherosa Ziehau ixgbe_stop_mac_link_on_d3_82599(hw); 170779251f5eSSepherosa Ziehau /* Turn off the laser - noop with no optics */ 170879251f5eSSepherosa Ziehau ixgbe_disable_tx_laser(hw); 170979251f5eSSepherosa Ziehau 171079251f5eSSepherosa Ziehau /* Update the stack */ 171179251f5eSSepherosa Ziehau sc->link_up = FALSE; 171279251f5eSSepherosa Ziehau ix_update_link_status(sc); 171379251f5eSSepherosa Ziehau 171479251f5eSSepherosa Ziehau /* Reprogram the RAR[0] in case user changed it. */ 171579251f5eSSepherosa Ziehau ixgbe_set_rar(hw, 0, hw->mac.addr, 0, IXGBE_RAH_AV); 171679251f5eSSepherosa Ziehau 171779251f5eSSepherosa Ziehau for (i = 0; i < sc->tx_ring_cnt; ++i) 171879251f5eSSepherosa Ziehau ix_free_tx_ring(&sc->tx_rings[i]); 171979251f5eSSepherosa Ziehau 172079251f5eSSepherosa Ziehau for (i = 0; i < sc->rx_ring_cnt; ++i) 172179251f5eSSepherosa Ziehau ix_free_rx_ring(&sc->rx_rings[i]); 172279251f5eSSepherosa Ziehau } 172379251f5eSSepherosa Ziehau 172479251f5eSSepherosa Ziehau static void 172579251f5eSSepherosa Ziehau ix_setup_ifp(struct ix_softc *sc) 172679251f5eSSepherosa Ziehau { 172779251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 172879251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 172979251f5eSSepherosa Ziehau int i; 173079251f5eSSepherosa Ziehau 173179251f5eSSepherosa Ziehau ifp->if_baudrate = IF_Gbps(10UL); 173279251f5eSSepherosa Ziehau 173379251f5eSSepherosa Ziehau ifp->if_softc = sc; 173479251f5eSSepherosa Ziehau ifp->if_flags = IFF_BROADCAST | IFF_SIMPLEX | IFF_MULTICAST; 173579251f5eSSepherosa Ziehau ifp->if_init = ix_init; 173679251f5eSSepherosa Ziehau ifp->if_ioctl = ix_ioctl; 173779251f5eSSepherosa Ziehau ifp->if_start = ix_start; 173879251f5eSSepherosa Ziehau ifp->if_serialize = ix_serialize; 173979251f5eSSepherosa Ziehau ifp->if_deserialize = ix_deserialize; 174079251f5eSSepherosa Ziehau ifp->if_tryserialize = ix_tryserialize; 174179251f5eSSepherosa Ziehau #ifdef INVARIANTS 174279251f5eSSepherosa Ziehau ifp->if_serialize_assert = ix_serialize_assert; 174379251f5eSSepherosa Ziehau #endif 17444a648aefSSepherosa Ziehau #ifdef IFPOLL_ENABLE 17454a648aefSSepherosa Ziehau ifp->if_npoll = ix_npoll; 17464a648aefSSepherosa Ziehau #endif 174779251f5eSSepherosa Ziehau 1748189a0ff3SSepherosa Ziehau /* Increase TSO burst length */ 1749189a0ff3SSepherosa Ziehau ifp->if_tsolen = (8 * ETHERMTU); 1750189a0ff3SSepherosa Ziehau 175114929979SSepherosa Ziehau ifp->if_nmbclusters = sc->rx_ring_cnt * sc->rx_rings[0].rx_ndesc; 175214929979SSepherosa Ziehau ifp->if_nmbjclusters = ifp->if_nmbclusters; 175314929979SSepherosa Ziehau 175479251f5eSSepherosa Ziehau ifq_set_maxlen(&ifp->if_snd, sc->tx_rings[0].tx_ndesc - 2); 175579251f5eSSepherosa Ziehau ifq_set_ready(&ifp->if_snd); 175679251f5eSSepherosa Ziehau ifq_set_subq_cnt(&ifp->if_snd, sc->tx_ring_cnt); 175779251f5eSSepherosa Ziehau 17583c37d13bSSepherosa Ziehau ifp->if_mapsubq = ifq_mapsubq_modulo; 17593c37d13bSSepherosa Ziehau ifq_set_subq_divisor(&ifp->if_snd, 1); 176079251f5eSSepherosa Ziehau 176179251f5eSSepherosa Ziehau ether_ifattach(ifp, hw->mac.addr, NULL); 176279251f5eSSepherosa Ziehau 176379251f5eSSepherosa Ziehau ifp->if_capabilities = 176479251f5eSSepherosa Ziehau IFCAP_HWCSUM | IFCAP_TSO | IFCAP_VLAN_HWTAGGING | IFCAP_VLAN_MTU; 176579251f5eSSepherosa Ziehau if (IX_ENABLE_HWRSS(sc)) 176679251f5eSSepherosa Ziehau ifp->if_capabilities |= IFCAP_RSS; 176779251f5eSSepherosa Ziehau ifp->if_capenable = ifp->if_capabilities; 176879251f5eSSepherosa Ziehau ifp->if_hwassist = CSUM_OFFLOAD | CSUM_TSO; 176979251f5eSSepherosa Ziehau 177079251f5eSSepherosa Ziehau /* 177179251f5eSSepherosa Ziehau * Tell the upper layer(s) we support long frames. 177279251f5eSSepherosa Ziehau */ 177379251f5eSSepherosa Ziehau ifp->if_data.ifi_hdrlen = sizeof(struct ether_vlan_header); 177479251f5eSSepherosa Ziehau 177579251f5eSSepherosa Ziehau /* Setup TX rings and subqueues */ 177679251f5eSSepherosa Ziehau for (i = 0; i < sc->tx_ring_cnt; ++i) { 177779251f5eSSepherosa Ziehau struct ifaltq_subque *ifsq = ifq_get_subq(&ifp->if_snd, i); 177879251f5eSSepherosa Ziehau struct ix_tx_ring *txr = &sc->tx_rings[i]; 177979251f5eSSepherosa Ziehau 178079251f5eSSepherosa Ziehau ifsq_set_cpuid(ifsq, txr->tx_intr_cpuid); 178179251f5eSSepherosa Ziehau ifsq_set_priv(ifsq, txr); 178279251f5eSSepherosa Ziehau ifsq_set_hw_serialize(ifsq, &txr->tx_serialize); 178379251f5eSSepherosa Ziehau txr->tx_ifsq = ifsq; 178479251f5eSSepherosa Ziehau 178579251f5eSSepherosa Ziehau ifsq_watchdog_init(&txr->tx_watchdog, ifsq, ix_watchdog); 178679251f5eSSepherosa Ziehau } 178779251f5eSSepherosa Ziehau 178863d483cdSSepherosa Ziehau /* Specify the media types supported by this adapter */ 178963d483cdSSepherosa Ziehau ix_init_media(sc); 179079251f5eSSepherosa Ziehau } 179179251f5eSSepherosa Ziehau 179279251f5eSSepherosa Ziehau static boolean_t 179379251f5eSSepherosa Ziehau ix_is_sfp(const struct ixgbe_hw *hw) 179479251f5eSSepherosa Ziehau { 179579251f5eSSepherosa Ziehau switch (hw->phy.type) { 179679251f5eSSepherosa Ziehau case ixgbe_phy_sfp_avago: 179779251f5eSSepherosa Ziehau case ixgbe_phy_sfp_ftl: 179879251f5eSSepherosa Ziehau case ixgbe_phy_sfp_intel: 179979251f5eSSepherosa Ziehau case ixgbe_phy_sfp_unknown: 180079251f5eSSepherosa Ziehau case ixgbe_phy_sfp_passive_tyco: 180179251f5eSSepherosa Ziehau case ixgbe_phy_sfp_passive_unknown: 180263d483cdSSepherosa Ziehau case ixgbe_phy_qsfp_passive_unknown: 180363d483cdSSepherosa Ziehau case ixgbe_phy_qsfp_active_unknown: 180463d483cdSSepherosa Ziehau case ixgbe_phy_qsfp_intel: 180563d483cdSSepherosa Ziehau case ixgbe_phy_qsfp_unknown: 180679251f5eSSepherosa Ziehau return TRUE; 180779251f5eSSepherosa Ziehau default: 180879251f5eSSepherosa Ziehau return FALSE; 180979251f5eSSepherosa Ziehau } 181079251f5eSSepherosa Ziehau } 181179251f5eSSepherosa Ziehau 181279251f5eSSepherosa Ziehau static void 181379251f5eSSepherosa Ziehau ix_config_link(struct ix_softc *sc) 181479251f5eSSepherosa Ziehau { 181579251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 181679251f5eSSepherosa Ziehau boolean_t sfp; 181779251f5eSSepherosa Ziehau 181879251f5eSSepherosa Ziehau sfp = ix_is_sfp(hw); 181979251f5eSSepherosa Ziehau if (sfp) { 182079251f5eSSepherosa Ziehau if (hw->phy.multispeed_fiber) { 182179251f5eSSepherosa Ziehau hw->mac.ops.setup_sfp(hw); 182279251f5eSSepherosa Ziehau ixgbe_enable_tx_laser(hw); 182379251f5eSSepherosa Ziehau ix_handle_msf(sc); 182479251f5eSSepherosa Ziehau } else { 182579251f5eSSepherosa Ziehau ix_handle_mod(sc); 182679251f5eSSepherosa Ziehau } 182779251f5eSSepherosa Ziehau } else { 182879251f5eSSepherosa Ziehau uint32_t autoneg, err = 0; 182979251f5eSSepherosa Ziehau 183079251f5eSSepherosa Ziehau if (hw->mac.ops.check_link != NULL) { 183179251f5eSSepherosa Ziehau err = ixgbe_check_link(hw, &sc->link_speed, 183279251f5eSSepherosa Ziehau &sc->link_up, FALSE); 183379251f5eSSepherosa Ziehau if (err) 183479251f5eSSepherosa Ziehau return; 183579251f5eSSepherosa Ziehau } 183679251f5eSSepherosa Ziehau 183763d483cdSSepherosa Ziehau if (sc->advspeed != IXGBE_LINK_SPEED_UNKNOWN) 183863d483cdSSepherosa Ziehau autoneg = sc->advspeed; 183963d483cdSSepherosa Ziehau else 184079251f5eSSepherosa Ziehau autoneg = hw->phy.autoneg_advertised; 184179251f5eSSepherosa Ziehau if (!autoneg && hw->mac.ops.get_link_capabilities != NULL) { 184279251f5eSSepherosa Ziehau bool negotiate; 184379251f5eSSepherosa Ziehau 184479251f5eSSepherosa Ziehau err = hw->mac.ops.get_link_capabilities(hw, 184579251f5eSSepherosa Ziehau &autoneg, &negotiate); 184679251f5eSSepherosa Ziehau if (err) 184779251f5eSSepherosa Ziehau return; 184879251f5eSSepherosa Ziehau } 184979251f5eSSepherosa Ziehau 185079251f5eSSepherosa Ziehau if (hw->mac.ops.setup_link != NULL) { 185179251f5eSSepherosa Ziehau err = hw->mac.ops.setup_link(hw, 185279251f5eSSepherosa Ziehau autoneg, sc->link_up); 185379251f5eSSepherosa Ziehau if (err) 185479251f5eSSepherosa Ziehau return; 185579251f5eSSepherosa Ziehau } 185679251f5eSSepherosa Ziehau } 185779251f5eSSepherosa Ziehau } 185879251f5eSSepherosa Ziehau 185979251f5eSSepherosa Ziehau static int 186079251f5eSSepherosa Ziehau ix_alloc_rings(struct ix_softc *sc) 186179251f5eSSepherosa Ziehau { 186279251f5eSSepherosa Ziehau int error, i; 186379251f5eSSepherosa Ziehau 186479251f5eSSepherosa Ziehau /* 186579251f5eSSepherosa Ziehau * Create top level busdma tag 186679251f5eSSepherosa Ziehau */ 186779251f5eSSepherosa Ziehau error = bus_dma_tag_create(NULL, 1, 0, 186879251f5eSSepherosa Ziehau BUS_SPACE_MAXADDR, BUS_SPACE_MAXADDR, NULL, NULL, 186979251f5eSSepherosa Ziehau BUS_SPACE_MAXSIZE_32BIT, 0, BUS_SPACE_MAXSIZE_32BIT, 0, 187079251f5eSSepherosa Ziehau &sc->parent_tag); 187179251f5eSSepherosa Ziehau if (error) { 187279251f5eSSepherosa Ziehau device_printf(sc->dev, "could not create top level DMA tag\n"); 187379251f5eSSepherosa Ziehau return error; 187479251f5eSSepherosa Ziehau } 187579251f5eSSepherosa Ziehau 187679251f5eSSepherosa Ziehau /* 187779251f5eSSepherosa Ziehau * Allocate TX descriptor rings and buffers 187879251f5eSSepherosa Ziehau */ 187979251f5eSSepherosa Ziehau sc->tx_rings = kmalloc_cachealign( 188079251f5eSSepherosa Ziehau sizeof(struct ix_tx_ring) * sc->tx_ring_cnt, 188179251f5eSSepherosa Ziehau M_DEVBUF, M_WAITOK | M_ZERO); 188279251f5eSSepherosa Ziehau for (i = 0; i < sc->tx_ring_cnt; ++i) { 188379251f5eSSepherosa Ziehau struct ix_tx_ring *txr = &sc->tx_rings[i]; 188479251f5eSSepherosa Ziehau 188579251f5eSSepherosa Ziehau txr->tx_sc = sc; 188679251f5eSSepherosa Ziehau txr->tx_idx = i; 1887189a0ff3SSepherosa Ziehau txr->tx_intr_vec = -1; 18883c37d13bSSepherosa Ziehau txr->tx_intr_cpuid = -1; 188979251f5eSSepherosa Ziehau lwkt_serialize_init(&txr->tx_serialize); 189082db96e9SSepherosa Ziehau callout_init_mp(&txr->tx_gc_timer); 189179251f5eSSepherosa Ziehau 189279251f5eSSepherosa Ziehau error = ix_create_tx_ring(txr); 189379251f5eSSepherosa Ziehau if (error) 189479251f5eSSepherosa Ziehau return error; 189579251f5eSSepherosa Ziehau } 189679251f5eSSepherosa Ziehau 189779251f5eSSepherosa Ziehau /* 189879251f5eSSepherosa Ziehau * Allocate RX descriptor rings and buffers 189979251f5eSSepherosa Ziehau */ 190079251f5eSSepherosa Ziehau sc->rx_rings = kmalloc_cachealign( 190179251f5eSSepherosa Ziehau sizeof(struct ix_rx_ring) * sc->rx_ring_cnt, 190279251f5eSSepherosa Ziehau M_DEVBUF, M_WAITOK | M_ZERO); 190379251f5eSSepherosa Ziehau for (i = 0; i < sc->rx_ring_cnt; ++i) { 190479251f5eSSepherosa Ziehau struct ix_rx_ring *rxr = &sc->rx_rings[i]; 190579251f5eSSepherosa Ziehau 190679251f5eSSepherosa Ziehau rxr->rx_sc = sc; 190779251f5eSSepherosa Ziehau rxr->rx_idx = i; 1908189a0ff3SSepherosa Ziehau rxr->rx_intr_vec = -1; 190979251f5eSSepherosa Ziehau lwkt_serialize_init(&rxr->rx_serialize); 191079251f5eSSepherosa Ziehau 191179251f5eSSepherosa Ziehau error = ix_create_rx_ring(rxr); 191279251f5eSSepherosa Ziehau if (error) 191379251f5eSSepherosa Ziehau return error; 191479251f5eSSepherosa Ziehau } 191579251f5eSSepherosa Ziehau 191679251f5eSSepherosa Ziehau return 0; 191779251f5eSSepherosa Ziehau } 191879251f5eSSepherosa Ziehau 191979251f5eSSepherosa Ziehau static int 192079251f5eSSepherosa Ziehau ix_create_tx_ring(struct ix_tx_ring *txr) 192179251f5eSSepherosa Ziehau { 192279251f5eSSepherosa Ziehau int error, i, tsize, ntxd; 192379251f5eSSepherosa Ziehau 192479251f5eSSepherosa Ziehau /* 192579251f5eSSepherosa Ziehau * Validate number of transmit descriptors. It must not exceed 192679251f5eSSepherosa Ziehau * hardware maximum, and must be multiple of IX_DBA_ALIGN. 192779251f5eSSepherosa Ziehau */ 192879251f5eSSepherosa Ziehau ntxd = device_getenv_int(txr->tx_sc->dev, "txd", ix_txd); 192979251f5eSSepherosa Ziehau if (((ntxd * sizeof(union ixgbe_adv_tx_desc)) % IX_DBA_ALIGN) != 0 || 193079251f5eSSepherosa Ziehau ntxd < IX_MIN_TXD || ntxd > IX_MAX_TXD) { 193179251f5eSSepherosa Ziehau device_printf(txr->tx_sc->dev, 193279251f5eSSepherosa Ziehau "Using %d TX descriptors instead of %d!\n", 193379251f5eSSepherosa Ziehau IX_DEF_TXD, ntxd); 193479251f5eSSepherosa Ziehau txr->tx_ndesc = IX_DEF_TXD; 193579251f5eSSepherosa Ziehau } else { 193679251f5eSSepherosa Ziehau txr->tx_ndesc = ntxd; 193779251f5eSSepherosa Ziehau } 193879251f5eSSepherosa Ziehau 193979251f5eSSepherosa Ziehau /* 194079251f5eSSepherosa Ziehau * Allocate TX head write-back buffer 194179251f5eSSepherosa Ziehau */ 194279251f5eSSepherosa Ziehau txr->tx_hdr = bus_dmamem_coherent_any(txr->tx_sc->parent_tag, 194379251f5eSSepherosa Ziehau __VM_CACHELINE_SIZE, __VM_CACHELINE_SIZE, BUS_DMA_WAITOK, 194479251f5eSSepherosa Ziehau &txr->tx_hdr_dtag, &txr->tx_hdr_map, &txr->tx_hdr_paddr); 194579251f5eSSepherosa Ziehau if (txr->tx_hdr == NULL) { 194679251f5eSSepherosa Ziehau device_printf(txr->tx_sc->dev, 194779251f5eSSepherosa Ziehau "Unable to allocate TX head write-back buffer\n"); 194879251f5eSSepherosa Ziehau return ENOMEM; 194979251f5eSSepherosa Ziehau } 195079251f5eSSepherosa Ziehau 195179251f5eSSepherosa Ziehau /* 195279251f5eSSepherosa Ziehau * Allocate TX descriptor ring 195379251f5eSSepherosa Ziehau */ 195479251f5eSSepherosa Ziehau tsize = roundup2(txr->tx_ndesc * sizeof(union ixgbe_adv_tx_desc), 195579251f5eSSepherosa Ziehau IX_DBA_ALIGN); 195679251f5eSSepherosa Ziehau txr->tx_base = bus_dmamem_coherent_any(txr->tx_sc->parent_tag, 195779251f5eSSepherosa Ziehau IX_DBA_ALIGN, tsize, BUS_DMA_WAITOK | BUS_DMA_ZERO, 195879251f5eSSepherosa Ziehau &txr->tx_base_dtag, &txr->tx_base_map, &txr->tx_base_paddr); 195979251f5eSSepherosa Ziehau if (txr->tx_base == NULL) { 196079251f5eSSepherosa Ziehau device_printf(txr->tx_sc->dev, 196179251f5eSSepherosa Ziehau "Unable to allocate TX Descriptor memory\n"); 196279251f5eSSepherosa Ziehau return ENOMEM; 196379251f5eSSepherosa Ziehau } 196479251f5eSSepherosa Ziehau 196579251f5eSSepherosa Ziehau tsize = __VM_CACHELINE_ALIGN(sizeof(struct ix_tx_buf) * txr->tx_ndesc); 196679251f5eSSepherosa Ziehau txr->tx_buf = kmalloc_cachealign(tsize, M_DEVBUF, M_WAITOK | M_ZERO); 196779251f5eSSepherosa Ziehau 196879251f5eSSepherosa Ziehau /* 196979251f5eSSepherosa Ziehau * Create DMA tag for TX buffers 197079251f5eSSepherosa Ziehau */ 197179251f5eSSepherosa Ziehau error = bus_dma_tag_create(txr->tx_sc->parent_tag, 197279251f5eSSepherosa Ziehau 1, 0, /* alignment, bounds */ 197379251f5eSSepherosa Ziehau BUS_SPACE_MAXADDR, /* lowaddr */ 197479251f5eSSepherosa Ziehau BUS_SPACE_MAXADDR, /* highaddr */ 197579251f5eSSepherosa Ziehau NULL, NULL, /* filter, filterarg */ 197679251f5eSSepherosa Ziehau IX_TSO_SIZE, /* maxsize */ 197779251f5eSSepherosa Ziehau IX_MAX_SCATTER, /* nsegments */ 197879251f5eSSepherosa Ziehau PAGE_SIZE, /* maxsegsize */ 197979251f5eSSepherosa Ziehau BUS_DMA_WAITOK | BUS_DMA_ALLOCNOW | 198079251f5eSSepherosa Ziehau BUS_DMA_ONEBPAGE, /* flags */ 198179251f5eSSepherosa Ziehau &txr->tx_tag); 198279251f5eSSepherosa Ziehau if (error) { 198379251f5eSSepherosa Ziehau device_printf(txr->tx_sc->dev, 198479251f5eSSepherosa Ziehau "Unable to allocate TX DMA tag\n"); 198579251f5eSSepherosa Ziehau kfree(txr->tx_buf, M_DEVBUF); 198679251f5eSSepherosa Ziehau txr->tx_buf = NULL; 198779251f5eSSepherosa Ziehau return error; 198879251f5eSSepherosa Ziehau } 198979251f5eSSepherosa Ziehau 199079251f5eSSepherosa Ziehau /* 199179251f5eSSepherosa Ziehau * Create DMA maps for TX buffers 199279251f5eSSepherosa Ziehau */ 199379251f5eSSepherosa Ziehau for (i = 0; i < txr->tx_ndesc; ++i) { 199479251f5eSSepherosa Ziehau struct ix_tx_buf *txbuf = &txr->tx_buf[i]; 199579251f5eSSepherosa Ziehau 199679251f5eSSepherosa Ziehau error = bus_dmamap_create(txr->tx_tag, 199779251f5eSSepherosa Ziehau BUS_DMA_WAITOK | BUS_DMA_ONEBPAGE, &txbuf->map); 199879251f5eSSepherosa Ziehau if (error) { 199979251f5eSSepherosa Ziehau device_printf(txr->tx_sc->dev, 200079251f5eSSepherosa Ziehau "Unable to create TX DMA map\n"); 200179251f5eSSepherosa Ziehau ix_destroy_tx_ring(txr, i); 200279251f5eSSepherosa Ziehau return error; 200379251f5eSSepherosa Ziehau } 200479251f5eSSepherosa Ziehau } 200579251f5eSSepherosa Ziehau 200679251f5eSSepherosa Ziehau /* 200779251f5eSSepherosa Ziehau * Initialize various watermark 200879251f5eSSepherosa Ziehau */ 200979251f5eSSepherosa Ziehau txr->tx_wreg_nsegs = IX_DEF_TXWREG_NSEGS; 201079251f5eSSepherosa Ziehau txr->tx_intr_nsegs = txr->tx_ndesc / 16; 201179251f5eSSepherosa Ziehau 201279251f5eSSepherosa Ziehau return 0; 201379251f5eSSepherosa Ziehau } 201479251f5eSSepherosa Ziehau 201579251f5eSSepherosa Ziehau static void 201679251f5eSSepherosa Ziehau ix_destroy_tx_ring(struct ix_tx_ring *txr, int ndesc) 201779251f5eSSepherosa Ziehau { 201879251f5eSSepherosa Ziehau int i; 201979251f5eSSepherosa Ziehau 202079251f5eSSepherosa Ziehau if (txr->tx_hdr != NULL) { 202179251f5eSSepherosa Ziehau bus_dmamap_unload(txr->tx_hdr_dtag, txr->tx_hdr_map); 202279251f5eSSepherosa Ziehau bus_dmamem_free(txr->tx_hdr_dtag, 202379251f5eSSepherosa Ziehau __DEVOLATILE(void *, txr->tx_hdr), txr->tx_hdr_map); 202479251f5eSSepherosa Ziehau bus_dma_tag_destroy(txr->tx_hdr_dtag); 202579251f5eSSepherosa Ziehau txr->tx_hdr = NULL; 202679251f5eSSepherosa Ziehau } 202779251f5eSSepherosa Ziehau 202879251f5eSSepherosa Ziehau if (txr->tx_base != NULL) { 202979251f5eSSepherosa Ziehau bus_dmamap_unload(txr->tx_base_dtag, txr->tx_base_map); 203079251f5eSSepherosa Ziehau bus_dmamem_free(txr->tx_base_dtag, txr->tx_base, 203179251f5eSSepherosa Ziehau txr->tx_base_map); 203279251f5eSSepherosa Ziehau bus_dma_tag_destroy(txr->tx_base_dtag); 203379251f5eSSepherosa Ziehau txr->tx_base = NULL; 203479251f5eSSepherosa Ziehau } 203579251f5eSSepherosa Ziehau 203679251f5eSSepherosa Ziehau if (txr->tx_buf == NULL) 203779251f5eSSepherosa Ziehau return; 203879251f5eSSepherosa Ziehau 203979251f5eSSepherosa Ziehau for (i = 0; i < ndesc; ++i) { 204079251f5eSSepherosa Ziehau struct ix_tx_buf *txbuf = &txr->tx_buf[i]; 204179251f5eSSepherosa Ziehau 204279251f5eSSepherosa Ziehau KKASSERT(txbuf->m_head == NULL); 204379251f5eSSepherosa Ziehau bus_dmamap_destroy(txr->tx_tag, txbuf->map); 204479251f5eSSepherosa Ziehau } 204579251f5eSSepherosa Ziehau bus_dma_tag_destroy(txr->tx_tag); 204679251f5eSSepherosa Ziehau 204779251f5eSSepherosa Ziehau kfree(txr->tx_buf, M_DEVBUF); 204879251f5eSSepherosa Ziehau txr->tx_buf = NULL; 204979251f5eSSepherosa Ziehau } 205079251f5eSSepherosa Ziehau 205179251f5eSSepherosa Ziehau static void 205279251f5eSSepherosa Ziehau ix_init_tx_ring(struct ix_tx_ring *txr) 205379251f5eSSepherosa Ziehau { 205479251f5eSSepherosa Ziehau /* Clear the old ring contents */ 205579251f5eSSepherosa Ziehau bzero(txr->tx_base, sizeof(union ixgbe_adv_tx_desc) * txr->tx_ndesc); 205679251f5eSSepherosa Ziehau 205779251f5eSSepherosa Ziehau /* Clear TX head write-back buffer */ 205879251f5eSSepherosa Ziehau *(txr->tx_hdr) = 0; 205979251f5eSSepherosa Ziehau 206079251f5eSSepherosa Ziehau /* Reset indices */ 206179251f5eSSepherosa Ziehau txr->tx_next_avail = 0; 206279251f5eSSepherosa Ziehau txr->tx_next_clean = 0; 206379251f5eSSepherosa Ziehau txr->tx_nsegs = 0; 206482db96e9SSepherosa Ziehau txr->tx_nmbuf = 0; 206582db96e9SSepherosa Ziehau txr->tx_running = 0; 206679251f5eSSepherosa Ziehau 206779251f5eSSepherosa Ziehau /* Set number of descriptors available */ 206879251f5eSSepherosa Ziehau txr->tx_avail = txr->tx_ndesc; 20694a648aefSSepherosa Ziehau 20704a648aefSSepherosa Ziehau /* Enable this TX ring */ 20714a648aefSSepherosa Ziehau txr->tx_flags |= IX_TXFLAG_ENABLED; 207279251f5eSSepherosa Ziehau } 207379251f5eSSepherosa Ziehau 207479251f5eSSepherosa Ziehau static void 207579251f5eSSepherosa Ziehau ix_init_tx_unit(struct ix_softc *sc) 207679251f5eSSepherosa Ziehau { 207779251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 207879251f5eSSepherosa Ziehau int i; 207979251f5eSSepherosa Ziehau 208079251f5eSSepherosa Ziehau /* 208179251f5eSSepherosa Ziehau * Setup the Base and Length of the Tx Descriptor Ring 208279251f5eSSepherosa Ziehau */ 208379251f5eSSepherosa Ziehau for (i = 0; i < sc->tx_ring_inuse; ++i) { 208479251f5eSSepherosa Ziehau struct ix_tx_ring *txr = &sc->tx_rings[i]; 208579251f5eSSepherosa Ziehau uint64_t tdba = txr->tx_base_paddr; 208679251f5eSSepherosa Ziehau uint64_t hdr_paddr = txr->tx_hdr_paddr; 208779251f5eSSepherosa Ziehau uint32_t txctrl; 208879251f5eSSepherosa Ziehau 208979251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_TDBAL(i), (uint32_t)tdba); 209079251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_TDBAH(i), (uint32_t)(tdba >> 32)); 209179251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_TDLEN(i), 209279251f5eSSepherosa Ziehau txr->tx_ndesc * sizeof(union ixgbe_adv_tx_desc)); 209379251f5eSSepherosa Ziehau 209479251f5eSSepherosa Ziehau /* Setup the HW Tx Head and Tail descriptor pointers */ 209579251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_TDH(i), 0); 209679251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_TDT(i), 0); 209779251f5eSSepherosa Ziehau 209879251f5eSSepherosa Ziehau /* Disable TX head write-back relax ordering */ 209979251f5eSSepherosa Ziehau switch (hw->mac.type) { 210079251f5eSSepherosa Ziehau case ixgbe_mac_82598EB: 210179251f5eSSepherosa Ziehau txctrl = IXGBE_READ_REG(hw, IXGBE_DCA_TXCTRL(i)); 210279251f5eSSepherosa Ziehau break; 210379251f5eSSepherosa Ziehau case ixgbe_mac_82599EB: 210479251f5eSSepherosa Ziehau case ixgbe_mac_X540: 210579251f5eSSepherosa Ziehau default: 210679251f5eSSepherosa Ziehau txctrl = IXGBE_READ_REG(hw, IXGBE_DCA_TXCTRL_82599(i)); 210779251f5eSSepherosa Ziehau break; 210879251f5eSSepherosa Ziehau } 210979251f5eSSepherosa Ziehau txctrl &= ~IXGBE_DCA_TXCTRL_DESC_WRO_EN; 211079251f5eSSepherosa Ziehau switch (hw->mac.type) { 211179251f5eSSepherosa Ziehau case ixgbe_mac_82598EB: 211279251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_DCA_TXCTRL(i), txctrl); 211379251f5eSSepherosa Ziehau break; 211479251f5eSSepherosa Ziehau case ixgbe_mac_82599EB: 211579251f5eSSepherosa Ziehau case ixgbe_mac_X540: 211679251f5eSSepherosa Ziehau default: 211779251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_DCA_TXCTRL_82599(i), txctrl); 211879251f5eSSepherosa Ziehau break; 211979251f5eSSepherosa Ziehau } 212079251f5eSSepherosa Ziehau 212179251f5eSSepherosa Ziehau /* Enable TX head write-back */ 212279251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_TDWBAH(i), 212379251f5eSSepherosa Ziehau (uint32_t)(hdr_paddr >> 32)); 212479251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_TDWBAL(i), 212579251f5eSSepherosa Ziehau ((uint32_t)hdr_paddr) | IXGBE_TDWBAL_HEAD_WB_ENABLE); 212679251f5eSSepherosa Ziehau } 212779251f5eSSepherosa Ziehau 212879251f5eSSepherosa Ziehau if (hw->mac.type != ixgbe_mac_82598EB) { 212979251f5eSSepherosa Ziehau uint32_t dmatxctl, rttdcs; 213079251f5eSSepherosa Ziehau 213179251f5eSSepherosa Ziehau dmatxctl = IXGBE_READ_REG(hw, IXGBE_DMATXCTL); 213279251f5eSSepherosa Ziehau dmatxctl |= IXGBE_DMATXCTL_TE; 213379251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_DMATXCTL, dmatxctl); 213479251f5eSSepherosa Ziehau 213579251f5eSSepherosa Ziehau /* Disable arbiter to set MTQC */ 213679251f5eSSepherosa Ziehau rttdcs = IXGBE_READ_REG(hw, IXGBE_RTTDCS); 213779251f5eSSepherosa Ziehau rttdcs |= IXGBE_RTTDCS_ARBDIS; 213879251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RTTDCS, rttdcs); 213979251f5eSSepherosa Ziehau 214079251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_MTQC, IXGBE_MTQC_64Q_1PB); 214179251f5eSSepherosa Ziehau 214279251f5eSSepherosa Ziehau /* Reenable aribter */ 214379251f5eSSepherosa Ziehau rttdcs &= ~IXGBE_RTTDCS_ARBDIS; 214479251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RTTDCS, rttdcs); 214579251f5eSSepherosa Ziehau } 214679251f5eSSepherosa Ziehau } 214779251f5eSSepherosa Ziehau 214879251f5eSSepherosa Ziehau static int 214979251f5eSSepherosa Ziehau ix_tx_ctx_setup(struct ix_tx_ring *txr, const struct mbuf *mp, 215079251f5eSSepherosa Ziehau uint32_t *cmd_type_len, uint32_t *olinfo_status) 215179251f5eSSepherosa Ziehau { 215279251f5eSSepherosa Ziehau struct ixgbe_adv_tx_context_desc *TXD; 215379251f5eSSepherosa Ziehau uint32_t vlan_macip_lens = 0, type_tucmd_mlhl = 0; 215479251f5eSSepherosa Ziehau int ehdrlen, ip_hlen = 0, ctxd; 215579251f5eSSepherosa Ziehau boolean_t offload = TRUE; 215679251f5eSSepherosa Ziehau 215779251f5eSSepherosa Ziehau /* First check if TSO is to be used */ 215879251f5eSSepherosa Ziehau if (mp->m_pkthdr.csum_flags & CSUM_TSO) { 215979251f5eSSepherosa Ziehau return ix_tso_ctx_setup(txr, mp, 216079251f5eSSepherosa Ziehau cmd_type_len, olinfo_status); 216179251f5eSSepherosa Ziehau } 216279251f5eSSepherosa Ziehau 216379251f5eSSepherosa Ziehau if ((mp->m_pkthdr.csum_flags & CSUM_OFFLOAD) == 0) 216479251f5eSSepherosa Ziehau offload = FALSE; 216579251f5eSSepherosa Ziehau 216679251f5eSSepherosa Ziehau /* Indicate the whole packet as payload when not doing TSO */ 216779251f5eSSepherosa Ziehau *olinfo_status |= mp->m_pkthdr.len << IXGBE_ADVTXD_PAYLEN_SHIFT; 216879251f5eSSepherosa Ziehau 216979251f5eSSepherosa Ziehau /* 217079251f5eSSepherosa Ziehau * In advanced descriptors the vlan tag must be placed into the 217179251f5eSSepherosa Ziehau * context descriptor. Hence we need to make one even if not 217279251f5eSSepherosa Ziehau * doing checksum offloads. 217379251f5eSSepherosa Ziehau */ 217479251f5eSSepherosa Ziehau if (mp->m_flags & M_VLANTAG) { 217579251f5eSSepherosa Ziehau vlan_macip_lens |= htole16(mp->m_pkthdr.ether_vlantag) << 217679251f5eSSepherosa Ziehau IXGBE_ADVTXD_VLAN_SHIFT; 217779251f5eSSepherosa Ziehau } else if (!offload) { 217879251f5eSSepherosa Ziehau /* No TX descriptor is consumed */ 217979251f5eSSepherosa Ziehau return 0; 218079251f5eSSepherosa Ziehau } 218179251f5eSSepherosa Ziehau 218279251f5eSSepherosa Ziehau /* Set the ether header length */ 218379251f5eSSepherosa Ziehau ehdrlen = mp->m_pkthdr.csum_lhlen; 218479251f5eSSepherosa Ziehau KASSERT(ehdrlen > 0, ("invalid ether hlen")); 218579251f5eSSepherosa Ziehau vlan_macip_lens |= ehdrlen << IXGBE_ADVTXD_MACLEN_SHIFT; 218679251f5eSSepherosa Ziehau 218779251f5eSSepherosa Ziehau if (mp->m_pkthdr.csum_flags & CSUM_IP) { 218879251f5eSSepherosa Ziehau *olinfo_status |= IXGBE_TXD_POPTS_IXSM << 8; 218979251f5eSSepherosa Ziehau type_tucmd_mlhl |= IXGBE_ADVTXD_TUCMD_IPV4; 219079251f5eSSepherosa Ziehau ip_hlen = mp->m_pkthdr.csum_iphlen; 219179251f5eSSepherosa Ziehau KASSERT(ip_hlen > 0, ("invalid ip hlen")); 219279251f5eSSepherosa Ziehau } 219379251f5eSSepherosa Ziehau vlan_macip_lens |= ip_hlen; 219479251f5eSSepherosa Ziehau 219579251f5eSSepherosa Ziehau type_tucmd_mlhl |= IXGBE_ADVTXD_DCMD_DEXT | IXGBE_ADVTXD_DTYP_CTXT; 219679251f5eSSepherosa Ziehau if (mp->m_pkthdr.csum_flags & CSUM_TCP) 219779251f5eSSepherosa Ziehau type_tucmd_mlhl |= IXGBE_ADVTXD_TUCMD_L4T_TCP; 219879251f5eSSepherosa Ziehau else if (mp->m_pkthdr.csum_flags & CSUM_UDP) 219979251f5eSSepherosa Ziehau type_tucmd_mlhl |= IXGBE_ADVTXD_TUCMD_L4T_UDP; 220079251f5eSSepherosa Ziehau 220179251f5eSSepherosa Ziehau if (mp->m_pkthdr.csum_flags & (CSUM_TCP | CSUM_UDP)) 220279251f5eSSepherosa Ziehau *olinfo_status |= IXGBE_TXD_POPTS_TXSM << 8; 220379251f5eSSepherosa Ziehau 220479251f5eSSepherosa Ziehau /* Now ready a context descriptor */ 220579251f5eSSepherosa Ziehau ctxd = txr->tx_next_avail; 220679251f5eSSepherosa Ziehau TXD = (struct ixgbe_adv_tx_context_desc *)&txr->tx_base[ctxd]; 220779251f5eSSepherosa Ziehau 220879251f5eSSepherosa Ziehau /* Now copy bits into descriptor */ 220979251f5eSSepherosa Ziehau TXD->vlan_macip_lens = htole32(vlan_macip_lens); 221079251f5eSSepherosa Ziehau TXD->type_tucmd_mlhl = htole32(type_tucmd_mlhl); 221179251f5eSSepherosa Ziehau TXD->seqnum_seed = htole32(0); 221279251f5eSSepherosa Ziehau TXD->mss_l4len_idx = htole32(0); 221379251f5eSSepherosa Ziehau 221479251f5eSSepherosa Ziehau /* We've consumed the first desc, adjust counters */ 221579251f5eSSepherosa Ziehau if (++ctxd == txr->tx_ndesc) 221679251f5eSSepherosa Ziehau ctxd = 0; 221779251f5eSSepherosa Ziehau txr->tx_next_avail = ctxd; 221879251f5eSSepherosa Ziehau --txr->tx_avail; 221979251f5eSSepherosa Ziehau 222079251f5eSSepherosa Ziehau /* One TX descriptor is consumed */ 222179251f5eSSepherosa Ziehau return 1; 222279251f5eSSepherosa Ziehau } 222379251f5eSSepherosa Ziehau 222479251f5eSSepherosa Ziehau static int 222579251f5eSSepherosa Ziehau ix_tso_ctx_setup(struct ix_tx_ring *txr, const struct mbuf *mp, 222679251f5eSSepherosa Ziehau uint32_t *cmd_type_len, uint32_t *olinfo_status) 222779251f5eSSepherosa Ziehau { 222879251f5eSSepherosa Ziehau struct ixgbe_adv_tx_context_desc *TXD; 222979251f5eSSepherosa Ziehau uint32_t vlan_macip_lens = 0, type_tucmd_mlhl = 0; 223079251f5eSSepherosa Ziehau uint32_t mss_l4len_idx = 0, paylen; 223179251f5eSSepherosa Ziehau int ctxd, ehdrlen, ip_hlen, tcp_hlen; 223279251f5eSSepherosa Ziehau 223379251f5eSSepherosa Ziehau ehdrlen = mp->m_pkthdr.csum_lhlen; 223479251f5eSSepherosa Ziehau KASSERT(ehdrlen > 0, ("invalid ether hlen")); 223579251f5eSSepherosa Ziehau 223679251f5eSSepherosa Ziehau ip_hlen = mp->m_pkthdr.csum_iphlen; 223779251f5eSSepherosa Ziehau KASSERT(ip_hlen > 0, ("invalid ip hlen")); 223879251f5eSSepherosa Ziehau 223979251f5eSSepherosa Ziehau tcp_hlen = mp->m_pkthdr.csum_thlen; 224079251f5eSSepherosa Ziehau KASSERT(tcp_hlen > 0, ("invalid tcp hlen")); 224179251f5eSSepherosa Ziehau 224279251f5eSSepherosa Ziehau ctxd = txr->tx_next_avail; 224379251f5eSSepherosa Ziehau TXD = (struct ixgbe_adv_tx_context_desc *) &txr->tx_base[ctxd]; 224479251f5eSSepherosa Ziehau 224579251f5eSSepherosa Ziehau if (mp->m_flags & M_VLANTAG) { 224679251f5eSSepherosa Ziehau vlan_macip_lens |= htole16(mp->m_pkthdr.ether_vlantag) << 224779251f5eSSepherosa Ziehau IXGBE_ADVTXD_VLAN_SHIFT; 224879251f5eSSepherosa Ziehau } 224979251f5eSSepherosa Ziehau vlan_macip_lens |= ehdrlen << IXGBE_ADVTXD_MACLEN_SHIFT; 225079251f5eSSepherosa Ziehau vlan_macip_lens |= ip_hlen; 225179251f5eSSepherosa Ziehau TXD->vlan_macip_lens = htole32(vlan_macip_lens); 225279251f5eSSepherosa Ziehau 225379251f5eSSepherosa Ziehau /* ADV DTYPE TUCMD */ 225479251f5eSSepherosa Ziehau type_tucmd_mlhl |= IXGBE_ADVTXD_TUCMD_IPV4; 225579251f5eSSepherosa Ziehau type_tucmd_mlhl |= IXGBE_ADVTXD_DCMD_DEXT | IXGBE_ADVTXD_DTYP_CTXT; 225679251f5eSSepherosa Ziehau type_tucmd_mlhl |= IXGBE_ADVTXD_TUCMD_L4T_TCP; 225779251f5eSSepherosa Ziehau TXD->type_tucmd_mlhl = htole32(type_tucmd_mlhl); 225879251f5eSSepherosa Ziehau 225979251f5eSSepherosa Ziehau /* MSS L4LEN IDX */ 226079251f5eSSepherosa Ziehau mss_l4len_idx |= (mp->m_pkthdr.tso_segsz << IXGBE_ADVTXD_MSS_SHIFT); 226179251f5eSSepherosa Ziehau mss_l4len_idx |= (tcp_hlen << IXGBE_ADVTXD_L4LEN_SHIFT); 226279251f5eSSepherosa Ziehau TXD->mss_l4len_idx = htole32(mss_l4len_idx); 226379251f5eSSepherosa Ziehau 226479251f5eSSepherosa Ziehau TXD->seqnum_seed = htole32(0); 226579251f5eSSepherosa Ziehau 226679251f5eSSepherosa Ziehau if (++ctxd == txr->tx_ndesc) 226779251f5eSSepherosa Ziehau ctxd = 0; 226879251f5eSSepherosa Ziehau 226979251f5eSSepherosa Ziehau txr->tx_avail--; 227079251f5eSSepherosa Ziehau txr->tx_next_avail = ctxd; 227179251f5eSSepherosa Ziehau 227279251f5eSSepherosa Ziehau *cmd_type_len |= IXGBE_ADVTXD_DCMD_TSE; 227379251f5eSSepherosa Ziehau 227479251f5eSSepherosa Ziehau /* This is used in the transmit desc in encap */ 227579251f5eSSepherosa Ziehau paylen = mp->m_pkthdr.len - ehdrlen - ip_hlen - tcp_hlen; 227679251f5eSSepherosa Ziehau 227779251f5eSSepherosa Ziehau *olinfo_status |= IXGBE_TXD_POPTS_IXSM << 8; 227879251f5eSSepherosa Ziehau *olinfo_status |= IXGBE_TXD_POPTS_TXSM << 8; 227979251f5eSSepherosa Ziehau *olinfo_status |= paylen << IXGBE_ADVTXD_PAYLEN_SHIFT; 228079251f5eSSepherosa Ziehau 228179251f5eSSepherosa Ziehau /* One TX descriptor is consumed */ 228279251f5eSSepherosa Ziehau return 1; 228379251f5eSSepherosa Ziehau } 228479251f5eSSepherosa Ziehau 228579251f5eSSepherosa Ziehau static void 2286189a0ff3SSepherosa Ziehau ix_txeof(struct ix_tx_ring *txr, int hdr) 228779251f5eSSepherosa Ziehau { 2288189a0ff3SSepherosa Ziehau int first, avail; 228979251f5eSSepherosa Ziehau 229079251f5eSSepherosa Ziehau if (txr->tx_avail == txr->tx_ndesc) 229179251f5eSSepherosa Ziehau return; 229279251f5eSSepherosa Ziehau 229379251f5eSSepherosa Ziehau first = txr->tx_next_clean; 229479251f5eSSepherosa Ziehau if (first == hdr) 229579251f5eSSepherosa Ziehau return; 229679251f5eSSepherosa Ziehau 229779251f5eSSepherosa Ziehau avail = txr->tx_avail; 229879251f5eSSepherosa Ziehau while (first != hdr) { 229979251f5eSSepherosa Ziehau struct ix_tx_buf *txbuf = &txr->tx_buf[first]; 230079251f5eSSepherosa Ziehau 230182db96e9SSepherosa Ziehau KKASSERT(avail < txr->tx_ndesc); 230279251f5eSSepherosa Ziehau ++avail; 230382db96e9SSepherosa Ziehau 230482db96e9SSepherosa Ziehau if (txbuf->m_head != NULL) 230582db96e9SSepherosa Ziehau ix_free_txbuf(txr, txbuf); 230679251f5eSSepherosa Ziehau if (++first == txr->tx_ndesc) 230779251f5eSSepherosa Ziehau first = 0; 230879251f5eSSepherosa Ziehau } 230979251f5eSSepherosa Ziehau txr->tx_next_clean = first; 231079251f5eSSepherosa Ziehau txr->tx_avail = avail; 231179251f5eSSepherosa Ziehau 231279251f5eSSepherosa Ziehau if (txr->tx_avail > IX_MAX_SCATTER + IX_TX_RESERVED) { 231379251f5eSSepherosa Ziehau ifsq_clr_oactive(txr->tx_ifsq); 231479251f5eSSepherosa Ziehau txr->tx_watchdog.wd_timer = 0; 231579251f5eSSepherosa Ziehau } 231682db96e9SSepherosa Ziehau txr->tx_running = IX_TX_RUNNING; 231782db96e9SSepherosa Ziehau } 231882db96e9SSepherosa Ziehau 231982db96e9SSepherosa Ziehau static void 232082db96e9SSepherosa Ziehau ix_txgc(struct ix_tx_ring *txr) 232182db96e9SSepherosa Ziehau { 232282db96e9SSepherosa Ziehau int first, hdr; 232382db96e9SSepherosa Ziehau #ifdef INVARIANTS 232482db96e9SSepherosa Ziehau int avail; 232582db96e9SSepherosa Ziehau #endif 232682db96e9SSepherosa Ziehau 232782db96e9SSepherosa Ziehau if (txr->tx_avail == txr->tx_ndesc) 232882db96e9SSepherosa Ziehau return; 232982db96e9SSepherosa Ziehau 233082db96e9SSepherosa Ziehau hdr = IXGBE_READ_REG(&txr->tx_sc->hw, IXGBE_TDH(txr->tx_idx)); 233182db96e9SSepherosa Ziehau first = txr->tx_next_clean; 233282db96e9SSepherosa Ziehau if (first == hdr) 2333*5ae4b2e8SSepherosa Ziehau goto done; 233482db96e9SSepherosa Ziehau txr->tx_gc++; 233582db96e9SSepherosa Ziehau 233682db96e9SSepherosa Ziehau #ifdef INVARIANTS 233782db96e9SSepherosa Ziehau avail = txr->tx_avail; 233882db96e9SSepherosa Ziehau #endif 233982db96e9SSepherosa Ziehau while (first != hdr) { 234082db96e9SSepherosa Ziehau struct ix_tx_buf *txbuf = &txr->tx_buf[first]; 234182db96e9SSepherosa Ziehau 234282db96e9SSepherosa Ziehau #ifdef INVARIANTS 234382db96e9SSepherosa Ziehau KKASSERT(avail < txr->tx_ndesc); 234482db96e9SSepherosa Ziehau ++avail; 234582db96e9SSepherosa Ziehau #endif 234682db96e9SSepherosa Ziehau if (txbuf->m_head != NULL) 234782db96e9SSepherosa Ziehau ix_free_txbuf(txr, txbuf); 234882db96e9SSepherosa Ziehau if (++first == txr->tx_ndesc) 234982db96e9SSepherosa Ziehau first = 0; 235082db96e9SSepherosa Ziehau } 2351*5ae4b2e8SSepherosa Ziehau done: 235282db96e9SSepherosa Ziehau if (txr->tx_nmbuf) 235382db96e9SSepherosa Ziehau txr->tx_running = IX_TX_RUNNING; 235479251f5eSSepherosa Ziehau } 235579251f5eSSepherosa Ziehau 235679251f5eSSepherosa Ziehau static int 235779251f5eSSepherosa Ziehau ix_create_rx_ring(struct ix_rx_ring *rxr) 235879251f5eSSepherosa Ziehau { 235979251f5eSSepherosa Ziehau int i, rsize, error, nrxd; 236079251f5eSSepherosa Ziehau 236179251f5eSSepherosa Ziehau /* 236279251f5eSSepherosa Ziehau * Validate number of receive descriptors. It must not exceed 236379251f5eSSepherosa Ziehau * hardware maximum, and must be multiple of IX_DBA_ALIGN. 236479251f5eSSepherosa Ziehau */ 236579251f5eSSepherosa Ziehau nrxd = device_getenv_int(rxr->rx_sc->dev, "rxd", ix_rxd); 236679251f5eSSepherosa Ziehau if (((nrxd * sizeof(union ixgbe_adv_rx_desc)) % IX_DBA_ALIGN) != 0 || 236779251f5eSSepherosa Ziehau nrxd < IX_MIN_RXD || nrxd > IX_MAX_RXD) { 236879251f5eSSepherosa Ziehau device_printf(rxr->rx_sc->dev, 236979251f5eSSepherosa Ziehau "Using %d RX descriptors instead of %d!\n", 237079251f5eSSepherosa Ziehau IX_DEF_RXD, nrxd); 237179251f5eSSepherosa Ziehau rxr->rx_ndesc = IX_DEF_RXD; 237279251f5eSSepherosa Ziehau } else { 237379251f5eSSepherosa Ziehau rxr->rx_ndesc = nrxd; 237479251f5eSSepherosa Ziehau } 237579251f5eSSepherosa Ziehau 237679251f5eSSepherosa Ziehau /* 237779251f5eSSepherosa Ziehau * Allocate RX descriptor ring 237879251f5eSSepherosa Ziehau */ 237979251f5eSSepherosa Ziehau rsize = roundup2(rxr->rx_ndesc * sizeof(union ixgbe_adv_rx_desc), 238079251f5eSSepherosa Ziehau IX_DBA_ALIGN); 238179251f5eSSepherosa Ziehau rxr->rx_base = bus_dmamem_coherent_any(rxr->rx_sc->parent_tag, 238279251f5eSSepherosa Ziehau IX_DBA_ALIGN, rsize, BUS_DMA_WAITOK | BUS_DMA_ZERO, 238379251f5eSSepherosa Ziehau &rxr->rx_base_dtag, &rxr->rx_base_map, &rxr->rx_base_paddr); 238479251f5eSSepherosa Ziehau if (rxr->rx_base == NULL) { 238579251f5eSSepherosa Ziehau device_printf(rxr->rx_sc->dev, 238679251f5eSSepherosa Ziehau "Unable to allocate TX Descriptor memory\n"); 238779251f5eSSepherosa Ziehau return ENOMEM; 238879251f5eSSepherosa Ziehau } 238979251f5eSSepherosa Ziehau 239079251f5eSSepherosa Ziehau rsize = __VM_CACHELINE_ALIGN(sizeof(struct ix_rx_buf) * rxr->rx_ndesc); 239179251f5eSSepherosa Ziehau rxr->rx_buf = kmalloc_cachealign(rsize, M_DEVBUF, M_WAITOK | M_ZERO); 239279251f5eSSepherosa Ziehau 239379251f5eSSepherosa Ziehau /* 239479251f5eSSepherosa Ziehau * Create DMA tag for RX buffers 239579251f5eSSepherosa Ziehau */ 239679251f5eSSepherosa Ziehau error = bus_dma_tag_create(rxr->rx_sc->parent_tag, 239779251f5eSSepherosa Ziehau 1, 0, /* alignment, bounds */ 239879251f5eSSepherosa Ziehau BUS_SPACE_MAXADDR, /* lowaddr */ 239979251f5eSSepherosa Ziehau BUS_SPACE_MAXADDR, /* highaddr */ 240079251f5eSSepherosa Ziehau NULL, NULL, /* filter, filterarg */ 240179251f5eSSepherosa Ziehau PAGE_SIZE, /* maxsize */ 240279251f5eSSepherosa Ziehau 1, /* nsegments */ 240379251f5eSSepherosa Ziehau PAGE_SIZE, /* maxsegsize */ 240479251f5eSSepherosa Ziehau BUS_DMA_WAITOK | BUS_DMA_ALLOCNOW, /* flags */ 240579251f5eSSepherosa Ziehau &rxr->rx_tag); 240679251f5eSSepherosa Ziehau if (error) { 240779251f5eSSepherosa Ziehau device_printf(rxr->rx_sc->dev, 240879251f5eSSepherosa Ziehau "Unable to create RX DMA tag\n"); 240979251f5eSSepherosa Ziehau kfree(rxr->rx_buf, M_DEVBUF); 241079251f5eSSepherosa Ziehau rxr->rx_buf = NULL; 241179251f5eSSepherosa Ziehau return error; 241279251f5eSSepherosa Ziehau } 241379251f5eSSepherosa Ziehau 241479251f5eSSepherosa Ziehau /* 241579251f5eSSepherosa Ziehau * Create spare DMA map for RX buffers 241679251f5eSSepherosa Ziehau */ 241779251f5eSSepherosa Ziehau error = bus_dmamap_create(rxr->rx_tag, BUS_DMA_WAITOK, 241879251f5eSSepherosa Ziehau &rxr->rx_sparemap); 241979251f5eSSepherosa Ziehau if (error) { 242079251f5eSSepherosa Ziehau device_printf(rxr->rx_sc->dev, 242179251f5eSSepherosa Ziehau "Unable to create spare RX DMA map\n"); 242279251f5eSSepherosa Ziehau bus_dma_tag_destroy(rxr->rx_tag); 242379251f5eSSepherosa Ziehau kfree(rxr->rx_buf, M_DEVBUF); 242479251f5eSSepherosa Ziehau rxr->rx_buf = NULL; 242579251f5eSSepherosa Ziehau return error; 242679251f5eSSepherosa Ziehau } 242779251f5eSSepherosa Ziehau 242879251f5eSSepherosa Ziehau /* 242979251f5eSSepherosa Ziehau * Create DMA maps for RX buffers 243079251f5eSSepherosa Ziehau */ 243179251f5eSSepherosa Ziehau for (i = 0; i < rxr->rx_ndesc; ++i) { 243279251f5eSSepherosa Ziehau struct ix_rx_buf *rxbuf = &rxr->rx_buf[i]; 243379251f5eSSepherosa Ziehau 243479251f5eSSepherosa Ziehau error = bus_dmamap_create(rxr->rx_tag, 243579251f5eSSepherosa Ziehau BUS_DMA_WAITOK, &rxbuf->map); 243679251f5eSSepherosa Ziehau if (error) { 243779251f5eSSepherosa Ziehau device_printf(rxr->rx_sc->dev, 243879251f5eSSepherosa Ziehau "Unable to create RX dma map\n"); 243979251f5eSSepherosa Ziehau ix_destroy_rx_ring(rxr, i); 244079251f5eSSepherosa Ziehau return error; 244179251f5eSSepherosa Ziehau } 244279251f5eSSepherosa Ziehau } 244379251f5eSSepherosa Ziehau 244479251f5eSSepherosa Ziehau /* 244579251f5eSSepherosa Ziehau * Initialize various watermark 244679251f5eSSepherosa Ziehau */ 244779251f5eSSepherosa Ziehau rxr->rx_wreg_nsegs = IX_DEF_RXWREG_NSEGS; 244879251f5eSSepherosa Ziehau 244979251f5eSSepherosa Ziehau return 0; 245079251f5eSSepherosa Ziehau } 245179251f5eSSepherosa Ziehau 245279251f5eSSepherosa Ziehau static void 245379251f5eSSepherosa Ziehau ix_destroy_rx_ring(struct ix_rx_ring *rxr, int ndesc) 245479251f5eSSepherosa Ziehau { 245579251f5eSSepherosa Ziehau int i; 245679251f5eSSepherosa Ziehau 245779251f5eSSepherosa Ziehau if (rxr->rx_base != NULL) { 245879251f5eSSepherosa Ziehau bus_dmamap_unload(rxr->rx_base_dtag, rxr->rx_base_map); 245979251f5eSSepherosa Ziehau bus_dmamem_free(rxr->rx_base_dtag, rxr->rx_base, 246079251f5eSSepherosa Ziehau rxr->rx_base_map); 246179251f5eSSepherosa Ziehau bus_dma_tag_destroy(rxr->rx_base_dtag); 246279251f5eSSepherosa Ziehau rxr->rx_base = NULL; 246379251f5eSSepherosa Ziehau } 246479251f5eSSepherosa Ziehau 246579251f5eSSepherosa Ziehau if (rxr->rx_buf == NULL) 246679251f5eSSepherosa Ziehau return; 246779251f5eSSepherosa Ziehau 246879251f5eSSepherosa Ziehau for (i = 0; i < ndesc; ++i) { 246979251f5eSSepherosa Ziehau struct ix_rx_buf *rxbuf = &rxr->rx_buf[i]; 247079251f5eSSepherosa Ziehau 247179251f5eSSepherosa Ziehau KKASSERT(rxbuf->m_head == NULL); 247279251f5eSSepherosa Ziehau bus_dmamap_destroy(rxr->rx_tag, rxbuf->map); 247379251f5eSSepherosa Ziehau } 247479251f5eSSepherosa Ziehau bus_dmamap_destroy(rxr->rx_tag, rxr->rx_sparemap); 247579251f5eSSepherosa Ziehau bus_dma_tag_destroy(rxr->rx_tag); 247679251f5eSSepherosa Ziehau 247779251f5eSSepherosa Ziehau kfree(rxr->rx_buf, M_DEVBUF); 247879251f5eSSepherosa Ziehau rxr->rx_buf = NULL; 247979251f5eSSepherosa Ziehau } 248079251f5eSSepherosa Ziehau 248179251f5eSSepherosa Ziehau /* 248279251f5eSSepherosa Ziehau ** Used to detect a descriptor that has 248379251f5eSSepherosa Ziehau ** been merged by Hardware RSC. 248479251f5eSSepherosa Ziehau */ 248579251f5eSSepherosa Ziehau static __inline uint32_t 248679251f5eSSepherosa Ziehau ix_rsc_count(union ixgbe_adv_rx_desc *rx) 248779251f5eSSepherosa Ziehau { 248879251f5eSSepherosa Ziehau return (le32toh(rx->wb.lower.lo_dword.data) & 248979251f5eSSepherosa Ziehau IXGBE_RXDADV_RSCCNT_MASK) >> IXGBE_RXDADV_RSCCNT_SHIFT; 249079251f5eSSepherosa Ziehau } 249179251f5eSSepherosa Ziehau 249279251f5eSSepherosa Ziehau #if 0 249379251f5eSSepherosa Ziehau /********************************************************************* 249479251f5eSSepherosa Ziehau * 249579251f5eSSepherosa Ziehau * Initialize Hardware RSC (LRO) feature on 82599 249679251f5eSSepherosa Ziehau * for an RX ring, this is toggled by the LRO capability 249779251f5eSSepherosa Ziehau * even though it is transparent to the stack. 249879251f5eSSepherosa Ziehau * 249979251f5eSSepherosa Ziehau * NOTE: since this HW feature only works with IPV4 and 250079251f5eSSepherosa Ziehau * our testing has shown soft LRO to be as effective 250179251f5eSSepherosa Ziehau * I have decided to disable this by default. 250279251f5eSSepherosa Ziehau * 250379251f5eSSepherosa Ziehau **********************************************************************/ 250479251f5eSSepherosa Ziehau static void 250579251f5eSSepherosa Ziehau ix_setup_hw_rsc(struct ix_rx_ring *rxr) 250679251f5eSSepherosa Ziehau { 250779251f5eSSepherosa Ziehau struct ix_softc *sc = rxr->rx_sc; 250879251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 250979251f5eSSepherosa Ziehau uint32_t rscctrl, rdrxctl; 251079251f5eSSepherosa Ziehau 251179251f5eSSepherosa Ziehau #if 0 251279251f5eSSepherosa Ziehau /* If turning LRO/RSC off we need to disable it */ 251379251f5eSSepherosa Ziehau if ((sc->arpcom.ac_if.if_capenable & IFCAP_LRO) == 0) { 251479251f5eSSepherosa Ziehau rscctrl = IXGBE_READ_REG(hw, IXGBE_RSCCTL(rxr->me)); 251579251f5eSSepherosa Ziehau rscctrl &= ~IXGBE_RSCCTL_RSCEN; 251679251f5eSSepherosa Ziehau return; 251779251f5eSSepherosa Ziehau } 251879251f5eSSepherosa Ziehau #endif 251979251f5eSSepherosa Ziehau 252079251f5eSSepherosa Ziehau rdrxctl = IXGBE_READ_REG(hw, IXGBE_RDRXCTL); 252179251f5eSSepherosa Ziehau rdrxctl &= ~IXGBE_RDRXCTL_RSCFRSTSIZE; 252279251f5eSSepherosa Ziehau rdrxctl |= IXGBE_RDRXCTL_CRCSTRIP; 252379251f5eSSepherosa Ziehau rdrxctl |= IXGBE_RDRXCTL_RSCACKC; 252479251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RDRXCTL, rdrxctl); 252579251f5eSSepherosa Ziehau 252679251f5eSSepherosa Ziehau rscctrl = IXGBE_READ_REG(hw, IXGBE_RSCCTL(rxr->me)); 252779251f5eSSepherosa Ziehau rscctrl |= IXGBE_RSCCTL_RSCEN; 252879251f5eSSepherosa Ziehau /* 252979251f5eSSepherosa Ziehau ** Limit the total number of descriptors that 253079251f5eSSepherosa Ziehau ** can be combined, so it does not exceed 64K 253179251f5eSSepherosa Ziehau */ 253279251f5eSSepherosa Ziehau if (rxr->mbuf_sz == MCLBYTES) 253379251f5eSSepherosa Ziehau rscctrl |= IXGBE_RSCCTL_MAXDESC_16; 253479251f5eSSepherosa Ziehau else if (rxr->mbuf_sz == MJUMPAGESIZE) 253579251f5eSSepherosa Ziehau rscctrl |= IXGBE_RSCCTL_MAXDESC_8; 253679251f5eSSepherosa Ziehau else if (rxr->mbuf_sz == MJUM9BYTES) 253779251f5eSSepherosa Ziehau rscctrl |= IXGBE_RSCCTL_MAXDESC_4; 253879251f5eSSepherosa Ziehau else /* Using 16K cluster */ 253979251f5eSSepherosa Ziehau rscctrl |= IXGBE_RSCCTL_MAXDESC_1; 254079251f5eSSepherosa Ziehau 254179251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RSCCTL(rxr->me), rscctrl); 254279251f5eSSepherosa Ziehau 254379251f5eSSepherosa Ziehau /* Enable TCP header recognition */ 254479251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_PSRTYPE(0), 254579251f5eSSepherosa Ziehau (IXGBE_READ_REG(hw, IXGBE_PSRTYPE(0)) | 254679251f5eSSepherosa Ziehau IXGBE_PSRTYPE_TCPHDR)); 254779251f5eSSepherosa Ziehau 254879251f5eSSepherosa Ziehau /* Disable RSC for ACK packets */ 254979251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RSCDBU, 255079251f5eSSepherosa Ziehau (IXGBE_RSCDBU_RSCACKDIS | IXGBE_READ_REG(hw, IXGBE_RSCDBU))); 255179251f5eSSepherosa Ziehau 255279251f5eSSepherosa Ziehau rxr->hw_rsc = TRUE; 255379251f5eSSepherosa Ziehau } 255479251f5eSSepherosa Ziehau #endif 255579251f5eSSepherosa Ziehau 255679251f5eSSepherosa Ziehau static int 255779251f5eSSepherosa Ziehau ix_init_rx_ring(struct ix_rx_ring *rxr) 255879251f5eSSepherosa Ziehau { 255979251f5eSSepherosa Ziehau int i; 256079251f5eSSepherosa Ziehau 256179251f5eSSepherosa Ziehau /* Clear the ring contents */ 256279251f5eSSepherosa Ziehau bzero(rxr->rx_base, rxr->rx_ndesc * sizeof(union ixgbe_adv_rx_desc)); 256379251f5eSSepherosa Ziehau 256479251f5eSSepherosa Ziehau /* XXX we need JUMPAGESIZE for RSC too */ 256579251f5eSSepherosa Ziehau if (rxr->rx_sc->max_frame_size <= MCLBYTES) 256679251f5eSSepherosa Ziehau rxr->rx_mbuf_sz = MCLBYTES; 256779251f5eSSepherosa Ziehau else 256879251f5eSSepherosa Ziehau rxr->rx_mbuf_sz = MJUMPAGESIZE; 256979251f5eSSepherosa Ziehau 257079251f5eSSepherosa Ziehau /* Now replenish the mbufs */ 257179251f5eSSepherosa Ziehau for (i = 0; i < rxr->rx_ndesc; ++i) { 257279251f5eSSepherosa Ziehau int error; 257379251f5eSSepherosa Ziehau 257479251f5eSSepherosa Ziehau error = ix_newbuf(rxr, i, TRUE); 257579251f5eSSepherosa Ziehau if (error) 257679251f5eSSepherosa Ziehau return error; 257779251f5eSSepherosa Ziehau } 257879251f5eSSepherosa Ziehau 257979251f5eSSepherosa Ziehau /* Setup our descriptor indices */ 258079251f5eSSepherosa Ziehau rxr->rx_next_check = 0; 258179251f5eSSepherosa Ziehau rxr->rx_flags &= ~IX_RXRING_FLAG_DISC; 258279251f5eSSepherosa Ziehau 258379251f5eSSepherosa Ziehau #if 0 258479251f5eSSepherosa Ziehau /* 258579251f5eSSepherosa Ziehau ** Now set up the LRO interface: 258679251f5eSSepherosa Ziehau */ 258779251f5eSSepherosa Ziehau if (ixgbe_rsc_enable) 258879251f5eSSepherosa Ziehau ix_setup_hw_rsc(rxr); 258979251f5eSSepherosa Ziehau #endif 259079251f5eSSepherosa Ziehau 259179251f5eSSepherosa Ziehau return 0; 259279251f5eSSepherosa Ziehau } 259379251f5eSSepherosa Ziehau 259479251f5eSSepherosa Ziehau #define IXGBE_SRRCTL_BSIZEHDRSIZE_SHIFT 2 259579251f5eSSepherosa Ziehau 259679251f5eSSepherosa Ziehau #define BSIZEPKT_ROUNDUP ((1<<IXGBE_SRRCTL_BSIZEPKT_SHIFT)-1) 259779251f5eSSepherosa Ziehau 259879251f5eSSepherosa Ziehau static void 25993c37d13bSSepherosa Ziehau ix_init_rx_unit(struct ix_softc *sc, boolean_t polling) 260079251f5eSSepherosa Ziehau { 260179251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 260279251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 260363d483cdSSepherosa Ziehau uint32_t bufsz, fctrl, rxcsum, hlreg; 260479251f5eSSepherosa Ziehau int i; 260579251f5eSSepherosa Ziehau 260679251f5eSSepherosa Ziehau /* 260779251f5eSSepherosa Ziehau * Make sure receives are disabled while setting up the descriptor ring 260879251f5eSSepherosa Ziehau */ 260963d483cdSSepherosa Ziehau ixgbe_disable_rx(hw); 261079251f5eSSepherosa Ziehau 261179251f5eSSepherosa Ziehau /* Enable broadcasts */ 261279251f5eSSepherosa Ziehau fctrl = IXGBE_READ_REG(hw, IXGBE_FCTRL); 261379251f5eSSepherosa Ziehau fctrl |= IXGBE_FCTRL_BAM; 261463d483cdSSepherosa Ziehau if (hw->mac.type == ixgbe_mac_82598EB) { 261579251f5eSSepherosa Ziehau fctrl |= IXGBE_FCTRL_DPF; 261679251f5eSSepherosa Ziehau fctrl |= IXGBE_FCTRL_PMCF; 261763d483cdSSepherosa Ziehau } 261879251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_FCTRL, fctrl); 261979251f5eSSepherosa Ziehau 262079251f5eSSepherosa Ziehau /* Set for Jumbo Frames? */ 262179251f5eSSepherosa Ziehau hlreg = IXGBE_READ_REG(hw, IXGBE_HLREG0); 262279251f5eSSepherosa Ziehau if (ifp->if_mtu > ETHERMTU) 262379251f5eSSepherosa Ziehau hlreg |= IXGBE_HLREG0_JUMBOEN; 262479251f5eSSepherosa Ziehau else 262579251f5eSSepherosa Ziehau hlreg &= ~IXGBE_HLREG0_JUMBOEN; 262679251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_HLREG0, hlreg); 262779251f5eSSepherosa Ziehau 262879251f5eSSepherosa Ziehau KKASSERT(sc->rx_rings[0].rx_mbuf_sz >= MCLBYTES); 262979251f5eSSepherosa Ziehau bufsz = (sc->rx_rings[0].rx_mbuf_sz + BSIZEPKT_ROUNDUP) >> 263079251f5eSSepherosa Ziehau IXGBE_SRRCTL_BSIZEPKT_SHIFT; 263179251f5eSSepherosa Ziehau 263279251f5eSSepherosa Ziehau for (i = 0; i < sc->rx_ring_inuse; ++i) { 263379251f5eSSepherosa Ziehau struct ix_rx_ring *rxr = &sc->rx_rings[i]; 263479251f5eSSepherosa Ziehau uint64_t rdba = rxr->rx_base_paddr; 263579251f5eSSepherosa Ziehau uint32_t srrctl; 263679251f5eSSepherosa Ziehau 263779251f5eSSepherosa Ziehau /* Setup the Base and Length of the Rx Descriptor Ring */ 263879251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RDBAL(i), (uint32_t)rdba); 263979251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RDBAH(i), (uint32_t)(rdba >> 32)); 264079251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RDLEN(i), 264179251f5eSSepherosa Ziehau rxr->rx_ndesc * sizeof(union ixgbe_adv_rx_desc)); 264279251f5eSSepherosa Ziehau 264379251f5eSSepherosa Ziehau /* 264479251f5eSSepherosa Ziehau * Set up the SRRCTL register 264579251f5eSSepherosa Ziehau */ 264679251f5eSSepherosa Ziehau srrctl = IXGBE_READ_REG(hw, IXGBE_SRRCTL(i)); 264779251f5eSSepherosa Ziehau 264879251f5eSSepherosa Ziehau srrctl &= ~IXGBE_SRRCTL_BSIZEHDR_MASK; 264979251f5eSSepherosa Ziehau srrctl &= ~IXGBE_SRRCTL_BSIZEPKT_MASK; 265079251f5eSSepherosa Ziehau srrctl |= bufsz; 265179251f5eSSepherosa Ziehau srrctl |= IXGBE_SRRCTL_DESCTYPE_ADV_ONEBUF; 265279251f5eSSepherosa Ziehau if (sc->rx_ring_inuse > 1) { 265379251f5eSSepherosa Ziehau /* See the commend near ix_enable_rx_drop() */ 265463d483cdSSepherosa Ziehau if (sc->ifm_media & 2655060fa21cSSepherosa Ziehau (IFM_ETH_RXPAUSE | IFM_ETH_TXPAUSE)) { 265679251f5eSSepherosa Ziehau srrctl &= ~IXGBE_SRRCTL_DROP_EN; 265779251f5eSSepherosa Ziehau if (i == 0 && bootverbose) { 2658060fa21cSSepherosa Ziehau if_printf(ifp, "flow control %s, " 2659060fa21cSSepherosa Ziehau "disable RX drop\n", 266063d483cdSSepherosa Ziehau ix_ifmedia2str(sc->ifm_media)); 266179251f5eSSepherosa Ziehau } 2662060fa21cSSepherosa Ziehau } else { 266379251f5eSSepherosa Ziehau srrctl |= IXGBE_SRRCTL_DROP_EN; 266479251f5eSSepherosa Ziehau if (i == 0 && bootverbose) { 2665060fa21cSSepherosa Ziehau if_printf(ifp, "flow control %s, " 2666060fa21cSSepherosa Ziehau "enable RX drop\n", 266763d483cdSSepherosa Ziehau ix_ifmedia2str(sc->ifm_media)); 266879251f5eSSepherosa Ziehau } 266979251f5eSSepherosa Ziehau } 267079251f5eSSepherosa Ziehau } 267179251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_SRRCTL(i), srrctl); 267279251f5eSSepherosa Ziehau 267379251f5eSSepherosa Ziehau /* Setup the HW Rx Head and Tail Descriptor Pointers */ 267479251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RDH(i), 0); 267579251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RDT(i), 0); 267679251f5eSSepherosa Ziehau } 267779251f5eSSepherosa Ziehau 267879251f5eSSepherosa Ziehau if (sc->hw.mac.type != ixgbe_mac_82598EB) 267979251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_PSRTYPE(0), 0); 268079251f5eSSepherosa Ziehau 268179251f5eSSepherosa Ziehau rxcsum = IXGBE_READ_REG(hw, IXGBE_RXCSUM); 268279251f5eSSepherosa Ziehau 268379251f5eSSepherosa Ziehau /* 268479251f5eSSepherosa Ziehau * Setup RSS 268579251f5eSSepherosa Ziehau */ 26863c37d13bSSepherosa Ziehau if (sc->rx_ring_inuse > 1) { 268779251f5eSSepherosa Ziehau uint8_t key[IX_NRSSRK * IX_RSSRK_SIZE]; 26883c37d13bSSepherosa Ziehau const struct if_ringmap *rm; 26893c37d13bSSepherosa Ziehau int j, r, nreta, table_nent; 269079251f5eSSepherosa Ziehau 269179251f5eSSepherosa Ziehau /* 269279251f5eSSepherosa Ziehau * NOTE: 269379251f5eSSepherosa Ziehau * When we reach here, RSS has already been disabled 269479251f5eSSepherosa Ziehau * in ix_stop(), so we could safely configure RSS key 269579251f5eSSepherosa Ziehau * and redirect table. 269679251f5eSSepherosa Ziehau */ 269779251f5eSSepherosa Ziehau 269879251f5eSSepherosa Ziehau /* 269979251f5eSSepherosa Ziehau * Configure RSS key 270079251f5eSSepherosa Ziehau */ 270179251f5eSSepherosa Ziehau toeplitz_get_key(key, sizeof(key)); 270279251f5eSSepherosa Ziehau for (i = 0; i < IX_NRSSRK; ++i) { 270379251f5eSSepherosa Ziehau uint32_t rssrk; 270479251f5eSSepherosa Ziehau 270579251f5eSSepherosa Ziehau rssrk = IX_RSSRK_VAL(key, i); 270679251f5eSSepherosa Ziehau IX_RSS_DPRINTF(sc, 1, "rssrk%d 0x%08x\n", 270779251f5eSSepherosa Ziehau i, rssrk); 270879251f5eSSepherosa Ziehau 270979251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RSSRK(i), rssrk); 271079251f5eSSepherosa Ziehau } 271179251f5eSSepherosa Ziehau 27123c37d13bSSepherosa Ziehau /* 27133c37d13bSSepherosa Ziehau * Configure RSS redirect table. 27143c37d13bSSepherosa Ziehau */ 27153c37d13bSSepherosa Ziehau 271663d483cdSSepherosa Ziehau /* Table size will differ based on MAC */ 271763d483cdSSepherosa Ziehau switch (hw->mac.type) { 271863d483cdSSepherosa Ziehau case ixgbe_mac_X550: 271963d483cdSSepherosa Ziehau case ixgbe_mac_X550EM_x: 272063d483cdSSepherosa Ziehau case ixgbe_mac_X550EM_a: 272163d483cdSSepherosa Ziehau nreta = IX_NRETA_X550; 272263d483cdSSepherosa Ziehau break; 272363d483cdSSepherosa Ziehau default: 272463d483cdSSepherosa Ziehau nreta = IX_NRETA; 272563d483cdSSepherosa Ziehau break; 272663d483cdSSepherosa Ziehau } 272763d483cdSSepherosa Ziehau 27283c37d13bSSepherosa Ziehau table_nent = nreta * IX_RETA_SIZE; 27293c37d13bSSepherosa Ziehau KASSERT(table_nent <= IX_RDRTABLE_SIZE, 27303c37d13bSSepherosa Ziehau ("invalid RETA count %d", nreta)); 27313c37d13bSSepherosa Ziehau if (polling) 27323c37d13bSSepherosa Ziehau rm = sc->rx_rmap; 27333c37d13bSSepherosa Ziehau else 27343c37d13bSSepherosa Ziehau rm = sc->rx_rmap_intr; 27353c37d13bSSepherosa Ziehau if_ringmap_rdrtable(rm, sc->rdr_table, table_nent); 27363c37d13bSSepherosa Ziehau 273779251f5eSSepherosa Ziehau r = 0; 273863d483cdSSepherosa Ziehau for (j = 0; j < nreta; ++j) { 273979251f5eSSepherosa Ziehau uint32_t reta = 0; 274079251f5eSSepherosa Ziehau 274179251f5eSSepherosa Ziehau for (i = 0; i < IX_RETA_SIZE; ++i) { 274279251f5eSSepherosa Ziehau uint32_t q; 274379251f5eSSepherosa Ziehau 27443c37d13bSSepherosa Ziehau q = sc->rdr_table[r]; 27453c37d13bSSepherosa Ziehau KASSERT(q < sc->rx_ring_inuse, 27463c37d13bSSepherosa Ziehau ("invalid RX ring index %d", q)); 274779251f5eSSepherosa Ziehau reta |= q << (8 * i); 274879251f5eSSepherosa Ziehau ++r; 274979251f5eSSepherosa Ziehau } 275079251f5eSSepherosa Ziehau IX_RSS_DPRINTF(sc, 1, "reta 0x%08x\n", reta); 275163d483cdSSepherosa Ziehau if (j < IX_NRETA) { 275279251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RETA(j), reta); 275363d483cdSSepherosa Ziehau } else { 275463d483cdSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_ERETA(j - IX_NRETA), 275563d483cdSSepherosa Ziehau reta); 275663d483cdSSepherosa Ziehau } 275779251f5eSSepherosa Ziehau } 275879251f5eSSepherosa Ziehau 275979251f5eSSepherosa Ziehau /* 276079251f5eSSepherosa Ziehau * Enable multiple receive queues. 276179251f5eSSepherosa Ziehau * Enable IPv4 RSS standard hash functions. 276279251f5eSSepherosa Ziehau */ 276379251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_MRQC, 276479251f5eSSepherosa Ziehau IXGBE_MRQC_RSSEN | 276579251f5eSSepherosa Ziehau IXGBE_MRQC_RSS_FIELD_IPV4 | 276679251f5eSSepherosa Ziehau IXGBE_MRQC_RSS_FIELD_IPV4_TCP); 276779251f5eSSepherosa Ziehau 276879251f5eSSepherosa Ziehau /* 276979251f5eSSepherosa Ziehau * NOTE: 277079251f5eSSepherosa Ziehau * PCSD must be enabled to enable multiple 277179251f5eSSepherosa Ziehau * receive queues. 277279251f5eSSepherosa Ziehau */ 277379251f5eSSepherosa Ziehau rxcsum |= IXGBE_RXCSUM_PCSD; 277479251f5eSSepherosa Ziehau } 277579251f5eSSepherosa Ziehau 277679251f5eSSepherosa Ziehau if (ifp->if_capenable & IFCAP_RXCSUM) 277779251f5eSSepherosa Ziehau rxcsum |= IXGBE_RXCSUM_PCSD; 277879251f5eSSepherosa Ziehau 277979251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RXCSUM, rxcsum); 278079251f5eSSepherosa Ziehau } 278179251f5eSSepherosa Ziehau 278279251f5eSSepherosa Ziehau static __inline void 278379251f5eSSepherosa Ziehau ix_rx_refresh(struct ix_rx_ring *rxr, int i) 278479251f5eSSepherosa Ziehau { 278579251f5eSSepherosa Ziehau if (--i < 0) 278679251f5eSSepherosa Ziehau i = rxr->rx_ndesc - 1; 278779251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&rxr->rx_sc->hw, IXGBE_RDT(rxr->rx_idx), i); 278879251f5eSSepherosa Ziehau } 278979251f5eSSepherosa Ziehau 279079251f5eSSepherosa Ziehau static __inline void 279179251f5eSSepherosa Ziehau ix_rxcsum(uint32_t staterr, struct mbuf *mp, uint32_t ptype) 279279251f5eSSepherosa Ziehau { 279379251f5eSSepherosa Ziehau if ((ptype & 279479251f5eSSepherosa Ziehau (IXGBE_RXDADV_PKTTYPE_IPV4 | IXGBE_RXDADV_PKTTYPE_IPV4_EX)) == 0) { 279579251f5eSSepherosa Ziehau /* Not IPv4 */ 279679251f5eSSepherosa Ziehau return; 279779251f5eSSepherosa Ziehau } 279879251f5eSSepherosa Ziehau 279979251f5eSSepherosa Ziehau if ((staterr & (IXGBE_RXD_STAT_IPCS | IXGBE_RXDADV_ERR_IPE)) == 280079251f5eSSepherosa Ziehau IXGBE_RXD_STAT_IPCS) 280179251f5eSSepherosa Ziehau mp->m_pkthdr.csum_flags |= CSUM_IP_CHECKED | CSUM_IP_VALID; 280279251f5eSSepherosa Ziehau 280379251f5eSSepherosa Ziehau if ((ptype & 280479251f5eSSepherosa Ziehau (IXGBE_RXDADV_PKTTYPE_TCP | IXGBE_RXDADV_PKTTYPE_UDP)) == 0) { 280579251f5eSSepherosa Ziehau /* 280679251f5eSSepherosa Ziehau * - Neither TCP nor UDP 280779251f5eSSepherosa Ziehau * - IPv4 fragment 280879251f5eSSepherosa Ziehau */ 280979251f5eSSepherosa Ziehau return; 281079251f5eSSepherosa Ziehau } 281179251f5eSSepherosa Ziehau 281279251f5eSSepherosa Ziehau if ((staterr & (IXGBE_RXD_STAT_L4CS | IXGBE_RXDADV_ERR_TCPE)) == 281379251f5eSSepherosa Ziehau IXGBE_RXD_STAT_L4CS) { 281479251f5eSSepherosa Ziehau mp->m_pkthdr.csum_flags |= CSUM_DATA_VALID | CSUM_PSEUDO_HDR | 281579251f5eSSepherosa Ziehau CSUM_FRAG_NOT_CHECKED; 281679251f5eSSepherosa Ziehau mp->m_pkthdr.csum_data = htons(0xffff); 281779251f5eSSepherosa Ziehau } 281879251f5eSSepherosa Ziehau } 281979251f5eSSepherosa Ziehau 282079251f5eSSepherosa Ziehau static __inline struct pktinfo * 282179251f5eSSepherosa Ziehau ix_rssinfo(struct mbuf *m, struct pktinfo *pi, 282279251f5eSSepherosa Ziehau uint32_t hash, uint32_t hashtype, uint32_t ptype) 282379251f5eSSepherosa Ziehau { 282479251f5eSSepherosa Ziehau switch (hashtype) { 282579251f5eSSepherosa Ziehau case IXGBE_RXDADV_RSSTYPE_IPV4_TCP: 282679251f5eSSepherosa Ziehau pi->pi_netisr = NETISR_IP; 282779251f5eSSepherosa Ziehau pi->pi_flags = 0; 282879251f5eSSepherosa Ziehau pi->pi_l3proto = IPPROTO_TCP; 282979251f5eSSepherosa Ziehau break; 283079251f5eSSepherosa Ziehau 283179251f5eSSepherosa Ziehau case IXGBE_RXDADV_RSSTYPE_IPV4: 283279251f5eSSepherosa Ziehau if ((ptype & IXGBE_RXDADV_PKTTYPE_UDP) == 0) { 283379251f5eSSepherosa Ziehau /* Not UDP or is fragment */ 283479251f5eSSepherosa Ziehau return NULL; 283579251f5eSSepherosa Ziehau } 283679251f5eSSepherosa Ziehau pi->pi_netisr = NETISR_IP; 283779251f5eSSepherosa Ziehau pi->pi_flags = 0; 283879251f5eSSepherosa Ziehau pi->pi_l3proto = IPPROTO_UDP; 283979251f5eSSepherosa Ziehau break; 284079251f5eSSepherosa Ziehau 284179251f5eSSepherosa Ziehau default: 284279251f5eSSepherosa Ziehau return NULL; 284379251f5eSSepherosa Ziehau } 284479251f5eSSepherosa Ziehau 28457558541bSSepherosa Ziehau m_sethash(m, toeplitz_hash(hash)); 284679251f5eSSepherosa Ziehau return pi; 284779251f5eSSepherosa Ziehau } 284879251f5eSSepherosa Ziehau 284979251f5eSSepherosa Ziehau static __inline void 285079251f5eSSepherosa Ziehau ix_setup_rxdesc(union ixgbe_adv_rx_desc *rxd, const struct ix_rx_buf *rxbuf) 285179251f5eSSepherosa Ziehau { 285279251f5eSSepherosa Ziehau rxd->read.pkt_addr = htole64(rxbuf->paddr); 285379251f5eSSepherosa Ziehau rxd->wb.upper.status_error = 0; 285479251f5eSSepherosa Ziehau } 285579251f5eSSepherosa Ziehau 285679251f5eSSepherosa Ziehau static void 285779251f5eSSepherosa Ziehau ix_rx_discard(struct ix_rx_ring *rxr, int i, boolean_t eop) 285879251f5eSSepherosa Ziehau { 285979251f5eSSepherosa Ziehau struct ix_rx_buf *rxbuf = &rxr->rx_buf[i]; 286079251f5eSSepherosa Ziehau 286179251f5eSSepherosa Ziehau /* 286279251f5eSSepherosa Ziehau * XXX discard may not be correct 286379251f5eSSepherosa Ziehau */ 286479251f5eSSepherosa Ziehau if (eop) { 286579251f5eSSepherosa Ziehau IFNET_STAT_INC(&rxr->rx_sc->arpcom.ac_if, ierrors, 1); 286679251f5eSSepherosa Ziehau rxr->rx_flags &= ~IX_RXRING_FLAG_DISC; 286779251f5eSSepherosa Ziehau } else { 286879251f5eSSepherosa Ziehau rxr->rx_flags |= IX_RXRING_FLAG_DISC; 286979251f5eSSepherosa Ziehau } 287079251f5eSSepherosa Ziehau if (rxbuf->fmp != NULL) { 287179251f5eSSepherosa Ziehau m_freem(rxbuf->fmp); 287279251f5eSSepherosa Ziehau rxbuf->fmp = NULL; 287379251f5eSSepherosa Ziehau rxbuf->lmp = NULL; 287479251f5eSSepherosa Ziehau } 287579251f5eSSepherosa Ziehau ix_setup_rxdesc(&rxr->rx_base[i], rxbuf); 287679251f5eSSepherosa Ziehau } 287779251f5eSSepherosa Ziehau 287879251f5eSSepherosa Ziehau static void 28794a648aefSSepherosa Ziehau ix_rxeof(struct ix_rx_ring *rxr, int count) 288079251f5eSSepherosa Ziehau { 288179251f5eSSepherosa Ziehau struct ifnet *ifp = &rxr->rx_sc->arpcom.ac_if; 2882ff37a356SSepherosa Ziehau int i, nsegs = 0, cpuid = mycpuid; 288379251f5eSSepherosa Ziehau 288479251f5eSSepherosa Ziehau i = rxr->rx_next_check; 28854a648aefSSepherosa Ziehau while (count != 0) { 288679251f5eSSepherosa Ziehau struct ix_rx_buf *rxbuf, *nbuf = NULL; 288779251f5eSSepherosa Ziehau union ixgbe_adv_rx_desc *cur; 288879251f5eSSepherosa Ziehau struct mbuf *sendmp = NULL, *mp; 288979251f5eSSepherosa Ziehau struct pktinfo *pi = NULL, pi0; 289079251f5eSSepherosa Ziehau uint32_t rsc = 0, ptype, staterr, hash, hashtype; 289179251f5eSSepherosa Ziehau uint16_t len; 289279251f5eSSepherosa Ziehau boolean_t eop; 289379251f5eSSepherosa Ziehau 289479251f5eSSepherosa Ziehau cur = &rxr->rx_base[i]; 289579251f5eSSepherosa Ziehau staterr = le32toh(cur->wb.upper.status_error); 289679251f5eSSepherosa Ziehau 289779251f5eSSepherosa Ziehau if ((staterr & IXGBE_RXD_STAT_DD) == 0) 289879251f5eSSepherosa Ziehau break; 289979251f5eSSepherosa Ziehau ++nsegs; 290079251f5eSSepherosa Ziehau 290179251f5eSSepherosa Ziehau rxbuf = &rxr->rx_buf[i]; 290279251f5eSSepherosa Ziehau mp = rxbuf->m_head; 290379251f5eSSepherosa Ziehau 290479251f5eSSepherosa Ziehau len = le16toh(cur->wb.upper.length); 290579251f5eSSepherosa Ziehau ptype = le32toh(cur->wb.lower.lo_dword.data) & 290679251f5eSSepherosa Ziehau IXGBE_RXDADV_PKTTYPE_MASK; 290779251f5eSSepherosa Ziehau hash = le32toh(cur->wb.lower.hi_dword.rss); 290879251f5eSSepherosa Ziehau hashtype = le32toh(cur->wb.lower.lo_dword.data) & 290979251f5eSSepherosa Ziehau IXGBE_RXDADV_RSSTYPE_MASK; 29104a648aefSSepherosa Ziehau 291179251f5eSSepherosa Ziehau eop = ((staterr & IXGBE_RXD_STAT_EOP) != 0); 29124a648aefSSepherosa Ziehau if (eop) 29134a648aefSSepherosa Ziehau --count; 291479251f5eSSepherosa Ziehau 291579251f5eSSepherosa Ziehau /* 291679251f5eSSepherosa Ziehau * Make sure bad packets are discarded 291779251f5eSSepherosa Ziehau */ 291879251f5eSSepherosa Ziehau if ((staterr & IXGBE_RXDADV_ERR_FRAME_ERR_MASK) || 291979251f5eSSepherosa Ziehau (rxr->rx_flags & IX_RXRING_FLAG_DISC)) { 292079251f5eSSepherosa Ziehau ix_rx_discard(rxr, i, eop); 292179251f5eSSepherosa Ziehau goto next_desc; 292279251f5eSSepherosa Ziehau } 292379251f5eSSepherosa Ziehau 292479251f5eSSepherosa Ziehau bus_dmamap_sync(rxr->rx_tag, rxbuf->map, BUS_DMASYNC_POSTREAD); 292579251f5eSSepherosa Ziehau if (ix_newbuf(rxr, i, FALSE) != 0) { 292679251f5eSSepherosa Ziehau ix_rx_discard(rxr, i, eop); 292779251f5eSSepherosa Ziehau goto next_desc; 292879251f5eSSepherosa Ziehau } 292979251f5eSSepherosa Ziehau 293079251f5eSSepherosa Ziehau /* 293179251f5eSSepherosa Ziehau * On 82599 which supports a hardware LRO, packets 293279251f5eSSepherosa Ziehau * need not be fragmented across sequential descriptors, 293379251f5eSSepherosa Ziehau * rather the next descriptor is indicated in bits 293479251f5eSSepherosa Ziehau * of the descriptor. This also means that we might 293579251f5eSSepherosa Ziehau * proceses more than one packet at a time, something 293679251f5eSSepherosa Ziehau * that has never been true before, it required 293779251f5eSSepherosa Ziehau * eliminating global chain pointers in favor of what 293879251f5eSSepherosa Ziehau * we are doing here. 293979251f5eSSepherosa Ziehau */ 294079251f5eSSepherosa Ziehau if (!eop) { 294179251f5eSSepherosa Ziehau int nextp; 294279251f5eSSepherosa Ziehau 294379251f5eSSepherosa Ziehau /* 294479251f5eSSepherosa Ziehau * Figure out the next descriptor 294579251f5eSSepherosa Ziehau * of this frame. 294679251f5eSSepherosa Ziehau */ 294779251f5eSSepherosa Ziehau if (rxr->rx_flags & IX_RXRING_FLAG_LRO) 294879251f5eSSepherosa Ziehau rsc = ix_rsc_count(cur); 294979251f5eSSepherosa Ziehau if (rsc) { /* Get hardware index */ 295079251f5eSSepherosa Ziehau nextp = ((staterr & 295179251f5eSSepherosa Ziehau IXGBE_RXDADV_NEXTP_MASK) >> 295279251f5eSSepherosa Ziehau IXGBE_RXDADV_NEXTP_SHIFT); 295379251f5eSSepherosa Ziehau } else { /* Just sequential */ 295479251f5eSSepherosa Ziehau nextp = i + 1; 295579251f5eSSepherosa Ziehau if (nextp == rxr->rx_ndesc) 295679251f5eSSepherosa Ziehau nextp = 0; 295779251f5eSSepherosa Ziehau } 295879251f5eSSepherosa Ziehau nbuf = &rxr->rx_buf[nextp]; 295979251f5eSSepherosa Ziehau prefetch(nbuf); 296079251f5eSSepherosa Ziehau } 296179251f5eSSepherosa Ziehau mp->m_len = len; 296279251f5eSSepherosa Ziehau 296379251f5eSSepherosa Ziehau /* 296479251f5eSSepherosa Ziehau * Rather than using the fmp/lmp global pointers 296579251f5eSSepherosa Ziehau * we now keep the head of a packet chain in the 296679251f5eSSepherosa Ziehau * buffer struct and pass this along from one 296779251f5eSSepherosa Ziehau * descriptor to the next, until we get EOP. 296879251f5eSSepherosa Ziehau */ 296979251f5eSSepherosa Ziehau if (rxbuf->fmp == NULL) { 297079251f5eSSepherosa Ziehau mp->m_pkthdr.len = len; 297179251f5eSSepherosa Ziehau rxbuf->fmp = mp; 297279251f5eSSepherosa Ziehau rxbuf->lmp = mp; 297379251f5eSSepherosa Ziehau } else { 297479251f5eSSepherosa Ziehau rxbuf->fmp->m_pkthdr.len += len; 297579251f5eSSepherosa Ziehau rxbuf->lmp->m_next = mp; 297679251f5eSSepherosa Ziehau rxbuf->lmp = mp; 297779251f5eSSepherosa Ziehau } 297879251f5eSSepherosa Ziehau 297979251f5eSSepherosa Ziehau if (nbuf != NULL) { 298079251f5eSSepherosa Ziehau /* 298179251f5eSSepherosa Ziehau * Not the last fragment of this frame, 298279251f5eSSepherosa Ziehau * pass this fragment list on 298379251f5eSSepherosa Ziehau */ 298479251f5eSSepherosa Ziehau nbuf->fmp = rxbuf->fmp; 298579251f5eSSepherosa Ziehau nbuf->lmp = rxbuf->lmp; 298679251f5eSSepherosa Ziehau } else { 298779251f5eSSepherosa Ziehau /* 298879251f5eSSepherosa Ziehau * Send this frame 298979251f5eSSepherosa Ziehau */ 299079251f5eSSepherosa Ziehau sendmp = rxbuf->fmp; 299179251f5eSSepherosa Ziehau 299279251f5eSSepherosa Ziehau sendmp->m_pkthdr.rcvif = ifp; 299379251f5eSSepherosa Ziehau IFNET_STAT_INC(ifp, ipackets, 1); 299479251f5eSSepherosa Ziehau #ifdef IX_RSS_DEBUG 299579251f5eSSepherosa Ziehau rxr->rx_pkts++; 299679251f5eSSepherosa Ziehau #endif 299779251f5eSSepherosa Ziehau 299879251f5eSSepherosa Ziehau /* Process vlan info */ 299979251f5eSSepherosa Ziehau if (staterr & IXGBE_RXD_STAT_VP) { 300079251f5eSSepherosa Ziehau sendmp->m_pkthdr.ether_vlantag = 300179251f5eSSepherosa Ziehau le16toh(cur->wb.upper.vlan); 300279251f5eSSepherosa Ziehau sendmp->m_flags |= M_VLANTAG; 300379251f5eSSepherosa Ziehau } 300479251f5eSSepherosa Ziehau if (ifp->if_capenable & IFCAP_RXCSUM) 300579251f5eSSepherosa Ziehau ix_rxcsum(staterr, sendmp, ptype); 300679251f5eSSepherosa Ziehau if (ifp->if_capenable & IFCAP_RSS) { 300779251f5eSSepherosa Ziehau pi = ix_rssinfo(sendmp, &pi0, 300879251f5eSSepherosa Ziehau hash, hashtype, ptype); 300979251f5eSSepherosa Ziehau } 301079251f5eSSepherosa Ziehau } 301179251f5eSSepherosa Ziehau rxbuf->fmp = NULL; 301279251f5eSSepherosa Ziehau rxbuf->lmp = NULL; 301379251f5eSSepherosa Ziehau next_desc: 301479251f5eSSepherosa Ziehau /* Advance our pointers to the next descriptor. */ 301579251f5eSSepherosa Ziehau if (++i == rxr->rx_ndesc) 301679251f5eSSepherosa Ziehau i = 0; 301779251f5eSSepherosa Ziehau 301879251f5eSSepherosa Ziehau if (sendmp != NULL) 3019be4134c6SFranco Fichtner ifp->if_input(ifp, sendmp, pi, cpuid); 302079251f5eSSepherosa Ziehau 302179251f5eSSepherosa Ziehau if (nsegs >= rxr->rx_wreg_nsegs) { 302279251f5eSSepherosa Ziehau ix_rx_refresh(rxr, i); 302379251f5eSSepherosa Ziehau nsegs = 0; 302479251f5eSSepherosa Ziehau } 302579251f5eSSepherosa Ziehau } 302679251f5eSSepherosa Ziehau rxr->rx_next_check = i; 302779251f5eSSepherosa Ziehau 302879251f5eSSepherosa Ziehau if (nsegs > 0) 302979251f5eSSepherosa Ziehau ix_rx_refresh(rxr, i); 303079251f5eSSepherosa Ziehau } 303179251f5eSSepherosa Ziehau 303279251f5eSSepherosa Ziehau static void 303379251f5eSSepherosa Ziehau ix_set_vlan(struct ix_softc *sc) 303479251f5eSSepherosa Ziehau { 303579251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 303679251f5eSSepherosa Ziehau uint32_t ctrl; 303779251f5eSSepherosa Ziehau 303879251f5eSSepherosa Ziehau if (hw->mac.type == ixgbe_mac_82598EB) { 303979251f5eSSepherosa Ziehau ctrl = IXGBE_READ_REG(hw, IXGBE_VLNCTRL); 304079251f5eSSepherosa Ziehau ctrl |= IXGBE_VLNCTRL_VME; 304179251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_VLNCTRL, ctrl); 304279251f5eSSepherosa Ziehau } else { 304379251f5eSSepherosa Ziehau int i; 304479251f5eSSepherosa Ziehau 304579251f5eSSepherosa Ziehau /* 304679251f5eSSepherosa Ziehau * On 82599 and later chips the VLAN enable is 304779251f5eSSepherosa Ziehau * per queue in RXDCTL 304879251f5eSSepherosa Ziehau */ 304979251f5eSSepherosa Ziehau for (i = 0; i < sc->rx_ring_inuse; ++i) { 305079251f5eSSepherosa Ziehau ctrl = IXGBE_READ_REG(hw, IXGBE_RXDCTL(i)); 305179251f5eSSepherosa Ziehau ctrl |= IXGBE_RXDCTL_VME; 305279251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_RXDCTL(i), ctrl); 305379251f5eSSepherosa Ziehau } 305479251f5eSSepherosa Ziehau } 305579251f5eSSepherosa Ziehau } 305679251f5eSSepherosa Ziehau 305779251f5eSSepherosa Ziehau static void 305879251f5eSSepherosa Ziehau ix_enable_intr(struct ix_softc *sc) 305979251f5eSSepherosa Ziehau { 306079251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 3061189a0ff3SSepherosa Ziehau uint32_t fwsm; 306279251f5eSSepherosa Ziehau int i; 306379251f5eSSepherosa Ziehau 306479251f5eSSepherosa Ziehau for (i = 0; i < sc->intr_cnt; ++i) 306579251f5eSSepherosa Ziehau lwkt_serialize_handler_enable(sc->intr_data[i].intr_serialize); 306679251f5eSSepherosa Ziehau 3067189a0ff3SSepherosa Ziehau sc->intr_mask = (IXGBE_EIMS_ENABLE_MASK & ~IXGBE_EIMS_RTX_QUEUE); 306879251f5eSSepherosa Ziehau 306979251f5eSSepherosa Ziehau /* Enable Fan Failure detection */ 307079251f5eSSepherosa Ziehau if (hw->device_id == IXGBE_DEV_ID_82598AT) 3071189a0ff3SSepherosa Ziehau sc->intr_mask |= IXGBE_EIMS_GPI_SDP1; 307279251f5eSSepherosa Ziehau 307363d483cdSSepherosa Ziehau switch (hw->mac.type) { 307479251f5eSSepherosa Ziehau case ixgbe_mac_82599EB: 3075189a0ff3SSepherosa Ziehau sc->intr_mask |= IXGBE_EIMS_ECC; 307663d483cdSSepherosa Ziehau /* Temperature sensor on some adapters */ 3077189a0ff3SSepherosa Ziehau sc->intr_mask |= IXGBE_EIMS_GPI_SDP0; 307863d483cdSSepherosa Ziehau /* SFP+ (RX_LOS_N & MOD_ABS_N) */ 3079189a0ff3SSepherosa Ziehau sc->intr_mask |= IXGBE_EIMS_GPI_SDP1; 3080189a0ff3SSepherosa Ziehau sc->intr_mask |= IXGBE_EIMS_GPI_SDP2; 308179251f5eSSepherosa Ziehau break; 3082189a0ff3SSepherosa Ziehau 308379251f5eSSepherosa Ziehau case ixgbe_mac_X540: 3084189a0ff3SSepherosa Ziehau sc->intr_mask |= IXGBE_EIMS_ECC; 308579251f5eSSepherosa Ziehau /* Detect if Thermal Sensor is enabled */ 308679251f5eSSepherosa Ziehau fwsm = IXGBE_READ_REG(hw, IXGBE_FWSM); 308779251f5eSSepherosa Ziehau if (fwsm & IXGBE_FWSM_TS_ENABLED) 3088189a0ff3SSepherosa Ziehau sc->intr_mask |= IXGBE_EIMS_TS; 308963d483cdSSepherosa Ziehau break; 309063d483cdSSepherosa Ziehau 309163d483cdSSepherosa Ziehau case ixgbe_mac_X550: 309263d483cdSSepherosa Ziehau case ixgbe_mac_X550EM_a: 309363d483cdSSepherosa Ziehau case ixgbe_mac_X550EM_x: 309463d483cdSSepherosa Ziehau sc->intr_mask |= IXGBE_EIMS_ECC; 309563d483cdSSepherosa Ziehau /* MAC thermal sensor is automatically enabled */ 309663d483cdSSepherosa Ziehau sc->intr_mask |= IXGBE_EIMS_TS; 309763d483cdSSepherosa Ziehau /* Some devices use SDP0 for important information */ 309863d483cdSSepherosa Ziehau if (hw->device_id == IXGBE_DEV_ID_X550EM_X_SFP || 309963d483cdSSepherosa Ziehau hw->device_id == IXGBE_DEV_ID_X550EM_X_10G_T) 310063d483cdSSepherosa Ziehau sc->intr_mask |= IXGBE_EIMS_GPI_SDP0_BY_MAC(hw); 310179251f5eSSepherosa Ziehau /* FALL THROUGH */ 310279251f5eSSepherosa Ziehau default: 310379251f5eSSepherosa Ziehau break; 310479251f5eSSepherosa Ziehau } 310579251f5eSSepherosa Ziehau 3106189a0ff3SSepherosa Ziehau /* With MSI-X we use auto clear for RX and TX rings */ 310779251f5eSSepherosa Ziehau if (sc->intr_type == PCI_INTR_TYPE_MSIX) { 3108189a0ff3SSepherosa Ziehau /* 3109189a0ff3SSepherosa Ziehau * There are no EIAC1/EIAC2 for newer chips; the related 3110189a0ff3SSepherosa Ziehau * bits for TX and RX rings > 16 are always auto clear. 3111189a0ff3SSepherosa Ziehau * 3112189a0ff3SSepherosa Ziehau * XXX which bits? There are _no_ documented EICR1 and 3113189a0ff3SSepherosa Ziehau * EICR2 at all; only EICR. 3114189a0ff3SSepherosa Ziehau */ 3115189a0ff3SSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_EIAC, IXGBE_EIMS_RTX_QUEUE); 311679251f5eSSepherosa Ziehau } else { 3117189a0ff3SSepherosa Ziehau sc->intr_mask |= IX_TX_INTR_MASK | IX_RX0_INTR_MASK; 311879251f5eSSepherosa Ziehau 311979251f5eSSepherosa Ziehau KKASSERT(sc->rx_ring_inuse <= IX_MIN_RXRING_RSS); 312079251f5eSSepherosa Ziehau if (sc->rx_ring_inuse == IX_MIN_RXRING_RSS) 312179251f5eSSepherosa Ziehau sc->intr_mask |= IX_RX1_INTR_MASK; 312279251f5eSSepherosa Ziehau } 312379251f5eSSepherosa Ziehau 312479251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_EIMS, sc->intr_mask); 3125189a0ff3SSepherosa Ziehau 3126189a0ff3SSepherosa Ziehau /* 3127189a0ff3SSepherosa Ziehau * Enable RX and TX rings for MSI-X 3128189a0ff3SSepherosa Ziehau */ 3129189a0ff3SSepherosa Ziehau if (sc->intr_type == PCI_INTR_TYPE_MSIX) { 3130189a0ff3SSepherosa Ziehau for (i = 0; i < sc->tx_ring_inuse; ++i) { 3131189a0ff3SSepherosa Ziehau const struct ix_tx_ring *txr = &sc->tx_rings[i]; 3132189a0ff3SSepherosa Ziehau 3133189a0ff3SSepherosa Ziehau if (txr->tx_intr_vec >= 0) { 3134189a0ff3SSepherosa Ziehau IXGBE_WRITE_REG(hw, txr->tx_eims, 3135189a0ff3SSepherosa Ziehau txr->tx_eims_val); 3136189a0ff3SSepherosa Ziehau } 3137189a0ff3SSepherosa Ziehau } 3138189a0ff3SSepherosa Ziehau for (i = 0; i < sc->rx_ring_inuse; ++i) { 3139189a0ff3SSepherosa Ziehau const struct ix_rx_ring *rxr = &sc->rx_rings[i]; 3140189a0ff3SSepherosa Ziehau 3141189a0ff3SSepherosa Ziehau KKASSERT(rxr->rx_intr_vec >= 0); 3142189a0ff3SSepherosa Ziehau IXGBE_WRITE_REG(hw, rxr->rx_eims, rxr->rx_eims_val); 3143189a0ff3SSepherosa Ziehau } 3144189a0ff3SSepherosa Ziehau } 314579251f5eSSepherosa Ziehau 314679251f5eSSepherosa Ziehau IXGBE_WRITE_FLUSH(hw); 314779251f5eSSepherosa Ziehau } 314879251f5eSSepherosa Ziehau 314979251f5eSSepherosa Ziehau static void 315079251f5eSSepherosa Ziehau ix_disable_intr(struct ix_softc *sc) 315179251f5eSSepherosa Ziehau { 315279251f5eSSepherosa Ziehau int i; 315379251f5eSSepherosa Ziehau 3154189a0ff3SSepherosa Ziehau if (sc->intr_type == PCI_INTR_TYPE_MSIX) 315579251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_EIAC, 0); 3156189a0ff3SSepherosa Ziehau 315779251f5eSSepherosa Ziehau if (sc->hw.mac.type == ixgbe_mac_82598EB) { 315879251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_EIMC, ~0); 315979251f5eSSepherosa Ziehau } else { 316079251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_EIMC, 0xFFFF0000); 316179251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_EIMC_EX(0), ~0); 316279251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_EIMC_EX(1), ~0); 316379251f5eSSepherosa Ziehau } 316479251f5eSSepherosa Ziehau IXGBE_WRITE_FLUSH(&sc->hw); 316579251f5eSSepherosa Ziehau 316679251f5eSSepherosa Ziehau for (i = 0; i < sc->intr_cnt; ++i) 316779251f5eSSepherosa Ziehau lwkt_serialize_handler_disable(sc->intr_data[i].intr_serialize); 316879251f5eSSepherosa Ziehau } 316979251f5eSSepherosa Ziehau 317079251f5eSSepherosa Ziehau uint16_t 317179251f5eSSepherosa Ziehau ixgbe_read_pci_cfg(struct ixgbe_hw *hw, uint32_t reg) 317279251f5eSSepherosa Ziehau { 317379251f5eSSepherosa Ziehau return pci_read_config(((struct ixgbe_osdep *)hw->back)->dev, 317479251f5eSSepherosa Ziehau reg, 2); 317579251f5eSSepherosa Ziehau } 317679251f5eSSepherosa Ziehau 317779251f5eSSepherosa Ziehau void 317879251f5eSSepherosa Ziehau ixgbe_write_pci_cfg(struct ixgbe_hw *hw, uint32_t reg, uint16_t value) 317979251f5eSSepherosa Ziehau { 318079251f5eSSepherosa Ziehau pci_write_config(((struct ixgbe_osdep *)hw->back)->dev, 318179251f5eSSepherosa Ziehau reg, value, 2); 318279251f5eSSepherosa Ziehau } 318379251f5eSSepherosa Ziehau 318479251f5eSSepherosa Ziehau static void 318579251f5eSSepherosa Ziehau ix_slot_info(struct ix_softc *sc) 318679251f5eSSepherosa Ziehau { 318779251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 318879251f5eSSepherosa Ziehau device_t dev = sc->dev; 318979251f5eSSepherosa Ziehau struct ixgbe_mac_info *mac = &hw->mac; 319079251f5eSSepherosa Ziehau uint16_t link; 319179251f5eSSepherosa Ziehau uint32_t offset; 319279251f5eSSepherosa Ziehau 319379251f5eSSepherosa Ziehau /* For most devices simply call the shared code routine */ 319479251f5eSSepherosa Ziehau if (hw->device_id != IXGBE_DEV_ID_82599_SFP_SF_QP) { 319579251f5eSSepherosa Ziehau ixgbe_get_bus_info(hw); 319663d483cdSSepherosa Ziehau /* These devices don't use PCI-E */ 319763d483cdSSepherosa Ziehau if (hw->mac.type == ixgbe_mac_X550EM_x || 319863d483cdSSepherosa Ziehau hw->mac.type == ixgbe_mac_X550EM_a) 319963d483cdSSepherosa Ziehau return; 320079251f5eSSepherosa Ziehau goto display; 320179251f5eSSepherosa Ziehau } 320279251f5eSSepherosa Ziehau 320379251f5eSSepherosa Ziehau /* 320479251f5eSSepherosa Ziehau * For the Quad port adapter we need to parse back 320579251f5eSSepherosa Ziehau * up the PCI tree to find the speed of the expansion 320679251f5eSSepherosa Ziehau * slot into which this adapter is plugged. A bit more work. 320779251f5eSSepherosa Ziehau */ 320879251f5eSSepherosa Ziehau dev = device_get_parent(device_get_parent(dev)); 320979251f5eSSepherosa Ziehau #ifdef IXGBE_DEBUG 321079251f5eSSepherosa Ziehau device_printf(dev, "parent pcib = %x,%x,%x\n", 321179251f5eSSepherosa Ziehau pci_get_bus(dev), pci_get_slot(dev), pci_get_function(dev)); 321279251f5eSSepherosa Ziehau #endif 321379251f5eSSepherosa Ziehau dev = device_get_parent(device_get_parent(dev)); 321479251f5eSSepherosa Ziehau #ifdef IXGBE_DEBUG 321579251f5eSSepherosa Ziehau device_printf(dev, "slot pcib = %x,%x,%x\n", 321679251f5eSSepherosa Ziehau pci_get_bus(dev), pci_get_slot(dev), pci_get_function(dev)); 321779251f5eSSepherosa Ziehau #endif 321879251f5eSSepherosa Ziehau /* Now get the PCI Express Capabilities offset */ 321979251f5eSSepherosa Ziehau offset = pci_get_pciecap_ptr(dev); 322079251f5eSSepherosa Ziehau /* ...and read the Link Status Register */ 322179251f5eSSepherosa Ziehau link = pci_read_config(dev, offset + PCIER_LINKSTAT, 2); 322279251f5eSSepherosa Ziehau switch (link & IXGBE_PCI_LINK_WIDTH) { 322379251f5eSSepherosa Ziehau case IXGBE_PCI_LINK_WIDTH_1: 322479251f5eSSepherosa Ziehau hw->bus.width = ixgbe_bus_width_pcie_x1; 322579251f5eSSepherosa Ziehau break; 322679251f5eSSepherosa Ziehau case IXGBE_PCI_LINK_WIDTH_2: 322779251f5eSSepherosa Ziehau hw->bus.width = ixgbe_bus_width_pcie_x2; 322879251f5eSSepherosa Ziehau break; 322979251f5eSSepherosa Ziehau case IXGBE_PCI_LINK_WIDTH_4: 323079251f5eSSepherosa Ziehau hw->bus.width = ixgbe_bus_width_pcie_x4; 323179251f5eSSepherosa Ziehau break; 323279251f5eSSepherosa Ziehau case IXGBE_PCI_LINK_WIDTH_8: 323379251f5eSSepherosa Ziehau hw->bus.width = ixgbe_bus_width_pcie_x8; 323479251f5eSSepherosa Ziehau break; 323579251f5eSSepherosa Ziehau default: 323679251f5eSSepherosa Ziehau hw->bus.width = ixgbe_bus_width_unknown; 323779251f5eSSepherosa Ziehau break; 323879251f5eSSepherosa Ziehau } 323979251f5eSSepherosa Ziehau 324079251f5eSSepherosa Ziehau switch (link & IXGBE_PCI_LINK_SPEED) { 324179251f5eSSepherosa Ziehau case IXGBE_PCI_LINK_SPEED_2500: 324279251f5eSSepherosa Ziehau hw->bus.speed = ixgbe_bus_speed_2500; 324379251f5eSSepherosa Ziehau break; 324479251f5eSSepherosa Ziehau case IXGBE_PCI_LINK_SPEED_5000: 324579251f5eSSepherosa Ziehau hw->bus.speed = ixgbe_bus_speed_5000; 324679251f5eSSepherosa Ziehau break; 324779251f5eSSepherosa Ziehau case IXGBE_PCI_LINK_SPEED_8000: 324879251f5eSSepherosa Ziehau hw->bus.speed = ixgbe_bus_speed_8000; 324979251f5eSSepherosa Ziehau break; 325079251f5eSSepherosa Ziehau default: 325179251f5eSSepherosa Ziehau hw->bus.speed = ixgbe_bus_speed_unknown; 325279251f5eSSepherosa Ziehau break; 325379251f5eSSepherosa Ziehau } 325479251f5eSSepherosa Ziehau 325579251f5eSSepherosa Ziehau mac->ops.set_lan_id(hw); 325679251f5eSSepherosa Ziehau 325779251f5eSSepherosa Ziehau display: 325879251f5eSSepherosa Ziehau device_printf(dev, "PCI Express Bus: Speed %s %s\n", 325979251f5eSSepherosa Ziehau hw->bus.speed == ixgbe_bus_speed_8000 ? "8.0GT/s" : 326079251f5eSSepherosa Ziehau hw->bus.speed == ixgbe_bus_speed_5000 ? "5.0GT/s" : 326179251f5eSSepherosa Ziehau hw->bus.speed == ixgbe_bus_speed_2500 ? "2.5GT/s" : "Unknown", 326279251f5eSSepherosa Ziehau hw->bus.width == ixgbe_bus_width_pcie_x8 ? "Width x8" : 326379251f5eSSepherosa Ziehau hw->bus.width == ixgbe_bus_width_pcie_x4 ? "Width x4" : 326479251f5eSSepherosa Ziehau hw->bus.width == ixgbe_bus_width_pcie_x1 ? "Width x1" : "Unknown"); 326579251f5eSSepherosa Ziehau 326679251f5eSSepherosa Ziehau if (hw->device_id != IXGBE_DEV_ID_82599_SFP_SF_QP && 326779251f5eSSepherosa Ziehau hw->bus.width <= ixgbe_bus_width_pcie_x4 && 326879251f5eSSepherosa Ziehau hw->bus.speed == ixgbe_bus_speed_2500) { 326979251f5eSSepherosa Ziehau device_printf(dev, "For optimal performance a x8 " 327079251f5eSSepherosa Ziehau "PCIE, or x4 PCIE Gen2 slot is required.\n"); 327179251f5eSSepherosa Ziehau } else if (hw->device_id == IXGBE_DEV_ID_82599_SFP_SF_QP && 327279251f5eSSepherosa Ziehau hw->bus.width <= ixgbe_bus_width_pcie_x8 && 327379251f5eSSepherosa Ziehau hw->bus.speed < ixgbe_bus_speed_8000) { 327479251f5eSSepherosa Ziehau device_printf(dev, "For optimal performance a x8 " 327579251f5eSSepherosa Ziehau "PCIE Gen3 slot is required.\n"); 327679251f5eSSepherosa Ziehau } 327779251f5eSSepherosa Ziehau } 327879251f5eSSepherosa Ziehau 327979251f5eSSepherosa Ziehau /* 328079251f5eSSepherosa Ziehau * TODO comment is incorrect 328179251f5eSSepherosa Ziehau * 328279251f5eSSepherosa Ziehau * Setup the correct IVAR register for a particular MSIX interrupt 328379251f5eSSepherosa Ziehau * - entry is the register array entry 328479251f5eSSepherosa Ziehau * - vector is the MSIX vector for this queue 328579251f5eSSepherosa Ziehau * - type is RX/TX/MISC 328679251f5eSSepherosa Ziehau */ 328779251f5eSSepherosa Ziehau static void 328879251f5eSSepherosa Ziehau ix_set_ivar(struct ix_softc *sc, uint8_t entry, uint8_t vector, 328979251f5eSSepherosa Ziehau int8_t type) 329079251f5eSSepherosa Ziehau { 329179251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 329279251f5eSSepherosa Ziehau uint32_t ivar, index; 329379251f5eSSepherosa Ziehau 329479251f5eSSepherosa Ziehau vector |= IXGBE_IVAR_ALLOC_VAL; 329579251f5eSSepherosa Ziehau 329679251f5eSSepherosa Ziehau switch (hw->mac.type) { 329779251f5eSSepherosa Ziehau case ixgbe_mac_82598EB: 329879251f5eSSepherosa Ziehau if (type == -1) 329979251f5eSSepherosa Ziehau entry = IXGBE_IVAR_OTHER_CAUSES_INDEX; 330079251f5eSSepherosa Ziehau else 330179251f5eSSepherosa Ziehau entry += (type * 64); 330279251f5eSSepherosa Ziehau index = (entry >> 2) & 0x1F; 330379251f5eSSepherosa Ziehau ivar = IXGBE_READ_REG(hw, IXGBE_IVAR(index)); 330479251f5eSSepherosa Ziehau ivar &= ~(0xFF << (8 * (entry & 0x3))); 330579251f5eSSepherosa Ziehau ivar |= (vector << (8 * (entry & 0x3))); 330679251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_IVAR(index), ivar); 330779251f5eSSepherosa Ziehau break; 330879251f5eSSepherosa Ziehau 330979251f5eSSepherosa Ziehau case ixgbe_mac_82599EB: 331079251f5eSSepherosa Ziehau case ixgbe_mac_X540: 331163d483cdSSepherosa Ziehau case ixgbe_mac_X550: 331263d483cdSSepherosa Ziehau case ixgbe_mac_X550EM_a: 331363d483cdSSepherosa Ziehau case ixgbe_mac_X550EM_x: 331479251f5eSSepherosa Ziehau if (type == -1) { /* MISC IVAR */ 331579251f5eSSepherosa Ziehau index = (entry & 1) * 8; 331679251f5eSSepherosa Ziehau ivar = IXGBE_READ_REG(hw, IXGBE_IVAR_MISC); 331779251f5eSSepherosa Ziehau ivar &= ~(0xFF << index); 331879251f5eSSepherosa Ziehau ivar |= (vector << index); 331979251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_IVAR_MISC, ivar); 332079251f5eSSepherosa Ziehau } else { /* RX/TX IVARS */ 332179251f5eSSepherosa Ziehau index = (16 * (entry & 1)) + (8 * type); 332279251f5eSSepherosa Ziehau ivar = IXGBE_READ_REG(hw, IXGBE_IVAR(entry >> 1)); 332379251f5eSSepherosa Ziehau ivar &= ~(0xFF << index); 332479251f5eSSepherosa Ziehau ivar |= (vector << index); 332579251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_IVAR(entry >> 1), ivar); 332679251f5eSSepherosa Ziehau } 332763d483cdSSepherosa Ziehau /* FALL THROUGH */ 332879251f5eSSepherosa Ziehau default: 332979251f5eSSepherosa Ziehau break; 333079251f5eSSepherosa Ziehau } 333179251f5eSSepherosa Ziehau } 333279251f5eSSepherosa Ziehau 333379251f5eSSepherosa Ziehau static boolean_t 333479251f5eSSepherosa Ziehau ix_sfp_probe(struct ix_softc *sc) 333579251f5eSSepherosa Ziehau { 333679251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 333779251f5eSSepherosa Ziehau 333879251f5eSSepherosa Ziehau if (hw->phy.type == ixgbe_phy_nl && 333979251f5eSSepherosa Ziehau hw->phy.sfp_type == ixgbe_sfp_type_not_present) { 334079251f5eSSepherosa Ziehau int32_t ret; 334179251f5eSSepherosa Ziehau 334279251f5eSSepherosa Ziehau ret = hw->phy.ops.identify_sfp(hw); 334379251f5eSSepherosa Ziehau if (ret) 334479251f5eSSepherosa Ziehau return FALSE; 334579251f5eSSepherosa Ziehau 334679251f5eSSepherosa Ziehau ret = hw->phy.ops.reset(hw); 334779251f5eSSepherosa Ziehau if (ret == IXGBE_ERR_SFP_NOT_SUPPORTED) { 334879251f5eSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, 334979251f5eSSepherosa Ziehau "Unsupported SFP+ module detected! " 335079251f5eSSepherosa Ziehau "Reload driver with supported module.\n"); 335179251f5eSSepherosa Ziehau sc->sfp_probe = FALSE; 335279251f5eSSepherosa Ziehau return FALSE; 335379251f5eSSepherosa Ziehau } 335479251f5eSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, "SFP+ module detected!\n"); 335579251f5eSSepherosa Ziehau 335679251f5eSSepherosa Ziehau /* We now have supported optics */ 335779251f5eSSepherosa Ziehau sc->sfp_probe = FALSE; 335879251f5eSSepherosa Ziehau 335979251f5eSSepherosa Ziehau return TRUE; 336079251f5eSSepherosa Ziehau } 336179251f5eSSepherosa Ziehau return FALSE; 336279251f5eSSepherosa Ziehau } 336379251f5eSSepherosa Ziehau 336479251f5eSSepherosa Ziehau static void 336579251f5eSSepherosa Ziehau ix_handle_link(struct ix_softc *sc) 336679251f5eSSepherosa Ziehau { 336779251f5eSSepherosa Ziehau ixgbe_check_link(&sc->hw, &sc->link_speed, &sc->link_up, 0); 336879251f5eSSepherosa Ziehau ix_update_link_status(sc); 336979251f5eSSepherosa Ziehau } 337079251f5eSSepherosa Ziehau 337179251f5eSSepherosa Ziehau /* 337279251f5eSSepherosa Ziehau * Handling SFP module 337379251f5eSSepherosa Ziehau */ 337479251f5eSSepherosa Ziehau static void 337579251f5eSSepherosa Ziehau ix_handle_mod(struct ix_softc *sc) 337679251f5eSSepherosa Ziehau { 337779251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 337879251f5eSSepherosa Ziehau uint32_t err; 337979251f5eSSepherosa Ziehau 338079251f5eSSepherosa Ziehau err = hw->phy.ops.identify_sfp(hw); 338179251f5eSSepherosa Ziehau if (err == IXGBE_ERR_SFP_NOT_SUPPORTED) { 338279251f5eSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, 338379251f5eSSepherosa Ziehau "Unsupported SFP+ module type was detected.\n"); 338479251f5eSSepherosa Ziehau return; 338579251f5eSSepherosa Ziehau } 338679251f5eSSepherosa Ziehau err = hw->mac.ops.setup_sfp(hw); 338779251f5eSSepherosa Ziehau if (err == IXGBE_ERR_SFP_NOT_SUPPORTED) { 338879251f5eSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, 338979251f5eSSepherosa Ziehau "Setup failure - unsupported SFP+ module type.\n"); 339079251f5eSSepherosa Ziehau return; 339179251f5eSSepherosa Ziehau } 339279251f5eSSepherosa Ziehau ix_handle_msf(sc); 339379251f5eSSepherosa Ziehau } 339479251f5eSSepherosa Ziehau 339579251f5eSSepherosa Ziehau /* 339679251f5eSSepherosa Ziehau * Handling MSF (multispeed fiber) 339779251f5eSSepherosa Ziehau */ 339879251f5eSSepherosa Ziehau static void 339979251f5eSSepherosa Ziehau ix_handle_msf(struct ix_softc *sc) 340079251f5eSSepherosa Ziehau { 340179251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 340279251f5eSSepherosa Ziehau uint32_t autoneg; 340379251f5eSSepherosa Ziehau 340463d483cdSSepherosa Ziehau hw->phy.ops.identify_sfp(hw); 340563d483cdSSepherosa Ziehau ix_init_media(sc); 340663d483cdSSepherosa Ziehau 340763d483cdSSepherosa Ziehau if (sc->advspeed != IXGBE_LINK_SPEED_UNKNOWN) 340863d483cdSSepherosa Ziehau autoneg = sc->advspeed; 340963d483cdSSepherosa Ziehau else 341079251f5eSSepherosa Ziehau autoneg = hw->phy.autoneg_advertised; 341179251f5eSSepherosa Ziehau if (!autoneg && hw->mac.ops.get_link_capabilities != NULL) { 341279251f5eSSepherosa Ziehau bool negotiate; 341379251f5eSSepherosa Ziehau 341479251f5eSSepherosa Ziehau hw->mac.ops.get_link_capabilities(hw, &autoneg, &negotiate); 341579251f5eSSepherosa Ziehau } 341679251f5eSSepherosa Ziehau if (hw->mac.ops.setup_link != NULL) 341779251f5eSSepherosa Ziehau hw->mac.ops.setup_link(hw, autoneg, TRUE); 341879251f5eSSepherosa Ziehau } 341979251f5eSSepherosa Ziehau 342079251f5eSSepherosa Ziehau static void 342163d483cdSSepherosa Ziehau ix_handle_phy(struct ix_softc *sc) 342263d483cdSSepherosa Ziehau { 342363d483cdSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 342463d483cdSSepherosa Ziehau int error; 342563d483cdSSepherosa Ziehau 342663d483cdSSepherosa Ziehau error = hw->phy.ops.handle_lasi(hw); 342763d483cdSSepherosa Ziehau if (error == IXGBE_ERR_OVERTEMP) { 342863d483cdSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, 342963d483cdSSepherosa Ziehau "CRITICAL: EXTERNAL PHY OVER TEMP!! " 343063d483cdSSepherosa Ziehau "PHY will downshift to lower power state!\n"); 343163d483cdSSepherosa Ziehau } else if (error) { 343263d483cdSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, 343363d483cdSSepherosa Ziehau "Error handling LASI interrupt: %d\n", error); 343463d483cdSSepherosa Ziehau } 343563d483cdSSepherosa Ziehau } 343663d483cdSSepherosa Ziehau 343763d483cdSSepherosa Ziehau static void 343879251f5eSSepherosa Ziehau ix_update_stats(struct ix_softc *sc) 343979251f5eSSepherosa Ziehau { 344079251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 344179251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 344279251f5eSSepherosa Ziehau uint32_t missed_rx = 0, bprc, lxon, lxoff, total; 344379251f5eSSepherosa Ziehau uint64_t total_missed_rx = 0; 344479251f5eSSepherosa Ziehau int i; 344579251f5eSSepherosa Ziehau 344679251f5eSSepherosa Ziehau sc->stats.crcerrs += IXGBE_READ_REG(hw, IXGBE_CRCERRS); 344779251f5eSSepherosa Ziehau sc->stats.illerrc += IXGBE_READ_REG(hw, IXGBE_ILLERRC); 344879251f5eSSepherosa Ziehau sc->stats.errbc += IXGBE_READ_REG(hw, IXGBE_ERRBC); 344979251f5eSSepherosa Ziehau sc->stats.mspdc += IXGBE_READ_REG(hw, IXGBE_MSPDC); 345079251f5eSSepherosa Ziehau 345179251f5eSSepherosa Ziehau for (i = 0; i < 16; i++) { 345279251f5eSSepherosa Ziehau sc->stats.qprc[i] += IXGBE_READ_REG(hw, IXGBE_QPRC(i)); 345379251f5eSSepherosa Ziehau sc->stats.qptc[i] += IXGBE_READ_REG(hw, IXGBE_QPTC(i)); 345479251f5eSSepherosa Ziehau sc->stats.qprdc[i] += IXGBE_READ_REG(hw, IXGBE_QPRDC(i)); 345579251f5eSSepherosa Ziehau } 345679251f5eSSepherosa Ziehau sc->stats.mlfc += IXGBE_READ_REG(hw, IXGBE_MLFC); 345779251f5eSSepherosa Ziehau sc->stats.mrfc += IXGBE_READ_REG(hw, IXGBE_MRFC); 345879251f5eSSepherosa Ziehau sc->stats.rlec += IXGBE_READ_REG(hw, IXGBE_RLEC); 345979251f5eSSepherosa Ziehau 346079251f5eSSepherosa Ziehau /* Hardware workaround, gprc counts missed packets */ 346179251f5eSSepherosa Ziehau sc->stats.gprc += IXGBE_READ_REG(hw, IXGBE_GPRC); 346279251f5eSSepherosa Ziehau sc->stats.gprc -= missed_rx; 346379251f5eSSepherosa Ziehau 346479251f5eSSepherosa Ziehau if (hw->mac.type != ixgbe_mac_82598EB) { 346579251f5eSSepherosa Ziehau sc->stats.gorc += IXGBE_READ_REG(hw, IXGBE_GORCL) + 346679251f5eSSepherosa Ziehau ((uint64_t)IXGBE_READ_REG(hw, IXGBE_GORCH) << 32); 346779251f5eSSepherosa Ziehau sc->stats.gotc += IXGBE_READ_REG(hw, IXGBE_GOTCL) + 346879251f5eSSepherosa Ziehau ((uint64_t)IXGBE_READ_REG(hw, IXGBE_GOTCH) << 32); 346979251f5eSSepherosa Ziehau sc->stats.tor += IXGBE_READ_REG(hw, IXGBE_TORL) + 347079251f5eSSepherosa Ziehau ((uint64_t)IXGBE_READ_REG(hw, IXGBE_TORH) << 32); 347179251f5eSSepherosa Ziehau sc->stats.lxonrxc += IXGBE_READ_REG(hw, IXGBE_LXONRXCNT); 347279251f5eSSepherosa Ziehau sc->stats.lxoffrxc += IXGBE_READ_REG(hw, IXGBE_LXOFFRXCNT); 347379251f5eSSepherosa Ziehau } else { 347479251f5eSSepherosa Ziehau sc->stats.lxonrxc += IXGBE_READ_REG(hw, IXGBE_LXONRXC); 347579251f5eSSepherosa Ziehau sc->stats.lxoffrxc += IXGBE_READ_REG(hw, IXGBE_LXOFFRXC); 347679251f5eSSepherosa Ziehau /* 82598 only has a counter in the high register */ 347779251f5eSSepherosa Ziehau sc->stats.gorc += IXGBE_READ_REG(hw, IXGBE_GORCH); 347879251f5eSSepherosa Ziehau sc->stats.gotc += IXGBE_READ_REG(hw, IXGBE_GOTCH); 347979251f5eSSepherosa Ziehau sc->stats.tor += IXGBE_READ_REG(hw, IXGBE_TORH); 348079251f5eSSepherosa Ziehau } 348179251f5eSSepherosa Ziehau 348279251f5eSSepherosa Ziehau /* 348379251f5eSSepherosa Ziehau * Workaround: mprc hardware is incorrectly counting 348479251f5eSSepherosa Ziehau * broadcasts, so for now we subtract those. 348579251f5eSSepherosa Ziehau */ 348679251f5eSSepherosa Ziehau bprc = IXGBE_READ_REG(hw, IXGBE_BPRC); 348779251f5eSSepherosa Ziehau sc->stats.bprc += bprc; 348879251f5eSSepherosa Ziehau sc->stats.mprc += IXGBE_READ_REG(hw, IXGBE_MPRC); 348979251f5eSSepherosa Ziehau if (hw->mac.type == ixgbe_mac_82598EB) 349079251f5eSSepherosa Ziehau sc->stats.mprc -= bprc; 349179251f5eSSepherosa Ziehau 349279251f5eSSepherosa Ziehau sc->stats.prc64 += IXGBE_READ_REG(hw, IXGBE_PRC64); 349379251f5eSSepherosa Ziehau sc->stats.prc127 += IXGBE_READ_REG(hw, IXGBE_PRC127); 349479251f5eSSepherosa Ziehau sc->stats.prc255 += IXGBE_READ_REG(hw, IXGBE_PRC255); 349579251f5eSSepherosa Ziehau sc->stats.prc511 += IXGBE_READ_REG(hw, IXGBE_PRC511); 349679251f5eSSepherosa Ziehau sc->stats.prc1023 += IXGBE_READ_REG(hw, IXGBE_PRC1023); 349779251f5eSSepherosa Ziehau sc->stats.prc1522 += IXGBE_READ_REG(hw, IXGBE_PRC1522); 349879251f5eSSepherosa Ziehau 349979251f5eSSepherosa Ziehau lxon = IXGBE_READ_REG(hw, IXGBE_LXONTXC); 350079251f5eSSepherosa Ziehau sc->stats.lxontxc += lxon; 350179251f5eSSepherosa Ziehau lxoff = IXGBE_READ_REG(hw, IXGBE_LXOFFTXC); 350279251f5eSSepherosa Ziehau sc->stats.lxofftxc += lxoff; 350379251f5eSSepherosa Ziehau total = lxon + lxoff; 350479251f5eSSepherosa Ziehau 350579251f5eSSepherosa Ziehau sc->stats.gptc += IXGBE_READ_REG(hw, IXGBE_GPTC); 350679251f5eSSepherosa Ziehau sc->stats.mptc += IXGBE_READ_REG(hw, IXGBE_MPTC); 350779251f5eSSepherosa Ziehau sc->stats.ptc64 += IXGBE_READ_REG(hw, IXGBE_PTC64); 350879251f5eSSepherosa Ziehau sc->stats.gptc -= total; 350979251f5eSSepherosa Ziehau sc->stats.mptc -= total; 351079251f5eSSepherosa Ziehau sc->stats.ptc64 -= total; 351179251f5eSSepherosa Ziehau sc->stats.gotc -= total * ETHER_MIN_LEN; 351279251f5eSSepherosa Ziehau 351379251f5eSSepherosa Ziehau sc->stats.ruc += IXGBE_READ_REG(hw, IXGBE_RUC); 351479251f5eSSepherosa Ziehau sc->stats.rfc += IXGBE_READ_REG(hw, IXGBE_RFC); 351579251f5eSSepherosa Ziehau sc->stats.roc += IXGBE_READ_REG(hw, IXGBE_ROC); 351679251f5eSSepherosa Ziehau sc->stats.rjc += IXGBE_READ_REG(hw, IXGBE_RJC); 351779251f5eSSepherosa Ziehau sc->stats.mngprc += IXGBE_READ_REG(hw, IXGBE_MNGPRC); 351879251f5eSSepherosa Ziehau sc->stats.mngpdc += IXGBE_READ_REG(hw, IXGBE_MNGPDC); 351979251f5eSSepherosa Ziehau sc->stats.mngptc += IXGBE_READ_REG(hw, IXGBE_MNGPTC); 352079251f5eSSepherosa Ziehau sc->stats.tpr += IXGBE_READ_REG(hw, IXGBE_TPR); 352179251f5eSSepherosa Ziehau sc->stats.tpt += IXGBE_READ_REG(hw, IXGBE_TPT); 352279251f5eSSepherosa Ziehau sc->stats.ptc127 += IXGBE_READ_REG(hw, IXGBE_PTC127); 352379251f5eSSepherosa Ziehau sc->stats.ptc255 += IXGBE_READ_REG(hw, IXGBE_PTC255); 352479251f5eSSepherosa Ziehau sc->stats.ptc511 += IXGBE_READ_REG(hw, IXGBE_PTC511); 352579251f5eSSepherosa Ziehau sc->stats.ptc1023 += IXGBE_READ_REG(hw, IXGBE_PTC1023); 352679251f5eSSepherosa Ziehau sc->stats.ptc1522 += IXGBE_READ_REG(hw, IXGBE_PTC1522); 352779251f5eSSepherosa Ziehau sc->stats.bptc += IXGBE_READ_REG(hw, IXGBE_BPTC); 352879251f5eSSepherosa Ziehau sc->stats.xec += IXGBE_READ_REG(hw, IXGBE_XEC); 352979251f5eSSepherosa Ziehau sc->stats.fccrc += IXGBE_READ_REG(hw, IXGBE_FCCRC); 353079251f5eSSepherosa Ziehau sc->stats.fclast += IXGBE_READ_REG(hw, IXGBE_FCLAST); 353179251f5eSSepherosa Ziehau /* Only read FCOE on 82599 */ 353279251f5eSSepherosa Ziehau if (hw->mac.type != ixgbe_mac_82598EB) { 353379251f5eSSepherosa Ziehau sc->stats.fcoerpdc += IXGBE_READ_REG(hw, IXGBE_FCOERPDC); 353479251f5eSSepherosa Ziehau sc->stats.fcoeprc += IXGBE_READ_REG(hw, IXGBE_FCOEPRC); 353579251f5eSSepherosa Ziehau sc->stats.fcoeptc += IXGBE_READ_REG(hw, IXGBE_FCOEPTC); 353679251f5eSSepherosa Ziehau sc->stats.fcoedwrc += IXGBE_READ_REG(hw, IXGBE_FCOEDWRC); 353779251f5eSSepherosa Ziehau sc->stats.fcoedwtc += IXGBE_READ_REG(hw, IXGBE_FCOEDWTC); 353879251f5eSSepherosa Ziehau } 353979251f5eSSepherosa Ziehau 354079251f5eSSepherosa Ziehau /* Rx Errors */ 354179251f5eSSepherosa Ziehau IFNET_STAT_SET(ifp, iqdrops, total_missed_rx); 354279251f5eSSepherosa Ziehau IFNET_STAT_SET(ifp, ierrors, sc->stats.crcerrs + sc->stats.rlec); 354379251f5eSSepherosa Ziehau } 354479251f5eSSepherosa Ziehau 354579251f5eSSepherosa Ziehau #if 0 354679251f5eSSepherosa Ziehau /* 354779251f5eSSepherosa Ziehau * Add sysctl variables, one per statistic, to the system. 354879251f5eSSepherosa Ziehau */ 354979251f5eSSepherosa Ziehau static void 355079251f5eSSepherosa Ziehau ix_add_hw_stats(struct ix_softc *sc) 355179251f5eSSepherosa Ziehau { 355279251f5eSSepherosa Ziehau 355379251f5eSSepherosa Ziehau device_t dev = sc->dev; 355479251f5eSSepherosa Ziehau 355579251f5eSSepherosa Ziehau struct ix_tx_ring *txr = sc->tx_rings; 355679251f5eSSepherosa Ziehau struct ix_rx_ring *rxr = sc->rx_rings; 355779251f5eSSepherosa Ziehau 355879251f5eSSepherosa Ziehau struct sysctl_ctx_list *ctx = device_get_sysctl_ctx(dev); 355979251f5eSSepherosa Ziehau struct sysctl_oid *tree = device_get_sysctl_tree(dev); 356079251f5eSSepherosa Ziehau struct sysctl_oid_list *child = SYSCTL_CHILDREN(tree); 356179251f5eSSepherosa Ziehau struct ixgbe_hw_stats *stats = &sc->stats; 356279251f5eSSepherosa Ziehau 356379251f5eSSepherosa Ziehau struct sysctl_oid *stat_node, *queue_node; 356479251f5eSSepherosa Ziehau struct sysctl_oid_list *stat_list, *queue_list; 356579251f5eSSepherosa Ziehau 356679251f5eSSepherosa Ziehau #define QUEUE_NAME_LEN 32 356779251f5eSSepherosa Ziehau char namebuf[QUEUE_NAME_LEN]; 356879251f5eSSepherosa Ziehau 356979251f5eSSepherosa Ziehau /* MAC stats get the own sub node */ 357079251f5eSSepherosa Ziehau 357179251f5eSSepherosa Ziehau stat_node = SYSCTL_ADD_NODE(ctx, child, OID_AUTO, "mac_stats", 357279251f5eSSepherosa Ziehau CTLFLAG_RD, NULL, "MAC Statistics"); 357379251f5eSSepherosa Ziehau stat_list = SYSCTL_CHILDREN(stat_node); 357479251f5eSSepherosa Ziehau 357579251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "crc_errs", 357679251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->crcerrs, 357779251f5eSSepherosa Ziehau "CRC Errors"); 357879251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "ill_errs", 357979251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->illerrc, 358079251f5eSSepherosa Ziehau "Illegal Byte Errors"); 358179251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "byte_errs", 358279251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->errbc, 358379251f5eSSepherosa Ziehau "Byte Errors"); 358479251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "short_discards", 358579251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->mspdc, 358679251f5eSSepherosa Ziehau "MAC Short Packets Discarded"); 358779251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "local_faults", 358879251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->mlfc, 358979251f5eSSepherosa Ziehau "MAC Local Faults"); 359079251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "remote_faults", 359179251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->mrfc, 359279251f5eSSepherosa Ziehau "MAC Remote Faults"); 359379251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rec_len_errs", 359479251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->rlec, 359579251f5eSSepherosa Ziehau "Receive Length Errors"); 359679251f5eSSepherosa Ziehau 359779251f5eSSepherosa Ziehau /* Flow Control stats */ 359879251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "xon_txd", 359979251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->lxontxc, 360079251f5eSSepherosa Ziehau "Link XON Transmitted"); 360179251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "xon_recvd", 360279251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->lxonrxc, 360379251f5eSSepherosa Ziehau "Link XON Received"); 360479251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "xoff_txd", 360579251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->lxofftxc, 360679251f5eSSepherosa Ziehau "Link XOFF Transmitted"); 360779251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "xoff_recvd", 360879251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->lxoffrxc, 360979251f5eSSepherosa Ziehau "Link XOFF Received"); 361079251f5eSSepherosa Ziehau 361179251f5eSSepherosa Ziehau /* Packet Reception Stats */ 361279251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "total_octets_rcvd", 361379251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->tor, 361479251f5eSSepherosa Ziehau "Total Octets Received"); 361579251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "good_octets_rcvd", 361679251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->gorc, 361779251f5eSSepherosa Ziehau "Good Octets Received"); 361879251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "total_pkts_rcvd", 361979251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->tpr, 362079251f5eSSepherosa Ziehau "Total Packets Received"); 362179251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "good_pkts_rcvd", 362279251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->gprc, 362379251f5eSSepherosa Ziehau "Good Packets Received"); 362479251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "mcast_pkts_rcvd", 362579251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->mprc, 362679251f5eSSepherosa Ziehau "Multicast Packets Received"); 362779251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "bcast_pkts_rcvd", 362879251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->bprc, 362979251f5eSSepherosa Ziehau "Broadcast Packets Received"); 363079251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_frames_64", 363179251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->prc64, 363279251f5eSSepherosa Ziehau "64 byte frames received "); 363379251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_frames_65_127", 363479251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->prc127, 363579251f5eSSepherosa Ziehau "65-127 byte frames received"); 363679251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_frames_128_255", 363779251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->prc255, 363879251f5eSSepherosa Ziehau "128-255 byte frames received"); 363979251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_frames_256_511", 364079251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->prc511, 364179251f5eSSepherosa Ziehau "256-511 byte frames received"); 364279251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_frames_512_1023", 364379251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->prc1023, 364479251f5eSSepherosa Ziehau "512-1023 byte frames received"); 364579251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_frames_1024_1522", 364679251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->prc1522, 364779251f5eSSepherosa Ziehau "1023-1522 byte frames received"); 364879251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "recv_undersized", 364979251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->ruc, 365079251f5eSSepherosa Ziehau "Receive Undersized"); 365179251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "recv_fragmented", 365279251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->rfc, 365379251f5eSSepherosa Ziehau "Fragmented Packets Received "); 365479251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "recv_oversized", 365579251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->roc, 365679251f5eSSepherosa Ziehau "Oversized Packets Received"); 365779251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "recv_jabberd", 365879251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->rjc, 365979251f5eSSepherosa Ziehau "Received Jabber"); 366079251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "management_pkts_rcvd", 366179251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->mngprc, 366279251f5eSSepherosa Ziehau "Management Packets Received"); 366379251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "management_pkts_drpd", 366479251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->mngptc, 366579251f5eSSepherosa Ziehau "Management Packets Dropped"); 366679251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "checksum_errs", 366779251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->xec, 366879251f5eSSepherosa Ziehau "Checksum Errors"); 366979251f5eSSepherosa Ziehau 367079251f5eSSepherosa Ziehau /* Packet Transmission Stats */ 367179251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "good_octets_txd", 367279251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->gotc, 367379251f5eSSepherosa Ziehau "Good Octets Transmitted"); 367479251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "total_pkts_txd", 367579251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->tpt, 367679251f5eSSepherosa Ziehau "Total Packets Transmitted"); 367779251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "good_pkts_txd", 367879251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->gptc, 367979251f5eSSepherosa Ziehau "Good Packets Transmitted"); 368079251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "bcast_pkts_txd", 368179251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->bptc, 368279251f5eSSepherosa Ziehau "Broadcast Packets Transmitted"); 368379251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "mcast_pkts_txd", 368479251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->mptc, 368579251f5eSSepherosa Ziehau "Multicast Packets Transmitted"); 368679251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "management_pkts_txd", 368779251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->mngptc, 368879251f5eSSepherosa Ziehau "Management Packets Transmitted"); 368979251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "tx_frames_64", 369079251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->ptc64, 369179251f5eSSepherosa Ziehau "64 byte frames transmitted "); 369279251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "tx_frames_65_127", 369379251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->ptc127, 369479251f5eSSepherosa Ziehau "65-127 byte frames transmitted"); 369579251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "tx_frames_128_255", 369679251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->ptc255, 369779251f5eSSepherosa Ziehau "128-255 byte frames transmitted"); 369879251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "tx_frames_256_511", 369979251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->ptc511, 370079251f5eSSepherosa Ziehau "256-511 byte frames transmitted"); 370179251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "tx_frames_512_1023", 370279251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->ptc1023, 370379251f5eSSepherosa Ziehau "512-1023 byte frames transmitted"); 370479251f5eSSepherosa Ziehau SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "tx_frames_1024_1522", 370579251f5eSSepherosa Ziehau CTLFLAG_RD, &stats->ptc1522, 370679251f5eSSepherosa Ziehau "1024-1522 byte frames transmitted"); 370779251f5eSSepherosa Ziehau } 370879251f5eSSepherosa Ziehau #endif 370979251f5eSSepherosa Ziehau 371079251f5eSSepherosa Ziehau /* 371179251f5eSSepherosa Ziehau * Enable the hardware to drop packets when the buffer is full. 371279251f5eSSepherosa Ziehau * This is useful when multiple RX rings are used, so that no 371379251f5eSSepherosa Ziehau * single RX ring being full stalls the entire RX engine. We 371479251f5eSSepherosa Ziehau * only enable this when multiple RX rings are used and when 371579251f5eSSepherosa Ziehau * flow control is disabled. 371679251f5eSSepherosa Ziehau */ 371779251f5eSSepherosa Ziehau static void 371879251f5eSSepherosa Ziehau ix_enable_rx_drop(struct ix_softc *sc) 371979251f5eSSepherosa Ziehau { 372079251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 372179251f5eSSepherosa Ziehau int i; 372279251f5eSSepherosa Ziehau 372379251f5eSSepherosa Ziehau if (bootverbose) { 372479251f5eSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, 3725060fa21cSSepherosa Ziehau "flow control %s, enable RX drop\n", 3726060fa21cSSepherosa Ziehau ix_fc2str(sc->hw.fc.current_mode)); 372779251f5eSSepherosa Ziehau } 372879251f5eSSepherosa Ziehau 372979251f5eSSepherosa Ziehau for (i = 0; i < sc->rx_ring_inuse; ++i) { 373079251f5eSSepherosa Ziehau uint32_t srrctl = IXGBE_READ_REG(hw, IXGBE_SRRCTL(i)); 373179251f5eSSepherosa Ziehau 373279251f5eSSepherosa Ziehau srrctl |= IXGBE_SRRCTL_DROP_EN; 373379251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_SRRCTL(i), srrctl); 373479251f5eSSepherosa Ziehau } 373579251f5eSSepherosa Ziehau } 373679251f5eSSepherosa Ziehau 373779251f5eSSepherosa Ziehau static void 373879251f5eSSepherosa Ziehau ix_disable_rx_drop(struct ix_softc *sc) 373979251f5eSSepherosa Ziehau { 374079251f5eSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 374179251f5eSSepherosa Ziehau int i; 374279251f5eSSepherosa Ziehau 374379251f5eSSepherosa Ziehau if (bootverbose) { 374479251f5eSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, 3745060fa21cSSepherosa Ziehau "flow control %s, disable RX drop\n", 3746060fa21cSSepherosa Ziehau ix_fc2str(sc->hw.fc.current_mode)); 374779251f5eSSepherosa Ziehau } 374879251f5eSSepherosa Ziehau 374979251f5eSSepherosa Ziehau for (i = 0; i < sc->rx_ring_inuse; ++i) { 375079251f5eSSepherosa Ziehau uint32_t srrctl = IXGBE_READ_REG(hw, IXGBE_SRRCTL(i)); 375179251f5eSSepherosa Ziehau 375279251f5eSSepherosa Ziehau srrctl &= ~IXGBE_SRRCTL_DROP_EN; 375379251f5eSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_SRRCTL(i), srrctl); 375479251f5eSSepherosa Ziehau } 375579251f5eSSepherosa Ziehau } 375679251f5eSSepherosa Ziehau 375779251f5eSSepherosa Ziehau static void 375879251f5eSSepherosa Ziehau ix_setup_serialize(struct ix_softc *sc) 375979251f5eSSepherosa Ziehau { 376079251f5eSSepherosa Ziehau int i = 0, j; 376179251f5eSSepherosa Ziehau 376279251f5eSSepherosa Ziehau /* Main + RX + TX */ 376379251f5eSSepherosa Ziehau sc->nserialize = 1 + sc->rx_ring_cnt + sc->tx_ring_cnt; 376479251f5eSSepherosa Ziehau sc->serializes = 376579251f5eSSepherosa Ziehau kmalloc(sc->nserialize * sizeof(struct lwkt_serialize *), 376679251f5eSSepherosa Ziehau M_DEVBUF, M_WAITOK | M_ZERO); 376779251f5eSSepherosa Ziehau 376879251f5eSSepherosa Ziehau /* 376979251f5eSSepherosa Ziehau * Setup serializes 377079251f5eSSepherosa Ziehau * 377179251f5eSSepherosa Ziehau * NOTE: Order is critical 377279251f5eSSepherosa Ziehau */ 377379251f5eSSepherosa Ziehau 377479251f5eSSepherosa Ziehau KKASSERT(i < sc->nserialize); 377579251f5eSSepherosa Ziehau sc->serializes[i++] = &sc->main_serialize; 377679251f5eSSepherosa Ziehau 377779251f5eSSepherosa Ziehau for (j = 0; j < sc->rx_ring_cnt; ++j) { 377879251f5eSSepherosa Ziehau KKASSERT(i < sc->nserialize); 377979251f5eSSepherosa Ziehau sc->serializes[i++] = &sc->rx_rings[j].rx_serialize; 378079251f5eSSepherosa Ziehau } 378179251f5eSSepherosa Ziehau 378279251f5eSSepherosa Ziehau for (j = 0; j < sc->tx_ring_cnt; ++j) { 378379251f5eSSepherosa Ziehau KKASSERT(i < sc->nserialize); 378479251f5eSSepherosa Ziehau sc->serializes[i++] = &sc->tx_rings[j].tx_serialize; 378579251f5eSSepherosa Ziehau } 378679251f5eSSepherosa Ziehau 378779251f5eSSepherosa Ziehau KKASSERT(i == sc->nserialize); 378879251f5eSSepherosa Ziehau } 378979251f5eSSepherosa Ziehau 379079251f5eSSepherosa Ziehau static int 379179251f5eSSepherosa Ziehau ix_alloc_intr(struct ix_softc *sc) 379279251f5eSSepherosa Ziehau { 379379251f5eSSepherosa Ziehau struct ix_intr_data *intr; 37943c37d13bSSepherosa Ziehau struct ix_tx_ring *txr; 379579251f5eSSepherosa Ziehau u_int intr_flags; 37963c37d13bSSepherosa Ziehau int i; 3797189a0ff3SSepherosa Ziehau 3798189a0ff3SSepherosa Ziehau ix_alloc_msix(sc); 3799189a0ff3SSepherosa Ziehau if (sc->intr_type == PCI_INTR_TYPE_MSIX) { 3800189a0ff3SSepherosa Ziehau ix_set_ring_inuse(sc, FALSE); 38013c37d13bSSepherosa Ziehau goto done; 3802189a0ff3SSepherosa Ziehau } 380379251f5eSSepherosa Ziehau 38043c37d13bSSepherosa Ziehau /* 38053c37d13bSSepherosa Ziehau * Reset some settings changed by ix_alloc_msix(). 38063c37d13bSSepherosa Ziehau */ 38073c37d13bSSepherosa Ziehau if (sc->rx_rmap_intr != NULL) { 38083c37d13bSSepherosa Ziehau if_ringmap_free(sc->rx_rmap_intr); 38093c37d13bSSepherosa Ziehau sc->rx_rmap_intr = NULL; 38103c37d13bSSepherosa Ziehau } 38113c37d13bSSepherosa Ziehau if (sc->tx_rmap_intr != NULL) { 38123c37d13bSSepherosa Ziehau if_ringmap_free(sc->tx_rmap_intr); 38133c37d13bSSepherosa Ziehau sc->tx_rmap_intr = NULL; 38143c37d13bSSepherosa Ziehau } 38153c37d13bSSepherosa Ziehau if (sc->intr_data != NULL) { 381679251f5eSSepherosa Ziehau kfree(sc->intr_data, M_DEVBUF); 38173c37d13bSSepherosa Ziehau sc->intr_data = NULL; 38183c37d13bSSepherosa Ziehau } 38193c37d13bSSepherosa Ziehau for (i = 0; i < sc->tx_ring_cnt; ++i) { 38203c37d13bSSepherosa Ziehau txr = &sc->tx_rings[i]; 38213c37d13bSSepherosa Ziehau txr->tx_intr_vec = -1; 38223c37d13bSSepherosa Ziehau txr->tx_intr_cpuid = -1; 38233c37d13bSSepherosa Ziehau } 38243c37d13bSSepherosa Ziehau for (i = 0; i < sc->rx_ring_cnt; ++i) { 38253c37d13bSSepherosa Ziehau struct ix_rx_ring *rxr = &sc->rx_rings[i]; 38263c37d13bSSepherosa Ziehau 38273c37d13bSSepherosa Ziehau rxr->rx_intr_vec = -1; 38283c37d13bSSepherosa Ziehau rxr->rx_txr = NULL; 38293c37d13bSSepherosa Ziehau } 383079251f5eSSepherosa Ziehau 383179251f5eSSepherosa Ziehau sc->intr_cnt = 1; 383279251f5eSSepherosa Ziehau sc->intr_data = kmalloc(sizeof(struct ix_intr_data), M_DEVBUF, 383379251f5eSSepherosa Ziehau M_WAITOK | M_ZERO); 383479251f5eSSepherosa Ziehau intr = &sc->intr_data[0]; 383579251f5eSSepherosa Ziehau 383679251f5eSSepherosa Ziehau /* 383779251f5eSSepherosa Ziehau * Allocate MSI/legacy interrupt resource 383879251f5eSSepherosa Ziehau */ 383979251f5eSSepherosa Ziehau sc->intr_type = pci_alloc_1intr(sc->dev, ix_msi_enable, 384079251f5eSSepherosa Ziehau &intr->intr_rid, &intr_flags); 384179251f5eSSepherosa Ziehau 384279251f5eSSepherosa Ziehau intr->intr_res = bus_alloc_resource_any(sc->dev, SYS_RES_IRQ, 384379251f5eSSepherosa Ziehau &intr->intr_rid, intr_flags); 384479251f5eSSepherosa Ziehau if (intr->intr_res == NULL) { 384579251f5eSSepherosa Ziehau device_printf(sc->dev, "Unable to allocate bus resource: " 384679251f5eSSepherosa Ziehau "interrupt\n"); 384779251f5eSSepherosa Ziehau return ENXIO; 384879251f5eSSepherosa Ziehau } 384979251f5eSSepherosa Ziehau 385079251f5eSSepherosa Ziehau intr->intr_serialize = &sc->main_serialize; 385179251f5eSSepherosa Ziehau intr->intr_cpuid = rman_get_cpuid(intr->intr_res); 385279251f5eSSepherosa Ziehau intr->intr_func = ix_intr; 385379251f5eSSepherosa Ziehau intr->intr_funcarg = sc; 385479251f5eSSepherosa Ziehau intr->intr_rate = IX_INTR_RATE; 385579251f5eSSepherosa Ziehau intr->intr_use = IX_INTR_USE_RXTX; 385679251f5eSSepherosa Ziehau 3857189a0ff3SSepherosa Ziehau sc->tx_rings[0].tx_intr_vec = IX_TX_INTR_VEC; 38583c37d13bSSepherosa Ziehau sc->tx_rings[0].tx_intr_cpuid = intr->intr_cpuid; 385979251f5eSSepherosa Ziehau 3860189a0ff3SSepherosa Ziehau sc->rx_rings[0].rx_intr_vec = IX_RX0_INTR_VEC; 386179251f5eSSepherosa Ziehau 386279251f5eSSepherosa Ziehau ix_set_ring_inuse(sc, FALSE); 386379251f5eSSepherosa Ziehau 386479251f5eSSepherosa Ziehau KKASSERT(sc->rx_ring_inuse <= IX_MIN_RXRING_RSS); 38653c37d13bSSepherosa Ziehau if (sc->rx_ring_inuse == IX_MIN_RXRING_RSS) { 386679251f5eSSepherosa Ziehau sc->rx_rings[1].rx_intr_vec = IX_RX1_INTR_VEC; 386779251f5eSSepherosa Ziehau 38683c37d13bSSepherosa Ziehau /* 38693c37d13bSSepherosa Ziehau * Allocate RX ring map for RSS setup. 38703c37d13bSSepherosa Ziehau */ 38713c37d13bSSepherosa Ziehau sc->rx_rmap_intr = if_ringmap_alloc(sc->dev, 38723c37d13bSSepherosa Ziehau IX_MIN_RXRING_RSS, IX_MIN_RXRING_RSS); 38733c37d13bSSepherosa Ziehau KASSERT(if_ringmap_count(sc->rx_rmap_intr) == 38743c37d13bSSepherosa Ziehau sc->rx_ring_inuse, ("RX ring inuse mismatch")); 38753c37d13bSSepherosa Ziehau } 38763c37d13bSSepherosa Ziehau done: 38773c37d13bSSepherosa Ziehau for (i = 0; i < sc->tx_ring_cnt; ++i) { 38783c37d13bSSepherosa Ziehau txr = &sc->tx_rings[i]; 38793c37d13bSSepherosa Ziehau if (txr->tx_intr_cpuid < 0) 38803c37d13bSSepherosa Ziehau txr->tx_intr_cpuid = 0; 38813c37d13bSSepherosa Ziehau } 388279251f5eSSepherosa Ziehau return 0; 388379251f5eSSepherosa Ziehau } 388479251f5eSSepherosa Ziehau 388579251f5eSSepherosa Ziehau static void 388679251f5eSSepherosa Ziehau ix_free_intr(struct ix_softc *sc) 388779251f5eSSepherosa Ziehau { 388879251f5eSSepherosa Ziehau if (sc->intr_data == NULL) 388979251f5eSSepherosa Ziehau return; 389079251f5eSSepherosa Ziehau 389179251f5eSSepherosa Ziehau if (sc->intr_type != PCI_INTR_TYPE_MSIX) { 389279251f5eSSepherosa Ziehau struct ix_intr_data *intr = &sc->intr_data[0]; 389379251f5eSSepherosa Ziehau 389479251f5eSSepherosa Ziehau KKASSERT(sc->intr_cnt == 1); 389579251f5eSSepherosa Ziehau if (intr->intr_res != NULL) { 389679251f5eSSepherosa Ziehau bus_release_resource(sc->dev, SYS_RES_IRQ, 389779251f5eSSepherosa Ziehau intr->intr_rid, intr->intr_res); 389879251f5eSSepherosa Ziehau } 389979251f5eSSepherosa Ziehau if (sc->intr_type == PCI_INTR_TYPE_MSI) 390079251f5eSSepherosa Ziehau pci_release_msi(sc->dev); 3901189a0ff3SSepherosa Ziehau 390279251f5eSSepherosa Ziehau kfree(sc->intr_data, M_DEVBUF); 3903189a0ff3SSepherosa Ziehau } else { 3904189a0ff3SSepherosa Ziehau ix_free_msix(sc, TRUE); 3905189a0ff3SSepherosa Ziehau } 390679251f5eSSepherosa Ziehau } 390779251f5eSSepherosa Ziehau 390879251f5eSSepherosa Ziehau static void 390979251f5eSSepherosa Ziehau ix_set_ring_inuse(struct ix_softc *sc, boolean_t polling) 391079251f5eSSepherosa Ziehau { 391179251f5eSSepherosa Ziehau sc->rx_ring_inuse = ix_get_rxring_inuse(sc, polling); 391279251f5eSSepherosa Ziehau sc->tx_ring_inuse = ix_get_txring_inuse(sc, polling); 391379251f5eSSepherosa Ziehau if (bootverbose) { 391479251f5eSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, 391579251f5eSSepherosa Ziehau "RX rings %d/%d, TX rings %d/%d\n", 391679251f5eSSepherosa Ziehau sc->rx_ring_inuse, sc->rx_ring_cnt, 391779251f5eSSepherosa Ziehau sc->tx_ring_inuse, sc->tx_ring_cnt); 391879251f5eSSepherosa Ziehau } 391979251f5eSSepherosa Ziehau } 392079251f5eSSepherosa Ziehau 392179251f5eSSepherosa Ziehau static int 392279251f5eSSepherosa Ziehau ix_get_rxring_inuse(const struct ix_softc *sc, boolean_t polling) 392379251f5eSSepherosa Ziehau { 392479251f5eSSepherosa Ziehau if (!IX_ENABLE_HWRSS(sc)) 392579251f5eSSepherosa Ziehau return 1; 392679251f5eSSepherosa Ziehau 392779251f5eSSepherosa Ziehau if (polling) 392879251f5eSSepherosa Ziehau return sc->rx_ring_cnt; 392979251f5eSSepherosa Ziehau else if (sc->intr_type != PCI_INTR_TYPE_MSIX) 393079251f5eSSepherosa Ziehau return IX_MIN_RXRING_RSS; 393179251f5eSSepherosa Ziehau else 3932189a0ff3SSepherosa Ziehau return sc->rx_ring_msix; 393379251f5eSSepherosa Ziehau } 393479251f5eSSepherosa Ziehau 393579251f5eSSepherosa Ziehau static int 393679251f5eSSepherosa Ziehau ix_get_txring_inuse(const struct ix_softc *sc, boolean_t polling) 393779251f5eSSepherosa Ziehau { 393879251f5eSSepherosa Ziehau if (!IX_ENABLE_HWTSS(sc)) 393979251f5eSSepherosa Ziehau return 1; 394079251f5eSSepherosa Ziehau 394179251f5eSSepherosa Ziehau if (polling) 394279251f5eSSepherosa Ziehau return sc->tx_ring_cnt; 394379251f5eSSepherosa Ziehau else if (sc->intr_type != PCI_INTR_TYPE_MSIX) 394479251f5eSSepherosa Ziehau return 1; 394579251f5eSSepherosa Ziehau else 3946189a0ff3SSepherosa Ziehau return sc->tx_ring_msix; 394779251f5eSSepherosa Ziehau } 394879251f5eSSepherosa Ziehau 394979251f5eSSepherosa Ziehau static int 395079251f5eSSepherosa Ziehau ix_setup_intr(struct ix_softc *sc) 395179251f5eSSepherosa Ziehau { 395279251f5eSSepherosa Ziehau int i; 395379251f5eSSepherosa Ziehau 395479251f5eSSepherosa Ziehau for (i = 0; i < sc->intr_cnt; ++i) { 395579251f5eSSepherosa Ziehau struct ix_intr_data *intr = &sc->intr_data[i]; 395679251f5eSSepherosa Ziehau int error; 395779251f5eSSepherosa Ziehau 395879251f5eSSepherosa Ziehau error = bus_setup_intr_descr(sc->dev, intr->intr_res, 395979251f5eSSepherosa Ziehau INTR_MPSAFE, intr->intr_func, intr->intr_funcarg, 396079251f5eSSepherosa Ziehau &intr->intr_hand, intr->intr_serialize, intr->intr_desc); 396179251f5eSSepherosa Ziehau if (error) { 396279251f5eSSepherosa Ziehau device_printf(sc->dev, "can't setup %dth intr\n", i); 396379251f5eSSepherosa Ziehau ix_teardown_intr(sc, i); 396479251f5eSSepherosa Ziehau return error; 396579251f5eSSepherosa Ziehau } 396679251f5eSSepherosa Ziehau } 396779251f5eSSepherosa Ziehau return 0; 396879251f5eSSepherosa Ziehau } 396979251f5eSSepherosa Ziehau 397079251f5eSSepherosa Ziehau static void 397179251f5eSSepherosa Ziehau ix_teardown_intr(struct ix_softc *sc, int intr_cnt) 397279251f5eSSepherosa Ziehau { 397379251f5eSSepherosa Ziehau int i; 397479251f5eSSepherosa Ziehau 397579251f5eSSepherosa Ziehau if (sc->intr_data == NULL) 397679251f5eSSepherosa Ziehau return; 397779251f5eSSepherosa Ziehau 397879251f5eSSepherosa Ziehau for (i = 0; i < intr_cnt; ++i) { 397979251f5eSSepherosa Ziehau struct ix_intr_data *intr = &sc->intr_data[i]; 398079251f5eSSepherosa Ziehau 398179251f5eSSepherosa Ziehau bus_teardown_intr(sc->dev, intr->intr_res, intr->intr_hand); 398279251f5eSSepherosa Ziehau } 398379251f5eSSepherosa Ziehau } 398479251f5eSSepherosa Ziehau 398579251f5eSSepherosa Ziehau static void 398679251f5eSSepherosa Ziehau ix_serialize(struct ifnet *ifp, enum ifnet_serialize slz) 398779251f5eSSepherosa Ziehau { 398879251f5eSSepherosa Ziehau struct ix_softc *sc = ifp->if_softc; 398979251f5eSSepherosa Ziehau 399079251f5eSSepherosa Ziehau ifnet_serialize_array_enter(sc->serializes, sc->nserialize, slz); 399179251f5eSSepherosa Ziehau } 399279251f5eSSepherosa Ziehau 399379251f5eSSepherosa Ziehau static void 399479251f5eSSepherosa Ziehau ix_deserialize(struct ifnet *ifp, enum ifnet_serialize slz) 399579251f5eSSepherosa Ziehau { 399679251f5eSSepherosa Ziehau struct ix_softc *sc = ifp->if_softc; 399779251f5eSSepherosa Ziehau 399879251f5eSSepherosa Ziehau ifnet_serialize_array_exit(sc->serializes, sc->nserialize, slz); 399979251f5eSSepherosa Ziehau } 400079251f5eSSepherosa Ziehau 400179251f5eSSepherosa Ziehau static int 400279251f5eSSepherosa Ziehau ix_tryserialize(struct ifnet *ifp, enum ifnet_serialize slz) 400379251f5eSSepherosa Ziehau { 400479251f5eSSepherosa Ziehau struct ix_softc *sc = ifp->if_softc; 400579251f5eSSepherosa Ziehau 400679251f5eSSepherosa Ziehau return ifnet_serialize_array_try(sc->serializes, sc->nserialize, slz); 400779251f5eSSepherosa Ziehau } 400879251f5eSSepherosa Ziehau 400979251f5eSSepherosa Ziehau #ifdef INVARIANTS 401079251f5eSSepherosa Ziehau 401179251f5eSSepherosa Ziehau static void 401279251f5eSSepherosa Ziehau ix_serialize_assert(struct ifnet *ifp, enum ifnet_serialize slz, 401379251f5eSSepherosa Ziehau boolean_t serialized) 401479251f5eSSepherosa Ziehau { 401579251f5eSSepherosa Ziehau struct ix_softc *sc = ifp->if_softc; 401679251f5eSSepherosa Ziehau 401779251f5eSSepherosa Ziehau ifnet_serialize_array_assert(sc->serializes, sc->nserialize, slz, 401879251f5eSSepherosa Ziehau serialized); 401979251f5eSSepherosa Ziehau } 402079251f5eSSepherosa Ziehau 402179251f5eSSepherosa Ziehau #endif /* INVARIANTS */ 402279251f5eSSepherosa Ziehau 402379251f5eSSepherosa Ziehau static void 402479251f5eSSepherosa Ziehau ix_free_rings(struct ix_softc *sc) 402579251f5eSSepherosa Ziehau { 402679251f5eSSepherosa Ziehau int i; 402779251f5eSSepherosa Ziehau 402879251f5eSSepherosa Ziehau if (sc->tx_rings != NULL) { 402979251f5eSSepherosa Ziehau for (i = 0; i < sc->tx_ring_cnt; ++i) { 403079251f5eSSepherosa Ziehau struct ix_tx_ring *txr = &sc->tx_rings[i]; 403179251f5eSSepherosa Ziehau 403279251f5eSSepherosa Ziehau ix_destroy_tx_ring(txr, txr->tx_ndesc); 403379251f5eSSepherosa Ziehau } 403479251f5eSSepherosa Ziehau kfree(sc->tx_rings, M_DEVBUF); 403579251f5eSSepherosa Ziehau } 403679251f5eSSepherosa Ziehau 403779251f5eSSepherosa Ziehau if (sc->rx_rings != NULL) { 403879251f5eSSepherosa Ziehau for (i =0; i < sc->rx_ring_cnt; ++i) { 403979251f5eSSepherosa Ziehau struct ix_rx_ring *rxr = &sc->rx_rings[i]; 404079251f5eSSepherosa Ziehau 404179251f5eSSepherosa Ziehau ix_destroy_rx_ring(rxr, rxr->rx_ndesc); 404279251f5eSSepherosa Ziehau } 404379251f5eSSepherosa Ziehau kfree(sc->rx_rings, M_DEVBUF); 404479251f5eSSepherosa Ziehau } 404579251f5eSSepherosa Ziehau 404679251f5eSSepherosa Ziehau if (sc->parent_tag != NULL) 404779251f5eSSepherosa Ziehau bus_dma_tag_destroy(sc->parent_tag); 404879251f5eSSepherosa Ziehau } 404979251f5eSSepherosa Ziehau 405079251f5eSSepherosa Ziehau static void 40518d0afa86SSepherosa Ziehau ix_watchdog_reset(struct ix_softc *sc) 40528d0afa86SSepherosa Ziehau { 40538d0afa86SSepherosa Ziehau int i; 40548d0afa86SSepherosa Ziehau 40558d0afa86SSepherosa Ziehau ASSERT_IFNET_SERIALIZED_ALL(&sc->arpcom.ac_if); 40568d0afa86SSepherosa Ziehau ix_init(sc); 40578d0afa86SSepherosa Ziehau for (i = 0; i < sc->tx_ring_inuse; ++i) 40588d0afa86SSepherosa Ziehau ifsq_devstart_sched(sc->tx_rings[i].tx_ifsq); 40598d0afa86SSepherosa Ziehau } 40608d0afa86SSepherosa Ziehau 40618d0afa86SSepherosa Ziehau static void 40628d0afa86SSepherosa Ziehau ix_sync_netisr(struct ix_softc *sc, int flags) 40638d0afa86SSepherosa Ziehau { 40648d0afa86SSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 40658d0afa86SSepherosa Ziehau 40668d0afa86SSepherosa Ziehau ifnet_serialize_all(ifp); 40678d0afa86SSepherosa Ziehau if (ifp->if_flags & IFF_RUNNING) { 40688d0afa86SSepherosa Ziehau ifp->if_flags &= ~(IFF_RUNNING | flags); 40698d0afa86SSepherosa Ziehau } else { 40708d0afa86SSepherosa Ziehau ifnet_deserialize_all(ifp); 40718d0afa86SSepherosa Ziehau return; 40728d0afa86SSepherosa Ziehau } 40738d0afa86SSepherosa Ziehau ifnet_deserialize_all(ifp); 40748d0afa86SSepherosa Ziehau 40758d0afa86SSepherosa Ziehau /* Make sure that polling stopped. */ 40768d0afa86SSepherosa Ziehau netmsg_service_sync(); 40778d0afa86SSepherosa Ziehau } 40788d0afa86SSepherosa Ziehau 40798d0afa86SSepherosa Ziehau static void 40808d0afa86SSepherosa Ziehau ix_watchdog_task(void *xsc, int pending __unused) 40818d0afa86SSepherosa Ziehau { 40828d0afa86SSepherosa Ziehau struct ix_softc *sc = xsc; 40838d0afa86SSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 40848d0afa86SSepherosa Ziehau 40858d0afa86SSepherosa Ziehau ix_sync_netisr(sc, 0); 40868d0afa86SSepherosa Ziehau 40878d0afa86SSepherosa Ziehau ifnet_serialize_all(ifp); 40888d0afa86SSepherosa Ziehau if ((ifp->if_flags & (IFF_UP | IFF_RUNNING)) == IFF_UP) 40898d0afa86SSepherosa Ziehau ix_watchdog_reset(sc); 40908d0afa86SSepherosa Ziehau ifnet_deserialize_all(ifp); 40918d0afa86SSepherosa Ziehau } 40928d0afa86SSepherosa Ziehau 40938d0afa86SSepherosa Ziehau static void 409479251f5eSSepherosa Ziehau ix_watchdog(struct ifaltq_subque *ifsq) 409579251f5eSSepherosa Ziehau { 409679251f5eSSepherosa Ziehau struct ix_tx_ring *txr = ifsq_get_priv(ifsq); 409779251f5eSSepherosa Ziehau struct ifnet *ifp = ifsq_get_ifp(ifsq); 409879251f5eSSepherosa Ziehau struct ix_softc *sc = ifp->if_softc; 409979251f5eSSepherosa Ziehau 410079251f5eSSepherosa Ziehau KKASSERT(txr->tx_ifsq == ifsq); 410179251f5eSSepherosa Ziehau ASSERT_IFNET_SERIALIZED_ALL(ifp); 410279251f5eSSepherosa Ziehau 410379251f5eSSepherosa Ziehau /* 410479251f5eSSepherosa Ziehau * If the interface has been paused then don't do the watchdog check 410579251f5eSSepherosa Ziehau */ 410679251f5eSSepherosa Ziehau if (IXGBE_READ_REG(&sc->hw, IXGBE_TFCS) & IXGBE_TFCS_TXOFF) { 410779251f5eSSepherosa Ziehau txr->tx_watchdog.wd_timer = 5; 410879251f5eSSepherosa Ziehau return; 410979251f5eSSepherosa Ziehau } 411079251f5eSSepherosa Ziehau 411179251f5eSSepherosa Ziehau if_printf(ifp, "Watchdog timeout -- resetting\n"); 411279251f5eSSepherosa Ziehau if_printf(ifp, "Queue(%d) tdh = %d, hw tdt = %d\n", txr->tx_idx, 411379251f5eSSepherosa Ziehau IXGBE_READ_REG(&sc->hw, IXGBE_TDH(txr->tx_idx)), 411479251f5eSSepherosa Ziehau IXGBE_READ_REG(&sc->hw, IXGBE_TDT(txr->tx_idx))); 411579251f5eSSepherosa Ziehau if_printf(ifp, "TX(%d) desc avail = %d, next TX to Clean = %d\n", 411679251f5eSSepherosa Ziehau txr->tx_idx, txr->tx_avail, txr->tx_next_clean); 411779251f5eSSepherosa Ziehau 41188d0afa86SSepherosa Ziehau if ((ifp->if_flags & (IFF_IDIRECT | IFF_NPOLLING | IFF_RUNNING)) == 41198d0afa86SSepherosa Ziehau (IFF_IDIRECT | IFF_NPOLLING | IFF_RUNNING)) 41208d0afa86SSepherosa Ziehau taskqueue_enqueue(taskqueue_thread[0], &sc->wdog_task); 41218d0afa86SSepherosa Ziehau else 41228d0afa86SSepherosa Ziehau ix_watchdog_reset(sc); 412379251f5eSSepherosa Ziehau } 412479251f5eSSepherosa Ziehau 412579251f5eSSepherosa Ziehau static void 412679251f5eSSepherosa Ziehau ix_free_tx_ring(struct ix_tx_ring *txr) 412779251f5eSSepherosa Ziehau { 412879251f5eSSepherosa Ziehau int i; 412979251f5eSSepherosa Ziehau 413079251f5eSSepherosa Ziehau for (i = 0; i < txr->tx_ndesc; ++i) { 413179251f5eSSepherosa Ziehau struct ix_tx_buf *txbuf = &txr->tx_buf[i]; 413279251f5eSSepherosa Ziehau 413382db96e9SSepherosa Ziehau if (txbuf->m_head != NULL) 413482db96e9SSepherosa Ziehau ix_free_txbuf(txr, txbuf); 413579251f5eSSepherosa Ziehau } 413679251f5eSSepherosa Ziehau } 413779251f5eSSepherosa Ziehau 413879251f5eSSepherosa Ziehau static void 413979251f5eSSepherosa Ziehau ix_free_rx_ring(struct ix_rx_ring *rxr) 414079251f5eSSepherosa Ziehau { 414179251f5eSSepherosa Ziehau int i; 414279251f5eSSepherosa Ziehau 414379251f5eSSepherosa Ziehau for (i = 0; i < rxr->rx_ndesc; ++i) { 414479251f5eSSepherosa Ziehau struct ix_rx_buf *rxbuf = &rxr->rx_buf[i]; 414579251f5eSSepherosa Ziehau 414679251f5eSSepherosa Ziehau if (rxbuf->fmp != NULL) { 414779251f5eSSepherosa Ziehau m_freem(rxbuf->fmp); 414879251f5eSSepherosa Ziehau rxbuf->fmp = NULL; 414979251f5eSSepherosa Ziehau rxbuf->lmp = NULL; 415079251f5eSSepherosa Ziehau } else { 415179251f5eSSepherosa Ziehau KKASSERT(rxbuf->lmp == NULL); 415279251f5eSSepherosa Ziehau } 415379251f5eSSepherosa Ziehau if (rxbuf->m_head != NULL) { 415479251f5eSSepherosa Ziehau bus_dmamap_unload(rxr->rx_tag, rxbuf->map); 415579251f5eSSepherosa Ziehau m_freem(rxbuf->m_head); 415679251f5eSSepherosa Ziehau rxbuf->m_head = NULL; 415779251f5eSSepherosa Ziehau } 415879251f5eSSepherosa Ziehau } 415979251f5eSSepherosa Ziehau } 416079251f5eSSepherosa Ziehau 416179251f5eSSepherosa Ziehau static int 416279251f5eSSepherosa Ziehau ix_newbuf(struct ix_rx_ring *rxr, int i, boolean_t wait) 416379251f5eSSepherosa Ziehau { 416479251f5eSSepherosa Ziehau struct mbuf *m; 416579251f5eSSepherosa Ziehau bus_dma_segment_t seg; 416679251f5eSSepherosa Ziehau bus_dmamap_t map; 416779251f5eSSepherosa Ziehau struct ix_rx_buf *rxbuf; 416879251f5eSSepherosa Ziehau int flags, error, nseg; 416979251f5eSSepherosa Ziehau 4170b5523eacSSascha Wildner flags = M_NOWAIT; 417179251f5eSSepherosa Ziehau if (__predict_false(wait)) 4172b5523eacSSascha Wildner flags = M_WAITOK; 417379251f5eSSepherosa Ziehau 417479251f5eSSepherosa Ziehau m = m_getjcl(flags, MT_DATA, M_PKTHDR, rxr->rx_mbuf_sz); 417579251f5eSSepherosa Ziehau if (m == NULL) { 417679251f5eSSepherosa Ziehau if (wait) { 417779251f5eSSepherosa Ziehau if_printf(&rxr->rx_sc->arpcom.ac_if, 417879251f5eSSepherosa Ziehau "Unable to allocate RX mbuf\n"); 417979251f5eSSepherosa Ziehau } 418079251f5eSSepherosa Ziehau return ENOBUFS; 418179251f5eSSepherosa Ziehau } 418279251f5eSSepherosa Ziehau m->m_len = m->m_pkthdr.len = rxr->rx_mbuf_sz; 418379251f5eSSepherosa Ziehau 418479251f5eSSepherosa Ziehau error = bus_dmamap_load_mbuf_segment(rxr->rx_tag, 418579251f5eSSepherosa Ziehau rxr->rx_sparemap, m, &seg, 1, &nseg, BUS_DMA_NOWAIT); 418679251f5eSSepherosa Ziehau if (error) { 418779251f5eSSepherosa Ziehau m_freem(m); 418879251f5eSSepherosa Ziehau if (wait) { 418979251f5eSSepherosa Ziehau if_printf(&rxr->rx_sc->arpcom.ac_if, 419079251f5eSSepherosa Ziehau "Unable to load RX mbuf\n"); 419179251f5eSSepherosa Ziehau } 419279251f5eSSepherosa Ziehau return error; 419379251f5eSSepherosa Ziehau } 419479251f5eSSepherosa Ziehau 419579251f5eSSepherosa Ziehau rxbuf = &rxr->rx_buf[i]; 419679251f5eSSepherosa Ziehau if (rxbuf->m_head != NULL) 419779251f5eSSepherosa Ziehau bus_dmamap_unload(rxr->rx_tag, rxbuf->map); 419879251f5eSSepherosa Ziehau 419979251f5eSSepherosa Ziehau map = rxbuf->map; 420079251f5eSSepherosa Ziehau rxbuf->map = rxr->rx_sparemap; 420179251f5eSSepherosa Ziehau rxr->rx_sparemap = map; 420279251f5eSSepherosa Ziehau 420379251f5eSSepherosa Ziehau rxbuf->m_head = m; 420479251f5eSSepherosa Ziehau rxbuf->paddr = seg.ds_addr; 420579251f5eSSepherosa Ziehau 420679251f5eSSepherosa Ziehau ix_setup_rxdesc(&rxr->rx_base[i], rxbuf); 420779251f5eSSepherosa Ziehau return 0; 420879251f5eSSepherosa Ziehau } 420979251f5eSSepherosa Ziehau 421079251f5eSSepherosa Ziehau static void 421179251f5eSSepherosa Ziehau ix_add_sysctl(struct ix_softc *sc) 421279251f5eSSepherosa Ziehau { 421326595b18SSascha Wildner struct sysctl_ctx_list *ctx = device_get_sysctl_ctx(sc->dev); 421426595b18SSascha Wildner struct sysctl_oid *tree = device_get_sysctl_tree(sc->dev); 421579251f5eSSepherosa Ziehau char node[32]; 4216020afcaaSSascha Wildner int i; 421779251f5eSSepherosa Ziehau 421826595b18SSascha Wildner SYSCTL_ADD_INT(ctx, SYSCTL_CHILDREN(tree), 421979251f5eSSepherosa Ziehau OID_AUTO, "rxr", CTLFLAG_RD, &sc->rx_ring_cnt, 0, "# of RX rings"); 422026595b18SSascha Wildner SYSCTL_ADD_INT(ctx, SYSCTL_CHILDREN(tree), 422179251f5eSSepherosa Ziehau OID_AUTO, "rxr_inuse", CTLFLAG_RD, &sc->rx_ring_inuse, 0, 422279251f5eSSepherosa Ziehau "# of RX rings used"); 422326595b18SSascha Wildner SYSCTL_ADD_INT(ctx, SYSCTL_CHILDREN(tree), 422479251f5eSSepherosa Ziehau OID_AUTO, "txr", CTLFLAG_RD, &sc->tx_ring_cnt, 0, "# of TX rings"); 422526595b18SSascha Wildner SYSCTL_ADD_INT(ctx, SYSCTL_CHILDREN(tree), 422679251f5eSSepherosa Ziehau OID_AUTO, "txr_inuse", CTLFLAG_RD, &sc->tx_ring_inuse, 0, 422779251f5eSSepherosa Ziehau "# of TX rings used"); 422826595b18SSascha Wildner SYSCTL_ADD_PROC(ctx, SYSCTL_CHILDREN(tree), 422979251f5eSSepherosa Ziehau OID_AUTO, "rxd", CTLTYPE_INT | CTLFLAG_RD, 423079251f5eSSepherosa Ziehau sc, 0, ix_sysctl_rxd, "I", 423179251f5eSSepherosa Ziehau "# of RX descs"); 423226595b18SSascha Wildner SYSCTL_ADD_PROC(ctx, SYSCTL_CHILDREN(tree), 423379251f5eSSepherosa Ziehau OID_AUTO, "txd", CTLTYPE_INT | CTLFLAG_RD, 423479251f5eSSepherosa Ziehau sc, 0, ix_sysctl_txd, "I", 423579251f5eSSepherosa Ziehau "# of TX descs"); 423626595b18SSascha Wildner SYSCTL_ADD_PROC(ctx, SYSCTL_CHILDREN(tree), 423779251f5eSSepherosa Ziehau OID_AUTO, "tx_wreg_nsegs", CTLTYPE_INT | CTLFLAG_RW, 423879251f5eSSepherosa Ziehau sc, 0, ix_sysctl_tx_wreg_nsegs, "I", 423979251f5eSSepherosa Ziehau "# of segments sent before write to hardware register"); 424026595b18SSascha Wildner SYSCTL_ADD_PROC(ctx, SYSCTL_CHILDREN(tree), 424179251f5eSSepherosa Ziehau OID_AUTO, "rx_wreg_nsegs", CTLTYPE_INT | CTLFLAG_RW, 424279251f5eSSepherosa Ziehau sc, 0, ix_sysctl_rx_wreg_nsegs, "I", 424379251f5eSSepherosa Ziehau "# of received segments sent before write to hardware register"); 424426595b18SSascha Wildner SYSCTL_ADD_PROC(ctx, SYSCTL_CHILDREN(tree), 424579251f5eSSepherosa Ziehau OID_AUTO, "tx_intr_nsegs", CTLTYPE_INT | CTLFLAG_RW, 424679251f5eSSepherosa Ziehau sc, 0, ix_sysctl_tx_intr_nsegs, "I", 424779251f5eSSepherosa Ziehau "# of segments per TX interrupt"); 42488d0afa86SSepherosa Ziehau SYSCTL_ADD_INT(ctx, SYSCTL_CHILDREN(tree), 42498d0afa86SSepherosa Ziehau OID_AUTO, "direct_input", CTLFLAG_RW, &sc->direct_input, 0, 42508d0afa86SSepherosa Ziehau "Enable direct input"); 42513c37d13bSSepherosa Ziehau if (sc->intr_type == PCI_INTR_TYPE_MSIX) { 42523c37d13bSSepherosa Ziehau SYSCTL_ADD_PROC(ctx, SYSCTL_CHILDREN(tree), 42533c37d13bSSepherosa Ziehau OID_AUTO, "tx_msix_cpumap", CTLTYPE_OPAQUE | CTLFLAG_RD, 42543c37d13bSSepherosa Ziehau sc->tx_rmap_intr, 0, if_ringmap_cpumap_sysctl, "I", 42553c37d13bSSepherosa Ziehau "TX MSI-X CPU map"); 42563c37d13bSSepherosa Ziehau SYSCTL_ADD_PROC(ctx, SYSCTL_CHILDREN(tree), 42573c37d13bSSepherosa Ziehau OID_AUTO, "rx_msix_cpumap", CTLTYPE_OPAQUE | CTLFLAG_RD, 42583c37d13bSSepherosa Ziehau sc->rx_rmap_intr, 0, if_ringmap_cpumap_sysctl, "I", 42593c37d13bSSepherosa Ziehau "RX MSI-X CPU map"); 42603c37d13bSSepherosa Ziehau } 42614a648aefSSepherosa Ziehau #ifdef IFPOLL_ENABLE 426226595b18SSascha Wildner SYSCTL_ADD_PROC(ctx, SYSCTL_CHILDREN(tree), 42633c37d13bSSepherosa Ziehau OID_AUTO, "tx_poll_cpumap", CTLTYPE_OPAQUE | CTLFLAG_RD, 42643c37d13bSSepherosa Ziehau sc->tx_rmap, 0, if_ringmap_cpumap_sysctl, "I", 42653c37d13bSSepherosa Ziehau "TX polling CPU map"); 426626595b18SSascha Wildner SYSCTL_ADD_PROC(ctx, SYSCTL_CHILDREN(tree), 42673c37d13bSSepherosa Ziehau OID_AUTO, "rx_poll_cpumap", CTLTYPE_OPAQUE | CTLFLAG_RD, 42683c37d13bSSepherosa Ziehau sc->rx_rmap, 0, if_ringmap_cpumap_sysctl, "I", 42693c37d13bSSepherosa Ziehau "RX polling CPU map"); 42704a648aefSSepherosa Ziehau #endif 42714a648aefSSepherosa Ziehau 4272189a0ff3SSepherosa Ziehau #define IX_ADD_INTR_RATE_SYSCTL(sc, use, name) \ 4273189a0ff3SSepherosa Ziehau do { \ 4274189a0ff3SSepherosa Ziehau ix_add_intr_rate_sysctl(sc, IX_INTR_USE_##use, #name, \ 4275189a0ff3SSepherosa Ziehau ix_sysctl_##name, #use " interrupt rate"); \ 4276189a0ff3SSepherosa Ziehau } while (0) 4277189a0ff3SSepherosa Ziehau 4278189a0ff3SSepherosa Ziehau IX_ADD_INTR_RATE_SYSCTL(sc, RXTX, rxtx_intr_rate); 4279189a0ff3SSepherosa Ziehau IX_ADD_INTR_RATE_SYSCTL(sc, RX, rx_intr_rate); 4280189a0ff3SSepherosa Ziehau IX_ADD_INTR_RATE_SYSCTL(sc, TX, tx_intr_rate); 4281189a0ff3SSepherosa Ziehau IX_ADD_INTR_RATE_SYSCTL(sc, STATUS, sts_intr_rate); 4282189a0ff3SSepherosa Ziehau 4283189a0ff3SSepherosa Ziehau #undef IX_ADD_INTR_RATE_SYSCTL 428479251f5eSSepherosa Ziehau 428579251f5eSSepherosa Ziehau #ifdef IX_RSS_DEBUG 428626595b18SSascha Wildner SYSCTL_ADD_INT(ctx, SYSCTL_CHILDREN(tree), 428779251f5eSSepherosa Ziehau OID_AUTO, "rss_debug", CTLFLAG_RW, &sc->rss_debug, 0, 428879251f5eSSepherosa Ziehau "RSS debug level"); 428979251f5eSSepherosa Ziehau for (i = 0; i < sc->rx_ring_cnt; ++i) { 429079251f5eSSepherosa Ziehau ksnprintf(node, sizeof(node), "rx%d_pkt", i); 429126595b18SSascha Wildner SYSCTL_ADD_ULONG(ctx, 429226595b18SSascha Wildner SYSCTL_CHILDREN(tree), OID_AUTO, node, 429379251f5eSSepherosa Ziehau CTLFLAG_RW, &sc->rx_rings[i].rx_pkts, "RXed packets"); 429479251f5eSSepherosa Ziehau } 429579251f5eSSepherosa Ziehau #endif 429682db96e9SSepherosa Ziehau for (i = 0; i < sc->tx_ring_cnt; ++i) { 429782db96e9SSepherosa Ziehau struct ix_tx_ring *txr = &sc->tx_rings[i]; 429882db96e9SSepherosa Ziehau 429982db96e9SSepherosa Ziehau ksnprintf(node, sizeof(node), "tx%d_nmbuf", i); 430082db96e9SSepherosa Ziehau SYSCTL_ADD_PROC(ctx, SYSCTL_CHILDREN(tree), OID_AUTO, node, 430182db96e9SSepherosa Ziehau CTLTYPE_INT | CTLFLAG_RD, txr, 0, ix_sysctl_tx_nmbuf, "I", 430282db96e9SSepherosa Ziehau "# of pending TX mbufs"); 430382db96e9SSepherosa Ziehau 430482db96e9SSepherosa Ziehau ksnprintf(node, sizeof(node), "tx%d_gc", i); 430582db96e9SSepherosa Ziehau SYSCTL_ADD_ULONG(ctx, SYSCTL_CHILDREN(tree), OID_AUTO, node, 430682db96e9SSepherosa Ziehau CTLFLAG_RW, &txr->tx_gc, "# of TX desc GC"); 430782db96e9SSepherosa Ziehau } 430879251f5eSSepherosa Ziehau 430979251f5eSSepherosa Ziehau #if 0 431079251f5eSSepherosa Ziehau ix_add_hw_stats(sc); 431179251f5eSSepherosa Ziehau #endif 431279251f5eSSepherosa Ziehau 431379251f5eSSepherosa Ziehau } 431479251f5eSSepherosa Ziehau 431579251f5eSSepherosa Ziehau static int 431682db96e9SSepherosa Ziehau ix_sysctl_tx_nmbuf(SYSCTL_HANDLER_ARGS) 431782db96e9SSepherosa Ziehau { 431882db96e9SSepherosa Ziehau struct ix_tx_ring *txr = (void *)arg1; 431982db96e9SSepherosa Ziehau int nmbuf; 432082db96e9SSepherosa Ziehau 432182db96e9SSepherosa Ziehau nmbuf = txr->tx_nmbuf; 432282db96e9SSepherosa Ziehau return (sysctl_handle_int(oidp, &nmbuf, 0, req)); 432382db96e9SSepherosa Ziehau } 432482db96e9SSepherosa Ziehau 432582db96e9SSepherosa Ziehau static int 432679251f5eSSepherosa Ziehau ix_sysctl_tx_wreg_nsegs(SYSCTL_HANDLER_ARGS) 432779251f5eSSepherosa Ziehau { 432879251f5eSSepherosa Ziehau struct ix_softc *sc = (void *)arg1; 432979251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 433079251f5eSSepherosa Ziehau int error, nsegs, i; 433179251f5eSSepherosa Ziehau 433279251f5eSSepherosa Ziehau nsegs = sc->tx_rings[0].tx_wreg_nsegs; 433379251f5eSSepherosa Ziehau error = sysctl_handle_int(oidp, &nsegs, 0, req); 433479251f5eSSepherosa Ziehau if (error || req->newptr == NULL) 433579251f5eSSepherosa Ziehau return error; 433679251f5eSSepherosa Ziehau if (nsegs < 0) 433779251f5eSSepherosa Ziehau return EINVAL; 433879251f5eSSepherosa Ziehau 433979251f5eSSepherosa Ziehau ifnet_serialize_all(ifp); 434079251f5eSSepherosa Ziehau for (i = 0; i < sc->tx_ring_cnt; ++i) 434179251f5eSSepherosa Ziehau sc->tx_rings[i].tx_wreg_nsegs = nsegs; 434279251f5eSSepherosa Ziehau ifnet_deserialize_all(ifp); 434379251f5eSSepherosa Ziehau 434479251f5eSSepherosa Ziehau return 0; 434579251f5eSSepherosa Ziehau } 434679251f5eSSepherosa Ziehau 434779251f5eSSepherosa Ziehau static int 434879251f5eSSepherosa Ziehau ix_sysctl_rx_wreg_nsegs(SYSCTL_HANDLER_ARGS) 434979251f5eSSepherosa Ziehau { 435079251f5eSSepherosa Ziehau struct ix_softc *sc = (void *)arg1; 435179251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 435279251f5eSSepherosa Ziehau int error, nsegs, i; 435379251f5eSSepherosa Ziehau 435479251f5eSSepherosa Ziehau nsegs = sc->rx_rings[0].rx_wreg_nsegs; 435579251f5eSSepherosa Ziehau error = sysctl_handle_int(oidp, &nsegs, 0, req); 435679251f5eSSepherosa Ziehau if (error || req->newptr == NULL) 435779251f5eSSepherosa Ziehau return error; 435879251f5eSSepherosa Ziehau if (nsegs < 0) 435979251f5eSSepherosa Ziehau return EINVAL; 436079251f5eSSepherosa Ziehau 436179251f5eSSepherosa Ziehau ifnet_serialize_all(ifp); 436279251f5eSSepherosa Ziehau for (i = 0; i < sc->rx_ring_cnt; ++i) 436379251f5eSSepherosa Ziehau sc->rx_rings[i].rx_wreg_nsegs =nsegs; 436479251f5eSSepherosa Ziehau ifnet_deserialize_all(ifp); 436579251f5eSSepherosa Ziehau 436679251f5eSSepherosa Ziehau return 0; 436779251f5eSSepherosa Ziehau } 436879251f5eSSepherosa Ziehau 436979251f5eSSepherosa Ziehau static int 437079251f5eSSepherosa Ziehau ix_sysctl_txd(SYSCTL_HANDLER_ARGS) 437179251f5eSSepherosa Ziehau { 437279251f5eSSepherosa Ziehau struct ix_softc *sc = (void *)arg1; 437379251f5eSSepherosa Ziehau int txd; 437479251f5eSSepherosa Ziehau 437579251f5eSSepherosa Ziehau txd = sc->tx_rings[0].tx_ndesc; 437679251f5eSSepherosa Ziehau return sysctl_handle_int(oidp, &txd, 0, req); 437779251f5eSSepherosa Ziehau } 437879251f5eSSepherosa Ziehau 437979251f5eSSepherosa Ziehau static int 438079251f5eSSepherosa Ziehau ix_sysctl_rxd(SYSCTL_HANDLER_ARGS) 438179251f5eSSepherosa Ziehau { 438279251f5eSSepherosa Ziehau struct ix_softc *sc = (void *)arg1; 438379251f5eSSepherosa Ziehau int rxd; 438479251f5eSSepherosa Ziehau 438579251f5eSSepherosa Ziehau rxd = sc->rx_rings[0].rx_ndesc; 438679251f5eSSepherosa Ziehau return sysctl_handle_int(oidp, &rxd, 0, req); 438779251f5eSSepherosa Ziehau } 438879251f5eSSepherosa Ziehau 438979251f5eSSepherosa Ziehau static int 439079251f5eSSepherosa Ziehau ix_sysctl_tx_intr_nsegs(SYSCTL_HANDLER_ARGS) 439179251f5eSSepherosa Ziehau { 439279251f5eSSepherosa Ziehau struct ix_softc *sc = (void *)arg1; 439379251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 439479251f5eSSepherosa Ziehau struct ix_tx_ring *txr = &sc->tx_rings[0]; 439579251f5eSSepherosa Ziehau int error, nsegs; 439679251f5eSSepherosa Ziehau 439779251f5eSSepherosa Ziehau nsegs = txr->tx_intr_nsegs; 439879251f5eSSepherosa Ziehau error = sysctl_handle_int(oidp, &nsegs, 0, req); 439979251f5eSSepherosa Ziehau if (error || req->newptr == NULL) 440079251f5eSSepherosa Ziehau return error; 440179251f5eSSepherosa Ziehau if (nsegs < 0) 440279251f5eSSepherosa Ziehau return EINVAL; 440379251f5eSSepherosa Ziehau 440479251f5eSSepherosa Ziehau ifnet_serialize_all(ifp); 440579251f5eSSepherosa Ziehau 440679251f5eSSepherosa Ziehau if (nsegs >= txr->tx_ndesc - IX_MAX_SCATTER - IX_TX_RESERVED) { 440779251f5eSSepherosa Ziehau error = EINVAL; 440879251f5eSSepherosa Ziehau } else { 440979251f5eSSepherosa Ziehau int i; 441079251f5eSSepherosa Ziehau 441179251f5eSSepherosa Ziehau error = 0; 441279251f5eSSepherosa Ziehau for (i = 0; i < sc->tx_ring_cnt; ++i) 441379251f5eSSepherosa Ziehau sc->tx_rings[i].tx_intr_nsegs = nsegs; 441479251f5eSSepherosa Ziehau } 441579251f5eSSepherosa Ziehau 441679251f5eSSepherosa Ziehau ifnet_deserialize_all(ifp); 441779251f5eSSepherosa Ziehau 441879251f5eSSepherosa Ziehau return error; 441979251f5eSSepherosa Ziehau } 442079251f5eSSepherosa Ziehau 442179251f5eSSepherosa Ziehau static void 442279251f5eSSepherosa Ziehau ix_set_eitr(struct ix_softc *sc, int idx, int rate) 442379251f5eSSepherosa Ziehau { 442479251f5eSSepherosa Ziehau uint32_t eitr, eitr_intvl; 442579251f5eSSepherosa Ziehau 442679251f5eSSepherosa Ziehau eitr = IXGBE_READ_REG(&sc->hw, IXGBE_EITR(idx)); 442779251f5eSSepherosa Ziehau eitr_intvl = 1000000000 / 256 / rate; 442879251f5eSSepherosa Ziehau 442979251f5eSSepherosa Ziehau if (sc->hw.mac.type == ixgbe_mac_82598EB) { 443079251f5eSSepherosa Ziehau eitr &= ~IX_EITR_INTVL_MASK_82598; 443179251f5eSSepherosa Ziehau if (eitr_intvl == 0) 443279251f5eSSepherosa Ziehau eitr_intvl = 1; 443379251f5eSSepherosa Ziehau else if (eitr_intvl > IX_EITR_INTVL_MASK_82598) 443479251f5eSSepherosa Ziehau eitr_intvl = IX_EITR_INTVL_MASK_82598; 443579251f5eSSepherosa Ziehau } else { 443679251f5eSSepherosa Ziehau eitr &= ~IX_EITR_INTVL_MASK; 443779251f5eSSepherosa Ziehau 443879251f5eSSepherosa Ziehau eitr_intvl &= ~IX_EITR_INTVL_RSVD_MASK; 443979251f5eSSepherosa Ziehau if (eitr_intvl == 0) 444079251f5eSSepherosa Ziehau eitr_intvl = IX_EITR_INTVL_MIN; 444179251f5eSSepherosa Ziehau else if (eitr_intvl > IX_EITR_INTVL_MAX) 444279251f5eSSepherosa Ziehau eitr_intvl = IX_EITR_INTVL_MAX; 444379251f5eSSepherosa Ziehau } 444479251f5eSSepherosa Ziehau eitr |= eitr_intvl; 444579251f5eSSepherosa Ziehau 444679251f5eSSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_EITR(idx), eitr); 444779251f5eSSepherosa Ziehau } 444879251f5eSSepherosa Ziehau 444979251f5eSSepherosa Ziehau static int 4450189a0ff3SSepherosa Ziehau ix_sysctl_rxtx_intr_rate(SYSCTL_HANDLER_ARGS) 4451189a0ff3SSepherosa Ziehau { 4452189a0ff3SSepherosa Ziehau return ix_sysctl_intr_rate(oidp, arg1, arg2, req, IX_INTR_USE_RXTX); 4453189a0ff3SSepherosa Ziehau } 4454189a0ff3SSepherosa Ziehau 4455189a0ff3SSepherosa Ziehau static int 4456189a0ff3SSepherosa Ziehau ix_sysctl_rx_intr_rate(SYSCTL_HANDLER_ARGS) 4457189a0ff3SSepherosa Ziehau { 4458189a0ff3SSepherosa Ziehau return ix_sysctl_intr_rate(oidp, arg1, arg2, req, IX_INTR_USE_RX); 4459189a0ff3SSepherosa Ziehau } 4460189a0ff3SSepherosa Ziehau 4461189a0ff3SSepherosa Ziehau static int 4462189a0ff3SSepherosa Ziehau ix_sysctl_tx_intr_rate(SYSCTL_HANDLER_ARGS) 4463189a0ff3SSepherosa Ziehau { 4464189a0ff3SSepherosa Ziehau return ix_sysctl_intr_rate(oidp, arg1, arg2, req, IX_INTR_USE_TX); 4465189a0ff3SSepherosa Ziehau } 4466189a0ff3SSepherosa Ziehau 4467189a0ff3SSepherosa Ziehau static int 4468189a0ff3SSepherosa Ziehau ix_sysctl_sts_intr_rate(SYSCTL_HANDLER_ARGS) 4469189a0ff3SSepherosa Ziehau { 4470189a0ff3SSepherosa Ziehau return ix_sysctl_intr_rate(oidp, arg1, arg2, req, IX_INTR_USE_STATUS); 4471189a0ff3SSepherosa Ziehau } 4472189a0ff3SSepherosa Ziehau 4473189a0ff3SSepherosa Ziehau static int 4474189a0ff3SSepherosa Ziehau ix_sysctl_intr_rate(SYSCTL_HANDLER_ARGS, int use) 447579251f5eSSepherosa Ziehau { 447679251f5eSSepherosa Ziehau struct ix_softc *sc = (void *)arg1; 447779251f5eSSepherosa Ziehau struct ifnet *ifp = &sc->arpcom.ac_if; 447879251f5eSSepherosa Ziehau int error, rate, i; 447979251f5eSSepherosa Ziehau 448079251f5eSSepherosa Ziehau rate = 0; 448179251f5eSSepherosa Ziehau for (i = 0; i < sc->intr_cnt; ++i) { 4482189a0ff3SSepherosa Ziehau if (sc->intr_data[i].intr_use == use) { 448379251f5eSSepherosa Ziehau rate = sc->intr_data[i].intr_rate; 448479251f5eSSepherosa Ziehau break; 448579251f5eSSepherosa Ziehau } 448679251f5eSSepherosa Ziehau } 448779251f5eSSepherosa Ziehau 448879251f5eSSepherosa Ziehau error = sysctl_handle_int(oidp, &rate, 0, req); 448979251f5eSSepherosa Ziehau if (error || req->newptr == NULL) 449079251f5eSSepherosa Ziehau return error; 449179251f5eSSepherosa Ziehau if (rate <= 0) 449279251f5eSSepherosa Ziehau return EINVAL; 449379251f5eSSepherosa Ziehau 449479251f5eSSepherosa Ziehau ifnet_serialize_all(ifp); 449579251f5eSSepherosa Ziehau 449679251f5eSSepherosa Ziehau for (i = 0; i < sc->intr_cnt; ++i) { 4497189a0ff3SSepherosa Ziehau if (sc->intr_data[i].intr_use == use) { 449879251f5eSSepherosa Ziehau sc->intr_data[i].intr_rate = rate; 449979251f5eSSepherosa Ziehau if (ifp->if_flags & IFF_RUNNING) 450079251f5eSSepherosa Ziehau ix_set_eitr(sc, i, rate); 450179251f5eSSepherosa Ziehau } 450279251f5eSSepherosa Ziehau } 450379251f5eSSepherosa Ziehau 450479251f5eSSepherosa Ziehau ifnet_deserialize_all(ifp); 450579251f5eSSepherosa Ziehau 450679251f5eSSepherosa Ziehau return error; 450779251f5eSSepherosa Ziehau } 450879251f5eSSepherosa Ziehau 450979251f5eSSepherosa Ziehau static void 4510189a0ff3SSepherosa Ziehau ix_add_intr_rate_sysctl(struct ix_softc *sc, int use, 4511189a0ff3SSepherosa Ziehau const char *name, int (*handler)(SYSCTL_HANDLER_ARGS), const char *desc) 4512189a0ff3SSepherosa Ziehau { 4513189a0ff3SSepherosa Ziehau int i; 4514189a0ff3SSepherosa Ziehau 4515189a0ff3SSepherosa Ziehau for (i = 0; i < sc->intr_cnt; ++i) { 4516189a0ff3SSepherosa Ziehau if (sc->intr_data[i].intr_use == use) { 451726595b18SSascha Wildner SYSCTL_ADD_PROC(device_get_sysctl_ctx(sc->dev), 451826595b18SSascha Wildner SYSCTL_CHILDREN(device_get_sysctl_tree(sc->dev)), 4519189a0ff3SSepherosa Ziehau OID_AUTO, name, CTLTYPE_INT | CTLFLAG_RW, 4520189a0ff3SSepherosa Ziehau sc, 0, handler, "I", desc); 4521189a0ff3SSepherosa Ziehau break; 4522189a0ff3SSepherosa Ziehau } 4523189a0ff3SSepherosa Ziehau } 4524189a0ff3SSepherosa Ziehau } 4525189a0ff3SSepherosa Ziehau 4526189a0ff3SSepherosa Ziehau static void 452779251f5eSSepherosa Ziehau ix_set_timer_cpuid(struct ix_softc *sc, boolean_t polling) 452879251f5eSSepherosa Ziehau { 452979251f5eSSepherosa Ziehau if (polling || sc->intr_type == PCI_INTR_TYPE_MSIX) 453079251f5eSSepherosa Ziehau sc->timer_cpuid = 0; /* XXX fixed */ 453179251f5eSSepherosa Ziehau else 453279251f5eSSepherosa Ziehau sc->timer_cpuid = rman_get_cpuid(sc->intr_data[0].intr_res); 453379251f5eSSepherosa Ziehau } 4534189a0ff3SSepherosa Ziehau 4535189a0ff3SSepherosa Ziehau static void 4536189a0ff3SSepherosa Ziehau ix_alloc_msix(struct ix_softc *sc) 4537189a0ff3SSepherosa Ziehau { 45383c37d13bSSepherosa Ziehau int msix_enable, msix_cnt, msix_ring, alloc_cnt; 4539189a0ff3SSepherosa Ziehau struct ix_intr_data *intr; 4540189a0ff3SSepherosa Ziehau int i, x, error; 45413c37d13bSSepherosa Ziehau int ring_cnt, ring_cntmax; 45423c37d13bSSepherosa Ziehau boolean_t setup = FALSE; 4543189a0ff3SSepherosa Ziehau 4544189a0ff3SSepherosa Ziehau msix_enable = ix_msix_enable; 4545189a0ff3SSepherosa Ziehau /* 4546189a0ff3SSepherosa Ziehau * Don't enable MSI-X on 82598 by default, see: 4547189a0ff3SSepherosa Ziehau * 82598 specification update errata #38 4548189a0ff3SSepherosa Ziehau */ 4549189a0ff3SSepherosa Ziehau if (sc->hw.mac.type == ixgbe_mac_82598EB) 4550189a0ff3SSepherosa Ziehau msix_enable = 0; 4551189a0ff3SSepherosa Ziehau msix_enable = device_getenv_int(sc->dev, "msix.enable", msix_enable); 4552189a0ff3SSepherosa Ziehau if (!msix_enable) 4553189a0ff3SSepherosa Ziehau return; 4554189a0ff3SSepherosa Ziehau 4555189a0ff3SSepherosa Ziehau msix_cnt = pci_msix_count(sc->dev); 4556189a0ff3SSepherosa Ziehau #ifdef IX_MSIX_DEBUG 4557189a0ff3SSepherosa Ziehau msix_cnt = device_getenv_int(sc->dev, "msix.count", msix_cnt); 4558189a0ff3SSepherosa Ziehau #endif 4559189a0ff3SSepherosa Ziehau if (msix_cnt <= 1) { 45603c37d13bSSepherosa Ziehau /* One MSI-X model does not make sense. */ 4561189a0ff3SSepherosa Ziehau return; 4562189a0ff3SSepherosa Ziehau } 4563189a0ff3SSepherosa Ziehau 4564189a0ff3SSepherosa Ziehau /* 4565189a0ff3SSepherosa Ziehau * Make sure that we don't break interrupt related registers 4566189a0ff3SSepherosa Ziehau * (EIMS, etc) limitation. 4567189a0ff3SSepherosa Ziehau */ 4568189a0ff3SSepherosa Ziehau if (sc->hw.mac.type == ixgbe_mac_82598EB) { 45693c37d13bSSepherosa Ziehau if (msix_cnt > IX_MAX_MSIX_82598) 45703c37d13bSSepherosa Ziehau msix_cnt = IX_MAX_MSIX_82598; 4571189a0ff3SSepherosa Ziehau } else { 45723c37d13bSSepherosa Ziehau if (msix_cnt > IX_MAX_MSIX) 45733c37d13bSSepherosa Ziehau msix_cnt = IX_MAX_MSIX; 4574189a0ff3SSepherosa Ziehau } 4575189a0ff3SSepherosa Ziehau if (bootverbose) 45763c37d13bSSepherosa Ziehau device_printf(sc->dev, "MSI-X count %d\n", msix_cnt); 45773c37d13bSSepherosa Ziehau msix_ring = msix_cnt - 1; /* -1 for status */ 45783c37d13bSSepherosa Ziehau 45793c37d13bSSepherosa Ziehau /* 45803c37d13bSSepherosa Ziehau * Configure # of RX/TX rings usable by MSI-X. 45813c37d13bSSepherosa Ziehau */ 45823c37d13bSSepherosa Ziehau ix_get_rxring_cnt(sc, &ring_cnt, &ring_cntmax); 45833c37d13bSSepherosa Ziehau if (ring_cntmax > msix_ring) 45843c37d13bSSepherosa Ziehau ring_cntmax = msix_ring; 45853c37d13bSSepherosa Ziehau sc->rx_rmap_intr = if_ringmap_alloc(sc->dev, ring_cnt, ring_cntmax); 45863c37d13bSSepherosa Ziehau 45873c37d13bSSepherosa Ziehau ix_get_txring_cnt(sc, &ring_cnt, &ring_cntmax); 45883c37d13bSSepherosa Ziehau if (ring_cntmax > msix_ring) 45893c37d13bSSepherosa Ziehau ring_cntmax = msix_ring; 45903c37d13bSSepherosa Ziehau sc->tx_rmap_intr = if_ringmap_alloc(sc->dev, ring_cnt, ring_cntmax); 45913c37d13bSSepherosa Ziehau 45923c37d13bSSepherosa Ziehau if_ringmap_match(sc->dev, sc->rx_rmap_intr, sc->tx_rmap_intr); 45933c37d13bSSepherosa Ziehau sc->rx_ring_msix = if_ringmap_count(sc->rx_rmap_intr); 45943c37d13bSSepherosa Ziehau KASSERT(sc->rx_ring_msix <= sc->rx_ring_cnt, 45953c37d13bSSepherosa Ziehau ("total RX ring count %d, MSI-X RX ring count %d", 45963c37d13bSSepherosa Ziehau sc->rx_ring_cnt, sc->rx_ring_msix)); 45973c37d13bSSepherosa Ziehau sc->tx_ring_msix = if_ringmap_count(sc->tx_rmap_intr); 45983c37d13bSSepherosa Ziehau KASSERT(sc->tx_ring_msix <= sc->tx_ring_cnt, 45993c37d13bSSepherosa Ziehau ("total TX ring count %d, MSI-X TX ring count %d", 46003c37d13bSSepherosa Ziehau sc->tx_ring_cnt, sc->tx_ring_msix)); 46013c37d13bSSepherosa Ziehau 4602189a0ff3SSepherosa Ziehau /* 4603189a0ff3SSepherosa Ziehau * Aggregate TX/RX MSI-X 4604189a0ff3SSepherosa Ziehau */ 46053c37d13bSSepherosa Ziehau ring_cntmax = sc->rx_ring_msix; 46063c37d13bSSepherosa Ziehau if (ring_cntmax < sc->tx_ring_msix) 46073c37d13bSSepherosa Ziehau ring_cntmax = sc->tx_ring_msix; 46083c37d13bSSepherosa Ziehau KASSERT(ring_cntmax <= msix_ring, 46093c37d13bSSepherosa Ziehau ("invalid ring count max %d, MSI-X count for rings %d", 46103c37d13bSSepherosa Ziehau ring_cntmax, msix_ring)); 4611189a0ff3SSepherosa Ziehau 46123c37d13bSSepherosa Ziehau alloc_cnt = ring_cntmax + 1; /* +1 for status */ 4613189a0ff3SSepherosa Ziehau if (bootverbose) { 4614189a0ff3SSepherosa Ziehau device_printf(sc->dev, "MSI-X alloc %d, " 4615189a0ff3SSepherosa Ziehau "RX ring %d, TX ring %d\n", alloc_cnt, 4616189a0ff3SSepherosa Ziehau sc->rx_ring_msix, sc->tx_ring_msix); 4617189a0ff3SSepherosa Ziehau } 4618189a0ff3SSepherosa Ziehau 4619189a0ff3SSepherosa Ziehau sc->msix_mem_rid = PCIR_BAR(IX_MSIX_BAR_82598); 4620189a0ff3SSepherosa Ziehau sc->msix_mem_res = bus_alloc_resource_any(sc->dev, SYS_RES_MEMORY, 4621189a0ff3SSepherosa Ziehau &sc->msix_mem_rid, RF_ACTIVE); 4622189a0ff3SSepherosa Ziehau if (sc->msix_mem_res == NULL) { 4623189a0ff3SSepherosa Ziehau sc->msix_mem_rid = PCIR_BAR(IX_MSIX_BAR_82599); 4624189a0ff3SSepherosa Ziehau sc->msix_mem_res = bus_alloc_resource_any(sc->dev, 4625189a0ff3SSepherosa Ziehau SYS_RES_MEMORY, &sc->msix_mem_rid, RF_ACTIVE); 4626189a0ff3SSepherosa Ziehau if (sc->msix_mem_res == NULL) { 4627189a0ff3SSepherosa Ziehau device_printf(sc->dev, "Unable to map MSI-X table\n"); 4628189a0ff3SSepherosa Ziehau return; 4629189a0ff3SSepherosa Ziehau } 4630189a0ff3SSepherosa Ziehau } 4631189a0ff3SSepherosa Ziehau 4632189a0ff3SSepherosa Ziehau sc->intr_cnt = alloc_cnt; 4633189a0ff3SSepherosa Ziehau sc->intr_data = kmalloc(sizeof(struct ix_intr_data) * sc->intr_cnt, 4634189a0ff3SSepherosa Ziehau M_DEVBUF, M_WAITOK | M_ZERO); 4635189a0ff3SSepherosa Ziehau for (x = 0; x < sc->intr_cnt; ++x) { 4636189a0ff3SSepherosa Ziehau intr = &sc->intr_data[x]; 4637189a0ff3SSepherosa Ziehau intr->intr_rid = -1; 4638189a0ff3SSepherosa Ziehau intr->intr_rate = IX_INTR_RATE; 4639189a0ff3SSepherosa Ziehau } 4640189a0ff3SSepherosa Ziehau 4641189a0ff3SSepherosa Ziehau x = 0; 46423c37d13bSSepherosa Ziehau for (i = 0; i < sc->rx_ring_msix; ++i) { 4643189a0ff3SSepherosa Ziehau struct ix_rx_ring *rxr = &sc->rx_rings[i]; 46443c37d13bSSepherosa Ziehau struct ix_tx_ring *txr = NULL; 46453c37d13bSSepherosa Ziehau int cpuid, j; 4646189a0ff3SSepherosa Ziehau 4647189a0ff3SSepherosa Ziehau KKASSERT(x < sc->intr_cnt); 4648189a0ff3SSepherosa Ziehau rxr->rx_intr_vec = x; 4649189a0ff3SSepherosa Ziehau ix_setup_msix_eims(sc, x, 4650189a0ff3SSepherosa Ziehau &rxr->rx_eims, &rxr->rx_eims_val); 46513c37d13bSSepherosa Ziehau 46523c37d13bSSepherosa Ziehau cpuid = if_ringmap_cpumap(sc->rx_rmap_intr, i); 46533c37d13bSSepherosa Ziehau 46543c37d13bSSepherosa Ziehau /* 46553c37d13bSSepherosa Ziehau * Try finding TX ring to piggyback. 46563c37d13bSSepherosa Ziehau */ 46573c37d13bSSepherosa Ziehau for (j = 0; j < sc->tx_ring_msix; ++j) { 46583c37d13bSSepherosa Ziehau if (cpuid == 46593c37d13bSSepherosa Ziehau if_ringmap_cpumap(sc->tx_rmap_intr, j)) { 46603c37d13bSSepherosa Ziehau txr = &sc->tx_rings[j]; 46613c37d13bSSepherosa Ziehau KKASSERT(txr->tx_intr_cpuid < 0); 46623c37d13bSSepherosa Ziehau break; 46633c37d13bSSepherosa Ziehau } 46643c37d13bSSepherosa Ziehau } 4665189a0ff3SSepherosa Ziehau rxr->rx_txr = txr; 4666189a0ff3SSepherosa Ziehau 4667189a0ff3SSepherosa Ziehau intr = &sc->intr_data[x++]; 4668189a0ff3SSepherosa Ziehau intr->intr_serialize = &rxr->rx_serialize; 46693c37d13bSSepherosa Ziehau if (txr != NULL) { 46703c37d13bSSepherosa Ziehau ksnprintf(intr->intr_desc0, 46713c37d13bSSepherosa Ziehau sizeof(intr->intr_desc0), "%s rx%dtx%d", 46723c37d13bSSepherosa Ziehau device_get_nameunit(sc->dev), i, txr->tx_idx); 4673189a0ff3SSepherosa Ziehau intr->intr_use = IX_INTR_USE_RXTX; 46743c37d13bSSepherosa Ziehau intr->intr_func = ix_msix_rxtx; 46753c37d13bSSepherosa Ziehau } else { 46763c37d13bSSepherosa Ziehau ksnprintf(intr->intr_desc0, 46773c37d13bSSepherosa Ziehau sizeof(intr->intr_desc0), "%s rx%d", 46783c37d13bSSepherosa Ziehau device_get_nameunit(sc->dev), i); 46793c37d13bSSepherosa Ziehau intr->intr_rate = IX_MSIX_RX_RATE; 46803c37d13bSSepherosa Ziehau intr->intr_use = IX_INTR_USE_RX; 46813c37d13bSSepherosa Ziehau intr->intr_func = ix_msix_rx; 46823c37d13bSSepherosa Ziehau } 46833c37d13bSSepherosa Ziehau intr->intr_funcarg = rxr; 46843c37d13bSSepherosa Ziehau intr->intr_cpuid = cpuid; 46853c37d13bSSepherosa Ziehau KKASSERT(intr->intr_cpuid < netisr_ncpus); 4686189a0ff3SSepherosa Ziehau intr->intr_desc = intr->intr_desc0; 46873c37d13bSSepherosa Ziehau 46883c37d13bSSepherosa Ziehau if (txr != NULL) { 46893c37d13bSSepherosa Ziehau txr->tx_intr_cpuid = intr->intr_cpuid; 46903c37d13bSSepherosa Ziehau /* NOTE: Leave TX ring's intr_vec negative. */ 46913c37d13bSSepherosa Ziehau } 4692189a0ff3SSepherosa Ziehau } 4693189a0ff3SSepherosa Ziehau 46943c37d13bSSepherosa Ziehau for (i = 0; i < sc->tx_ring_msix; ++i) { 46953c37d13bSSepherosa Ziehau struct ix_tx_ring *txr = &sc->tx_rings[i]; 46963c37d13bSSepherosa Ziehau 46973c37d13bSSepherosa Ziehau if (txr->tx_intr_cpuid >= 0) { 46983c37d13bSSepherosa Ziehau /* Piggybacked by RX ring. */ 46993c37d13bSSepherosa Ziehau continue; 4700189a0ff3SSepherosa Ziehau } 47013c37d13bSSepherosa Ziehau 47023c37d13bSSepherosa Ziehau KKASSERT(x < sc->intr_cnt); 47033c37d13bSSepherosa Ziehau txr->tx_intr_vec = x; 47043c37d13bSSepherosa Ziehau ix_setup_msix_eims(sc, x, &txr->tx_eims, &txr->tx_eims_val); 47053c37d13bSSepherosa Ziehau 47063c37d13bSSepherosa Ziehau intr = &sc->intr_data[x++]; 47073c37d13bSSepherosa Ziehau intr->intr_serialize = &txr->tx_serialize; 47083c37d13bSSepherosa Ziehau intr->intr_rate = IX_MSIX_TX_RATE; 47093c37d13bSSepherosa Ziehau intr->intr_use = IX_INTR_USE_TX; 47103c37d13bSSepherosa Ziehau intr->intr_func = ix_msix_tx; 47113c37d13bSSepherosa Ziehau intr->intr_funcarg = txr; 47123c37d13bSSepherosa Ziehau intr->intr_cpuid = if_ringmap_cpumap(sc->tx_rmap_intr, i); 47133c37d13bSSepherosa Ziehau KKASSERT(intr->intr_cpuid < netisr_ncpus); 47143c37d13bSSepherosa Ziehau ksnprintf(intr->intr_desc0, sizeof(intr->intr_desc0), "%s tx%d", 47153c37d13bSSepherosa Ziehau device_get_nameunit(sc->dev), i); 47163c37d13bSSepherosa Ziehau intr->intr_desc = intr->intr_desc0; 47173c37d13bSSepherosa Ziehau 47183c37d13bSSepherosa Ziehau txr->tx_intr_cpuid = intr->intr_cpuid; 4719189a0ff3SSepherosa Ziehau } 4720189a0ff3SSepherosa Ziehau 4721189a0ff3SSepherosa Ziehau /* 4722189a0ff3SSepherosa Ziehau * Status MSI-X 4723189a0ff3SSepherosa Ziehau */ 4724189a0ff3SSepherosa Ziehau KKASSERT(x < sc->intr_cnt); 4725189a0ff3SSepherosa Ziehau sc->sts_msix_vec = x; 4726189a0ff3SSepherosa Ziehau 4727189a0ff3SSepherosa Ziehau intr = &sc->intr_data[x++]; 4728189a0ff3SSepherosa Ziehau 4729189a0ff3SSepherosa Ziehau intr->intr_serialize = &sc->main_serialize; 4730189a0ff3SSepherosa Ziehau intr->intr_func = ix_msix_status; 4731189a0ff3SSepherosa Ziehau intr->intr_funcarg = sc; 4732189a0ff3SSepherosa Ziehau intr->intr_cpuid = 0; 4733189a0ff3SSepherosa Ziehau intr->intr_use = IX_INTR_USE_STATUS; 4734189a0ff3SSepherosa Ziehau 4735189a0ff3SSepherosa Ziehau ksnprintf(intr->intr_desc0, sizeof(intr->intr_desc0), "%s sts", 4736189a0ff3SSepherosa Ziehau device_get_nameunit(sc->dev)); 4737189a0ff3SSepherosa Ziehau intr->intr_desc = intr->intr_desc0; 4738189a0ff3SSepherosa Ziehau 4739189a0ff3SSepherosa Ziehau KKASSERT(x == sc->intr_cnt); 4740189a0ff3SSepherosa Ziehau 4741189a0ff3SSepherosa Ziehau error = pci_setup_msix(sc->dev); 4742189a0ff3SSepherosa Ziehau if (error) { 4743189a0ff3SSepherosa Ziehau device_printf(sc->dev, "Setup MSI-X failed\n"); 4744189a0ff3SSepherosa Ziehau goto back; 4745189a0ff3SSepherosa Ziehau } 4746189a0ff3SSepherosa Ziehau setup = TRUE; 4747189a0ff3SSepherosa Ziehau 4748189a0ff3SSepherosa Ziehau for (i = 0; i < sc->intr_cnt; ++i) { 4749189a0ff3SSepherosa Ziehau intr = &sc->intr_data[i]; 4750189a0ff3SSepherosa Ziehau 4751189a0ff3SSepherosa Ziehau error = pci_alloc_msix_vector(sc->dev, i, &intr->intr_rid, 4752189a0ff3SSepherosa Ziehau intr->intr_cpuid); 4753189a0ff3SSepherosa Ziehau if (error) { 4754189a0ff3SSepherosa Ziehau device_printf(sc->dev, 4755189a0ff3SSepherosa Ziehau "Unable to allocate MSI-X %d on cpu%d\n", i, 4756189a0ff3SSepherosa Ziehau intr->intr_cpuid); 4757189a0ff3SSepherosa Ziehau goto back; 4758189a0ff3SSepherosa Ziehau } 4759189a0ff3SSepherosa Ziehau 4760189a0ff3SSepherosa Ziehau intr->intr_res = bus_alloc_resource_any(sc->dev, SYS_RES_IRQ, 4761189a0ff3SSepherosa Ziehau &intr->intr_rid, RF_ACTIVE); 4762189a0ff3SSepherosa Ziehau if (intr->intr_res == NULL) { 4763189a0ff3SSepherosa Ziehau device_printf(sc->dev, 4764189a0ff3SSepherosa Ziehau "Unable to allocate MSI-X %d resource\n", i); 4765189a0ff3SSepherosa Ziehau error = ENOMEM; 4766189a0ff3SSepherosa Ziehau goto back; 4767189a0ff3SSepherosa Ziehau } 4768189a0ff3SSepherosa Ziehau } 4769189a0ff3SSepherosa Ziehau 4770189a0ff3SSepherosa Ziehau pci_enable_msix(sc->dev); 4771189a0ff3SSepherosa Ziehau sc->intr_type = PCI_INTR_TYPE_MSIX; 4772189a0ff3SSepherosa Ziehau back: 4773189a0ff3SSepherosa Ziehau if (error) 4774189a0ff3SSepherosa Ziehau ix_free_msix(sc, setup); 4775189a0ff3SSepherosa Ziehau } 4776189a0ff3SSepherosa Ziehau 4777189a0ff3SSepherosa Ziehau static void 4778189a0ff3SSepherosa Ziehau ix_free_msix(struct ix_softc *sc, boolean_t setup) 4779189a0ff3SSepherosa Ziehau { 4780189a0ff3SSepherosa Ziehau int i; 4781189a0ff3SSepherosa Ziehau 4782189a0ff3SSepherosa Ziehau KKASSERT(sc->intr_cnt > 1); 4783189a0ff3SSepherosa Ziehau 4784189a0ff3SSepherosa Ziehau for (i = 0; i < sc->intr_cnt; ++i) { 4785189a0ff3SSepherosa Ziehau struct ix_intr_data *intr = &sc->intr_data[i]; 4786189a0ff3SSepherosa Ziehau 4787189a0ff3SSepherosa Ziehau if (intr->intr_res != NULL) { 4788189a0ff3SSepherosa Ziehau bus_release_resource(sc->dev, SYS_RES_IRQ, 4789189a0ff3SSepherosa Ziehau intr->intr_rid, intr->intr_res); 4790189a0ff3SSepherosa Ziehau } 4791189a0ff3SSepherosa Ziehau if (intr->intr_rid >= 0) 4792189a0ff3SSepherosa Ziehau pci_release_msix_vector(sc->dev, intr->intr_rid); 4793189a0ff3SSepherosa Ziehau } 4794189a0ff3SSepherosa Ziehau if (setup) 4795189a0ff3SSepherosa Ziehau pci_teardown_msix(sc->dev); 4796189a0ff3SSepherosa Ziehau 4797189a0ff3SSepherosa Ziehau sc->intr_cnt = 0; 4798189a0ff3SSepherosa Ziehau kfree(sc->intr_data, M_DEVBUF); 4799189a0ff3SSepherosa Ziehau sc->intr_data = NULL; 4800189a0ff3SSepherosa Ziehau } 4801189a0ff3SSepherosa Ziehau 4802189a0ff3SSepherosa Ziehau static void 4803189a0ff3SSepherosa Ziehau ix_msix_rx(void *xrxr) 4804189a0ff3SSepherosa Ziehau { 4805189a0ff3SSepherosa Ziehau struct ix_rx_ring *rxr = xrxr; 4806189a0ff3SSepherosa Ziehau 4807189a0ff3SSepherosa Ziehau ASSERT_SERIALIZED(&rxr->rx_serialize); 4808189a0ff3SSepherosa Ziehau 48094a648aefSSepherosa Ziehau ix_rxeof(rxr, -1); 4810189a0ff3SSepherosa Ziehau IXGBE_WRITE_REG(&rxr->rx_sc->hw, rxr->rx_eims, rxr->rx_eims_val); 4811189a0ff3SSepherosa Ziehau } 4812189a0ff3SSepherosa Ziehau 4813189a0ff3SSepherosa Ziehau static void 4814189a0ff3SSepherosa Ziehau ix_msix_tx(void *xtxr) 4815189a0ff3SSepherosa Ziehau { 4816189a0ff3SSepherosa Ziehau struct ix_tx_ring *txr = xtxr; 4817189a0ff3SSepherosa Ziehau 4818189a0ff3SSepherosa Ziehau ASSERT_SERIALIZED(&txr->tx_serialize); 4819189a0ff3SSepherosa Ziehau 482082db96e9SSepherosa Ziehau ix_tx_intr(txr, *(txr->tx_hdr)); 4821189a0ff3SSepherosa Ziehau IXGBE_WRITE_REG(&txr->tx_sc->hw, txr->tx_eims, txr->tx_eims_val); 4822189a0ff3SSepherosa Ziehau } 4823189a0ff3SSepherosa Ziehau 4824189a0ff3SSepherosa Ziehau static void 4825189a0ff3SSepherosa Ziehau ix_msix_rxtx(void *xrxr) 4826189a0ff3SSepherosa Ziehau { 4827189a0ff3SSepherosa Ziehau struct ix_rx_ring *rxr = xrxr; 4828189a0ff3SSepherosa Ziehau struct ix_tx_ring *txr; 4829189a0ff3SSepherosa Ziehau int hdr; 4830189a0ff3SSepherosa Ziehau 4831189a0ff3SSepherosa Ziehau ASSERT_SERIALIZED(&rxr->rx_serialize); 4832189a0ff3SSepherosa Ziehau 48334a648aefSSepherosa Ziehau ix_rxeof(rxr, -1); 4834189a0ff3SSepherosa Ziehau 4835189a0ff3SSepherosa Ziehau /* 4836189a0ff3SSepherosa Ziehau * NOTE: 4837189a0ff3SSepherosa Ziehau * Since tx_next_clean is only changed by ix_txeof(), 4838189a0ff3SSepherosa Ziehau * which is called only in interrupt handler, the 4839189a0ff3SSepherosa Ziehau * check w/o holding tx serializer is MPSAFE. 4840189a0ff3SSepherosa Ziehau */ 4841189a0ff3SSepherosa Ziehau txr = rxr->rx_txr; 4842189a0ff3SSepherosa Ziehau hdr = *(txr->tx_hdr); 4843189a0ff3SSepherosa Ziehau if (hdr != txr->tx_next_clean) { 4844189a0ff3SSepherosa Ziehau lwkt_serialize_enter(&txr->tx_serialize); 484582db96e9SSepherosa Ziehau ix_tx_intr(txr, hdr); 4846189a0ff3SSepherosa Ziehau lwkt_serialize_exit(&txr->tx_serialize); 4847189a0ff3SSepherosa Ziehau } 4848189a0ff3SSepherosa Ziehau 4849189a0ff3SSepherosa Ziehau IXGBE_WRITE_REG(&rxr->rx_sc->hw, rxr->rx_eims, rxr->rx_eims_val); 4850189a0ff3SSepherosa Ziehau } 4851189a0ff3SSepherosa Ziehau 4852189a0ff3SSepherosa Ziehau static void 4853189a0ff3SSepherosa Ziehau ix_intr_status(struct ix_softc *sc, uint32_t eicr) 4854189a0ff3SSepherosa Ziehau { 4855189a0ff3SSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 4856189a0ff3SSepherosa Ziehau 4857189a0ff3SSepherosa Ziehau /* Link status change */ 4858189a0ff3SSepherosa Ziehau if (eicr & IXGBE_EICR_LSC) 4859189a0ff3SSepherosa Ziehau ix_handle_link(sc); 4860189a0ff3SSepherosa Ziehau 4861189a0ff3SSepherosa Ziehau if (hw->mac.type != ixgbe_mac_82598EB) { 4862189a0ff3SSepherosa Ziehau if (eicr & IXGBE_EICR_ECC) 4863189a0ff3SSepherosa Ziehau if_printf(&sc->arpcom.ac_if, "ECC ERROR!! Reboot!!\n"); 486463d483cdSSepherosa Ziehau 486563d483cdSSepherosa Ziehau /* Check for over temp condition */ 486663d483cdSSepherosa Ziehau if (eicr & IXGBE_EICR_TS) { 486763d483cdSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, "CRITICAL: OVER TEMP!! " 486863d483cdSSepherosa Ziehau "PHY IS SHUT DOWN!! Shutdown!!\n"); 486963d483cdSSepherosa Ziehau } 487063d483cdSSepherosa Ziehau } 487163d483cdSSepherosa Ziehau 487263d483cdSSepherosa Ziehau if (ix_is_sfp(hw)) { 487363d483cdSSepherosa Ziehau uint32_t mod_mask; 487463d483cdSSepherosa Ziehau 487563d483cdSSepherosa Ziehau /* Pluggable optics-related interrupt */ 487663d483cdSSepherosa Ziehau if (hw->device_id == IXGBE_DEV_ID_X550EM_X_SFP) 487763d483cdSSepherosa Ziehau mod_mask = IXGBE_EICR_GPI_SDP0_X540; 487863d483cdSSepherosa Ziehau else 487963d483cdSSepherosa Ziehau mod_mask = IXGBE_EICR_GPI_SDP2_BY_MAC(hw); 488063d483cdSSepherosa Ziehau if (eicr & IXGBE_EICR_GPI_SDP1_BY_MAC(hw)) 4881189a0ff3SSepherosa Ziehau ix_handle_msf(sc); 488263d483cdSSepherosa Ziehau else if (eicr & mod_mask) 4883189a0ff3SSepherosa Ziehau ix_handle_mod(sc); 4884189a0ff3SSepherosa Ziehau } 4885189a0ff3SSepherosa Ziehau 4886189a0ff3SSepherosa Ziehau /* Check for fan failure */ 4887189a0ff3SSepherosa Ziehau if (hw->device_id == IXGBE_DEV_ID_82598AT && 4888189a0ff3SSepherosa Ziehau (eicr & IXGBE_EICR_GPI_SDP1)) 4889189a0ff3SSepherosa Ziehau if_printf(&sc->arpcom.ac_if, "FAN FAILURE!! Replace!!\n"); 4890189a0ff3SSepherosa Ziehau 489163d483cdSSepherosa Ziehau /* External PHY interrupt */ 489263d483cdSSepherosa Ziehau if (hw->device_id == IXGBE_DEV_ID_X550EM_X_10G_T && 489363d483cdSSepherosa Ziehau (eicr & IXGBE_EICR_GPI_SDP0_X540)) 489463d483cdSSepherosa Ziehau ix_handle_phy(sc); 4895189a0ff3SSepherosa Ziehau } 4896189a0ff3SSepherosa Ziehau 4897189a0ff3SSepherosa Ziehau static void 4898189a0ff3SSepherosa Ziehau ix_msix_status(void *xsc) 4899189a0ff3SSepherosa Ziehau { 4900189a0ff3SSepherosa Ziehau struct ix_softc *sc = xsc; 4901189a0ff3SSepherosa Ziehau uint32_t eicr; 4902189a0ff3SSepherosa Ziehau 4903189a0ff3SSepherosa Ziehau ASSERT_SERIALIZED(&sc->main_serialize); 4904189a0ff3SSepherosa Ziehau 4905189a0ff3SSepherosa Ziehau eicr = IXGBE_READ_REG(&sc->hw, IXGBE_EICR); 4906189a0ff3SSepherosa Ziehau ix_intr_status(sc, eicr); 4907189a0ff3SSepherosa Ziehau 4908189a0ff3SSepherosa Ziehau IXGBE_WRITE_REG(&sc->hw, IXGBE_EIMS, sc->intr_mask); 4909189a0ff3SSepherosa Ziehau } 4910189a0ff3SSepherosa Ziehau 4911189a0ff3SSepherosa Ziehau static void 4912189a0ff3SSepherosa Ziehau ix_setup_msix_eims(const struct ix_softc *sc, int x, 4913189a0ff3SSepherosa Ziehau uint32_t *eims, uint32_t *eims_val) 4914189a0ff3SSepherosa Ziehau { 4915189a0ff3SSepherosa Ziehau if (x < 32) { 4916189a0ff3SSepherosa Ziehau if (sc->hw.mac.type == ixgbe_mac_82598EB) { 4917189a0ff3SSepherosa Ziehau KASSERT(x < IX_MAX_MSIX_82598, 4918189a0ff3SSepherosa Ziehau ("%s: invalid vector %d for 82598", 4919189a0ff3SSepherosa Ziehau device_get_nameunit(sc->dev), x)); 4920189a0ff3SSepherosa Ziehau *eims = IXGBE_EIMS; 4921189a0ff3SSepherosa Ziehau } else { 4922189a0ff3SSepherosa Ziehau *eims = IXGBE_EIMS_EX(0); 4923189a0ff3SSepherosa Ziehau } 4924189a0ff3SSepherosa Ziehau *eims_val = 1 << x; 4925189a0ff3SSepherosa Ziehau } else { 4926189a0ff3SSepherosa Ziehau KASSERT(x < IX_MAX_MSIX, ("%s: invalid vector %d", 4927189a0ff3SSepherosa Ziehau device_get_nameunit(sc->dev), x)); 4928189a0ff3SSepherosa Ziehau KASSERT(sc->hw.mac.type != ixgbe_mac_82598EB, 4929189a0ff3SSepherosa Ziehau ("%s: invalid vector %d for 82598", 4930189a0ff3SSepherosa Ziehau device_get_nameunit(sc->dev), x)); 4931189a0ff3SSepherosa Ziehau *eims = IXGBE_EIMS_EX(1); 4932189a0ff3SSepherosa Ziehau *eims_val = 1 << (x - 32); 4933189a0ff3SSepherosa Ziehau } 4934189a0ff3SSepherosa Ziehau } 49354a648aefSSepherosa Ziehau 49364a648aefSSepherosa Ziehau #ifdef IFPOLL_ENABLE 49374a648aefSSepherosa Ziehau 49384a648aefSSepherosa Ziehau static void 49394a648aefSSepherosa Ziehau ix_npoll_status(struct ifnet *ifp) 49404a648aefSSepherosa Ziehau { 49414a648aefSSepherosa Ziehau struct ix_softc *sc = ifp->if_softc; 49424a648aefSSepherosa Ziehau uint32_t eicr; 49434a648aefSSepherosa Ziehau 49444a648aefSSepherosa Ziehau ASSERT_SERIALIZED(&sc->main_serialize); 49454a648aefSSepherosa Ziehau 49464a648aefSSepherosa Ziehau eicr = IXGBE_READ_REG(&sc->hw, IXGBE_EICR); 49474a648aefSSepherosa Ziehau ix_intr_status(sc, eicr); 49484a648aefSSepherosa Ziehau } 49494a648aefSSepherosa Ziehau 49504a648aefSSepherosa Ziehau static void 49514a648aefSSepherosa Ziehau ix_npoll_tx(struct ifnet *ifp, void *arg, int cycle __unused) 49524a648aefSSepherosa Ziehau { 49534a648aefSSepherosa Ziehau struct ix_tx_ring *txr = arg; 49544a648aefSSepherosa Ziehau 49554a648aefSSepherosa Ziehau ASSERT_SERIALIZED(&txr->tx_serialize); 49564a648aefSSepherosa Ziehau 495782db96e9SSepherosa Ziehau ix_tx_intr(txr, *(txr->tx_hdr)); 495882db96e9SSepherosa Ziehau ix_try_txgc(txr, 1); 49594a648aefSSepherosa Ziehau } 49604a648aefSSepherosa Ziehau 49614a648aefSSepherosa Ziehau static void 49624a648aefSSepherosa Ziehau ix_npoll_rx(struct ifnet *ifp __unused, void *arg, int cycle) 49634a648aefSSepherosa Ziehau { 49644a648aefSSepherosa Ziehau struct ix_rx_ring *rxr = arg; 49654a648aefSSepherosa Ziehau 49664a648aefSSepherosa Ziehau ASSERT_SERIALIZED(&rxr->rx_serialize); 49678d0afa86SSepherosa Ziehau ix_rxeof(rxr, cycle); 49688d0afa86SSepherosa Ziehau } 49694a648aefSSepherosa Ziehau 49708d0afa86SSepherosa Ziehau static void 49718d0afa86SSepherosa Ziehau ix_npoll_rx_direct(struct ifnet *ifp __unused, void *arg, int cycle) 49728d0afa86SSepherosa Ziehau { 49738d0afa86SSepherosa Ziehau struct ix_rx_ring *rxr = arg; 49748d0afa86SSepherosa Ziehau 49758d0afa86SSepherosa Ziehau ASSERT_NOT_SERIALIZED(&rxr->rx_serialize); 49764a648aefSSepherosa Ziehau ix_rxeof(rxr, cycle); 49774a648aefSSepherosa Ziehau } 49784a648aefSSepherosa Ziehau 49794a648aefSSepherosa Ziehau static void 49804a648aefSSepherosa Ziehau ix_npoll(struct ifnet *ifp, struct ifpoll_info *info) 49814a648aefSSepherosa Ziehau { 49824a648aefSSepherosa Ziehau struct ix_softc *sc = ifp->if_softc; 49838d0afa86SSepherosa Ziehau int i, txr_cnt, rxr_cnt, idirect; 49844a648aefSSepherosa Ziehau 49854a648aefSSepherosa Ziehau ASSERT_IFNET_SERIALIZED_ALL(ifp); 49864a648aefSSepherosa Ziehau 49878d0afa86SSepherosa Ziehau idirect = sc->direct_input; 49888d0afa86SSepherosa Ziehau cpu_ccfence(); 49898d0afa86SSepherosa Ziehau 49904a648aefSSepherosa Ziehau if (info) { 49913c37d13bSSepherosa Ziehau int cpu; 49924a648aefSSepherosa Ziehau 49934a648aefSSepherosa Ziehau info->ifpi_status.status_func = ix_npoll_status; 49944a648aefSSepherosa Ziehau info->ifpi_status.serializer = &sc->main_serialize; 49954a648aefSSepherosa Ziehau 49964a648aefSSepherosa Ziehau txr_cnt = ix_get_txring_inuse(sc, TRUE); 49974a648aefSSepherosa Ziehau for (i = 0; i < txr_cnt; ++i) { 49984a648aefSSepherosa Ziehau struct ix_tx_ring *txr = &sc->tx_rings[i]; 49994a648aefSSepherosa Ziehau 50003c37d13bSSepherosa Ziehau cpu = if_ringmap_cpumap(sc->tx_rmap, i); 50013c37d13bSSepherosa Ziehau KKASSERT(cpu < netisr_ncpus); 50023c37d13bSSepherosa Ziehau info->ifpi_tx[cpu].poll_func = ix_npoll_tx; 50033c37d13bSSepherosa Ziehau info->ifpi_tx[cpu].arg = txr; 50043c37d13bSSepherosa Ziehau info->ifpi_tx[cpu].serializer = &txr->tx_serialize; 50053c37d13bSSepherosa Ziehau ifsq_set_cpuid(txr->tx_ifsq, cpu); 50064a648aefSSepherosa Ziehau } 50074a648aefSSepherosa Ziehau 50084a648aefSSepherosa Ziehau rxr_cnt = ix_get_rxring_inuse(sc, TRUE); 50094a648aefSSepherosa Ziehau for (i = 0; i < rxr_cnt; ++i) { 50104a648aefSSepherosa Ziehau struct ix_rx_ring *rxr = &sc->rx_rings[i]; 50114a648aefSSepherosa Ziehau 50123c37d13bSSepherosa Ziehau cpu = if_ringmap_cpumap(sc->rx_rmap, i); 50133c37d13bSSepherosa Ziehau KKASSERT(cpu < netisr_ncpus); 50143c37d13bSSepherosa Ziehau info->ifpi_rx[cpu].arg = rxr; 50158d0afa86SSepherosa Ziehau if (idirect) { 50168d0afa86SSepherosa Ziehau info->ifpi_rx[cpu].poll_func = 50178d0afa86SSepherosa Ziehau ix_npoll_rx_direct; 50188d0afa86SSepherosa Ziehau info->ifpi_rx[cpu].serializer = NULL; 50194a648aefSSepherosa Ziehau } else { 50208d0afa86SSepherosa Ziehau info->ifpi_rx[cpu].poll_func = ix_npoll_rx; 50218d0afa86SSepherosa Ziehau info->ifpi_rx[cpu].serializer = 50228d0afa86SSepherosa Ziehau &rxr->rx_serialize; 50238d0afa86SSepherosa Ziehau } 50248d0afa86SSepherosa Ziehau } 50258d0afa86SSepherosa Ziehau if (idirect) 50268d0afa86SSepherosa Ziehau ifp->if_flags |= IFF_IDIRECT; 50278d0afa86SSepherosa Ziehau } else { 50288d0afa86SSepherosa Ziehau ifp->if_flags &= ~IFF_IDIRECT; 50294a648aefSSepherosa Ziehau for (i = 0; i < sc->tx_ring_cnt; ++i) { 50304a648aefSSepherosa Ziehau struct ix_tx_ring *txr = &sc->tx_rings[i]; 50314a648aefSSepherosa Ziehau 50324a648aefSSepherosa Ziehau ifsq_set_cpuid(txr->tx_ifsq, txr->tx_intr_cpuid); 50334a648aefSSepherosa Ziehau } 50343c37d13bSSepherosa Ziehau } 50353c37d13bSSepherosa Ziehau if (ifp->if_flags & IFF_RUNNING) 50364a648aefSSepherosa Ziehau ix_init(sc); 50374a648aefSSepherosa Ziehau } 50384a648aefSSepherosa Ziehau 50394a648aefSSepherosa Ziehau #endif /* IFPOLL_ENABLE */ 5040060fa21cSSepherosa Ziehau 5041060fa21cSSepherosa Ziehau static enum ixgbe_fc_mode 5042060fa21cSSepherosa Ziehau ix_ifmedia2fc(int ifm) 5043060fa21cSSepherosa Ziehau { 5044060fa21cSSepherosa Ziehau int fc_opt = ifm & (IFM_ETH_RXPAUSE | IFM_ETH_TXPAUSE); 5045060fa21cSSepherosa Ziehau 5046060fa21cSSepherosa Ziehau switch (fc_opt) { 5047060fa21cSSepherosa Ziehau case (IFM_ETH_RXPAUSE | IFM_ETH_TXPAUSE): 5048060fa21cSSepherosa Ziehau return ixgbe_fc_full; 5049060fa21cSSepherosa Ziehau 5050060fa21cSSepherosa Ziehau case IFM_ETH_RXPAUSE: 5051060fa21cSSepherosa Ziehau return ixgbe_fc_rx_pause; 5052060fa21cSSepherosa Ziehau 5053060fa21cSSepherosa Ziehau case IFM_ETH_TXPAUSE: 5054060fa21cSSepherosa Ziehau return ixgbe_fc_tx_pause; 5055060fa21cSSepherosa Ziehau 5056060fa21cSSepherosa Ziehau default: 5057060fa21cSSepherosa Ziehau return ixgbe_fc_none; 5058060fa21cSSepherosa Ziehau } 5059060fa21cSSepherosa Ziehau } 5060060fa21cSSepherosa Ziehau 5061060fa21cSSepherosa Ziehau static const char * 5062060fa21cSSepherosa Ziehau ix_ifmedia2str(int ifm) 5063060fa21cSSepherosa Ziehau { 5064060fa21cSSepherosa Ziehau int fc_opt = ifm & (IFM_ETH_RXPAUSE | IFM_ETH_TXPAUSE); 5065060fa21cSSepherosa Ziehau 5066060fa21cSSepherosa Ziehau switch (fc_opt) { 5067060fa21cSSepherosa Ziehau case (IFM_ETH_RXPAUSE | IFM_ETH_TXPAUSE): 5068060fa21cSSepherosa Ziehau return IFM_ETH_FC_FULL; 5069060fa21cSSepherosa Ziehau 5070060fa21cSSepherosa Ziehau case IFM_ETH_RXPAUSE: 5071060fa21cSSepherosa Ziehau return IFM_ETH_FC_RXPAUSE; 5072060fa21cSSepherosa Ziehau 5073060fa21cSSepherosa Ziehau case IFM_ETH_TXPAUSE: 5074060fa21cSSepherosa Ziehau return IFM_ETH_FC_TXPAUSE; 5075060fa21cSSepherosa Ziehau 5076060fa21cSSepherosa Ziehau default: 5077060fa21cSSepherosa Ziehau return IFM_ETH_FC_NONE; 5078060fa21cSSepherosa Ziehau } 5079060fa21cSSepherosa Ziehau } 5080060fa21cSSepherosa Ziehau 5081060fa21cSSepherosa Ziehau static const char * 5082060fa21cSSepherosa Ziehau ix_fc2str(enum ixgbe_fc_mode fc) 5083060fa21cSSepherosa Ziehau { 5084060fa21cSSepherosa Ziehau switch (fc) { 5085060fa21cSSepherosa Ziehau case ixgbe_fc_full: 5086060fa21cSSepherosa Ziehau return IFM_ETH_FC_FULL; 5087060fa21cSSepherosa Ziehau 5088060fa21cSSepherosa Ziehau case ixgbe_fc_rx_pause: 5089060fa21cSSepherosa Ziehau return IFM_ETH_FC_RXPAUSE; 5090060fa21cSSepherosa Ziehau 5091060fa21cSSepherosa Ziehau case ixgbe_fc_tx_pause: 5092060fa21cSSepherosa Ziehau return IFM_ETH_FC_TXPAUSE; 5093060fa21cSSepherosa Ziehau 5094060fa21cSSepherosa Ziehau default: 5095060fa21cSSepherosa Ziehau return IFM_ETH_FC_NONE; 5096060fa21cSSepherosa Ziehau } 5097060fa21cSSepherosa Ziehau } 509863d483cdSSepherosa Ziehau 509963d483cdSSepherosa Ziehau static int 510063d483cdSSepherosa Ziehau ix_powerdown(struct ix_softc *sc) 510163d483cdSSepherosa Ziehau { 510263d483cdSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 510363d483cdSSepherosa Ziehau int error = 0; 510463d483cdSSepherosa Ziehau 510563d483cdSSepherosa Ziehau /* Limit power managment flow to X550EM baseT */ 510663d483cdSSepherosa Ziehau if (hw->device_id == IXGBE_DEV_ID_X550EM_X_10G_T && 510763d483cdSSepherosa Ziehau hw->phy.ops.enter_lplu) { 510863d483cdSSepherosa Ziehau /* Turn off support for APM wakeup. (Using ACPI instead) */ 510963d483cdSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_GRC, 511063d483cdSSepherosa Ziehau IXGBE_READ_REG(hw, IXGBE_GRC) & ~(uint32_t)2); 511163d483cdSSepherosa Ziehau 511263d483cdSSepherosa Ziehau /* 511363d483cdSSepherosa Ziehau * Clear Wake Up Status register to prevent any previous wakeup 511463d483cdSSepherosa Ziehau * events from waking us up immediately after we suspend. 511563d483cdSSepherosa Ziehau */ 511663d483cdSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_WUS, 0xffffffff); 511763d483cdSSepherosa Ziehau 511863d483cdSSepherosa Ziehau /* 511963d483cdSSepherosa Ziehau * Program the Wakeup Filter Control register with user filter 512063d483cdSSepherosa Ziehau * settings 512163d483cdSSepherosa Ziehau */ 512263d483cdSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_WUFC, sc->wufc); 512363d483cdSSepherosa Ziehau 512463d483cdSSepherosa Ziehau /* Enable wakeups and power management in Wakeup Control */ 512563d483cdSSepherosa Ziehau IXGBE_WRITE_REG(hw, IXGBE_WUC, 512663d483cdSSepherosa Ziehau IXGBE_WUC_WKEN | IXGBE_WUC_PME_EN); 512763d483cdSSepherosa Ziehau 512863d483cdSSepherosa Ziehau /* X550EM baseT adapters need a special LPLU flow */ 512963d483cdSSepherosa Ziehau hw->phy.reset_disable = true; 513063d483cdSSepherosa Ziehau ix_stop(sc); 513163d483cdSSepherosa Ziehau error = hw->phy.ops.enter_lplu(hw); 513263d483cdSSepherosa Ziehau if (error) { 513363d483cdSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, 513463d483cdSSepherosa Ziehau "Error entering LPLU: %d\n", error); 513563d483cdSSepherosa Ziehau } 513663d483cdSSepherosa Ziehau hw->phy.reset_disable = false; 513763d483cdSSepherosa Ziehau } else { 513863d483cdSSepherosa Ziehau /* Just stop for other adapters */ 513963d483cdSSepherosa Ziehau ix_stop(sc); 514063d483cdSSepherosa Ziehau } 514163d483cdSSepherosa Ziehau return error; 514263d483cdSSepherosa Ziehau } 514363d483cdSSepherosa Ziehau 514463d483cdSSepherosa Ziehau static void 514563d483cdSSepherosa Ziehau ix_config_flowctrl(struct ix_softc *sc) 514663d483cdSSepherosa Ziehau { 514763d483cdSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 514863d483cdSSepherosa Ziehau uint32_t rxpb, frame, size, tmp; 514963d483cdSSepherosa Ziehau 515063d483cdSSepherosa Ziehau frame = sc->max_frame_size; 515163d483cdSSepherosa Ziehau 515263d483cdSSepherosa Ziehau /* Calculate High Water */ 515363d483cdSSepherosa Ziehau switch (hw->mac.type) { 515463d483cdSSepherosa Ziehau case ixgbe_mac_X540: 515563d483cdSSepherosa Ziehau case ixgbe_mac_X550: 515663d483cdSSepherosa Ziehau case ixgbe_mac_X550EM_a: 515763d483cdSSepherosa Ziehau case ixgbe_mac_X550EM_x: 515863d483cdSSepherosa Ziehau tmp = IXGBE_DV_X540(frame, frame); 515963d483cdSSepherosa Ziehau break; 516063d483cdSSepherosa Ziehau default: 516163d483cdSSepherosa Ziehau tmp = IXGBE_DV(frame, frame); 516263d483cdSSepherosa Ziehau break; 516363d483cdSSepherosa Ziehau } 516463d483cdSSepherosa Ziehau size = IXGBE_BT2KB(tmp); 516563d483cdSSepherosa Ziehau rxpb = IXGBE_READ_REG(hw, IXGBE_RXPBSIZE(0)) >> 10; 516663d483cdSSepherosa Ziehau hw->fc.high_water[0] = rxpb - size; 516763d483cdSSepherosa Ziehau 516863d483cdSSepherosa Ziehau /* Now calculate Low Water */ 516963d483cdSSepherosa Ziehau switch (hw->mac.type) { 517063d483cdSSepherosa Ziehau case ixgbe_mac_X540: 517163d483cdSSepherosa Ziehau case ixgbe_mac_X550: 517263d483cdSSepherosa Ziehau case ixgbe_mac_X550EM_a: 517363d483cdSSepherosa Ziehau case ixgbe_mac_X550EM_x: 517463d483cdSSepherosa Ziehau tmp = IXGBE_LOW_DV_X540(frame); 517563d483cdSSepherosa Ziehau break; 517663d483cdSSepherosa Ziehau default: 517763d483cdSSepherosa Ziehau tmp = IXGBE_LOW_DV(frame); 517863d483cdSSepherosa Ziehau break; 517963d483cdSSepherosa Ziehau } 518063d483cdSSepherosa Ziehau hw->fc.low_water[0] = IXGBE_BT2KB(tmp); 518163d483cdSSepherosa Ziehau 518263d483cdSSepherosa Ziehau hw->fc.requested_mode = ix_ifmedia2fc(sc->ifm_media); 518363d483cdSSepherosa Ziehau if (sc->ifm_media & IFM_ETH_FORCEPAUSE) 518463d483cdSSepherosa Ziehau hw->fc.disable_fc_autoneg = TRUE; 518563d483cdSSepherosa Ziehau else 518663d483cdSSepherosa Ziehau hw->fc.disable_fc_autoneg = FALSE; 518763d483cdSSepherosa Ziehau hw->fc.pause_time = IX_FC_PAUSE; 518863d483cdSSepherosa Ziehau hw->fc.send_xon = TRUE; 518963d483cdSSepherosa Ziehau } 519063d483cdSSepherosa Ziehau 519163d483cdSSepherosa Ziehau static void 519263d483cdSSepherosa Ziehau ix_config_dmac(struct ix_softc *sc) 519363d483cdSSepherosa Ziehau { 519463d483cdSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 519563d483cdSSepherosa Ziehau struct ixgbe_dmac_config *dcfg = &hw->mac.dmac_config; 519663d483cdSSepherosa Ziehau 519763d483cdSSepherosa Ziehau if (hw->mac.type < ixgbe_mac_X550 || !hw->mac.ops.dmac_config) 519863d483cdSSepherosa Ziehau return; 519963d483cdSSepherosa Ziehau 520063d483cdSSepherosa Ziehau if ((dcfg->watchdog_timer ^ sc->dmac) || 520163d483cdSSepherosa Ziehau (dcfg->link_speed ^ sc->link_speed)) { 520263d483cdSSepherosa Ziehau dcfg->watchdog_timer = sc->dmac; 520363d483cdSSepherosa Ziehau dcfg->fcoe_en = false; 520463d483cdSSepherosa Ziehau dcfg->link_speed = sc->link_speed; 520563d483cdSSepherosa Ziehau dcfg->num_tcs = 1; 520663d483cdSSepherosa Ziehau 520763d483cdSSepherosa Ziehau if (bootverbose) { 520863d483cdSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, "dmac settings: " 520963d483cdSSepherosa Ziehau "watchdog %d, link speed %d\n", 521063d483cdSSepherosa Ziehau dcfg->watchdog_timer, dcfg->link_speed); 521163d483cdSSepherosa Ziehau } 521263d483cdSSepherosa Ziehau 521363d483cdSSepherosa Ziehau hw->mac.ops.dmac_config(hw); 521463d483cdSSepherosa Ziehau } 521563d483cdSSepherosa Ziehau } 521663d483cdSSepherosa Ziehau 521763d483cdSSepherosa Ziehau static void 521863d483cdSSepherosa Ziehau ix_init_media(struct ix_softc *sc) 521963d483cdSSepherosa Ziehau { 522063d483cdSSepherosa Ziehau struct ixgbe_hw *hw = &sc->hw; 522163d483cdSSepherosa Ziehau int layer, msf_ifm = IFM_NONE; 522263d483cdSSepherosa Ziehau 522363d483cdSSepherosa Ziehau ifmedia_removeall(&sc->media); 522463d483cdSSepherosa Ziehau 522563d483cdSSepherosa Ziehau layer = ixgbe_get_supported_physical_layer(hw); 522663d483cdSSepherosa Ziehau 522763d483cdSSepherosa Ziehau /* 522863d483cdSSepherosa Ziehau * Media types with matching DragonFlyBSD media defines 522963d483cdSSepherosa Ziehau */ 523063d483cdSSepherosa Ziehau if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_T) { 523163d483cdSSepherosa Ziehau ifmedia_add_nodup(&sc->media, IFM_ETHER | IFM_10G_T | IFM_FDX, 523263d483cdSSepherosa Ziehau 0, NULL); 523363d483cdSSepherosa Ziehau } 523463d483cdSSepherosa Ziehau if (layer & IXGBE_PHYSICAL_LAYER_1000BASE_T) { 523563d483cdSSepherosa Ziehau ifmedia_add_nodup(&sc->media, IFM_ETHER | IFM_1000_T | IFM_FDX, 523663d483cdSSepherosa Ziehau 0, NULL); 523763d483cdSSepherosa Ziehau } 523863d483cdSSepherosa Ziehau if (layer & IXGBE_PHYSICAL_LAYER_100BASE_TX) { 523963d483cdSSepherosa Ziehau ifmedia_add_nodup(&sc->media, IFM_ETHER | IFM_100_TX | IFM_FDX, 524063d483cdSSepherosa Ziehau 0, NULL); 524163d483cdSSepherosa Ziehau /* No half-duplex support */ 524263d483cdSSepherosa Ziehau } 524363d483cdSSepherosa Ziehau 524463d483cdSSepherosa Ziehau if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_LR) { 524563d483cdSSepherosa Ziehau ifmedia_add_nodup(&sc->media, IFM_ETHER | IFM_10G_LR | IFM_FDX, 524663d483cdSSepherosa Ziehau 0, NULL); 524763d483cdSSepherosa Ziehau msf_ifm = IFM_1000_LX; 524863d483cdSSepherosa Ziehau } 524963d483cdSSepherosa Ziehau if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_LRM) { 525063d483cdSSepherosa Ziehau ifmedia_add_nodup(&sc->media, IFM_ETHER | IFM_10G_LRM | IFM_FDX, 525163d483cdSSepherosa Ziehau 0, NULL); 525263d483cdSSepherosa Ziehau msf_ifm = IFM_1000_LX; 525363d483cdSSepherosa Ziehau } 525463d483cdSSepherosa Ziehau if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_SR) { 525563d483cdSSepherosa Ziehau ifmedia_add_nodup(&sc->media, IFM_ETHER | IFM_10G_SR | IFM_FDX, 525663d483cdSSepherosa Ziehau 0, NULL); 525763d483cdSSepherosa Ziehau msf_ifm = IFM_1000_SX; 525863d483cdSSepherosa Ziehau } 525963d483cdSSepherosa Ziehau 526063d483cdSSepherosa Ziehau /* Add media for multispeed fiber */ 526163d483cdSSepherosa Ziehau if (ix_is_sfp(hw) && hw->phy.multispeed_fiber && msf_ifm != IFM_NONE) { 526263d483cdSSepherosa Ziehau uint32_t linkcap; 526363d483cdSSepherosa Ziehau bool autoneg; 526463d483cdSSepherosa Ziehau 526563d483cdSSepherosa Ziehau hw->mac.ops.get_link_capabilities(hw, &linkcap, &autoneg); 526663d483cdSSepherosa Ziehau if (linkcap & IXGBE_LINK_SPEED_1GB_FULL) 526763d483cdSSepherosa Ziehau ifmedia_add_nodup(&sc->media, 526863d483cdSSepherosa Ziehau IFM_ETHER | msf_ifm | IFM_FDX, 0, NULL); 526963d483cdSSepherosa Ziehau } 527063d483cdSSepherosa Ziehau 527163d483cdSSepherosa Ziehau if ((layer & IXGBE_PHYSICAL_LAYER_SFP_PLUS_CU) || 527263d483cdSSepherosa Ziehau (layer & IXGBE_PHYSICAL_LAYER_SFP_ACTIVE_DA)) { 527363d483cdSSepherosa Ziehau ifmedia_add_nodup(&sc->media, 527463d483cdSSepherosa Ziehau IFM_ETHER | IFM_10G_TWINAX | IFM_FDX, 0, NULL); 527563d483cdSSepherosa Ziehau } 527663d483cdSSepherosa Ziehau if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_CX4) { 527763d483cdSSepherosa Ziehau ifmedia_add_nodup(&sc->media, IFM_ETHER | IFM_10G_CX4 | IFM_FDX, 527863d483cdSSepherosa Ziehau 0, NULL); 527963d483cdSSepherosa Ziehau } 528063d483cdSSepherosa Ziehau if (layer & IXGBE_PHYSICAL_LAYER_1000BASE_SX) { 528163d483cdSSepherosa Ziehau ifmedia_add_nodup(&sc->media, IFM_ETHER | IFM_1000_SX | IFM_FDX, 528263d483cdSSepherosa Ziehau 0, NULL); 528363d483cdSSepherosa Ziehau } 528463d483cdSSepherosa Ziehau 528563d483cdSSepherosa Ziehau /* 528663d483cdSSepherosa Ziehau * XXX Other (no matching DragonFlyBSD media type): 528763d483cdSSepherosa Ziehau * To workaround this, we'll assign these completely 528863d483cdSSepherosa Ziehau * inappropriate media types. 528963d483cdSSepherosa Ziehau */ 529063d483cdSSepherosa Ziehau if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_KR) { 529163d483cdSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, "Media supported: 10GbaseKR\n"); 529263d483cdSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, "10GbaseKR mapped to 10GbaseSR\n"); 529363d483cdSSepherosa Ziehau ifmedia_add_nodup(&sc->media, IFM_ETHER | IFM_10G_SR | IFM_FDX, 529463d483cdSSepherosa Ziehau 0, NULL); 529563d483cdSSepherosa Ziehau } 529663d483cdSSepherosa Ziehau if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_KX4) { 529763d483cdSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, "Media supported: 10GbaseKX4\n"); 529863d483cdSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, 529963d483cdSSepherosa Ziehau "10GbaseKX4 mapped to 10GbaseCX4\n"); 530063d483cdSSepherosa Ziehau ifmedia_add_nodup(&sc->media, IFM_ETHER | IFM_10G_CX4 | IFM_FDX, 530163d483cdSSepherosa Ziehau 0, NULL); 530263d483cdSSepherosa Ziehau } 530363d483cdSSepherosa Ziehau if (layer & IXGBE_PHYSICAL_LAYER_1000BASE_KX) { 530463d483cdSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, "Media supported: 1000baseKX\n"); 530563d483cdSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, 530663d483cdSSepherosa Ziehau "1000baseKX mapped to 1000baseCX\n"); 530763d483cdSSepherosa Ziehau ifmedia_add_nodup(&sc->media, IFM_ETHER | IFM_1000_CX | IFM_FDX, 530863d483cdSSepherosa Ziehau 0, NULL); 530963d483cdSSepherosa Ziehau } 531063d483cdSSepherosa Ziehau if (layer & IXGBE_PHYSICAL_LAYER_1000BASE_BX) { 531163d483cdSSepherosa Ziehau /* Someday, someone will care about you... */ 531263d483cdSSepherosa Ziehau if_printf(&sc->arpcom.ac_if, 531363d483cdSSepherosa Ziehau "Media supported: 1000baseBX, ignored\n"); 531463d483cdSSepherosa Ziehau } 531563d483cdSSepherosa Ziehau 531663d483cdSSepherosa Ziehau /* XXX we probably don't need this */ 531763d483cdSSepherosa Ziehau if (hw->device_id == IXGBE_DEV_ID_82598AT) { 531863d483cdSSepherosa Ziehau ifmedia_add_nodup(&sc->media, 531963d483cdSSepherosa Ziehau IFM_ETHER | IFM_1000_T | IFM_FDX, 0, NULL); 532063d483cdSSepherosa Ziehau } 532163d483cdSSepherosa Ziehau 532263d483cdSSepherosa Ziehau ifmedia_add_nodup(&sc->media, IFM_ETHER | IFM_AUTO, 0, NULL); 532363d483cdSSepherosa Ziehau 532463d483cdSSepherosa Ziehau if (ifmedia_tryset(&sc->media, sc->ifm_media)) { 532563d483cdSSepherosa Ziehau int flowctrl = (sc->ifm_media & IFM_ETH_FCMASK); 532663d483cdSSepherosa Ziehau 532763d483cdSSepherosa Ziehau sc->advspeed = IXGBE_LINK_SPEED_UNKNOWN; 532863d483cdSSepherosa Ziehau sc->ifm_media = IX_IFM_DEFAULT | flowctrl; 532963d483cdSSepherosa Ziehau ifmedia_set(&sc->media, sc->ifm_media); 533063d483cdSSepherosa Ziehau } 533163d483cdSSepherosa Ziehau } 5332