1517904deSPeter Grehan /*- 2517904deSPeter Grehan * SPDX-License-Identifier: BSD-2-Clause 3517904deSPeter Grehan * 4bc9402abSKevin Bowling * Copyright (c) 2001-2024, Intel Corporation 5517904deSPeter Grehan * Copyright (c) 2016 Nicole Graziano <nicole@nextbsd.org> 6bc9402abSKevin Bowling * Copyright (c) 2021-2024 Rubicon Communications, LLC (Netgate) 7517904deSPeter Grehan * 8517904deSPeter Grehan * Redistribution and use in source and binary forms, with or without 9517904deSPeter Grehan * modification, are permitted provided that the following conditions 10517904deSPeter Grehan * are met: 11517904deSPeter Grehan * 1. Redistributions of source code must retain the above copyright 12517904deSPeter Grehan * notice, this list of conditions and the following disclaimer. 13517904deSPeter Grehan * 2. Redistributions in binary form must reproduce the above copyright 14517904deSPeter Grehan * notice, this list of conditions and the following disclaimer in the 15517904deSPeter Grehan * documentation and/or other materials provided with the distribution. 16517904deSPeter Grehan * 17517904deSPeter Grehan * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 18517904deSPeter Grehan * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 19517904deSPeter Grehan * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 20517904deSPeter Grehan * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 21517904deSPeter Grehan * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 22517904deSPeter Grehan * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 23517904deSPeter Grehan * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 24517904deSPeter Grehan * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 25517904deSPeter Grehan * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 26517904deSPeter Grehan * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 27517904deSPeter Grehan * SUCH DAMAGE. 28517904deSPeter Grehan */ 29517904deSPeter Grehan 30517904deSPeter Grehan #include <sys/cdefs.h> 31517904deSPeter Grehan #include "if_igc.h" 32517904deSPeter Grehan #include <sys/sbuf.h> 33517904deSPeter Grehan #include <machine/_inttypes.h> 34517904deSPeter Grehan 35517904deSPeter Grehan #ifdef RSS 36517904deSPeter Grehan #include <net/rss_config.h> 37517904deSPeter Grehan #include <netinet/in_rss.h> 38517904deSPeter Grehan #endif 39517904deSPeter Grehan 40517904deSPeter Grehan /********************************************************************* 41517904deSPeter Grehan * PCI Device ID Table 42517904deSPeter Grehan * 43517904deSPeter Grehan * Used by probe to select devices to load on 44517904deSPeter Grehan * Last entry must be all 0s 45517904deSPeter Grehan * 46517904deSPeter Grehan * { Vendor ID, Device ID, String } 47517904deSPeter Grehan *********************************************************************/ 48517904deSPeter Grehan 4951e23514SMarius Strobl static const pci_vendor_info_t igc_vendor_info_array[] = 50517904deSPeter Grehan { 51517904deSPeter Grehan /* Intel(R) PRO/1000 Network Connection - igc */ 529efc7325SKevin Bowling PVID(0x8086, IGC_DEV_ID_I225_LM, 539efc7325SKevin Bowling "Intel(R) Ethernet Controller I225-LM"), 549efc7325SKevin Bowling PVID(0x8086, IGC_DEV_ID_I225_V, 559efc7325SKevin Bowling "Intel(R) Ethernet Controller I225-V"), 569efc7325SKevin Bowling PVID(0x8086, IGC_DEV_ID_I225_K, 579efc7325SKevin Bowling "Intel(R) Ethernet Controller I225-K"), 589efc7325SKevin Bowling PVID(0x8086, IGC_DEV_ID_I225_I, 59*850f78d5SKevin Bowling "Intel(R) Ethernet Controller I225-IT"), 609efc7325SKevin Bowling PVID(0x8086, IGC_DEV_ID_I220_V, 619efc7325SKevin Bowling "Intel(R) Ethernet Controller I220-V"), 629efc7325SKevin Bowling PVID(0x8086, IGC_DEV_ID_I225_K2, 639efc7325SKevin Bowling "Intel(R) Ethernet Controller I225-K(2)"), 649efc7325SKevin Bowling PVID(0x8086, IGC_DEV_ID_I225_LMVP, 659efc7325SKevin Bowling "Intel(R) Ethernet Controller I225-LMvP(2)"), 669efc7325SKevin Bowling PVID(0x8086, IGC_DEV_ID_I226_K, 679efc7325SKevin Bowling "Intel(R) Ethernet Controller I226-K"), 689efc7325SKevin Bowling PVID(0x8086, IGC_DEV_ID_I226_LMVP, 699efc7325SKevin Bowling "Intel(R) Ethernet Controller I226-LMvP"), 709efc7325SKevin Bowling PVID(0x8086, IGC_DEV_ID_I225_IT, 719efc7325SKevin Bowling "Intel(R) Ethernet Controller I225-IT(2)"), 729efc7325SKevin Bowling PVID(0x8086, IGC_DEV_ID_I226_LM, 739efc7325SKevin Bowling "Intel(R) Ethernet Controller I226-LM"), 749efc7325SKevin Bowling PVID(0x8086, IGC_DEV_ID_I226_V, 759efc7325SKevin Bowling "Intel(R) Ethernet Controller I226-V"), 769efc7325SKevin Bowling PVID(0x8086, IGC_DEV_ID_I226_IT, 779efc7325SKevin Bowling "Intel(R) Ethernet Controller I226-IT"), 789efc7325SKevin Bowling PVID(0x8086, IGC_DEV_ID_I221_V, 799efc7325SKevin Bowling "Intel(R) Ethernet Controller I221-V"), 809efc7325SKevin Bowling PVID(0x8086, IGC_DEV_ID_I226_BLANK_NVM, 819efc7325SKevin Bowling "Intel(R) Ethernet Controller I226(blankNVM)"), 829efc7325SKevin Bowling PVID(0x8086, IGC_DEV_ID_I225_BLANK_NVM, 839efc7325SKevin Bowling "Intel(R) Ethernet Controller I225(blankNVM)"), 84517904deSPeter Grehan /* required last entry */ 85517904deSPeter Grehan PVID_END 86517904deSPeter Grehan }; 87517904deSPeter Grehan 88517904deSPeter Grehan /********************************************************************* 89517904deSPeter Grehan * Function prototypes 90517904deSPeter Grehan *********************************************************************/ 911b0e41ddSKevin Bowling static void *igc_register(device_t); 921b0e41ddSKevin Bowling static int igc_if_attach_pre(if_ctx_t); 931b0e41ddSKevin Bowling static int igc_if_attach_post(if_ctx_t); 941b0e41ddSKevin Bowling static int igc_if_detach(if_ctx_t); 951b0e41ddSKevin Bowling static int igc_if_shutdown(if_ctx_t); 961b0e41ddSKevin Bowling static int igc_if_suspend(if_ctx_t); 971b0e41ddSKevin Bowling static int igc_if_resume(if_ctx_t); 98517904deSPeter Grehan 999efc7325SKevin Bowling static int igc_if_tx_queues_alloc(if_ctx_t, caddr_t *, uint64_t *, int, 1009efc7325SKevin Bowling int); 1019efc7325SKevin Bowling static int igc_if_rx_queues_alloc(if_ctx_t, caddr_t *, uint64_t *, int, 1029efc7325SKevin Bowling int); 1031b0e41ddSKevin Bowling static void igc_if_queues_free(if_ctx_t); 104517904deSPeter Grehan 105517904deSPeter Grehan static uint64_t igc_if_get_counter(if_ctx_t, ift_counter); 1061b0e41ddSKevin Bowling static void igc_if_init(if_ctx_t); 1071b0e41ddSKevin Bowling static void igc_if_stop(if_ctx_t); 108517904deSPeter Grehan static void igc_if_media_status(if_ctx_t, struct ifmediareq *); 1091b0e41ddSKevin Bowling static int igc_if_media_change(if_ctx_t); 1101b0e41ddSKevin Bowling static int igc_if_mtu_set(if_ctx_t, uint32_t); 1111b0e41ddSKevin Bowling static void igc_if_timer(if_ctx_t, uint16_t); 1121b0e41ddSKevin Bowling static void igc_if_watchdog_reset(if_ctx_t); 1131b0e41ddSKevin Bowling static bool igc_if_needs_restart(if_ctx_t, enum iflib_restart_event); 114517904deSPeter Grehan 1151b0e41ddSKevin Bowling static void igc_identify_hardware(if_ctx_t); 1161b0e41ddSKevin Bowling static int igc_allocate_pci_resources(if_ctx_t); 1171b0e41ddSKevin Bowling static void igc_free_pci_resources(if_ctx_t); 1181b0e41ddSKevin Bowling static void igc_reset(if_ctx_t); 1191b0e41ddSKevin Bowling static int igc_setup_interface(if_ctx_t); 1201b0e41ddSKevin Bowling static int igc_setup_msix(if_ctx_t); 121517904deSPeter Grehan 1221b0e41ddSKevin Bowling static void igc_initialize_transmit_unit(if_ctx_t); 1231b0e41ddSKevin Bowling static void igc_initialize_receive_unit(if_ctx_t); 124517904deSPeter Grehan 1251b0e41ddSKevin Bowling static void igc_if_intr_enable(if_ctx_t); 1261b0e41ddSKevin Bowling static void igc_if_intr_disable(if_ctx_t); 1271b0e41ddSKevin Bowling static int igc_if_rx_queue_intr_enable(if_ctx_t, uint16_t); 1281b0e41ddSKevin Bowling static int igc_if_tx_queue_intr_enable(if_ctx_t, uint16_t); 1291b0e41ddSKevin Bowling static void igc_if_multi_set(if_ctx_t); 1301b0e41ddSKevin Bowling static void igc_if_update_admin_status(if_ctx_t); 1311b0e41ddSKevin Bowling static void igc_if_debug(if_ctx_t); 132542f5d56SKevin Bowling static void igc_update_stats_counters(struct igc_softc *); 1331b0e41ddSKevin Bowling static void igc_add_hw_stats(struct igc_softc *); 1341b0e41ddSKevin Bowling static int igc_if_set_promisc(if_ctx_t, int); 1351b0e41ddSKevin Bowling static void igc_setup_vlan_hw_support(if_ctx_t); 136542f5d56SKevin Bowling static void igc_fw_version(struct igc_softc *); 13733ed9bdcSKevin Bowling static void igc_sbuf_fw_version(struct igc_fw_version *, struct sbuf *); 138542f5d56SKevin Bowling static void igc_print_fw_version(struct igc_softc *); 13933ed9bdcSKevin Bowling static int igc_sysctl_print_fw_version(SYSCTL_HANDLER_ARGS); 140517904deSPeter Grehan static int igc_sysctl_nvm_info(SYSCTL_HANDLER_ARGS); 141542f5d56SKevin Bowling static void igc_print_nvm_info(struct igc_softc *); 142517904deSPeter Grehan static int igc_sysctl_debug_info(SYSCTL_HANDLER_ARGS); 143517904deSPeter Grehan static int igc_get_rs(SYSCTL_HANDLER_ARGS); 144542f5d56SKevin Bowling static void igc_print_debug_info(struct igc_softc *); 145517904deSPeter Grehan static int igc_is_valid_ether_addr(u8 *); 146542f5d56SKevin Bowling static void igc_neweitr(struct igc_softc *, struct igc_rx_queue *, 147bc9402abSKevin Bowling struct tx_ring *, struct rx_ring *); 148ab540d44SKevin Bowling static int igc_sysctl_tso_tcp_flags_mask(SYSCTL_HANDLER_ARGS); 149517904deSPeter Grehan /* Management and WOL Support */ 150542f5d56SKevin Bowling static void igc_get_hw_control(struct igc_softc *); 151542f5d56SKevin Bowling static void igc_release_hw_control(struct igc_softc *); 1521b0e41ddSKevin Bowling static void igc_get_wakeup(if_ctx_t); 1531b0e41ddSKevin Bowling static void igc_enable_wakeup(if_ctx_t); 154517904deSPeter Grehan 1551b0e41ddSKevin Bowling int igc_intr(void *); 156517904deSPeter Grehan 157517904deSPeter Grehan /* MSI-X handlers */ 158517904deSPeter Grehan static int igc_if_msix_intr_assign(if_ctx_t, int); 159517904deSPeter Grehan static int igc_msix_link(void *); 160517904deSPeter Grehan static void igc_handle_link(void *context); 161517904deSPeter Grehan 162517904deSPeter Grehan static int igc_set_flowcntl(SYSCTL_HANDLER_ARGS); 16368b1f5dcSKevin Bowling static int igc_sysctl_dmac(SYSCTL_HANDLER_ARGS); 164517904deSPeter Grehan static int igc_sysctl_eee(SYSCTL_HANDLER_ARGS); 165517904deSPeter Grehan 166517904deSPeter Grehan static int igc_get_regs(SYSCTL_HANDLER_ARGS); 167517904deSPeter Grehan 1681b0e41ddSKevin Bowling static void igc_configure_queues(struct igc_softc *); 169517904deSPeter Grehan 170517904deSPeter Grehan 171517904deSPeter Grehan /********************************************************************* 172517904deSPeter Grehan * FreeBSD Device Interface Entry Points 173517904deSPeter Grehan *********************************************************************/ 174517904deSPeter Grehan static device_method_t igc_methods[] = { 175517904deSPeter Grehan /* Device interface */ 176517904deSPeter Grehan DEVMETHOD(device_register, igc_register), 177517904deSPeter Grehan DEVMETHOD(device_probe, iflib_device_probe), 178517904deSPeter Grehan DEVMETHOD(device_attach, iflib_device_attach), 179517904deSPeter Grehan DEVMETHOD(device_detach, iflib_device_detach), 180517904deSPeter Grehan DEVMETHOD(device_shutdown, iflib_device_shutdown), 181517904deSPeter Grehan DEVMETHOD(device_suspend, iflib_device_suspend), 182517904deSPeter Grehan DEVMETHOD(device_resume, iflib_device_resume), 183517904deSPeter Grehan DEVMETHOD_END 184517904deSPeter Grehan }; 185517904deSPeter Grehan 186517904deSPeter Grehan static driver_t igc_driver = { 187542f5d56SKevin Bowling "igc", igc_methods, sizeof(struct igc_softc), 188517904deSPeter Grehan }; 189517904deSPeter Grehan 19005a86e7cSJohn Baldwin DRIVER_MODULE(igc, pci, igc_driver, 0, 0); 191517904deSPeter Grehan 192517904deSPeter Grehan MODULE_DEPEND(igc, pci, 1, 1, 1); 193517904deSPeter Grehan MODULE_DEPEND(igc, ether, 1, 1, 1); 194517904deSPeter Grehan MODULE_DEPEND(igc, iflib, 1, 1, 1); 195517904deSPeter Grehan 196517904deSPeter Grehan IFLIB_PNP_INFO(pci, igc, igc_vendor_info_array); 197517904deSPeter Grehan 198517904deSPeter Grehan static device_method_t igc_if_methods[] = { 199517904deSPeter Grehan DEVMETHOD(ifdi_attach_pre, igc_if_attach_pre), 200517904deSPeter Grehan DEVMETHOD(ifdi_attach_post, igc_if_attach_post), 201517904deSPeter Grehan DEVMETHOD(ifdi_detach, igc_if_detach), 202517904deSPeter Grehan DEVMETHOD(ifdi_shutdown, igc_if_shutdown), 203517904deSPeter Grehan DEVMETHOD(ifdi_suspend, igc_if_suspend), 204517904deSPeter Grehan DEVMETHOD(ifdi_resume, igc_if_resume), 205517904deSPeter Grehan DEVMETHOD(ifdi_init, igc_if_init), 206517904deSPeter Grehan DEVMETHOD(ifdi_stop, igc_if_stop), 207517904deSPeter Grehan DEVMETHOD(ifdi_msix_intr_assign, igc_if_msix_intr_assign), 208517904deSPeter Grehan DEVMETHOD(ifdi_intr_enable, igc_if_intr_enable), 209517904deSPeter Grehan DEVMETHOD(ifdi_intr_disable, igc_if_intr_disable), 210517904deSPeter Grehan DEVMETHOD(ifdi_tx_queues_alloc, igc_if_tx_queues_alloc), 211517904deSPeter Grehan DEVMETHOD(ifdi_rx_queues_alloc, igc_if_rx_queues_alloc), 212517904deSPeter Grehan DEVMETHOD(ifdi_queues_free, igc_if_queues_free), 213517904deSPeter Grehan DEVMETHOD(ifdi_update_admin_status, igc_if_update_admin_status), 214517904deSPeter Grehan DEVMETHOD(ifdi_multi_set, igc_if_multi_set), 215517904deSPeter Grehan DEVMETHOD(ifdi_media_status, igc_if_media_status), 216517904deSPeter Grehan DEVMETHOD(ifdi_media_change, igc_if_media_change), 217517904deSPeter Grehan DEVMETHOD(ifdi_mtu_set, igc_if_mtu_set), 218517904deSPeter Grehan DEVMETHOD(ifdi_promisc_set, igc_if_set_promisc), 219517904deSPeter Grehan DEVMETHOD(ifdi_timer, igc_if_timer), 220517904deSPeter Grehan DEVMETHOD(ifdi_watchdog_reset, igc_if_watchdog_reset), 221517904deSPeter Grehan DEVMETHOD(ifdi_get_counter, igc_if_get_counter), 222517904deSPeter Grehan DEVMETHOD(ifdi_rx_queue_intr_enable, igc_if_rx_queue_intr_enable), 223517904deSPeter Grehan DEVMETHOD(ifdi_tx_queue_intr_enable, igc_if_tx_queue_intr_enable), 224517904deSPeter Grehan DEVMETHOD(ifdi_debug, igc_if_debug), 225517904deSPeter Grehan DEVMETHOD(ifdi_needs_restart, igc_if_needs_restart), 226517904deSPeter Grehan DEVMETHOD_END 227517904deSPeter Grehan }; 228517904deSPeter Grehan 229517904deSPeter Grehan static driver_t igc_if_driver = { 230542f5d56SKevin Bowling "igc_if", igc_if_methods, sizeof(struct igc_softc) 231517904deSPeter Grehan }; 232517904deSPeter Grehan 233517904deSPeter Grehan /********************************************************************* 234517904deSPeter Grehan * Tunable default values. 235517904deSPeter Grehan *********************************************************************/ 236517904deSPeter Grehan 237517904deSPeter Grehan /* Allow common code without TSO */ 238517904deSPeter Grehan #ifndef CSUM_TSO 239517904deSPeter Grehan #define CSUM_TSO 0 240517904deSPeter Grehan #endif 241517904deSPeter Grehan 242517904deSPeter Grehan static SYSCTL_NODE(_hw, OID_AUTO, igc, CTLFLAG_RD | CTLFLAG_MPSAFE, 0, 243517904deSPeter Grehan "igc driver parameters"); 244517904deSPeter Grehan 245517904deSPeter Grehan static int igc_disable_crc_stripping = 0; 246517904deSPeter Grehan SYSCTL_INT(_hw_igc, OID_AUTO, disable_crc_stripping, CTLFLAG_RDTUN, 247517904deSPeter Grehan &igc_disable_crc_stripping, 0, "Disable CRC Stripping"); 248517904deSPeter Grehan 249517904deSPeter Grehan static int igc_smart_pwr_down = false; 2509efc7325SKevin Bowling SYSCTL_INT(_hw_igc, OID_AUTO, smart_pwr_down, CTLFLAG_RDTUN, 2519efc7325SKevin Bowling &igc_smart_pwr_down, 252517904deSPeter Grehan 0, "Set to true to leave smart power down enabled on newer adapters"); 253517904deSPeter Grehan 254517904deSPeter Grehan /* Controls whether promiscuous also shows bad packets */ 255e80419daSKevin Bowling static int igc_debug_sbp = false; 256517904deSPeter Grehan SYSCTL_INT(_hw_igc, OID_AUTO, sbp, CTLFLAG_RDTUN, &igc_debug_sbp, 0, 257517904deSPeter Grehan "Show bad packets in promiscuous mode"); 258517904deSPeter Grehan 259517904deSPeter Grehan /* Energy efficient ethernet - default to OFF */ 260517904deSPeter Grehan static int igc_eee_setting = 1; 261517904deSPeter Grehan SYSCTL_INT(_hw_igc, OID_AUTO, eee_setting, CTLFLAG_RDTUN, &igc_eee_setting, 0, 262517904deSPeter Grehan "Enable Energy Efficient Ethernet"); 263517904deSPeter Grehan 264517904deSPeter Grehan /* 265bc9402abSKevin Bowling * AIM: Adaptive Interrupt Moderation 266bc9402abSKevin Bowling * which means that the interrupt rate is varied over time based on the 267bc9402abSKevin Bowling * traffic for that interrupt vector 268bc9402abSKevin Bowling */ 269bc9402abSKevin Bowling static int igc_enable_aim = 1; 270bc9402abSKevin Bowling SYSCTL_INT(_hw_igc, OID_AUTO, enable_aim, CTLFLAG_RWTUN, &igc_enable_aim, 271bc9402abSKevin Bowling 0, "Enable adaptive interrupt moderation (1=normal, 2=lowlatency)"); 272bc9402abSKevin Bowling 273bc9402abSKevin Bowling /* 274517904deSPeter Grehan ** Tuneable Interrupt rate 275517904deSPeter Grehan */ 276bc9402abSKevin Bowling static int igc_max_interrupt_rate = IGC_INTS_DEFAULT; 277517904deSPeter Grehan SYSCTL_INT(_hw_igc, OID_AUTO, max_interrupt_rate, CTLFLAG_RDTUN, 278517904deSPeter Grehan &igc_max_interrupt_rate, 0, "Maximum interrupts per second"); 279517904deSPeter Grehan 280517904deSPeter Grehan extern struct if_txrx igc_txrx; 281517904deSPeter Grehan 282517904deSPeter Grehan static struct if_shared_ctx igc_sctx_init = { 283517904deSPeter Grehan .isc_magic = IFLIB_MAGIC, 284517904deSPeter Grehan .isc_q_align = PAGE_SIZE, 285517904deSPeter Grehan .isc_tx_maxsize = IGC_TSO_SIZE + sizeof(struct ether_vlan_header), 286517904deSPeter Grehan .isc_tx_maxsegsize = PAGE_SIZE, 287517904deSPeter Grehan .isc_tso_maxsize = IGC_TSO_SIZE + sizeof(struct ether_vlan_header), 288517904deSPeter Grehan .isc_tso_maxsegsize = IGC_TSO_SEG_SIZE, 289517904deSPeter Grehan .isc_rx_maxsize = MAX_JUMBO_FRAME_SIZE, 290517904deSPeter Grehan .isc_rx_nsegments = 1, 291517904deSPeter Grehan .isc_rx_maxsegsize = MJUM9BYTES, 292517904deSPeter Grehan .isc_nfl = 1, 293517904deSPeter Grehan .isc_nrxqs = 1, 294517904deSPeter Grehan .isc_ntxqs = 1, 295517904deSPeter Grehan .isc_admin_intrcnt = 1, 296517904deSPeter Grehan .isc_vendor_info = igc_vendor_info_array, 297517904deSPeter Grehan .isc_driver_version = "1", 298517904deSPeter Grehan .isc_driver = &igc_if_driver, 2999efc7325SKevin Bowling .isc_flags = 3009efc7325SKevin Bowling IFLIB_NEED_SCRATCH | IFLIB_TSO_INIT_IP | IFLIB_NEED_ZERO_CSUM, 301517904deSPeter Grehan 302517904deSPeter Grehan .isc_nrxd_min = {IGC_MIN_RXD}, 303517904deSPeter Grehan .isc_ntxd_min = {IGC_MIN_TXD}, 304517904deSPeter Grehan .isc_nrxd_max = {IGC_MAX_RXD}, 305517904deSPeter Grehan .isc_ntxd_max = {IGC_MAX_TXD}, 306517904deSPeter Grehan .isc_nrxd_default = {IGC_DEFAULT_RXD}, 307517904deSPeter Grehan .isc_ntxd_default = {IGC_DEFAULT_TXD}, 308517904deSPeter Grehan }; 309517904deSPeter Grehan 310517904deSPeter Grehan /***************************************************************** 311517904deSPeter Grehan * 312517904deSPeter Grehan * Dump Registers 313517904deSPeter Grehan * 314517904deSPeter Grehan ****************************************************************/ 315517904deSPeter Grehan #define IGC_REGS_LEN 739 316517904deSPeter Grehan 317517904deSPeter Grehan static int igc_get_regs(SYSCTL_HANDLER_ARGS) 318517904deSPeter Grehan { 319542f5d56SKevin Bowling struct igc_softc *sc = (struct igc_softc *)arg1; 320542f5d56SKevin Bowling struct igc_hw *hw = &sc->hw; 321517904deSPeter Grehan struct sbuf *sb; 322517904deSPeter Grehan u32 *regs_buff; 323517904deSPeter Grehan int rc; 324517904deSPeter Grehan 325517904deSPeter Grehan regs_buff = malloc(sizeof(u32) * IGC_REGS_LEN, M_DEVBUF, M_WAITOK); 326517904deSPeter Grehan memset(regs_buff, 0, IGC_REGS_LEN * sizeof(u32)); 327517904deSPeter Grehan 328517904deSPeter Grehan rc = sysctl_wire_old_buffer(req, 0); 329517904deSPeter Grehan MPASS(rc == 0); 330517904deSPeter Grehan if (rc != 0) { 331517904deSPeter Grehan free(regs_buff, M_DEVBUF); 332517904deSPeter Grehan return (rc); 333517904deSPeter Grehan } 334517904deSPeter Grehan 335517904deSPeter Grehan sb = sbuf_new_for_sysctl(NULL, NULL, 32*400, req); 336517904deSPeter Grehan MPASS(sb != NULL); 337517904deSPeter Grehan if (sb == NULL) { 338517904deSPeter Grehan free(regs_buff, M_DEVBUF); 339517904deSPeter Grehan return (ENOMEM); 340517904deSPeter Grehan } 341517904deSPeter Grehan 342517904deSPeter Grehan /* General Registers */ 343517904deSPeter Grehan regs_buff[0] = IGC_READ_REG(hw, IGC_CTRL); 344517904deSPeter Grehan regs_buff[1] = IGC_READ_REG(hw, IGC_STATUS); 345517904deSPeter Grehan regs_buff[2] = IGC_READ_REG(hw, IGC_CTRL_EXT); 346517904deSPeter Grehan regs_buff[3] = IGC_READ_REG(hw, IGC_ICR); 347517904deSPeter Grehan regs_buff[4] = IGC_READ_REG(hw, IGC_RCTL); 348517904deSPeter Grehan regs_buff[5] = IGC_READ_REG(hw, IGC_RDLEN(0)); 349517904deSPeter Grehan regs_buff[6] = IGC_READ_REG(hw, IGC_RDH(0)); 350517904deSPeter Grehan regs_buff[7] = IGC_READ_REG(hw, IGC_RDT(0)); 351517904deSPeter Grehan regs_buff[8] = IGC_READ_REG(hw, IGC_RXDCTL(0)); 352517904deSPeter Grehan regs_buff[9] = IGC_READ_REG(hw, IGC_RDBAL(0)); 353517904deSPeter Grehan regs_buff[10] = IGC_READ_REG(hw, IGC_RDBAH(0)); 354517904deSPeter Grehan regs_buff[11] = IGC_READ_REG(hw, IGC_TCTL); 355517904deSPeter Grehan regs_buff[12] = IGC_READ_REG(hw, IGC_TDBAL(0)); 356517904deSPeter Grehan regs_buff[13] = IGC_READ_REG(hw, IGC_TDBAH(0)); 357517904deSPeter Grehan regs_buff[14] = IGC_READ_REG(hw, IGC_TDLEN(0)); 358517904deSPeter Grehan regs_buff[15] = IGC_READ_REG(hw, IGC_TDH(0)); 359517904deSPeter Grehan regs_buff[16] = IGC_READ_REG(hw, IGC_TDT(0)); 360517904deSPeter Grehan regs_buff[17] = IGC_READ_REG(hw, IGC_TXDCTL(0)); 361517904deSPeter Grehan 362517904deSPeter Grehan sbuf_printf(sb, "General Registers\n"); 363517904deSPeter Grehan sbuf_printf(sb, "\tCTRL\t %08x\n", regs_buff[0]); 364517904deSPeter Grehan sbuf_printf(sb, "\tSTATUS\t %08x\n", regs_buff[1]); 365517904deSPeter Grehan sbuf_printf(sb, "\tCTRL_EXIT\t %08x\n\n", regs_buff[2]); 366517904deSPeter Grehan 367517904deSPeter Grehan sbuf_printf(sb, "Interrupt Registers\n"); 368517904deSPeter Grehan sbuf_printf(sb, "\tICR\t %08x\n\n", regs_buff[3]); 369517904deSPeter Grehan 370517904deSPeter Grehan sbuf_printf(sb, "RX Registers\n"); 371517904deSPeter Grehan sbuf_printf(sb, "\tRCTL\t %08x\n", regs_buff[4]); 372517904deSPeter Grehan sbuf_printf(sb, "\tRDLEN\t %08x\n", regs_buff[5]); 373517904deSPeter Grehan sbuf_printf(sb, "\tRDH\t %08x\n", regs_buff[6]); 374517904deSPeter Grehan sbuf_printf(sb, "\tRDT\t %08x\n", regs_buff[7]); 375517904deSPeter Grehan sbuf_printf(sb, "\tRXDCTL\t %08x\n", regs_buff[8]); 376517904deSPeter Grehan sbuf_printf(sb, "\tRDBAL\t %08x\n", regs_buff[9]); 377517904deSPeter Grehan sbuf_printf(sb, "\tRDBAH\t %08x\n\n", regs_buff[10]); 378517904deSPeter Grehan 379517904deSPeter Grehan sbuf_printf(sb, "TX Registers\n"); 380517904deSPeter Grehan sbuf_printf(sb, "\tTCTL\t %08x\n", regs_buff[11]); 381517904deSPeter Grehan sbuf_printf(sb, "\tTDBAL\t %08x\n", regs_buff[12]); 382517904deSPeter Grehan sbuf_printf(sb, "\tTDBAH\t %08x\n", regs_buff[13]); 383517904deSPeter Grehan sbuf_printf(sb, "\tTDLEN\t %08x\n", regs_buff[14]); 384517904deSPeter Grehan sbuf_printf(sb, "\tTDH\t %08x\n", regs_buff[15]); 385517904deSPeter Grehan sbuf_printf(sb, "\tTDT\t %08x\n", regs_buff[16]); 386517904deSPeter Grehan sbuf_printf(sb, "\tTXDCTL\t %08x\n", regs_buff[17]); 387517904deSPeter Grehan sbuf_printf(sb, "\tTDFH\t %08x\n", regs_buff[18]); 388517904deSPeter Grehan sbuf_printf(sb, "\tTDFT\t %08x\n", regs_buff[19]); 389517904deSPeter Grehan sbuf_printf(sb, "\tTDFHS\t %08x\n", regs_buff[20]); 390517904deSPeter Grehan sbuf_printf(sb, "\tTDFPC\t %08x\n\n", regs_buff[21]); 391517904deSPeter Grehan 392517904deSPeter Grehan free(regs_buff, M_DEVBUF); 393517904deSPeter Grehan 394517904deSPeter Grehan #ifdef DUMP_DESCS 395517904deSPeter Grehan { 396542f5d56SKevin Bowling if_softc_ctx_t scctx = sc->shared; 397517904deSPeter Grehan struct rx_ring *rxr = &rx_que->rxr; 398517904deSPeter Grehan struct tx_ring *txr = &tx_que->txr; 399517904deSPeter Grehan int ntxd = scctx->isc_ntxd[0]; 400517904deSPeter Grehan int nrxd = scctx->isc_nrxd[0]; 401517904deSPeter Grehan int j; 402517904deSPeter Grehan 403517904deSPeter Grehan for (j = 0; j < nrxd; j++) { 404517904deSPeter Grehan u32 staterr = le32toh(rxr->rx_base[j].wb.upper.status_error); 405517904deSPeter Grehan u32 length = le32toh(rxr->rx_base[j].wb.upper.length); 4069efc7325SKevin Bowling sbuf_printf(sb, "\tReceive Descriptor Address %d: %08" 4079efc7325SKevin Bowling PRIx64 " Error:%d Length:%d\n", 4089efc7325SKevin Bowling j, rxr->rx_base[j].read.buffer_addr, staterr, length); 409517904deSPeter Grehan } 410517904deSPeter Grehan 411517904deSPeter Grehan for (j = 0; j < min(ntxd, 256); j++) { 412517904deSPeter Grehan unsigned int *ptr = (unsigned int *)&txr->tx_base[j]; 413517904deSPeter Grehan 4149efc7325SKevin Bowling sbuf_printf(sb, "\tTXD[%03d] [0]: %08x [1]: %08x [2]: %08x" 4159efc7325SKevin Bowling "[3]: %08x eop: %d DD=%d\n", 416517904deSPeter Grehan j, ptr[0], ptr[1], ptr[2], ptr[3], buf->eop, 4179efc7325SKevin Bowling buf->eop != -1 ? 4189efc7325SKevin Bowling txr->tx_base[buf->eop].upper.fields.status & 4199efc7325SKevin Bowling IGC_TXD_STAT_DD : 0); 420517904deSPeter Grehan 421517904deSPeter Grehan } 422517904deSPeter Grehan } 423517904deSPeter Grehan #endif 424517904deSPeter Grehan 425517904deSPeter Grehan rc = sbuf_finish(sb); 426517904deSPeter Grehan sbuf_delete(sb); 427517904deSPeter Grehan return(rc); 428517904deSPeter Grehan } 429517904deSPeter Grehan 430517904deSPeter Grehan static void * 431517904deSPeter Grehan igc_register(device_t dev) 432517904deSPeter Grehan { 433517904deSPeter Grehan return (&igc_sctx_init); 434517904deSPeter Grehan } 435517904deSPeter Grehan 436517904deSPeter Grehan static int 437517904deSPeter Grehan igc_set_num_queues(if_ctx_t ctx) 438517904deSPeter Grehan { 439517904deSPeter Grehan int maxqueues; 440517904deSPeter Grehan 441517904deSPeter Grehan maxqueues = 4; 442517904deSPeter Grehan 443517904deSPeter Grehan return (maxqueues); 444517904deSPeter Grehan } 445517904deSPeter Grehan 446517904deSPeter Grehan #define IGC_CAPS \ 447517904deSPeter Grehan IFCAP_HWCSUM | IFCAP_VLAN_MTU | IFCAP_VLAN_HWTAGGING | \ 4482eaef8ecSKevin Bowling IFCAP_VLAN_HWCSUM | IFCAP_WOL | IFCAP_TSO4 | IFCAP_LRO | \ 4492eaef8ecSKevin Bowling IFCAP_VLAN_HWTSO | IFCAP_JUMBO_MTU | IFCAP_HWCSUM_IPV6 | IFCAP_TSO6 450517904deSPeter Grehan 451517904deSPeter Grehan /********************************************************************* 452517904deSPeter Grehan * Device initialization routine 453517904deSPeter Grehan * 454517904deSPeter Grehan * The attach entry point is called when the driver is being loaded. 455517904deSPeter Grehan * This routine identifies the type of hardware, allocates all resources 456517904deSPeter Grehan * and initializes the hardware. 457517904deSPeter Grehan * 458517904deSPeter Grehan * return 0 on success, positive on failure 459517904deSPeter Grehan *********************************************************************/ 460517904deSPeter Grehan static int 461517904deSPeter Grehan igc_if_attach_pre(if_ctx_t ctx) 462517904deSPeter Grehan { 463542f5d56SKevin Bowling struct igc_softc *sc; 464517904deSPeter Grehan if_softc_ctx_t scctx; 465517904deSPeter Grehan device_t dev; 466517904deSPeter Grehan struct igc_hw *hw; 467517904deSPeter Grehan int error = 0; 468517904deSPeter Grehan 469517904deSPeter Grehan INIT_DEBUGOUT("igc_if_attach_pre: begin"); 470517904deSPeter Grehan dev = iflib_get_dev(ctx); 471542f5d56SKevin Bowling sc = iflib_get_softc(ctx); 472517904deSPeter Grehan 473542f5d56SKevin Bowling sc->ctx = sc->osdep.ctx = ctx; 474542f5d56SKevin Bowling sc->dev = sc->osdep.dev = dev; 475542f5d56SKevin Bowling scctx = sc->shared = iflib_get_softc_ctx(ctx); 476542f5d56SKevin Bowling sc->media = iflib_get_media(ctx); 477542f5d56SKevin Bowling hw = &sc->hw; 478517904deSPeter Grehan 479517904deSPeter Grehan /* SYSCTL stuff */ 480517904deSPeter Grehan SYSCTL_ADD_PROC(device_get_sysctl_ctx(dev), 481517904deSPeter Grehan SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 482517904deSPeter Grehan OID_AUTO, "nvm", CTLTYPE_INT | CTLFLAG_RW | CTLFLAG_NEEDGIANT, 483542f5d56SKevin Bowling sc, 0, igc_sysctl_nvm_info, "I", "NVM Information"); 484517904deSPeter Grehan 485542f5d56SKevin Bowling sc->enable_aim = igc_enable_aim; 486bc9402abSKevin Bowling SYSCTL_ADD_INT(device_get_sysctl_ctx(dev), 487bc9402abSKevin Bowling SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 488bc9402abSKevin Bowling OID_AUTO, "enable_aim", CTLFLAG_RW, 489542f5d56SKevin Bowling &sc->enable_aim, 0, 490bc9402abSKevin Bowling "Interrupt Moderation (1=normal, 2=lowlatency)"); 491bc9402abSKevin Bowling 492517904deSPeter Grehan SYSCTL_ADD_PROC(device_get_sysctl_ctx(dev), 493517904deSPeter Grehan SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 49433ed9bdcSKevin Bowling OID_AUTO, "fw_version", CTLTYPE_STRING | CTLFLAG_RD, 495542f5d56SKevin Bowling sc, 0, igc_sysctl_print_fw_version, "A", 49633ed9bdcSKevin Bowling "Prints FW/NVM Versions"); 49733ed9bdcSKevin Bowling 49833ed9bdcSKevin Bowling SYSCTL_ADD_PROC(device_get_sysctl_ctx(dev), 49933ed9bdcSKevin Bowling SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 500517904deSPeter Grehan OID_AUTO, "debug", CTLTYPE_INT | CTLFLAG_RW | CTLFLAG_NEEDGIANT, 501542f5d56SKevin Bowling sc, 0, igc_sysctl_debug_info, "I", "Debug Information"); 502517904deSPeter Grehan 503517904deSPeter Grehan SYSCTL_ADD_PROC(device_get_sysctl_ctx(dev), 504517904deSPeter Grehan SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 505517904deSPeter Grehan OID_AUTO, "fc", CTLTYPE_INT | CTLFLAG_RW | CTLFLAG_NEEDGIANT, 506542f5d56SKevin Bowling sc, 0, igc_set_flowcntl, "I", "Flow Control"); 507517904deSPeter Grehan 508517904deSPeter Grehan SYSCTL_ADD_PROC(device_get_sysctl_ctx(dev), 509517904deSPeter Grehan SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 510517904deSPeter Grehan OID_AUTO, "reg_dump", 511542f5d56SKevin Bowling CTLTYPE_STRING | CTLFLAG_RD | CTLFLAG_NEEDGIANT, sc, 0, 512517904deSPeter Grehan igc_get_regs, "A", "Dump Registers"); 513517904deSPeter Grehan 514517904deSPeter Grehan SYSCTL_ADD_PROC(device_get_sysctl_ctx(dev), 515517904deSPeter Grehan SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 516517904deSPeter Grehan OID_AUTO, "rs_dump", 517542f5d56SKevin Bowling CTLTYPE_INT | CTLFLAG_RW | CTLFLAG_NEEDGIANT, sc, 0, 518517904deSPeter Grehan igc_get_rs, "I", "Dump RS indexes"); 519517904deSPeter Grehan 52068b1f5dcSKevin Bowling SYSCTL_ADD_PROC(device_get_sysctl_ctx(dev), 52168b1f5dcSKevin Bowling SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 52268b1f5dcSKevin Bowling OID_AUTO, "dmac", 523542f5d56SKevin Bowling CTLTYPE_INT | CTLFLAG_RW, sc, 0, 52468b1f5dcSKevin Bowling igc_sysctl_dmac, "I", "DMA Coalesce"); 52568b1f5dcSKevin Bowling 526ab540d44SKevin Bowling SYSCTL_ADD_PROC(device_get_sysctl_ctx(dev), 527ab540d44SKevin Bowling SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 528ab540d44SKevin Bowling OID_AUTO, "tso_tcp_flags_mask_first_segment", 529ab540d44SKevin Bowling CTLTYPE_UINT | CTLFLAG_RW | CTLFLAG_NEEDGIANT, 530ab540d44SKevin Bowling sc, 0, igc_sysctl_tso_tcp_flags_mask, "IU", 531ab540d44SKevin Bowling "TSO TCP flags mask for first segment"); 532ab540d44SKevin Bowling 533ab540d44SKevin Bowling SYSCTL_ADD_PROC(device_get_sysctl_ctx(dev), 534ab540d44SKevin Bowling SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 535ab540d44SKevin Bowling OID_AUTO, "tso_tcp_flags_mask_middle_segment", 536ab540d44SKevin Bowling CTLTYPE_UINT | CTLFLAG_RW | CTLFLAG_NEEDGIANT, 537ab540d44SKevin Bowling sc, 1, igc_sysctl_tso_tcp_flags_mask, "IU", 538ab540d44SKevin Bowling "TSO TCP flags mask for middle segment"); 539ab540d44SKevin Bowling 540ab540d44SKevin Bowling SYSCTL_ADD_PROC(device_get_sysctl_ctx(dev), 541ab540d44SKevin Bowling SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 542ab540d44SKevin Bowling OID_AUTO, "tso_tcp_flags_mask_last_segment", 543ab540d44SKevin Bowling CTLTYPE_UINT | CTLFLAG_RW | CTLFLAG_NEEDGIANT, 544ab540d44SKevin Bowling sc, 2, igc_sysctl_tso_tcp_flags_mask, "IU", 545ab540d44SKevin Bowling "TSO TCP flags mask for last segment"); 546ab540d44SKevin Bowling 547517904deSPeter Grehan /* Determine hardware and mac info */ 548517904deSPeter Grehan igc_identify_hardware(ctx); 549517904deSPeter Grehan 550517904deSPeter Grehan scctx->isc_tx_nsegments = IGC_MAX_SCATTER; 5519efc7325SKevin Bowling scctx->isc_nrxqsets_max = 5529efc7325SKevin Bowling scctx->isc_ntxqsets_max = igc_set_num_queues(ctx); 553517904deSPeter Grehan if (bootverbose) 554517904deSPeter Grehan device_printf(dev, "attach_pre capping queues at %d\n", 555517904deSPeter Grehan scctx->isc_ntxqsets_max); 556517904deSPeter Grehan 5579efc7325SKevin Bowling scctx->isc_txqsizes[0] = roundup2(scctx->isc_ntxd[0] * 5589efc7325SKevin Bowling sizeof(union igc_adv_tx_desc), IGC_DBA_ALIGN); 5599efc7325SKevin Bowling scctx->isc_rxqsizes[0] = roundup2(scctx->isc_nrxd[0] * 5609efc7325SKevin Bowling sizeof(union igc_adv_rx_desc), IGC_DBA_ALIGN); 561517904deSPeter Grehan scctx->isc_txd_size[0] = sizeof(union igc_adv_tx_desc); 562517904deSPeter Grehan scctx->isc_rxd_size[0] = sizeof(union igc_adv_rx_desc); 563517904deSPeter Grehan scctx->isc_txrx = &igc_txrx; 564517904deSPeter Grehan scctx->isc_tx_tso_segments_max = IGC_MAX_SCATTER; 565517904deSPeter Grehan scctx->isc_tx_tso_size_max = IGC_TSO_SIZE; 566517904deSPeter Grehan scctx->isc_tx_tso_segsize_max = IGC_TSO_SEG_SIZE; 567517904deSPeter Grehan scctx->isc_capabilities = scctx->isc_capenable = IGC_CAPS; 568517904deSPeter Grehan scctx->isc_tx_csum_flags = CSUM_TCP | CSUM_UDP | CSUM_TSO | 569517904deSPeter Grehan CSUM_IP6_TCP | CSUM_IP6_UDP | CSUM_SCTP | CSUM_IP6_SCTP; 570517904deSPeter Grehan 571517904deSPeter Grehan /* 572517904deSPeter Grehan ** Some new devices, as with ixgbe, now may 573517904deSPeter Grehan ** use a different BAR, so we need to keep 574517904deSPeter Grehan ** track of which is used. 575517904deSPeter Grehan */ 576517904deSPeter Grehan scctx->isc_msix_bar = PCIR_BAR(IGC_MSIX_BAR); 577517904deSPeter Grehan if (pci_read_config(dev, scctx->isc_msix_bar, 4) == 0) 578517904deSPeter Grehan scctx->isc_msix_bar += 4; 579517904deSPeter Grehan 580517904deSPeter Grehan /* Setup PCI resources */ 581517904deSPeter Grehan if (igc_allocate_pci_resources(ctx)) { 582517904deSPeter Grehan device_printf(dev, "Allocation of PCI resources failed\n"); 583517904deSPeter Grehan error = ENXIO; 584517904deSPeter Grehan goto err_pci; 585517904deSPeter Grehan } 586517904deSPeter Grehan 587517904deSPeter Grehan /* Do Shared Code initialization */ 588517904deSPeter Grehan error = igc_setup_init_funcs(hw, true); 589517904deSPeter Grehan if (error) { 590517904deSPeter Grehan device_printf(dev, "Setup of Shared code failed, error %d\n", 591517904deSPeter Grehan error); 592517904deSPeter Grehan error = ENXIO; 593517904deSPeter Grehan goto err_pci; 594517904deSPeter Grehan } 595517904deSPeter Grehan 596517904deSPeter Grehan igc_setup_msix(ctx); 597517904deSPeter Grehan igc_get_bus_info(hw); 598517904deSPeter Grehan 599517904deSPeter Grehan hw->mac.autoneg = DO_AUTO_NEG; 600517904deSPeter Grehan hw->phy.autoneg_wait_to_complete = false; 601517904deSPeter Grehan hw->phy.autoneg_advertised = AUTONEG_ADV_DEFAULT; 602517904deSPeter Grehan 603517904deSPeter Grehan /* Copper options */ 604517904deSPeter Grehan if (hw->phy.media_type == igc_media_type_copper) { 605517904deSPeter Grehan hw->phy.mdix = AUTO_ALL_MODES; 606517904deSPeter Grehan } 607517904deSPeter Grehan 608517904deSPeter Grehan /* 609517904deSPeter Grehan * Set the frame limits assuming 610517904deSPeter Grehan * standard ethernet sized frames. 611517904deSPeter Grehan */ 612542f5d56SKevin Bowling scctx->isc_max_frame_size = sc->hw.mac.max_frame_size = 613517904deSPeter Grehan ETHERMTU + ETHER_HDR_LEN + ETHERNET_FCS_SIZE; 614517904deSPeter Grehan 615517904deSPeter Grehan /* Allocate multicast array memory. */ 616542f5d56SKevin Bowling sc->mta = malloc(sizeof(u8) * ETHER_ADDR_LEN * 617517904deSPeter Grehan MAX_NUM_MULTICAST_ADDRESSES, M_DEVBUF, M_NOWAIT); 618542f5d56SKevin Bowling if (sc->mta == NULL) { 6199efc7325SKevin Bowling device_printf(dev, 6209efc7325SKevin Bowling "Can not allocate multicast setup array\n"); 621517904deSPeter Grehan error = ENOMEM; 622517904deSPeter Grehan goto err_late; 623517904deSPeter Grehan } 624517904deSPeter Grehan 625517904deSPeter Grehan /* Check SOL/IDER usage */ 626517904deSPeter Grehan if (igc_check_reset_block(hw)) 627517904deSPeter Grehan device_printf(dev, "PHY reset is blocked" 628517904deSPeter Grehan " due to SOL/IDER session.\n"); 629517904deSPeter Grehan 630517904deSPeter Grehan /* Sysctl for setting Energy Efficient Ethernet */ 631542f5d56SKevin Bowling sc->hw.dev_spec._i225.eee_disable = igc_eee_setting; 632517904deSPeter Grehan SYSCTL_ADD_PROC(device_get_sysctl_ctx(dev), 633517904deSPeter Grehan SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 634517904deSPeter Grehan OID_AUTO, "eee_control", 635517904deSPeter Grehan CTLTYPE_INT | CTLFLAG_RW | CTLFLAG_NEEDGIANT, 636542f5d56SKevin Bowling sc, 0, igc_sysctl_eee, "I", 637517904deSPeter Grehan "Disable Energy Efficient Ethernet"); 638517904deSPeter Grehan 639517904deSPeter Grehan /* 640517904deSPeter Grehan ** Start from a known state, this is 641517904deSPeter Grehan ** important in reading the nvm and 642517904deSPeter Grehan ** mac from that. 643517904deSPeter Grehan */ 644517904deSPeter Grehan igc_reset_hw(hw); 645517904deSPeter Grehan 646517904deSPeter Grehan /* Make sure we have a good EEPROM before we read from it */ 647517904deSPeter Grehan if (igc_validate_nvm_checksum(hw) < 0) { 648517904deSPeter Grehan /* 649517904deSPeter Grehan ** Some PCI-E parts fail the first check due to 650517904deSPeter Grehan ** the link being in sleep state, call it again, 651517904deSPeter Grehan ** if it fails a second time its a real issue. 652517904deSPeter Grehan */ 653517904deSPeter Grehan if (igc_validate_nvm_checksum(hw) < 0) { 654517904deSPeter Grehan device_printf(dev, 655517904deSPeter Grehan "The EEPROM Checksum Is Not Valid\n"); 656517904deSPeter Grehan error = EIO; 657517904deSPeter Grehan goto err_late; 658517904deSPeter Grehan } 659517904deSPeter Grehan } 660517904deSPeter Grehan 661517904deSPeter Grehan /* Copy the permanent MAC address out of the EEPROM */ 662517904deSPeter Grehan if (igc_read_mac_addr(hw) < 0) { 663517904deSPeter Grehan device_printf(dev, "EEPROM read error while reading MAC" 664517904deSPeter Grehan " address\n"); 665517904deSPeter Grehan error = EIO; 666517904deSPeter Grehan goto err_late; 667517904deSPeter Grehan } 668517904deSPeter Grehan 669517904deSPeter Grehan if (!igc_is_valid_ether_addr(hw->mac.addr)) { 670517904deSPeter Grehan device_printf(dev, "Invalid MAC address\n"); 671517904deSPeter Grehan error = EIO; 672517904deSPeter Grehan goto err_late; 673517904deSPeter Grehan } 674517904deSPeter Grehan 67533ed9bdcSKevin Bowling /* Save the EEPROM/NVM versions */ 676542f5d56SKevin Bowling igc_fw_version(sc); 67733ed9bdcSKevin Bowling 678542f5d56SKevin Bowling igc_print_fw_version(sc); 67933ed9bdcSKevin Bowling 680517904deSPeter Grehan /* 681517904deSPeter Grehan * Get Wake-on-Lan and Management info for later use 682517904deSPeter Grehan */ 683517904deSPeter Grehan igc_get_wakeup(ctx); 684517904deSPeter Grehan 685517904deSPeter Grehan /* Enable only WOL MAGIC by default */ 686517904deSPeter Grehan scctx->isc_capenable &= ~IFCAP_WOL; 687542f5d56SKevin Bowling if (sc->wol != 0) 688517904deSPeter Grehan scctx->isc_capenable |= IFCAP_WOL_MAGIC; 689517904deSPeter Grehan 690517904deSPeter Grehan iflib_set_mac(ctx, hw->mac.addr); 691517904deSPeter Grehan 692517904deSPeter Grehan return (0); 693517904deSPeter Grehan 694517904deSPeter Grehan err_late: 695542f5d56SKevin Bowling igc_release_hw_control(sc); 696517904deSPeter Grehan err_pci: 697517904deSPeter Grehan igc_free_pci_resources(ctx); 698542f5d56SKevin Bowling free(sc->mta, M_DEVBUF); 699517904deSPeter Grehan 700517904deSPeter Grehan return (error); 701517904deSPeter Grehan } 702517904deSPeter Grehan 703517904deSPeter Grehan static int 704517904deSPeter Grehan igc_if_attach_post(if_ctx_t ctx) 705517904deSPeter Grehan { 706542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 707542f5d56SKevin Bowling struct igc_hw *hw = &sc->hw; 708517904deSPeter Grehan int error = 0; 709517904deSPeter Grehan 710517904deSPeter Grehan /* Setup OS specific network interface */ 711517904deSPeter Grehan error = igc_setup_interface(ctx); 712517904deSPeter Grehan if (error != 0) { 713517904deSPeter Grehan goto err_late; 714517904deSPeter Grehan } 715517904deSPeter Grehan 716517904deSPeter Grehan igc_reset(ctx); 717517904deSPeter Grehan 718517904deSPeter Grehan /* Initialize statistics */ 719542f5d56SKevin Bowling igc_update_stats_counters(sc); 720517904deSPeter Grehan hw->mac.get_link_status = true; 721517904deSPeter Grehan igc_if_update_admin_status(ctx); 722542f5d56SKevin Bowling igc_add_hw_stats(sc); 723517904deSPeter Grehan 724517904deSPeter Grehan /* the driver can now take control from firmware */ 725542f5d56SKevin Bowling igc_get_hw_control(sc); 726517904deSPeter Grehan 727517904deSPeter Grehan INIT_DEBUGOUT("igc_if_attach_post: end"); 728517904deSPeter Grehan 729517904deSPeter Grehan return (error); 730517904deSPeter Grehan 731517904deSPeter Grehan err_late: 732542f5d56SKevin Bowling igc_release_hw_control(sc); 733517904deSPeter Grehan igc_free_pci_resources(ctx); 734517904deSPeter Grehan igc_if_queues_free(ctx); 735542f5d56SKevin Bowling free(sc->mta, M_DEVBUF); 736517904deSPeter Grehan 737517904deSPeter Grehan return (error); 738517904deSPeter Grehan } 739517904deSPeter Grehan 740517904deSPeter Grehan /********************************************************************* 741517904deSPeter Grehan * Device removal routine 742517904deSPeter Grehan * 743517904deSPeter Grehan * The detach entry point is called when the driver is being removed. 744517904deSPeter Grehan * This routine stops the adapter and deallocates all the resources 745517904deSPeter Grehan * that were allocated for driver operation. 746517904deSPeter Grehan * 747517904deSPeter Grehan * return 0 on success, positive on failure 748517904deSPeter Grehan *********************************************************************/ 749517904deSPeter Grehan static int 750517904deSPeter Grehan igc_if_detach(if_ctx_t ctx) 751517904deSPeter Grehan { 752542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 753517904deSPeter Grehan 754517904deSPeter Grehan INIT_DEBUGOUT("igc_if_detach: begin"); 755517904deSPeter Grehan 756542f5d56SKevin Bowling igc_phy_hw_reset(&sc->hw); 757517904deSPeter Grehan 758542f5d56SKevin Bowling igc_release_hw_control(sc); 759517904deSPeter Grehan igc_free_pci_resources(ctx); 760517904deSPeter Grehan 761517904deSPeter Grehan return (0); 762517904deSPeter Grehan } 763517904deSPeter Grehan 764517904deSPeter Grehan /********************************************************************* 765517904deSPeter Grehan * 766517904deSPeter Grehan * Shutdown entry point 767517904deSPeter Grehan * 768517904deSPeter Grehan **********************************************************************/ 769517904deSPeter Grehan 770517904deSPeter Grehan static int 771517904deSPeter Grehan igc_if_shutdown(if_ctx_t ctx) 772517904deSPeter Grehan { 773517904deSPeter Grehan return igc_if_suspend(ctx); 774517904deSPeter Grehan } 775517904deSPeter Grehan 776517904deSPeter Grehan /* 777517904deSPeter Grehan * Suspend/resume device methods. 778517904deSPeter Grehan */ 779517904deSPeter Grehan static int 780517904deSPeter Grehan igc_if_suspend(if_ctx_t ctx) 781517904deSPeter Grehan { 782542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 783517904deSPeter Grehan 784542f5d56SKevin Bowling igc_release_hw_control(sc); 785517904deSPeter Grehan igc_enable_wakeup(ctx); 786517904deSPeter Grehan return (0); 787517904deSPeter Grehan } 788517904deSPeter Grehan 789517904deSPeter Grehan static int 790517904deSPeter Grehan igc_if_resume(if_ctx_t ctx) 791517904deSPeter Grehan { 792517904deSPeter Grehan igc_if_init(ctx); 793517904deSPeter Grehan 794517904deSPeter Grehan return(0); 795517904deSPeter Grehan } 796517904deSPeter Grehan 797517904deSPeter Grehan static int 798517904deSPeter Grehan igc_if_mtu_set(if_ctx_t ctx, uint32_t mtu) 799517904deSPeter Grehan { 800517904deSPeter Grehan int max_frame_size; 801542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 802517904deSPeter Grehan if_softc_ctx_t scctx = iflib_get_softc_ctx(ctx); 803517904deSPeter Grehan 804517904deSPeter Grehan IOCTL_DEBUGOUT("ioctl rcv'd: SIOCSIFMTU (Set Interface MTU)"); 805517904deSPeter Grehan 806517904deSPeter Grehan /* 9K Jumbo Frame size */ 807517904deSPeter Grehan max_frame_size = 9234; 808517904deSPeter Grehan 809517904deSPeter Grehan if (mtu > max_frame_size - ETHER_HDR_LEN - ETHER_CRC_LEN) { 810517904deSPeter Grehan return (EINVAL); 811517904deSPeter Grehan } 812517904deSPeter Grehan 813542f5d56SKevin Bowling scctx->isc_max_frame_size = sc->hw.mac.max_frame_size = 814517904deSPeter Grehan mtu + ETHER_HDR_LEN + ETHER_CRC_LEN; 815517904deSPeter Grehan return (0); 816517904deSPeter Grehan } 817517904deSPeter Grehan 818517904deSPeter Grehan /********************************************************************* 819517904deSPeter Grehan * Init entry point 820517904deSPeter Grehan * 821517904deSPeter Grehan * This routine is used in two ways. It is used by the stack as 822517904deSPeter Grehan * init entry point in network interface structure. It is also used 823517904deSPeter Grehan * by the driver as a hw/sw initialization routine to get to a 824517904deSPeter Grehan * consistent state. 825517904deSPeter Grehan * 826517904deSPeter Grehan **********************************************************************/ 827517904deSPeter Grehan static void 828517904deSPeter Grehan igc_if_init(if_ctx_t ctx) 829517904deSPeter Grehan { 830542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 831542f5d56SKevin Bowling if_softc_ctx_t scctx = sc->shared; 832ec22a3a2SJustin Hibbits if_t ifp = iflib_get_ifp(ctx); 833517904deSPeter Grehan struct igc_tx_queue *tx_que; 834517904deSPeter Grehan int i; 835517904deSPeter Grehan 836517904deSPeter Grehan INIT_DEBUGOUT("igc_if_init: begin"); 837517904deSPeter Grehan 838517904deSPeter Grehan /* Get the latest mac address, User can use a LAA */ 839542f5d56SKevin Bowling bcopy(if_getlladdr(ifp), sc->hw.mac.addr, 840517904deSPeter Grehan ETHER_ADDR_LEN); 841517904deSPeter Grehan 842517904deSPeter Grehan /* Put the address into the Receive Address Array */ 843542f5d56SKevin Bowling igc_rar_set(&sc->hw, sc->hw.mac.addr, 0); 844517904deSPeter Grehan 845517904deSPeter Grehan /* Initialize the hardware */ 846517904deSPeter Grehan igc_reset(ctx); 847517904deSPeter Grehan igc_if_update_admin_status(ctx); 848517904deSPeter Grehan 8499efc7325SKevin Bowling for (i = 0, tx_que = sc->tx_queues; i < sc->tx_num_queues; 8509efc7325SKevin Bowling i++, tx_que++) { 851517904deSPeter Grehan struct tx_ring *txr = &tx_que->txr; 852517904deSPeter Grehan 853517904deSPeter Grehan txr->tx_rs_cidx = txr->tx_rs_pidx; 854517904deSPeter Grehan 855517904deSPeter Grehan /* Initialize the last processed descriptor to be the end of 856517904deSPeter Grehan * the ring, rather than the start, so that we avoid an 857517904deSPeter Grehan * off-by-one error when calculating how many descriptors are 858517904deSPeter Grehan * done in the credits_update function. 859517904deSPeter Grehan */ 860517904deSPeter Grehan txr->tx_cidx_processed = scctx->isc_ntxd[0] - 1; 861517904deSPeter Grehan } 862517904deSPeter Grehan 863517904deSPeter Grehan /* Setup VLAN support, basic and offload if available */ 864542f5d56SKevin Bowling IGC_WRITE_REG(&sc->hw, IGC_VET, ETHERTYPE_VLAN); 865517904deSPeter Grehan 866517904deSPeter Grehan /* Prepare transmit descriptors and buffers */ 867517904deSPeter Grehan igc_initialize_transmit_unit(ctx); 868517904deSPeter Grehan 869517904deSPeter Grehan /* Setup Multicast table */ 870517904deSPeter Grehan igc_if_multi_set(ctx); 871517904deSPeter Grehan 872542f5d56SKevin Bowling sc->rx_mbuf_sz = iflib_get_rx_mbuf_sz(ctx); 873517904deSPeter Grehan igc_initialize_receive_unit(ctx); 874517904deSPeter Grehan 8752eaef8ecSKevin Bowling /* Set up VLAN support */ 8762eaef8ecSKevin Bowling igc_setup_vlan_hw_support(ctx); 877517904deSPeter Grehan 878517904deSPeter Grehan /* Don't lose promiscuous settings */ 87955760984SHubert Mazur igc_if_set_promisc(ctx, if_getflags(ifp)); 880542f5d56SKevin Bowling igc_clear_hw_cntrs_base_generic(&sc->hw); 881517904deSPeter Grehan 882542f5d56SKevin Bowling if (sc->intr_type == IFLIB_INTR_MSIX) /* Set up queue routing */ 883542f5d56SKevin Bowling igc_configure_queues(sc); 884517904deSPeter Grehan 885517904deSPeter Grehan /* this clears any pending interrupts */ 886542f5d56SKevin Bowling IGC_READ_REG(&sc->hw, IGC_ICR); 887542f5d56SKevin Bowling IGC_WRITE_REG(&sc->hw, IGC_ICS, IGC_ICS_LSC); 888517904deSPeter Grehan 889517904deSPeter Grehan /* the driver can now take control from firmware */ 890542f5d56SKevin Bowling igc_get_hw_control(sc); 891517904deSPeter Grehan 892517904deSPeter Grehan /* Set Energy Efficient Ethernet */ 893542f5d56SKevin Bowling igc_set_eee_i225(&sc->hw, true, true, true); 894517904deSPeter Grehan } 895517904deSPeter Grehan 896bc9402abSKevin Bowling enum eitr_latency_target { 897bc9402abSKevin Bowling eitr_latency_disabled = 0, 898bc9402abSKevin Bowling eitr_latency_lowest = 1, 899bc9402abSKevin Bowling eitr_latency_low = 2, 900bc9402abSKevin Bowling eitr_latency_bulk = 3 901bc9402abSKevin Bowling }; 902bc9402abSKevin Bowling /********************************************************************* 903bc9402abSKevin Bowling * 904bc9402abSKevin Bowling * Helper to calculate next EITR value for AIM 905bc9402abSKevin Bowling * 906bc9402abSKevin Bowling *********************************************************************/ 907bc9402abSKevin Bowling static void 908542f5d56SKevin Bowling igc_neweitr(struct igc_softc *sc, struct igc_rx_queue *que, 909bc9402abSKevin Bowling struct tx_ring *txr, struct rx_ring *rxr) 910bc9402abSKevin Bowling { 911bc9402abSKevin Bowling struct igc_hw *hw = &sc->hw; 912bc9402abSKevin Bowling u32 neweitr; 913bc9402abSKevin Bowling u32 bytes; 914bc9402abSKevin Bowling u32 bytes_packets; 915bc9402abSKevin Bowling u32 packets; 916bc9402abSKevin Bowling u8 nextlatency; 917bc9402abSKevin Bowling 918bc9402abSKevin Bowling /* Idle, do nothing */ 919bc9402abSKevin Bowling if ((txr->tx_bytes == 0) && (rxr->rx_bytes == 0)) 920bc9402abSKevin Bowling return; 921bc9402abSKevin Bowling 922bc9402abSKevin Bowling neweitr = 0; 923bc9402abSKevin Bowling 924bc9402abSKevin Bowling if (sc->enable_aim) { 925bc9402abSKevin Bowling nextlatency = rxr->rx_nextlatency; 926bc9402abSKevin Bowling 927bc9402abSKevin Bowling /* Use half default (4K) ITR if sub-gig */ 928669d26e5SKevin Bowling if (sc->link_speed < 1000) { 929bc9402abSKevin Bowling neweitr = IGC_INTS_4K; 930bc9402abSKevin Bowling goto igc_set_next_eitr; 931bc9402abSKevin Bowling } 932bc9402abSKevin Bowling /* Want at least enough packet buffer for two frames to AIM */ 933bc9402abSKevin Bowling if (sc->shared->isc_max_frame_size * 2 > (sc->pba << 10)) { 934bc9402abSKevin Bowling neweitr = igc_max_interrupt_rate; 935bc9402abSKevin Bowling sc->enable_aim = 0; 936bc9402abSKevin Bowling goto igc_set_next_eitr; 937bc9402abSKevin Bowling } 938bc9402abSKevin Bowling 9399efc7325SKevin Bowling /* Get largest values from the associated tx and rx ring */ 940bc9402abSKevin Bowling if (txr->tx_bytes && txr->tx_packets) { 941bc9402abSKevin Bowling bytes = txr->tx_bytes; 942bc9402abSKevin Bowling bytes_packets = txr->tx_bytes/txr->tx_packets; 943bc9402abSKevin Bowling packets = txr->tx_packets; 944bc9402abSKevin Bowling } 945bc9402abSKevin Bowling if (rxr->rx_bytes && rxr->rx_packets) { 946bc9402abSKevin Bowling bytes = max(bytes, rxr->rx_bytes); 9479efc7325SKevin Bowling bytes_packets = max(bytes_packets, 9489efc7325SKevin Bowling rxr->rx_bytes/rxr->rx_packets); 949bc9402abSKevin Bowling packets = max(packets, rxr->rx_packets); 950bc9402abSKevin Bowling } 951bc9402abSKevin Bowling 952bc9402abSKevin Bowling /* Latency state machine */ 953bc9402abSKevin Bowling switch (nextlatency) { 954bc9402abSKevin Bowling case eitr_latency_disabled: /* Bootstrapping */ 955bc9402abSKevin Bowling nextlatency = eitr_latency_low; 956bc9402abSKevin Bowling break; 957bc9402abSKevin Bowling case eitr_latency_lowest: /* 70k ints/s */ 958bc9402abSKevin Bowling /* TSO and jumbo frames */ 959bc9402abSKevin Bowling if (bytes_packets > 8000) 960bc9402abSKevin Bowling nextlatency = eitr_latency_bulk; 961bc9402abSKevin Bowling else if ((packets < 5) && (bytes > 512)) 962bc9402abSKevin Bowling nextlatency = eitr_latency_low; 963bc9402abSKevin Bowling break; 964bc9402abSKevin Bowling case eitr_latency_low: /* 20k ints/s */ 965bc9402abSKevin Bowling if (bytes > 10000) { 966bc9402abSKevin Bowling /* Handle TSO */ 967bc9402abSKevin Bowling if (bytes_packets > 8000) 968bc9402abSKevin Bowling nextlatency = eitr_latency_bulk; 9699efc7325SKevin Bowling else if ((packets < 10) || 9709efc7325SKevin Bowling (bytes_packets > 1200)) 971bc9402abSKevin Bowling nextlatency = eitr_latency_bulk; 972bc9402abSKevin Bowling else if (packets > 35) 973bc9402abSKevin Bowling nextlatency = eitr_latency_lowest; 974bc9402abSKevin Bowling } else if (bytes_packets > 2000) { 975bc9402abSKevin Bowling nextlatency = eitr_latency_bulk; 976bc9402abSKevin Bowling } else if (packets < 3 && bytes < 512) { 977bc9402abSKevin Bowling nextlatency = eitr_latency_lowest; 978bc9402abSKevin Bowling } 979bc9402abSKevin Bowling break; 980bc9402abSKevin Bowling case eitr_latency_bulk: /* 4k ints/s */ 981bc9402abSKevin Bowling if (bytes > 25000) { 982bc9402abSKevin Bowling if (packets > 35) 983bc9402abSKevin Bowling nextlatency = eitr_latency_low; 984bc9402abSKevin Bowling } else if (bytes < 1500) 985bc9402abSKevin Bowling nextlatency = eitr_latency_low; 986bc9402abSKevin Bowling break; 987bc9402abSKevin Bowling default: 988bc9402abSKevin Bowling nextlatency = eitr_latency_low; 9899efc7325SKevin Bowling device_printf(sc->dev, 9909efc7325SKevin Bowling "Unexpected neweitr transition %d\n", 991bc9402abSKevin Bowling nextlatency); 992bc9402abSKevin Bowling break; 993bc9402abSKevin Bowling } 994bc9402abSKevin Bowling 995bc9402abSKevin Bowling /* Trim itr_latency_lowest for default AIM setting */ 996bc9402abSKevin Bowling if (sc->enable_aim == 1 && nextlatency == eitr_latency_lowest) 997bc9402abSKevin Bowling nextlatency = eitr_latency_low; 998bc9402abSKevin Bowling 999bc9402abSKevin Bowling /* Request new latency */ 1000bc9402abSKevin Bowling rxr->rx_nextlatency = nextlatency; 1001bc9402abSKevin Bowling } else { 1002bc9402abSKevin Bowling /* We may have toggled to AIM disabled */ 1003bc9402abSKevin Bowling nextlatency = eitr_latency_disabled; 1004bc9402abSKevin Bowling rxr->rx_nextlatency = nextlatency; 1005bc9402abSKevin Bowling } 1006bc9402abSKevin Bowling 1007bc9402abSKevin Bowling /* ITR state machine */ 1008bc9402abSKevin Bowling switch(nextlatency) { 1009bc9402abSKevin Bowling case eitr_latency_lowest: 1010bc9402abSKevin Bowling neweitr = IGC_INTS_70K; 1011bc9402abSKevin Bowling break; 1012bc9402abSKevin Bowling case eitr_latency_low: 1013bc9402abSKevin Bowling neweitr = IGC_INTS_20K; 1014bc9402abSKevin Bowling break; 1015bc9402abSKevin Bowling case eitr_latency_bulk: 1016bc9402abSKevin Bowling neweitr = IGC_INTS_4K; 1017bc9402abSKevin Bowling break; 1018bc9402abSKevin Bowling case eitr_latency_disabled: 1019bc9402abSKevin Bowling default: 1020bc9402abSKevin Bowling neweitr = igc_max_interrupt_rate; 1021bc9402abSKevin Bowling break; 1022bc9402abSKevin Bowling } 1023bc9402abSKevin Bowling 1024bc9402abSKevin Bowling igc_set_next_eitr: 1025bc9402abSKevin Bowling neweitr = IGC_INTS_TO_EITR(neweitr); 1026bc9402abSKevin Bowling 1027bc9402abSKevin Bowling neweitr |= IGC_EITR_CNT_IGNR; 1028bc9402abSKevin Bowling 1029bc9402abSKevin Bowling if (neweitr != que->eitr_setting) { 1030bc9402abSKevin Bowling que->eitr_setting = neweitr; 1031bc9402abSKevin Bowling IGC_WRITE_REG(hw, IGC_EITR(que->msix), que->eitr_setting); 1032bc9402abSKevin Bowling } 1033bc9402abSKevin Bowling } 1034bc9402abSKevin Bowling 1035517904deSPeter Grehan /********************************************************************* 1036517904deSPeter Grehan * 1037517904deSPeter Grehan * Fast Legacy/MSI Combined Interrupt Service routine 1038517904deSPeter Grehan * 1039517904deSPeter Grehan *********************************************************************/ 1040517904deSPeter Grehan int 1041517904deSPeter Grehan igc_intr(void *arg) 1042517904deSPeter Grehan { 1043542f5d56SKevin Bowling struct igc_softc *sc = arg; 1044542f5d56SKevin Bowling struct igc_hw *hw = &sc->hw; 1045542f5d56SKevin Bowling struct igc_rx_queue *que = &sc->rx_queues[0]; 1046542f5d56SKevin Bowling struct tx_ring *txr = &sc->tx_queues[0].txr; 1047bc9402abSKevin Bowling struct rx_ring *rxr = &que->rxr; 1048542f5d56SKevin Bowling if_ctx_t ctx = sc->ctx; 1049517904deSPeter Grehan u32 reg_icr; 1050517904deSPeter Grehan 1051bc9402abSKevin Bowling reg_icr = IGC_READ_REG(hw, IGC_ICR); 1052517904deSPeter Grehan 1053517904deSPeter Grehan /* Hot eject? */ 1054517904deSPeter Grehan if (reg_icr == 0xffffffff) 1055517904deSPeter Grehan return FILTER_STRAY; 1056517904deSPeter Grehan 1057517904deSPeter Grehan /* Definitely not our interrupt. */ 1058517904deSPeter Grehan if (reg_icr == 0x0) 1059517904deSPeter Grehan return FILTER_STRAY; 1060517904deSPeter Grehan 1061517904deSPeter Grehan if ((reg_icr & IGC_ICR_INT_ASSERTED) == 0) 1062517904deSPeter Grehan return FILTER_STRAY; 1063517904deSPeter Grehan 1064517904deSPeter Grehan /* 1065517904deSPeter Grehan * Only MSI-X interrupts have one-shot behavior by taking advantage 1066517904deSPeter Grehan * of the EIAC register. Thus, explicitly disable interrupts. This 1067517904deSPeter Grehan * also works around the MSI message reordering errata on certain 1068517904deSPeter Grehan * systems. 1069517904deSPeter Grehan */ 1070517904deSPeter Grehan IFDI_INTR_DISABLE(ctx); 1071517904deSPeter Grehan 1072517904deSPeter Grehan /* Link status change */ 1073517904deSPeter Grehan if (reg_icr & (IGC_ICR_RXSEQ | IGC_ICR_LSC)) 1074517904deSPeter Grehan igc_handle_link(ctx); 1075517904deSPeter Grehan 1076517904deSPeter Grehan if (reg_icr & IGC_ICR_RXO) 1077542f5d56SKevin Bowling sc->rx_overruns++; 1078517904deSPeter Grehan 1079542f5d56SKevin Bowling igc_neweitr(sc, que, txr, rxr); 1080bc9402abSKevin Bowling 1081bc9402abSKevin Bowling /* Reset state */ 1082bc9402abSKevin Bowling txr->tx_bytes = 0; 1083bc9402abSKevin Bowling txr->tx_packets = 0; 1084bc9402abSKevin Bowling rxr->rx_bytes = 0; 1085bc9402abSKevin Bowling rxr->rx_packets = 0; 1086bc9402abSKevin Bowling 1087517904deSPeter Grehan return (FILTER_SCHEDULE_THREAD); 1088517904deSPeter Grehan } 1089517904deSPeter Grehan 1090517904deSPeter Grehan static int 1091517904deSPeter Grehan igc_if_rx_queue_intr_enable(if_ctx_t ctx, uint16_t rxqid) 1092517904deSPeter Grehan { 1093542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 1094542f5d56SKevin Bowling struct igc_rx_queue *rxq = &sc->rx_queues[rxqid]; 1095517904deSPeter Grehan 1096542f5d56SKevin Bowling IGC_WRITE_REG(&sc->hw, IGC_EIMS, rxq->eims); 1097517904deSPeter Grehan return (0); 1098517904deSPeter Grehan } 1099517904deSPeter Grehan 1100517904deSPeter Grehan static int 1101517904deSPeter Grehan igc_if_tx_queue_intr_enable(if_ctx_t ctx, uint16_t txqid) 1102517904deSPeter Grehan { 1103542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 1104542f5d56SKevin Bowling struct igc_tx_queue *txq = &sc->tx_queues[txqid]; 1105517904deSPeter Grehan 1106542f5d56SKevin Bowling IGC_WRITE_REG(&sc->hw, IGC_EIMS, txq->eims); 1107517904deSPeter Grehan return (0); 1108517904deSPeter Grehan } 1109517904deSPeter Grehan 1110517904deSPeter Grehan /********************************************************************* 1111517904deSPeter Grehan * 1112517904deSPeter Grehan * MSI-X RX Interrupt Service routine 1113517904deSPeter Grehan * 1114517904deSPeter Grehan **********************************************************************/ 1115517904deSPeter Grehan static int 1116517904deSPeter Grehan igc_msix_que(void *arg) 1117517904deSPeter Grehan { 1118517904deSPeter Grehan struct igc_rx_queue *que = arg; 1119542f5d56SKevin Bowling struct igc_softc *sc = que->sc; 1120bc9402abSKevin Bowling struct tx_ring *txr = &sc->tx_queues[que->msix].txr; 1121bc9402abSKevin Bowling struct rx_ring *rxr = &que->rxr; 1122517904deSPeter Grehan 1123517904deSPeter Grehan ++que->irqs; 1124517904deSPeter Grehan 1125bc9402abSKevin Bowling igc_neweitr(sc, que, txr, rxr); 1126bc9402abSKevin Bowling 1127bc9402abSKevin Bowling /* Reset state */ 1128bc9402abSKevin Bowling txr->tx_bytes = 0; 1129bc9402abSKevin Bowling txr->tx_packets = 0; 1130bc9402abSKevin Bowling rxr->rx_bytes = 0; 1131bc9402abSKevin Bowling rxr->rx_packets = 0; 1132bc9402abSKevin Bowling 1133517904deSPeter Grehan return (FILTER_SCHEDULE_THREAD); 1134517904deSPeter Grehan } 1135517904deSPeter Grehan 1136517904deSPeter Grehan /********************************************************************* 1137517904deSPeter Grehan * 1138517904deSPeter Grehan * MSI-X Link Fast Interrupt Service routine 1139517904deSPeter Grehan * 1140517904deSPeter Grehan **********************************************************************/ 1141517904deSPeter Grehan static int 1142517904deSPeter Grehan igc_msix_link(void *arg) 1143517904deSPeter Grehan { 1144542f5d56SKevin Bowling struct igc_softc *sc = arg; 1145517904deSPeter Grehan u32 reg_icr; 1146517904deSPeter Grehan 1147542f5d56SKevin Bowling ++sc->link_irq; 1148542f5d56SKevin Bowling MPASS(sc->hw.back != NULL); 1149542f5d56SKevin Bowling reg_icr = IGC_READ_REG(&sc->hw, IGC_ICR); 1150517904deSPeter Grehan 1151517904deSPeter Grehan if (reg_icr & IGC_ICR_RXO) 1152542f5d56SKevin Bowling sc->rx_overruns++; 1153517904deSPeter Grehan 1154517904deSPeter Grehan if (reg_icr & (IGC_ICR_RXSEQ | IGC_ICR_LSC)) { 1155542f5d56SKevin Bowling igc_handle_link(sc->ctx); 1156517904deSPeter Grehan } 1157517904deSPeter Grehan 1158542f5d56SKevin Bowling IGC_WRITE_REG(&sc->hw, IGC_IMS, IGC_IMS_LSC); 1159542f5d56SKevin Bowling IGC_WRITE_REG(&sc->hw, IGC_EIMS, sc->link_mask); 1160517904deSPeter Grehan 1161517904deSPeter Grehan return (FILTER_HANDLED); 1162517904deSPeter Grehan } 1163517904deSPeter Grehan 1164517904deSPeter Grehan static void 1165517904deSPeter Grehan igc_handle_link(void *context) 1166517904deSPeter Grehan { 1167517904deSPeter Grehan if_ctx_t ctx = context; 1168542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 1169517904deSPeter Grehan 1170542f5d56SKevin Bowling sc->hw.mac.get_link_status = true; 1171517904deSPeter Grehan iflib_admin_intr_deferred(ctx); 1172517904deSPeter Grehan } 1173517904deSPeter Grehan 1174517904deSPeter Grehan /********************************************************************* 1175517904deSPeter Grehan * 1176517904deSPeter Grehan * Media Ioctl callback 1177517904deSPeter Grehan * 1178517904deSPeter Grehan * This routine is called whenever the user queries the status of 1179517904deSPeter Grehan * the interface using ifconfig. 1180517904deSPeter Grehan * 1181517904deSPeter Grehan **********************************************************************/ 1182517904deSPeter Grehan static void 1183517904deSPeter Grehan igc_if_media_status(if_ctx_t ctx, struct ifmediareq *ifmr) 1184517904deSPeter Grehan { 1185542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 1186517904deSPeter Grehan 1187517904deSPeter Grehan INIT_DEBUGOUT("igc_if_media_status: begin"); 1188517904deSPeter Grehan 1189517904deSPeter Grehan iflib_admin_intr_deferred(ctx); 1190517904deSPeter Grehan 1191517904deSPeter Grehan ifmr->ifm_status = IFM_AVALID; 1192517904deSPeter Grehan ifmr->ifm_active = IFM_ETHER; 1193517904deSPeter Grehan 1194542f5d56SKevin Bowling if (!sc->link_active) { 1195517904deSPeter Grehan return; 1196517904deSPeter Grehan } 1197517904deSPeter Grehan 1198517904deSPeter Grehan ifmr->ifm_status |= IFM_ACTIVE; 1199517904deSPeter Grehan 1200542f5d56SKevin Bowling switch (sc->link_speed) { 1201517904deSPeter Grehan case 10: 1202517904deSPeter Grehan ifmr->ifm_active |= IFM_10_T; 1203517904deSPeter Grehan break; 1204517904deSPeter Grehan case 100: 1205517904deSPeter Grehan ifmr->ifm_active |= IFM_100_TX; 1206517904deSPeter Grehan break; 1207517904deSPeter Grehan case 1000: 1208517904deSPeter Grehan ifmr->ifm_active |= IFM_1000_T; 1209517904deSPeter Grehan break; 1210517904deSPeter Grehan case 2500: 1211517904deSPeter Grehan ifmr->ifm_active |= IFM_2500_T; 1212517904deSPeter Grehan break; 1213517904deSPeter Grehan } 1214517904deSPeter Grehan 1215542f5d56SKevin Bowling if (sc->link_duplex == FULL_DUPLEX) 1216517904deSPeter Grehan ifmr->ifm_active |= IFM_FDX; 1217517904deSPeter Grehan else 1218517904deSPeter Grehan ifmr->ifm_active |= IFM_HDX; 1219517904deSPeter Grehan } 1220517904deSPeter Grehan 1221517904deSPeter Grehan /********************************************************************* 1222517904deSPeter Grehan * 1223517904deSPeter Grehan * Media Ioctl callback 1224517904deSPeter Grehan * 1225517904deSPeter Grehan * This routine is called when the user changes speed/duplex using 1226517904deSPeter Grehan * media/mediopt option with ifconfig. 1227517904deSPeter Grehan * 1228517904deSPeter Grehan **********************************************************************/ 1229517904deSPeter Grehan static int 1230517904deSPeter Grehan igc_if_media_change(if_ctx_t ctx) 1231517904deSPeter Grehan { 1232542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 1233517904deSPeter Grehan struct ifmedia *ifm = iflib_get_media(ctx); 1234517904deSPeter Grehan 1235517904deSPeter Grehan INIT_DEBUGOUT("igc_if_media_change: begin"); 1236517904deSPeter Grehan 1237517904deSPeter Grehan if (IFM_TYPE(ifm->ifm_media) != IFM_ETHER) 1238517904deSPeter Grehan return (EINVAL); 1239517904deSPeter Grehan 1240542f5d56SKevin Bowling sc->hw.mac.autoneg = DO_AUTO_NEG; 1241517904deSPeter Grehan 1242517904deSPeter Grehan switch (IFM_SUBTYPE(ifm->ifm_media)) { 1243517904deSPeter Grehan case IFM_AUTO: 1244542f5d56SKevin Bowling sc->hw.phy.autoneg_advertised = AUTONEG_ADV_DEFAULT; 1245517904deSPeter Grehan break; 1246517904deSPeter Grehan case IFM_2500_T: 1247542f5d56SKevin Bowling sc->hw.phy.autoneg_advertised = ADVERTISE_2500_FULL; 1248517904deSPeter Grehan break; 1249517904deSPeter Grehan case IFM_1000_T: 1250542f5d56SKevin Bowling sc->hw.phy.autoneg_advertised = ADVERTISE_1000_FULL; 1251517904deSPeter Grehan break; 1252517904deSPeter Grehan case IFM_100_TX: 12533b8d04f8SKornel Dulęba if ((ifm->ifm_media & IFM_GMASK) == IFM_FDX) 1254542f5d56SKevin Bowling sc->hw.phy.autoneg_advertised = ADVERTISE_100_FULL; 12553b8d04f8SKornel Dulęba else 1256542f5d56SKevin Bowling sc->hw.phy.autoneg_advertised = ADVERTISE_100_HALF; 1257517904deSPeter Grehan break; 1258517904deSPeter Grehan case IFM_10_T: 12593b8d04f8SKornel Dulęba if ((ifm->ifm_media & IFM_GMASK) == IFM_FDX) 1260542f5d56SKevin Bowling sc->hw.phy.autoneg_advertised = ADVERTISE_10_FULL; 12613b8d04f8SKornel Dulęba else 1262542f5d56SKevin Bowling sc->hw.phy.autoneg_advertised = ADVERTISE_10_HALF; 1263517904deSPeter Grehan break; 1264517904deSPeter Grehan default: 1265542f5d56SKevin Bowling device_printf(sc->dev, "Unsupported media type\n"); 1266517904deSPeter Grehan } 1267517904deSPeter Grehan 1268517904deSPeter Grehan igc_if_init(ctx); 1269517904deSPeter Grehan 1270517904deSPeter Grehan return (0); 1271517904deSPeter Grehan } 1272517904deSPeter Grehan 1273517904deSPeter Grehan static int 1274517904deSPeter Grehan igc_if_set_promisc(if_ctx_t ctx, int flags) 1275517904deSPeter Grehan { 1276542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 1277ec22a3a2SJustin Hibbits if_t ifp = iflib_get_ifp(ctx); 1278517904deSPeter Grehan u32 reg_rctl; 1279517904deSPeter Grehan int mcnt = 0; 1280517904deSPeter Grehan 1281542f5d56SKevin Bowling reg_rctl = IGC_READ_REG(&sc->hw, IGC_RCTL); 1282517904deSPeter Grehan reg_rctl &= ~(IGC_RCTL_SBP | IGC_RCTL_UPE); 1283517904deSPeter Grehan if (flags & IFF_ALLMULTI) 1284517904deSPeter Grehan mcnt = MAX_NUM_MULTICAST_ADDRESSES; 1285517904deSPeter Grehan else 1286517904deSPeter Grehan mcnt = min(if_llmaddr_count(ifp), MAX_NUM_MULTICAST_ADDRESSES); 1287517904deSPeter Grehan 1288517904deSPeter Grehan /* Don't disable if in MAX groups */ 1289517904deSPeter Grehan if (mcnt < MAX_NUM_MULTICAST_ADDRESSES) 1290517904deSPeter Grehan reg_rctl &= (~IGC_RCTL_MPE); 1291542f5d56SKevin Bowling IGC_WRITE_REG(&sc->hw, IGC_RCTL, reg_rctl); 1292517904deSPeter Grehan 1293517904deSPeter Grehan if (flags & IFF_PROMISC) { 1294517904deSPeter Grehan reg_rctl |= (IGC_RCTL_UPE | IGC_RCTL_MPE); 1295517904deSPeter Grehan /* Turn this on if you want to see bad packets */ 1296517904deSPeter Grehan if (igc_debug_sbp) 1297517904deSPeter Grehan reg_rctl |= IGC_RCTL_SBP; 1298542f5d56SKevin Bowling IGC_WRITE_REG(&sc->hw, IGC_RCTL, reg_rctl); 1299517904deSPeter Grehan } else if (flags & IFF_ALLMULTI) { 1300517904deSPeter Grehan reg_rctl |= IGC_RCTL_MPE; 1301517904deSPeter Grehan reg_rctl &= ~IGC_RCTL_UPE; 1302542f5d56SKevin Bowling IGC_WRITE_REG(&sc->hw, IGC_RCTL, reg_rctl); 1303517904deSPeter Grehan } 1304517904deSPeter Grehan return (0); 1305517904deSPeter Grehan } 1306517904deSPeter Grehan 1307517904deSPeter Grehan static u_int 1308517904deSPeter Grehan igc_copy_maddr(void *arg, struct sockaddr_dl *sdl, u_int idx) 1309517904deSPeter Grehan { 1310517904deSPeter Grehan u8 *mta = arg; 1311517904deSPeter Grehan 1312517904deSPeter Grehan if (idx == MAX_NUM_MULTICAST_ADDRESSES) 1313517904deSPeter Grehan return (0); 1314517904deSPeter Grehan 1315517904deSPeter Grehan bcopy(LLADDR(sdl), &mta[idx * ETHER_ADDR_LEN], ETHER_ADDR_LEN); 1316517904deSPeter Grehan 1317517904deSPeter Grehan return (1); 1318517904deSPeter Grehan } 1319517904deSPeter Grehan 1320517904deSPeter Grehan /********************************************************************* 1321517904deSPeter Grehan * Multicast Update 1322517904deSPeter Grehan * 1323517904deSPeter Grehan * This routine is called whenever multicast address list is updated. 1324517904deSPeter Grehan * 1325517904deSPeter Grehan **********************************************************************/ 1326517904deSPeter Grehan 1327517904deSPeter Grehan static void 1328517904deSPeter Grehan igc_if_multi_set(if_ctx_t ctx) 1329517904deSPeter Grehan { 1330542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 1331ec22a3a2SJustin Hibbits if_t ifp = iflib_get_ifp(ctx); 1332517904deSPeter Grehan u8 *mta; /* Multicast array memory */ 1333517904deSPeter Grehan u32 reg_rctl = 0; 1334517904deSPeter Grehan int mcnt = 0; 1335517904deSPeter Grehan 1336517904deSPeter Grehan IOCTL_DEBUGOUT("igc_set_multi: begin"); 1337517904deSPeter Grehan 1338542f5d56SKevin Bowling mta = sc->mta; 1339517904deSPeter Grehan bzero(mta, sizeof(u8) * ETHER_ADDR_LEN * MAX_NUM_MULTICAST_ADDRESSES); 1340517904deSPeter Grehan 1341517904deSPeter Grehan mcnt = if_foreach_llmaddr(ifp, igc_copy_maddr, mta); 1342517904deSPeter Grehan 1343542f5d56SKevin Bowling reg_rctl = IGC_READ_REG(&sc->hw, IGC_RCTL); 1344517904deSPeter Grehan 1345517904deSPeter Grehan if (if_getflags(ifp) & IFF_PROMISC) { 1346517904deSPeter Grehan reg_rctl |= (IGC_RCTL_UPE | IGC_RCTL_MPE); 1347517904deSPeter Grehan /* Turn this on if you want to see bad packets */ 1348517904deSPeter Grehan if (igc_debug_sbp) 1349517904deSPeter Grehan reg_rctl |= IGC_RCTL_SBP; 1350517904deSPeter Grehan } else if (mcnt >= MAX_NUM_MULTICAST_ADDRESSES || 1351517904deSPeter Grehan if_getflags(ifp) & IFF_ALLMULTI) { 1352517904deSPeter Grehan reg_rctl |= IGC_RCTL_MPE; 1353517904deSPeter Grehan reg_rctl &= ~IGC_RCTL_UPE; 1354517904deSPeter Grehan } else 13555a3eb620SPeter Grehan reg_rctl &= ~(IGC_RCTL_UPE | IGC_RCTL_MPE); 1356517904deSPeter Grehan 1357517904deSPeter Grehan if (mcnt < MAX_NUM_MULTICAST_ADDRESSES) 1358542f5d56SKevin Bowling igc_update_mc_addr_list(&sc->hw, mta, mcnt); 13595a3eb620SPeter Grehan 1360542f5d56SKevin Bowling IGC_WRITE_REG(&sc->hw, IGC_RCTL, reg_rctl); 1361517904deSPeter Grehan } 1362517904deSPeter Grehan 1363517904deSPeter Grehan /********************************************************************* 1364517904deSPeter Grehan * Timer routine 1365517904deSPeter Grehan * 1366517904deSPeter Grehan * This routine schedules igc_if_update_admin_status() to check for 1367517904deSPeter Grehan * link status and to gather statistics as well as to perform some 1368517904deSPeter Grehan * controller-specific hardware patting. 1369517904deSPeter Grehan * 1370517904deSPeter Grehan **********************************************************************/ 1371517904deSPeter Grehan static void 1372517904deSPeter Grehan igc_if_timer(if_ctx_t ctx, uint16_t qid) 1373517904deSPeter Grehan { 1374517904deSPeter Grehan 1375517904deSPeter Grehan if (qid != 0) 1376517904deSPeter Grehan return; 1377517904deSPeter Grehan 1378517904deSPeter Grehan iflib_admin_intr_deferred(ctx); 1379517904deSPeter Grehan } 1380517904deSPeter Grehan 1381517904deSPeter Grehan static void 1382517904deSPeter Grehan igc_if_update_admin_status(if_ctx_t ctx) 1383517904deSPeter Grehan { 1384542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 1385542f5d56SKevin Bowling struct igc_hw *hw = &sc->hw; 1386517904deSPeter Grehan device_t dev = iflib_get_dev(ctx); 1387517904deSPeter Grehan u32 link_check, thstat, ctrl; 1388517904deSPeter Grehan 1389517904deSPeter Grehan link_check = thstat = ctrl = 0; 1390517904deSPeter Grehan /* Get the cached link value or read phy for real */ 1391517904deSPeter Grehan switch (hw->phy.media_type) { 1392517904deSPeter Grehan case igc_media_type_copper: 1393517904deSPeter Grehan if (hw->mac.get_link_status == true) { 1394517904deSPeter Grehan /* Do the work to read phy */ 1395517904deSPeter Grehan igc_check_for_link(hw); 1396517904deSPeter Grehan link_check = !hw->mac.get_link_status; 1397517904deSPeter Grehan } else 1398517904deSPeter Grehan link_check = true; 1399517904deSPeter Grehan break; 1400517904deSPeter Grehan case igc_media_type_unknown: 1401517904deSPeter Grehan igc_check_for_link(hw); 1402517904deSPeter Grehan link_check = !hw->mac.get_link_status; 1403517904deSPeter Grehan /* FALLTHROUGH */ 1404517904deSPeter Grehan default: 1405517904deSPeter Grehan break; 1406517904deSPeter Grehan } 1407517904deSPeter Grehan 1408517904deSPeter Grehan /* Now check for a transition */ 1409542f5d56SKevin Bowling if (link_check && (sc->link_active == 0)) { 1410542f5d56SKevin Bowling igc_get_speed_and_duplex(hw, &sc->link_speed, 1411542f5d56SKevin Bowling &sc->link_duplex); 1412517904deSPeter Grehan if (bootverbose) 1413517904deSPeter Grehan device_printf(dev, "Link is up %d Mbps %s\n", 1414542f5d56SKevin Bowling sc->link_speed, 1415542f5d56SKevin Bowling ((sc->link_duplex == FULL_DUPLEX) ? 1416517904deSPeter Grehan "Full Duplex" : "Half Duplex")); 1417542f5d56SKevin Bowling sc->link_active = 1; 1418517904deSPeter Grehan iflib_link_state_change(ctx, LINK_STATE_UP, 1419542f5d56SKevin Bowling IF_Mbps(sc->link_speed)); 1420542f5d56SKevin Bowling } else if (!link_check && (sc->link_active == 1)) { 1421542f5d56SKevin Bowling sc->link_speed = 0; 1422542f5d56SKevin Bowling sc->link_duplex = 0; 1423542f5d56SKevin Bowling sc->link_active = 0; 1424517904deSPeter Grehan iflib_link_state_change(ctx, LINK_STATE_DOWN, 0); 1425517904deSPeter Grehan } 1426542f5d56SKevin Bowling igc_update_stats_counters(sc); 1427517904deSPeter Grehan } 1428517904deSPeter Grehan 1429517904deSPeter Grehan static void 1430517904deSPeter Grehan igc_if_watchdog_reset(if_ctx_t ctx) 1431517904deSPeter Grehan { 1432542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 1433517904deSPeter Grehan 1434517904deSPeter Grehan /* 1435517904deSPeter Grehan * Just count the event; iflib(4) will already trigger a 1436517904deSPeter Grehan * sufficient reset of the controller. 1437517904deSPeter Grehan */ 1438542f5d56SKevin Bowling sc->watchdog_events++; 1439517904deSPeter Grehan } 1440517904deSPeter Grehan 1441517904deSPeter Grehan /********************************************************************* 1442517904deSPeter Grehan * 1443517904deSPeter Grehan * This routine disables all traffic on the adapter by issuing a 1444517904deSPeter Grehan * global reset on the MAC. 1445517904deSPeter Grehan * 1446517904deSPeter Grehan **********************************************************************/ 1447517904deSPeter Grehan static void 1448517904deSPeter Grehan igc_if_stop(if_ctx_t ctx) 1449517904deSPeter Grehan { 1450542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 1451517904deSPeter Grehan 1452517904deSPeter Grehan INIT_DEBUGOUT("igc_if_stop: begin"); 1453517904deSPeter Grehan 1454542f5d56SKevin Bowling igc_reset_hw(&sc->hw); 1455542f5d56SKevin Bowling IGC_WRITE_REG(&sc->hw, IGC_WUC, 0); 1456517904deSPeter Grehan } 1457517904deSPeter Grehan 1458517904deSPeter Grehan /********************************************************************* 1459517904deSPeter Grehan * 1460517904deSPeter Grehan * Determine hardware revision. 1461517904deSPeter Grehan * 1462517904deSPeter Grehan **********************************************************************/ 1463517904deSPeter Grehan static void 1464517904deSPeter Grehan igc_identify_hardware(if_ctx_t ctx) 1465517904deSPeter Grehan { 1466517904deSPeter Grehan device_t dev = iflib_get_dev(ctx); 1467542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 1468517904deSPeter Grehan 1469517904deSPeter Grehan /* Make sure our PCI config space has the necessary stuff set */ 1470542f5d56SKevin Bowling sc->hw.bus.pci_cmd_word = pci_read_config(dev, PCIR_COMMAND, 2); 1471517904deSPeter Grehan 1472517904deSPeter Grehan /* Save off the information about this board */ 1473542f5d56SKevin Bowling sc->hw.vendor_id = pci_get_vendor(dev); 1474542f5d56SKevin Bowling sc->hw.device_id = pci_get_device(dev); 1475542f5d56SKevin Bowling sc->hw.revision_id = pci_read_config(dev, PCIR_REVID, 1); 1476542f5d56SKevin Bowling sc->hw.subsystem_vendor_id = 1477517904deSPeter Grehan pci_read_config(dev, PCIR_SUBVEND_0, 2); 1478542f5d56SKevin Bowling sc->hw.subsystem_device_id = 1479517904deSPeter Grehan pci_read_config(dev, PCIR_SUBDEV_0, 2); 1480517904deSPeter Grehan 1481517904deSPeter Grehan /* Do Shared Code Init and Setup */ 1482542f5d56SKevin Bowling if (igc_set_mac_type(&sc->hw)) { 1483517904deSPeter Grehan device_printf(dev, "Setup init failure\n"); 1484517904deSPeter Grehan return; 1485517904deSPeter Grehan } 1486517904deSPeter Grehan } 1487517904deSPeter Grehan 1488517904deSPeter Grehan static int 1489517904deSPeter Grehan igc_allocate_pci_resources(if_ctx_t ctx) 1490517904deSPeter Grehan { 1491542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 1492517904deSPeter Grehan device_t dev = iflib_get_dev(ctx); 1493517904deSPeter Grehan int rid; 1494517904deSPeter Grehan 1495517904deSPeter Grehan rid = PCIR_BAR(0); 1496542f5d56SKevin Bowling sc->memory = bus_alloc_resource_any(dev, SYS_RES_MEMORY, 1497517904deSPeter Grehan &rid, RF_ACTIVE); 1498542f5d56SKevin Bowling if (sc->memory == NULL) { 14999efc7325SKevin Bowling device_printf(dev, 15009efc7325SKevin Bowling "Unable to allocate bus resource: memory\n"); 1501517904deSPeter Grehan return (ENXIO); 1502517904deSPeter Grehan } 1503542f5d56SKevin Bowling sc->osdep.mem_bus_space_tag = rman_get_bustag(sc->memory); 1504542f5d56SKevin Bowling sc->osdep.mem_bus_space_handle = 1505542f5d56SKevin Bowling rman_get_bushandle(sc->memory); 1506542f5d56SKevin Bowling sc->hw.hw_addr = (u8 *)&sc->osdep.mem_bus_space_handle; 1507517904deSPeter Grehan 1508542f5d56SKevin Bowling sc->hw.back = &sc->osdep; 1509517904deSPeter Grehan 1510517904deSPeter Grehan return (0); 1511517904deSPeter Grehan } 1512517904deSPeter Grehan 1513517904deSPeter Grehan /********************************************************************* 1514517904deSPeter Grehan * 1515517904deSPeter Grehan * Set up the MSI-X Interrupt handlers 1516517904deSPeter Grehan * 1517517904deSPeter Grehan **********************************************************************/ 1518517904deSPeter Grehan static int 1519517904deSPeter Grehan igc_if_msix_intr_assign(if_ctx_t ctx, int msix) 1520517904deSPeter Grehan { 1521542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 1522542f5d56SKevin Bowling struct igc_rx_queue *rx_que = sc->rx_queues; 1523542f5d56SKevin Bowling struct igc_tx_queue *tx_que = sc->tx_queues; 1524517904deSPeter Grehan int error, rid, i, vector = 0, rx_vectors; 1525517904deSPeter Grehan char buf[16]; 1526517904deSPeter Grehan 1527517904deSPeter Grehan /* First set up ring resources */ 1528542f5d56SKevin Bowling for (i = 0; i < sc->rx_num_queues; i++, rx_que++, vector++) { 1529517904deSPeter Grehan rid = vector + 1; 1530517904deSPeter Grehan snprintf(buf, sizeof(buf), "rxq%d", i); 15319efc7325SKevin Bowling error = iflib_irq_alloc_generic(ctx, &rx_que->que_irq, rid, 15329efc7325SKevin Bowling IFLIB_INTR_RXTX, igc_msix_que, rx_que, rx_que->me, buf); 1533517904deSPeter Grehan if (error) { 15349efc7325SKevin Bowling device_printf(iflib_get_dev(ctx), 15359efc7325SKevin Bowling "Failed to allocate que int %d err: %d", 15369efc7325SKevin Bowling i, error); 1537542f5d56SKevin Bowling sc->rx_num_queues = i + 1; 1538517904deSPeter Grehan goto fail; 1539517904deSPeter Grehan } 1540517904deSPeter Grehan 1541517904deSPeter Grehan rx_que->msix = vector; 1542517904deSPeter Grehan 1543517904deSPeter Grehan /* 1544517904deSPeter Grehan * Set the bit to enable interrupt 1545517904deSPeter Grehan * in IGC_IMS -- bits 20 and 21 1546517904deSPeter Grehan * are for RX0 and RX1, note this has 1547517904deSPeter Grehan * NOTHING to do with the MSI-X vector 1548517904deSPeter Grehan */ 1549517904deSPeter Grehan rx_que->eims = 1 << vector; 1550517904deSPeter Grehan } 1551517904deSPeter Grehan rx_vectors = vector; 1552517904deSPeter Grehan 1553517904deSPeter Grehan vector = 0; 1554542f5d56SKevin Bowling for (i = 0; i < sc->tx_num_queues; i++, tx_que++, vector++) { 1555517904deSPeter Grehan snprintf(buf, sizeof(buf), "txq%d", i); 1556542f5d56SKevin Bowling tx_que = &sc->tx_queues[i]; 1557517904deSPeter Grehan iflib_softirq_alloc_generic(ctx, 1558542f5d56SKevin Bowling &sc->rx_queues[i % sc->rx_num_queues].que_irq, 1559517904deSPeter Grehan IFLIB_INTR_TX, tx_que, tx_que->me, buf); 1560517904deSPeter Grehan 1561542f5d56SKevin Bowling tx_que->msix = (vector % sc->rx_num_queues); 1562517904deSPeter Grehan 1563517904deSPeter Grehan /* 1564517904deSPeter Grehan * Set the bit to enable interrupt 1565517904deSPeter Grehan * in IGC_IMS -- bits 22 and 23 1566517904deSPeter Grehan * are for TX0 and TX1, note this has 1567517904deSPeter Grehan * NOTHING to do with the MSI-X vector 1568517904deSPeter Grehan */ 1569517904deSPeter Grehan tx_que->eims = 1 << i; 1570517904deSPeter Grehan } 1571517904deSPeter Grehan 1572517904deSPeter Grehan /* Link interrupt */ 1573517904deSPeter Grehan rid = rx_vectors + 1; 15749efc7325SKevin Bowling error = iflib_irq_alloc_generic(ctx, &sc->irq, rid, IFLIB_INTR_ADMIN, 15759efc7325SKevin Bowling igc_msix_link, sc, 0, "aq"); 1576517904deSPeter Grehan 1577517904deSPeter Grehan if (error) { 15789efc7325SKevin Bowling device_printf(iflib_get_dev(ctx), 15799efc7325SKevin Bowling "Failed to register admin handler"); 1580517904deSPeter Grehan goto fail; 1581517904deSPeter Grehan } 1582542f5d56SKevin Bowling sc->linkvec = rx_vectors; 1583517904deSPeter Grehan return (0); 1584517904deSPeter Grehan fail: 1585542f5d56SKevin Bowling iflib_irq_free(ctx, &sc->irq); 1586542f5d56SKevin Bowling rx_que = sc->rx_queues; 1587542f5d56SKevin Bowling for (int i = 0; i < sc->rx_num_queues; i++, rx_que++) 1588517904deSPeter Grehan iflib_irq_free(ctx, &rx_que->que_irq); 1589517904deSPeter Grehan return (error); 1590517904deSPeter Grehan } 1591517904deSPeter Grehan 1592517904deSPeter Grehan static void 1593542f5d56SKevin Bowling igc_configure_queues(struct igc_softc *sc) 1594517904deSPeter Grehan { 1595542f5d56SKevin Bowling struct igc_hw *hw = &sc->hw; 1596517904deSPeter Grehan struct igc_rx_queue *rx_que; 1597517904deSPeter Grehan struct igc_tx_queue *tx_que; 1598517904deSPeter Grehan u32 ivar = 0, newitr = 0; 1599517904deSPeter Grehan 1600517904deSPeter Grehan /* First turn on RSS capability */ 1601517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_GPIE, 1602517904deSPeter Grehan IGC_GPIE_MSIX_MODE | IGC_GPIE_EIAME | IGC_GPIE_PBA | 1603517904deSPeter Grehan IGC_GPIE_NSICR); 1604517904deSPeter Grehan 1605517904deSPeter Grehan /* Turn on MSI-X */ 1606517904deSPeter Grehan /* RX entries */ 1607542f5d56SKevin Bowling for (int i = 0; i < sc->rx_num_queues; i++) { 1608517904deSPeter Grehan u32 index = i >> 1; 1609517904deSPeter Grehan ivar = IGC_READ_REG_ARRAY(hw, IGC_IVAR0, index); 1610542f5d56SKevin Bowling rx_que = &sc->rx_queues[i]; 1611517904deSPeter Grehan if (i & 1) { 1612517904deSPeter Grehan ivar &= 0xFF00FFFF; 1613517904deSPeter Grehan ivar |= (rx_que->msix | IGC_IVAR_VALID) << 16; 1614517904deSPeter Grehan } else { 1615517904deSPeter Grehan ivar &= 0xFFFFFF00; 1616517904deSPeter Grehan ivar |= rx_que->msix | IGC_IVAR_VALID; 1617517904deSPeter Grehan } 1618517904deSPeter Grehan IGC_WRITE_REG_ARRAY(hw, IGC_IVAR0, index, ivar); 1619517904deSPeter Grehan } 1620517904deSPeter Grehan /* TX entries */ 1621542f5d56SKevin Bowling for (int i = 0; i < sc->tx_num_queues; i++) { 1622517904deSPeter Grehan u32 index = i >> 1; 1623517904deSPeter Grehan ivar = IGC_READ_REG_ARRAY(hw, IGC_IVAR0, index); 1624542f5d56SKevin Bowling tx_que = &sc->tx_queues[i]; 1625517904deSPeter Grehan if (i & 1) { 1626517904deSPeter Grehan ivar &= 0x00FFFFFF; 1627517904deSPeter Grehan ivar |= (tx_que->msix | IGC_IVAR_VALID) << 24; 1628517904deSPeter Grehan } else { 1629517904deSPeter Grehan ivar &= 0xFFFF00FF; 1630517904deSPeter Grehan ivar |= (tx_que->msix | IGC_IVAR_VALID) << 8; 1631517904deSPeter Grehan } 1632517904deSPeter Grehan IGC_WRITE_REG_ARRAY(hw, IGC_IVAR0, index, ivar); 1633542f5d56SKevin Bowling sc->que_mask |= tx_que->eims; 1634517904deSPeter Grehan } 1635517904deSPeter Grehan 1636517904deSPeter Grehan /* And for the link interrupt */ 1637542f5d56SKevin Bowling ivar = (sc->linkvec | IGC_IVAR_VALID) << 8; 1638542f5d56SKevin Bowling sc->link_mask = 1 << sc->linkvec; 1639517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_IVAR_MISC, ivar); 1640517904deSPeter Grehan 1641517904deSPeter Grehan /* Set the starting interrupt rate */ 1642517904deSPeter Grehan if (igc_max_interrupt_rate > 0) 1643bc9402abSKevin Bowling newitr = IGC_INTS_TO_EITR(igc_max_interrupt_rate); 1644517904deSPeter Grehan 1645517904deSPeter Grehan newitr |= IGC_EITR_CNT_IGNR; 1646517904deSPeter Grehan 1647542f5d56SKevin Bowling for (int i = 0; i < sc->rx_num_queues; i++) { 1648542f5d56SKevin Bowling rx_que = &sc->rx_queues[i]; 1649517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_EITR(rx_que->msix), newitr); 1650517904deSPeter Grehan } 1651517904deSPeter Grehan 1652517904deSPeter Grehan return; 1653517904deSPeter Grehan } 1654517904deSPeter Grehan 1655517904deSPeter Grehan static void 1656517904deSPeter Grehan igc_free_pci_resources(if_ctx_t ctx) 1657517904deSPeter Grehan { 1658542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 1659542f5d56SKevin Bowling struct igc_rx_queue *que = sc->rx_queues; 1660517904deSPeter Grehan device_t dev = iflib_get_dev(ctx); 1661517904deSPeter Grehan 1662517904deSPeter Grehan /* Release all MSI-X queue resources */ 1663542f5d56SKevin Bowling if (sc->intr_type == IFLIB_INTR_MSIX) 1664542f5d56SKevin Bowling iflib_irq_free(ctx, &sc->irq); 1665517904deSPeter Grehan 1666542f5d56SKevin Bowling for (int i = 0; i < sc->rx_num_queues; i++, que++) { 1667517904deSPeter Grehan iflib_irq_free(ctx, &que->que_irq); 1668517904deSPeter Grehan } 1669517904deSPeter Grehan 1670542f5d56SKevin Bowling if (sc->memory != NULL) { 1671517904deSPeter Grehan bus_release_resource(dev, SYS_RES_MEMORY, 1672542f5d56SKevin Bowling rman_get_rid(sc->memory), sc->memory); 1673542f5d56SKevin Bowling sc->memory = NULL; 1674517904deSPeter Grehan } 1675517904deSPeter Grehan 1676542f5d56SKevin Bowling if (sc->flash != NULL) { 1677517904deSPeter Grehan bus_release_resource(dev, SYS_RES_MEMORY, 1678542f5d56SKevin Bowling rman_get_rid(sc->flash), sc->flash); 1679542f5d56SKevin Bowling sc->flash = NULL; 1680517904deSPeter Grehan } 1681517904deSPeter Grehan 1682542f5d56SKevin Bowling if (sc->ioport != NULL) { 1683517904deSPeter Grehan bus_release_resource(dev, SYS_RES_IOPORT, 1684542f5d56SKevin Bowling rman_get_rid(sc->ioport), sc->ioport); 1685542f5d56SKevin Bowling sc->ioport = NULL; 1686517904deSPeter Grehan } 1687517904deSPeter Grehan } 1688517904deSPeter Grehan 1689517904deSPeter Grehan /* Set up MSI or MSI-X */ 1690517904deSPeter Grehan static int 1691517904deSPeter Grehan igc_setup_msix(if_ctx_t ctx) 1692517904deSPeter Grehan { 1693517904deSPeter Grehan return (0); 1694517904deSPeter Grehan } 1695517904deSPeter Grehan 1696517904deSPeter Grehan /********************************************************************* 1697517904deSPeter Grehan * 1698517904deSPeter Grehan * Initialize the DMA Coalescing feature 1699517904deSPeter Grehan * 1700517904deSPeter Grehan **********************************************************************/ 1701517904deSPeter Grehan static void 1702542f5d56SKevin Bowling igc_init_dmac(struct igc_softc *sc, u32 pba) 1703517904deSPeter Grehan { 1704542f5d56SKevin Bowling device_t dev = sc->dev; 1705542f5d56SKevin Bowling struct igc_hw *hw = &sc->hw; 1706517904deSPeter Grehan u32 dmac, reg = ~IGC_DMACR_DMAC_EN; 1707517904deSPeter Grehan u16 hwm; 1708517904deSPeter Grehan u16 max_frame_size; 1709517904deSPeter Grehan int status; 1710517904deSPeter Grehan 1711542f5d56SKevin Bowling max_frame_size = sc->shared->isc_max_frame_size; 1712517904deSPeter Grehan 1713542f5d56SKevin Bowling if (sc->dmac == 0) { /* Disabling it */ 1714517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_DMACR, reg); 1715517904deSPeter Grehan return; 1716517904deSPeter Grehan } else 1717517904deSPeter Grehan device_printf(dev, "DMA Coalescing enabled\n"); 1718517904deSPeter Grehan 1719517904deSPeter Grehan /* Set starting threshold */ 1720517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_DMCTXTH, 0); 1721517904deSPeter Grehan 1722517904deSPeter Grehan hwm = 64 * pba - max_frame_size / 16; 1723517904deSPeter Grehan if (hwm < 64 * (pba - 6)) 1724517904deSPeter Grehan hwm = 64 * (pba - 6); 1725517904deSPeter Grehan reg = IGC_READ_REG(hw, IGC_FCRTC); 1726517904deSPeter Grehan reg &= ~IGC_FCRTC_RTH_COAL_MASK; 1727517904deSPeter Grehan reg |= ((hwm << IGC_FCRTC_RTH_COAL_SHIFT) 1728517904deSPeter Grehan & IGC_FCRTC_RTH_COAL_MASK); 1729517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_FCRTC, reg); 1730517904deSPeter Grehan 1731517904deSPeter Grehan dmac = pba - max_frame_size / 512; 1732517904deSPeter Grehan if (dmac < pba - 10) 1733517904deSPeter Grehan dmac = pba - 10; 1734517904deSPeter Grehan reg = IGC_READ_REG(hw, IGC_DMACR); 1735517904deSPeter Grehan reg &= ~IGC_DMACR_DMACTHR_MASK; 1736517904deSPeter Grehan reg |= ((dmac << IGC_DMACR_DMACTHR_SHIFT) 1737517904deSPeter Grehan & IGC_DMACR_DMACTHR_MASK); 1738517904deSPeter Grehan 1739517904deSPeter Grehan /* transition to L0x or L1 if available..*/ 1740517904deSPeter Grehan reg |= (IGC_DMACR_DMAC_EN | IGC_DMACR_DMAC_LX_MASK); 1741517904deSPeter Grehan 1742517904deSPeter Grehan /* Check if status is 2.5Gb backplane connection 1743517904deSPeter Grehan * before configuration of watchdog timer, which is 1744517904deSPeter Grehan * in msec values in 12.8usec intervals 1745517904deSPeter Grehan * watchdog timer= msec values in 32usec intervals 1746517904deSPeter Grehan * for non 2.5Gb connection 1747517904deSPeter Grehan */ 1748517904deSPeter Grehan status = IGC_READ_REG(hw, IGC_STATUS); 1749517904deSPeter Grehan if ((status & IGC_STATUS_2P5_SKU) && 1750517904deSPeter Grehan (!(status & IGC_STATUS_2P5_SKU_OVER))) 1751542f5d56SKevin Bowling reg |= ((sc->dmac * 5) >> 6); 1752517904deSPeter Grehan else 1753542f5d56SKevin Bowling reg |= (sc->dmac >> 5); 1754517904deSPeter Grehan 1755517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_DMACR, reg); 1756517904deSPeter Grehan 1757517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_DMCRTRH, 0); 1758517904deSPeter Grehan 1759517904deSPeter Grehan /* Set the interval before transition */ 1760517904deSPeter Grehan reg = IGC_READ_REG(hw, IGC_DMCTLX); 1761517904deSPeter Grehan reg |= IGC_DMCTLX_DCFLUSH_DIS; 1762517904deSPeter Grehan 1763517904deSPeter Grehan /* 1764517904deSPeter Grehan ** in 2.5Gb connection, TTLX unit is 0.4 usec 1765517904deSPeter Grehan ** which is 0x4*2 = 0xA. But delay is still 4 usec 1766517904deSPeter Grehan */ 1767517904deSPeter Grehan status = IGC_READ_REG(hw, IGC_STATUS); 1768517904deSPeter Grehan if ((status & IGC_STATUS_2P5_SKU) && 1769517904deSPeter Grehan (!(status & IGC_STATUS_2P5_SKU_OVER))) 1770517904deSPeter Grehan reg |= 0xA; 1771517904deSPeter Grehan else 1772517904deSPeter Grehan reg |= 0x4; 1773517904deSPeter Grehan 1774517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_DMCTLX, reg); 1775517904deSPeter Grehan 1776517904deSPeter Grehan /* free space in tx packet buffer to wake from DMA coal */ 1777517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_DMCTXTH, (IGC_TXPBSIZE - 1778517904deSPeter Grehan (2 * max_frame_size)) >> 6); 1779517904deSPeter Grehan 1780517904deSPeter Grehan /* make low power state decision controlled by DMA coal */ 1781517904deSPeter Grehan reg = IGC_READ_REG(hw, IGC_PCIEMISC); 1782517904deSPeter Grehan reg &= ~IGC_PCIEMISC_LX_DECISION; 1783517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_PCIEMISC, reg); 1784517904deSPeter Grehan } 1785517904deSPeter Grehan 1786517904deSPeter Grehan /********************************************************************* 1787517904deSPeter Grehan * 1788517904deSPeter Grehan * Initialize the hardware to a configuration as specified by the 1789542f5d56SKevin Bowling * softc structure. 1790517904deSPeter Grehan * 1791517904deSPeter Grehan **********************************************************************/ 1792517904deSPeter Grehan static void 1793517904deSPeter Grehan igc_reset(if_ctx_t ctx) 1794517904deSPeter Grehan { 1795517904deSPeter Grehan device_t dev = iflib_get_dev(ctx); 1796542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 1797542f5d56SKevin Bowling struct igc_hw *hw = &sc->hw; 17989b88ecd6SKevin Bowling u32 rx_buffer_size; 1799517904deSPeter Grehan u32 pba; 1800517904deSPeter Grehan 1801517904deSPeter Grehan INIT_DEBUGOUT("igc_reset: begin"); 1802517904deSPeter Grehan /* Let the firmware know the OS is in control */ 1803542f5d56SKevin Bowling igc_get_hw_control(sc); 1804517904deSPeter Grehan 1805517904deSPeter Grehan /* 1806517904deSPeter Grehan * Packet Buffer Allocation (PBA) 1807517904deSPeter Grehan * Writing PBA sets the receive portion of the buffer 1808517904deSPeter Grehan * the remainder is used for the transmit buffer. 1809517904deSPeter Grehan */ 1810517904deSPeter Grehan pba = IGC_PBA_34K; 1811517904deSPeter Grehan 1812517904deSPeter Grehan INIT_DEBUGOUT1("igc_reset: pba=%dK",pba); 1813517904deSPeter Grehan 1814517904deSPeter Grehan /* 1815517904deSPeter Grehan * These parameters control the automatic generation (Tx) and 1816517904deSPeter Grehan * response (Rx) to Ethernet PAUSE frames. 1817517904deSPeter Grehan * - High water mark should allow for at least two frames to be 1818517904deSPeter Grehan * received after sending an XOFF. 18199efc7325SKevin Bowling * - Low water mark works best when it is very near the high water 18209efc7325SKevin Bowling * mark. 1821517904deSPeter Grehan * This allows the receiver to restart by sending XON when it has 1822517904deSPeter Grehan * drained a bit. Here we use an arbitrary value of 1500 which will 1823517904deSPeter Grehan * restart after one full frame is pulled from the buffer. There 1824517904deSPeter Grehan * could be several smaller frames in the buffer and if so they will 1825517904deSPeter Grehan * not trigger the XON until their total number reduces the buffer 1826517904deSPeter Grehan * by 1500. 1827517904deSPeter Grehan * - The pause time is fairly large at 1000 x 512ns = 512 usec. 1828517904deSPeter Grehan */ 1829517904deSPeter Grehan rx_buffer_size = (pba & 0xffff) << 10; 1830517904deSPeter Grehan hw->fc.high_water = rx_buffer_size - 1831542f5d56SKevin Bowling roundup2(sc->hw.mac.max_frame_size, 1024); 1832517904deSPeter Grehan /* 16-byte granularity */ 1833517904deSPeter Grehan hw->fc.low_water = hw->fc.high_water - 16; 1834517904deSPeter Grehan 1835542f5d56SKevin Bowling if (sc->fc) /* locally set flow control value? */ 1836542f5d56SKevin Bowling hw->fc.requested_mode = sc->fc; 1837517904deSPeter Grehan else 1838517904deSPeter Grehan hw->fc.requested_mode = igc_fc_full; 1839517904deSPeter Grehan 1840517904deSPeter Grehan hw->fc.pause_time = IGC_FC_PAUSE_TIME; 1841517904deSPeter Grehan 1842517904deSPeter Grehan hw->fc.send_xon = true; 1843517904deSPeter Grehan 1844517904deSPeter Grehan /* Issue a global reset */ 1845517904deSPeter Grehan igc_reset_hw(hw); 1846517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_WUC, 0); 1847517904deSPeter Grehan 1848517904deSPeter Grehan /* and a re-init */ 1849517904deSPeter Grehan if (igc_init_hw(hw) < 0) { 1850517904deSPeter Grehan device_printf(dev, "Hardware Initialization Failed\n"); 1851517904deSPeter Grehan return; 1852517904deSPeter Grehan } 1853517904deSPeter Grehan 1854517904deSPeter Grehan /* Setup DMA Coalescing */ 1855542f5d56SKevin Bowling igc_init_dmac(sc, pba); 1856517904deSPeter Grehan 1857bc9402abSKevin Bowling /* Save the final PBA off if it needs to be used elsewhere i.e. AIM */ 1858542f5d56SKevin Bowling sc->pba = pba; 1859bc9402abSKevin Bowling 1860517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_VET, ETHERTYPE_VLAN); 1861517904deSPeter Grehan igc_get_phy_info(hw); 1862517904deSPeter Grehan igc_check_for_link(hw); 1863517904deSPeter Grehan } 1864517904deSPeter Grehan 1865517904deSPeter Grehan /* 1866517904deSPeter Grehan * Initialise the RSS mapping for NICs that support multiple transmit/ 1867517904deSPeter Grehan * receive rings. 1868517904deSPeter Grehan */ 1869517904deSPeter Grehan 1870517904deSPeter Grehan #define RSSKEYLEN 10 1871517904deSPeter Grehan static void 1872542f5d56SKevin Bowling igc_initialize_rss_mapping(struct igc_softc *sc) 1873517904deSPeter Grehan { 1874542f5d56SKevin Bowling struct igc_hw *hw = &sc->hw; 1875517904deSPeter Grehan int i; 1876517904deSPeter Grehan int queue_id; 1877517904deSPeter Grehan u32 reta; 1878517904deSPeter Grehan u32 rss_key[RSSKEYLEN], mrqc, shift = 0; 1879517904deSPeter Grehan 1880517904deSPeter Grehan /* 1881517904deSPeter Grehan * The redirection table controls which destination 1882517904deSPeter Grehan * queue each bucket redirects traffic to. 1883517904deSPeter Grehan * Each DWORD represents four queues, with the LSB 1884517904deSPeter Grehan * being the first queue in the DWORD. 1885517904deSPeter Grehan * 1886517904deSPeter Grehan * This just allocates buckets to queues using round-robin 1887517904deSPeter Grehan * allocation. 1888517904deSPeter Grehan * 1889517904deSPeter Grehan * NOTE: It Just Happens to line up with the default 1890517904deSPeter Grehan * RSS allocation method. 1891517904deSPeter Grehan */ 1892517904deSPeter Grehan 1893517904deSPeter Grehan /* Warning FM follows */ 1894517904deSPeter Grehan reta = 0; 1895517904deSPeter Grehan for (i = 0; i < 128; i++) { 1896517904deSPeter Grehan #ifdef RSS 1897517904deSPeter Grehan queue_id = rss_get_indirection_to_bucket(i); 1898517904deSPeter Grehan /* 1899517904deSPeter Grehan * If we have more queues than buckets, we'll 1900517904deSPeter Grehan * end up mapping buckets to a subset of the 1901517904deSPeter Grehan * queues. 1902517904deSPeter Grehan * 1903517904deSPeter Grehan * If we have more buckets than queues, we'll 1904517904deSPeter Grehan * end up instead assigning multiple buckets 1905517904deSPeter Grehan * to queues. 1906517904deSPeter Grehan * 1907517904deSPeter Grehan * Both are suboptimal, but we need to handle 1908517904deSPeter Grehan * the case so we don't go out of bounds 1909517904deSPeter Grehan * indexing arrays and such. 1910517904deSPeter Grehan */ 1911542f5d56SKevin Bowling queue_id = queue_id % sc->rx_num_queues; 1912517904deSPeter Grehan #else 1913542f5d56SKevin Bowling queue_id = (i % sc->rx_num_queues); 1914517904deSPeter Grehan #endif 1915517904deSPeter Grehan /* Adjust if required */ 1916517904deSPeter Grehan queue_id = queue_id << shift; 1917517904deSPeter Grehan 1918517904deSPeter Grehan /* 1919517904deSPeter Grehan * The low 8 bits are for hash value (n+0); 1920517904deSPeter Grehan * The next 8 bits are for hash value (n+1), etc. 1921517904deSPeter Grehan */ 1922517904deSPeter Grehan reta = reta >> 8; 1923517904deSPeter Grehan reta = reta | ( ((uint32_t) queue_id) << 24); 1924517904deSPeter Grehan if ((i & 3) == 3) { 1925517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_RETA(i >> 2), reta); 1926517904deSPeter Grehan reta = 0; 1927517904deSPeter Grehan } 1928517904deSPeter Grehan } 1929517904deSPeter Grehan 1930517904deSPeter Grehan /* Now fill in hash table */ 1931517904deSPeter Grehan 1932517904deSPeter Grehan /* 1933517904deSPeter Grehan * MRQC: Multiple Receive Queues Command 1934517904deSPeter Grehan * Set queuing to RSS control, number depends on the device. 1935517904deSPeter Grehan */ 1936517904deSPeter Grehan mrqc = IGC_MRQC_ENABLE_RSS_4Q; 1937517904deSPeter Grehan 1938517904deSPeter Grehan #ifdef RSS 1939517904deSPeter Grehan /* XXX ew typecasting */ 1940517904deSPeter Grehan rss_getkey((uint8_t *) &rss_key); 1941517904deSPeter Grehan #else 1942517904deSPeter Grehan arc4rand(&rss_key, sizeof(rss_key), 0); 1943517904deSPeter Grehan #endif 1944517904deSPeter Grehan for (i = 0; i < RSSKEYLEN; i++) 1945517904deSPeter Grehan IGC_WRITE_REG_ARRAY(hw, IGC_RSSRK(0), i, rss_key[i]); 1946517904deSPeter Grehan 1947517904deSPeter Grehan /* 1948517904deSPeter Grehan * Configure the RSS fields to hash upon. 1949517904deSPeter Grehan */ 1950517904deSPeter Grehan mrqc |= (IGC_MRQC_RSS_FIELD_IPV4 | 1951517904deSPeter Grehan IGC_MRQC_RSS_FIELD_IPV4_TCP); 1952517904deSPeter Grehan mrqc |= (IGC_MRQC_RSS_FIELD_IPV6 | 1953517904deSPeter Grehan IGC_MRQC_RSS_FIELD_IPV6_TCP); 1954517904deSPeter Grehan mrqc |=( IGC_MRQC_RSS_FIELD_IPV4_UDP | 1955517904deSPeter Grehan IGC_MRQC_RSS_FIELD_IPV6_UDP); 1956517904deSPeter Grehan mrqc |=( IGC_MRQC_RSS_FIELD_IPV6_UDP_EX | 1957517904deSPeter Grehan IGC_MRQC_RSS_FIELD_IPV6_TCP_EX); 1958517904deSPeter Grehan 1959517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_MRQC, mrqc); 1960517904deSPeter Grehan } 1961517904deSPeter Grehan 1962517904deSPeter Grehan /********************************************************************* 1963517904deSPeter Grehan * 1964517904deSPeter Grehan * Setup networking device structure and register interface media. 1965517904deSPeter Grehan * 1966517904deSPeter Grehan **********************************************************************/ 1967517904deSPeter Grehan static int 1968517904deSPeter Grehan igc_setup_interface(if_ctx_t ctx) 1969517904deSPeter Grehan { 1970ec22a3a2SJustin Hibbits if_t ifp = iflib_get_ifp(ctx); 1971542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 1972542f5d56SKevin Bowling if_softc_ctx_t scctx = sc->shared; 1973517904deSPeter Grehan 1974517904deSPeter Grehan INIT_DEBUGOUT("igc_setup_interface: begin"); 1975517904deSPeter Grehan 1976517904deSPeter Grehan /* Single Queue */ 1977542f5d56SKevin Bowling if (sc->tx_num_queues == 1) { 1978517904deSPeter Grehan if_setsendqlen(ifp, scctx->isc_ntxd[0] - 1); 1979517904deSPeter Grehan if_setsendqready(ifp); 1980517904deSPeter Grehan } 1981517904deSPeter Grehan 1982517904deSPeter Grehan /* 1983517904deSPeter Grehan * Specify the media types supported by this adapter and register 1984517904deSPeter Grehan * callbacks to update media and link information 1985517904deSPeter Grehan */ 1986542f5d56SKevin Bowling ifmedia_add(sc->media, IFM_ETHER | IFM_10_T, 0, NULL); 1987542f5d56SKevin Bowling ifmedia_add(sc->media, IFM_ETHER | IFM_10_T | IFM_FDX, 0, NULL); 1988542f5d56SKevin Bowling ifmedia_add(sc->media, IFM_ETHER | IFM_100_TX, 0, NULL); 1989542f5d56SKevin Bowling ifmedia_add(sc->media, IFM_ETHER | IFM_100_TX | IFM_FDX, 0, NULL); 1990542f5d56SKevin Bowling ifmedia_add(sc->media, IFM_ETHER | IFM_1000_T | IFM_FDX, 0, NULL); 1991542f5d56SKevin Bowling ifmedia_add(sc->media, IFM_ETHER | IFM_1000_T, 0, NULL); 1992542f5d56SKevin Bowling ifmedia_add(sc->media, IFM_ETHER | IFM_2500_T, 0, NULL); 1993517904deSPeter Grehan 1994542f5d56SKevin Bowling ifmedia_add(sc->media, IFM_ETHER | IFM_AUTO, 0, NULL); 1995542f5d56SKevin Bowling ifmedia_set(sc->media, IFM_ETHER | IFM_AUTO); 1996517904deSPeter Grehan return (0); 1997517904deSPeter Grehan } 1998517904deSPeter Grehan 1999517904deSPeter Grehan static int 20009efc7325SKevin Bowling igc_if_tx_queues_alloc(if_ctx_t ctx, caddr_t *vaddrs, uint64_t *paddrs, 20019efc7325SKevin Bowling int ntxqs, int ntxqsets) 2002517904deSPeter Grehan { 2003542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 2004542f5d56SKevin Bowling if_softc_ctx_t scctx = sc->shared; 2005517904deSPeter Grehan int error = IGC_SUCCESS; 2006517904deSPeter Grehan struct igc_tx_queue *que; 2007517904deSPeter Grehan int i, j; 2008517904deSPeter Grehan 2009542f5d56SKevin Bowling MPASS(sc->tx_num_queues > 0); 2010542f5d56SKevin Bowling MPASS(sc->tx_num_queues == ntxqsets); 2011517904deSPeter Grehan 2012517904deSPeter Grehan /* First allocate the top level queue structs */ 2013542f5d56SKevin Bowling if (!(sc->tx_queues = 2014517904deSPeter Grehan (struct igc_tx_queue *) malloc(sizeof(struct igc_tx_queue) * 2015542f5d56SKevin Bowling sc->tx_num_queues, M_DEVBUF, M_NOWAIT | M_ZERO))) { 20169efc7325SKevin Bowling device_printf(iflib_get_dev(ctx), 20179efc7325SKevin Bowling "Unable to allocate queue memory\n"); 2018517904deSPeter Grehan return(ENOMEM); 2019517904deSPeter Grehan } 2020517904deSPeter Grehan 2021542f5d56SKevin Bowling for (i = 0, que = sc->tx_queues; i < sc->tx_num_queues; i++, que++) { 2022517904deSPeter Grehan /* Set up some basics */ 2023517904deSPeter Grehan 2024517904deSPeter Grehan struct tx_ring *txr = &que->txr; 2025542f5d56SKevin Bowling txr->sc = que->sc = sc; 2026517904deSPeter Grehan que->me = txr->me = i; 2027517904deSPeter Grehan 2028517904deSPeter Grehan /* Allocate report status array */ 20299efc7325SKevin Bowling if (!(txr->tx_rsq = (qidx_t *) malloc(sizeof(qidx_t) * 20309efc7325SKevin Bowling scctx->isc_ntxd[0], M_DEVBUF, M_NOWAIT | M_ZERO))) { 20319efc7325SKevin Bowling device_printf(iflib_get_dev(ctx), 20329efc7325SKevin Bowling "failed to allocate rs_idxs memory\n"); 2033517904deSPeter Grehan error = ENOMEM; 2034517904deSPeter Grehan goto fail; 2035517904deSPeter Grehan } 2036517904deSPeter Grehan for (j = 0; j < scctx->isc_ntxd[0]; j++) 2037517904deSPeter Grehan txr->tx_rsq[j] = QIDX_INVALID; 20389efc7325SKevin Bowling /* get virtual and physical address of the hardware queues */ 2039517904deSPeter Grehan txr->tx_base = (struct igc_tx_desc *)vaddrs[i*ntxqs]; 2040517904deSPeter Grehan txr->tx_paddr = paddrs[i*ntxqs]; 2041517904deSPeter Grehan } 2042517904deSPeter Grehan 2043517904deSPeter Grehan if (bootverbose) 2044517904deSPeter Grehan device_printf(iflib_get_dev(ctx), 2045542f5d56SKevin Bowling "allocated for %d tx_queues\n", sc->tx_num_queues); 2046517904deSPeter Grehan return (0); 2047517904deSPeter Grehan fail: 2048517904deSPeter Grehan igc_if_queues_free(ctx); 2049517904deSPeter Grehan return (error); 2050517904deSPeter Grehan } 2051517904deSPeter Grehan 2052517904deSPeter Grehan static int 20539efc7325SKevin Bowling igc_if_rx_queues_alloc(if_ctx_t ctx, caddr_t *vaddrs, uint64_t *paddrs, 20549efc7325SKevin Bowling int nrxqs, int nrxqsets) 2055517904deSPeter Grehan { 2056542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 2057517904deSPeter Grehan int error = IGC_SUCCESS; 2058517904deSPeter Grehan struct igc_rx_queue *que; 2059517904deSPeter Grehan int i; 2060517904deSPeter Grehan 2061542f5d56SKevin Bowling MPASS(sc->rx_num_queues > 0); 2062542f5d56SKevin Bowling MPASS(sc->rx_num_queues == nrxqsets); 2063517904deSPeter Grehan 2064517904deSPeter Grehan /* First allocate the top level queue structs */ 2065542f5d56SKevin Bowling if (!(sc->rx_queues = 2066517904deSPeter Grehan (struct igc_rx_queue *) malloc(sizeof(struct igc_rx_queue) * 2067542f5d56SKevin Bowling sc->rx_num_queues, M_DEVBUF, M_NOWAIT | M_ZERO))) { 20689efc7325SKevin Bowling device_printf(iflib_get_dev(ctx), 20699efc7325SKevin Bowling "Unable to allocate queue memory\n"); 2070517904deSPeter Grehan error = ENOMEM; 2071517904deSPeter Grehan goto fail; 2072517904deSPeter Grehan } 2073517904deSPeter Grehan 2074542f5d56SKevin Bowling for (i = 0, que = sc->rx_queues; i < nrxqsets; i++, que++) { 2075517904deSPeter Grehan /* Set up some basics */ 2076517904deSPeter Grehan struct rx_ring *rxr = &que->rxr; 2077542f5d56SKevin Bowling rxr->sc = que->sc = sc; 2078517904deSPeter Grehan rxr->que = que; 2079517904deSPeter Grehan que->me = rxr->me = i; 2080517904deSPeter Grehan 20819efc7325SKevin Bowling /* get virtual and physical address of the hardware queues */ 2082517904deSPeter Grehan rxr->rx_base = (union igc_rx_desc_extended *)vaddrs[i*nrxqs]; 2083517904deSPeter Grehan rxr->rx_paddr = paddrs[i*nrxqs]; 2084517904deSPeter Grehan } 2085517904deSPeter Grehan 2086517904deSPeter Grehan if (bootverbose) 2087517904deSPeter Grehan device_printf(iflib_get_dev(ctx), 2088542f5d56SKevin Bowling "allocated for %d rx_queues\n", sc->rx_num_queues); 2089517904deSPeter Grehan 2090517904deSPeter Grehan return (0); 2091517904deSPeter Grehan fail: 2092517904deSPeter Grehan igc_if_queues_free(ctx); 2093517904deSPeter Grehan return (error); 2094517904deSPeter Grehan } 2095517904deSPeter Grehan 2096517904deSPeter Grehan static void 2097517904deSPeter Grehan igc_if_queues_free(if_ctx_t ctx) 2098517904deSPeter Grehan { 2099542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 2100542f5d56SKevin Bowling struct igc_tx_queue *tx_que = sc->tx_queues; 2101542f5d56SKevin Bowling struct igc_rx_queue *rx_que = sc->rx_queues; 2102517904deSPeter Grehan 2103517904deSPeter Grehan if (tx_que != NULL) { 2104542f5d56SKevin Bowling for (int i = 0; i < sc->tx_num_queues; i++, tx_que++) { 2105517904deSPeter Grehan struct tx_ring *txr = &tx_que->txr; 2106517904deSPeter Grehan if (txr->tx_rsq == NULL) 2107517904deSPeter Grehan break; 2108517904deSPeter Grehan 2109517904deSPeter Grehan free(txr->tx_rsq, M_DEVBUF); 2110517904deSPeter Grehan txr->tx_rsq = NULL; 2111517904deSPeter Grehan } 2112542f5d56SKevin Bowling free(sc->tx_queues, M_DEVBUF); 2113542f5d56SKevin Bowling sc->tx_queues = NULL; 2114517904deSPeter Grehan } 2115517904deSPeter Grehan 2116517904deSPeter Grehan if (rx_que != NULL) { 2117542f5d56SKevin Bowling free(sc->rx_queues, M_DEVBUF); 2118542f5d56SKevin Bowling sc->rx_queues = NULL; 2119517904deSPeter Grehan } 2120517904deSPeter Grehan 2121542f5d56SKevin Bowling if (sc->mta != NULL) { 2122542f5d56SKevin Bowling free(sc->mta, M_DEVBUF); 2123517904deSPeter Grehan } 2124517904deSPeter Grehan } 2125517904deSPeter Grehan 2126517904deSPeter Grehan /********************************************************************* 2127517904deSPeter Grehan * 2128517904deSPeter Grehan * Enable transmit unit. 2129517904deSPeter Grehan * 2130517904deSPeter Grehan **********************************************************************/ 2131517904deSPeter Grehan static void 2132517904deSPeter Grehan igc_initialize_transmit_unit(if_ctx_t ctx) 2133517904deSPeter Grehan { 2134542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 2135542f5d56SKevin Bowling if_softc_ctx_t scctx = sc->shared; 2136517904deSPeter Grehan struct igc_tx_queue *que; 2137517904deSPeter Grehan struct tx_ring *txr; 2138542f5d56SKevin Bowling struct igc_hw *hw = &sc->hw; 2139517904deSPeter Grehan u32 tctl, txdctl = 0; 2140517904deSPeter Grehan 2141517904deSPeter Grehan INIT_DEBUGOUT("igc_initialize_transmit_unit: begin"); 2142517904deSPeter Grehan 2143542f5d56SKevin Bowling for (int i = 0; i < sc->tx_num_queues; i++, txr++) { 2144517904deSPeter Grehan u64 bus_addr; 2145517904deSPeter Grehan caddr_t offp, endp; 2146517904deSPeter Grehan 2147542f5d56SKevin Bowling que = &sc->tx_queues[i]; 2148517904deSPeter Grehan txr = &que->txr; 2149517904deSPeter Grehan bus_addr = txr->tx_paddr; 2150517904deSPeter Grehan 2151517904deSPeter Grehan /* Clear checksum offload context. */ 2152517904deSPeter Grehan offp = (caddr_t)&txr->csum_flags; 2153517904deSPeter Grehan endp = (caddr_t)(txr + 1); 2154517904deSPeter Grehan bzero(offp, endp - offp); 2155517904deSPeter Grehan 2156517904deSPeter Grehan /* Base and Len of TX Ring */ 2157517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_TDLEN(i), 2158517904deSPeter Grehan scctx->isc_ntxd[0] * sizeof(struct igc_tx_desc)); 2159517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_TDBAH(i), 2160517904deSPeter Grehan (u32)(bus_addr >> 32)); 2161517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_TDBAL(i), 2162517904deSPeter Grehan (u32)bus_addr); 2163517904deSPeter Grehan /* Init the HEAD/TAIL indices */ 2164517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_TDT(i), 0); 2165517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_TDH(i), 0); 2166517904deSPeter Grehan 2167517904deSPeter Grehan HW_DEBUGOUT2("Base = %x, Length = %x\n", 2168542f5d56SKevin Bowling IGC_READ_REG(&sc->hw, IGC_TDBAL(i)), 2169542f5d56SKevin Bowling IGC_READ_REG(&sc->hw, IGC_TDLEN(i))); 2170517904deSPeter Grehan 2171517904deSPeter Grehan txdctl = 0; /* clear txdctl */ 2172517904deSPeter Grehan txdctl |= 0x1f; /* PTHRESH */ 2173517904deSPeter Grehan txdctl |= 1 << 8; /* HTHRESH */ 2174517904deSPeter Grehan txdctl |= 1 << 16;/* WTHRESH */ 2175517904deSPeter Grehan txdctl |= 1 << 22; /* Reserved bit 22 must always be 1 */ 2176517904deSPeter Grehan txdctl |= IGC_TXDCTL_GRAN; 2177517904deSPeter Grehan txdctl |= 1 << 25; /* LWTHRESH */ 2178517904deSPeter Grehan 2179517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_TXDCTL(i), txdctl); 2180517904deSPeter Grehan } 2181517904deSPeter Grehan 2182517904deSPeter Grehan /* Program the Transmit Control Register */ 2183542f5d56SKevin Bowling tctl = IGC_READ_REG(&sc->hw, IGC_TCTL); 2184517904deSPeter Grehan tctl &= ~IGC_TCTL_CT; 2185517904deSPeter Grehan tctl |= (IGC_TCTL_PSP | IGC_TCTL_RTLC | IGC_TCTL_EN | 2186517904deSPeter Grehan (IGC_COLLISION_THRESHOLD << IGC_CT_SHIFT)); 2187517904deSPeter Grehan 2188517904deSPeter Grehan /* This write will effectively turn on the transmit unit. */ 2189542f5d56SKevin Bowling IGC_WRITE_REG(&sc->hw, IGC_TCTL, tctl); 2190517904deSPeter Grehan } 2191517904deSPeter Grehan 2192517904deSPeter Grehan /********************************************************************* 2193517904deSPeter Grehan * 2194517904deSPeter Grehan * Enable receive unit. 2195517904deSPeter Grehan * 2196517904deSPeter Grehan **********************************************************************/ 21970eb8cd1dSKevin Bowling #define BSIZEPKT_ROUNDUP ((1<<IGC_SRRCTL_BSIZEPKT_SHIFT)-1) 2198517904deSPeter Grehan 2199517904deSPeter Grehan static void 2200517904deSPeter Grehan igc_initialize_receive_unit(if_ctx_t ctx) 2201517904deSPeter Grehan { 2202542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 2203542f5d56SKevin Bowling if_softc_ctx_t scctx = sc->shared; 2204ec22a3a2SJustin Hibbits if_t ifp = iflib_get_ifp(ctx); 2205542f5d56SKevin Bowling struct igc_hw *hw = &sc->hw; 2206517904deSPeter Grehan struct igc_rx_queue *que; 2207517904deSPeter Grehan int i; 2208517904deSPeter Grehan u32 psize, rctl, rxcsum, srrctl = 0; 2209517904deSPeter Grehan 2210517904deSPeter Grehan INIT_DEBUGOUT("igc_initialize_receive_units: begin"); 2211517904deSPeter Grehan 2212517904deSPeter Grehan /* 2213517904deSPeter Grehan * Make sure receives are disabled while setting 2214517904deSPeter Grehan * up the descriptor ring 2215517904deSPeter Grehan */ 2216517904deSPeter Grehan rctl = IGC_READ_REG(hw, IGC_RCTL); 2217517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_RCTL, rctl & ~IGC_RCTL_EN); 2218517904deSPeter Grehan 2219517904deSPeter Grehan /* Setup the Receive Control Register */ 2220517904deSPeter Grehan rctl &= ~(3 << IGC_RCTL_MO_SHIFT); 2221517904deSPeter Grehan rctl |= IGC_RCTL_EN | IGC_RCTL_BAM | 2222517904deSPeter Grehan IGC_RCTL_LBM_NO | IGC_RCTL_RDMTS_HALF | 2223517904deSPeter Grehan (hw->mac.mc_filter_type << IGC_RCTL_MO_SHIFT); 2224517904deSPeter Grehan 2225517904deSPeter Grehan /* Do not store bad packets */ 2226517904deSPeter Grehan rctl &= ~IGC_RCTL_SBP; 2227517904deSPeter Grehan 2228517904deSPeter Grehan /* Enable Long Packet receive */ 2229517904deSPeter Grehan if (if_getmtu(ifp) > ETHERMTU) 2230517904deSPeter Grehan rctl |= IGC_RCTL_LPE; 2231517904deSPeter Grehan else 2232517904deSPeter Grehan rctl &= ~IGC_RCTL_LPE; 2233517904deSPeter Grehan 2234517904deSPeter Grehan /* Strip the CRC */ 2235517904deSPeter Grehan if (!igc_disable_crc_stripping) 2236517904deSPeter Grehan rctl |= IGC_RCTL_SECRC; 2237517904deSPeter Grehan 2238517904deSPeter Grehan rxcsum = IGC_READ_REG(hw, IGC_RXCSUM); 2239517904deSPeter Grehan if (if_getcapenable(ifp) & IFCAP_RXCSUM) { 2240517904deSPeter Grehan rxcsum |= IGC_RXCSUM_CRCOFL; 2241542f5d56SKevin Bowling if (sc->tx_num_queues > 1) 2242517904deSPeter Grehan rxcsum |= IGC_RXCSUM_PCSD; 2243517904deSPeter Grehan else 2244517904deSPeter Grehan rxcsum |= IGC_RXCSUM_IPPCSE; 2245517904deSPeter Grehan } else { 2246542f5d56SKevin Bowling if (sc->tx_num_queues > 1) 2247517904deSPeter Grehan rxcsum |= IGC_RXCSUM_PCSD; 2248517904deSPeter Grehan else 2249517904deSPeter Grehan rxcsum &= ~IGC_RXCSUM_TUOFL; 2250517904deSPeter Grehan } 2251517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_RXCSUM, rxcsum); 2252517904deSPeter Grehan 2253542f5d56SKevin Bowling if (sc->rx_num_queues > 1) 2254542f5d56SKevin Bowling igc_initialize_rss_mapping(sc); 2255517904deSPeter Grehan 2256517904deSPeter Grehan if (if_getmtu(ifp) > ETHERMTU) { 2257517904deSPeter Grehan psize = scctx->isc_max_frame_size; 2258517904deSPeter Grehan /* are we on a vlan? */ 2259ec22a3a2SJustin Hibbits if (if_vlantrunkinuse(ifp)) 2260517904deSPeter Grehan psize += VLAN_TAG_SIZE; 2261542f5d56SKevin Bowling IGC_WRITE_REG(&sc->hw, IGC_RLPML, psize); 2262517904deSPeter Grehan } 2263517904deSPeter Grehan 22640eb8cd1dSKevin Bowling /* Set maximum packet buffer len */ 2265542f5d56SKevin Bowling srrctl |= (sc->rx_mbuf_sz + BSIZEPKT_ROUNDUP) >> 22660eb8cd1dSKevin Bowling IGC_SRRCTL_BSIZEPKT_SHIFT; 22670eb8cd1dSKevin Bowling /* srrctl above overrides this but set the register to a sane value */ 22680eb8cd1dSKevin Bowling rctl |= IGC_RCTL_SZ_2048; 22690eb8cd1dSKevin Bowling 2270517904deSPeter Grehan /* 2271517904deSPeter Grehan * If TX flow control is disabled and there's >1 queue defined, 2272517904deSPeter Grehan * enable DROP. 2273517904deSPeter Grehan * 2274517904deSPeter Grehan * This drops frames rather than hanging the RX MAC for all queues. 2275517904deSPeter Grehan */ 2276542f5d56SKevin Bowling if ((sc->rx_num_queues > 1) && 2277542f5d56SKevin Bowling (sc->fc == igc_fc_none || 2278542f5d56SKevin Bowling sc->fc == igc_fc_rx_pause)) { 2279517904deSPeter Grehan srrctl |= IGC_SRRCTL_DROP_EN; 2280517904deSPeter Grehan } 2281517904deSPeter Grehan 2282517904deSPeter Grehan /* Setup the Base and Length of the Rx Descriptor Rings */ 2283542f5d56SKevin Bowling for (i = 0, que = sc->rx_queues; i < sc->rx_num_queues; i++, que++) { 2284517904deSPeter Grehan struct rx_ring *rxr = &que->rxr; 2285517904deSPeter Grehan u64 bus_addr = rxr->rx_paddr; 2286517904deSPeter Grehan u32 rxdctl; 2287517904deSPeter Grehan 2288517904deSPeter Grehan #ifdef notyet 2289517904deSPeter Grehan /* Configure for header split? -- ignore for now */ 2290517904deSPeter Grehan rxr->hdr_split = igc_header_split; 2291517904deSPeter Grehan #else 2292517904deSPeter Grehan srrctl |= IGC_SRRCTL_DESCTYPE_ADV_ONEBUF; 2293517904deSPeter Grehan #endif 2294517904deSPeter Grehan 2295517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_RDLEN(i), 2296517904deSPeter Grehan scctx->isc_nrxd[0] * sizeof(struct igc_rx_desc)); 22979efc7325SKevin Bowling IGC_WRITE_REG(hw, IGC_RDBAH(i), (uint32_t)(bus_addr >> 32)); 22989efc7325SKevin Bowling IGC_WRITE_REG(hw, IGC_RDBAL(i), (uint32_t)bus_addr); 2299517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_SRRCTL(i), srrctl); 2300517904deSPeter Grehan /* Setup the Head and Tail Descriptor Pointers */ 2301517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_RDH(i), 0); 2302517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_RDT(i), 0); 2303517904deSPeter Grehan /* Enable this Queue */ 2304517904deSPeter Grehan rxdctl = IGC_READ_REG(hw, IGC_RXDCTL(i)); 2305517904deSPeter Grehan rxdctl |= IGC_RXDCTL_QUEUE_ENABLE; 2306517904deSPeter Grehan rxdctl &= 0xFFF00000; 2307517904deSPeter Grehan rxdctl |= IGC_RX_PTHRESH; 2308517904deSPeter Grehan rxdctl |= IGC_RX_HTHRESH << 8; 2309517904deSPeter Grehan rxdctl |= IGC_RX_WTHRESH << 16; 2310517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_RXDCTL(i), rxdctl); 2311517904deSPeter Grehan } 2312517904deSPeter Grehan 2313517904deSPeter Grehan /* Make sure VLAN Filters are off */ 2314517904deSPeter Grehan rctl &= ~IGC_RCTL_VFE; 2315517904deSPeter Grehan 2316517904deSPeter Grehan /* Write out the settings */ 2317517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_RCTL, rctl); 2318517904deSPeter Grehan 2319517904deSPeter Grehan return; 2320517904deSPeter Grehan } 2321517904deSPeter Grehan 2322517904deSPeter Grehan static void 23232eaef8ecSKevin Bowling igc_setup_vlan_hw_support(if_ctx_t ctx) 2324517904deSPeter Grehan { 2325542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 2326542f5d56SKevin Bowling struct igc_hw *hw = &sc->hw; 23272eaef8ecSKevin Bowling struct ifnet *ifp = iflib_get_ifp(ctx); 2328517904deSPeter Grehan u32 reg; 2329517904deSPeter Grehan 23302eaef8ecSKevin Bowling /* igc hardware doesn't seem to implement VFTA for HWFILTER */ 2331517904deSPeter Grehan 23322eaef8ecSKevin Bowling if (if_getcapenable(ifp) & IFCAP_VLAN_HWTAGGING && 23332eaef8ecSKevin Bowling !igc_disable_crc_stripping) { 2334517904deSPeter Grehan reg = IGC_READ_REG(hw, IGC_CTRL); 2335517904deSPeter Grehan reg |= IGC_CTRL_VME; 2336517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_CTRL, reg); 23372eaef8ecSKevin Bowling } else { 23382eaef8ecSKevin Bowling reg = IGC_READ_REG(hw, IGC_CTRL); 23392eaef8ecSKevin Bowling reg &= ~IGC_CTRL_VME; 23402eaef8ecSKevin Bowling IGC_WRITE_REG(hw, IGC_CTRL, reg); 23412eaef8ecSKevin Bowling } 2342517904deSPeter Grehan } 2343517904deSPeter Grehan 2344517904deSPeter Grehan static void 2345517904deSPeter Grehan igc_if_intr_enable(if_ctx_t ctx) 2346517904deSPeter Grehan { 2347542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 2348542f5d56SKevin Bowling struct igc_hw *hw = &sc->hw; 2349517904deSPeter Grehan u32 mask; 2350517904deSPeter Grehan 2351542f5d56SKevin Bowling if (__predict_true(sc->intr_type == IFLIB_INTR_MSIX)) { 2352542f5d56SKevin Bowling mask = (sc->que_mask | sc->link_mask); 2353517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_EIAC, mask); 2354517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_EIAM, mask); 2355517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_EIMS, mask); 2356517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_IMS, IGC_IMS_LSC); 2357517904deSPeter Grehan } else 2358517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_IMS, IMS_ENABLE_MASK); 2359517904deSPeter Grehan IGC_WRITE_FLUSH(hw); 2360517904deSPeter Grehan } 2361517904deSPeter Grehan 2362517904deSPeter Grehan static void 2363517904deSPeter Grehan igc_if_intr_disable(if_ctx_t ctx) 2364517904deSPeter Grehan { 2365542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 2366542f5d56SKevin Bowling struct igc_hw *hw = &sc->hw; 2367517904deSPeter Grehan 2368542f5d56SKevin Bowling if (__predict_true(sc->intr_type == IFLIB_INTR_MSIX)) { 2369517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_EIMC, 0xffffffff); 2370517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_EIAC, 0); 2371517904deSPeter Grehan } 2372517904deSPeter Grehan IGC_WRITE_REG(hw, IGC_IMC, 0xffffffff); 2373517904deSPeter Grehan IGC_WRITE_FLUSH(hw); 2374517904deSPeter Grehan } 2375517904deSPeter Grehan 2376517904deSPeter Grehan /* 2377517904deSPeter Grehan * igc_get_hw_control sets the {CTRL_EXT|FWSM}:DRV_LOAD bit. 2378517904deSPeter Grehan * For ASF and Pass Through versions of f/w this means 2379517904deSPeter Grehan * that the driver is loaded. For AMT version type f/w 2380517904deSPeter Grehan * this means that the network i/f is open. 2381517904deSPeter Grehan */ 2382517904deSPeter Grehan static void 2383542f5d56SKevin Bowling igc_get_hw_control(struct igc_softc *sc) 2384517904deSPeter Grehan { 2385517904deSPeter Grehan u32 ctrl_ext; 2386517904deSPeter Grehan 2387542f5d56SKevin Bowling if (sc->vf_ifp) 2388517904deSPeter Grehan return; 2389517904deSPeter Grehan 2390542f5d56SKevin Bowling ctrl_ext = IGC_READ_REG(&sc->hw, IGC_CTRL_EXT); 2391542f5d56SKevin Bowling IGC_WRITE_REG(&sc->hw, IGC_CTRL_EXT, 2392517904deSPeter Grehan ctrl_ext | IGC_CTRL_EXT_DRV_LOAD); 2393517904deSPeter Grehan } 2394517904deSPeter Grehan 2395517904deSPeter Grehan /* 2396517904deSPeter Grehan * igc_release_hw_control resets {CTRL_EXT|FWSM}:DRV_LOAD bit. 2397517904deSPeter Grehan * For ASF and Pass Through versions of f/w this means that 2398517904deSPeter Grehan * the driver is no longer loaded. For AMT versions of the 2399517904deSPeter Grehan * f/w this means that the network i/f is closed. 2400517904deSPeter Grehan */ 2401517904deSPeter Grehan static void 2402542f5d56SKevin Bowling igc_release_hw_control(struct igc_softc *sc) 2403517904deSPeter Grehan { 2404517904deSPeter Grehan u32 ctrl_ext; 2405517904deSPeter Grehan 2406542f5d56SKevin Bowling ctrl_ext = IGC_READ_REG(&sc->hw, IGC_CTRL_EXT); 2407542f5d56SKevin Bowling IGC_WRITE_REG(&sc->hw, IGC_CTRL_EXT, 2408517904deSPeter Grehan ctrl_ext & ~IGC_CTRL_EXT_DRV_LOAD); 2409517904deSPeter Grehan return; 2410517904deSPeter Grehan } 2411517904deSPeter Grehan 2412517904deSPeter Grehan static int 2413517904deSPeter Grehan igc_is_valid_ether_addr(u8 *addr) 2414517904deSPeter Grehan { 2415517904deSPeter Grehan char zero_addr[6] = { 0, 0, 0, 0, 0, 0 }; 2416517904deSPeter Grehan 2417517904deSPeter Grehan if ((addr[0] & 1) || (!bcmp(addr, zero_addr, ETHER_ADDR_LEN))) { 2418517904deSPeter Grehan return (false); 2419517904deSPeter Grehan } 2420517904deSPeter Grehan 2421517904deSPeter Grehan return (true); 2422517904deSPeter Grehan } 2423517904deSPeter Grehan 2424517904deSPeter Grehan /* 2425517904deSPeter Grehan ** Parse the interface capabilities with regard 2426517904deSPeter Grehan ** to both system management and wake-on-lan for 2427517904deSPeter Grehan ** later use. 2428517904deSPeter Grehan */ 2429517904deSPeter Grehan static void 2430517904deSPeter Grehan igc_get_wakeup(if_ctx_t ctx) 2431517904deSPeter Grehan { 2432542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 2433517904deSPeter Grehan u16 eeprom_data = 0, apme_mask; 2434517904deSPeter Grehan 2435517904deSPeter Grehan apme_mask = IGC_WUC_APME; 2436542f5d56SKevin Bowling eeprom_data = IGC_READ_REG(&sc->hw, IGC_WUC); 2437517904deSPeter Grehan 2438517904deSPeter Grehan if (eeprom_data & apme_mask) 2439542f5d56SKevin Bowling sc->wol = IGC_WUFC_LNKC; 2440517904deSPeter Grehan } 2441517904deSPeter Grehan 2442517904deSPeter Grehan 2443517904deSPeter Grehan /* 2444517904deSPeter Grehan * Enable PCI Wake On Lan capability 2445517904deSPeter Grehan */ 2446517904deSPeter Grehan static void 2447517904deSPeter Grehan igc_enable_wakeup(if_ctx_t ctx) 2448517904deSPeter Grehan { 2449542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 2450517904deSPeter Grehan device_t dev = iflib_get_dev(ctx); 2451517904deSPeter Grehan if_t ifp = iflib_get_ifp(ctx); 2452517904deSPeter Grehan int error = 0; 2453517904deSPeter Grehan u32 pmc, ctrl, rctl; 2454517904deSPeter Grehan u16 status; 2455517904deSPeter Grehan 2456517904deSPeter Grehan if (pci_find_cap(dev, PCIY_PMG, &pmc) != 0) 2457517904deSPeter Grehan return; 2458517904deSPeter Grehan 2459517904deSPeter Grehan /* 2460517904deSPeter Grehan * Determine type of Wakeup: note that wol 2461517904deSPeter Grehan * is set with all bits on by default. 2462517904deSPeter Grehan */ 2463517904deSPeter Grehan if ((if_getcapenable(ifp) & IFCAP_WOL_MAGIC) == 0) 2464542f5d56SKevin Bowling sc->wol &= ~IGC_WUFC_MAG; 2465517904deSPeter Grehan 2466517904deSPeter Grehan if ((if_getcapenable(ifp) & IFCAP_WOL_UCAST) == 0) 2467542f5d56SKevin Bowling sc->wol &= ~IGC_WUFC_EX; 2468517904deSPeter Grehan 2469517904deSPeter Grehan if ((if_getcapenable(ifp) & IFCAP_WOL_MCAST) == 0) 2470542f5d56SKevin Bowling sc->wol &= ~IGC_WUFC_MC; 2471517904deSPeter Grehan else { 2472542f5d56SKevin Bowling rctl = IGC_READ_REG(&sc->hw, IGC_RCTL); 2473517904deSPeter Grehan rctl |= IGC_RCTL_MPE; 2474542f5d56SKevin Bowling IGC_WRITE_REG(&sc->hw, IGC_RCTL, rctl); 2475517904deSPeter Grehan } 2476517904deSPeter Grehan 2477542f5d56SKevin Bowling if (!(sc->wol & (IGC_WUFC_EX | IGC_WUFC_MAG | IGC_WUFC_MC))) 2478517904deSPeter Grehan goto pme; 2479517904deSPeter Grehan 2480517904deSPeter Grehan /* Advertise the wakeup capability */ 2481542f5d56SKevin Bowling ctrl = IGC_READ_REG(&sc->hw, IGC_CTRL); 2482517904deSPeter Grehan ctrl |= IGC_CTRL_ADVD3WUC; 2483542f5d56SKevin Bowling IGC_WRITE_REG(&sc->hw, IGC_CTRL, ctrl); 2484517904deSPeter Grehan 2485517904deSPeter Grehan /* Enable wakeup by the MAC */ 2486542f5d56SKevin Bowling IGC_WRITE_REG(&sc->hw, IGC_WUC, IGC_WUC_PME_EN); 2487542f5d56SKevin Bowling IGC_WRITE_REG(&sc->hw, IGC_WUFC, sc->wol); 2488517904deSPeter Grehan 2489517904deSPeter Grehan pme: 2490517904deSPeter Grehan status = pci_read_config(dev, pmc + PCIR_POWER_STATUS, 2); 2491517904deSPeter Grehan status &= ~(PCIM_PSTAT_PME | PCIM_PSTAT_PMEENABLE); 2492517904deSPeter Grehan if (!error && (if_getcapenable(ifp) & IFCAP_WOL)) 2493517904deSPeter Grehan status |= PCIM_PSTAT_PME | PCIM_PSTAT_PMEENABLE; 2494517904deSPeter Grehan pci_write_config(dev, pmc + PCIR_POWER_STATUS, status, 2); 2495517904deSPeter Grehan 2496517904deSPeter Grehan return; 2497517904deSPeter Grehan } 2498517904deSPeter Grehan 2499517904deSPeter Grehan /********************************************************************** 2500517904deSPeter Grehan * 2501517904deSPeter Grehan * Update the board statistics counters. 2502517904deSPeter Grehan * 2503517904deSPeter Grehan **********************************************************************/ 2504517904deSPeter Grehan static void 2505542f5d56SKevin Bowling igc_update_stats_counters(struct igc_softc *sc) 2506517904deSPeter Grehan { 2507542f5d56SKevin Bowling u64 prev_xoffrxc = sc->stats.xoffrxc; 2508517904deSPeter Grehan 2509542f5d56SKevin Bowling sc->stats.crcerrs += IGC_READ_REG(&sc->hw, IGC_CRCERRS); 2510542f5d56SKevin Bowling sc->stats.mpc += IGC_READ_REG(&sc->hw, IGC_MPC); 2511542f5d56SKevin Bowling sc->stats.scc += IGC_READ_REG(&sc->hw, IGC_SCC); 2512542f5d56SKevin Bowling sc->stats.ecol += IGC_READ_REG(&sc->hw, IGC_ECOL); 2513517904deSPeter Grehan 2514542f5d56SKevin Bowling sc->stats.mcc += IGC_READ_REG(&sc->hw, IGC_MCC); 2515542f5d56SKevin Bowling sc->stats.latecol += IGC_READ_REG(&sc->hw, IGC_LATECOL); 2516542f5d56SKevin Bowling sc->stats.colc += IGC_READ_REG(&sc->hw, IGC_COLC); 2517542f5d56SKevin Bowling sc->stats.colc += IGC_READ_REG(&sc->hw, IGC_RERC); 2518542f5d56SKevin Bowling sc->stats.dc += IGC_READ_REG(&sc->hw, IGC_DC); 2519542f5d56SKevin Bowling sc->stats.rlec += IGC_READ_REG(&sc->hw, IGC_RLEC); 2520542f5d56SKevin Bowling sc->stats.xonrxc += IGC_READ_REG(&sc->hw, IGC_XONRXC); 2521542f5d56SKevin Bowling sc->stats.xontxc += IGC_READ_REG(&sc->hw, IGC_XONTXC); 2522542f5d56SKevin Bowling sc->stats.xoffrxc += IGC_READ_REG(&sc->hw, IGC_XOFFRXC); 2523517904deSPeter Grehan /* 2524517904deSPeter Grehan * For watchdog management we need to know if we have been 2525517904deSPeter Grehan * paused during the last interval, so capture that here. 2526517904deSPeter Grehan */ 2527542f5d56SKevin Bowling if (sc->stats.xoffrxc != prev_xoffrxc) 2528542f5d56SKevin Bowling sc->shared->isc_pause_frames = 1; 2529542f5d56SKevin Bowling sc->stats.xofftxc += IGC_READ_REG(&sc->hw, IGC_XOFFTXC); 2530542f5d56SKevin Bowling sc->stats.fcruc += IGC_READ_REG(&sc->hw, IGC_FCRUC); 2531542f5d56SKevin Bowling sc->stats.prc64 += IGC_READ_REG(&sc->hw, IGC_PRC64); 2532542f5d56SKevin Bowling sc->stats.prc127 += IGC_READ_REG(&sc->hw, IGC_PRC127); 2533542f5d56SKevin Bowling sc->stats.prc255 += IGC_READ_REG(&sc->hw, IGC_PRC255); 2534542f5d56SKevin Bowling sc->stats.prc511 += IGC_READ_REG(&sc->hw, IGC_PRC511); 2535542f5d56SKevin Bowling sc->stats.prc1023 += IGC_READ_REG(&sc->hw, IGC_PRC1023); 2536542f5d56SKevin Bowling sc->stats.prc1522 += IGC_READ_REG(&sc->hw, IGC_PRC1522); 2537542f5d56SKevin Bowling sc->stats.tlpic += IGC_READ_REG(&sc->hw, IGC_TLPIC); 2538542f5d56SKevin Bowling sc->stats.rlpic += IGC_READ_REG(&sc->hw, IGC_RLPIC); 2539542f5d56SKevin Bowling sc->stats.gprc += IGC_READ_REG(&sc->hw, IGC_GPRC); 2540542f5d56SKevin Bowling sc->stats.bprc += IGC_READ_REG(&sc->hw, IGC_BPRC); 2541542f5d56SKevin Bowling sc->stats.mprc += IGC_READ_REG(&sc->hw, IGC_MPRC); 2542542f5d56SKevin Bowling sc->stats.gptc += IGC_READ_REG(&sc->hw, IGC_GPTC); 2543517904deSPeter Grehan 2544517904deSPeter Grehan /* For the 64-bit byte counters the low dword must be read first. */ 2545517904deSPeter Grehan /* Both registers clear on the read of the high dword */ 2546517904deSPeter Grehan 2547542f5d56SKevin Bowling sc->stats.gorc += IGC_READ_REG(&sc->hw, IGC_GORCL) + 2548542f5d56SKevin Bowling ((u64)IGC_READ_REG(&sc->hw, IGC_GORCH) << 32); 2549542f5d56SKevin Bowling sc->stats.gotc += IGC_READ_REG(&sc->hw, IGC_GOTCL) + 2550542f5d56SKevin Bowling ((u64)IGC_READ_REG(&sc->hw, IGC_GOTCH) << 32); 2551517904deSPeter Grehan 2552542f5d56SKevin Bowling sc->stats.rnbc += IGC_READ_REG(&sc->hw, IGC_RNBC); 2553542f5d56SKevin Bowling sc->stats.ruc += IGC_READ_REG(&sc->hw, IGC_RUC); 2554542f5d56SKevin Bowling sc->stats.rfc += IGC_READ_REG(&sc->hw, IGC_RFC); 2555542f5d56SKevin Bowling sc->stats.roc += IGC_READ_REG(&sc->hw, IGC_ROC); 2556542f5d56SKevin Bowling sc->stats.rjc += IGC_READ_REG(&sc->hw, IGC_RJC); 2557517904deSPeter Grehan 2558542f5d56SKevin Bowling sc->stats.mgprc += IGC_READ_REG(&sc->hw, IGC_MGTPRC); 2559542f5d56SKevin Bowling sc->stats.mgpdc += IGC_READ_REG(&sc->hw, IGC_MGTPDC); 2560542f5d56SKevin Bowling sc->stats.mgptc += IGC_READ_REG(&sc->hw, IGC_MGTPTC); 256109526a77SKevin Bowling 2562542f5d56SKevin Bowling sc->stats.tor += IGC_READ_REG(&sc->hw, IGC_TORH); 2563542f5d56SKevin Bowling sc->stats.tot += IGC_READ_REG(&sc->hw, IGC_TOTH); 2564517904deSPeter Grehan 2565542f5d56SKevin Bowling sc->stats.tpr += IGC_READ_REG(&sc->hw, IGC_TPR); 2566542f5d56SKevin Bowling sc->stats.tpt += IGC_READ_REG(&sc->hw, IGC_TPT); 2567542f5d56SKevin Bowling sc->stats.ptc64 += IGC_READ_REG(&sc->hw, IGC_PTC64); 2568542f5d56SKevin Bowling sc->stats.ptc127 += IGC_READ_REG(&sc->hw, IGC_PTC127); 2569542f5d56SKevin Bowling sc->stats.ptc255 += IGC_READ_REG(&sc->hw, IGC_PTC255); 2570542f5d56SKevin Bowling sc->stats.ptc511 += IGC_READ_REG(&sc->hw, IGC_PTC511); 2571542f5d56SKevin Bowling sc->stats.ptc1023 += IGC_READ_REG(&sc->hw, IGC_PTC1023); 2572542f5d56SKevin Bowling sc->stats.ptc1522 += IGC_READ_REG(&sc->hw, IGC_PTC1522); 2573542f5d56SKevin Bowling sc->stats.mptc += IGC_READ_REG(&sc->hw, IGC_MPTC); 2574542f5d56SKevin Bowling sc->stats.bptc += IGC_READ_REG(&sc->hw, IGC_BPTC); 2575517904deSPeter Grehan 2576517904deSPeter Grehan /* Interrupt Counts */ 2577542f5d56SKevin Bowling sc->stats.iac += IGC_READ_REG(&sc->hw, IGC_IAC); 2578542f5d56SKevin Bowling sc->stats.rxdmtc += IGC_READ_REG(&sc->hw, IGC_RXDMTC); 2579517904deSPeter Grehan 2580542f5d56SKevin Bowling sc->stats.algnerrc += IGC_READ_REG(&sc->hw, IGC_ALGNERRC); 2581542f5d56SKevin Bowling sc->stats.tncrs += IGC_READ_REG(&sc->hw, IGC_TNCRS); 2582542f5d56SKevin Bowling sc->stats.htdpmc += IGC_READ_REG(&sc->hw, IGC_HTDPMC); 2583542f5d56SKevin Bowling sc->stats.tsctc += IGC_READ_REG(&sc->hw, IGC_TSCTC); 2584517904deSPeter Grehan } 2585517904deSPeter Grehan 2586517904deSPeter Grehan static uint64_t 2587517904deSPeter Grehan igc_if_get_counter(if_ctx_t ctx, ift_counter cnt) 2588517904deSPeter Grehan { 2589542f5d56SKevin Bowling struct igc_softc *sc = iflib_get_softc(ctx); 2590ec22a3a2SJustin Hibbits if_t ifp = iflib_get_ifp(ctx); 2591517904deSPeter Grehan 2592517904deSPeter Grehan switch (cnt) { 2593517904deSPeter Grehan case IFCOUNTER_COLLISIONS: 2594542f5d56SKevin Bowling return (sc->stats.colc); 2595517904deSPeter Grehan case IFCOUNTER_IERRORS: 2596542f5d56SKevin Bowling return (sc->dropped_pkts + sc->stats.rxerrc + 2597542f5d56SKevin Bowling sc->stats.crcerrs + sc->stats.algnerrc + 2598542f5d56SKevin Bowling sc->stats.ruc + sc->stats.roc + 2599542f5d56SKevin Bowling sc->stats.mpc + sc->stats.htdpmc); 2600517904deSPeter Grehan case IFCOUNTER_OERRORS: 2601542f5d56SKevin Bowling return (sc->stats.ecol + sc->stats.latecol + 2602542f5d56SKevin Bowling sc->watchdog_events); 2603517904deSPeter Grehan default: 2604517904deSPeter Grehan return (if_get_counter_default(ifp, cnt)); 2605517904deSPeter Grehan } 2606517904deSPeter Grehan } 2607517904deSPeter Grehan 2608517904deSPeter Grehan /* igc_if_needs_restart - Tell iflib when the driver needs to be reinitialized 2609517904deSPeter Grehan * @ctx: iflib context 2610517904deSPeter Grehan * @event: event code to check 2611517904deSPeter Grehan * 2612725e4008SKevin Bowling * Defaults to returning false for unknown events. 2613517904deSPeter Grehan * 2614517904deSPeter Grehan * @returns true if iflib needs to reinit the interface 2615517904deSPeter Grehan */ 2616517904deSPeter Grehan static bool 2617517904deSPeter Grehan igc_if_needs_restart(if_ctx_t ctx __unused, enum iflib_restart_event event) 2618517904deSPeter Grehan { 2619517904deSPeter Grehan switch (event) { 2620517904deSPeter Grehan case IFLIB_RESTART_VLAN_CONFIG: 2621517904deSPeter Grehan default: 2622725e4008SKevin Bowling return (false); 2623517904deSPeter Grehan } 2624517904deSPeter Grehan } 2625517904deSPeter Grehan 2626517904deSPeter Grehan /* Export a single 32-bit register via a read-only sysctl. */ 2627517904deSPeter Grehan static int 2628517904deSPeter Grehan igc_sysctl_reg_handler(SYSCTL_HANDLER_ARGS) 2629517904deSPeter Grehan { 2630542f5d56SKevin Bowling struct igc_softc *sc; 2631517904deSPeter Grehan u_int val; 2632517904deSPeter Grehan 2633542f5d56SKevin Bowling sc = oidp->oid_arg1; 2634542f5d56SKevin Bowling val = IGC_READ_REG(&sc->hw, oidp->oid_arg2); 2635517904deSPeter Grehan return (sysctl_handle_int(oidp, &val, 0, req)); 2636517904deSPeter Grehan } 2637517904deSPeter Grehan 2638bc9402abSKevin Bowling /* Per queue holdoff interrupt rate handler */ 2639bc9402abSKevin Bowling static int 2640bc9402abSKevin Bowling igc_sysctl_interrupt_rate_handler(SYSCTL_HANDLER_ARGS) 2641bc9402abSKevin Bowling { 2642bc9402abSKevin Bowling struct igc_rx_queue *rque; 2643bc9402abSKevin Bowling struct igc_tx_queue *tque; 2644bc9402abSKevin Bowling struct igc_hw *hw; 2645bc9402abSKevin Bowling int error; 2646bc9402abSKevin Bowling u32 reg, usec, rate; 2647bc9402abSKevin Bowling 2648bc9402abSKevin Bowling bool tx = oidp->oid_arg2; 2649bc9402abSKevin Bowling 2650bc9402abSKevin Bowling if (tx) { 2651bc9402abSKevin Bowling tque = oidp->oid_arg1; 2652542f5d56SKevin Bowling hw = &tque->sc->hw; 2653bc9402abSKevin Bowling reg = IGC_READ_REG(hw, IGC_EITR(tque->me)); 2654bc9402abSKevin Bowling } else { 2655bc9402abSKevin Bowling rque = oidp->oid_arg1; 2656542f5d56SKevin Bowling hw = &rque->sc->hw; 2657bc9402abSKevin Bowling reg = IGC_READ_REG(hw, IGC_EITR(rque->msix)); 2658bc9402abSKevin Bowling } 2659bc9402abSKevin Bowling 2660bc9402abSKevin Bowling usec = (reg & IGC_QVECTOR_MASK); 2661bc9402abSKevin Bowling if (usec > 0) 2662bc9402abSKevin Bowling rate = IGC_INTS_TO_EITR(usec); 2663bc9402abSKevin Bowling else 2664bc9402abSKevin Bowling rate = 0; 2665bc9402abSKevin Bowling 2666bc9402abSKevin Bowling error = sysctl_handle_int(oidp, &rate, 0, req); 2667bc9402abSKevin Bowling if (error || !req->newptr) 2668bc9402abSKevin Bowling return error; 2669bc9402abSKevin Bowling return 0; 2670bc9402abSKevin Bowling } 2671bc9402abSKevin Bowling 2672517904deSPeter Grehan /* 2673517904deSPeter Grehan * Add sysctl variables, one per statistic, to the system. 2674517904deSPeter Grehan */ 2675517904deSPeter Grehan static void 2676542f5d56SKevin Bowling igc_add_hw_stats(struct igc_softc *sc) 2677517904deSPeter Grehan { 2678542f5d56SKevin Bowling device_t dev = iflib_get_dev(sc->ctx); 2679542f5d56SKevin Bowling struct igc_tx_queue *tx_que = sc->tx_queues; 2680542f5d56SKevin Bowling struct igc_rx_queue *rx_que = sc->rx_queues; 2681517904deSPeter Grehan 2682517904deSPeter Grehan struct sysctl_ctx_list *ctx = device_get_sysctl_ctx(dev); 2683517904deSPeter Grehan struct sysctl_oid *tree = device_get_sysctl_tree(dev); 2684517904deSPeter Grehan struct sysctl_oid_list *child = SYSCTL_CHILDREN(tree); 2685542f5d56SKevin Bowling struct igc_hw_stats *stats = &sc->stats; 2686517904deSPeter Grehan 2687517904deSPeter Grehan struct sysctl_oid *stat_node, *queue_node, *int_node; 2688517904deSPeter Grehan struct sysctl_oid_list *stat_list, *queue_list, *int_list; 2689517904deSPeter Grehan 2690517904deSPeter Grehan #define QUEUE_NAME_LEN 32 2691517904deSPeter Grehan char namebuf[QUEUE_NAME_LEN]; 2692517904deSPeter Grehan 2693517904deSPeter Grehan /* Driver Statistics */ 2694517904deSPeter Grehan SYSCTL_ADD_ULONG(ctx, child, OID_AUTO, "dropped", 2695542f5d56SKevin Bowling CTLFLAG_RD, &sc->dropped_pkts, 2696517904deSPeter Grehan "Driver dropped packets"); 2697517904deSPeter Grehan SYSCTL_ADD_ULONG(ctx, child, OID_AUTO, "link_irq", 2698542f5d56SKevin Bowling CTLFLAG_RD, &sc->link_irq, 2699517904deSPeter Grehan "Link MSI-X IRQ Handled"); 2700517904deSPeter Grehan SYSCTL_ADD_ULONG(ctx, child, OID_AUTO, "rx_overruns", 2701542f5d56SKevin Bowling CTLFLAG_RD, &sc->rx_overruns, 2702517904deSPeter Grehan "RX overruns"); 2703517904deSPeter Grehan SYSCTL_ADD_ULONG(ctx, child, OID_AUTO, "watchdog_timeouts", 2704542f5d56SKevin Bowling CTLFLAG_RD, &sc->watchdog_events, 2705517904deSPeter Grehan "Watchdog timeouts"); 2706517904deSPeter Grehan SYSCTL_ADD_PROC(ctx, child, OID_AUTO, "device_control", 2707517904deSPeter Grehan CTLTYPE_UINT | CTLFLAG_RD | CTLFLAG_NEEDGIANT, 2708542f5d56SKevin Bowling sc, IGC_CTRL, igc_sysctl_reg_handler, "IU", 2709517904deSPeter Grehan "Device Control Register"); 2710517904deSPeter Grehan SYSCTL_ADD_PROC(ctx, child, OID_AUTO, "rx_control", 2711517904deSPeter Grehan CTLTYPE_UINT | CTLFLAG_RD | CTLFLAG_NEEDGIANT, 2712542f5d56SKevin Bowling sc, IGC_RCTL, igc_sysctl_reg_handler, "IU", 2713517904deSPeter Grehan "Receiver Control Register"); 2714517904deSPeter Grehan SYSCTL_ADD_UINT(ctx, child, OID_AUTO, "fc_high_water", 2715542f5d56SKevin Bowling CTLFLAG_RD, &sc->hw.fc.high_water, 0, 2716517904deSPeter Grehan "Flow Control High Watermark"); 2717517904deSPeter Grehan SYSCTL_ADD_UINT(ctx, child, OID_AUTO, "fc_low_water", 2718542f5d56SKevin Bowling CTLFLAG_RD, &sc->hw.fc.low_water, 0, 2719517904deSPeter Grehan "Flow Control Low Watermark"); 2720517904deSPeter Grehan 2721542f5d56SKevin Bowling for (int i = 0; i < sc->tx_num_queues; i++, tx_que++) { 2722517904deSPeter Grehan struct tx_ring *txr = &tx_que->txr; 2723517904deSPeter Grehan snprintf(namebuf, QUEUE_NAME_LEN, "queue_tx_%d", i); 2724517904deSPeter Grehan queue_node = SYSCTL_ADD_NODE(ctx, child, OID_AUTO, namebuf, 2725517904deSPeter Grehan CTLFLAG_RD | CTLFLAG_MPSAFE, NULL, "TX Queue Name"); 2726517904deSPeter Grehan queue_list = SYSCTL_CHILDREN(queue_node); 2727517904deSPeter Grehan 2728bc9402abSKevin Bowling SYSCTL_ADD_PROC(ctx, queue_list, OID_AUTO, "interrupt_rate", 2729bc9402abSKevin Bowling CTLTYPE_UINT | CTLFLAG_RD, tx_que, 2730bc9402abSKevin Bowling true, igc_sysctl_interrupt_rate_handler, "IU", 2731bc9402abSKevin Bowling "Interrupt Rate"); 2732517904deSPeter Grehan SYSCTL_ADD_PROC(ctx, queue_list, OID_AUTO, "txd_head", 2733542f5d56SKevin Bowling CTLTYPE_UINT | CTLFLAG_RD | CTLFLAG_NEEDGIANT, sc, 2734517904deSPeter Grehan IGC_TDH(txr->me), igc_sysctl_reg_handler, "IU", 2735517904deSPeter Grehan "Transmit Descriptor Head"); 2736517904deSPeter Grehan SYSCTL_ADD_PROC(ctx, queue_list, OID_AUTO, "txd_tail", 2737542f5d56SKevin Bowling CTLTYPE_UINT | CTLFLAG_RD | CTLFLAG_NEEDGIANT, sc, 2738517904deSPeter Grehan IGC_TDT(txr->me), igc_sysctl_reg_handler, "IU", 2739517904deSPeter Grehan "Transmit Descriptor Tail"); 2740517904deSPeter Grehan SYSCTL_ADD_ULONG(ctx, queue_list, OID_AUTO, "tx_irq", 2741517904deSPeter Grehan CTLFLAG_RD, &txr->tx_irq, 2742517904deSPeter Grehan "Queue MSI-X Transmit Interrupts"); 2743517904deSPeter Grehan } 2744517904deSPeter Grehan 2745542f5d56SKevin Bowling for (int j = 0; j < sc->rx_num_queues; j++, rx_que++) { 2746517904deSPeter Grehan struct rx_ring *rxr = &rx_que->rxr; 2747517904deSPeter Grehan snprintf(namebuf, QUEUE_NAME_LEN, "queue_rx_%d", j); 2748517904deSPeter Grehan queue_node = SYSCTL_ADD_NODE(ctx, child, OID_AUTO, namebuf, 2749517904deSPeter Grehan CTLFLAG_RD | CTLFLAG_MPSAFE, NULL, "RX Queue Name"); 2750517904deSPeter Grehan queue_list = SYSCTL_CHILDREN(queue_node); 2751517904deSPeter Grehan 2752bc9402abSKevin Bowling SYSCTL_ADD_PROC(ctx, queue_list, OID_AUTO, "interrupt_rate", 2753bc9402abSKevin Bowling CTLTYPE_UINT | CTLFLAG_RD, rx_que, 2754bc9402abSKevin Bowling false, igc_sysctl_interrupt_rate_handler, "IU", 2755bc9402abSKevin Bowling "Interrupt Rate"); 2756517904deSPeter Grehan SYSCTL_ADD_PROC(ctx, queue_list, OID_AUTO, "rxd_head", 2757542f5d56SKevin Bowling CTLTYPE_UINT | CTLFLAG_RD | CTLFLAG_NEEDGIANT, sc, 2758517904deSPeter Grehan IGC_RDH(rxr->me), igc_sysctl_reg_handler, "IU", 2759517904deSPeter Grehan "Receive Descriptor Head"); 2760517904deSPeter Grehan SYSCTL_ADD_PROC(ctx, queue_list, OID_AUTO, "rxd_tail", 2761542f5d56SKevin Bowling CTLTYPE_UINT | CTLFLAG_RD | CTLFLAG_NEEDGIANT, sc, 2762517904deSPeter Grehan IGC_RDT(rxr->me), igc_sysctl_reg_handler, "IU", 2763517904deSPeter Grehan "Receive Descriptor Tail"); 2764517904deSPeter Grehan SYSCTL_ADD_ULONG(ctx, queue_list, OID_AUTO, "rx_irq", 2765517904deSPeter Grehan CTLFLAG_RD, &rxr->rx_irq, 2766517904deSPeter Grehan "Queue MSI-X Receive Interrupts"); 2767517904deSPeter Grehan } 2768517904deSPeter Grehan 2769517904deSPeter Grehan /* MAC stats get their own sub node */ 2770517904deSPeter Grehan stat_node = SYSCTL_ADD_NODE(ctx, child, OID_AUTO, "mac_stats", 2771517904deSPeter Grehan CTLFLAG_RD | CTLFLAG_MPSAFE, NULL, "Statistics"); 2772517904deSPeter Grehan stat_list = SYSCTL_CHILDREN(stat_node); 2773517904deSPeter Grehan 2774517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "excess_coll", 2775517904deSPeter Grehan CTLFLAG_RD, &stats->ecol, 2776517904deSPeter Grehan "Excessive collisions"); 2777517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "single_coll", 2778517904deSPeter Grehan CTLFLAG_RD, &stats->scc, 2779517904deSPeter Grehan "Single collisions"); 2780517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "multiple_coll", 2781517904deSPeter Grehan CTLFLAG_RD, &stats->mcc, 2782517904deSPeter Grehan "Multiple collisions"); 2783517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "late_coll", 2784517904deSPeter Grehan CTLFLAG_RD, &stats->latecol, 2785517904deSPeter Grehan "Late collisions"); 2786517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "collision_count", 2787517904deSPeter Grehan CTLFLAG_RD, &stats->colc, 2788517904deSPeter Grehan "Collision Count"); 2789517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "symbol_errors", 2790542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.symerrs, 2791517904deSPeter Grehan "Symbol Errors"); 2792517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "sequence_errors", 2793542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.sec, 2794517904deSPeter Grehan "Sequence Errors"); 2795517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "defer_count", 2796542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.dc, 2797517904deSPeter Grehan "Defer Count"); 2798517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "missed_packets", 2799542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.mpc, 2800517904deSPeter Grehan "Missed Packets"); 280109526a77SKevin Bowling SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "recv_length_errors", 2802542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.rlec, 280309526a77SKevin Bowling "Receive Length Errors"); 2804517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "recv_no_buff", 2805542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.rnbc, 2806517904deSPeter Grehan "Receive No Buffers"); 2807517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "recv_undersize", 2808542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.ruc, 2809517904deSPeter Grehan "Receive Undersize"); 2810517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "recv_fragmented", 2811542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.rfc, 2812517904deSPeter Grehan "Fragmented Packets Received "); 2813517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "recv_oversize", 2814542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.roc, 2815517904deSPeter Grehan "Oversized Packets Received"); 2816517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "recv_jabber", 2817542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.rjc, 2818517904deSPeter Grehan "Recevied Jabber"); 2819517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "recv_errs", 2820542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.rxerrc, 2821517904deSPeter Grehan "Receive Errors"); 2822517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "crc_errs", 2823542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.crcerrs, 2824517904deSPeter Grehan "CRC errors"); 2825517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "alignment_errs", 2826542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.algnerrc, 2827517904deSPeter Grehan "Alignment Errors"); 2828517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "xon_recvd", 2829542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.xonrxc, 2830517904deSPeter Grehan "XON Received"); 2831517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "xon_txd", 2832542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.xontxc, 2833517904deSPeter Grehan "XON Transmitted"); 2834517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "xoff_recvd", 2835542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.xoffrxc, 2836517904deSPeter Grehan "XOFF Received"); 2837517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "xoff_txd", 2838542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.xofftxc, 2839517904deSPeter Grehan "XOFF Transmitted"); 284009526a77SKevin Bowling SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "unsupported_fc_recvd", 2841542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.fcruc, 284209526a77SKevin Bowling "Unsupported Flow Control Received"); 284309526a77SKevin Bowling SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "mgmt_pkts_recvd", 2844542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.mgprc, 284509526a77SKevin Bowling "Management Packets Received"); 284609526a77SKevin Bowling SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "mgmt_pkts_drop", 2847542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.mgpdc, 284809526a77SKevin Bowling "Management Packets Dropped"); 284909526a77SKevin Bowling SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "mgmt_pkts_txd", 2850542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.mgptc, 285109526a77SKevin Bowling "Management Packets Transmitted"); 2852517904deSPeter Grehan 2853517904deSPeter Grehan /* Packet Reception Stats */ 2854517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "total_pkts_recvd", 2855542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.tpr, 2856517904deSPeter Grehan "Total Packets Received "); 2857517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "good_pkts_recvd", 2858542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.gprc, 2859517904deSPeter Grehan "Good Packets Received"); 2860517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "bcast_pkts_recvd", 2861542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.bprc, 2862517904deSPeter Grehan "Broadcast Packets Received"); 2863517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "mcast_pkts_recvd", 2864542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.mprc, 2865517904deSPeter Grehan "Multicast Packets Received"); 2866517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_frames_64", 2867542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.prc64, 2868517904deSPeter Grehan "64 byte frames received "); 2869517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_frames_65_127", 2870542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.prc127, 2871517904deSPeter Grehan "65-127 byte frames received"); 2872517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_frames_128_255", 2873542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.prc255, 2874517904deSPeter Grehan "128-255 byte frames received"); 2875517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_frames_256_511", 2876542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.prc511, 2877517904deSPeter Grehan "256-511 byte frames received"); 2878517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_frames_512_1023", 2879542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.prc1023, 2880517904deSPeter Grehan "512-1023 byte frames received"); 2881517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_frames_1024_1522", 2882542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.prc1522, 2883517904deSPeter Grehan "1023-1522 byte frames received"); 2884517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "good_octets_recvd", 2885542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.gorc, 2886517904deSPeter Grehan "Good Octets Received"); 2887517904deSPeter Grehan 2888517904deSPeter Grehan /* Packet Transmission Stats */ 2889517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "good_octets_txd", 2890542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.gotc, 2891517904deSPeter Grehan "Good Octets Transmitted"); 2892517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "total_pkts_txd", 2893542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.tpt, 2894517904deSPeter Grehan "Total Packets Transmitted"); 2895517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "good_pkts_txd", 2896542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.gptc, 2897517904deSPeter Grehan "Good Packets Transmitted"); 2898517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "bcast_pkts_txd", 2899542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.bptc, 2900517904deSPeter Grehan "Broadcast Packets Transmitted"); 2901517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "mcast_pkts_txd", 2902542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.mptc, 2903517904deSPeter Grehan "Multicast Packets Transmitted"); 2904517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "tx_frames_64", 2905542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.ptc64, 2906517904deSPeter Grehan "64 byte frames transmitted "); 2907517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "tx_frames_65_127", 2908542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.ptc127, 2909517904deSPeter Grehan "65-127 byte frames transmitted"); 2910517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "tx_frames_128_255", 2911542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.ptc255, 2912517904deSPeter Grehan "128-255 byte frames transmitted"); 2913517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "tx_frames_256_511", 2914542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.ptc511, 2915517904deSPeter Grehan "256-511 byte frames transmitted"); 2916517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "tx_frames_512_1023", 2917542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.ptc1023, 2918517904deSPeter Grehan "512-1023 byte frames transmitted"); 2919517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "tx_frames_1024_1522", 2920542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.ptc1522, 2921517904deSPeter Grehan "1024-1522 byte frames transmitted"); 2922517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "tso_txd", 2923542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.tsctc, 2924517904deSPeter Grehan "TSO Contexts Transmitted"); 2925517904deSPeter Grehan 2926517904deSPeter Grehan /* Interrupt Stats */ 2927517904deSPeter Grehan int_node = SYSCTL_ADD_NODE(ctx, child, OID_AUTO, "interrupts", 2928517904deSPeter Grehan CTLFLAG_RD | CTLFLAG_MPSAFE, NULL, "Interrupt Statistics"); 2929517904deSPeter Grehan int_list = SYSCTL_CHILDREN(int_node); 2930517904deSPeter Grehan 2931517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, int_list, OID_AUTO, "asserts", 2932542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.iac, 2933517904deSPeter Grehan "Interrupt Assertion Count"); 2934517904deSPeter Grehan 2935517904deSPeter Grehan SYSCTL_ADD_UQUAD(ctx, int_list, OID_AUTO, "rx_desc_min_thresh", 2936542f5d56SKevin Bowling CTLFLAG_RD, &sc->stats.rxdmtc, 2937517904deSPeter Grehan "Rx Desc Min Thresh Count"); 2938517904deSPeter Grehan } 2939517904deSPeter Grehan 294033ed9bdcSKevin Bowling static void 2941542f5d56SKevin Bowling igc_fw_version(struct igc_softc *sc) 294233ed9bdcSKevin Bowling { 294333ed9bdcSKevin Bowling struct igc_hw *hw = &sc->hw; 294433ed9bdcSKevin Bowling struct igc_fw_version *fw_ver = &sc->fw_ver; 294533ed9bdcSKevin Bowling 294633ed9bdcSKevin Bowling *fw_ver = (struct igc_fw_version){0}; 294733ed9bdcSKevin Bowling 294833ed9bdcSKevin Bowling igc_get_fw_version(hw, fw_ver); 294933ed9bdcSKevin Bowling } 295033ed9bdcSKevin Bowling 295133ed9bdcSKevin Bowling static void 295233ed9bdcSKevin Bowling igc_sbuf_fw_version(struct igc_fw_version *fw_ver, struct sbuf *buf) 295333ed9bdcSKevin Bowling { 295433ed9bdcSKevin Bowling const char *space = ""; 295533ed9bdcSKevin Bowling 295633ed9bdcSKevin Bowling if (fw_ver->eep_major || fw_ver->eep_minor || fw_ver->eep_build) { 295733ed9bdcSKevin Bowling sbuf_printf(buf, "EEPROM V%d.%d-%d", fw_ver->eep_major, 295833ed9bdcSKevin Bowling fw_ver->eep_minor, fw_ver->eep_build); 295933ed9bdcSKevin Bowling space = " "; 296033ed9bdcSKevin Bowling } 296133ed9bdcSKevin Bowling 29629efc7325SKevin Bowling if (fw_ver->invm_major || fw_ver->invm_minor || 29639efc7325SKevin Bowling fw_ver->invm_img_type) { 296433ed9bdcSKevin Bowling sbuf_printf(buf, "%sNVM V%d.%d imgtype%d", 296533ed9bdcSKevin Bowling space, fw_ver->invm_major, fw_ver->invm_minor, 296633ed9bdcSKevin Bowling fw_ver->invm_img_type); 296733ed9bdcSKevin Bowling space = " "; 296833ed9bdcSKevin Bowling } 296933ed9bdcSKevin Bowling 297033ed9bdcSKevin Bowling if (fw_ver->or_valid) { 297133ed9bdcSKevin Bowling sbuf_printf(buf, "%sOption ROM V%d-b%d-p%d", 297233ed9bdcSKevin Bowling space, fw_ver->or_major, fw_ver->or_build, 297333ed9bdcSKevin Bowling fw_ver->or_patch); 297433ed9bdcSKevin Bowling space = " "; 297533ed9bdcSKevin Bowling } 297633ed9bdcSKevin Bowling 297733ed9bdcSKevin Bowling if (fw_ver->etrack_id) 297833ed9bdcSKevin Bowling sbuf_printf(buf, "%seTrack 0x%08x", space, fw_ver->etrack_id); 297933ed9bdcSKevin Bowling } 298033ed9bdcSKevin Bowling 298133ed9bdcSKevin Bowling static void 2982542f5d56SKevin Bowling igc_print_fw_version(struct igc_softc *sc ) 298333ed9bdcSKevin Bowling { 298433ed9bdcSKevin Bowling device_t dev = sc->dev; 298533ed9bdcSKevin Bowling struct sbuf *buf; 298633ed9bdcSKevin Bowling int error = 0; 298733ed9bdcSKevin Bowling 298833ed9bdcSKevin Bowling buf = sbuf_new_auto(); 298933ed9bdcSKevin Bowling if (!buf) { 299033ed9bdcSKevin Bowling device_printf(dev, "Could not allocate sbuf for output.\n"); 299133ed9bdcSKevin Bowling return; 299233ed9bdcSKevin Bowling } 299333ed9bdcSKevin Bowling 299433ed9bdcSKevin Bowling igc_sbuf_fw_version(&sc->fw_ver, buf); 299533ed9bdcSKevin Bowling 299633ed9bdcSKevin Bowling error = sbuf_finish(buf); 299733ed9bdcSKevin Bowling if (error) 299833ed9bdcSKevin Bowling device_printf(dev, "Error finishing sbuf: %d\n", error); 299933ed9bdcSKevin Bowling else if (sbuf_len(buf)) 300033ed9bdcSKevin Bowling device_printf(dev, "%s\n", sbuf_data(buf)); 300133ed9bdcSKevin Bowling 300233ed9bdcSKevin Bowling sbuf_delete(buf); 300333ed9bdcSKevin Bowling } 300433ed9bdcSKevin Bowling 300533ed9bdcSKevin Bowling static int 300633ed9bdcSKevin Bowling igc_sysctl_print_fw_version(SYSCTL_HANDLER_ARGS) 300733ed9bdcSKevin Bowling { 3008542f5d56SKevin Bowling struct igc_softc *sc = (struct igc_softc *)arg1; 300933ed9bdcSKevin Bowling device_t dev = sc->dev; 301033ed9bdcSKevin Bowling struct sbuf *buf; 301133ed9bdcSKevin Bowling int error = 0; 301233ed9bdcSKevin Bowling 301333ed9bdcSKevin Bowling buf = sbuf_new_for_sysctl(NULL, NULL, 128, req); 301433ed9bdcSKevin Bowling if (!buf) { 301533ed9bdcSKevin Bowling device_printf(dev, "Could not allocate sbuf for output.\n"); 301633ed9bdcSKevin Bowling return (ENOMEM); 301733ed9bdcSKevin Bowling } 301833ed9bdcSKevin Bowling 301933ed9bdcSKevin Bowling igc_sbuf_fw_version(&sc->fw_ver, buf); 302033ed9bdcSKevin Bowling 302133ed9bdcSKevin Bowling error = sbuf_finish(buf); 302233ed9bdcSKevin Bowling if (error) 302333ed9bdcSKevin Bowling device_printf(dev, "Error finishing sbuf: %d\n", error); 302433ed9bdcSKevin Bowling 302533ed9bdcSKevin Bowling sbuf_delete(buf); 302633ed9bdcSKevin Bowling 302733ed9bdcSKevin Bowling return (0); 302833ed9bdcSKevin Bowling } 302933ed9bdcSKevin Bowling 3030517904deSPeter Grehan /********************************************************************** 3031517904deSPeter Grehan * 3032517904deSPeter Grehan * This routine provides a way to dump out the adapter eeprom, 3033517904deSPeter Grehan * often a useful debug/service tool. This only dumps the first 3034517904deSPeter Grehan * 32 words, stuff that matters is in that extent. 3035517904deSPeter Grehan * 3036517904deSPeter Grehan **********************************************************************/ 3037517904deSPeter Grehan static int 3038517904deSPeter Grehan igc_sysctl_nvm_info(SYSCTL_HANDLER_ARGS) 3039517904deSPeter Grehan { 3040542f5d56SKevin Bowling struct igc_softc *sc = (struct igc_softc *)arg1; 3041517904deSPeter Grehan int error; 3042517904deSPeter Grehan int result; 3043517904deSPeter Grehan 3044517904deSPeter Grehan result = -1; 3045517904deSPeter Grehan error = sysctl_handle_int(oidp, &result, 0, req); 3046517904deSPeter Grehan 3047517904deSPeter Grehan if (error || !req->newptr) 3048517904deSPeter Grehan return (error); 3049517904deSPeter Grehan 3050517904deSPeter Grehan /* 3051517904deSPeter Grehan * This value will cause a hex dump of the 3052517904deSPeter Grehan * first 32 16-bit words of the EEPROM to 3053517904deSPeter Grehan * the screen. 3054517904deSPeter Grehan */ 3055517904deSPeter Grehan if (result == 1) 3056542f5d56SKevin Bowling igc_print_nvm_info(sc); 3057517904deSPeter Grehan 3058517904deSPeter Grehan return (error); 3059517904deSPeter Grehan } 3060517904deSPeter Grehan 3061517904deSPeter Grehan static void 3062542f5d56SKevin Bowling igc_print_nvm_info(struct igc_softc *sc) 3063517904deSPeter Grehan { 3064517904deSPeter Grehan u16 eeprom_data; 3065517904deSPeter Grehan int i, j, row = 0; 3066517904deSPeter Grehan 3067517904deSPeter Grehan /* Its a bit crude, but it gets the job done */ 3068517904deSPeter Grehan printf("\nInterface EEPROM Dump:\n"); 3069517904deSPeter Grehan printf("Offset\n0x0000 "); 3070517904deSPeter Grehan for (i = 0, j = 0; i < 32; i++, j++) { 3071517904deSPeter Grehan if (j == 8) { /* Make the offset block */ 3072517904deSPeter Grehan j = 0; ++row; 3073517904deSPeter Grehan printf("\n0x00%x0 ",row); 3074517904deSPeter Grehan } 3075542f5d56SKevin Bowling igc_read_nvm(&sc->hw, i, 1, &eeprom_data); 3076517904deSPeter Grehan printf("%04x ", eeprom_data); 3077517904deSPeter Grehan } 3078517904deSPeter Grehan printf("\n"); 3079517904deSPeter Grehan } 3080517904deSPeter Grehan 3081ab540d44SKevin Bowling static int 3082ab540d44SKevin Bowling igc_sysctl_tso_tcp_flags_mask(SYSCTL_HANDLER_ARGS) 3083ab540d44SKevin Bowling { 3084ab540d44SKevin Bowling struct igc_softc *sc; 3085ab540d44SKevin Bowling u32 reg, val, shift; 3086ab540d44SKevin Bowling int error, mask; 3087ab540d44SKevin Bowling 3088ab540d44SKevin Bowling sc = oidp->oid_arg1; 3089ab540d44SKevin Bowling switch (oidp->oid_arg2) { 3090ab540d44SKevin Bowling case 0: 3091ab540d44SKevin Bowling reg = IGC_DTXTCPFLGL; 3092ab540d44SKevin Bowling shift = 0; 3093ab540d44SKevin Bowling break; 3094ab540d44SKevin Bowling case 1: 3095ab540d44SKevin Bowling reg = IGC_DTXTCPFLGL; 3096ab540d44SKevin Bowling shift = 16; 3097ab540d44SKevin Bowling break; 3098ab540d44SKevin Bowling case 2: 3099ab540d44SKevin Bowling reg = IGC_DTXTCPFLGH; 3100ab540d44SKevin Bowling shift = 0; 3101ab540d44SKevin Bowling break; 3102ab540d44SKevin Bowling default: 3103ab540d44SKevin Bowling return (EINVAL); 3104ab540d44SKevin Bowling break; 3105ab540d44SKevin Bowling } 3106ab540d44SKevin Bowling val = IGC_READ_REG(&sc->hw, reg); 3107ab540d44SKevin Bowling mask = (val >> shift) & 0xfff; 3108ab540d44SKevin Bowling error = sysctl_handle_int(oidp, &mask, 0, req); 3109ab540d44SKevin Bowling if (error != 0 || req->newptr == NULL) 3110ab540d44SKevin Bowling return (error); 3111ab540d44SKevin Bowling if (mask < 0 || mask > 0xfff) 3112ab540d44SKevin Bowling return (EINVAL); 3113ab540d44SKevin Bowling val = (val & ~(0xfff << shift)) | (mask << shift); 3114ab540d44SKevin Bowling IGC_WRITE_REG(&sc->hw, reg, val); 3115ab540d44SKevin Bowling return (0); 3116ab540d44SKevin Bowling } 3117ab540d44SKevin Bowling 3118517904deSPeter Grehan /* 3119517904deSPeter Grehan * Set flow control using sysctl: 3120517904deSPeter Grehan * Flow control values: 3121517904deSPeter Grehan * 0 - off 3122517904deSPeter Grehan * 1 - rx pause 3123517904deSPeter Grehan * 2 - tx pause 3124517904deSPeter Grehan * 3 - full 3125517904deSPeter Grehan */ 3126517904deSPeter Grehan static int 3127517904deSPeter Grehan igc_set_flowcntl(SYSCTL_HANDLER_ARGS) 3128517904deSPeter Grehan { 3129517904deSPeter Grehan int error; 3130517904deSPeter Grehan static int input = 3; /* default is full */ 3131542f5d56SKevin Bowling struct igc_softc *sc = (struct igc_softc *) arg1; 3132517904deSPeter Grehan 3133517904deSPeter Grehan error = sysctl_handle_int(oidp, &input, 0, req); 3134517904deSPeter Grehan 3135517904deSPeter Grehan if ((error) || (req->newptr == NULL)) 3136517904deSPeter Grehan return (error); 3137517904deSPeter Grehan 3138542f5d56SKevin Bowling if (input == sc->fc) /* no change? */ 3139517904deSPeter Grehan return (error); 3140517904deSPeter Grehan 3141517904deSPeter Grehan switch (input) { 3142517904deSPeter Grehan case igc_fc_rx_pause: 3143517904deSPeter Grehan case igc_fc_tx_pause: 3144517904deSPeter Grehan case igc_fc_full: 3145517904deSPeter Grehan case igc_fc_none: 3146542f5d56SKevin Bowling sc->hw.fc.requested_mode = input; 3147542f5d56SKevin Bowling sc->fc = input; 3148517904deSPeter Grehan break; 3149517904deSPeter Grehan default: 3150517904deSPeter Grehan /* Do nothing */ 3151517904deSPeter Grehan return (error); 3152517904deSPeter Grehan } 3153517904deSPeter Grehan 3154542f5d56SKevin Bowling sc->hw.fc.current_mode = sc->hw.fc.requested_mode; 3155542f5d56SKevin Bowling igc_force_mac_fc(&sc->hw); 3156517904deSPeter Grehan return (error); 3157517904deSPeter Grehan } 3158517904deSPeter Grehan 3159517904deSPeter Grehan /* 316068b1f5dcSKevin Bowling * Manage DMA Coalesce: 316168b1f5dcSKevin Bowling * Control values: 316268b1f5dcSKevin Bowling * 0/1 - off/on 316368b1f5dcSKevin Bowling * Legal timer values are: 316468b1f5dcSKevin Bowling * 250,500,1000-10000 in thousands 316568b1f5dcSKevin Bowling */ 316668b1f5dcSKevin Bowling static int 316768b1f5dcSKevin Bowling igc_sysctl_dmac(SYSCTL_HANDLER_ARGS) 316868b1f5dcSKevin Bowling { 3169542f5d56SKevin Bowling struct igc_softc *sc = (struct igc_softc *) arg1; 317068b1f5dcSKevin Bowling int error; 317168b1f5dcSKevin Bowling 317268b1f5dcSKevin Bowling error = sysctl_handle_int(oidp, &sc->dmac, 0, req); 317368b1f5dcSKevin Bowling 317468b1f5dcSKevin Bowling if ((error) || (req->newptr == NULL)) 317568b1f5dcSKevin Bowling return (error); 317668b1f5dcSKevin Bowling 317768b1f5dcSKevin Bowling switch (sc->dmac) { 317868b1f5dcSKevin Bowling case 0: 317968b1f5dcSKevin Bowling /* Disabling */ 318068b1f5dcSKevin Bowling break; 318168b1f5dcSKevin Bowling case 1: /* Just enable and use default */ 318268b1f5dcSKevin Bowling sc->dmac = 1000; 318368b1f5dcSKevin Bowling break; 318468b1f5dcSKevin Bowling case 250: 318568b1f5dcSKevin Bowling case 500: 318668b1f5dcSKevin Bowling case 1000: 318768b1f5dcSKevin Bowling case 2000: 318868b1f5dcSKevin Bowling case 3000: 318968b1f5dcSKevin Bowling case 4000: 319068b1f5dcSKevin Bowling case 5000: 319168b1f5dcSKevin Bowling case 6000: 319268b1f5dcSKevin Bowling case 7000: 319368b1f5dcSKevin Bowling case 8000: 319468b1f5dcSKevin Bowling case 9000: 319568b1f5dcSKevin Bowling case 10000: 319668b1f5dcSKevin Bowling /* Legal values - allow */ 319768b1f5dcSKevin Bowling break; 319868b1f5dcSKevin Bowling default: 319968b1f5dcSKevin Bowling /* Do nothing, illegal value */ 320068b1f5dcSKevin Bowling sc->dmac = 0; 320168b1f5dcSKevin Bowling return (EINVAL); 320268b1f5dcSKevin Bowling } 320368b1f5dcSKevin Bowling /* Reinit the interface */ 320468b1f5dcSKevin Bowling igc_if_init(sc->ctx); 320568b1f5dcSKevin Bowling return (error); 320668b1f5dcSKevin Bowling } 320768b1f5dcSKevin Bowling 320868b1f5dcSKevin Bowling /* 3209517904deSPeter Grehan * Manage Energy Efficient Ethernet: 3210517904deSPeter Grehan * Control values: 3211517904deSPeter Grehan * 0/1 - enabled/disabled 3212517904deSPeter Grehan */ 3213517904deSPeter Grehan static int 3214517904deSPeter Grehan igc_sysctl_eee(SYSCTL_HANDLER_ARGS) 3215517904deSPeter Grehan { 3216542f5d56SKevin Bowling struct igc_softc *sc = (struct igc_softc *) arg1; 3217517904deSPeter Grehan int error, value; 3218517904deSPeter Grehan 3219542f5d56SKevin Bowling value = sc->hw.dev_spec._i225.eee_disable; 3220517904deSPeter Grehan error = sysctl_handle_int(oidp, &value, 0, req); 3221517904deSPeter Grehan if (error || req->newptr == NULL) 3222517904deSPeter Grehan return (error); 3223517904deSPeter Grehan 3224542f5d56SKevin Bowling sc->hw.dev_spec._i225.eee_disable = (value != 0); 3225542f5d56SKevin Bowling igc_if_init(sc->ctx); 3226517904deSPeter Grehan 3227517904deSPeter Grehan return (0); 3228517904deSPeter Grehan } 3229517904deSPeter Grehan 3230517904deSPeter Grehan static int 3231517904deSPeter Grehan igc_sysctl_debug_info(SYSCTL_HANDLER_ARGS) 3232517904deSPeter Grehan { 3233542f5d56SKevin Bowling struct igc_softc *sc; 3234517904deSPeter Grehan int error; 3235517904deSPeter Grehan int result; 3236517904deSPeter Grehan 3237517904deSPeter Grehan result = -1; 3238517904deSPeter Grehan error = sysctl_handle_int(oidp, &result, 0, req); 3239517904deSPeter Grehan 3240517904deSPeter Grehan if (error || !req->newptr) 3241517904deSPeter Grehan return (error); 3242517904deSPeter Grehan 3243517904deSPeter Grehan if (result == 1) { 3244542f5d56SKevin Bowling sc = (struct igc_softc *) arg1; 3245542f5d56SKevin Bowling igc_print_debug_info(sc); 3246517904deSPeter Grehan } 3247517904deSPeter Grehan 3248517904deSPeter Grehan return (error); 3249517904deSPeter Grehan } 3250517904deSPeter Grehan 3251517904deSPeter Grehan static int 3252517904deSPeter Grehan igc_get_rs(SYSCTL_HANDLER_ARGS) 3253517904deSPeter Grehan { 3254542f5d56SKevin Bowling struct igc_softc *sc = (struct igc_softc *) arg1; 3255517904deSPeter Grehan int error; 3256517904deSPeter Grehan int result; 3257517904deSPeter Grehan 3258517904deSPeter Grehan result = 0; 3259517904deSPeter Grehan error = sysctl_handle_int(oidp, &result, 0, req); 3260517904deSPeter Grehan 3261517904deSPeter Grehan if (error || !req->newptr || result != 1) 3262517904deSPeter Grehan return (error); 3263542f5d56SKevin Bowling igc_dump_rs(sc); 3264517904deSPeter Grehan 3265517904deSPeter Grehan return (error); 3266517904deSPeter Grehan } 3267517904deSPeter Grehan 3268517904deSPeter Grehan static void 3269517904deSPeter Grehan igc_if_debug(if_ctx_t ctx) 3270517904deSPeter Grehan { 3271517904deSPeter Grehan igc_dump_rs(iflib_get_softc(ctx)); 3272517904deSPeter Grehan } 3273517904deSPeter Grehan 3274517904deSPeter Grehan /* 3275517904deSPeter Grehan * This routine is meant to be fluid, add whatever is 3276517904deSPeter Grehan * needed for debugging a problem. -jfv 3277517904deSPeter Grehan */ 3278517904deSPeter Grehan static void 3279542f5d56SKevin Bowling igc_print_debug_info(struct igc_softc *sc) 3280517904deSPeter Grehan { 3281542f5d56SKevin Bowling device_t dev = iflib_get_dev(sc->ctx); 3282542f5d56SKevin Bowling if_t ifp = iflib_get_ifp(sc->ctx); 3283542f5d56SKevin Bowling struct tx_ring *txr = &sc->tx_queues->txr; 3284542f5d56SKevin Bowling struct rx_ring *rxr = &sc->rx_queues->rxr; 3285517904deSPeter Grehan 3286517904deSPeter Grehan if (if_getdrvflags(ifp) & IFF_DRV_RUNNING) 3287517904deSPeter Grehan printf("Interface is RUNNING "); 3288517904deSPeter Grehan else 3289517904deSPeter Grehan printf("Interface is NOT RUNNING\n"); 3290517904deSPeter Grehan 3291517904deSPeter Grehan if (if_getdrvflags(ifp) & IFF_DRV_OACTIVE) 3292517904deSPeter Grehan printf("and INACTIVE\n"); 3293517904deSPeter Grehan else 3294517904deSPeter Grehan printf("and ACTIVE\n"); 3295517904deSPeter Grehan 3296542f5d56SKevin Bowling for (int i = 0; i < sc->tx_num_queues; i++, txr++) { 3297517904deSPeter Grehan device_printf(dev, "TX Queue %d ------\n", i); 3298517904deSPeter Grehan device_printf(dev, "hw tdh = %d, hw tdt = %d\n", 3299542f5d56SKevin Bowling IGC_READ_REG(&sc->hw, IGC_TDH(i)), 3300542f5d56SKevin Bowling IGC_READ_REG(&sc->hw, IGC_TDT(i))); 3301517904deSPeter Grehan 3302517904deSPeter Grehan } 3303542f5d56SKevin Bowling for (int j=0; j < sc->rx_num_queues; j++, rxr++) { 3304517904deSPeter Grehan device_printf(dev, "RX Queue %d ------\n", j); 3305517904deSPeter Grehan device_printf(dev, "hw rdh = %d, hw rdt = %d\n", 3306542f5d56SKevin Bowling IGC_READ_REG(&sc->hw, IGC_RDH(j)), 3307542f5d56SKevin Bowling IGC_READ_REG(&sc->hw, IGC_RDT(j))); 3308517904deSPeter Grehan } 3309517904deSPeter Grehan } 3310