xref: /freebsd/sys/dev/ixgbe/if_ix.c (revision 89d4096950c4db748e39758c941cfb708c2ff808)
1c58d34ddSKevin Bowling /*****************************************************************************
2758cc3dcSJack F Vogel 
38eb6488eSEric Joyner   Copyright (c) 2001-2017, Intel Corporation
4758cc3dcSJack F Vogel   All rights reserved.
5758cc3dcSJack F Vogel 
6758cc3dcSJack F Vogel   Redistribution and use in source and binary forms, with or without
7758cc3dcSJack F Vogel   modification, are permitted provided that the following conditions are met:
8758cc3dcSJack F Vogel 
9758cc3dcSJack F Vogel    1. Redistributions of source code must retain the above copyright notice,
10758cc3dcSJack F Vogel       this list of conditions and the following disclaimer.
11758cc3dcSJack F Vogel 
12758cc3dcSJack F Vogel    2. Redistributions in binary form must reproduce the above copyright
13758cc3dcSJack F Vogel       notice, this list of conditions and the following disclaimer in the
14758cc3dcSJack F Vogel       documentation and/or other materials provided with the distribution.
15758cc3dcSJack F Vogel 
16758cc3dcSJack F Vogel    3. Neither the name of the Intel Corporation nor the names of its
17758cc3dcSJack F Vogel       contributors may be used to endorse or promote products derived from
18758cc3dcSJack F Vogel       this software without specific prior written permission.
19758cc3dcSJack F Vogel 
20758cc3dcSJack F Vogel   THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
21758cc3dcSJack F Vogel   AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22758cc3dcSJack F Vogel   IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
23758cc3dcSJack F Vogel   ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
24758cc3dcSJack F Vogel   LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
25758cc3dcSJack F Vogel   CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
26758cc3dcSJack F Vogel   SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
27758cc3dcSJack F Vogel   INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
28758cc3dcSJack F Vogel   CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
29758cc3dcSJack F Vogel   ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
30758cc3dcSJack F Vogel   POSSIBILITY OF SUCH DAMAGE.
31758cc3dcSJack F Vogel 
32c58d34ddSKevin Bowling *****************************************************************************/
33758cc3dcSJack F Vogel 
34758cc3dcSJack F Vogel #include "opt_inet.h"
35758cc3dcSJack F Vogel #include "opt_inet6.h"
36758cc3dcSJack F Vogel #include "opt_rss.h"
37758cc3dcSJack F Vogel 
38758cc3dcSJack F Vogel #include "ixgbe.h"
39c19c7afeSEric Joyner #include "ixgbe_sriov.h"
40c19c7afeSEric Joyner #include "ifdi_if.h"
41c19c7afeSEric Joyner 
42c19c7afeSEric Joyner #include <net/netmap.h>
43c19c7afeSEric Joyner #include <dev/netmap/netmap_kern.h>
44758cc3dcSJack F Vogel 
458eb6488eSEric Joyner /************************************************************************
46758cc3dcSJack F Vogel  * Driver version
478eb6488eSEric Joyner  ************************************************************************/
4851e23514SMarius Strobl static const char ixgbe_driver_version[] = "4.0.1-k";
49a9ca1c79SSean Bruno 
508eb6488eSEric Joyner /************************************************************************
51758cc3dcSJack F Vogel  * PCI Device ID Table
52758cc3dcSJack F Vogel  *
53758cc3dcSJack F Vogel  *   Used by probe to select devices to load on
54758cc3dcSJack F Vogel  *   Last field stores an index into ixgbe_strings
55758cc3dcSJack F Vogel  *   Last entry must be all 0s
56758cc3dcSJack F Vogel  *
57758cc3dcSJack F Vogel  *   { Vendor ID, Device ID, SubVendor ID, SubDevice ID, String Index }
588eb6488eSEric Joyner  ************************************************************************/
5951e23514SMarius Strobl static const pci_vendor_info_t ixgbe_vendor_info_array[] =
60758cc3dcSJack F Vogel {
61c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82598AF_DUAL_PORT,
62c58d34ddSKevin Bowling     "Intel(R) 82598EB AF (Dual Fiber)"),
63c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82598AF_SINGLE_PORT,
64c58d34ddSKevin Bowling     "Intel(R) 82598EB AF (Fiber)"),
65c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82598EB_CX4,
66c58d34ddSKevin Bowling     "Intel(R) 82598EB AT (CX4)"),
67c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82598AT,
68c58d34ddSKevin Bowling     "Intel(R) 82598EB AT"),
69c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82598AT2,
70c58d34ddSKevin Bowling     "Intel(R) 82598EB AT2"),
71fdbcd35aSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82598, "Intel(R) 82598"),
72c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82598_DA_DUAL_PORT,
73c58d34ddSKevin Bowling     "Intel(R) 82598EB AF DA (Dual Fiber)"),
74c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82598_CX4_DUAL_PORT,
75c58d34ddSKevin Bowling     "Intel(R) 82598EB AT (Dual CX4)"),
76c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82598EB_XF_LR,
77c58d34ddSKevin Bowling     "Intel(R) 82598EB AF (Dual Fiber LR)"),
78c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82598_SR_DUAL_PORT_EM,
79c58d34ddSKevin Bowling     "Intel(R) 82598EB AF (Dual Fiber SR)"),
80c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82598EB_SFP_LOM,
81c58d34ddSKevin Bowling     "Intel(R) 82598EB LOM"),
82c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82599_KX4,
83c58d34ddSKevin Bowling     "Intel(R) X520 82599 (KX4)"),
84c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82599_KX4_MEZZ,
85c58d34ddSKevin Bowling     "Intel(R) X520 82599 (KX4 Mezzanine)"),
86c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82599_SFP,
87c58d34ddSKevin Bowling     "Intel(R) X520 82599ES (SFI/SFP+)"),
88c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82599_XAUI_LOM,
89c58d34ddSKevin Bowling     "Intel(R) X520 82599 (XAUI/BX4)"),
90c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82599_CX4,
91c58d34ddSKevin Bowling     "Intel(R) X520 82599 (Dual CX4)"),
92c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82599_T3_LOM,
93c58d34ddSKevin Bowling     "Intel(R) X520-T 82599 LOM"),
94c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82599_LS,
95c58d34ddSKevin Bowling     "Intel(R) X520 82599 LS"),
96c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82599_COMBO_BACKPLANE,
97c58d34ddSKevin Bowling     "Intel(R) X520 82599 (Combined Backplane)"),
98c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82599_BACKPLANE_FCOE,
99c58d34ddSKevin Bowling     "Intel(R) X520 82599 (Backplane w/FCoE)"),
100c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82599_SFP_SF2,
101c58d34ddSKevin Bowling     "Intel(R) X520 82599 (Dual SFP+)"),
102c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82599_SFP_FCOE,
103c58d34ddSKevin Bowling     "Intel(R) X520 82599 (Dual SFP+ w/FCoE)"),
104c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82599EN_SFP,
105c58d34ddSKevin Bowling     "Intel(R) X520-1 82599EN (SFP+)"),
106c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82599_SFP_SF_QP,
107c58d34ddSKevin Bowling     "Intel(R) X520-4 82599 (Quad SFP+)"),
108c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82599_QSFP_SF_QP,
109c58d34ddSKevin Bowling     "Intel(R) X520-Q1 82599 (QSFP+)"),
110c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_X540T,
111c58d34ddSKevin Bowling     "Intel(R) X540-AT2"),
112fdbcd35aSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_X540T1,  "Intel(R) X540-T1"),
113fdbcd35aSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_X550T,  "Intel(R) X550-T2"),
114fdbcd35aSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_X550T1, "Intel(R) X550-T1"),
115c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_X550EM_X_KR,
116c58d34ddSKevin Bowling     "Intel(R) X552 (KR Backplane)"),
117c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_X550EM_X_KX4,
118c58d34ddSKevin Bowling     "Intel(R) X552 (KX4 Backplane)"),
119c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_X550EM_X_10G_T,
120c58d34ddSKevin Bowling     "Intel(R) X552/X557-AT (10GBASE-T)"),
121c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_X550EM_X_1G_T,
122c58d34ddSKevin Bowling     "Intel(R) X552 (1000BASE-T)"),
123c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_X550EM_X_SFP,
124c58d34ddSKevin Bowling     "Intel(R) X552 (SFP+)"),
125c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_X550EM_A_KR,
126c58d34ddSKevin Bowling     "Intel(R) X553 (KR Backplane)"),
127c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_X550EM_A_KR_L,
128c58d34ddSKevin Bowling     "Intel(R) X553 L (KR Backplane)"),
129c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_X550EM_A_SFP,
130c58d34ddSKevin Bowling     "Intel(R) X553 (SFP+)"),
131c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_X550EM_A_SFP_N,
132c58d34ddSKevin Bowling     "Intel(R) X553 N (SFP+)"),
133c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_X550EM_A_SGMII,
134c58d34ddSKevin Bowling     "Intel(R) X553 (1GbE SGMII)"),
135c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_X550EM_A_SGMII_L,
136c58d34ddSKevin Bowling     "Intel(R) X553 L (1GbE SGMII)"),
137c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_X550EM_A_10G_T,
138c58d34ddSKevin Bowling     "Intel(R) X553/X557-AT (10GBASE-T)"),
139c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_X550EM_A_1G_T,
140c58d34ddSKevin Bowling     "Intel(R) X553 (1GbE)"),
141c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_X550EM_A_1G_T_L,
142c58d34ddSKevin Bowling     "Intel(R) X553 L (1GbE)"),
143c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_X540_BYPASS,
144c58d34ddSKevin Bowling     "Intel(R) X540-T2 (Bypass)"),
145c58d34ddSKevin Bowling 	PVID(IXGBE_INTEL_VENDOR_ID, IXGBE_DEV_ID_82599_BYPASS,
146c58d34ddSKevin Bowling     "Intel(R) X520 82599 (Bypass)"),
147758cc3dcSJack F Vogel 	/* required last entry */
148c19c7afeSEric Joyner 	PVID_END
149758cc3dcSJack F Vogel };
150758cc3dcSJack F Vogel 
151b1d5caf3SKevin Bowling static void *ixgbe_register(device_t);
152b1d5caf3SKevin Bowling static int  ixgbe_if_attach_pre(if_ctx_t);
153b1d5caf3SKevin Bowling static int  ixgbe_if_attach_post(if_ctx_t);
154b1d5caf3SKevin Bowling static int  ixgbe_if_detach(if_ctx_t);
155b1d5caf3SKevin Bowling static int  ixgbe_if_shutdown(if_ctx_t);
156b1d5caf3SKevin Bowling static int  ixgbe_if_suspend(if_ctx_t);
157b1d5caf3SKevin Bowling static int  ixgbe_if_resume(if_ctx_t);
158c19c7afeSEric Joyner 
159b1d5caf3SKevin Bowling static void ixgbe_if_stop(if_ctx_t);
160b1d5caf3SKevin Bowling void ixgbe_if_enable_intr(if_ctx_t);
161b1d5caf3SKevin Bowling static void ixgbe_if_disable_intr(if_ctx_t);
162b1d5caf3SKevin Bowling static void ixgbe_link_intr_enable(if_ctx_t);
163b1d5caf3SKevin Bowling static int  ixgbe_if_rx_queue_intr_enable(if_ctx_t, uint16_t);
164b1d5caf3SKevin Bowling static void ixgbe_if_media_status(if_ctx_t, struct ifmediareq *);
165b1d5caf3SKevin Bowling static int  ixgbe_if_media_change(if_ctx_t);
166c19c7afeSEric Joyner static int  ixgbe_if_msix_intr_assign(if_ctx_t, int);
167b1d5caf3SKevin Bowling static int  ixgbe_if_mtu_set(if_ctx_t, uint32_t);
168b1d5caf3SKevin Bowling static void ixgbe_if_crcstrip_set(if_ctx_t, int, int);
169b1d5caf3SKevin Bowling static void ixgbe_if_multi_set(if_ctx_t);
170b1d5caf3SKevin Bowling static int  ixgbe_if_promisc_set(if_ctx_t, int);
171c58d34ddSKevin Bowling static int  ixgbe_if_tx_queues_alloc(if_ctx_t, caddr_t *, uint64_t *, int,
172c58d34ddSKevin Bowling     int);
173c58d34ddSKevin Bowling static int  ixgbe_if_rx_queues_alloc(if_ctx_t, caddr_t *, uint64_t *, int,
174c58d34ddSKevin Bowling    int);
175b1d5caf3SKevin Bowling static void ixgbe_if_queues_free(if_ctx_t);
176b1d5caf3SKevin Bowling static void ixgbe_if_timer(if_ctx_t, uint16_t);
177b1d5caf3SKevin Bowling static void ixgbe_if_update_admin_status(if_ctx_t);
178b1d5caf3SKevin Bowling static void ixgbe_if_vlan_register(if_ctx_t, u16);
179b1d5caf3SKevin Bowling static void ixgbe_if_vlan_unregister(if_ctx_t, u16);
180b1d5caf3SKevin Bowling static int  ixgbe_if_i2c_req(if_ctx_t, struct ifi2creq *);
181b1d5caf3SKevin Bowling static bool ixgbe_if_needs_restart(if_ctx_t, enum iflib_restart_event);
182b1d5caf3SKevin Bowling int ixgbe_intr(void *);
183758cc3dcSJack F Vogel 
1848eb6488eSEric Joyner /************************************************************************
185758cc3dcSJack F Vogel  * Function prototypes
1868eb6488eSEric Joyner  ************************************************************************/
187c19c7afeSEric Joyner static uint64_t ixgbe_if_get_counter(if_ctx_t, ift_counter);
188758cc3dcSJack F Vogel 
189b1d5caf3SKevin Bowling static void ixgbe_enable_queue(struct ixgbe_softc *, u32);
190b1d5caf3SKevin Bowling static void ixgbe_disable_queue(struct ixgbe_softc *, u32);
191b1d5caf3SKevin Bowling static void ixgbe_add_device_sysctls(if_ctx_t);
192b1d5caf3SKevin Bowling static int  ixgbe_allocate_pci_resources(if_ctx_t);
193b1d5caf3SKevin Bowling static int  ixgbe_setup_low_power_mode(if_ctx_t);
194c19c7afeSEric Joyner 
195b1d5caf3SKevin Bowling static void ixgbe_config_dmac(struct ixgbe_softc *);
196b1d5caf3SKevin Bowling static void ixgbe_configure_ivars(struct ixgbe_softc *);
197b1d5caf3SKevin Bowling static void ixgbe_set_ivar(struct ixgbe_softc *, u8, u8, s8);
198c19c7afeSEric Joyner static u8   *ixgbe_mc_array_itr(struct ixgbe_hw *, u8 **, u32 *);
199b1d5caf3SKevin Bowling static bool ixgbe_sfp_probe(if_ctx_t);
200c19c7afeSEric Joyner 
201b1d5caf3SKevin Bowling static void ixgbe_free_pci_resources(if_ctx_t);
202c19c7afeSEric Joyner 
203b1d5caf3SKevin Bowling static int  ixgbe_msix_link(void *);
204b1d5caf3SKevin Bowling static int  ixgbe_msix_que(void *);
205b1d5caf3SKevin Bowling static void ixgbe_initialize_rss_mapping(struct ixgbe_softc *);
206b1d5caf3SKevin Bowling static void ixgbe_initialize_receive_units(if_ctx_t);
207b1d5caf3SKevin Bowling static void ixgbe_initialize_transmit_units(if_ctx_t);
208c19c7afeSEric Joyner 
209b1d5caf3SKevin Bowling static int  ixgbe_setup_interface(if_ctx_t);
210b1d5caf3SKevin Bowling static void ixgbe_init_device_features(struct ixgbe_softc *);
211b1d5caf3SKevin Bowling static void ixgbe_check_fan_failure(struct ixgbe_softc *, u32, bool);
2127660e4eaSKevin Bowling static void ixgbe_sbuf_fw_version(struct ixgbe_hw *, struct sbuf *);
213b1d5caf3SKevin Bowling static void ixgbe_print_fw_version(if_ctx_t);
214b1d5caf3SKevin Bowling static void ixgbe_add_media_types(if_ctx_t);
215b1d5caf3SKevin Bowling static void ixgbe_update_stats_counters(struct ixgbe_softc *);
216b1d5caf3SKevin Bowling static void ixgbe_config_link(if_ctx_t);
217b1d5caf3SKevin Bowling static void ixgbe_get_slot_info(struct ixgbe_softc *);
218f72de14eSKevin Bowling static void ixgbe_fw_mode_timer(void *);
219b1d5caf3SKevin Bowling static void ixgbe_check_wol_support(struct ixgbe_softc *);
220b1d5caf3SKevin Bowling static void ixgbe_enable_rx_drop(struct ixgbe_softc *);
221b1d5caf3SKevin Bowling static void ixgbe_disable_rx_drop(struct ixgbe_softc *);
222758cc3dcSJack F Vogel 
223b1d5caf3SKevin Bowling static void ixgbe_add_hw_stats(struct ixgbe_softc *);
224b1d5caf3SKevin Bowling static int  ixgbe_set_flowcntl(struct ixgbe_softc *, int);
225b1d5caf3SKevin Bowling static int  ixgbe_set_advertise(struct ixgbe_softc *, int);
226d381c807SPiotr Pietruszewski static int  ixgbe_get_default_advertise(struct ixgbe_softc *);
227b1d5caf3SKevin Bowling static void ixgbe_setup_vlan_hw_support(if_ctx_t);
228b1d5caf3SKevin Bowling static void ixgbe_config_gpie(struct ixgbe_softc *);
229b1d5caf3SKevin Bowling static void ixgbe_config_delay_values(struct ixgbe_softc *);
2306f37f232SEric Joyner 
2316f37f232SEric Joyner /* Sysctl handlers */
232f2c4db54SSteven Hartland static int  ixgbe_sysctl_flowcntl(SYSCTL_HANDLER_ARGS);
233f2c4db54SSteven Hartland static int  ixgbe_sysctl_advertise(SYSCTL_HANDLER_ARGS);
2348eb6488eSEric Joyner static int  ixgbe_sysctl_interrupt_rate_handler(SYSCTL_HANDLER_ARGS);
2356f37f232SEric Joyner static int  ixgbe_sysctl_dmac(SYSCTL_HANDLER_ARGS);
2366f37f232SEric Joyner static int  ixgbe_sysctl_phy_temp(SYSCTL_HANDLER_ARGS);
2376f37f232SEric Joyner static int  ixgbe_sysctl_phy_overtemp_occurred(SYSCTL_HANDLER_ARGS);
2387660e4eaSKevin Bowling static int  ixgbe_sysctl_print_fw_version(SYSCTL_HANDLER_ARGS);
239a9ca1c79SSean Bruno #ifdef IXGBE_DEBUG
240a9ca1c79SSean Bruno static int  ixgbe_sysctl_power_state(SYSCTL_HANDLER_ARGS);
241a9ca1c79SSean Bruno static int  ixgbe_sysctl_print_rss_config(SYSCTL_HANDLER_ARGS);
242a9ca1c79SSean Bruno #endif
2438eb6488eSEric Joyner static int  ixgbe_sysctl_rdh_handler(SYSCTL_HANDLER_ARGS);
2448eb6488eSEric Joyner static int  ixgbe_sysctl_rdt_handler(SYSCTL_HANDLER_ARGS);
2458eb6488eSEric Joyner static int  ixgbe_sysctl_tdt_handler(SYSCTL_HANDLER_ARGS);
2468eb6488eSEric Joyner static int  ixgbe_sysctl_tdh_handler(SYSCTL_HANDLER_ARGS);
2478eb6488eSEric Joyner static int  ixgbe_sysctl_eee_state(SYSCTL_HANDLER_ARGS);
2486f37f232SEric Joyner static int  ixgbe_sysctl_wol_enable(SYSCTL_HANDLER_ARGS);
2496f37f232SEric Joyner static int  ixgbe_sysctl_wufc(SYSCTL_HANDLER_ARGS);
250eea2e089SMichael Tuexen static int  ixgbe_sysctl_tso_tcp_flags_mask(SYSCTL_HANDLER_ARGS);
251758cc3dcSJack F Vogel 
252758cc3dcSJack F Vogel /* Deferred interrupt tasklets */
253c19c7afeSEric Joyner static void ixgbe_handle_msf(void *);
254c19c7afeSEric Joyner static void ixgbe_handle_mod(void *);
255c19c7afeSEric Joyner static void ixgbe_handle_phy(void *);
256758cc3dcSJack F Vogel 
2578eb6488eSEric Joyner /************************************************************************
258758cc3dcSJack F Vogel  *  FreeBSD Device Interface Entry Points
2598eb6488eSEric Joyner  ************************************************************************/
260a1edda90SAdrian Chadd static device_method_t ix_methods[] = {
261758cc3dcSJack F Vogel 	/* Device interface */
262c19c7afeSEric Joyner 	DEVMETHOD(device_register, ixgbe_register),
263c19c7afeSEric Joyner 	DEVMETHOD(device_probe, iflib_device_probe),
264c19c7afeSEric Joyner 	DEVMETHOD(device_attach, iflib_device_attach),
265c19c7afeSEric Joyner 	DEVMETHOD(device_detach, iflib_device_detach),
266c19c7afeSEric Joyner 	DEVMETHOD(device_shutdown, iflib_device_shutdown),
267c19c7afeSEric Joyner 	DEVMETHOD(device_suspend, iflib_device_suspend),
268c19c7afeSEric Joyner 	DEVMETHOD(device_resume, iflib_device_resume),
26948056c88SJack F Vogel #ifdef PCI_IOV
270c19c7afeSEric Joyner 	DEVMETHOD(pci_iov_init, iflib_device_iov_init),
271c19c7afeSEric Joyner 	DEVMETHOD(pci_iov_uninit, iflib_device_iov_uninit),
272c19c7afeSEric Joyner 	DEVMETHOD(pci_iov_add_vf, iflib_device_iov_add_vf),
27348056c88SJack F Vogel #endif /* PCI_IOV */
274758cc3dcSJack F Vogel 	DEVMETHOD_END
275758cc3dcSJack F Vogel };
276758cc3dcSJack F Vogel 
277a1edda90SAdrian Chadd static driver_t ix_driver = {
278b1d5caf3SKevin Bowling 	"ix", ix_methods, sizeof(struct ixgbe_softc),
279758cc3dcSJack F Vogel };
280758cc3dcSJack F Vogel 
281964c2b3aSJohn Baldwin DRIVER_MODULE(ix, pci, ix_driver, 0, 0);
282937b0f25SYuri Pankov IFLIB_PNP_INFO(pci, ix_driver, ixgbe_vendor_info_array);
283a1edda90SAdrian Chadd MODULE_DEPEND(ix, pci, 1, 1, 1);
284a1edda90SAdrian Chadd MODULE_DEPEND(ix, ether, 1, 1, 1);
285c19c7afeSEric Joyner MODULE_DEPEND(ix, iflib, 1, 1, 1);
286c19c7afeSEric Joyner 
287c19c7afeSEric Joyner static device_method_t ixgbe_if_methods[] = {
288c19c7afeSEric Joyner 	DEVMETHOD(ifdi_attach_pre, ixgbe_if_attach_pre),
289c19c7afeSEric Joyner 	DEVMETHOD(ifdi_attach_post, ixgbe_if_attach_post),
290c19c7afeSEric Joyner 	DEVMETHOD(ifdi_detach, ixgbe_if_detach),
291c19c7afeSEric Joyner 	DEVMETHOD(ifdi_shutdown, ixgbe_if_shutdown),
292c19c7afeSEric Joyner 	DEVMETHOD(ifdi_suspend, ixgbe_if_suspend),
293c19c7afeSEric Joyner 	DEVMETHOD(ifdi_resume, ixgbe_if_resume),
294c19c7afeSEric Joyner 	DEVMETHOD(ifdi_init, ixgbe_if_init),
295c19c7afeSEric Joyner 	DEVMETHOD(ifdi_stop, ixgbe_if_stop),
296c19c7afeSEric Joyner 	DEVMETHOD(ifdi_msix_intr_assign, ixgbe_if_msix_intr_assign),
297c19c7afeSEric Joyner 	DEVMETHOD(ifdi_intr_enable, ixgbe_if_enable_intr),
298c19c7afeSEric Joyner 	DEVMETHOD(ifdi_intr_disable, ixgbe_if_disable_intr),
299b2c1e8e6SEric Joyner 	DEVMETHOD(ifdi_link_intr_enable, ixgbe_link_intr_enable),
300c19c7afeSEric Joyner 	DEVMETHOD(ifdi_tx_queue_intr_enable, ixgbe_if_rx_queue_intr_enable),
301c19c7afeSEric Joyner 	DEVMETHOD(ifdi_rx_queue_intr_enable, ixgbe_if_rx_queue_intr_enable),
302c19c7afeSEric Joyner 	DEVMETHOD(ifdi_tx_queues_alloc, ixgbe_if_tx_queues_alloc),
303c19c7afeSEric Joyner 	DEVMETHOD(ifdi_rx_queues_alloc, ixgbe_if_rx_queues_alloc),
304c19c7afeSEric Joyner 	DEVMETHOD(ifdi_queues_free, ixgbe_if_queues_free),
305c19c7afeSEric Joyner 	DEVMETHOD(ifdi_update_admin_status, ixgbe_if_update_admin_status),
306c19c7afeSEric Joyner 	DEVMETHOD(ifdi_multi_set, ixgbe_if_multi_set),
307c19c7afeSEric Joyner 	DEVMETHOD(ifdi_mtu_set, ixgbe_if_mtu_set),
308c19c7afeSEric Joyner 	DEVMETHOD(ifdi_crcstrip_set, ixgbe_if_crcstrip_set),
309c19c7afeSEric Joyner 	DEVMETHOD(ifdi_media_status, ixgbe_if_media_status),
310c19c7afeSEric Joyner 	DEVMETHOD(ifdi_media_change, ixgbe_if_media_change),
311c19c7afeSEric Joyner 	DEVMETHOD(ifdi_promisc_set, ixgbe_if_promisc_set),
312c19c7afeSEric Joyner 	DEVMETHOD(ifdi_timer, ixgbe_if_timer),
313c19c7afeSEric Joyner 	DEVMETHOD(ifdi_vlan_register, ixgbe_if_vlan_register),
314c19c7afeSEric Joyner 	DEVMETHOD(ifdi_vlan_unregister, ixgbe_if_vlan_unregister),
315c19c7afeSEric Joyner 	DEVMETHOD(ifdi_get_counter, ixgbe_if_get_counter),
316bca38080SAndrew Gallatin 	DEVMETHOD(ifdi_i2c_req, ixgbe_if_i2c_req),
317cf150917SEric Joyner 	DEVMETHOD(ifdi_needs_restart, ixgbe_if_needs_restart),
318c19c7afeSEric Joyner #ifdef PCI_IOV
319c19c7afeSEric Joyner 	DEVMETHOD(ifdi_iov_init, ixgbe_if_iov_init),
320c19c7afeSEric Joyner 	DEVMETHOD(ifdi_iov_uninit, ixgbe_if_iov_uninit),
321c19c7afeSEric Joyner 	DEVMETHOD(ifdi_iov_vf_add, ixgbe_if_iov_vf_add),
322c19c7afeSEric Joyner #endif /* PCI_IOV */
323c19c7afeSEric Joyner 	DEVMETHOD_END
324c19c7afeSEric Joyner };
325758cc3dcSJack F Vogel 
326758cc3dcSJack F Vogel /*
3278eb6488eSEric Joyner  * TUNEABLE PARAMETERS:
328758cc3dcSJack F Vogel  */
329758cc3dcSJack F Vogel 
33020b91f0aSPawel Biernacki static SYSCTL_NODE(_hw, OID_AUTO, ix, CTLFLAG_RD | CTLFLAG_MPSAFE, 0,
33120b91f0aSPawel Biernacki     "IXGBE driver parameters");
332c19c7afeSEric Joyner static driver_t ixgbe_if_driver = {
333b1d5caf3SKevin Bowling   "ixgbe_if", ixgbe_if_methods, sizeof(struct ixgbe_softc)
334c19c7afeSEric Joyner };
335758cc3dcSJack F Vogel 
336758cc3dcSJack F Vogel static int ixgbe_max_interrupt_rate = (4000000 / IXGBE_LOW_LATENCY);
337758cc3dcSJack F Vogel SYSCTL_INT(_hw_ix, OID_AUTO, max_interrupt_rate, CTLFLAG_RDTUN,
338758cc3dcSJack F Vogel     &ixgbe_max_interrupt_rate, 0, "Maximum interrupts per second");
339758cc3dcSJack F Vogel 
340f2c4db54SSteven Hartland /* Flow control setting, default to full */
341f2c4db54SSteven Hartland static int ixgbe_flow_control = ixgbe_fc_full;
342f2c4db54SSteven Hartland SYSCTL_INT(_hw_ix, OID_AUTO, flow_control, CTLFLAG_RDTUN,
343f2c4db54SSteven Hartland     &ixgbe_flow_control, 0, "Default flow control used for all adapters");
344f2c4db54SSteven Hartland 
345f2c4db54SSteven Hartland /* Advertise Speed, default to 0 (auto) */
346f2c4db54SSteven Hartland static int ixgbe_advertise_speed = 0;
347f2c4db54SSteven Hartland SYSCTL_INT(_hw_ix, OID_AUTO, advertise_speed, CTLFLAG_RDTUN,
348f2c4db54SSteven Hartland     &ixgbe_advertise_speed, 0, "Default advertised speed for all adapters");
349f2c4db54SSteven Hartland 
350758cc3dcSJack F Vogel /*
3518eb6488eSEric Joyner  * Smart speed setting, default to on
3528eb6488eSEric Joyner  * this only works as a compile option
3538eb6488eSEric Joyner  * right now as its during attach, set
3548eb6488eSEric Joyner  * this to 'ixgbe_smart_speed_off' to
3558eb6488eSEric Joyner  * disable.
356758cc3dcSJack F Vogel  */
357758cc3dcSJack F Vogel static int ixgbe_smart_speed = ixgbe_smart_speed_on;
358758cc3dcSJack F Vogel 
359758cc3dcSJack F Vogel /*
3608eb6488eSEric Joyner  * MSI-X should be the default for best performance,
361758cc3dcSJack F Vogel  * but this allows it to be forced off for testing.
362758cc3dcSJack F Vogel  */
363758cc3dcSJack F Vogel static int ixgbe_enable_msix = 1;
364c58d34ddSKevin Bowling SYSCTL_INT(_hw_ix, OID_AUTO, enable_msix, CTLFLAG_RDTUN, &ixgbe_enable_msix,
365c58d34ddSKevin Bowling     0,
366758cc3dcSJack F Vogel     "Enable MSI-X interrupts");
367758cc3dcSJack F Vogel 
368758cc3dcSJack F Vogel /*
3698eb6488eSEric Joyner  * Defining this on will allow the use
3708eb6488eSEric Joyner  * of unsupported SFP+ modules, note that
3718eb6488eSEric Joyner  * doing so you are on your own :)
372758cc3dcSJack F Vogel  */
37379b36ec9SKevin Bowling static int allow_unsupported_sfp = false;
374fb6aa95dSSean Bruno SYSCTL_INT(_hw_ix, OID_AUTO, unsupported_sfp, CTLFLAG_RDTUN,
3758eb6488eSEric Joyner     &allow_unsupported_sfp, 0,
3768eb6488eSEric Joyner     "Allow unsupported SFP modules...use at your own risk");
3778eb6488eSEric Joyner 
3788eb6488eSEric Joyner /*
3798eb6488eSEric Joyner  * Not sure if Flow Director is fully baked,
3808eb6488eSEric Joyner  * so we'll default to turning it off.
3818eb6488eSEric Joyner  */
3828eb6488eSEric Joyner static int ixgbe_enable_fdir = 0;
383c58d34ddSKevin Bowling SYSCTL_INT(_hw_ix, OID_AUTO, enable_fdir, CTLFLAG_RDTUN, &ixgbe_enable_fdir,
384c58d34ddSKevin Bowling     0,
3858eb6488eSEric Joyner     "Enable Flow Director");
3868eb6488eSEric Joyner 
3878eb6488eSEric Joyner /* Receive-Side Scaling */
3888eb6488eSEric Joyner static int ixgbe_enable_rss = 1;
389c58d34ddSKevin Bowling SYSCTL_INT(_hw_ix, OID_AUTO, enable_rss, CTLFLAG_RDTUN, &ixgbe_enable_rss,
390c58d34ddSKevin Bowling     0,
3918eb6488eSEric Joyner     "Enable Receive-Side Scaling (RSS)");
392758cc3dcSJack F Vogel 
39364881da4SSai Rajesh Tallamraju /*
39464881da4SSai Rajesh Tallamraju  * AIM: Adaptive Interrupt Moderation
39564881da4SSai Rajesh Tallamraju  * which means that the interrupt rate
39664881da4SSai Rajesh Tallamraju  * is varied over time based on the
39764881da4SSai Rajesh Tallamraju  * traffic for that interrupt vector
39864881da4SSai Rajesh Tallamraju  */
39979b36ec9SKevin Bowling static int ixgbe_enable_aim = false;
400c58d34ddSKevin Bowling SYSCTL_INT(_hw_ix, OID_AUTO, enable_aim, CTLFLAG_RWTUN, &ixgbe_enable_aim,
401c58d34ddSKevin Bowling     0,
40264881da4SSai Rajesh Tallamraju     "Enable adaptive interrupt moderation");
40364881da4SSai Rajesh Tallamraju 
404c19c7afeSEric Joyner #if 0
405758cc3dcSJack F Vogel /* Keep running tab on them for sanity check */
406758cc3dcSJack F Vogel static int ixgbe_total_ports;
407c19c7afeSEric Joyner #endif
408758cc3dcSJack F Vogel 
4098eb6488eSEric Joyner MALLOC_DEFINE(M_IXGBE, "ix", "ix driver allocations");
410758cc3dcSJack F Vogel 
411c19c7afeSEric Joyner /*
412c19c7afeSEric Joyner  * For Flow Director: this is the number of TX packets we sample
413c19c7afeSEric Joyner  * for the filter pool, this means every 20th packet will be probed.
414c19c7afeSEric Joyner  *
415c19c7afeSEric Joyner  * This feature can be disabled by setting this to 0.
416c19c7afeSEric Joyner  */
417c19c7afeSEric Joyner static int atr_sample_rate = 20;
418c19c7afeSEric Joyner 
419c19c7afeSEric Joyner extern struct if_txrx ixgbe_txrx;
420c19c7afeSEric Joyner 
421c19c7afeSEric Joyner static struct if_shared_ctx ixgbe_sctx_init = {
422c19c7afeSEric Joyner 	.isc_magic = IFLIB_MAGIC,
423c19c7afeSEric Joyner 	.isc_q_align = PAGE_SIZE,/* max(DBA_ALIGN, PAGE_SIZE) */
4247f87c040SMarius Strobl 	.isc_tx_maxsize = IXGBE_TSO_SIZE + sizeof(struct ether_vlan_header),
425c19c7afeSEric Joyner 	.isc_tx_maxsegsize = PAGE_SIZE,
4267f87c040SMarius Strobl 	.isc_tso_maxsize = IXGBE_TSO_SIZE + sizeof(struct ether_vlan_header),
4277f87c040SMarius Strobl 	.isc_tso_maxsegsize = PAGE_SIZE,
428c19c7afeSEric Joyner 	.isc_rx_maxsize = PAGE_SIZE*4,
429c19c7afeSEric Joyner 	.isc_rx_nsegments = 1,
430c19c7afeSEric Joyner 	.isc_rx_maxsegsize = PAGE_SIZE*4,
431c19c7afeSEric Joyner 	.isc_nfl = 1,
432c19c7afeSEric Joyner 	.isc_ntxqs = 1,
433c19c7afeSEric Joyner 	.isc_nrxqs = 1,
434c19c7afeSEric Joyner 
435c19c7afeSEric Joyner 	.isc_admin_intrcnt = 1,
436c19c7afeSEric Joyner 	.isc_vendor_info = ixgbe_vendor_info_array,
437c19c7afeSEric Joyner 	.isc_driver_version = ixgbe_driver_version,
438c19c7afeSEric Joyner 	.isc_driver = &ixgbe_if_driver,
4397aad1f4eSEric Joyner 	.isc_flags = IFLIB_TSO_INIT_IP,
440c19c7afeSEric Joyner 
441c19c7afeSEric Joyner 	.isc_nrxd_min = {MIN_RXD},
442c19c7afeSEric Joyner 	.isc_ntxd_min = {MIN_TXD},
443c19c7afeSEric Joyner 	.isc_nrxd_max = {MAX_RXD},
444c19c7afeSEric Joyner 	.isc_ntxd_max = {MAX_TXD},
445c19c7afeSEric Joyner 	.isc_nrxd_default = {DEFAULT_RXD},
446c19c7afeSEric Joyner 	.isc_ntxd_default = {DEFAULT_TXD},
447c19c7afeSEric Joyner };
448c19c7afeSEric Joyner 
449c19c7afeSEric Joyner /************************************************************************
450c19c7afeSEric Joyner  * ixgbe_if_tx_queues_alloc
451c19c7afeSEric Joyner  ************************************************************************/
452c19c7afeSEric Joyner static int
ixgbe_if_tx_queues_alloc(if_ctx_t ctx,caddr_t * vaddrs,uint64_t * paddrs,int ntxqs,int ntxqsets)453c19c7afeSEric Joyner ixgbe_if_tx_queues_alloc(if_ctx_t ctx, caddr_t *vaddrs, uint64_t *paddrs,
454c19c7afeSEric Joyner     int ntxqs, int ntxqsets)
455c19c7afeSEric Joyner {
456b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
457b1d5caf3SKevin Bowling 	if_softc_ctx_t scctx = sc->shared;
458c19c7afeSEric Joyner 	struct ix_tx_queue *que;
459c19c7afeSEric Joyner 	int i, j, error;
460c19c7afeSEric Joyner 
461b1d5caf3SKevin Bowling 	MPASS(sc->num_tx_queues > 0);
462b1d5caf3SKevin Bowling 	MPASS(sc->num_tx_queues == ntxqsets);
463c19c7afeSEric Joyner 	MPASS(ntxqs == 1);
464c19c7afeSEric Joyner 
465c19c7afeSEric Joyner 	/* Allocate queue structure memory */
466b1d5caf3SKevin Bowling 	sc->tx_queues =
467c58d34ddSKevin Bowling 	    (struct ix_tx_queue *)malloc(sizeof(struct ix_tx_queue) *
468c58d34ddSKevin Bowling 	    ntxqsets, M_IXGBE, M_NOWAIT | M_ZERO);
469b1d5caf3SKevin Bowling 	if (!sc->tx_queues) {
470c19c7afeSEric Joyner 		device_printf(iflib_get_dev(ctx),
471c19c7afeSEric Joyner 		    "Unable to allocate TX ring memory\n");
472c19c7afeSEric Joyner 		return (ENOMEM);
473c19c7afeSEric Joyner 	}
474c19c7afeSEric Joyner 
475b1d5caf3SKevin Bowling 	for (i = 0, que = sc->tx_queues; i < ntxqsets; i++, que++) {
476c19c7afeSEric Joyner 		struct tx_ring *txr = &que->txr;
477c19c7afeSEric Joyner 
478c19c7afeSEric Joyner 		/* In case SR-IOV is enabled, align the index properly */
479c58d34ddSKevin Bowling 		txr->me = ixgbe_vf_que_index(sc->iov_mode, sc->pool, i);
480c19c7afeSEric Joyner 
481b1d5caf3SKevin Bowling 		txr->sc = que->sc = sc;
482c19c7afeSEric Joyner 
483c19c7afeSEric Joyner 		/* Allocate report status array */
484c58d34ddSKevin Bowling 		txr->tx_rsq = (qidx_t *)malloc(sizeof(qidx_t) *
485c58d34ddSKevin Bowling 		    scctx->isc_ntxd[0], M_IXGBE, M_NOWAIT | M_ZERO);
486c19c7afeSEric Joyner 		if (txr->tx_rsq == NULL) {
487c19c7afeSEric Joyner 			error = ENOMEM;
488c19c7afeSEric Joyner 			goto fail;
489c19c7afeSEric Joyner 		}
490c19c7afeSEric Joyner 		for (j = 0; j < scctx->isc_ntxd[0]; j++)
491c19c7afeSEric Joyner 			txr->tx_rsq[j] = QIDX_INVALID;
492c58d34ddSKevin Bowling 		/* get virtual and physical address of the hardware queues */
493c19c7afeSEric Joyner 		txr->tail = IXGBE_TDT(txr->me);
494c19c7afeSEric Joyner 		txr->tx_base = (union ixgbe_adv_tx_desc *)vaddrs[i];
495c19c7afeSEric Joyner 		txr->tx_paddr = paddrs[i];
496c19c7afeSEric Joyner 
497c19c7afeSEric Joyner 		txr->bytes = 0;
498c19c7afeSEric Joyner 		txr->total_packets = 0;
499c19c7afeSEric Joyner 
500c19c7afeSEric Joyner 		/* Set the rate at which we sample packets */
501b1d5caf3SKevin Bowling 		if (sc->feat_en & IXGBE_FEATURE_FDIR)
502c19c7afeSEric Joyner 			txr->atr_sample = atr_sample_rate;
503c19c7afeSEric Joyner 
504c19c7afeSEric Joyner 	}
505c19c7afeSEric Joyner 
506c19c7afeSEric Joyner 	device_printf(iflib_get_dev(ctx), "allocated for %d queues\n",
507b1d5caf3SKevin Bowling 	    sc->num_tx_queues);
508c19c7afeSEric Joyner 
509c19c7afeSEric Joyner 	return (0);
510c19c7afeSEric Joyner 
511c19c7afeSEric Joyner fail:
512c19c7afeSEric Joyner 	ixgbe_if_queues_free(ctx);
513c19c7afeSEric Joyner 
514c19c7afeSEric Joyner 	return (error);
515c19c7afeSEric Joyner } /* ixgbe_if_tx_queues_alloc */
516c19c7afeSEric Joyner 
517c19c7afeSEric Joyner /************************************************************************
518c19c7afeSEric Joyner  * ixgbe_if_rx_queues_alloc
519c19c7afeSEric Joyner  ************************************************************************/
520c19c7afeSEric Joyner static int
ixgbe_if_rx_queues_alloc(if_ctx_t ctx,caddr_t * vaddrs,uint64_t * paddrs,int nrxqs,int nrxqsets)521c19c7afeSEric Joyner ixgbe_if_rx_queues_alloc(if_ctx_t ctx, caddr_t *vaddrs, uint64_t *paddrs,
522c19c7afeSEric Joyner     int nrxqs, int nrxqsets)
523c19c7afeSEric Joyner {
524b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
525c19c7afeSEric Joyner 	struct ix_rx_queue *que;
526c19c7afeSEric Joyner 	int i;
527c19c7afeSEric Joyner 
528b1d5caf3SKevin Bowling 	MPASS(sc->num_rx_queues > 0);
529b1d5caf3SKevin Bowling 	MPASS(sc->num_rx_queues == nrxqsets);
530c19c7afeSEric Joyner 	MPASS(nrxqs == 1);
531c19c7afeSEric Joyner 
532c19c7afeSEric Joyner 	/* Allocate queue structure memory */
533b1d5caf3SKevin Bowling 	sc->rx_queues =
534c19c7afeSEric Joyner 	    (struct ix_rx_queue *)malloc(sizeof(struct ix_rx_queue)*nrxqsets,
535c19c7afeSEric Joyner 	    M_IXGBE, M_NOWAIT | M_ZERO);
536b1d5caf3SKevin Bowling 	if (!sc->rx_queues) {
537c19c7afeSEric Joyner 		device_printf(iflib_get_dev(ctx),
538c19c7afeSEric Joyner 		    "Unable to allocate TX ring memory\n");
539c19c7afeSEric Joyner 		return (ENOMEM);
540c19c7afeSEric Joyner 	}
541c19c7afeSEric Joyner 
542b1d5caf3SKevin Bowling 	for (i = 0, que = sc->rx_queues; i < nrxqsets; i++, que++) {
543c19c7afeSEric Joyner 		struct rx_ring *rxr = &que->rxr;
544c19c7afeSEric Joyner 
545c19c7afeSEric Joyner 		/* In case SR-IOV is enabled, align the index properly */
546c58d34ddSKevin Bowling 		rxr->me = ixgbe_vf_que_index(sc->iov_mode, sc->pool, i);
547c19c7afeSEric Joyner 
548b1d5caf3SKevin Bowling 		rxr->sc = que->sc = sc;
549c19c7afeSEric Joyner 
550c19c7afeSEric Joyner 		/* get the virtual and physical address of the hw queues */
551c19c7afeSEric Joyner 		rxr->tail = IXGBE_RDT(rxr->me);
552c19c7afeSEric Joyner 		rxr->rx_base = (union ixgbe_adv_rx_desc *)vaddrs[i];
553c19c7afeSEric Joyner 		rxr->rx_paddr = paddrs[i];
554c19c7afeSEric Joyner 		rxr->bytes = 0;
555c19c7afeSEric Joyner 		rxr->que = que;
556c19c7afeSEric Joyner 	}
557c19c7afeSEric Joyner 
558c19c7afeSEric Joyner 	device_printf(iflib_get_dev(ctx), "allocated for %d rx queues\n",
559b1d5caf3SKevin Bowling 	    sc->num_rx_queues);
560c19c7afeSEric Joyner 
561c19c7afeSEric Joyner 	return (0);
562c19c7afeSEric Joyner } /* ixgbe_if_rx_queues_alloc */
563c19c7afeSEric Joyner 
564c19c7afeSEric Joyner /************************************************************************
565c19c7afeSEric Joyner  * ixgbe_if_queues_free
566c19c7afeSEric Joyner  ************************************************************************/
567c19c7afeSEric Joyner static void
ixgbe_if_queues_free(if_ctx_t ctx)568c19c7afeSEric Joyner ixgbe_if_queues_free(if_ctx_t ctx)
569c19c7afeSEric Joyner {
570b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
571b1d5caf3SKevin Bowling 	struct ix_tx_queue *tx_que = sc->tx_queues;
572b1d5caf3SKevin Bowling 	struct ix_rx_queue *rx_que = sc->rx_queues;
573c19c7afeSEric Joyner 	int i;
574c19c7afeSEric Joyner 
575c19c7afeSEric Joyner 	if (tx_que != NULL) {
576b1d5caf3SKevin Bowling 		for (i = 0; i < sc->num_tx_queues; i++, tx_que++) {
577c19c7afeSEric Joyner 			struct tx_ring *txr = &tx_que->txr;
578c19c7afeSEric Joyner 			if (txr->tx_rsq == NULL)
579c19c7afeSEric Joyner 				break;
580c19c7afeSEric Joyner 
581c19c7afeSEric Joyner 			free(txr->tx_rsq, M_IXGBE);
582c19c7afeSEric Joyner 			txr->tx_rsq = NULL;
583c19c7afeSEric Joyner 		}
584c19c7afeSEric Joyner 
585b1d5caf3SKevin Bowling 		free(sc->tx_queues, M_IXGBE);
586b1d5caf3SKevin Bowling 		sc->tx_queues = NULL;
587c19c7afeSEric Joyner 	}
588c19c7afeSEric Joyner 	if (rx_que != NULL) {
589b1d5caf3SKevin Bowling 		free(sc->rx_queues, M_IXGBE);
590b1d5caf3SKevin Bowling 		sc->rx_queues = NULL;
591c19c7afeSEric Joyner 	}
592c19c7afeSEric Joyner } /* ixgbe_if_queues_free */
593c19c7afeSEric Joyner 
5948eb6488eSEric Joyner /************************************************************************
5958eb6488eSEric Joyner  * ixgbe_initialize_rss_mapping
5968eb6488eSEric Joyner  ************************************************************************/
5978eb6488eSEric Joyner static void
ixgbe_initialize_rss_mapping(struct ixgbe_softc * sc)598b1d5caf3SKevin Bowling ixgbe_initialize_rss_mapping(struct ixgbe_softc *sc)
599758cc3dcSJack F Vogel {
600b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
6018eb6488eSEric Joyner 	u32 reta = 0, mrqc, rss_key[10];
6028eb6488eSEric Joyner 	int queue_id, table_size, index_mult;
6038eb6488eSEric Joyner 	int i, j;
6048eb6488eSEric Joyner 	u32 rss_hash_config;
6056f37f232SEric Joyner 
606b1d5caf3SKevin Bowling 	if (sc->feat_en & IXGBE_FEATURE_RSS) {
6078eb6488eSEric Joyner 		/* Fetch the configured RSS key */
6088eb6488eSEric Joyner 		rss_getkey((uint8_t *)&rss_key);
609758cc3dcSJack F Vogel 	} else {
6108eb6488eSEric Joyner 		/* set up random bits */
6118eb6488eSEric Joyner 		arc4rand(&rss_key, sizeof(rss_key), 0);
612758cc3dcSJack F Vogel 	}
613758cc3dcSJack F Vogel 
6148eb6488eSEric Joyner 	/* Set multiplier for RETA setup and table size based on MAC */
6158eb6488eSEric Joyner 	index_mult = 0x1;
6168eb6488eSEric Joyner 	table_size = 128;
617b1d5caf3SKevin Bowling 	switch (sc->hw.mac.type) {
6188eb6488eSEric Joyner 	case ixgbe_mac_82598EB:
6198eb6488eSEric Joyner 		index_mult = 0x11;
620758cc3dcSJack F Vogel 		break;
6218eb6488eSEric Joyner 	case ixgbe_mac_X550:
6228eb6488eSEric Joyner 	case ixgbe_mac_X550EM_x:
6238eb6488eSEric Joyner 	case ixgbe_mac_X550EM_a:
6248eb6488eSEric Joyner 		table_size = 512;
6258eb6488eSEric Joyner 		break;
626758cc3dcSJack F Vogel 	default:
627758cc3dcSJack F Vogel 		break;
628758cc3dcSJack F Vogel 	}
629758cc3dcSJack F Vogel 
6308eb6488eSEric Joyner 	/* Set up the redirection table */
6318eb6488eSEric Joyner 	for (i = 0, j = 0; i < table_size; i++, j++) {
632b1d5caf3SKevin Bowling 		if (j == sc->num_rx_queues)
6338eb6488eSEric Joyner 			j = 0;
634758cc3dcSJack F Vogel 
635b1d5caf3SKevin Bowling 		if (sc->feat_en & IXGBE_FEATURE_RSS) {
636a9ca1c79SSean Bruno 			/*
6378eb6488eSEric Joyner 			 * Fetch the RSS bucket id for the given indirection
6388eb6488eSEric Joyner 			 * entry. Cap it at the number of configured buckets
639c19c7afeSEric Joyner 			 * (which is num_rx_queues.)
640a9ca1c79SSean Bruno 			 */
6418eb6488eSEric Joyner 			queue_id = rss_get_indirection_to_bucket(i);
642b1d5caf3SKevin Bowling 			queue_id = queue_id % sc->num_rx_queues;
643758cc3dcSJack F Vogel 		} else
6448eb6488eSEric Joyner 			queue_id = (j * index_mult);
6458eb6488eSEric Joyner 
6468eb6488eSEric Joyner 		/*
6478eb6488eSEric Joyner 		 * The low 8 bits are for hash value (n+0);
6488eb6488eSEric Joyner 		 * The next 8 bits are for hash value (n+1), etc.
6498eb6488eSEric Joyner 		 */
6508eb6488eSEric Joyner 		reta = reta >> 8;
6518eb6488eSEric Joyner 		reta = reta | (((uint32_t)queue_id) << 24);
6528eb6488eSEric Joyner 		if ((i & 3) == 3) {
6538eb6488eSEric Joyner 			if (i < 128)
6548eb6488eSEric Joyner 				IXGBE_WRITE_REG(hw, IXGBE_RETA(i >> 2), reta);
6558eb6488eSEric Joyner 			else
656c58d34ddSKevin Bowling 				IXGBE_WRITE_REG(hw,
657c58d34ddSKevin Bowling 				    IXGBE_ERETA((i >> 2) - 32), reta);
6588eb6488eSEric Joyner 			reta = 0;
6598eb6488eSEric Joyner 		}
660758cc3dcSJack F Vogel 	}
661758cc3dcSJack F Vogel 
6628eb6488eSEric Joyner 	/* Now fill our hash function seeds */
6638eb6488eSEric Joyner 	for (i = 0; i < 10; i++)
6648eb6488eSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_RSSRK(i), rss_key[i]);
665758cc3dcSJack F Vogel 
6668eb6488eSEric Joyner 	/* Perform hash on these packet types */
667b1d5caf3SKevin Bowling 	if (sc->feat_en & IXGBE_FEATURE_RSS)
6688eb6488eSEric Joyner 		rss_hash_config = rss_gethashconfig();
669758cc3dcSJack F Vogel 	else {
670758cc3dcSJack F Vogel 		/*
6718eb6488eSEric Joyner 		 * Disable UDP - IP fragments aren't currently being handled
6728eb6488eSEric Joyner 		 * and so we end up with a mix of 2-tuple and 4-tuple
6738eb6488eSEric Joyner 		 * traffic.
674758cc3dcSJack F Vogel 		 */
675c58d34ddSKevin Bowling 		rss_hash_config = RSS_HASHTYPE_RSS_IPV4 |
676c58d34ddSKevin Bowling 		    RSS_HASHTYPE_RSS_TCP_IPV4 |
677c58d34ddSKevin Bowling 		    RSS_HASHTYPE_RSS_IPV6 |
678c58d34ddSKevin Bowling 		    RSS_HASHTYPE_RSS_TCP_IPV6 |
679c58d34ddSKevin Bowling 		    RSS_HASHTYPE_RSS_IPV6_EX |
680c58d34ddSKevin Bowling 		    RSS_HASHTYPE_RSS_TCP_IPV6_EX;
681758cc3dcSJack F Vogel 	}
682758cc3dcSJack F Vogel 
6838eb6488eSEric Joyner 	mrqc = IXGBE_MRQC_RSSEN;
6848eb6488eSEric Joyner 	if (rss_hash_config & RSS_HASHTYPE_RSS_IPV4)
6858eb6488eSEric Joyner 		mrqc |= IXGBE_MRQC_RSS_FIELD_IPV4;
6868eb6488eSEric Joyner 	if (rss_hash_config & RSS_HASHTYPE_RSS_TCP_IPV4)
6878eb6488eSEric Joyner 		mrqc |= IXGBE_MRQC_RSS_FIELD_IPV4_TCP;
6888eb6488eSEric Joyner 	if (rss_hash_config & RSS_HASHTYPE_RSS_IPV6)
6898eb6488eSEric Joyner 		mrqc |= IXGBE_MRQC_RSS_FIELD_IPV6;
6908eb6488eSEric Joyner 	if (rss_hash_config & RSS_HASHTYPE_RSS_TCP_IPV6)
6918eb6488eSEric Joyner 		mrqc |= IXGBE_MRQC_RSS_FIELD_IPV6_TCP;
6928eb6488eSEric Joyner 	if (rss_hash_config & RSS_HASHTYPE_RSS_IPV6_EX)
6938eb6488eSEric Joyner 		mrqc |= IXGBE_MRQC_RSS_FIELD_IPV6_EX;
6948eb6488eSEric Joyner 	if (rss_hash_config & RSS_HASHTYPE_RSS_TCP_IPV6_EX)
6958eb6488eSEric Joyner 		mrqc |= IXGBE_MRQC_RSS_FIELD_IPV6_EX_TCP;
6968eb6488eSEric Joyner 	if (rss_hash_config & RSS_HASHTYPE_RSS_UDP_IPV4)
6978eb6488eSEric Joyner 		mrqc |= IXGBE_MRQC_RSS_FIELD_IPV4_UDP;
6988eb6488eSEric Joyner 	if (rss_hash_config & RSS_HASHTYPE_RSS_UDP_IPV6)
6998eb6488eSEric Joyner 		mrqc |= IXGBE_MRQC_RSS_FIELD_IPV6_UDP;
7008eb6488eSEric Joyner 	if (rss_hash_config & RSS_HASHTYPE_RSS_UDP_IPV6_EX)
7018eb6488eSEric Joyner 		mrqc |= IXGBE_MRQC_RSS_FIELD_IPV6_EX_UDP;
702b1d5caf3SKevin Bowling 	mrqc |= ixgbe_get_mrqc(sc->iov_mode);
7038eb6488eSEric Joyner 	IXGBE_WRITE_REG(hw, IXGBE_MRQC, mrqc);
7048eb6488eSEric Joyner } /* ixgbe_initialize_rss_mapping */
705758cc3dcSJack F Vogel 
7068eb6488eSEric Joyner /************************************************************************
7078eb6488eSEric Joyner  * ixgbe_initialize_receive_units - Setup receive registers and features.
7088eb6488eSEric Joyner  ************************************************************************/
7098eb6488eSEric Joyner #define BSIZEPKT_ROUNDUP ((1<<IXGBE_SRRCTL_BSIZEPKT_SHIFT)-1)
710758cc3dcSJack F Vogel 
711758cc3dcSJack F Vogel static void
ixgbe_initialize_receive_units(if_ctx_t ctx)712c19c7afeSEric Joyner ixgbe_initialize_receive_units(if_ctx_t ctx)
713758cc3dcSJack F Vogel {
714b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
715b1d5caf3SKevin Bowling 	if_softc_ctx_t scctx = sc->shared;
716b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
717ff06a8dbSJustin Hibbits 	if_t ifp = iflib_get_ifp(ctx);
718c19c7afeSEric Joyner 	struct ix_rx_queue *que;
7198eb6488eSEric Joyner 	int i, j;
7208eb6488eSEric Joyner 	u32 bufsz, fctrl, srrctl, rxcsum;
7218eb6488eSEric Joyner 	u32 hlreg;
72248056c88SJack F Vogel 
72348056c88SJack F Vogel 	/*
7248eb6488eSEric Joyner 	 * Make sure receives are disabled while
7258eb6488eSEric Joyner 	 * setting up the descriptor ring
72648056c88SJack F Vogel 	 */
7278eb6488eSEric Joyner 	ixgbe_disable_rx(hw);
7288eb6488eSEric Joyner 
7298eb6488eSEric Joyner 	/* Enable broadcasts */
7308eb6488eSEric Joyner 	fctrl = IXGBE_READ_REG(hw, IXGBE_FCTRL);
7318eb6488eSEric Joyner 	fctrl |= IXGBE_FCTRL_BAM;
732b1d5caf3SKevin Bowling 	if (sc->hw.mac.type == ixgbe_mac_82598EB) {
7338eb6488eSEric Joyner 		fctrl |= IXGBE_FCTRL_DPF;
7348eb6488eSEric Joyner 		fctrl |= IXGBE_FCTRL_PMCF;
73548056c88SJack F Vogel 	}
7368eb6488eSEric Joyner 	IXGBE_WRITE_REG(hw, IXGBE_FCTRL, fctrl);
7378eb6488eSEric Joyner 
7388eb6488eSEric Joyner 	/* Set for Jumbo Frames? */
7398eb6488eSEric Joyner 	hlreg = IXGBE_READ_REG(hw, IXGBE_HLREG0);
740ff06a8dbSJustin Hibbits 	if (if_getmtu(ifp) > ETHERMTU)
7418eb6488eSEric Joyner 		hlreg |= IXGBE_HLREG0_JUMBOEN;
7428eb6488eSEric Joyner 	else
7438eb6488eSEric Joyner 		hlreg &= ~IXGBE_HLREG0_JUMBOEN;
7448eb6488eSEric Joyner 	IXGBE_WRITE_REG(hw, IXGBE_HLREG0, hlreg);
7458eb6488eSEric Joyner 
746b1d5caf3SKevin Bowling 	bufsz = (sc->rx_mbuf_sz + BSIZEPKT_ROUNDUP) >>
7478eb6488eSEric Joyner 	    IXGBE_SRRCTL_BSIZEPKT_SHIFT;
7488eb6488eSEric Joyner 
749c19c7afeSEric Joyner 	/* Setup the Base and Length of the Rx Descriptor Ring */
750b1d5caf3SKevin Bowling 	for (i = 0, que = sc->rx_queues; i < sc->num_rx_queues; i++, que++) {
751c19c7afeSEric Joyner 		struct rx_ring *rxr = &que->rxr;
752c19c7afeSEric Joyner 		u64 rdba = rxr->rx_paddr;
753c19c7afeSEric Joyner 
7548eb6488eSEric Joyner 		j = rxr->me;
7558eb6488eSEric Joyner 
7568eb6488eSEric Joyner 		/* Setup the Base and Length of the Rx Descriptor Ring */
7578eb6488eSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_RDBAL(j),
7588eb6488eSEric Joyner 		    (rdba & 0x00000000ffffffffULL));
7598eb6488eSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_RDBAH(j), (rdba >> 32));
7608eb6488eSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_RDLEN(j),
761c19c7afeSEric Joyner 		     scctx->isc_nrxd[0] * sizeof(union ixgbe_adv_rx_desc));
7628eb6488eSEric Joyner 
7638eb6488eSEric Joyner 		/* Set up the SRRCTL register */
7648eb6488eSEric Joyner 		srrctl = IXGBE_READ_REG(hw, IXGBE_SRRCTL(j));
7658eb6488eSEric Joyner 		srrctl &= ~IXGBE_SRRCTL_BSIZEHDR_MASK;
7668eb6488eSEric Joyner 		srrctl &= ~IXGBE_SRRCTL_BSIZEPKT_MASK;
7678eb6488eSEric Joyner 		srrctl |= bufsz;
7688eb6488eSEric Joyner 		srrctl |= IXGBE_SRRCTL_DESCTYPE_ADV_ONEBUF;
76948056c88SJack F Vogel 
77048056c88SJack F Vogel 		/*
7718eb6488eSEric Joyner 		 * Set DROP_EN iff we have no flow control and >1 queue.
7728eb6488eSEric Joyner 		 * Note that srrctl was cleared shortly before during reset,
7738eb6488eSEric Joyner 		 * so we do not need to clear the bit, but do it just in case
7748eb6488eSEric Joyner 		 * this code is moved elsewhere.
77548056c88SJack F Vogel 		 */
776b1d5caf3SKevin Bowling 		if (sc->num_rx_queues > 1 &&
777b1d5caf3SKevin Bowling 		    sc->hw.fc.requested_mode == ixgbe_fc_none) {
7788eb6488eSEric Joyner 			srrctl |= IXGBE_SRRCTL_DROP_EN;
779758cc3dcSJack F Vogel 		} else {
7808eb6488eSEric Joyner 			srrctl &= ~IXGBE_SRRCTL_DROP_EN;
781758cc3dcSJack F Vogel 		}
782758cc3dcSJack F Vogel 
7838eb6488eSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_SRRCTL(j), srrctl);
784758cc3dcSJack F Vogel 
7858eb6488eSEric Joyner 		/* Setup the HW Rx Head and Tail Descriptor Pointers */
7868eb6488eSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_RDH(j), 0);
7878eb6488eSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_RDT(j), 0);
788758cc3dcSJack F Vogel 
7898eb6488eSEric Joyner 		/* Set the driver rx tail address */
7908eb6488eSEric Joyner 		rxr->tail =  IXGBE_RDT(rxr->me);
791a9ca1c79SSean Bruno 	}
792758cc3dcSJack F Vogel 
793b1d5caf3SKevin Bowling 	if (sc->hw.mac.type != ixgbe_mac_82598EB) {
794c58d34ddSKevin Bowling 		u32 psrtype = IXGBE_PSRTYPE_TCPHDR |
795c58d34ddSKevin Bowling 		    IXGBE_PSRTYPE_UDPHDR |
796c58d34ddSKevin Bowling 		    IXGBE_PSRTYPE_IPV4HDR |
797c58d34ddSKevin Bowling 		    IXGBE_PSRTYPE_IPV6HDR;
7988eb6488eSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_PSRTYPE(0), psrtype);
7996f37f232SEric Joyner 	}
8006f37f232SEric Joyner 
8018eb6488eSEric Joyner 	rxcsum = IXGBE_READ_REG(hw, IXGBE_RXCSUM);
8028eb6488eSEric Joyner 
803b1d5caf3SKevin Bowling 	ixgbe_initialize_rss_mapping(sc);
8048eb6488eSEric Joyner 
805156424fcSPrzemyslaw Lewandowski 	if (sc->feat_en & IXGBE_FEATURE_RSS) {
8068eb6488eSEric Joyner 		/* RSS and RX IPP Checksum are mutually exclusive */
8078eb6488eSEric Joyner 		rxcsum |= IXGBE_RXCSUM_PCSD;
8086f37f232SEric Joyner 	}
8096f37f232SEric Joyner 
810ff06a8dbSJustin Hibbits 	if (if_getcapenable(ifp) & IFCAP_RXCSUM)
8118eb6488eSEric Joyner 		rxcsum |= IXGBE_RXCSUM_PCSD;
812758cc3dcSJack F Vogel 
8138eb6488eSEric Joyner 	/* This is useful for calculating UDP/IP fragment checksums */
8148eb6488eSEric Joyner 	if (!(rxcsum & IXGBE_RXCSUM_PCSD))
8158eb6488eSEric Joyner 		rxcsum |= IXGBE_RXCSUM_IPPCSE;
816758cc3dcSJack F Vogel 
8178eb6488eSEric Joyner 	IXGBE_WRITE_REG(hw, IXGBE_RXCSUM, rxcsum);
818758cc3dcSJack F Vogel 
8198eb6488eSEric Joyner } /* ixgbe_initialize_receive_units */
820758cc3dcSJack F Vogel 
8218eb6488eSEric Joyner /************************************************************************
8228eb6488eSEric Joyner  * ixgbe_initialize_transmit_units - Enable transmit units.
8238eb6488eSEric Joyner  ************************************************************************/
824758cc3dcSJack F Vogel static void
ixgbe_initialize_transmit_units(if_ctx_t ctx)825c19c7afeSEric Joyner ixgbe_initialize_transmit_units(if_ctx_t ctx)
826758cc3dcSJack F Vogel {
827b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
828b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
829b1d5caf3SKevin Bowling 	if_softc_ctx_t scctx = sc->shared;
830c19c7afeSEric Joyner 	struct ix_tx_queue *que;
831c19c7afeSEric Joyner 	int i;
832758cc3dcSJack F Vogel 
833758cc3dcSJack F Vogel 	/* Setup the Base and Length of the Tx Descriptor Ring */
834b1d5caf3SKevin Bowling 	for (i = 0, que = sc->tx_queues; i < sc->num_tx_queues;
835c19c7afeSEric Joyner 	    i++, que++) {
836c19c7afeSEric Joyner 		struct tx_ring	   *txr = &que->txr;
837c19c7afeSEric Joyner 		u64 tdba = txr->tx_paddr;
838758cc3dcSJack F Vogel 		u32 txctrl = 0;
83948056c88SJack F Vogel 		int j = txr->me;
840758cc3dcSJack F Vogel 
84148056c88SJack F Vogel 		IXGBE_WRITE_REG(hw, IXGBE_TDBAL(j),
842758cc3dcSJack F Vogel 		    (tdba & 0x00000000ffffffffULL));
84348056c88SJack F Vogel 		IXGBE_WRITE_REG(hw, IXGBE_TDBAH(j), (tdba >> 32));
84448056c88SJack F Vogel 		IXGBE_WRITE_REG(hw, IXGBE_TDLEN(j),
845c19c7afeSEric Joyner 		    scctx->isc_ntxd[0] * sizeof(union ixgbe_adv_tx_desc));
846758cc3dcSJack F Vogel 
847758cc3dcSJack F Vogel 		/* Setup the HW Tx Head and Tail descriptor pointers */
84848056c88SJack F Vogel 		IXGBE_WRITE_REG(hw, IXGBE_TDH(j), 0);
84948056c88SJack F Vogel 		IXGBE_WRITE_REG(hw, IXGBE_TDT(j), 0);
850758cc3dcSJack F Vogel 
851758cc3dcSJack F Vogel 		/* Cache the tail address */
8522dc2d580SEric Joyner 		txr->tail = IXGBE_TDT(txr->me);
8532dc2d580SEric Joyner 
854088a0b27SEric Joyner 		txr->tx_rs_cidx = txr->tx_rs_pidx;
855088a0b27SEric Joyner 		txr->tx_cidx_processed = scctx->isc_ntxd[0] - 1;
856c19c7afeSEric Joyner 		for (int k = 0; k < scctx->isc_ntxd[0]; k++)
857c19c7afeSEric Joyner 			txr->tx_rsq[k] = QIDX_INVALID;
858758cc3dcSJack F Vogel 
859758cc3dcSJack F Vogel 		/* Disable Head Writeback */
860a9ca1c79SSean Bruno 		/*
861a9ca1c79SSean Bruno 		 * Note: for X550 series devices, these registers are actually
862a9ca1c79SSean Bruno 		 * prefixed with TPH_ isntead of DCA_, but the addresses and
863a9ca1c79SSean Bruno 		 * fields remain the same.
864a9ca1c79SSean Bruno 		 */
865758cc3dcSJack F Vogel 		switch (hw->mac.type) {
866758cc3dcSJack F Vogel 		case ixgbe_mac_82598EB:
86748056c88SJack F Vogel 			txctrl = IXGBE_READ_REG(hw, IXGBE_DCA_TXCTRL(j));
868758cc3dcSJack F Vogel 			break;
869758cc3dcSJack F Vogel 		default:
870c58d34ddSKevin Bowling 			txctrl =
871c58d34ddSKevin Bowling 			    IXGBE_READ_REG(hw, IXGBE_DCA_TXCTRL_82599(j));
872758cc3dcSJack F Vogel 			break;
873758cc3dcSJack F Vogel 		}
874758cc3dcSJack F Vogel 		txctrl &= ~IXGBE_DCA_TXCTRL_DESC_WRO_EN;
875758cc3dcSJack F Vogel 		switch (hw->mac.type) {
876758cc3dcSJack F Vogel 		case ixgbe_mac_82598EB:
87748056c88SJack F Vogel 			IXGBE_WRITE_REG(hw, IXGBE_DCA_TXCTRL(j), txctrl);
878758cc3dcSJack F Vogel 			break;
879758cc3dcSJack F Vogel 		default:
880c58d34ddSKevin Bowling 			IXGBE_WRITE_REG(hw, IXGBE_DCA_TXCTRL_82599(j),
881c58d34ddSKevin Bowling 			    txctrl);
882758cc3dcSJack F Vogel 			break;
883758cc3dcSJack F Vogel 		}
884758cc3dcSJack F Vogel 
885758cc3dcSJack F Vogel 	}
886758cc3dcSJack F Vogel 
887758cc3dcSJack F Vogel 	if (hw->mac.type != ixgbe_mac_82598EB) {
888758cc3dcSJack F Vogel 		u32 dmatxctl, rttdcs;
8898eb6488eSEric Joyner 
890758cc3dcSJack F Vogel 		dmatxctl = IXGBE_READ_REG(hw, IXGBE_DMATXCTL);
891758cc3dcSJack F Vogel 		dmatxctl |= IXGBE_DMATXCTL_TE;
892758cc3dcSJack F Vogel 		IXGBE_WRITE_REG(hw, IXGBE_DMATXCTL, dmatxctl);
893758cc3dcSJack F Vogel 		/* Disable arbiter to set MTQC */
894758cc3dcSJack F Vogel 		rttdcs = IXGBE_READ_REG(hw, IXGBE_RTTDCS);
895758cc3dcSJack F Vogel 		rttdcs |= IXGBE_RTTDCS_ARBDIS;
896758cc3dcSJack F Vogel 		IXGBE_WRITE_REG(hw, IXGBE_RTTDCS, rttdcs);
8978eb6488eSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_MTQC,
898b1d5caf3SKevin Bowling 		    ixgbe_get_mtqc(sc->iov_mode));
899758cc3dcSJack F Vogel 		rttdcs &= ~IXGBE_RTTDCS_ARBDIS;
900758cc3dcSJack F Vogel 		IXGBE_WRITE_REG(hw, IXGBE_RTTDCS, rttdcs);
901758cc3dcSJack F Vogel 	}
902758cc3dcSJack F Vogel 
9038eb6488eSEric Joyner } /* ixgbe_initialize_transmit_units */
904758cc3dcSJack F Vogel 
9058eb6488eSEric Joyner /************************************************************************
906c19c7afeSEric Joyner  * ixgbe_register
907c19c7afeSEric Joyner  ************************************************************************/
908c19c7afeSEric Joyner static void *
ixgbe_register(device_t dev)909c19c7afeSEric Joyner ixgbe_register(device_t dev)
910c19c7afeSEric Joyner {
911ffe3def9SMark Johnston 	return (&ixgbe_sctx_init);
912c19c7afeSEric Joyner } /* ixgbe_register */
913c19c7afeSEric Joyner 
914c19c7afeSEric Joyner /************************************************************************
915c19c7afeSEric Joyner  * ixgbe_if_attach_pre - Device initialization routine, part 1
916758cc3dcSJack F Vogel  *
9178eb6488eSEric Joyner  *   Called when the driver is being loaded.
918c19c7afeSEric Joyner  *   Identifies the type of hardware, initializes the hardware,
919c19c7afeSEric Joyner  *   and initializes iflib structures.
920758cc3dcSJack F Vogel  *
9218eb6488eSEric Joyner  *   return 0 on success, positive on failure
9228eb6488eSEric Joyner  ************************************************************************/
9238eb6488eSEric Joyner static int
ixgbe_if_attach_pre(if_ctx_t ctx)924c19c7afeSEric Joyner ixgbe_if_attach_pre(if_ctx_t ctx)
925758cc3dcSJack F Vogel {
926b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc;
927c19c7afeSEric Joyner 	device_t dev;
928c19c7afeSEric Joyner 	if_softc_ctx_t scctx;
9298eb6488eSEric Joyner 	struct ixgbe_hw *hw;
9308eb6488eSEric Joyner 	int error = 0;
9318eb6488eSEric Joyner 	u32 ctrl_ext;
9327234c309SJakub Chylkowski 	size_t i;
9338eb6488eSEric Joyner 
9348eb6488eSEric Joyner 	INIT_DEBUGOUT("ixgbe_attach: begin");
9358eb6488eSEric Joyner 
9368eb6488eSEric Joyner 	/* Allocate, clear, and link in our adapter structure */
937c19c7afeSEric Joyner 	dev = iflib_get_dev(ctx);
938b1d5caf3SKevin Bowling 	sc = iflib_get_softc(ctx);
939b1d5caf3SKevin Bowling 	sc->hw.back = sc;
940b1d5caf3SKevin Bowling 	sc->ctx = ctx;
941b1d5caf3SKevin Bowling 	sc->dev = dev;
942b1d5caf3SKevin Bowling 	scctx = sc->shared = iflib_get_softc_ctx(ctx);
943b1d5caf3SKevin Bowling 	sc->media = iflib_get_media(ctx);
944b1d5caf3SKevin Bowling 	hw = &sc->hw;
9458eb6488eSEric Joyner 
9468eb6488eSEric Joyner 	/* Determine hardware revision */
9478eb6488eSEric Joyner 	hw->vendor_id = pci_get_vendor(dev);
9488eb6488eSEric Joyner 	hw->device_id = pci_get_device(dev);
9498eb6488eSEric Joyner 	hw->revision_id = pci_get_revid(dev);
9508eb6488eSEric Joyner 	hw->subsystem_vendor_id = pci_get_subvendor(dev);
9518eb6488eSEric Joyner 	hw->subsystem_device_id = pci_get_subdevice(dev);
952758cc3dcSJack F Vogel 
9538eb6488eSEric Joyner 	/* Do base PCI setup - map BAR0 */
954c19c7afeSEric Joyner 	if (ixgbe_allocate_pci_resources(ctx)) {
9558eb6488eSEric Joyner 		device_printf(dev, "Allocation of PCI resources failed\n");
956c19c7afeSEric Joyner 		return (ENXIO);
9576f37f232SEric Joyner 	}
958758cc3dcSJack F Vogel 
9598eb6488eSEric Joyner 	/* let hardware know driver is loaded */
9608eb6488eSEric Joyner 	ctrl_ext = IXGBE_READ_REG(hw, IXGBE_CTRL_EXT);
9618eb6488eSEric Joyner 	ctrl_ext |= IXGBE_CTRL_EXT_DRV_LOAD;
9628eb6488eSEric Joyner 	IXGBE_WRITE_REG(hw, IXGBE_CTRL_EXT, ctrl_ext);
963758cc3dcSJack F Vogel 
964758cc3dcSJack F Vogel 	/*
9658eb6488eSEric Joyner 	 * Initialize the shared code
966758cc3dcSJack F Vogel 	 */
967c19c7afeSEric Joyner 	if (ixgbe_init_shared_code(hw) != 0) {
9688eb6488eSEric Joyner 		device_printf(dev, "Unable to initialize the shared code\n");
9698eb6488eSEric Joyner 		error = ENXIO;
970c19c7afeSEric Joyner 		goto err_pci;
97130126537SJack F Vogel 	}
972758cc3dcSJack F Vogel 
973c58d34ddSKevin Bowling 	if (hw->mac.ops.fw_recovery_mode &&
974c58d34ddSKevin Bowling 	    hw->mac.ops.fw_recovery_mode(hw)) {
975c58d34ddSKevin Bowling 		device_printf(dev,
976c58d34ddSKevin Bowling 		    "Firmware recovery mode detected. Limiting "
9778b4a3fbdSKevin Bowling 		    "functionality.\nRefer to the Intel(R) Ethernet Adapters "
9788b4a3fbdSKevin Bowling 		    "and Devices User Guide for details on firmware recovery "
9798b4a3fbdSKevin Bowling 		    "mode.");
9808b4a3fbdSKevin Bowling 		error = ENOSYS;
9818b4a3fbdSKevin Bowling 		goto err_pci;
9828b4a3fbdSKevin Bowling 	}
9838b4a3fbdSKevin Bowling 
9847234c309SJakub Chylkowski 	/* 82598 Does not support SR-IOV, initialize everything else */
9857234c309SJakub Chylkowski 	if (hw->mac.type >= ixgbe_mac_82599_vf) {
9867234c309SJakub Chylkowski 		for (i = 0; i < sc->num_vfs; i++)
9877234c309SJakub Chylkowski 			hw->mbx.ops[i].init_params(hw);
9887234c309SJakub Chylkowski 	}
989758cc3dcSJack F Vogel 
9908eb6488eSEric Joyner 	hw->allow_unsupported_sfp = allow_unsupported_sfp;
991758cc3dcSJack F Vogel 
992c19c7afeSEric Joyner 	if (hw->mac.type != ixgbe_mac_82598EB)
9938eb6488eSEric Joyner 		hw->phy.smart_speed = ixgbe_smart_speed;
9948eb6488eSEric Joyner 
995b1d5caf3SKevin Bowling 	ixgbe_init_device_features(sc);
9968eb6488eSEric Joyner 
9978eb6488eSEric Joyner 	/* Enable WoL (if supported) */
998b1d5caf3SKevin Bowling 	ixgbe_check_wol_support(sc);
9998eb6488eSEric Joyner 
10008eb6488eSEric Joyner 	/* Verify adapter fan is still functional (if applicable) */
1001b1d5caf3SKevin Bowling 	if (sc->feat_en & IXGBE_FEATURE_FAN_FAIL) {
10028eb6488eSEric Joyner 		u32 esdp = IXGBE_READ_REG(hw, IXGBE_ESDP);
1003b1d5caf3SKevin Bowling 		ixgbe_check_fan_failure(sc, esdp, false);
10048eb6488eSEric Joyner 	}
10058eb6488eSEric Joyner 
10068eb6488eSEric Joyner 	/* Ensure SW/FW semaphore is free */
10078eb6488eSEric Joyner 	ixgbe_init_swfw_semaphore(hw);
10088eb6488eSEric Joyner 
10098eb6488eSEric Joyner 	/* Set an initial default flow control value */
10108eb6488eSEric Joyner 	hw->fc.requested_mode = ixgbe_flow_control;
10118eb6488eSEric Joyner 
101279b36ec9SKevin Bowling 	hw->phy.reset_if_overtemp = true;
10138eb6488eSEric Joyner 	error = ixgbe_reset_hw(hw);
101479b36ec9SKevin Bowling 	hw->phy.reset_if_overtemp = false;
10158eb6488eSEric Joyner 	if (error == IXGBE_ERR_SFP_NOT_PRESENT) {
1016758cc3dcSJack F Vogel 		/*
10178eb6488eSEric Joyner 		 * No optics in this port, set up
10188eb6488eSEric Joyner 		 * so the timer routine will probe
10198eb6488eSEric Joyner 		 * for later insertion.
1020758cc3dcSJack F Vogel 		 */
1021b1d5caf3SKevin Bowling 		sc->sfp_probe = true;
1022c19c7afeSEric Joyner 		error = 0;
10238eb6488eSEric Joyner 	} else if (error == IXGBE_ERR_SFP_NOT_SUPPORTED) {
10248eb6488eSEric Joyner 		device_printf(dev, "Unsupported SFP+ module detected!\n");
10258eb6488eSEric Joyner 		error = EIO;
1026c19c7afeSEric Joyner 		goto err_pci;
10278eb6488eSEric Joyner 	} else if (error) {
10288eb6488eSEric Joyner 		device_printf(dev, "Hardware initialization failed\n");
10298eb6488eSEric Joyner 		error = EIO;
1030c19c7afeSEric Joyner 		goto err_pci;
103197f9586eSSean Bruno 	}
103297f9586eSSean Bruno 
10338eb6488eSEric Joyner 	/* Make sure we have a good EEPROM before we read from it */
1034b1d5caf3SKevin Bowling 	if (ixgbe_validate_eeprom_checksum(&sc->hw, NULL) < 0) {
10358eb6488eSEric Joyner 		device_printf(dev, "The EEPROM Checksum Is Not Valid\n");
10368eb6488eSEric Joyner 		error = EIO;
1037c19c7afeSEric Joyner 		goto err_pci;
103897f9586eSSean Bruno 	}
103997f9586eSSean Bruno 
10408eb6488eSEric Joyner 	error = ixgbe_start_hw(hw);
10418eb6488eSEric Joyner 	switch (error) {
10428eb6488eSEric Joyner 	case IXGBE_ERR_EEPROM_VERSION:
1043c58d34ddSKevin Bowling 		device_printf(dev,
1044c58d34ddSKevin Bowling 		    "This device is a pre-production adapter/LOM.  Please be"
1045c58d34ddSKevin Bowling 		    " aware there may be issues associated with your"
1046c58d34ddSKevin Bowling 		    " hardware.\nIf you are experiencing problems please"
1047c58d34ddSKevin Bowling 		    " contact your Intel or hardware representative who"
1048c58d34ddSKevin Bowling 		    " provided you with this hardware.\n");
104997f9586eSSean Bruno 		break;
10508eb6488eSEric Joyner 	case IXGBE_ERR_SFP_NOT_SUPPORTED:
10518eb6488eSEric Joyner 		device_printf(dev, "Unsupported SFP+ Module\n");
10528eb6488eSEric Joyner 		error = EIO;
1053c19c7afeSEric Joyner 		goto err_pci;
10548eb6488eSEric Joyner 	case IXGBE_ERR_SFP_NOT_PRESENT:
10558eb6488eSEric Joyner 		device_printf(dev, "No SFP+ Module found\n");
10568eb6488eSEric Joyner 		/* falls thru */
105797f9586eSSean Bruno 	default:
105897f9586eSSean Bruno 		break;
105997f9586eSSean Bruno 	}
106097f9586eSSean Bruno 
1061c19c7afeSEric Joyner 	/* Most of the iflib initialization... */
1062c19c7afeSEric Joyner 
1063c19c7afeSEric Joyner 	iflib_set_mac(ctx, hw->mac.addr);
1064b1d5caf3SKevin Bowling 	switch (sc->hw.mac.type) {
1065c19c7afeSEric Joyner 	case ixgbe_mac_X550:
1066c19c7afeSEric Joyner 	case ixgbe_mac_X550EM_x:
1067c19c7afeSEric Joyner 	case ixgbe_mac_X550EM_a:
1068c19c7afeSEric Joyner 		scctx->isc_rss_table_size = 512;
1069c19c7afeSEric Joyner 		scctx->isc_ntxqsets_max = scctx->isc_nrxqsets_max = 64;
1070c19c7afeSEric Joyner 		break;
1071c19c7afeSEric Joyner 	default:
1072c19c7afeSEric Joyner 		scctx->isc_rss_table_size = 128;
1073c19c7afeSEric Joyner 		scctx->isc_ntxqsets_max = scctx->isc_nrxqsets_max = 16;
1074c19c7afeSEric Joyner 	}
1075c19c7afeSEric Joyner 
1076c19c7afeSEric Joyner 	/* Allow legacy interrupts */
1077c19c7afeSEric Joyner 	ixgbe_txrx.ift_legacy_intr = ixgbe_intr;
1078c19c7afeSEric Joyner 
1079c19c7afeSEric Joyner 	scctx->isc_txqsizes[0] =
1080c19c7afeSEric Joyner 	    roundup2(scctx->isc_ntxd[0] * sizeof(union ixgbe_adv_tx_desc) +
1081c19c7afeSEric Joyner 	    sizeof(u32), DBA_ALIGN),
1082c19c7afeSEric Joyner 	scctx->isc_rxqsizes[0] =
1083c19c7afeSEric Joyner 	    roundup2(scctx->isc_nrxd[0] * sizeof(union ixgbe_adv_rx_desc),
1084c19c7afeSEric Joyner 	    DBA_ALIGN);
1085c19c7afeSEric Joyner 
1086c19c7afeSEric Joyner 	/* XXX */
1087c19c7afeSEric Joyner 	scctx->isc_tx_csum_flags = CSUM_IP | CSUM_TCP | CSUM_UDP | CSUM_TSO |
1088c19c7afeSEric Joyner 	    CSUM_IP6_TCP | CSUM_IP6_UDP | CSUM_IP6_TSO;
1089b1d5caf3SKevin Bowling 	if (sc->hw.mac.type == ixgbe_mac_82598EB) {
1090c19c7afeSEric Joyner 		scctx->isc_tx_nsegments = IXGBE_82598_SCATTER;
1091c19c7afeSEric Joyner 	} else {
1092c19c7afeSEric Joyner 		scctx->isc_tx_csum_flags |= CSUM_SCTP |CSUM_IP6_SCTP;
1093c19c7afeSEric Joyner 		scctx->isc_tx_nsegments = IXGBE_82599_SCATTER;
1094c19c7afeSEric Joyner 	}
1095749597dcSEric Joyner 
1096749597dcSEric Joyner 	scctx->isc_msix_bar = pci_msix_table_bar(dev);
1097749597dcSEric Joyner 
1098c19c7afeSEric Joyner 	scctx->isc_tx_tso_segments_max = scctx->isc_tx_nsegments;
1099c19c7afeSEric Joyner 	scctx->isc_tx_tso_size_max = IXGBE_TSO_SIZE;
1100c19c7afeSEric Joyner 	scctx->isc_tx_tso_segsize_max = PAGE_SIZE;
1101c19c7afeSEric Joyner 
1102c19c7afeSEric Joyner 	scctx->isc_txrx = &ixgbe_txrx;
1103c19c7afeSEric Joyner 
11047f87c040SMarius Strobl 	scctx->isc_capabilities = scctx->isc_capenable = IXGBE_CAPS;
1105c19c7afeSEric Joyner 
1106c19c7afeSEric Joyner 	return (0);
1107c19c7afeSEric Joyner 
1108c19c7afeSEric Joyner err_pci:
1109b1d5caf3SKevin Bowling 	ctrl_ext = IXGBE_READ_REG(&sc->hw, IXGBE_CTRL_EXT);
1110c19c7afeSEric Joyner 	ctrl_ext &= ~IXGBE_CTRL_EXT_DRV_LOAD;
1111b1d5caf3SKevin Bowling 	IXGBE_WRITE_REG(&sc->hw, IXGBE_CTRL_EXT, ctrl_ext);
1112c19c7afeSEric Joyner 	ixgbe_free_pci_resources(ctx);
1113c19c7afeSEric Joyner 
1114c19c7afeSEric Joyner 	return (error);
1115c19c7afeSEric Joyner } /* ixgbe_if_attach_pre */
1116c19c7afeSEric Joyner 
1117c19c7afeSEric Joyner  /*********************************************************************
1118c19c7afeSEric Joyner  * ixgbe_if_attach_post - Device initialization routine, part 2
1119c19c7afeSEric Joyner  *
1120c19c7afeSEric Joyner  *   Called during driver load, but after interrupts and
1121c19c7afeSEric Joyner  *   resources have been allocated and configured.
1122c19c7afeSEric Joyner  *   Sets up some data structures not relevant to iflib.
1123c19c7afeSEric Joyner  *
1124c19c7afeSEric Joyner  *   return 0 on success, positive on failure
1125c19c7afeSEric Joyner  *********************************************************************/
1126c19c7afeSEric Joyner static int
ixgbe_if_attach_post(if_ctx_t ctx)1127c19c7afeSEric Joyner ixgbe_if_attach_post(if_ctx_t ctx)
1128c19c7afeSEric Joyner {
1129c19c7afeSEric Joyner 	device_t dev;
1130b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc;
1131c19c7afeSEric Joyner 	struct ixgbe_hw *hw;
1132c19c7afeSEric Joyner 	int error = 0;
1133c19c7afeSEric Joyner 
1134c19c7afeSEric Joyner 	dev = iflib_get_dev(ctx);
1135b1d5caf3SKevin Bowling 	sc = iflib_get_softc(ctx);
1136b1d5caf3SKevin Bowling 	hw = &sc->hw;
1137c19c7afeSEric Joyner 
1138b1d5caf3SKevin Bowling 	if (sc->intr_type == IFLIB_INTR_LEGACY &&
1139b1d5caf3SKevin Bowling 		(sc->feat_cap & IXGBE_FEATURE_LEGACY_IRQ) == 0) {
1140c19c7afeSEric Joyner 		device_printf(dev, "Device does not support legacy interrupts");
1141c19c7afeSEric Joyner 		error = ENXIO;
1142c19c7afeSEric Joyner 		goto err;
1143c19c7afeSEric Joyner 	}
1144c19c7afeSEric Joyner 
1145c19c7afeSEric Joyner 	/* Allocate multicast array memory. */
1146c58d34ddSKevin Bowling 	sc->mta = malloc(sizeof(*sc->mta) * MAX_NUM_MULTICAST_ADDRESSES,
1147c58d34ddSKevin Bowling 	    M_IXGBE, M_NOWAIT);
1148b1d5caf3SKevin Bowling 	if (sc->mta == NULL) {
1149c58d34ddSKevin Bowling 		device_printf(dev,
1150c58d34ddSKevin Bowling 		    "Can not allocate multicast setup array\n");
1151c19c7afeSEric Joyner 		error = ENOMEM;
1152c19c7afeSEric Joyner 		goto err;
1153c19c7afeSEric Joyner 	}
1154c19c7afeSEric Joyner 
1155c19c7afeSEric Joyner 	/* hw.ix defaults init */
1156b1d5caf3SKevin Bowling 	ixgbe_set_advertise(sc, ixgbe_advertise_speed);
1157c19c7afeSEric Joyner 
11588eb6488eSEric Joyner 	/* Enable the optics for 82599 SFP+ fiber */
11598eb6488eSEric Joyner 	ixgbe_enable_tx_laser(hw);
1160758cc3dcSJack F Vogel 
11618eb6488eSEric Joyner 	/* Enable power to the phy. */
116279b36ec9SKevin Bowling 	ixgbe_set_phy_power(hw, true);
11638eb6488eSEric Joyner 
1164b1d5caf3SKevin Bowling 	ixgbe_initialize_iov(sc);
1165c19c7afeSEric Joyner 
1166c19c7afeSEric Joyner 	error = ixgbe_setup_interface(ctx);
1167c19c7afeSEric Joyner 	if (error) {
1168c19c7afeSEric Joyner 		device_printf(dev, "Interface setup failed: %d\n", error);
1169c19c7afeSEric Joyner 		goto err;
1170c19c7afeSEric Joyner 	}
1171c19c7afeSEric Joyner 
1172c19c7afeSEric Joyner 	ixgbe_if_update_admin_status(ctx);
1173c19c7afeSEric Joyner 
11748eb6488eSEric Joyner 	/* Initialize statistics */
1175b1d5caf3SKevin Bowling 	ixgbe_update_stats_counters(sc);
1176b1d5caf3SKevin Bowling 	ixgbe_add_hw_stats(sc);
11778eb6488eSEric Joyner 
11788eb6488eSEric Joyner 	/* Check PCIE slot type/speed/width */
1179b1d5caf3SKevin Bowling 	ixgbe_get_slot_info(sc);
1180758cc3dcSJack F Vogel 
1181758cc3dcSJack F Vogel 	/*
11828eb6488eSEric Joyner 	 * Do time init and sysctl init here, but
1183b1d5caf3SKevin Bowling 	 * only on the first port of a bypass sc.
1184758cc3dcSJack F Vogel 	 */
1185b1d5caf3SKevin Bowling 	ixgbe_bypass_init(sc);
1186758cc3dcSJack F Vogel 
11877660e4eaSKevin Bowling 	/* Display NVM and Option ROM versions */
11887660e4eaSKevin Bowling 	ixgbe_print_fw_version(ctx);
11897660e4eaSKevin Bowling 
11908eb6488eSEric Joyner 	/* Set an initial dmac value */
1191b1d5caf3SKevin Bowling 	sc->dmac = 0;
11928eb6488eSEric Joyner 	/* Set initial advertised speeds (if applicable) */
1193d381c807SPiotr Pietruszewski 	sc->advertise = ixgbe_get_default_advertise(sc);
1194758cc3dcSJack F Vogel 
1195b1d5caf3SKevin Bowling 	if (sc->feat_cap & IXGBE_FEATURE_SRIOV)
11968eb6488eSEric Joyner 		ixgbe_define_iov_schemas(dev, &error);
1197758cc3dcSJack F Vogel 
11988eb6488eSEric Joyner 	/* Add sysctls */
1199c19c7afeSEric Joyner 	ixgbe_add_device_sysctls(ctx);
1200758cc3dcSJack F Vogel 
1201f72de14eSKevin Bowling 	/* Init recovery mode timer and state variable */
1202f72de14eSKevin Bowling 	if (sc->feat_en & IXGBE_FEATURE_RECOVERY_MODE) {
1203f72de14eSKevin Bowling 		sc->recovery_mode = 0;
1204f72de14eSKevin Bowling 
1205f72de14eSKevin Bowling 		/* Set up the timer callout */
1206f72de14eSKevin Bowling 		callout_init(&sc->fw_mode_timer, true);
1207f72de14eSKevin Bowling 
1208f72de14eSKevin Bowling 		/* Start the task */
1209a924b5eeSKevin Bowling 		callout_reset(&sc->fw_mode_timer, hz, ixgbe_fw_mode_timer, sc);
1210f72de14eSKevin Bowling 	}
1211f72de14eSKevin Bowling 
12128eb6488eSEric Joyner 	return (0);
1213c19c7afeSEric Joyner err:
12148eb6488eSEric Joyner 	return (error);
1215c19c7afeSEric Joyner } /* ixgbe_if_attach_post */
12168eb6488eSEric Joyner 
12178eb6488eSEric Joyner /************************************************************************
12188eb6488eSEric Joyner  * ixgbe_check_wol_support
12196f37f232SEric Joyner  *
12206f37f232SEric Joyner  *   Checks whether the adapter's ports are capable of
12216f37f232SEric Joyner  *   Wake On LAN by reading the adapter's NVM.
12226f37f232SEric Joyner  *
12236f37f232SEric Joyner  *   Sets each port's hw->wol_enabled value depending
12246f37f232SEric Joyner  *   on the value read here.
12258eb6488eSEric Joyner  ************************************************************************/
12266f37f232SEric Joyner static void
ixgbe_check_wol_support(struct ixgbe_softc * sc)1227b1d5caf3SKevin Bowling ixgbe_check_wol_support(struct ixgbe_softc *sc)
12286f37f232SEric Joyner {
1229b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
12306f37f232SEric Joyner 	u16 dev_caps = 0;
12316f37f232SEric Joyner 
12326f37f232SEric Joyner 	/* Find out WoL support for port */
1233b1d5caf3SKevin Bowling 	sc->wol_support = hw->wol_enabled = 0;
12346f37f232SEric Joyner 	ixgbe_get_device_caps(hw, &dev_caps);
12356f37f232SEric Joyner 	if ((dev_caps & IXGBE_DEVICE_CAPS_WOL_PORT0_1) ||
12366f37f232SEric Joyner 	    ((dev_caps & IXGBE_DEVICE_CAPS_WOL_PORT0) &&
12376f37f232SEric Joyner 	     hw->bus.func == 0))
1238b1d5caf3SKevin Bowling 		sc->wol_support = hw->wol_enabled = 1;
12396f37f232SEric Joyner 
12406f37f232SEric Joyner 	/* Save initial wake up filter configuration */
1241b1d5caf3SKevin Bowling 	sc->wufc = IXGBE_READ_REG(hw, IXGBE_WUFC);
12426f37f232SEric Joyner 
12436f37f232SEric Joyner 	return;
12448eb6488eSEric Joyner } /* ixgbe_check_wol_support */
12456f37f232SEric Joyner 
12468eb6488eSEric Joyner /************************************************************************
12478eb6488eSEric Joyner  * ixgbe_setup_interface
12488eb6488eSEric Joyner  *
12498eb6488eSEric Joyner  *   Setup networking device structure and register an interface.
12508eb6488eSEric Joyner  ************************************************************************/
12516f37f232SEric Joyner static int
ixgbe_setup_interface(if_ctx_t ctx)1252c19c7afeSEric Joyner ixgbe_setup_interface(if_ctx_t ctx)
12536f37f232SEric Joyner {
1254ff06a8dbSJustin Hibbits 	if_t ifp = iflib_get_ifp(ctx);
1255b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
12566f37f232SEric Joyner 
12578eb6488eSEric Joyner 	INIT_DEBUGOUT("ixgbe_setup_interface: begin");
12586f37f232SEric Joyner 
1259c19c7afeSEric Joyner 	if_setbaudrate(ifp, IF_Gbps(10));
12606f37f232SEric Joyner 
1261ff06a8dbSJustin Hibbits 	sc->max_frame_size = if_getmtu(ifp) + ETHER_HDR_LEN + ETHER_CRC_LEN;
1262758cc3dcSJack F Vogel 
1263b1d5caf3SKevin Bowling 	sc->phy_layer = ixgbe_get_supported_physical_layer(&sc->hw);
12648eb6488eSEric Joyner 
1265c19c7afeSEric Joyner 	ixgbe_add_media_types(ctx);
1266c19c7afeSEric Joyner 
1267c19c7afeSEric Joyner 	/* Autoselect media by default */
1268b1d5caf3SKevin Bowling 	ifmedia_set(sc->media, IFM_ETHER | IFM_AUTO);
12698eb6488eSEric Joyner 
12708eb6488eSEric Joyner 	return (0);
12718eb6488eSEric Joyner } /* ixgbe_setup_interface */
1272758cc3dcSJack F Vogel 
12738eb6488eSEric Joyner /************************************************************************
1274c19c7afeSEric Joyner  * ixgbe_if_get_counter
12758eb6488eSEric Joyner  ************************************************************************/
1276758cc3dcSJack F Vogel static uint64_t
ixgbe_if_get_counter(if_ctx_t ctx,ift_counter cnt)1277c19c7afeSEric Joyner ixgbe_if_get_counter(if_ctx_t ctx, ift_counter cnt)
1278758cc3dcSJack F Vogel {
1279b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
1280c19c7afeSEric Joyner 	if_t ifp = iflib_get_ifp(ctx);
1281758cc3dcSJack F Vogel 
1282758cc3dcSJack F Vogel 	switch (cnt) {
1283758cc3dcSJack F Vogel 	case IFCOUNTER_IPACKETS:
1284b1d5caf3SKevin Bowling 		return (sc->ipackets);
1285758cc3dcSJack F Vogel 	case IFCOUNTER_OPACKETS:
1286b1d5caf3SKevin Bowling 		return (sc->opackets);
1287758cc3dcSJack F Vogel 	case IFCOUNTER_IBYTES:
1288b1d5caf3SKevin Bowling 		return (sc->ibytes);
1289758cc3dcSJack F Vogel 	case IFCOUNTER_OBYTES:
1290b1d5caf3SKevin Bowling 		return (sc->obytes);
1291758cc3dcSJack F Vogel 	case IFCOUNTER_IMCASTS:
1292b1d5caf3SKevin Bowling 		return (sc->imcasts);
1293758cc3dcSJack F Vogel 	case IFCOUNTER_OMCASTS:
1294b1d5caf3SKevin Bowling 		return (sc->omcasts);
1295758cc3dcSJack F Vogel 	case IFCOUNTER_COLLISIONS:
1296758cc3dcSJack F Vogel 		return (0);
1297758cc3dcSJack F Vogel 	case IFCOUNTER_IQDROPS:
1298b1d5caf3SKevin Bowling 		return (sc->iqdrops);
1299625d12c6SJohn Baldwin 	case IFCOUNTER_OQDROPS:
1300c19c7afeSEric Joyner 		return (0);
1301758cc3dcSJack F Vogel 	case IFCOUNTER_IERRORS:
1302b1d5caf3SKevin Bowling 		return (sc->ierrors);
1303758cc3dcSJack F Vogel 	default:
1304758cc3dcSJack F Vogel 		return (if_get_counter_default(ifp, cnt));
1305758cc3dcSJack F Vogel 	}
1306c19c7afeSEric Joyner } /* ixgbe_if_get_counter */
1307758cc3dcSJack F Vogel 
13088eb6488eSEric Joyner /************************************************************************
1309bca38080SAndrew Gallatin  * ixgbe_if_i2c_req
1310bca38080SAndrew Gallatin  ************************************************************************/
1311bca38080SAndrew Gallatin static int
ixgbe_if_i2c_req(if_ctx_t ctx,struct ifi2creq * req)1312bca38080SAndrew Gallatin ixgbe_if_i2c_req(if_ctx_t ctx, struct ifi2creq *req)
1313bca38080SAndrew Gallatin {
1314b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
1315b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
1316bca38080SAndrew Gallatin 	int i;
1317bca38080SAndrew Gallatin 
1318bca38080SAndrew Gallatin 	if (hw->phy.ops.read_i2c_byte == NULL)
1319bca38080SAndrew Gallatin 		return (ENXIO);
1320bca38080SAndrew Gallatin 	for (i = 0; i < req->len; i++)
1321bca38080SAndrew Gallatin 		hw->phy.ops.read_i2c_byte(hw, req->offset + i,
1322bca38080SAndrew Gallatin 		    req->dev_addr, &req->data[i]);
1323bca38080SAndrew Gallatin 	return (0);
1324bca38080SAndrew Gallatin } /* ixgbe_if_i2c_req */
1325bca38080SAndrew Gallatin 
1326c58d34ddSKevin Bowling /* ixgbe_if_needs_restart - Tell iflib when the driver needs to be
1327c58d34ddSKevin Bowling  * reinitialized
1328cf150917SEric Joyner  * @ctx: iflib context
1329cf150917SEric Joyner  * @event: event code to check
1330cf150917SEric Joyner  *
1331725e4008SKevin Bowling  * Defaults to returning false for unknown events.
1332cf150917SEric Joyner  *
1333cf150917SEric Joyner  * @returns true if iflib needs to reinit the interface
1334cf150917SEric Joyner  */
1335cf150917SEric Joyner static bool
ixgbe_if_needs_restart(if_ctx_t ctx __unused,enum iflib_restart_event event)1336cf150917SEric Joyner ixgbe_if_needs_restart(if_ctx_t ctx __unused, enum iflib_restart_event event)
1337cf150917SEric Joyner {
1338cf150917SEric Joyner 	switch (event) {
1339cf150917SEric Joyner 	case IFLIB_RESTART_VLAN_CONFIG:
1340cf150917SEric Joyner 	default:
1341725e4008SKevin Bowling 		return (false);
1342cf150917SEric Joyner 	}
1343cf150917SEric Joyner }
1344cf150917SEric Joyner 
1345bca38080SAndrew Gallatin /************************************************************************
13468eb6488eSEric Joyner  * ixgbe_add_media_types
13478eb6488eSEric Joyner  ************************************************************************/
13488eb6488eSEric Joyner static void
ixgbe_add_media_types(if_ctx_t ctx)1349c19c7afeSEric Joyner ixgbe_add_media_types(if_ctx_t ctx)
13508eb6488eSEric Joyner {
1351b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
1352b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
1353c19c7afeSEric Joyner 	device_t dev = iflib_get_dev(ctx);
13548eb6488eSEric Joyner 	u64 layer;
13558eb6488eSEric Joyner 
1356b1d5caf3SKevin Bowling 	layer = sc->phy_layer = ixgbe_get_supported_physical_layer(hw);
13578eb6488eSEric Joyner 
13588eb6488eSEric Joyner 	/* Media types with matching FreeBSD media defines */
13598eb6488eSEric Joyner 	if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_T)
1360b1d5caf3SKevin Bowling 		ifmedia_add(sc->media, IFM_ETHER | IFM_10G_T, 0, NULL);
13618eb6488eSEric Joyner 	if (layer & IXGBE_PHYSICAL_LAYER_1000BASE_T)
1362b1d5caf3SKevin Bowling 		ifmedia_add(sc->media, IFM_ETHER | IFM_1000_T, 0, NULL);
13638eb6488eSEric Joyner 	if (layer & IXGBE_PHYSICAL_LAYER_100BASE_TX)
1364b1d5caf3SKevin Bowling 		ifmedia_add(sc->media, IFM_ETHER | IFM_100_TX, 0, NULL);
13658eb6488eSEric Joyner 	if (layer & IXGBE_PHYSICAL_LAYER_10BASE_T)
1366b1d5caf3SKevin Bowling 		ifmedia_add(sc->media, IFM_ETHER | IFM_10_T, 0, NULL);
13678eb6488eSEric Joyner 
1368d381c807SPiotr Pietruszewski 	if (hw->mac.type == ixgbe_mac_X550) {
1369d381c807SPiotr Pietruszewski 		ifmedia_add(sc->media, IFM_ETHER | IFM_2500_T, 0, NULL);
1370d381c807SPiotr Pietruszewski 		ifmedia_add(sc->media, IFM_ETHER | IFM_5000_T, 0, NULL);
1371d381c807SPiotr Pietruszewski 	}
1372d381c807SPiotr Pietruszewski 
13738eb6488eSEric Joyner 	if (layer & IXGBE_PHYSICAL_LAYER_SFP_PLUS_CU ||
137448ddd1b9SKevin Bowling 	    layer & IXGBE_PHYSICAL_LAYER_SFP_ACTIVE_DA) {
1375b1d5caf3SKevin Bowling 		ifmedia_add(sc->media, IFM_ETHER | IFM_10G_TWINAX, 0,
13768eb6488eSEric Joyner 		    NULL);
137748ddd1b9SKevin Bowling 		ifmedia_add(sc->media, IFM_ETHER | IFM_1000_KX, 0, NULL);
137848ddd1b9SKevin Bowling 	}
13798eb6488eSEric Joyner 
13808eb6488eSEric Joyner 	if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_LR) {
1381b1d5caf3SKevin Bowling 		ifmedia_add(sc->media, IFM_ETHER | IFM_10G_LR, 0, NULL);
13828eb6488eSEric Joyner 		if (hw->phy.multispeed_fiber)
1383b1d5caf3SKevin Bowling 			ifmedia_add(sc->media, IFM_ETHER | IFM_1000_LX, 0,
13848eb6488eSEric Joyner 			    NULL);
13858eb6488eSEric Joyner 	}
13868eb6488eSEric Joyner 	if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_SR) {
1387b1d5caf3SKevin Bowling 		ifmedia_add(sc->media, IFM_ETHER | IFM_10G_SR, 0, NULL);
13888eb6488eSEric Joyner 		if (hw->phy.multispeed_fiber)
1389b1d5caf3SKevin Bowling 			ifmedia_add(sc->media, IFM_ETHER | IFM_1000_SX, 0,
13908eb6488eSEric Joyner 			    NULL);
13918eb6488eSEric Joyner 	} else if (layer & IXGBE_PHYSICAL_LAYER_1000BASE_SX)
1392b1d5caf3SKevin Bowling 		ifmedia_add(sc->media, IFM_ETHER | IFM_1000_SX, 0, NULL);
13938eb6488eSEric Joyner 	if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_CX4)
1394b1d5caf3SKevin Bowling 		ifmedia_add(sc->media, IFM_ETHER | IFM_10G_CX4, 0, NULL);
13958eb6488eSEric Joyner 
13968eb6488eSEric Joyner #ifdef IFM_ETH_XTYPE
13978eb6488eSEric Joyner 	if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_KR)
1398b1d5caf3SKevin Bowling 		ifmedia_add(sc->media, IFM_ETHER | IFM_10G_KR, 0, NULL);
13998eb6488eSEric Joyner 	if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_KX4)
1400b1d5caf3SKevin Bowling 		ifmedia_add( sc->media, IFM_ETHER | IFM_10G_KX4, 0, NULL);
14018eb6488eSEric Joyner 	if (layer & IXGBE_PHYSICAL_LAYER_1000BASE_KX)
1402b1d5caf3SKevin Bowling 		ifmedia_add(sc->media, IFM_ETHER | IFM_1000_KX, 0, NULL);
14038eb6488eSEric Joyner 	if (layer & IXGBE_PHYSICAL_LAYER_2500BASE_KX)
1404b1d5caf3SKevin Bowling 		ifmedia_add(sc->media, IFM_ETHER | IFM_2500_KX, 0, NULL);
14058eb6488eSEric Joyner #else
14068eb6488eSEric Joyner 	if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_KR) {
14078eb6488eSEric Joyner 		device_printf(dev, "Media supported: 10GbaseKR\n");
14088eb6488eSEric Joyner 		device_printf(dev, "10GbaseKR mapped to 10GbaseSR\n");
1409b1d5caf3SKevin Bowling 		ifmedia_add(sc->media, IFM_ETHER | IFM_10G_SR, 0, NULL);
14108eb6488eSEric Joyner 	}
14118eb6488eSEric Joyner 	if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_KX4) {
14128eb6488eSEric Joyner 		device_printf(dev, "Media supported: 10GbaseKX4\n");
14138eb6488eSEric Joyner 		device_printf(dev, "10GbaseKX4 mapped to 10GbaseCX4\n");
1414b1d5caf3SKevin Bowling 		ifmedia_add(sc->media, IFM_ETHER | IFM_10G_CX4, 0, NULL);
14158eb6488eSEric Joyner 	}
14168eb6488eSEric Joyner 	if (layer & IXGBE_PHYSICAL_LAYER_1000BASE_KX) {
14178eb6488eSEric Joyner 		device_printf(dev, "Media supported: 1000baseKX\n");
14188eb6488eSEric Joyner 		device_printf(dev, "1000baseKX mapped to 1000baseCX\n");
1419b1d5caf3SKevin Bowling 		ifmedia_add(sc->media, IFM_ETHER | IFM_1000_CX, 0, NULL);
14208eb6488eSEric Joyner 	}
14218eb6488eSEric Joyner 	if (layer & IXGBE_PHYSICAL_LAYER_2500BASE_KX) {
14228eb6488eSEric Joyner 		device_printf(dev, "Media supported: 2500baseKX\n");
14238eb6488eSEric Joyner 		device_printf(dev, "2500baseKX mapped to 2500baseSX\n");
1424b1d5caf3SKevin Bowling 		ifmedia_add(sc->media, IFM_ETHER | IFM_2500_SX, 0, NULL);
14258eb6488eSEric Joyner 	}
14268eb6488eSEric Joyner #endif
1427*89d40969STore Amundsen 	if (layer & IXGBE_PHYSICAL_LAYER_1000BASE_BX) {
14288eb6488eSEric Joyner 		device_printf(dev, "Media supported: 1000baseBX\n");
1429*89d40969STore Amundsen 		ifmedia_add(sc->media, IFM_ETHER | IFM_1000_BX, 0, NULL);
1430*89d40969STore Amundsen 	}
14318eb6488eSEric Joyner 
14328eb6488eSEric Joyner 	if (hw->device_id == IXGBE_DEV_ID_82598AT) {
1433b1d5caf3SKevin Bowling 		ifmedia_add(sc->media, IFM_ETHER | IFM_1000_T | IFM_FDX,
14348eb6488eSEric Joyner 		    0, NULL);
1435b1d5caf3SKevin Bowling 		ifmedia_add(sc->media, IFM_ETHER | IFM_1000_T, 0, NULL);
14368eb6488eSEric Joyner 	}
14378eb6488eSEric Joyner 
1438b1d5caf3SKevin Bowling 	ifmedia_add(sc->media, IFM_ETHER | IFM_AUTO, 0, NULL);
14398eb6488eSEric Joyner } /* ixgbe_add_media_types */
14408eb6488eSEric Joyner 
14418eb6488eSEric Joyner /************************************************************************
14428eb6488eSEric Joyner  * ixgbe_is_sfp
14438eb6488eSEric Joyner  ************************************************************************/
14448eb6488eSEric Joyner static inline bool
ixgbe_is_sfp(struct ixgbe_hw * hw)14458eb6488eSEric Joyner ixgbe_is_sfp(struct ixgbe_hw *hw)
14468eb6488eSEric Joyner {
14478eb6488eSEric Joyner 	switch (hw->mac.type) {
14488eb6488eSEric Joyner 	case ixgbe_mac_82598EB:
14498eb6488eSEric Joyner 		if (hw->phy.type == ixgbe_phy_nl)
145079b36ec9SKevin Bowling 			return (true);
145179b36ec9SKevin Bowling 		return (false);
14528eb6488eSEric Joyner 	case ixgbe_mac_82599EB:
14538eb6488eSEric Joyner 		switch (hw->mac.ops.get_media_type(hw)) {
14548eb6488eSEric Joyner 		case ixgbe_media_type_fiber:
14558eb6488eSEric Joyner 		case ixgbe_media_type_fiber_qsfp:
145679b36ec9SKevin Bowling 			return (true);
14578eb6488eSEric Joyner 		default:
145879b36ec9SKevin Bowling 			return (false);
14598eb6488eSEric Joyner 		}
14608eb6488eSEric Joyner 	case ixgbe_mac_X550EM_x:
14618eb6488eSEric Joyner 	case ixgbe_mac_X550EM_a:
14628eb6488eSEric Joyner 		if (hw->mac.ops.get_media_type(hw) == ixgbe_media_type_fiber)
146379b36ec9SKevin Bowling 			return (true);
146479b36ec9SKevin Bowling 		return (false);
14658eb6488eSEric Joyner 	default:
146679b36ec9SKevin Bowling 		return (false);
14678eb6488eSEric Joyner 	}
14688eb6488eSEric Joyner } /* ixgbe_is_sfp */
14698eb6488eSEric Joyner 
14708eb6488eSEric Joyner /************************************************************************
14718eb6488eSEric Joyner  * ixgbe_config_link
14728eb6488eSEric Joyner  ************************************************************************/
14738eb6488eSEric Joyner static void
ixgbe_config_link(if_ctx_t ctx)1474b2c1e8e6SEric Joyner ixgbe_config_link(if_ctx_t ctx)
14758eb6488eSEric Joyner {
1476b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
1477b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
14788eb6488eSEric Joyner 	u32 autoneg, err = 0;
14798eb6488eSEric Joyner 	bool sfp, negotiate;
14808eb6488eSEric Joyner 
14818eb6488eSEric Joyner 	sfp = ixgbe_is_sfp(hw);
14828eb6488eSEric Joyner 
14838eb6488eSEric Joyner 	if (sfp) {
1484b1d5caf3SKevin Bowling 		sc->task_requests |= IXGBE_REQUEST_TASK_MOD;
1485b2c1e8e6SEric Joyner 		iflib_admin_intr_deferred(ctx);
14868eb6488eSEric Joyner 	} else {
14878eb6488eSEric Joyner 		if (hw->mac.ops.check_link)
1488b1d5caf3SKevin Bowling 			err = ixgbe_check_link(hw, &sc->link_speed,
1489b1d5caf3SKevin Bowling 			    &sc->link_up, false);
14908eb6488eSEric Joyner 		if (err)
1491c19c7afeSEric Joyner 			return;
14928eb6488eSEric Joyner 		autoneg = hw->phy.autoneg_advertised;
14938eb6488eSEric Joyner 		if ((!autoneg) && (hw->mac.ops.get_link_capabilities))
14948eb6488eSEric Joyner 			err = hw->mac.ops.get_link_capabilities(hw, &autoneg,
14958eb6488eSEric Joyner 			    &negotiate);
14968eb6488eSEric Joyner 		if (err)
1497c19c7afeSEric Joyner 			return;
1498d381c807SPiotr Pietruszewski 
1499d381c807SPiotr Pietruszewski 		if (hw->mac.type == ixgbe_mac_X550 &&
1500d381c807SPiotr Pietruszewski 		    hw->phy.autoneg_advertised == 0) {
1501d381c807SPiotr Pietruszewski 			/*
1502d381c807SPiotr Pietruszewski 			 * 2.5G and 5G autonegotiation speeds on X550
1503d381c807SPiotr Pietruszewski 			 * are disabled by default due to reported
1504d381c807SPiotr Pietruszewski 			 * interoperability issues with some switches.
1505d381c807SPiotr Pietruszewski 			 *
1506d381c807SPiotr Pietruszewski 			 * The second condition checks if any operations
1507d381c807SPiotr Pietruszewski 			 * involving setting autonegotiation speeds have
1508d381c807SPiotr Pietruszewski 			 * been performed prior to this ixgbe_config_link()
1509d381c807SPiotr Pietruszewski 			 * call.
1510d381c807SPiotr Pietruszewski 			 *
1511d381c807SPiotr Pietruszewski 			 * If hw->phy.autoneg_advertised does not
1512d381c807SPiotr Pietruszewski 			 * equal 0, this means that the user might have
1513d381c807SPiotr Pietruszewski 			 * set autonegotiation speeds via the sysctl
1514d381c807SPiotr Pietruszewski 			 * before bringing the interface up. In this
1515d381c807SPiotr Pietruszewski 			 * case, we should not disable 2.5G and 5G
1516d381c807SPiotr Pietruszewski 			 * since that speeds might be selected by the
1517d381c807SPiotr Pietruszewski 			 * user.
1518d381c807SPiotr Pietruszewski 			 *
1519d381c807SPiotr Pietruszewski 			 * Otherwise (i.e. if hw->phy.autoneg_advertised
1520d381c807SPiotr Pietruszewski 			 * is set to 0), it is the first time we set
1521d381c807SPiotr Pietruszewski 			 * autonegotiation preferences and the default
1522d381c807SPiotr Pietruszewski 			 * set of speeds should exclude 2.5G and 5G.
1523d381c807SPiotr Pietruszewski 			 */
1524d381c807SPiotr Pietruszewski 			autoneg &= ~(IXGBE_LINK_SPEED_2_5GB_FULL |
1525d381c807SPiotr Pietruszewski 			    IXGBE_LINK_SPEED_5GB_FULL);
1526d381c807SPiotr Pietruszewski 		}
1527d381c807SPiotr Pietruszewski 
15288eb6488eSEric Joyner 		if (hw->mac.ops.setup_link)
15298eb6488eSEric Joyner 			err = hw->mac.ops.setup_link(hw, autoneg,
1530b1d5caf3SKevin Bowling 			    sc->link_up);
15318eb6488eSEric Joyner 	}
15328eb6488eSEric Joyner } /* ixgbe_config_link */
15338eb6488eSEric Joyner 
15348eb6488eSEric Joyner /************************************************************************
15358eb6488eSEric Joyner  * ixgbe_update_stats_counters - Update board statistics counters.
15368eb6488eSEric Joyner  ************************************************************************/
15378eb6488eSEric Joyner static void
ixgbe_update_stats_counters(struct ixgbe_softc * sc)1538b1d5caf3SKevin Bowling ixgbe_update_stats_counters(struct ixgbe_softc *sc)
15398eb6488eSEric Joyner {
1540b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
1541b1d5caf3SKevin Bowling 	struct ixgbe_hw_stats *stats = &sc->stats.pf;
15428eb6488eSEric Joyner 	u32 missed_rx = 0, bprc, lxon, lxoff, total;
1543e37d3dc1SEric Joyner 	u32 lxoffrxc;
15448eb6488eSEric Joyner 	u64 total_missed_rx = 0;
15458eb6488eSEric Joyner 
15468eb6488eSEric Joyner 	stats->crcerrs += IXGBE_READ_REG(hw, IXGBE_CRCERRS);
15478eb6488eSEric Joyner 	stats->illerrc += IXGBE_READ_REG(hw, IXGBE_ILLERRC);
15488eb6488eSEric Joyner 	stats->errbc += IXGBE_READ_REG(hw, IXGBE_ERRBC);
15498eb6488eSEric Joyner 	stats->mspdc += IXGBE_READ_REG(hw, IXGBE_MSPDC);
15508eb6488eSEric Joyner 	stats->mpc[0] += IXGBE_READ_REG(hw, IXGBE_MPC(0));
15518eb6488eSEric Joyner 
15528eb6488eSEric Joyner 	for (int i = 0; i < 16; i++) {
15538eb6488eSEric Joyner 		stats->qprc[i] += IXGBE_READ_REG(hw, IXGBE_QPRC(i));
15548eb6488eSEric Joyner 		stats->qptc[i] += IXGBE_READ_REG(hw, IXGBE_QPTC(i));
15558eb6488eSEric Joyner 		stats->qprdc[i] += IXGBE_READ_REG(hw, IXGBE_QPRDC(i));
15568eb6488eSEric Joyner 	}
15578eb6488eSEric Joyner 	stats->mlfc += IXGBE_READ_REG(hw, IXGBE_MLFC);
15588eb6488eSEric Joyner 	stats->mrfc += IXGBE_READ_REG(hw, IXGBE_MRFC);
15598eb6488eSEric Joyner 	stats->rlec += IXGBE_READ_REG(hw, IXGBE_RLEC);
15608eb6488eSEric Joyner 
15618eb6488eSEric Joyner 	/* Hardware workaround, gprc counts missed packets */
15628eb6488eSEric Joyner 	stats->gprc += IXGBE_READ_REG(hw, IXGBE_GPRC);
15638eb6488eSEric Joyner 	stats->gprc -= missed_rx;
15648eb6488eSEric Joyner 
15658eb6488eSEric Joyner 	if (hw->mac.type != ixgbe_mac_82598EB) {
15668eb6488eSEric Joyner 		stats->gorc += IXGBE_READ_REG(hw, IXGBE_GORCL) +
15678eb6488eSEric Joyner 		    ((u64)IXGBE_READ_REG(hw, IXGBE_GORCH) << 32);
15688eb6488eSEric Joyner 		stats->gotc += IXGBE_READ_REG(hw, IXGBE_GOTCL) +
15698eb6488eSEric Joyner 		    ((u64)IXGBE_READ_REG(hw, IXGBE_GOTCH) << 32);
15708eb6488eSEric Joyner 		stats->tor += IXGBE_READ_REG(hw, IXGBE_TORL) +
15718eb6488eSEric Joyner 		    ((u64)IXGBE_READ_REG(hw, IXGBE_TORH) << 32);
15728eb6488eSEric Joyner 		stats->lxonrxc += IXGBE_READ_REG(hw, IXGBE_LXONRXCNT);
1573e37d3dc1SEric Joyner 		lxoffrxc = IXGBE_READ_REG(hw, IXGBE_LXOFFRXCNT);
1574e37d3dc1SEric Joyner 		stats->lxoffrxc += lxoffrxc;
15758eb6488eSEric Joyner 	} else {
15768eb6488eSEric Joyner 		stats->lxonrxc += IXGBE_READ_REG(hw, IXGBE_LXONRXC);
1577e37d3dc1SEric Joyner 		lxoffrxc = IXGBE_READ_REG(hw, IXGBE_LXOFFRXC);
1578e37d3dc1SEric Joyner 		stats->lxoffrxc += lxoffrxc;
15798eb6488eSEric Joyner 		/* 82598 only has a counter in the high register */
15808eb6488eSEric Joyner 		stats->gorc += IXGBE_READ_REG(hw, IXGBE_GORCH);
15818eb6488eSEric Joyner 		stats->gotc += IXGBE_READ_REG(hw, IXGBE_GOTCH);
15828eb6488eSEric Joyner 		stats->tor += IXGBE_READ_REG(hw, IXGBE_TORH);
15838eb6488eSEric Joyner 	}
15848eb6488eSEric Joyner 
15858eb6488eSEric Joyner 	/*
1586e37d3dc1SEric Joyner 	 * For watchdog management we need to know if we have been paused
1587e37d3dc1SEric Joyner 	 * during the last interval, so capture that here.
1588e37d3dc1SEric Joyner 	*/
1589e37d3dc1SEric Joyner 	if (lxoffrxc)
1590b1d5caf3SKevin Bowling 		sc->shared->isc_pause_frames = 1;
1591e37d3dc1SEric Joyner 
1592e37d3dc1SEric Joyner 	/*
15938eb6488eSEric Joyner 	 * Workaround: mprc hardware is incorrectly counting
15948eb6488eSEric Joyner 	 * broadcasts, so for now we subtract those.
1595758cc3dcSJack F Vogel 	 */
15968eb6488eSEric Joyner 	bprc = IXGBE_READ_REG(hw, IXGBE_BPRC);
15978eb6488eSEric Joyner 	stats->bprc += bprc;
15988eb6488eSEric Joyner 	stats->mprc += IXGBE_READ_REG(hw, IXGBE_MPRC);
15998eb6488eSEric Joyner 	if (hw->mac.type == ixgbe_mac_82598EB)
16008eb6488eSEric Joyner 		stats->mprc -= bprc;
16018eb6488eSEric Joyner 
16028eb6488eSEric Joyner 	stats->prc64 += IXGBE_READ_REG(hw, IXGBE_PRC64);
16038eb6488eSEric Joyner 	stats->prc127 += IXGBE_READ_REG(hw, IXGBE_PRC127);
16048eb6488eSEric Joyner 	stats->prc255 += IXGBE_READ_REG(hw, IXGBE_PRC255);
16058eb6488eSEric Joyner 	stats->prc511 += IXGBE_READ_REG(hw, IXGBE_PRC511);
16068eb6488eSEric Joyner 	stats->prc1023 += IXGBE_READ_REG(hw, IXGBE_PRC1023);
16078eb6488eSEric Joyner 	stats->prc1522 += IXGBE_READ_REG(hw, IXGBE_PRC1522);
16088eb6488eSEric Joyner 
16098eb6488eSEric Joyner 	lxon = IXGBE_READ_REG(hw, IXGBE_LXONTXC);
16108eb6488eSEric Joyner 	stats->lxontxc += lxon;
16118eb6488eSEric Joyner 	lxoff = IXGBE_READ_REG(hw, IXGBE_LXOFFTXC);
16128eb6488eSEric Joyner 	stats->lxofftxc += lxoff;
16138eb6488eSEric Joyner 	total = lxon + lxoff;
16148eb6488eSEric Joyner 
16158eb6488eSEric Joyner 	stats->gptc += IXGBE_READ_REG(hw, IXGBE_GPTC);
16168eb6488eSEric Joyner 	stats->mptc += IXGBE_READ_REG(hw, IXGBE_MPTC);
16178eb6488eSEric Joyner 	stats->ptc64 += IXGBE_READ_REG(hw, IXGBE_PTC64);
16188eb6488eSEric Joyner 	stats->gptc -= total;
16198eb6488eSEric Joyner 	stats->mptc -= total;
16208eb6488eSEric Joyner 	stats->ptc64 -= total;
16218eb6488eSEric Joyner 	stats->gotc -= total * ETHER_MIN_LEN;
16228eb6488eSEric Joyner 
16238eb6488eSEric Joyner 	stats->ruc += IXGBE_READ_REG(hw, IXGBE_RUC);
16248eb6488eSEric Joyner 	stats->rfc += IXGBE_READ_REG(hw, IXGBE_RFC);
16258eb6488eSEric Joyner 	stats->roc += IXGBE_READ_REG(hw, IXGBE_ROC);
16268eb6488eSEric Joyner 	stats->rjc += IXGBE_READ_REG(hw, IXGBE_RJC);
16278eb6488eSEric Joyner 	stats->mngprc += IXGBE_READ_REG(hw, IXGBE_MNGPRC);
16288eb6488eSEric Joyner 	stats->mngpdc += IXGBE_READ_REG(hw, IXGBE_MNGPDC);
16298eb6488eSEric Joyner 	stats->mngptc += IXGBE_READ_REG(hw, IXGBE_MNGPTC);
16308eb6488eSEric Joyner 	stats->tpr += IXGBE_READ_REG(hw, IXGBE_TPR);
16318eb6488eSEric Joyner 	stats->tpt += IXGBE_READ_REG(hw, IXGBE_TPT);
16328eb6488eSEric Joyner 	stats->ptc127 += IXGBE_READ_REG(hw, IXGBE_PTC127);
16338eb6488eSEric Joyner 	stats->ptc255 += IXGBE_READ_REG(hw, IXGBE_PTC255);
16348eb6488eSEric Joyner 	stats->ptc511 += IXGBE_READ_REG(hw, IXGBE_PTC511);
16358eb6488eSEric Joyner 	stats->ptc1023 += IXGBE_READ_REG(hw, IXGBE_PTC1023);
16368eb6488eSEric Joyner 	stats->ptc1522 += IXGBE_READ_REG(hw, IXGBE_PTC1522);
16378eb6488eSEric Joyner 	stats->bptc += IXGBE_READ_REG(hw, IXGBE_BPTC);
16388eb6488eSEric Joyner 	stats->xec += IXGBE_READ_REG(hw, IXGBE_XEC);
16398eb6488eSEric Joyner 	stats->fccrc += IXGBE_READ_REG(hw, IXGBE_FCCRC);
16408eb6488eSEric Joyner 	stats->fclast += IXGBE_READ_REG(hw, IXGBE_FCLAST);
16418eb6488eSEric Joyner 	/* Only read FCOE on 82599 */
16428eb6488eSEric Joyner 	if (hw->mac.type != ixgbe_mac_82598EB) {
16438eb6488eSEric Joyner 		stats->fcoerpdc += IXGBE_READ_REG(hw, IXGBE_FCOERPDC);
16448eb6488eSEric Joyner 		stats->fcoeprc += IXGBE_READ_REG(hw, IXGBE_FCOEPRC);
16458eb6488eSEric Joyner 		stats->fcoeptc += IXGBE_READ_REG(hw, IXGBE_FCOEPTC);
16468eb6488eSEric Joyner 		stats->fcoedwrc += IXGBE_READ_REG(hw, IXGBE_FCOEDWRC);
16478eb6488eSEric Joyner 		stats->fcoedwtc += IXGBE_READ_REG(hw, IXGBE_FCOEDWTC);
16488eb6488eSEric Joyner 	}
16498eb6488eSEric Joyner 
16508eb6488eSEric Joyner 	/* Fill out the OS statistics structure */
1651b1d5caf3SKevin Bowling 	IXGBE_SET_IPACKETS(sc, stats->gprc);
1652b1d5caf3SKevin Bowling 	IXGBE_SET_OPACKETS(sc, stats->gptc);
1653b1d5caf3SKevin Bowling 	IXGBE_SET_IBYTES(sc, stats->gorc);
1654b1d5caf3SKevin Bowling 	IXGBE_SET_OBYTES(sc, stats->gotc);
1655b1d5caf3SKevin Bowling 	IXGBE_SET_IMCASTS(sc, stats->mprc);
1656b1d5caf3SKevin Bowling 	IXGBE_SET_OMCASTS(sc, stats->mptc);
1657b1d5caf3SKevin Bowling 	IXGBE_SET_COLLISIONS(sc, 0);
1658b1d5caf3SKevin Bowling 	IXGBE_SET_IQDROPS(sc, total_missed_rx);
1659afb1aa4eSPiotr Pietruszewski 
1660afb1aa4eSPiotr Pietruszewski 	/*
1661afb1aa4eSPiotr Pietruszewski 	 * Aggregate following types of errors as RX errors:
1662afb1aa4eSPiotr Pietruszewski 	 * - CRC error count,
1663afb1aa4eSPiotr Pietruszewski 	 * - illegal byte error count,
1664afb1aa4eSPiotr Pietruszewski 	 * - missed packets count,
1665afb1aa4eSPiotr Pietruszewski 	 * - length error count,
1666afb1aa4eSPiotr Pietruszewski 	 * - undersized packets count,
1667afb1aa4eSPiotr Pietruszewski 	 * - fragmented packets count,
1668afb1aa4eSPiotr Pietruszewski 	 * - oversized packets count,
1669afb1aa4eSPiotr Pietruszewski 	 * - jabber count.
1670afb1aa4eSPiotr Pietruszewski 	 */
16718526120aSAndrey V. Elsukov 	IXGBE_SET_IERRORS(sc, stats->crcerrs + stats->illerrc +
1672c58d34ddSKevin Bowling 	    stats->mpc[0] + stats->rlec + stats->ruc + stats->rfc +
1673c58d34ddSKevin Bowling 	    stats->roc + stats->rjc);
16748eb6488eSEric Joyner } /* ixgbe_update_stats_counters */
16758eb6488eSEric Joyner 
16768eb6488eSEric Joyner /************************************************************************
16778eb6488eSEric Joyner  * ixgbe_add_hw_stats
16788eb6488eSEric Joyner  *
16798eb6488eSEric Joyner  *   Add sysctl variables, one per statistic, to the system.
16808eb6488eSEric Joyner  ************************************************************************/
16818eb6488eSEric Joyner static void
ixgbe_add_hw_stats(struct ixgbe_softc * sc)1682b1d5caf3SKevin Bowling ixgbe_add_hw_stats(struct ixgbe_softc *sc)
16838eb6488eSEric Joyner {
1684b1d5caf3SKevin Bowling 	device_t dev = iflib_get_dev(sc->ctx);
1685c19c7afeSEric Joyner 	struct ix_rx_queue *rx_que;
1686c19c7afeSEric Joyner 	struct ix_tx_queue *tx_que;
16878eb6488eSEric Joyner 	struct sysctl_ctx_list *ctx = device_get_sysctl_ctx(dev);
16888eb6488eSEric Joyner 	struct sysctl_oid *tree = device_get_sysctl_tree(dev);
16898eb6488eSEric Joyner 	struct sysctl_oid_list *child = SYSCTL_CHILDREN(tree);
1690b1d5caf3SKevin Bowling 	struct ixgbe_hw_stats *stats = &sc->stats.pf;
16918eb6488eSEric Joyner 	struct sysctl_oid *stat_node, *queue_node;
16928eb6488eSEric Joyner 	struct sysctl_oid_list *stat_list, *queue_list;
1693c19c7afeSEric Joyner 	int i;
16948eb6488eSEric Joyner 
16958eb6488eSEric Joyner #define QUEUE_NAME_LEN 32
16968eb6488eSEric Joyner 	char namebuf[QUEUE_NAME_LEN];
16978eb6488eSEric Joyner 
16988eb6488eSEric Joyner 	/* Driver Statistics */
16998eb6488eSEric Joyner 	SYSCTL_ADD_ULONG(ctx, child, OID_AUTO, "dropped",
1700b1d5caf3SKevin Bowling 	    CTLFLAG_RD, &sc->dropped_pkts, "Driver dropped packets");
17018eb6488eSEric Joyner 	SYSCTL_ADD_ULONG(ctx, child, OID_AUTO, "watchdog_events",
1702b1d5caf3SKevin Bowling 	    CTLFLAG_RD, &sc->watchdog_events, "Watchdog timeouts");
17038eb6488eSEric Joyner 	SYSCTL_ADD_ULONG(ctx, child, OID_AUTO, "link_irq",
1704b1d5caf3SKevin Bowling 	    CTLFLAG_RD, &sc->link_irq, "Link MSI-X IRQ Handled");
17058eb6488eSEric Joyner 
1706c58d34ddSKevin Bowling 	for (i = 0, tx_que = sc->tx_queues; i < sc->num_tx_queues;
1707c58d34ddSKevin Bowling 	    i++, tx_que++) {
1708c19c7afeSEric Joyner 		struct tx_ring *txr = &tx_que->txr;
17098eb6488eSEric Joyner 		snprintf(namebuf, QUEUE_NAME_LEN, "queue%d", i);
17108eb6488eSEric Joyner 		queue_node = SYSCTL_ADD_NODE(ctx, child, OID_AUTO, namebuf,
171120b91f0aSPawel Biernacki 		    CTLFLAG_RD | CTLFLAG_MPSAFE, NULL, "Queue Name");
17128eb6488eSEric Joyner 		queue_list = SYSCTL_CHILDREN(queue_node);
17138eb6488eSEric Joyner 
17148eb6488eSEric Joyner 		SYSCTL_ADD_PROC(ctx, queue_list, OID_AUTO, "txd_head",
1715f72de14eSKevin Bowling 		    CTLTYPE_UINT | CTLFLAG_RD, txr, 0,
1716c58d34ddSKevin Bowling 		    ixgbe_sysctl_tdh_handler, "IU",
1717c58d34ddSKevin Bowling 		    "Transmit Descriptor Head");
17188eb6488eSEric Joyner 		SYSCTL_ADD_PROC(ctx, queue_list, OID_AUTO, "txd_tail",
1719f72de14eSKevin Bowling 		    CTLTYPE_UINT | CTLFLAG_RD, txr, 0,
1720c58d34ddSKevin Bowling 		    ixgbe_sysctl_tdt_handler, "IU",
1721c58d34ddSKevin Bowling 		    "Transmit Descriptor Tail");
17228eb6488eSEric Joyner 		SYSCTL_ADD_UQUAD(ctx, queue_list, OID_AUTO, "tso_tx",
17238eb6488eSEric Joyner 		    CTLFLAG_RD, &txr->tso_tx, "TSO");
17248eb6488eSEric Joyner 		SYSCTL_ADD_UQUAD(ctx, queue_list, OID_AUTO, "tx_packets",
17258eb6488eSEric Joyner 		    CTLFLAG_RD, &txr->total_packets,
17268eb6488eSEric Joyner 		    "Queue Packets Transmitted");
17278eb6488eSEric Joyner 	}
17288eb6488eSEric Joyner 
1729c58d34ddSKevin Bowling 	for (i = 0, rx_que = sc->rx_queues; i < sc->num_rx_queues;
1730c58d34ddSKevin Bowling 	    i++, rx_que++) {
1731c19c7afeSEric Joyner 		struct rx_ring *rxr = &rx_que->rxr;
17328eb6488eSEric Joyner 		snprintf(namebuf, QUEUE_NAME_LEN, "queue%d", i);
17338eb6488eSEric Joyner 		queue_node = SYSCTL_ADD_NODE(ctx, child, OID_AUTO, namebuf,
173420b91f0aSPawel Biernacki 		    CTLFLAG_RD | CTLFLAG_MPSAFE, NULL, "Queue Name");
17358eb6488eSEric Joyner 		queue_list = SYSCTL_CHILDREN(queue_node);
17368eb6488eSEric Joyner 
1737c19c7afeSEric Joyner 		SYSCTL_ADD_PROC(ctx, queue_list, OID_AUTO, "interrupt_rate",
1738f72de14eSKevin Bowling 		    CTLTYPE_UINT | CTLFLAG_RW,
1739b1d5caf3SKevin Bowling 		    &sc->rx_queues[i], 0,
1740c19c7afeSEric Joyner 		    ixgbe_sysctl_interrupt_rate_handler, "IU",
1741c19c7afeSEric Joyner 		    "Interrupt Rate");
1742c19c7afeSEric Joyner 		SYSCTL_ADD_UQUAD(ctx, queue_list, OID_AUTO, "irqs",
1743b1d5caf3SKevin Bowling 		    CTLFLAG_RD, &(sc->rx_queues[i].irqs),
1744c19c7afeSEric Joyner 		    "irqs on this queue");
17458eb6488eSEric Joyner 		SYSCTL_ADD_PROC(ctx, queue_list, OID_AUTO, "rxd_head",
1746f72de14eSKevin Bowling 		    CTLTYPE_UINT | CTLFLAG_RD, rxr, 0,
1747c58d34ddSKevin Bowling 		    ixgbe_sysctl_rdh_handler, "IU",
1748c58d34ddSKevin Bowling 		    "Receive Descriptor Head");
17498eb6488eSEric Joyner 		SYSCTL_ADD_PROC(ctx, queue_list, OID_AUTO, "rxd_tail",
1750f72de14eSKevin Bowling 		    CTLTYPE_UINT | CTLFLAG_RD, rxr, 0,
1751c58d34ddSKevin Bowling 		    ixgbe_sysctl_rdt_handler, "IU",
1752c58d34ddSKevin Bowling 		    "Receive Descriptor Tail");
17538eb6488eSEric Joyner 		SYSCTL_ADD_UQUAD(ctx, queue_list, OID_AUTO, "rx_packets",
17548eb6488eSEric Joyner 		    CTLFLAG_RD, &rxr->rx_packets, "Queue Packets Received");
17558eb6488eSEric Joyner 		SYSCTL_ADD_UQUAD(ctx, queue_list, OID_AUTO, "rx_bytes",
17568eb6488eSEric Joyner 		    CTLFLAG_RD, &rxr->rx_bytes, "Queue Bytes Received");
17578eb6488eSEric Joyner 		SYSCTL_ADD_UQUAD(ctx, queue_list, OID_AUTO, "rx_copies",
17588eb6488eSEric Joyner 		    CTLFLAG_RD, &rxr->rx_copies, "Copied RX Frames");
17598eb6488eSEric Joyner 		SYSCTL_ADD_UQUAD(ctx, queue_list, OID_AUTO, "rx_discarded",
17608eb6488eSEric Joyner 		    CTLFLAG_RD, &rxr->rx_discarded, "Discarded RX packets");
17618eb6488eSEric Joyner 	}
17628eb6488eSEric Joyner 
17638eb6488eSEric Joyner 	/* MAC stats get their own sub node */
17648eb6488eSEric Joyner 	stat_node = SYSCTL_ADD_NODE(ctx, child, OID_AUTO, "mac_stats",
176520b91f0aSPawel Biernacki 	    CTLFLAG_RD | CTLFLAG_MPSAFE, NULL, "MAC Statistics");
17668eb6488eSEric Joyner 	stat_list = SYSCTL_CHILDREN(stat_node);
17678eb6488eSEric Joyner 
1768afb1aa4eSPiotr Pietruszewski 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_errs",
1769b1d5caf3SKevin Bowling 	    CTLFLAG_RD, &sc->ierrors, IXGBE_SYSCTL_DESC_RX_ERRS);
17708eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "crc_errs",
17718eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->crcerrs, "CRC Errors");
17728eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "ill_errs",
17738eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->illerrc, "Illegal Byte Errors");
17748eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "byte_errs",
17758eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->errbc, "Byte Errors");
17768eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "short_discards",
17778eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->mspdc, "MAC Short Packets Discarded");
17788eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "local_faults",
17798eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->mlfc, "MAC Local Faults");
17808eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "remote_faults",
17818eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->mrfc, "MAC Remote Faults");
17828eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rec_len_errs",
17838eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->rlec, "Receive Length Errors");
17848eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_missed_packets",
17858eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->mpc[0], "RX Missed Packet Count");
17868eb6488eSEric Joyner 
17878eb6488eSEric Joyner 	/* Flow Control stats */
17888eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "xon_txd",
17898eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->lxontxc, "Link XON Transmitted");
17908eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "xon_recvd",
17918eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->lxonrxc, "Link XON Received");
17928eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "xoff_txd",
17938eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->lxofftxc, "Link XOFF Transmitted");
17948eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "xoff_recvd",
17958eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->lxoffrxc, "Link XOFF Received");
17968eb6488eSEric Joyner 
17978eb6488eSEric Joyner 	/* Packet Reception Stats */
17988eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "total_octets_rcvd",
17998eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->tor, "Total Octets Received");
18008eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "good_octets_rcvd",
18018eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->gorc, "Good Octets Received");
18028eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "total_pkts_rcvd",
18038eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->tpr, "Total Packets Received");
18048eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "good_pkts_rcvd",
18058eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->gprc, "Good Packets Received");
18068eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "mcast_pkts_rcvd",
18078eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->mprc, "Multicast Packets Received");
18088eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "bcast_pkts_rcvd",
18098eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->bprc, "Broadcast Packets Received");
18108eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_frames_64",
18118eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->prc64, "64 byte frames received ");
18128eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_frames_65_127",
18138eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->prc127, "65-127 byte frames received");
18148eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_frames_128_255",
18158eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->prc255, "128-255 byte frames received");
18168eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_frames_256_511",
18178eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->prc511, "256-511 byte frames received");
18188eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_frames_512_1023",
18198eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->prc1023, "512-1023 byte frames received");
18208eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "rx_frames_1024_1522",
18218eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->prc1522, "1023-1522 byte frames received");
18228eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "recv_undersized",
18238eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->ruc, "Receive Undersized");
18248eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "recv_fragmented",
18258eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->rfc, "Fragmented Packets Received ");
18268eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "recv_oversized",
18278eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->roc, "Oversized Packets Received");
18288eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "recv_jabberd",
18298eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->rjc, "Received Jabber");
18308eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "management_pkts_rcvd",
18318eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->mngprc, "Management Packets Received");
18328eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "management_pkts_drpd",
18338eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->mngptc, "Management Packets Dropped");
18348eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "checksum_errs",
18358eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->xec, "Checksum Errors");
18368eb6488eSEric Joyner 
18378eb6488eSEric Joyner 	/* Packet Transmission Stats */
18388eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "good_octets_txd",
18398eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->gotc, "Good Octets Transmitted");
18408eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "total_pkts_txd",
18418eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->tpt, "Total Packets Transmitted");
18428eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "good_pkts_txd",
18438eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->gptc, "Good Packets Transmitted");
18448eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "bcast_pkts_txd",
18458eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->bptc, "Broadcast Packets Transmitted");
18468eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "mcast_pkts_txd",
18478eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->mptc, "Multicast Packets Transmitted");
18488eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "management_pkts_txd",
18498eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->mngptc, "Management Packets Transmitted");
18508eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "tx_frames_64",
18518eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->ptc64, "64 byte frames transmitted ");
18528eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "tx_frames_65_127",
18538eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->ptc127, "65-127 byte frames transmitted");
18548eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "tx_frames_128_255",
18558eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->ptc255, "128-255 byte frames transmitted");
18568eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "tx_frames_256_511",
18578eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->ptc511, "256-511 byte frames transmitted");
18588eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "tx_frames_512_1023",
18598eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->ptc1023, "512-1023 byte frames transmitted");
18608eb6488eSEric Joyner 	SYSCTL_ADD_UQUAD(ctx, stat_list, OID_AUTO, "tx_frames_1024_1522",
18618eb6488eSEric Joyner 	    CTLFLAG_RD, &stats->ptc1522, "1024-1522 byte frames transmitted");
18628eb6488eSEric Joyner } /* ixgbe_add_hw_stats */
18638eb6488eSEric Joyner 
18648eb6488eSEric Joyner /************************************************************************
18658eb6488eSEric Joyner  * ixgbe_sysctl_tdh_handler - Transmit Descriptor Head handler function
18668eb6488eSEric Joyner  *
18678eb6488eSEric Joyner  *   Retrieves the TDH value from the hardware
18688eb6488eSEric Joyner  ************************************************************************/
1869758cc3dcSJack F Vogel static int
ixgbe_sysctl_tdh_handler(SYSCTL_HANDLER_ARGS)1870758cc3dcSJack F Vogel ixgbe_sysctl_tdh_handler(SYSCTL_HANDLER_ARGS)
1871758cc3dcSJack F Vogel {
1872758cc3dcSJack F Vogel 	struct tx_ring *txr = ((struct tx_ring *)oidp->oid_arg1);
18738eb6488eSEric Joyner 	int error;
18748eb6488eSEric Joyner 	unsigned int val;
1875758cc3dcSJack F Vogel 
18768eb6488eSEric Joyner 	if (!txr)
18778eb6488eSEric Joyner 		return (0);
18788eb6488eSEric Joyner 
1879f72de14eSKevin Bowling 
1880f72de14eSKevin Bowling 	if (atomic_load_acq_int(&txr->sc->recovery_mode))
1881f72de14eSKevin Bowling 		return (EPERM);
1882f72de14eSKevin Bowling 
1883b1d5caf3SKevin Bowling 	val = IXGBE_READ_REG(&txr->sc->hw, IXGBE_TDH(txr->me));
1884758cc3dcSJack F Vogel 	error = sysctl_handle_int(oidp, &val, 0, req);
1885758cc3dcSJack F Vogel 	if (error || !req->newptr)
1886758cc3dcSJack F Vogel 		return error;
1887758cc3dcSJack F Vogel 
18888eb6488eSEric Joyner 	return (0);
18898eb6488eSEric Joyner } /* ixgbe_sysctl_tdh_handler */
18908eb6488eSEric Joyner 
18918eb6488eSEric Joyner /************************************************************************
18928eb6488eSEric Joyner  * ixgbe_sysctl_tdt_handler - Transmit Descriptor Tail handler function
18938eb6488eSEric Joyner  *
1894758cc3dcSJack F Vogel  *   Retrieves the TDT value from the hardware
18958eb6488eSEric Joyner  ************************************************************************/
1896758cc3dcSJack F Vogel static int
ixgbe_sysctl_tdt_handler(SYSCTL_HANDLER_ARGS)1897758cc3dcSJack F Vogel ixgbe_sysctl_tdt_handler(SYSCTL_HANDLER_ARGS)
1898758cc3dcSJack F Vogel {
1899758cc3dcSJack F Vogel 	struct tx_ring *txr = ((struct tx_ring *)oidp->oid_arg1);
19008eb6488eSEric Joyner 	int error;
19018eb6488eSEric Joyner 	unsigned int val;
1902758cc3dcSJack F Vogel 
19038eb6488eSEric Joyner 	if (!txr)
19048eb6488eSEric Joyner 		return (0);
19058eb6488eSEric Joyner 
1906f72de14eSKevin Bowling 	if (atomic_load_acq_int(&txr->sc->recovery_mode))
1907f72de14eSKevin Bowling 		return (EPERM);
1908f72de14eSKevin Bowling 
1909b1d5caf3SKevin Bowling 	val = IXGBE_READ_REG(&txr->sc->hw, IXGBE_TDT(txr->me));
1910758cc3dcSJack F Vogel 	error = sysctl_handle_int(oidp, &val, 0, req);
1911758cc3dcSJack F Vogel 	if (error || !req->newptr)
1912758cc3dcSJack F Vogel 		return error;
1913758cc3dcSJack F Vogel 
19148eb6488eSEric Joyner 	return (0);
19158eb6488eSEric Joyner } /* ixgbe_sysctl_tdt_handler */
19168eb6488eSEric Joyner 
19178eb6488eSEric Joyner /************************************************************************
19188eb6488eSEric Joyner  * ixgbe_sysctl_rdh_handler - Receive Descriptor Head handler function
19198eb6488eSEric Joyner  *
1920758cc3dcSJack F Vogel  *   Retrieves the RDH value from the hardware
19218eb6488eSEric Joyner  ************************************************************************/
1922758cc3dcSJack F Vogel static int
ixgbe_sysctl_rdh_handler(SYSCTL_HANDLER_ARGS)1923758cc3dcSJack F Vogel ixgbe_sysctl_rdh_handler(SYSCTL_HANDLER_ARGS)
1924758cc3dcSJack F Vogel {
1925758cc3dcSJack F Vogel 	struct rx_ring *rxr = ((struct rx_ring *)oidp->oid_arg1);
19268eb6488eSEric Joyner 	int error;
19278eb6488eSEric Joyner 	unsigned int val;
1928758cc3dcSJack F Vogel 
19298eb6488eSEric Joyner 	if (!rxr)
19308eb6488eSEric Joyner 		return (0);
19318eb6488eSEric Joyner 
1932f72de14eSKevin Bowling 	if (atomic_load_acq_int(&rxr->sc->recovery_mode))
1933f72de14eSKevin Bowling 		return (EPERM);
1934f72de14eSKevin Bowling 
1935b1d5caf3SKevin Bowling 	val = IXGBE_READ_REG(&rxr->sc->hw, IXGBE_RDH(rxr->me));
1936758cc3dcSJack F Vogel 	error = sysctl_handle_int(oidp, &val, 0, req);
1937758cc3dcSJack F Vogel 	if (error || !req->newptr)
1938758cc3dcSJack F Vogel 		return error;
1939758cc3dcSJack F Vogel 
19408eb6488eSEric Joyner 	return (0);
19418eb6488eSEric Joyner } /* ixgbe_sysctl_rdh_handler */
19428eb6488eSEric Joyner 
19438eb6488eSEric Joyner /************************************************************************
19448eb6488eSEric Joyner  * ixgbe_sysctl_rdt_handler - Receive Descriptor Tail handler function
19458eb6488eSEric Joyner  *
1946758cc3dcSJack F Vogel  *   Retrieves the RDT value from the hardware
19478eb6488eSEric Joyner  ************************************************************************/
1948758cc3dcSJack F Vogel static int
ixgbe_sysctl_rdt_handler(SYSCTL_HANDLER_ARGS)1949758cc3dcSJack F Vogel ixgbe_sysctl_rdt_handler(SYSCTL_HANDLER_ARGS)
1950758cc3dcSJack F Vogel {
1951758cc3dcSJack F Vogel 	struct rx_ring *rxr = ((struct rx_ring *)oidp->oid_arg1);
19528eb6488eSEric Joyner 	int error;
19538eb6488eSEric Joyner 	unsigned int val;
1954758cc3dcSJack F Vogel 
19558eb6488eSEric Joyner 	if (!rxr)
19568eb6488eSEric Joyner 		return (0);
19578eb6488eSEric Joyner 
1958f72de14eSKevin Bowling 	if (atomic_load_acq_int(&rxr->sc->recovery_mode))
1959f72de14eSKevin Bowling 		return (EPERM);
1960f72de14eSKevin Bowling 
1961b1d5caf3SKevin Bowling 	val = IXGBE_READ_REG(&rxr->sc->hw, IXGBE_RDT(rxr->me));
1962758cc3dcSJack F Vogel 	error = sysctl_handle_int(oidp, &val, 0, req);
1963758cc3dcSJack F Vogel 	if (error || !req->newptr)
1964758cc3dcSJack F Vogel 		return error;
19658eb6488eSEric Joyner 
19668eb6488eSEric Joyner 	return (0);
19678eb6488eSEric Joyner } /* ixgbe_sysctl_rdt_handler */
19688eb6488eSEric Joyner 
19698eb6488eSEric Joyner /************************************************************************
1970c19c7afeSEric Joyner  * ixgbe_if_vlan_register
19718eb6488eSEric Joyner  *
19728eb6488eSEric Joyner  *   Run via vlan config EVENT, it enables us to use the
19738eb6488eSEric Joyner  *   HW Filter table since we can get the vlan id. This
19748eb6488eSEric Joyner  *   just creates the entry in the soft version of the
19758eb6488eSEric Joyner  *   VFTA, init will repopulate the real table.
19768eb6488eSEric Joyner  ************************************************************************/
19778eb6488eSEric Joyner static void
ixgbe_if_vlan_register(if_ctx_t ctx,u16 vtag)1978c19c7afeSEric Joyner ixgbe_if_vlan_register(if_ctx_t ctx, u16 vtag)
19798eb6488eSEric Joyner {
1980b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
19818eb6488eSEric Joyner 	u16 index, bit;
19828eb6488eSEric Joyner 
19838eb6488eSEric Joyner 	index = (vtag >> 5) & 0x7F;
19848eb6488eSEric Joyner 	bit = vtag & 0x1F;
1985b1d5caf3SKevin Bowling 	sc->shadow_vfta[index] |= (1 << bit);
1986b1d5caf3SKevin Bowling 	++sc->num_vlans;
1987c19c7afeSEric Joyner 	ixgbe_setup_vlan_hw_support(ctx);
1988c19c7afeSEric Joyner } /* ixgbe_if_vlan_register */
19898eb6488eSEric Joyner 
19908eb6488eSEric Joyner /************************************************************************
1991c19c7afeSEric Joyner  * ixgbe_if_vlan_unregister
19928eb6488eSEric Joyner  *
19938eb6488eSEric Joyner  *   Run via vlan unconfig EVENT, remove our entry in the soft vfta.
19948eb6488eSEric Joyner  ************************************************************************/
19958eb6488eSEric Joyner static void
ixgbe_if_vlan_unregister(if_ctx_t ctx,u16 vtag)1996c19c7afeSEric Joyner ixgbe_if_vlan_unregister(if_ctx_t ctx, u16 vtag)
19978eb6488eSEric Joyner {
1998b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
19998eb6488eSEric Joyner 	u16 index, bit;
20008eb6488eSEric Joyner 
20018eb6488eSEric Joyner 	index = (vtag >> 5) & 0x7F;
20028eb6488eSEric Joyner 	bit = vtag & 0x1F;
2003b1d5caf3SKevin Bowling 	sc->shadow_vfta[index] &= ~(1 << bit);
2004b1d5caf3SKevin Bowling 	--sc->num_vlans;
20058eb6488eSEric Joyner 	/* Re-init to load the changes */
2006c19c7afeSEric Joyner 	ixgbe_setup_vlan_hw_support(ctx);
2007c19c7afeSEric Joyner } /* ixgbe_if_vlan_unregister */
20088eb6488eSEric Joyner 
20098eb6488eSEric Joyner /************************************************************************
20108eb6488eSEric Joyner  * ixgbe_setup_vlan_hw_support
20118eb6488eSEric Joyner  ************************************************************************/
20128eb6488eSEric Joyner static void
ixgbe_setup_vlan_hw_support(if_ctx_t ctx)2013c19c7afeSEric Joyner ixgbe_setup_vlan_hw_support(if_ctx_t ctx)
20148eb6488eSEric Joyner {
2015ff06a8dbSJustin Hibbits 	if_t ifp = iflib_get_ifp(ctx);
2016b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
2017b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
20188eb6488eSEric Joyner 	struct rx_ring *rxr;
20198eb6488eSEric Joyner 	int i;
20208eb6488eSEric Joyner 	u32 ctrl;
20218eb6488eSEric Joyner 
20228eb6488eSEric Joyner 
20238eb6488eSEric Joyner 	/*
20248eb6488eSEric Joyner 	 * We get here thru init_locked, meaning
20258eb6488eSEric Joyner 	 * a soft reset, this has already cleared
20268eb6488eSEric Joyner 	 * the VFTA and other state, so if there
20278eb6488eSEric Joyner 	 * have been no vlan's registered do nothing.
20288eb6488eSEric Joyner 	 */
2029c58d34ddSKevin Bowling 	if (sc->num_vlans == 0 ||
2030c58d34ddSKevin Bowling 	    (if_getcapenable(ifp) & IFCAP_VLAN_HWTAGGING) == 0) {
2031e7abb897SKristof Provost 		/* Clear the vlan hw flag */
2032e7abb897SKristof Provost 		for (i = 0; i < sc->num_rx_queues; i++) {
2033e7abb897SKristof Provost 			rxr = &sc->rx_queues[i].rxr;
2034e7abb897SKristof Provost 			/* On 82599 the VLAN enable is per/queue in RXDCTL */
2035e7abb897SKristof Provost 			if (hw->mac.type != ixgbe_mac_82598EB) {
2036c58d34ddSKevin Bowling 				ctrl = IXGBE_READ_REG(hw,
2037c58d34ddSKevin Bowling 				    IXGBE_RXDCTL(rxr->me));
2038e7abb897SKristof Provost 				ctrl &= ~IXGBE_RXDCTL_VME;
2039c58d34ddSKevin Bowling 				IXGBE_WRITE_REG(hw, IXGBE_RXDCTL(rxr->me),
2040c58d34ddSKevin Bowling 				    ctrl);
2041e7abb897SKristof Provost 			}
2042e7abb897SKristof Provost 			rxr->vtag_strip = false;
2043e7abb897SKristof Provost 		}
2044e7abb897SKristof Provost 		ctrl = IXGBE_READ_REG(hw, IXGBE_VLNCTRL);
2045e7abb897SKristof Provost 		/* Enable the Filter Table if enabled */
2046e7abb897SKristof Provost 		ctrl |= IXGBE_VLNCTRL_CFIEN;
2047e7abb897SKristof Provost 		ctrl &= ~IXGBE_VLNCTRL_VFE;
2048e7abb897SKristof Provost 		if (hw->mac.type == ixgbe_mac_82598EB)
2049e7abb897SKristof Provost 			ctrl &= ~IXGBE_VLNCTRL_VME;
2050e7abb897SKristof Provost 		IXGBE_WRITE_REG(hw, IXGBE_VLNCTRL, ctrl);
20518eb6488eSEric Joyner 		return;
2052e7abb897SKristof Provost 	}
20538eb6488eSEric Joyner 
20548eb6488eSEric Joyner 	/* Setup the queues for vlans */
2055ff06a8dbSJustin Hibbits 	if (if_getcapenable(ifp) & IFCAP_VLAN_HWTAGGING) {
2056b1d5caf3SKevin Bowling 		for (i = 0; i < sc->num_rx_queues; i++) {
2057b1d5caf3SKevin Bowling 			rxr = &sc->rx_queues[i].rxr;
20588eb6488eSEric Joyner 			/* On 82599 the VLAN enable is per/queue in RXDCTL */
20598eb6488eSEric Joyner 			if (hw->mac.type != ixgbe_mac_82598EB) {
2060c58d34ddSKevin Bowling 				ctrl = IXGBE_READ_REG(hw,
2061c58d34ddSKevin Bowling 				    IXGBE_RXDCTL(rxr->me));
20628eb6488eSEric Joyner 				ctrl |= IXGBE_RXDCTL_VME;
2063c58d34ddSKevin Bowling 				IXGBE_WRITE_REG(hw, IXGBE_RXDCTL(rxr->me),
2064c58d34ddSKevin Bowling 				    ctrl);
20658eb6488eSEric Joyner 			}
206679b36ec9SKevin Bowling 			rxr->vtag_strip = true;
2067758cc3dcSJack F Vogel 		}
20689c92bb71SSean Bruno 	}
2069758cc3dcSJack F Vogel 
2070ff06a8dbSJustin Hibbits 	if ((if_getcapenable(ifp) & IFCAP_VLAN_HWFILTER) == 0)
20718eb6488eSEric Joyner 		return;
20728eb6488eSEric Joyner 	/*
20738eb6488eSEric Joyner 	 * A soft reset zero's out the VFTA, so
20748eb6488eSEric Joyner 	 * we need to repopulate it now.
20758eb6488eSEric Joyner 	 */
20768eb6488eSEric Joyner 	for (i = 0; i < IXGBE_VFTA_SIZE; i++)
2077b1d5caf3SKevin Bowling 		if (sc->shadow_vfta[i] != 0)
20788eb6488eSEric Joyner 			IXGBE_WRITE_REG(hw, IXGBE_VFTA(i),
2079b1d5caf3SKevin Bowling 			    sc->shadow_vfta[i]);
20808eb6488eSEric Joyner 
20818eb6488eSEric Joyner 	ctrl = IXGBE_READ_REG(hw, IXGBE_VLNCTRL);
20828eb6488eSEric Joyner 	/* Enable the Filter Table if enabled */
2083ff06a8dbSJustin Hibbits 	if (if_getcapenable(ifp) & IFCAP_VLAN_HWFILTER) {
20848eb6488eSEric Joyner 		ctrl &= ~IXGBE_VLNCTRL_CFIEN;
20858eb6488eSEric Joyner 		ctrl |= IXGBE_VLNCTRL_VFE;
20868eb6488eSEric Joyner 	}
20878eb6488eSEric Joyner 	if (hw->mac.type == ixgbe_mac_82598EB)
20888eb6488eSEric Joyner 		ctrl |= IXGBE_VLNCTRL_VME;
20898eb6488eSEric Joyner 	IXGBE_WRITE_REG(hw, IXGBE_VLNCTRL, ctrl);
20908eb6488eSEric Joyner } /* ixgbe_setup_vlan_hw_support */
20918eb6488eSEric Joyner 
20928eb6488eSEric Joyner /************************************************************************
20938eb6488eSEric Joyner  * ixgbe_get_slot_info
20948eb6488eSEric Joyner  *
20958eb6488eSEric Joyner  *   Get the width and transaction speed of
20968eb6488eSEric Joyner  *   the slot this adapter is plugged into.
20978eb6488eSEric Joyner  ************************************************************************/
20988eb6488eSEric Joyner static void
ixgbe_get_slot_info(struct ixgbe_softc * sc)2099b1d5caf3SKevin Bowling ixgbe_get_slot_info(struct ixgbe_softc *sc)
21008eb6488eSEric Joyner {
2101b1d5caf3SKevin Bowling 	device_t dev = iflib_get_dev(sc->ctx);
2102b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
210379b36ec9SKevin Bowling 	int bus_info_valid = true;
21048eb6488eSEric Joyner 	u32 offset;
21058eb6488eSEric Joyner 	u16 link;
21068eb6488eSEric Joyner 
21078eb6488eSEric Joyner 	/* Some devices are behind an internal bridge */
21088eb6488eSEric Joyner 	switch (hw->device_id) {
21098eb6488eSEric Joyner 	case IXGBE_DEV_ID_82599_SFP_SF_QP:
21108eb6488eSEric Joyner 	case IXGBE_DEV_ID_82599_QSFP_SF_QP:
21118eb6488eSEric Joyner 		goto get_parent_info;
21128eb6488eSEric Joyner 	default:
21138eb6488eSEric Joyner 		break;
21148eb6488eSEric Joyner 	}
21158eb6488eSEric Joyner 
21168eb6488eSEric Joyner 	ixgbe_get_bus_info(hw);
21178eb6488eSEric Joyner 
21188eb6488eSEric Joyner 	/*
21198eb6488eSEric Joyner 	 * Some devices don't use PCI-E, but there is no need
21208eb6488eSEric Joyner 	 * to display "Unknown" for bus speed and width.
21218eb6488eSEric Joyner 	 */
21228eb6488eSEric Joyner 	switch (hw->mac.type) {
21238eb6488eSEric Joyner 	case ixgbe_mac_X550EM_x:
21248eb6488eSEric Joyner 	case ixgbe_mac_X550EM_a:
21258eb6488eSEric Joyner 		return;
21268eb6488eSEric Joyner 	default:
21278eb6488eSEric Joyner 		goto display;
21288eb6488eSEric Joyner 	}
21298eb6488eSEric Joyner 
21308eb6488eSEric Joyner get_parent_info:
21318eb6488eSEric Joyner 	/*
21328eb6488eSEric Joyner 	 * For the Quad port adapter we need to parse back
21338eb6488eSEric Joyner 	 * up the PCI tree to find the speed of the expansion
21348eb6488eSEric Joyner 	 * slot into which this adapter is plugged. A bit more work.
21358eb6488eSEric Joyner 	 */
21368eb6488eSEric Joyner 	dev = device_get_parent(device_get_parent(dev));
21378eb6488eSEric Joyner #ifdef IXGBE_DEBUG
21388eb6488eSEric Joyner 	device_printf(dev, "parent pcib = %x,%x,%x\n", pci_get_bus(dev),
21398eb6488eSEric Joyner 	    pci_get_slot(dev), pci_get_function(dev));
21408eb6488eSEric Joyner #endif
21418eb6488eSEric Joyner 	dev = device_get_parent(device_get_parent(dev));
21428eb6488eSEric Joyner #ifdef IXGBE_DEBUG
21438eb6488eSEric Joyner 	device_printf(dev, "slot pcib = %x,%x,%x\n", pci_get_bus(dev),
21448eb6488eSEric Joyner 	    pci_get_slot(dev), pci_get_function(dev));
21458eb6488eSEric Joyner #endif
21468eb6488eSEric Joyner 	/* Now get the PCI Express Capabilities offset */
21478eb6488eSEric Joyner 	if (pci_find_cap(dev, PCIY_EXPRESS, &offset)) {
21488eb6488eSEric Joyner 		/*
21498eb6488eSEric Joyner 		 * Hmm...can't get PCI-Express capabilities.
21508eb6488eSEric Joyner 		 * Falling back to default method.
21518eb6488eSEric Joyner 		 */
215279b36ec9SKevin Bowling 		bus_info_valid = false;
21538eb6488eSEric Joyner 		ixgbe_get_bus_info(hw);
21548eb6488eSEric Joyner 		goto display;
21558eb6488eSEric Joyner 	}
21568eb6488eSEric Joyner 	/* ...and read the Link Status Register */
21578eb6488eSEric Joyner 	link = pci_read_config(dev, offset + PCIER_LINK_STA, 2);
21588eb6488eSEric Joyner 	ixgbe_set_pci_config_data_generic(hw, link);
21598eb6488eSEric Joyner 
21608eb6488eSEric Joyner display:
21618eb6488eSEric Joyner 	device_printf(dev, "PCI Express Bus: Speed %s %s\n",
21628eb6488eSEric Joyner 	    ((hw->bus.speed == ixgbe_bus_speed_8000)    ? "8.0GT/s"  :
21638eb6488eSEric Joyner 	     (hw->bus.speed == ixgbe_bus_speed_5000)    ? "5.0GT/s"  :
21648eb6488eSEric Joyner 	     (hw->bus.speed == ixgbe_bus_speed_2500)    ? "2.5GT/s"  :
21658eb6488eSEric Joyner 	     "Unknown"),
21668eb6488eSEric Joyner 	    ((hw->bus.width == ixgbe_bus_width_pcie_x8) ? "Width x8" :
21678eb6488eSEric Joyner 	     (hw->bus.width == ixgbe_bus_width_pcie_x4) ? "Width x4" :
21688eb6488eSEric Joyner 	     (hw->bus.width == ixgbe_bus_width_pcie_x1) ? "Width x1" :
21698eb6488eSEric Joyner 	     "Unknown"));
21708eb6488eSEric Joyner 
21718eb6488eSEric Joyner 	if (bus_info_valid) {
21728eb6488eSEric Joyner 		if ((hw->device_id != IXGBE_DEV_ID_82599_SFP_SF_QP) &&
21738eb6488eSEric Joyner 		    ((hw->bus.width <= ixgbe_bus_width_pcie_x4) &&
21748eb6488eSEric Joyner 		    (hw->bus.speed == ixgbe_bus_speed_2500))) {
2175c58d34ddSKevin Bowling 			device_printf(dev,
2176c58d34ddSKevin Bowling 			    "PCI-Express bandwidth available for this card"
2177c58d34ddSKevin Bowling 			    " is not sufficient for optimal performance.\n");
2178c58d34ddSKevin Bowling 			device_printf(dev,
2179c58d34ddSKevin Bowling 			    "For optimal performance a x8 PCIE, or x4 PCIE"
2180c58d34ddSKevin Bowling 			    " Gen2 slot is required.\n");
21818eb6488eSEric Joyner 		}
21828eb6488eSEric Joyner 		if ((hw->device_id == IXGBE_DEV_ID_82599_SFP_SF_QP) &&
21838eb6488eSEric Joyner 		    ((hw->bus.width <= ixgbe_bus_width_pcie_x8) &&
21848eb6488eSEric Joyner 		    (hw->bus.speed < ixgbe_bus_speed_8000))) {
2185c58d34ddSKevin Bowling 			device_printf(dev,
2186c58d34ddSKevin Bowling 			    "PCI-Express bandwidth available for this card"
2187c58d34ddSKevin Bowling 			    " is not sufficient for optimal performance.\n");
2188c58d34ddSKevin Bowling 			device_printf(dev,
2189c58d34ddSKevin Bowling 			    "For optimal performance a x8 PCIE Gen3 slot is"
2190c58d34ddSKevin Bowling 			    " required.\n");
21918eb6488eSEric Joyner 		}
21928eb6488eSEric Joyner 	} else
2193c58d34ddSKevin Bowling 		device_printf(dev,
2194c58d34ddSKevin Bowling 		    "Unable to determine slot speed/width. The speed/width"
2195c58d34ddSKevin Bowling 		    " reported are that of the internal switch.\n");
21968eb6488eSEric Joyner 
21978eb6488eSEric Joyner 	return;
21988eb6488eSEric Joyner } /* ixgbe_get_slot_info */
21998eb6488eSEric Joyner 
22008eb6488eSEric Joyner /************************************************************************
2201c19c7afeSEric Joyner  * ixgbe_if_msix_intr_assign
2202c19c7afeSEric Joyner  *
2203c19c7afeSEric Joyner  *   Setup MSI-X Interrupt resources and handlers
22048eb6488eSEric Joyner  ************************************************************************/
2205c19c7afeSEric Joyner static int
ixgbe_if_msix_intr_assign(if_ctx_t ctx,int msix)2206c19c7afeSEric Joyner ixgbe_if_msix_intr_assign(if_ctx_t ctx, int msix)
22078eb6488eSEric Joyner {
2208b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
2209b1d5caf3SKevin Bowling 	struct ix_rx_queue *rx_que = sc->rx_queues;
2210c19c7afeSEric Joyner 	struct ix_tx_queue *tx_que;
2211c19c7afeSEric Joyner 	int error, rid, vector = 0;
2212c19c7afeSEric Joyner 	char buf[16];
22138eb6488eSEric Joyner 
2214c19c7afeSEric Joyner 	/* Admin Que is vector 0*/
2215c19c7afeSEric Joyner 	rid = vector + 1;
2216b1d5caf3SKevin Bowling 	for (int i = 0; i < sc->num_rx_queues; i++, vector++, rx_que++) {
2217c19c7afeSEric Joyner 		rid = vector + 1;
2218c19c7afeSEric Joyner 
2219c19c7afeSEric Joyner 		snprintf(buf, sizeof(buf), "rxq%d", i);
2220c19c7afeSEric Joyner 		error = iflib_irq_alloc_generic(ctx, &rx_que->que_irq, rid,
2221c58d34ddSKevin Bowling 		    IFLIB_INTR_RXTX, ixgbe_msix_que, rx_que, rx_que->rxr.me,
2222c58d34ddSKevin Bowling 		    buf);
2223c19c7afeSEric Joyner 
2224c19c7afeSEric Joyner 		if (error) {
2225c19c7afeSEric Joyner 			device_printf(iflib_get_dev(ctx),
2226c58d34ddSKevin Bowling 			    "Failed to allocate que int %d err: %d",
2227c58d34ddSKevin Bowling 			    i,error);
2228b1d5caf3SKevin Bowling 			sc->num_rx_queues = i + 1;
2229c19c7afeSEric Joyner 			goto fail;
22308eb6488eSEric Joyner 		}
22318eb6488eSEric Joyner 
2232c19c7afeSEric Joyner 		rx_que->msix = vector;
2233c19c7afeSEric Joyner 	}
2234b1d5caf3SKevin Bowling 	for (int i = 0; i < sc->num_tx_queues; i++) {
2235c19c7afeSEric Joyner 		snprintf(buf, sizeof(buf), "txq%d", i);
2236b1d5caf3SKevin Bowling 		tx_que = &sc->tx_queues[i];
2237b1d5caf3SKevin Bowling 		tx_que->msix = i % sc->num_rx_queues;
2238c19c7afeSEric Joyner 		iflib_softirq_alloc_generic(ctx,
2239b1d5caf3SKevin Bowling 		    &sc->rx_queues[tx_que->msix].que_irq,
2240c19c7afeSEric Joyner 		    IFLIB_INTR_TX, tx_que, tx_que->txr.me, buf);
2241c19c7afeSEric Joyner 	}
2242c19c7afeSEric Joyner 	rid = vector + 1;
2243b1d5caf3SKevin Bowling 	error = iflib_irq_alloc_generic(ctx, &sc->irq, rid,
2244b1d5caf3SKevin Bowling 	    IFLIB_INTR_ADMIN, ixgbe_msix_link, sc, 0, "aq");
2245c19c7afeSEric Joyner 	if (error) {
2246c19c7afeSEric Joyner 		device_printf(iflib_get_dev(ctx),
2247c19c7afeSEric Joyner 		    "Failed to register admin handler");
2248c19c7afeSEric Joyner 		return (error);
2249c19c7afeSEric Joyner 	}
2250c19c7afeSEric Joyner 
2251b1d5caf3SKevin Bowling 	sc->vector = vector;
2252c19c7afeSEric Joyner 
2253c19c7afeSEric Joyner 	return (0);
2254c19c7afeSEric Joyner fail:
2255b1d5caf3SKevin Bowling 	iflib_irq_free(ctx, &sc->irq);
2256b1d5caf3SKevin Bowling 	rx_que = sc->rx_queues;
2257b1d5caf3SKevin Bowling 	for (int i = 0; i < sc->num_rx_queues; i++, rx_que++)
2258c19c7afeSEric Joyner 		iflib_irq_free(ctx, &rx_que->que_irq);
2259c19c7afeSEric Joyner 
2260c19c7afeSEric Joyner 	return (error);
2261c19c7afeSEric Joyner } /* ixgbe_if_msix_intr_assign */
2262c19c7afeSEric Joyner 
226364881da4SSai Rajesh Tallamraju static inline void
ixgbe_perform_aim(struct ixgbe_softc * sc,struct ix_rx_queue * que)2264b1d5caf3SKevin Bowling ixgbe_perform_aim(struct ixgbe_softc *sc, struct ix_rx_queue *que)
226564881da4SSai Rajesh Tallamraju {
226664881da4SSai Rajesh Tallamraju 	uint32_t newitr = 0;
226764881da4SSai Rajesh Tallamraju 	struct rx_ring *rxr = &que->rxr;
2268f72de14eSKevin Bowling 	/* FIXME struct tx_ring *txr = ... ->txr; */
226964881da4SSai Rajesh Tallamraju 
227064881da4SSai Rajesh Tallamraju 	/*
227164881da4SSai Rajesh Tallamraju 	 * Do Adaptive Interrupt Moderation:
227264881da4SSai Rajesh Tallamraju 	 *  - Write out last calculated setting
227364881da4SSai Rajesh Tallamraju 	 *  - Calculate based on average size over
227464881da4SSai Rajesh Tallamraju 	 *    the last interval.
227564881da4SSai Rajesh Tallamraju 	 */
227664881da4SSai Rajesh Tallamraju 	if (que->eitr_setting) {
2277b1d5caf3SKevin Bowling 		IXGBE_WRITE_REG(&sc->hw, IXGBE_EITR(que->msix),
227864881da4SSai Rajesh Tallamraju 		    que->eitr_setting);
227964881da4SSai Rajesh Tallamraju 	}
228064881da4SSai Rajesh Tallamraju 
228164881da4SSai Rajesh Tallamraju 	que->eitr_setting = 0;
228264881da4SSai Rajesh Tallamraju 	/* Idle, do nothing */
228364881da4SSai Rajesh Tallamraju 	if (rxr->bytes == 0) {
2284f72de14eSKevin Bowling 		/* FIXME && txr->bytes == 0 */
228564881da4SSai Rajesh Tallamraju 		return;
228664881da4SSai Rajesh Tallamraju 	}
228764881da4SSai Rajesh Tallamraju 
2288f72de14eSKevin Bowling 	if ((rxr->bytes) && (rxr->packets))
2289f72de14eSKevin Bowling 		newitr = rxr->bytes / rxr->packets;
2290f72de14eSKevin Bowling 	/* FIXME for transmit accounting
2291f72de14eSKevin Bowling 	 * if ((txr->bytes) && (txr->packets))
2292f72de14eSKevin Bowling 	 * 	newitr = txr->bytes/txr->packets;
2293f72de14eSKevin Bowling 	 * if ((rxr->bytes) && (rxr->packets))
2294f72de14eSKevin Bowling 	 * 	newitr = max(newitr, (rxr->bytes / rxr->packets));
2295f72de14eSKevin Bowling 	 */
229664881da4SSai Rajesh Tallamraju 
229764881da4SSai Rajesh Tallamraju 	newitr += 24; /* account for hardware frame, crc */
229864881da4SSai Rajesh Tallamraju 	/* set an upper boundary */
229964881da4SSai Rajesh Tallamraju 	newitr = min(newitr, 3000);
230064881da4SSai Rajesh Tallamraju 
230164881da4SSai Rajesh Tallamraju 	/* Be nice to the mid range */
230264881da4SSai Rajesh Tallamraju 	if ((newitr > 300) && (newitr < 1200)) {
230364881da4SSai Rajesh Tallamraju 		newitr = (newitr / 3);
230464881da4SSai Rajesh Tallamraju 	} else {
230564881da4SSai Rajesh Tallamraju 		newitr = (newitr / 2);
230664881da4SSai Rajesh Tallamraju 	}
230764881da4SSai Rajesh Tallamraju 
2308b1d5caf3SKevin Bowling 	if (sc->hw.mac.type == ixgbe_mac_82598EB) {
230964881da4SSai Rajesh Tallamraju 		newitr |= newitr << 16;
231064881da4SSai Rajesh Tallamraju 	} else {
231164881da4SSai Rajesh Tallamraju 		newitr |= IXGBE_EITR_CNT_WDIS;
231264881da4SSai Rajesh Tallamraju 	}
231364881da4SSai Rajesh Tallamraju 
231464881da4SSai Rajesh Tallamraju 	/* save for next interrupt */
231564881da4SSai Rajesh Tallamraju 	que->eitr_setting = newitr;
231664881da4SSai Rajesh Tallamraju 
231764881da4SSai Rajesh Tallamraju 	/* Reset state */
2318f72de14eSKevin Bowling 	/* FIXME txr->bytes = 0; */
2319f72de14eSKevin Bowling 	/* FIXME txr->packets = 0; */
232064881da4SSai Rajesh Tallamraju 	rxr->bytes = 0;
232164881da4SSai Rajesh Tallamraju 	rxr->packets = 0;
232264881da4SSai Rajesh Tallamraju 
232364881da4SSai Rajesh Tallamraju 	return;
232464881da4SSai Rajesh Tallamraju }
232564881da4SSai Rajesh Tallamraju 
2326c19c7afeSEric Joyner /*********************************************************************
23278eb6488eSEric Joyner  * ixgbe_msix_que - MSI-X Queue Interrupt Service routine
2328c19c7afeSEric Joyner  **********************************************************************/
2329c19c7afeSEric Joyner static int
ixgbe_msix_que(void * arg)23308eb6488eSEric Joyner ixgbe_msix_que(void *arg)
23318eb6488eSEric Joyner {
2332c19c7afeSEric Joyner 	struct ix_rx_queue *que = arg;
2333b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = que->sc;
2334ff06a8dbSJustin Hibbits 	if_t ifp = iflib_get_ifp(que->sc->ctx);
23358eb6488eSEric Joyner 
23368eb6488eSEric Joyner 	/* Protect against spurious interrupts */
2337ff06a8dbSJustin Hibbits 	if ((if_getdrvflags(ifp) & IFF_DRV_RUNNING) == 0)
23386c351041SEric Joyner 		return (FILTER_HANDLED);
23398eb6488eSEric Joyner 
2340b1d5caf3SKevin Bowling 	ixgbe_disable_queue(sc, que->msix);
23418eb6488eSEric Joyner 	++que->irqs;
23428eb6488eSEric Joyner 
234364881da4SSai Rajesh Tallamraju 	/* Check for AIM */
2344b1d5caf3SKevin Bowling 	if (sc->enable_aim) {
2345b1d5caf3SKevin Bowling 		ixgbe_perform_aim(sc, que);
234664881da4SSai Rajesh Tallamraju 	}
234764881da4SSai Rajesh Tallamraju 
2348c19c7afeSEric Joyner 	return (FILTER_SCHEDULE_THREAD);
23498eb6488eSEric Joyner } /* ixgbe_msix_que */
23508eb6488eSEric Joyner 
23518eb6488eSEric Joyner /************************************************************************
23528eb6488eSEric Joyner  * ixgbe_media_status - Media Ioctl callback
23538eb6488eSEric Joyner  *
23548eb6488eSEric Joyner  *   Called whenever the user queries the status of
23558eb6488eSEric Joyner  *   the interface using ifconfig.
23568eb6488eSEric Joyner  ************************************************************************/
23578eb6488eSEric Joyner static void
ixgbe_if_media_status(if_ctx_t ctx,struct ifmediareq * ifmr)2358c19c7afeSEric Joyner ixgbe_if_media_status(if_ctx_t ctx, struct ifmediareq * ifmr)
23598eb6488eSEric Joyner {
2360b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
2361b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
23628eb6488eSEric Joyner 	int layer;
23638eb6488eSEric Joyner 
2364c19c7afeSEric Joyner 	INIT_DEBUGOUT("ixgbe_if_media_status: begin");
2365c19c7afeSEric Joyner 
23668eb6488eSEric Joyner 	ifmr->ifm_status = IFM_AVALID;
23678eb6488eSEric Joyner 	ifmr->ifm_active = IFM_ETHER;
23688eb6488eSEric Joyner 
2369b1d5caf3SKevin Bowling 	if (!sc->link_active)
23708eb6488eSEric Joyner 		return;
23718eb6488eSEric Joyner 
23728eb6488eSEric Joyner 	ifmr->ifm_status |= IFM_ACTIVE;
2373b1d5caf3SKevin Bowling 	layer = sc->phy_layer;
23748eb6488eSEric Joyner 
23758eb6488eSEric Joyner 	if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_T ||
23768eb6488eSEric Joyner 	    layer & IXGBE_PHYSICAL_LAYER_1000BASE_T ||
23778eb6488eSEric Joyner 	    layer & IXGBE_PHYSICAL_LAYER_100BASE_TX ||
23788eb6488eSEric Joyner 	    layer & IXGBE_PHYSICAL_LAYER_10BASE_T)
2379b1d5caf3SKevin Bowling 		switch (sc->link_speed) {
23808eb6488eSEric Joyner 		case IXGBE_LINK_SPEED_10GB_FULL:
23818eb6488eSEric Joyner 			ifmr->ifm_active |= IFM_10G_T | IFM_FDX;
23828eb6488eSEric Joyner 			break;
23838eb6488eSEric Joyner 		case IXGBE_LINK_SPEED_1GB_FULL:
23848eb6488eSEric Joyner 			ifmr->ifm_active |= IFM_1000_T | IFM_FDX;
23858eb6488eSEric Joyner 			break;
23868eb6488eSEric Joyner 		case IXGBE_LINK_SPEED_100_FULL:
23878eb6488eSEric Joyner 			ifmr->ifm_active |= IFM_100_TX | IFM_FDX;
23888eb6488eSEric Joyner 			break;
23898eb6488eSEric Joyner 		case IXGBE_LINK_SPEED_10_FULL:
23908eb6488eSEric Joyner 			ifmr->ifm_active |= IFM_10_T | IFM_FDX;
23918eb6488eSEric Joyner 			break;
23928eb6488eSEric Joyner 		}
2393d381c807SPiotr Pietruszewski 	if (hw->mac.type == ixgbe_mac_X550)
2394d381c807SPiotr Pietruszewski 		switch (sc->link_speed) {
2395d381c807SPiotr Pietruszewski 		case IXGBE_LINK_SPEED_5GB_FULL:
2396d381c807SPiotr Pietruszewski 			ifmr->ifm_active |= IFM_5000_T | IFM_FDX;
2397d381c807SPiotr Pietruszewski 			break;
2398d381c807SPiotr Pietruszewski 		case IXGBE_LINK_SPEED_2_5GB_FULL:
2399d381c807SPiotr Pietruszewski 			ifmr->ifm_active |= IFM_2500_T | IFM_FDX;
2400d381c807SPiotr Pietruszewski 			break;
2401d381c807SPiotr Pietruszewski 		}
24028eb6488eSEric Joyner 	if (layer & IXGBE_PHYSICAL_LAYER_SFP_PLUS_CU ||
24038eb6488eSEric Joyner 	    layer & IXGBE_PHYSICAL_LAYER_SFP_ACTIVE_DA)
2404b1d5caf3SKevin Bowling 		switch (sc->link_speed) {
24058eb6488eSEric Joyner 		case IXGBE_LINK_SPEED_10GB_FULL:
24068eb6488eSEric Joyner 			ifmr->ifm_active |= IFM_10G_TWINAX | IFM_FDX;
24078eb6488eSEric Joyner 			break;
240848ddd1b9SKevin Bowling 		case IXGBE_LINK_SPEED_1GB_FULL:
240948ddd1b9SKevin Bowling 			ifmr->ifm_active |= IFM_1000_KX | IFM_FDX;
241048ddd1b9SKevin Bowling 			break;
24118eb6488eSEric Joyner 		}
24128eb6488eSEric Joyner 	if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_LR)
2413b1d5caf3SKevin Bowling 		switch (sc->link_speed) {
24148eb6488eSEric Joyner 		case IXGBE_LINK_SPEED_10GB_FULL:
24158eb6488eSEric Joyner 			ifmr->ifm_active |= IFM_10G_LR | IFM_FDX;
24168eb6488eSEric Joyner 			break;
24178eb6488eSEric Joyner 		case IXGBE_LINK_SPEED_1GB_FULL:
24188eb6488eSEric Joyner 			ifmr->ifm_active |= IFM_1000_LX | IFM_FDX;
24198eb6488eSEric Joyner 			break;
24208eb6488eSEric Joyner 		}
24218eb6488eSEric Joyner 	if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_LRM)
2422b1d5caf3SKevin Bowling 		switch (sc->link_speed) {
24238eb6488eSEric Joyner 		case IXGBE_LINK_SPEED_10GB_FULL:
24248eb6488eSEric Joyner 			ifmr->ifm_active |= IFM_10G_LRM | IFM_FDX;
24258eb6488eSEric Joyner 			break;
24268eb6488eSEric Joyner 		case IXGBE_LINK_SPEED_1GB_FULL:
24278eb6488eSEric Joyner 			ifmr->ifm_active |= IFM_1000_LX | IFM_FDX;
24288eb6488eSEric Joyner 			break;
24298eb6488eSEric Joyner 		}
24308eb6488eSEric Joyner 	if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_SR ||
24318eb6488eSEric Joyner 	    layer & IXGBE_PHYSICAL_LAYER_1000BASE_SX)
2432b1d5caf3SKevin Bowling 		switch (sc->link_speed) {
24338eb6488eSEric Joyner 		case IXGBE_LINK_SPEED_10GB_FULL:
24348eb6488eSEric Joyner 			ifmr->ifm_active |= IFM_10G_SR | IFM_FDX;
24358eb6488eSEric Joyner 			break;
24368eb6488eSEric Joyner 		case IXGBE_LINK_SPEED_1GB_FULL:
24378eb6488eSEric Joyner 			ifmr->ifm_active |= IFM_1000_SX | IFM_FDX;
24388eb6488eSEric Joyner 			break;
24398eb6488eSEric Joyner 		}
24408eb6488eSEric Joyner 	if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_CX4)
2441b1d5caf3SKevin Bowling 		switch (sc->link_speed) {
24428eb6488eSEric Joyner 		case IXGBE_LINK_SPEED_10GB_FULL:
24438eb6488eSEric Joyner 			ifmr->ifm_active |= IFM_10G_CX4 | IFM_FDX;
24448eb6488eSEric Joyner 			break;
24458eb6488eSEric Joyner 		}
24468eb6488eSEric Joyner 	/*
24478eb6488eSEric Joyner 	 * XXX: These need to use the proper media types once
24488eb6488eSEric Joyner 	 * they're added.
24498eb6488eSEric Joyner 	 */
24508eb6488eSEric Joyner #ifndef IFM_ETH_XTYPE
24518eb6488eSEric Joyner 	if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_KR)
2452b1d5caf3SKevin Bowling 		switch (sc->link_speed) {
24538eb6488eSEric Joyner 		case IXGBE_LINK_SPEED_10GB_FULL:
24548eb6488eSEric Joyner 			ifmr->ifm_active |= IFM_10G_SR | IFM_FDX;
24558eb6488eSEric Joyner 			break;
24568eb6488eSEric Joyner 		case IXGBE_LINK_SPEED_2_5GB_FULL:
24578eb6488eSEric Joyner 			ifmr->ifm_active |= IFM_2500_SX | IFM_FDX;
24588eb6488eSEric Joyner 			break;
24598eb6488eSEric Joyner 		case IXGBE_LINK_SPEED_1GB_FULL:
24608eb6488eSEric Joyner 			ifmr->ifm_active |= IFM_1000_CX | IFM_FDX;
24618eb6488eSEric Joyner 			break;
24628eb6488eSEric Joyner 		}
24638eb6488eSEric Joyner 	else if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_KX4 ||
24648eb6488eSEric Joyner 	    layer & IXGBE_PHYSICAL_LAYER_2500BASE_KX ||
24658eb6488eSEric Joyner 	    layer & IXGBE_PHYSICAL_LAYER_1000BASE_KX)
2466b1d5caf3SKevin Bowling 		switch (sc->link_speed) {
24678eb6488eSEric Joyner 		case IXGBE_LINK_SPEED_10GB_FULL:
24688eb6488eSEric Joyner 			ifmr->ifm_active |= IFM_10G_CX4 | IFM_FDX;
24698eb6488eSEric Joyner 			break;
24708eb6488eSEric Joyner 		case IXGBE_LINK_SPEED_2_5GB_FULL:
24718eb6488eSEric Joyner 			ifmr->ifm_active |= IFM_2500_SX | IFM_FDX;
24728eb6488eSEric Joyner 			break;
24738eb6488eSEric Joyner 		case IXGBE_LINK_SPEED_1GB_FULL:
24748eb6488eSEric Joyner 			ifmr->ifm_active |= IFM_1000_CX | IFM_FDX;
24758eb6488eSEric Joyner 			break;
24768eb6488eSEric Joyner 		}
24778eb6488eSEric Joyner #else
24788eb6488eSEric Joyner 	if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_KR)
2479b1d5caf3SKevin Bowling 		switch (sc->link_speed) {
24808eb6488eSEric Joyner 		case IXGBE_LINK_SPEED_10GB_FULL:
24818eb6488eSEric Joyner 			ifmr->ifm_active |= IFM_10G_KR | IFM_FDX;
24828eb6488eSEric Joyner 			break;
24838eb6488eSEric Joyner 		case IXGBE_LINK_SPEED_2_5GB_FULL:
24848eb6488eSEric Joyner 			ifmr->ifm_active |= IFM_2500_KX | IFM_FDX;
24858eb6488eSEric Joyner 			break;
24868eb6488eSEric Joyner 		case IXGBE_LINK_SPEED_1GB_FULL:
24878eb6488eSEric Joyner 			ifmr->ifm_active |= IFM_1000_KX | IFM_FDX;
24888eb6488eSEric Joyner 			break;
24898eb6488eSEric Joyner 		}
24908eb6488eSEric Joyner 	else if (layer & IXGBE_PHYSICAL_LAYER_10GBASE_KX4 ||
24918eb6488eSEric Joyner 	    layer & IXGBE_PHYSICAL_LAYER_2500BASE_KX ||
24928eb6488eSEric Joyner 	    layer & IXGBE_PHYSICAL_LAYER_1000BASE_KX)
2493b1d5caf3SKevin Bowling 		switch (sc->link_speed) {
24948eb6488eSEric Joyner 		case IXGBE_LINK_SPEED_10GB_FULL:
24958eb6488eSEric Joyner 			ifmr->ifm_active |= IFM_10G_KX4 | IFM_FDX;
24968eb6488eSEric Joyner 			break;
24978eb6488eSEric Joyner 		case IXGBE_LINK_SPEED_2_5GB_FULL:
24988eb6488eSEric Joyner 			ifmr->ifm_active |= IFM_2500_KX | IFM_FDX;
24998eb6488eSEric Joyner 			break;
25008eb6488eSEric Joyner 		case IXGBE_LINK_SPEED_1GB_FULL:
25018eb6488eSEric Joyner 			ifmr->ifm_active |= IFM_1000_KX | IFM_FDX;
25028eb6488eSEric Joyner 			break;
25038eb6488eSEric Joyner 		}
25048eb6488eSEric Joyner #endif
25058eb6488eSEric Joyner 
25068eb6488eSEric Joyner 	/* If nothing is recognized... */
25078eb6488eSEric Joyner 	if (IFM_SUBTYPE(ifmr->ifm_active) == 0)
25088eb6488eSEric Joyner 		ifmr->ifm_active |= IFM_UNKNOWN;
25098eb6488eSEric Joyner 
25108eb6488eSEric Joyner 	/* Display current flow control setting used on link */
25118eb6488eSEric Joyner 	if (hw->fc.current_mode == ixgbe_fc_rx_pause ||
25128eb6488eSEric Joyner 	    hw->fc.current_mode == ixgbe_fc_full)
25138eb6488eSEric Joyner 		ifmr->ifm_active |= IFM_ETH_RXPAUSE;
25148eb6488eSEric Joyner 	if (hw->fc.current_mode == ixgbe_fc_tx_pause ||
25158eb6488eSEric Joyner 	    hw->fc.current_mode == ixgbe_fc_full)
25168eb6488eSEric Joyner 		ifmr->ifm_active |= IFM_ETH_TXPAUSE;
25178eb6488eSEric Joyner } /* ixgbe_media_status */
25188eb6488eSEric Joyner 
25198eb6488eSEric Joyner /************************************************************************
25208eb6488eSEric Joyner  * ixgbe_media_change - Media Ioctl callback
25218eb6488eSEric Joyner  *
25228eb6488eSEric Joyner  *   Called when the user changes speed/duplex using
25238eb6488eSEric Joyner  *   media/mediopt option with ifconfig.
25248eb6488eSEric Joyner  ************************************************************************/
25258eb6488eSEric Joyner static int
ixgbe_if_media_change(if_ctx_t ctx)2526c19c7afeSEric Joyner ixgbe_if_media_change(if_ctx_t ctx)
25278eb6488eSEric Joyner {
2528b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
2529c19c7afeSEric Joyner 	struct ifmedia *ifm = iflib_get_media(ctx);
2530b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
25318eb6488eSEric Joyner 	ixgbe_link_speed speed = 0;
25328eb6488eSEric Joyner 
2533c19c7afeSEric Joyner 	INIT_DEBUGOUT("ixgbe_if_media_change: begin");
25348eb6488eSEric Joyner 
25358eb6488eSEric Joyner 	if (IFM_TYPE(ifm->ifm_media) != IFM_ETHER)
25368eb6488eSEric Joyner 		return (EINVAL);
25378eb6488eSEric Joyner 
25388eb6488eSEric Joyner 	if (hw->phy.media_type == ixgbe_media_type_backplane)
2539c19c7afeSEric Joyner 		return (EPERM);
25408eb6488eSEric Joyner 
25418eb6488eSEric Joyner 	/*
25428eb6488eSEric Joyner 	 * We don't actually need to check against the supported
25438eb6488eSEric Joyner 	 * media types of the adapter; ifmedia will take care of
25448eb6488eSEric Joyner 	 * that for us.
25458eb6488eSEric Joyner 	 */
25468eb6488eSEric Joyner 	switch (IFM_SUBTYPE(ifm->ifm_media)) {
25478eb6488eSEric Joyner 	case IFM_AUTO:
25488eb6488eSEric Joyner 	case IFM_10G_T:
25498eb6488eSEric Joyner 		speed |= IXGBE_LINK_SPEED_100_FULL;
25508eb6488eSEric Joyner 		speed |= IXGBE_LINK_SPEED_1GB_FULL;
25518eb6488eSEric Joyner 		speed |= IXGBE_LINK_SPEED_10GB_FULL;
25528eb6488eSEric Joyner 		break;
25538eb6488eSEric Joyner 	case IFM_10G_LRM:
25548eb6488eSEric Joyner 	case IFM_10G_LR:
25558eb6488eSEric Joyner #ifndef IFM_ETH_XTYPE
25568eb6488eSEric Joyner 	case IFM_10G_SR: /* KR, too */
25578eb6488eSEric Joyner 	case IFM_10G_CX4: /* KX4 */
25588eb6488eSEric Joyner #else
25598eb6488eSEric Joyner 	case IFM_10G_KR:
25608eb6488eSEric Joyner 	case IFM_10G_KX4:
25618eb6488eSEric Joyner #endif
25628eb6488eSEric Joyner 		speed |= IXGBE_LINK_SPEED_1GB_FULL;
25638eb6488eSEric Joyner 		speed |= IXGBE_LINK_SPEED_10GB_FULL;
25648eb6488eSEric Joyner 		break;
25658eb6488eSEric Joyner #ifndef IFM_ETH_XTYPE
25668eb6488eSEric Joyner 	case IFM_1000_CX: /* KX */
25678eb6488eSEric Joyner #else
25688eb6488eSEric Joyner 	case IFM_1000_KX:
25698eb6488eSEric Joyner #endif
25708eb6488eSEric Joyner 	case IFM_1000_LX:
25718eb6488eSEric Joyner 	case IFM_1000_SX:
2572*89d40969STore Amundsen 	case IFM_1000_BX:
25738eb6488eSEric Joyner 		speed |= IXGBE_LINK_SPEED_1GB_FULL;
25748eb6488eSEric Joyner 		break;
25758eb6488eSEric Joyner 	case IFM_1000_T:
25768eb6488eSEric Joyner 		speed |= IXGBE_LINK_SPEED_100_FULL;
25778eb6488eSEric Joyner 		speed |= IXGBE_LINK_SPEED_1GB_FULL;
25788eb6488eSEric Joyner 		break;
25798eb6488eSEric Joyner 	case IFM_10G_TWINAX:
25808eb6488eSEric Joyner 		speed |= IXGBE_LINK_SPEED_10GB_FULL;
25818eb6488eSEric Joyner 		break;
2582d381c807SPiotr Pietruszewski 	case IFM_5000_T:
2583d381c807SPiotr Pietruszewski 		speed |= IXGBE_LINK_SPEED_5GB_FULL;
2584d381c807SPiotr Pietruszewski 		break;
2585d381c807SPiotr Pietruszewski 	case IFM_2500_T:
2586d381c807SPiotr Pietruszewski 		speed |= IXGBE_LINK_SPEED_2_5GB_FULL;
2587d381c807SPiotr Pietruszewski 		break;
25888eb6488eSEric Joyner 	case IFM_100_TX:
25898eb6488eSEric Joyner 		speed |= IXGBE_LINK_SPEED_100_FULL;
25908eb6488eSEric Joyner 		break;
25918eb6488eSEric Joyner 	case IFM_10_T:
25928eb6488eSEric Joyner 		speed |= IXGBE_LINK_SPEED_10_FULL;
25938eb6488eSEric Joyner 		break;
25948eb6488eSEric Joyner 	default:
25958eb6488eSEric Joyner 		goto invalid;
25968eb6488eSEric Joyner 	}
25978eb6488eSEric Joyner 
259879b36ec9SKevin Bowling 	hw->mac.autotry_restart = true;
259979b36ec9SKevin Bowling 	hw->mac.ops.setup_link(hw, speed, true);
2600b1d5caf3SKevin Bowling 	sc->advertise =
2601d381c807SPiotr Pietruszewski 	    ((speed & IXGBE_LINK_SPEED_10GB_FULL)  ? 0x4  : 0) |
2602d381c807SPiotr Pietruszewski 	    ((speed & IXGBE_LINK_SPEED_5GB_FULL)   ? 0x20 : 0) |
2603d381c807SPiotr Pietruszewski 	    ((speed & IXGBE_LINK_SPEED_2_5GB_FULL) ? 0x10 : 0) |
2604d381c807SPiotr Pietruszewski 	    ((speed & IXGBE_LINK_SPEED_1GB_FULL)   ? 0x2  : 0) |
2605d381c807SPiotr Pietruszewski 	    ((speed & IXGBE_LINK_SPEED_100_FULL)   ? 0x1  : 0) |
2606d381c807SPiotr Pietruszewski 	    ((speed & IXGBE_LINK_SPEED_10_FULL)    ? 0x8  : 0);
26078eb6488eSEric Joyner 
26088eb6488eSEric Joyner 	return (0);
26098eb6488eSEric Joyner 
26108eb6488eSEric Joyner invalid:
2611c19c7afeSEric Joyner 	device_printf(iflib_get_dev(ctx), "Invalid media type!\n");
26128eb6488eSEric Joyner 
26138eb6488eSEric Joyner 	return (EINVAL);
2614c19c7afeSEric Joyner } /* ixgbe_if_media_change */
26158eb6488eSEric Joyner 
26168eb6488eSEric Joyner /************************************************************************
26178eb6488eSEric Joyner  * ixgbe_set_promisc
26188eb6488eSEric Joyner  ************************************************************************/
2619c19c7afeSEric Joyner static int
ixgbe_if_promisc_set(if_ctx_t ctx,int flags)2620c19c7afeSEric Joyner ixgbe_if_promisc_set(if_ctx_t ctx, int flags)
26218eb6488eSEric Joyner {
2622b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
2623ff06a8dbSJustin Hibbits 	if_t ifp = iflib_get_ifp(ctx);
26248eb6488eSEric Joyner 	u32 rctl;
2625c19c7afeSEric Joyner 	int mcnt = 0;
26268eb6488eSEric Joyner 
2627b1d5caf3SKevin Bowling 	rctl = IXGBE_READ_REG(&sc->hw, IXGBE_FCTRL);
26288eb6488eSEric Joyner 	rctl &= (~IXGBE_FCTRL_UPE);
2629ff06a8dbSJustin Hibbits 	if (if_getflags(ifp) & IFF_ALLMULTI)
26308eb6488eSEric Joyner 		mcnt = MAX_NUM_MULTICAST_ADDRESSES;
26318eb6488eSEric Joyner 	else {
2632c58d34ddSKevin Bowling 		mcnt = min(if_llmaddr_count(ifp),
2633c58d34ddSKevin Bowling 		    MAX_NUM_MULTICAST_ADDRESSES);
26348eb6488eSEric Joyner 	}
26358eb6488eSEric Joyner 	if (mcnt < MAX_NUM_MULTICAST_ADDRESSES)
26368eb6488eSEric Joyner 		rctl &= (~IXGBE_FCTRL_MPE);
2637b1d5caf3SKevin Bowling 	IXGBE_WRITE_REG(&sc->hw, IXGBE_FCTRL, rctl);
26388eb6488eSEric Joyner 
2639ff06a8dbSJustin Hibbits 	if (if_getflags(ifp) & IFF_PROMISC) {
26408eb6488eSEric Joyner 		rctl |= (IXGBE_FCTRL_UPE | IXGBE_FCTRL_MPE);
2641b1d5caf3SKevin Bowling 		IXGBE_WRITE_REG(&sc->hw, IXGBE_FCTRL, rctl);
2642ff06a8dbSJustin Hibbits 	} else if (if_getflags(ifp) & IFF_ALLMULTI) {
26438eb6488eSEric Joyner 		rctl |= IXGBE_FCTRL_MPE;
26448eb6488eSEric Joyner 		rctl &= ~IXGBE_FCTRL_UPE;
2645b1d5caf3SKevin Bowling 		IXGBE_WRITE_REG(&sc->hw, IXGBE_FCTRL, rctl);
26468eb6488eSEric Joyner 	}
2647c19c7afeSEric Joyner 	return (0);
2648c19c7afeSEric Joyner } /* ixgbe_if_promisc_set */
26498eb6488eSEric Joyner 
26508eb6488eSEric Joyner /************************************************************************
26518eb6488eSEric Joyner  * ixgbe_msix_link - Link status change ISR (MSI/MSI-X)
26528eb6488eSEric Joyner  ************************************************************************/
2653c19c7afeSEric Joyner static int
ixgbe_msix_link(void * arg)26548eb6488eSEric Joyner ixgbe_msix_link(void *arg)
26558eb6488eSEric Joyner {
2656b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = arg;
2657b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
26588eb6488eSEric Joyner 	u32 eicr, eicr_mask;
26598eb6488eSEric Joyner 	s32 retval;
26608eb6488eSEric Joyner 
2661b1d5caf3SKevin Bowling 	++sc->link_irq;
26628eb6488eSEric Joyner 
26638eb6488eSEric Joyner 	/* Pause other interrupts */
26648eb6488eSEric Joyner 	IXGBE_WRITE_REG(hw, IXGBE_EIMC, IXGBE_EIMC_OTHER);
26658eb6488eSEric Joyner 
26668eb6488eSEric Joyner 	/* First get the cause */
26678eb6488eSEric Joyner 	eicr = IXGBE_READ_REG(hw, IXGBE_EICS);
26688eb6488eSEric Joyner 	/* Be sure the queue bits are not cleared */
26698eb6488eSEric Joyner 	eicr &= ~IXGBE_EICR_RTX_QUEUE;
26708eb6488eSEric Joyner 	/* Clear interrupt with write */
26718eb6488eSEric Joyner 	IXGBE_WRITE_REG(hw, IXGBE_EICR, eicr);
26728eb6488eSEric Joyner 
26738eb6488eSEric Joyner 	/* Link status change */
26748eb6488eSEric Joyner 	if (eicr & IXGBE_EICR_LSC) {
26758eb6488eSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_EIMC, IXGBE_EIMC_LSC);
2676b1d5caf3SKevin Bowling 		sc->task_requests |= IXGBE_REQUEST_TASK_LSC;
26778eb6488eSEric Joyner 	}
26788eb6488eSEric Joyner 
2679b1d5caf3SKevin Bowling 	if (sc->hw.mac.type != ixgbe_mac_82598EB) {
2680b1d5caf3SKevin Bowling 		if ((sc->feat_en & IXGBE_FEATURE_FDIR) &&
26818eb6488eSEric Joyner 		    (eicr & IXGBE_EICR_FLOW_DIR)) {
26828eb6488eSEric Joyner 			/* This is probably overkill :) */
2683b1d5caf3SKevin Bowling 			if (!atomic_cmpset_int(&sc->fdir_reinit, 0, 1))
2684c19c7afeSEric Joyner 				return (FILTER_HANDLED);
26858eb6488eSEric Joyner 			/* Disable the interrupt */
2686c19c7afeSEric Joyner 			IXGBE_WRITE_REG(hw, IXGBE_EIMC, IXGBE_EICR_FLOW_DIR);
2687b1d5caf3SKevin Bowling 			sc->task_requests |= IXGBE_REQUEST_TASK_FDIR;
2688c19c7afeSEric Joyner 		} else
26898eb6488eSEric Joyner 			if (eicr & IXGBE_EICR_ECC) {
2690b1d5caf3SKevin Bowling 				device_printf(iflib_get_dev(sc->ctx),
26914f1d91e4SNeel Chauhan 				    "Received ECC Err, initiating reset\n");
2692c58d34ddSKevin Bowling 				hw->mac.flags |=
2693c58d34ddSKevin Bowling 				    ~IXGBE_FLAGS_DOUBLE_RESET_REQUIRED;
26944f1d91e4SNeel Chauhan 				ixgbe_reset_hw(hw);
2695c58d34ddSKevin Bowling 				IXGBE_WRITE_REG(hw, IXGBE_EICR,
2696c58d34ddSKevin Bowling 				    IXGBE_EICR_ECC);
26978eb6488eSEric Joyner 			}
26988eb6488eSEric Joyner 
26998eb6488eSEric Joyner 		/* Check for over temp condition */
2700b1d5caf3SKevin Bowling 		if (sc->feat_en & IXGBE_FEATURE_TEMP_SENSOR) {
2701b1d5caf3SKevin Bowling 			switch (sc->hw.mac.type) {
27028eb6488eSEric Joyner 			case ixgbe_mac_X550EM_a:
27038eb6488eSEric Joyner 				if (!(eicr & IXGBE_EICR_GPI_SDP0_X550EM_a))
27048eb6488eSEric Joyner 					break;
27058eb6488eSEric Joyner 				IXGBE_WRITE_REG(hw, IXGBE_EIMC,
27068eb6488eSEric Joyner 				    IXGBE_EICR_GPI_SDP0_X550EM_a);
27078eb6488eSEric Joyner 				IXGBE_WRITE_REG(hw, IXGBE_EICR,
27088eb6488eSEric Joyner 				    IXGBE_EICR_GPI_SDP0_X550EM_a);
27098eb6488eSEric Joyner 				retval = hw->phy.ops.check_overtemp(hw);
27108eb6488eSEric Joyner 				if (retval != IXGBE_ERR_OVERTEMP)
27118eb6488eSEric Joyner 					break;
2712b1d5caf3SKevin Bowling 				device_printf(iflib_get_dev(sc->ctx),
2713c58d34ddSKevin Bowling 				    "\nCRITICAL: OVER TEMP!!"
2714c58d34ddSKevin Bowling 				    " PHY IS SHUT DOWN!!\n");
2715b1d5caf3SKevin Bowling 				device_printf(iflib_get_dev(sc->ctx),
2716c19c7afeSEric Joyner 				    "System shutdown required!\n");
27178eb6488eSEric Joyner 				break;
27188eb6488eSEric Joyner 			default:
27198eb6488eSEric Joyner 				if (!(eicr & IXGBE_EICR_TS))
27208eb6488eSEric Joyner 					break;
27218eb6488eSEric Joyner 				retval = hw->phy.ops.check_overtemp(hw);
27228eb6488eSEric Joyner 				if (retval != IXGBE_ERR_OVERTEMP)
27238eb6488eSEric Joyner 					break;
2724b1d5caf3SKevin Bowling 				device_printf(iflib_get_dev(sc->ctx),
2725c58d34ddSKevin Bowling 				    "\nCRITICAL: OVER TEMP!!"
2726c58d34ddSKevin Bowling 				    " PHY IS SHUT DOWN!!\n");
2727b1d5caf3SKevin Bowling 				device_printf(iflib_get_dev(sc->ctx),
2728c19c7afeSEric Joyner 				    "System shutdown required!\n");
2729c58d34ddSKevin Bowling 				IXGBE_WRITE_REG(hw, IXGBE_EICR,
2730c58d34ddSKevin Bowling 				    IXGBE_EICR_TS);
27318eb6488eSEric Joyner 				break;
27328eb6488eSEric Joyner 			}
27338eb6488eSEric Joyner 		}
27348eb6488eSEric Joyner 
27358eb6488eSEric Joyner 		/* Check for VF message */
2736b1d5caf3SKevin Bowling 		if ((sc->feat_en & IXGBE_FEATURE_SRIOV) &&
27378eb6488eSEric Joyner 		    (eicr & IXGBE_EICR_MAILBOX))
2738b1d5caf3SKevin Bowling 			sc->task_requests |= IXGBE_REQUEST_TASK_MBX;
27398eb6488eSEric Joyner 	}
27408eb6488eSEric Joyner 
27418eb6488eSEric Joyner 	if (ixgbe_is_sfp(hw)) {
27428eb6488eSEric Joyner 		/* Pluggable optics-related interrupt */
27438eb6488eSEric Joyner 		if (hw->mac.type >= ixgbe_mac_X540)
27448eb6488eSEric Joyner 			eicr_mask = IXGBE_EICR_GPI_SDP0_X540;
27458eb6488eSEric Joyner 		else
27468eb6488eSEric Joyner 			eicr_mask = IXGBE_EICR_GPI_SDP2_BY_MAC(hw);
27478eb6488eSEric Joyner 
27488eb6488eSEric Joyner 		if (eicr & eicr_mask) {
27498eb6488eSEric Joyner 			IXGBE_WRITE_REG(hw, IXGBE_EICR, eicr_mask);
2750b1d5caf3SKevin Bowling 			sc->task_requests |= IXGBE_REQUEST_TASK_MOD;
27518eb6488eSEric Joyner 		}
27528eb6488eSEric Joyner 
27538eb6488eSEric Joyner 		if ((hw->mac.type == ixgbe_mac_82599EB) &&
27548eb6488eSEric Joyner 		    (eicr & IXGBE_EICR_GPI_SDP1_BY_MAC(hw))) {
27558eb6488eSEric Joyner 			IXGBE_WRITE_REG(hw, IXGBE_EICR,
27568eb6488eSEric Joyner 			    IXGBE_EICR_GPI_SDP1_BY_MAC(hw));
2757b1d5caf3SKevin Bowling 			sc->task_requests |= IXGBE_REQUEST_TASK_MSF;
27588eb6488eSEric Joyner 		}
27598eb6488eSEric Joyner 	}
27608eb6488eSEric Joyner 
27618eb6488eSEric Joyner 	/* Check for fan failure */
2762b1d5caf3SKevin Bowling 	if (sc->feat_en & IXGBE_FEATURE_FAN_FAIL) {
2763b1d5caf3SKevin Bowling 		ixgbe_check_fan_failure(sc, eicr, true);
2764c58d34ddSKevin Bowling 		IXGBE_WRITE_REG(hw, IXGBE_EICR,
2765c58d34ddSKevin Bowling 		    IXGBE_EICR_GPI_SDP1_BY_MAC(hw));
27668eb6488eSEric Joyner 	}
27678eb6488eSEric Joyner 
27688eb6488eSEric Joyner 	/* External PHY interrupt */
27698eb6488eSEric Joyner 	if ((hw->phy.type == ixgbe_phy_x550em_ext_t) &&
27708eb6488eSEric Joyner 	    (eicr & IXGBE_EICR_GPI_SDP0_X540)) {
27718eb6488eSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_EICR, IXGBE_EICR_GPI_SDP0_X540);
2772b1d5caf3SKevin Bowling 		sc->task_requests |= IXGBE_REQUEST_TASK_PHY;
27738eb6488eSEric Joyner 	}
27748eb6488eSEric Joyner 
2775c58d34ddSKevin Bowling 	return (sc->task_requests != 0) ?
2776c58d34ddSKevin Bowling 	    FILTER_SCHEDULE_THREAD : FILTER_HANDLED;
27778eb6488eSEric Joyner } /* ixgbe_msix_link */
27788eb6488eSEric Joyner 
27798eb6488eSEric Joyner /************************************************************************
27808eb6488eSEric Joyner  * ixgbe_sysctl_interrupt_rate_handler
27818eb6488eSEric Joyner  ************************************************************************/
2782758cc3dcSJack F Vogel static int
ixgbe_sysctl_interrupt_rate_handler(SYSCTL_HANDLER_ARGS)2783758cc3dcSJack F Vogel ixgbe_sysctl_interrupt_rate_handler(SYSCTL_HANDLER_ARGS)
2784758cc3dcSJack F Vogel {
2785c19c7afeSEric Joyner 	struct ix_rx_queue *que = ((struct ix_rx_queue *)oidp->oid_arg1);
27868eb6488eSEric Joyner 	int error;
2787758cc3dcSJack F Vogel 	unsigned int reg, usec, rate;
2788758cc3dcSJack F Vogel 
2789f72de14eSKevin Bowling 	if (atomic_load_acq_int(&que->sc->recovery_mode))
2790f72de14eSKevin Bowling 		return (EPERM);
2791f72de14eSKevin Bowling 
2792b1d5caf3SKevin Bowling 	reg = IXGBE_READ_REG(&que->sc->hw, IXGBE_EITR(que->msix));
2793758cc3dcSJack F Vogel 	usec = ((reg & 0x0FF8) >> 3);
2794758cc3dcSJack F Vogel 	if (usec > 0)
2795758cc3dcSJack F Vogel 		rate = 500000 / usec;
2796758cc3dcSJack F Vogel 	else
2797758cc3dcSJack F Vogel 		rate = 0;
2798758cc3dcSJack F Vogel 	error = sysctl_handle_int(oidp, &rate, 0, req);
2799758cc3dcSJack F Vogel 	if (error || !req->newptr)
2800758cc3dcSJack F Vogel 		return error;
2801758cc3dcSJack F Vogel 	reg &= ~0xfff; /* default, no limitation */
2802758cc3dcSJack F Vogel 	ixgbe_max_interrupt_rate = 0;
2803758cc3dcSJack F Vogel 	if (rate > 0 && rate < 500000) {
2804758cc3dcSJack F Vogel 		if (rate < 1000)
2805758cc3dcSJack F Vogel 			rate = 1000;
2806758cc3dcSJack F Vogel 		ixgbe_max_interrupt_rate = rate;
2807758cc3dcSJack F Vogel 		reg |= ((4000000/rate) & 0xff8);
2808758cc3dcSJack F Vogel 	}
2809b1d5caf3SKevin Bowling 	IXGBE_WRITE_REG(&que->sc->hw, IXGBE_EITR(que->msix), reg);
2810758cc3dcSJack F Vogel 
28118eb6488eSEric Joyner 	return (0);
28128eb6488eSEric Joyner } /* ixgbe_sysctl_interrupt_rate_handler */
28138eb6488eSEric Joyner 
28148eb6488eSEric Joyner /************************************************************************
28158eb6488eSEric Joyner  * ixgbe_add_device_sysctls
28168eb6488eSEric Joyner  ************************************************************************/
28176f37f232SEric Joyner static void
ixgbe_add_device_sysctls(if_ctx_t ctx)2818c19c7afeSEric Joyner ixgbe_add_device_sysctls(if_ctx_t ctx)
28196f37f232SEric Joyner {
2820b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
2821c19c7afeSEric Joyner 	device_t dev = iflib_get_dev(ctx);
2822b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
28236f37f232SEric Joyner 	struct sysctl_oid_list *child;
2824c19c7afeSEric Joyner 	struct sysctl_ctx_list *ctx_list;
28256f37f232SEric Joyner 
2826c19c7afeSEric Joyner 	ctx_list = device_get_sysctl_ctx(dev);
28276f37f232SEric Joyner 	child = SYSCTL_CHILDREN(device_get_sysctl_tree(dev));
28286f37f232SEric Joyner 
28296f37f232SEric Joyner 	/* Sysctls for all devices */
2830c19c7afeSEric Joyner 	SYSCTL_ADD_PROC(ctx_list, child, OID_AUTO, "fc",
2831f72de14eSKevin Bowling 	    CTLTYPE_INT | CTLFLAG_RW,
2832b1d5caf3SKevin Bowling 	    sc, 0, ixgbe_sysctl_flowcntl, "I",
2833c19c7afeSEric Joyner 	    IXGBE_SYSCTL_DESC_SET_FC);
28346f37f232SEric Joyner 
2835c19c7afeSEric Joyner 	SYSCTL_ADD_PROC(ctx_list, child, OID_AUTO, "advertise_speed",
2836f72de14eSKevin Bowling 	    CTLTYPE_INT | CTLFLAG_RW,
2837b1d5caf3SKevin Bowling 	    sc, 0, ixgbe_sysctl_advertise, "I",
28388eb6488eSEric Joyner 	    IXGBE_SYSCTL_DESC_ADV_SPEED);
28396f37f232SEric Joyner 
2840b1d5caf3SKevin Bowling 	sc->enable_aim = ixgbe_enable_aim;
284164881da4SSai Rajesh Tallamraju 	SYSCTL_ADD_INT(ctx_list, child, OID_AUTO, "enable_aim", CTLFLAG_RW,
2842b1d5caf3SKevin Bowling 	    &sc->enable_aim, 0, "Interrupt Moderation");
284364881da4SSai Rajesh Tallamraju 
28447660e4eaSKevin Bowling 	SYSCTL_ADD_PROC(ctx_list, child, OID_AUTO, "fw_version",
2845f72de14eSKevin Bowling 	    CTLTYPE_STRING | CTLFLAG_RD, sc, 0,
28467660e4eaSKevin Bowling 	    ixgbe_sysctl_print_fw_version, "A", "Prints FW/NVM Versions");
28477660e4eaSKevin Bowling 
2848eea2e089SMichael Tuexen 	SYSCTL_ADD_PROC(ctx_list, child, OID_AUTO,
2849eea2e089SMichael Tuexen 	    "tso_tcp_flags_mask_first_segment",
2850eea2e089SMichael Tuexen 	    CTLTYPE_UINT | CTLFLAG_RW | CTLFLAG_NEEDGIANT,
2851eea2e089SMichael Tuexen 	    sc, 0, ixgbe_sysctl_tso_tcp_flags_mask, "IU",
2852eea2e089SMichael Tuexen 	    "TSO TCP flags mask for first segment");
2853eea2e089SMichael Tuexen 
2854eea2e089SMichael Tuexen 	SYSCTL_ADD_PROC(ctx_list, child, OID_AUTO,
2855eea2e089SMichael Tuexen 	    "tso_tcp_flags_mask_middle_segment",
2856eea2e089SMichael Tuexen 	    CTLTYPE_UINT | CTLFLAG_RW | CTLFLAG_NEEDGIANT,
2857eea2e089SMichael Tuexen 	    sc, 1, ixgbe_sysctl_tso_tcp_flags_mask, "IU",
2858eea2e089SMichael Tuexen 	    "TSO TCP flags mask for middle segment");
2859eea2e089SMichael Tuexen 
2860eea2e089SMichael Tuexen 	SYSCTL_ADD_PROC(ctx_list, child, OID_AUTO,
2861eea2e089SMichael Tuexen 	    "tso_tcp_flags_mask_last_segment",
2862eea2e089SMichael Tuexen 	    CTLTYPE_UINT | CTLFLAG_RW | CTLFLAG_NEEDGIANT,
2863eea2e089SMichael Tuexen 	    sc, 2, ixgbe_sysctl_tso_tcp_flags_mask, "IU",
2864eea2e089SMichael Tuexen 	    "TSO TCP flags mask for last segment");
2865eea2e089SMichael Tuexen 
2866a9ca1c79SSean Bruno #ifdef IXGBE_DEBUG
2867a9ca1c79SSean Bruno 	/* testing sysctls (for all devices) */
2868c19c7afeSEric Joyner 	SYSCTL_ADD_PROC(ctx_list, child, OID_AUTO, "power_state",
2869f72de14eSKevin Bowling 	    CTLTYPE_INT | CTLFLAG_RW,
2870b1d5caf3SKevin Bowling 	    sc, 0, ixgbe_sysctl_power_state,
28718eb6488eSEric Joyner 	    "I", "PCI Power State");
2872a9ca1c79SSean Bruno 
2873c19c7afeSEric Joyner 	SYSCTL_ADD_PROC(ctx_list, child, OID_AUTO, "print_rss_config",
2874f72de14eSKevin Bowling 	    CTLTYPE_STRING | CTLFLAG_RD, sc, 0,
2875a9ca1c79SSean Bruno 	    ixgbe_sysctl_print_rss_config, "A", "Prints RSS Configuration");
2876a9ca1c79SSean Bruno #endif
2877a9ca1c79SSean Bruno 	/* for X550 series devices */
28786f37f232SEric Joyner 	if (hw->mac.type >= ixgbe_mac_X550)
2879c19c7afeSEric Joyner 		SYSCTL_ADD_PROC(ctx_list, child, OID_AUTO, "dmac",
2880f72de14eSKevin Bowling 		    CTLTYPE_U16 | CTLFLAG_RW,
2881b1d5caf3SKevin Bowling 		    sc, 0, ixgbe_sysctl_dmac,
28828eb6488eSEric Joyner 		    "I", "DMA Coalesce");
28836f37f232SEric Joyner 
2884a9ca1c79SSean Bruno 	/* for WoL-capable devices */
2885a9ca1c79SSean Bruno 	if (hw->device_id == IXGBE_DEV_ID_X550EM_X_10G_T) {
2886c19c7afeSEric Joyner 		SYSCTL_ADD_PROC(ctx_list, child, OID_AUTO, "wol_enable",
2887f72de14eSKevin Bowling 		    CTLTYPE_INT | CTLFLAG_RW, sc, 0,
2888c58d34ddSKevin Bowling 		    ixgbe_sysctl_wol_enable, "I",
2889c58d34ddSKevin Bowling 		    "Enable/Disable Wake on LAN");
28906f37f232SEric Joyner 
2891c19c7afeSEric Joyner 		SYSCTL_ADD_PROC(ctx_list, child, OID_AUTO, "wufc",
2892f72de14eSKevin Bowling 		    CTLTYPE_U32 | CTLFLAG_RW,
2893b1d5caf3SKevin Bowling 		    sc, 0, ixgbe_sysctl_wufc,
28948eb6488eSEric Joyner 		    "I", "Enable/Disable Wake Up Filters");
28956f37f232SEric Joyner 	}
28966f37f232SEric Joyner 
2897a9ca1c79SSean Bruno 	/* for X552/X557-AT devices */
28986f37f232SEric Joyner 	if (hw->device_id == IXGBE_DEV_ID_X550EM_X_10G_T) {
28996f37f232SEric Joyner 		struct sysctl_oid *phy_node;
29006f37f232SEric Joyner 		struct sysctl_oid_list *phy_list;
29016f37f232SEric Joyner 
2902c19c7afeSEric Joyner 		phy_node = SYSCTL_ADD_NODE(ctx_list, child, OID_AUTO, "phy",
2903c58d34ddSKevin Bowling 		    CTLFLAG_RD | CTLFLAG_MPSAFE, NULL,
2904c58d34ddSKevin Bowling 		    "External PHY sysctls");
29056f37f232SEric Joyner 		phy_list = SYSCTL_CHILDREN(phy_node);
29066f37f232SEric Joyner 
2907c19c7afeSEric Joyner 		SYSCTL_ADD_PROC(ctx_list, phy_list, OID_AUTO, "temp",
2908f72de14eSKevin Bowling 		    CTLTYPE_U16 | CTLFLAG_RD,
2909b1d5caf3SKevin Bowling 		    sc, 0, ixgbe_sysctl_phy_temp,
29108eb6488eSEric Joyner 		    "I", "Current External PHY Temperature (Celsius)");
29116f37f232SEric Joyner 
2912c19c7afeSEric Joyner 		SYSCTL_ADD_PROC(ctx_list, phy_list, OID_AUTO,
291320b91f0aSPawel Biernacki 		    "overtemp_occurred",
2914f72de14eSKevin Bowling 		    CTLTYPE_U16 | CTLFLAG_RD, sc, 0,
29156f37f232SEric Joyner 		    ixgbe_sysctl_phy_overtemp_occurred, "I",
29166f37f232SEric Joyner 		    "External PHY High Temperature Event Occurred");
29176f37f232SEric Joyner 	}
29188eb6488eSEric Joyner 
2919b1d5caf3SKevin Bowling 	if (sc->feat_cap & IXGBE_FEATURE_EEE) {
2920c19c7afeSEric Joyner 		SYSCTL_ADD_PROC(ctx_list, child, OID_AUTO, "eee_state",
2921f72de14eSKevin Bowling 		    CTLTYPE_INT | CTLFLAG_RW, sc, 0,
29228eb6488eSEric Joyner 		    ixgbe_sysctl_eee_state, "I", "EEE Power Save State");
29238eb6488eSEric Joyner 	}
29248eb6488eSEric Joyner } /* ixgbe_add_device_sysctls */
29258eb6488eSEric Joyner 
29268eb6488eSEric Joyner /************************************************************************
29278eb6488eSEric Joyner  * ixgbe_allocate_pci_resources
29288eb6488eSEric Joyner  ************************************************************************/
29298eb6488eSEric Joyner static int
ixgbe_allocate_pci_resources(if_ctx_t ctx)2930c19c7afeSEric Joyner ixgbe_allocate_pci_resources(if_ctx_t ctx)
29318eb6488eSEric Joyner {
2932b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
2933c19c7afeSEric Joyner 	device_t dev = iflib_get_dev(ctx);
29348eb6488eSEric Joyner 	int rid;
29358eb6488eSEric Joyner 
29368eb6488eSEric Joyner 	rid = PCIR_BAR(0);
2937b1d5caf3SKevin Bowling 	sc->pci_mem = bus_alloc_resource_any(dev, SYS_RES_MEMORY, &rid,
29388eb6488eSEric Joyner 	    RF_ACTIVE);
29398eb6488eSEric Joyner 
2940b1d5caf3SKevin Bowling 	if (!(sc->pci_mem)) {
2941c58d34ddSKevin Bowling 		device_printf(dev,
2942c58d34ddSKevin Bowling 		    "Unable to allocate bus resource: memory\n");
29438eb6488eSEric Joyner 		return (ENXIO);
29448eb6488eSEric Joyner 	}
29458eb6488eSEric Joyner 
29468eb6488eSEric Joyner 	/* Save bus_space values for READ/WRITE_REG macros */
2947b1d5caf3SKevin Bowling 	sc->osdep.mem_bus_space_tag = rman_get_bustag(sc->pci_mem);
2948b1d5caf3SKevin Bowling 	sc->osdep.mem_bus_space_handle =
2949b1d5caf3SKevin Bowling 	    rman_get_bushandle(sc->pci_mem);
29508eb6488eSEric Joyner 	/* Set hw values for shared code */
2951b1d5caf3SKevin Bowling 	sc->hw.hw_addr = (u8 *)&sc->osdep.mem_bus_space_handle;
29528eb6488eSEric Joyner 
29538eb6488eSEric Joyner 	return (0);
29548eb6488eSEric Joyner } /* ixgbe_allocate_pci_resources */
29558eb6488eSEric Joyner 
29568eb6488eSEric Joyner /************************************************************************
29578eb6488eSEric Joyner  * ixgbe_detach - Device removal routine
29588eb6488eSEric Joyner  *
29598eb6488eSEric Joyner  *   Called when the driver is being removed.
29608eb6488eSEric Joyner  *   Stops the adapter and deallocates all the resources
29618eb6488eSEric Joyner  *   that were allocated for driver operation.
29628eb6488eSEric Joyner  *
29638eb6488eSEric Joyner  *   return 0 on success, positive on failure
29648eb6488eSEric Joyner  ************************************************************************/
29658eb6488eSEric Joyner static int
ixgbe_if_detach(if_ctx_t ctx)2966c19c7afeSEric Joyner ixgbe_if_detach(if_ctx_t ctx)
29678eb6488eSEric Joyner {
2968b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
2969c19c7afeSEric Joyner 	device_t dev = iflib_get_dev(ctx);
29708eb6488eSEric Joyner 	u32 ctrl_ext;
29718eb6488eSEric Joyner 
29728eb6488eSEric Joyner 	INIT_DEBUGOUT("ixgbe_detach: begin");
29738eb6488eSEric Joyner 
29748eb6488eSEric Joyner 	if (ixgbe_pci_iov_detach(dev) != 0) {
29758eb6488eSEric Joyner 		device_printf(dev, "SR-IOV in use; detach first.\n");
29768eb6488eSEric Joyner 		return (EBUSY);
29778eb6488eSEric Joyner 	}
29788eb6488eSEric Joyner 
2979c19c7afeSEric Joyner 	ixgbe_setup_low_power_mode(ctx);
29808eb6488eSEric Joyner 
29818eb6488eSEric Joyner 	/* let hardware know driver is unloading */
2982b1d5caf3SKevin Bowling 	ctrl_ext = IXGBE_READ_REG(&sc->hw, IXGBE_CTRL_EXT);
29838eb6488eSEric Joyner 	ctrl_ext &= ~IXGBE_CTRL_EXT_DRV_LOAD;
2984b1d5caf3SKevin Bowling 	IXGBE_WRITE_REG(&sc->hw, IXGBE_CTRL_EXT, ctrl_ext);
29858eb6488eSEric Joyner 
2986f72de14eSKevin Bowling 	callout_drain(&sc->fw_mode_timer);
2987f72de14eSKevin Bowling 
2988c19c7afeSEric Joyner 	ixgbe_free_pci_resources(ctx);
2989b1d5caf3SKevin Bowling 	free(sc->mta, M_IXGBE);
29908eb6488eSEric Joyner 
29918eb6488eSEric Joyner 	return (0);
2992c19c7afeSEric Joyner } /* ixgbe_if_detach */
29938eb6488eSEric Joyner 
29948eb6488eSEric Joyner /************************************************************************
29958eb6488eSEric Joyner  * ixgbe_setup_low_power_mode - LPLU/WoL preparation
29968eb6488eSEric Joyner  *
29978eb6488eSEric Joyner  *   Prepare the adapter/port for LPLU and/or WoL
29988eb6488eSEric Joyner  ************************************************************************/
29998eb6488eSEric Joyner static int
ixgbe_setup_low_power_mode(if_ctx_t ctx)3000c19c7afeSEric Joyner ixgbe_setup_low_power_mode(if_ctx_t ctx)
30018eb6488eSEric Joyner {
3002b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
3003b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
3004c19c7afeSEric Joyner 	device_t dev = iflib_get_dev(ctx);
30058eb6488eSEric Joyner 	s32 error = 0;
30068eb6488eSEric Joyner 
3007c19c7afeSEric Joyner 	if (!hw->wol_enabled)
300879b36ec9SKevin Bowling 		ixgbe_set_phy_power(hw, false);
30098eb6488eSEric Joyner 
30108eb6488eSEric Joyner 	/* Limit power management flow to X550EM baseT */
30118eb6488eSEric Joyner 	if (hw->device_id == IXGBE_DEV_ID_X550EM_X_10G_T &&
30128eb6488eSEric Joyner 	    hw->phy.ops.enter_lplu) {
30138eb6488eSEric Joyner 		/* Turn off support for APM wakeup. (Using ACPI instead) */
301473fa89e0SPiotr Pietruszewski 		IXGBE_WRITE_REG(hw, IXGBE_GRC_BY_MAC(hw),
301573fa89e0SPiotr Pietruszewski 		    IXGBE_READ_REG(hw, IXGBE_GRC_BY_MAC(hw)) & ~(u32)2);
30168eb6488eSEric Joyner 
30178eb6488eSEric Joyner 		/*
3018c58d34ddSKevin Bowling 		 * Clear Wake Up Status register to prevent any previous
3019c58d34ddSKevin Bowling 		 * wakeup events from waking us up immediately after we
3020c58d34ddSKevin Bowling 		 * suspend.
30218eb6488eSEric Joyner 		 */
30228eb6488eSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_WUS, 0xffffffff);
30238eb6488eSEric Joyner 
30248eb6488eSEric Joyner 		/*
30258eb6488eSEric Joyner 		 * Program the Wakeup Filter Control register with user filter
30268eb6488eSEric Joyner 		 * settings
30278eb6488eSEric Joyner 		 */
3028b1d5caf3SKevin Bowling 		IXGBE_WRITE_REG(hw, IXGBE_WUFC, sc->wufc);
30298eb6488eSEric Joyner 
30308eb6488eSEric Joyner 		/* Enable wakeups and power management in Wakeup Control */
30318eb6488eSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_WUC,
30328eb6488eSEric Joyner 		    IXGBE_WUC_WKEN | IXGBE_WUC_PME_EN);
30338eb6488eSEric Joyner 
30348eb6488eSEric Joyner 		/* X550EM baseT adapters need a special LPLU flow */
303579b36ec9SKevin Bowling 		hw->phy.reset_disable = true;
3036c19c7afeSEric Joyner 		ixgbe_if_stop(ctx);
30378eb6488eSEric Joyner 		error = hw->phy.ops.enter_lplu(hw);
30388eb6488eSEric Joyner 		if (error)
3039c58d34ddSKevin Bowling 			device_printf(dev, "Error entering LPLU: %d\n",
3040c58d34ddSKevin Bowling 			    error);
304179b36ec9SKevin Bowling 		hw->phy.reset_disable = false;
30428eb6488eSEric Joyner 	} else {
30438eb6488eSEric Joyner 		/* Just stop for other adapters */
3044c19c7afeSEric Joyner 		ixgbe_if_stop(ctx);
30458eb6488eSEric Joyner 	}
30468eb6488eSEric Joyner 
30478eb6488eSEric Joyner 	return error;
30488eb6488eSEric Joyner } /* ixgbe_setup_low_power_mode */
30498eb6488eSEric Joyner 
30508eb6488eSEric Joyner /************************************************************************
30518eb6488eSEric Joyner  * ixgbe_shutdown - Shutdown entry point
30528eb6488eSEric Joyner  ************************************************************************/
30538eb6488eSEric Joyner static int
ixgbe_if_shutdown(if_ctx_t ctx)3054c19c7afeSEric Joyner ixgbe_if_shutdown(if_ctx_t ctx)
30558eb6488eSEric Joyner {
30568eb6488eSEric Joyner 	int error = 0;
30578eb6488eSEric Joyner 
30588eb6488eSEric Joyner 	INIT_DEBUGOUT("ixgbe_shutdown: begin");
30598eb6488eSEric Joyner 
3060c19c7afeSEric Joyner 	error = ixgbe_setup_low_power_mode(ctx);
30618eb6488eSEric Joyner 
30628eb6488eSEric Joyner 	return (error);
3063c19c7afeSEric Joyner } /* ixgbe_if_shutdown */
30648eb6488eSEric Joyner 
30658eb6488eSEric Joyner /************************************************************************
30668eb6488eSEric Joyner  * ixgbe_suspend
30678eb6488eSEric Joyner  *
30688eb6488eSEric Joyner  *   From D0 to D3
30698eb6488eSEric Joyner  ************************************************************************/
30708eb6488eSEric Joyner static int
ixgbe_if_suspend(if_ctx_t ctx)3071c19c7afeSEric Joyner ixgbe_if_suspend(if_ctx_t ctx)
30728eb6488eSEric Joyner {
30738eb6488eSEric Joyner 	int error = 0;
30748eb6488eSEric Joyner 
30758eb6488eSEric Joyner 	INIT_DEBUGOUT("ixgbe_suspend: begin");
30768eb6488eSEric Joyner 
3077c19c7afeSEric Joyner 	error = ixgbe_setup_low_power_mode(ctx);
30788eb6488eSEric Joyner 
30798eb6488eSEric Joyner 	return (error);
3080c19c7afeSEric Joyner } /* ixgbe_if_suspend */
30818eb6488eSEric Joyner 
30828eb6488eSEric Joyner /************************************************************************
30838eb6488eSEric Joyner  * ixgbe_resume
30848eb6488eSEric Joyner  *
30858eb6488eSEric Joyner  *   From D3 to D0
30868eb6488eSEric Joyner  ************************************************************************/
30878eb6488eSEric Joyner static int
ixgbe_if_resume(if_ctx_t ctx)3088c19c7afeSEric Joyner ixgbe_if_resume(if_ctx_t ctx)
30898eb6488eSEric Joyner {
3090b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
3091c19c7afeSEric Joyner 	device_t dev = iflib_get_dev(ctx);
3092ff06a8dbSJustin Hibbits 	if_t ifp = iflib_get_ifp(ctx);
3093b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
30948eb6488eSEric Joyner 	u32 wus;
30958eb6488eSEric Joyner 
30968eb6488eSEric Joyner 	INIT_DEBUGOUT("ixgbe_resume: begin");
30978eb6488eSEric Joyner 
30988eb6488eSEric Joyner 	/* Read & clear WUS register */
30998eb6488eSEric Joyner 	wus = IXGBE_READ_REG(hw, IXGBE_WUS);
31008eb6488eSEric Joyner 	if (wus)
31018eb6488eSEric Joyner 		device_printf(dev, "Woken up by (WUS): %#010x\n",
31028eb6488eSEric Joyner 		    IXGBE_READ_REG(hw, IXGBE_WUS));
31038eb6488eSEric Joyner 	IXGBE_WRITE_REG(hw, IXGBE_WUS, 0xffffffff);
31048eb6488eSEric Joyner 	/* And clear WUFC until next low-power transition */
31058eb6488eSEric Joyner 	IXGBE_WRITE_REG(hw, IXGBE_WUFC, 0);
31068eb6488eSEric Joyner 
31078eb6488eSEric Joyner 	/*
31088eb6488eSEric Joyner 	 * Required after D3->D0 transition;
31098eb6488eSEric Joyner 	 * will re-advertise all previous advertised speeds
31108eb6488eSEric Joyner 	 */
3111ff06a8dbSJustin Hibbits 	if (if_getflags(ifp) & IFF_UP)
3112c19c7afeSEric Joyner 		ixgbe_if_init(ctx);
31138eb6488eSEric Joyner 
31148eb6488eSEric Joyner 	return (0);
3115c19c7afeSEric Joyner } /* ixgbe_if_resume */
31168eb6488eSEric Joyner 
31178eb6488eSEric Joyner /************************************************************************
3118c19c7afeSEric Joyner  * ixgbe_if_mtu_set - Ioctl mtu entry point
31198eb6488eSEric Joyner  *
3120c19c7afeSEric Joyner  *   Return 0 on success, EINVAL on failure
3121c19c7afeSEric Joyner  ************************************************************************/
3122c19c7afeSEric Joyner static int
ixgbe_if_mtu_set(if_ctx_t ctx,uint32_t mtu)3123c19c7afeSEric Joyner ixgbe_if_mtu_set(if_ctx_t ctx, uint32_t mtu)
3124c19c7afeSEric Joyner {
3125b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
3126c19c7afeSEric Joyner 	int error = 0;
3127c19c7afeSEric Joyner 
3128c19c7afeSEric Joyner 	IOCTL_DEBUGOUT("ioctl: SIOCIFMTU (Set Interface MTU)");
3129c19c7afeSEric Joyner 
3130c19c7afeSEric Joyner 	if (mtu > IXGBE_MAX_MTU) {
3131c19c7afeSEric Joyner 		error = EINVAL;
3132c19c7afeSEric Joyner 	} else {
3133b1d5caf3SKevin Bowling 		sc->max_frame_size = mtu + IXGBE_MTU_HDR;
3134c19c7afeSEric Joyner 	}
3135c19c7afeSEric Joyner 
3136c19c7afeSEric Joyner 	return error;
3137c19c7afeSEric Joyner } /* ixgbe_if_mtu_set */
3138c19c7afeSEric Joyner 
3139c19c7afeSEric Joyner /************************************************************************
3140c19c7afeSEric Joyner  * ixgbe_if_crcstrip_set
31418eb6488eSEric Joyner  ************************************************************************/
31428eb6488eSEric Joyner static void
ixgbe_if_crcstrip_set(if_ctx_t ctx,int onoff,int crcstrip)3143c19c7afeSEric Joyner ixgbe_if_crcstrip_set(if_ctx_t ctx, int onoff, int crcstrip)
31448eb6488eSEric Joyner {
3145b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
3146c19c7afeSEric Joyner 	struct ixgbe_hw *hw = &sc->hw;
3147c19c7afeSEric Joyner 	/* crc stripping is set in two places:
3148c19c7afeSEric Joyner 	 * IXGBE_HLREG0 (modified on init_locked and hw reset)
3149c19c7afeSEric Joyner 	 * IXGBE_RDRXCTL (set by the original driver in
3150c19c7afeSEric Joyner 	 *	ixgbe_setup_hw_rsc() called in init_locked.
3151c19c7afeSEric Joyner 	 *	We disable the setting when netmap is compiled in).
3152c19c7afeSEric Joyner 	 * We update the values here, but also in ixgbe.c because
3153c19c7afeSEric Joyner 	 * init_locked sometimes is called outside our control.
3154c19c7afeSEric Joyner 	 */
3155c19c7afeSEric Joyner 	uint32_t hl, rxc;
31568eb6488eSEric Joyner 
3157c19c7afeSEric Joyner 	hl = IXGBE_READ_REG(hw, IXGBE_HLREG0);
3158c19c7afeSEric Joyner 	rxc = IXGBE_READ_REG(hw, IXGBE_RDRXCTL);
3159c19c7afeSEric Joyner #ifdef NETMAP
3160c19c7afeSEric Joyner 	if (netmap_verbose)
3161c19c7afeSEric Joyner 		D("%s read  HLREG 0x%x rxc 0x%x",
3162c19c7afeSEric Joyner 			onoff ? "enter" : "exit", hl, rxc);
31638eb6488eSEric Joyner #endif
3164c19c7afeSEric Joyner 	/* hw requirements ... */
3165c19c7afeSEric Joyner 	rxc &= ~IXGBE_RDRXCTL_RSCFRSTSIZE;
3166c19c7afeSEric Joyner 	rxc |= IXGBE_RDRXCTL_RSCACKC;
3167c19c7afeSEric Joyner 	if (onoff && !crcstrip) {
3168c19c7afeSEric Joyner 		/* keep the crc. Fast rx */
3169c19c7afeSEric Joyner 		hl &= ~IXGBE_HLREG0_RXCRCSTRP;
3170c19c7afeSEric Joyner 		rxc &= ~IXGBE_RDRXCTL_CRCSTRIP;
3171c19c7afeSEric Joyner 	} else {
3172c19c7afeSEric Joyner 		/* reset default mode */
3173c19c7afeSEric Joyner 		hl |= IXGBE_HLREG0_RXCRCSTRP;
3174c19c7afeSEric Joyner 		rxc |= IXGBE_RDRXCTL_CRCSTRIP;
3175c19c7afeSEric Joyner 	}
3176c19c7afeSEric Joyner #ifdef NETMAP
3177c19c7afeSEric Joyner 	if (netmap_verbose)
3178c19c7afeSEric Joyner 		D("%s write HLREG 0x%x rxc 0x%x",
3179c19c7afeSEric Joyner 			onoff ? "enter" : "exit", hl, rxc);
3180c19c7afeSEric Joyner #endif
3181c19c7afeSEric Joyner 	IXGBE_WRITE_REG(hw, IXGBE_HLREG0, hl);
3182c19c7afeSEric Joyner 	IXGBE_WRITE_REG(hw, IXGBE_RDRXCTL, rxc);
3183c19c7afeSEric Joyner } /* ixgbe_if_crcstrip_set */
31848eb6488eSEric Joyner 
3185c19c7afeSEric Joyner /*********************************************************************
3186c19c7afeSEric Joyner  * ixgbe_if_init - Init entry point
31878eb6488eSEric Joyner  *
31888eb6488eSEric Joyner  *   Used in two ways: It is used by the stack as an init
31898eb6488eSEric Joyner  *   entry point in network interface structure. It is also
31908eb6488eSEric Joyner  *   used by the driver as a hw/sw initialization routine to
31918eb6488eSEric Joyner  *   get to a consistent state.
31928eb6488eSEric Joyner  *
3193c19c7afeSEric Joyner  *   Return 0 on success, positive on failure
3194c19c7afeSEric Joyner  **********************************************************************/
31958eb6488eSEric Joyner void
ixgbe_if_init(if_ctx_t ctx)3196c19c7afeSEric Joyner ixgbe_if_init(if_ctx_t ctx)
31978eb6488eSEric Joyner {
3198b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
3199ff06a8dbSJustin Hibbits 	if_t ifp = iflib_get_ifp(ctx);
3200c19c7afeSEric Joyner 	device_t dev = iflib_get_dev(ctx);
3201b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
3202c19c7afeSEric Joyner 	struct ix_rx_queue *rx_que;
3203c19c7afeSEric Joyner 	struct ix_tx_queue *tx_que;
32048eb6488eSEric Joyner 	u32 txdctl, mhadd;
32058eb6488eSEric Joyner 	u32 rxdctl, rxctrl;
32068eb6488eSEric Joyner 	u32 ctrl_ext;
32078eb6488eSEric Joyner 
3208c19c7afeSEric Joyner 	int i, j, err;
32098eb6488eSEric Joyner 
3210c19c7afeSEric Joyner 	INIT_DEBUGOUT("ixgbe_if_init: begin");
32118eb6488eSEric Joyner 
32128eb6488eSEric Joyner 	/* Queue indices may change with IOV mode */
3213b1d5caf3SKevin Bowling 	ixgbe_align_all_queue_indices(sc);
32148eb6488eSEric Joyner 
32158eb6488eSEric Joyner 	/* reprogram the RAR[0] in case user changed it. */
3216b1d5caf3SKevin Bowling 	ixgbe_set_rar(hw, 0, hw->mac.addr, sc->pool, IXGBE_RAH_AV);
32178eb6488eSEric Joyner 
32188eb6488eSEric Joyner 	/* Get the latest mac address, User can use a LAA */
3219ff06a8dbSJustin Hibbits 	bcopy(if_getlladdr(ifp), hw->mac.addr, IXGBE_ETH_LENGTH_OF_ADDRESS);
3220b1d5caf3SKevin Bowling 	ixgbe_set_rar(hw, 0, hw->mac.addr, sc->pool, 1);
32218eb6488eSEric Joyner 	hw->addr_ctrl.rar_used_count = 1;
32228eb6488eSEric Joyner 
32238eb6488eSEric Joyner 	ixgbe_init_hw(hw);
3224c19c7afeSEric Joyner 
3225b1d5caf3SKevin Bowling 	ixgbe_initialize_iov(sc);
3226c19c7afeSEric Joyner 
3227c19c7afeSEric Joyner 	ixgbe_initialize_transmit_units(ctx);
32288eb6488eSEric Joyner 
32298eb6488eSEric Joyner 	/* Setup Multicast table */
3230c19c7afeSEric Joyner 	ixgbe_if_multi_set(ctx);
32318eb6488eSEric Joyner 
32328eb6488eSEric Joyner 	/* Determine the correct mbuf pool, based on frame size */
3233b1d5caf3SKevin Bowling 	sc->rx_mbuf_sz = iflib_get_rx_mbuf_sz(ctx);
32348eb6488eSEric Joyner 
32358eb6488eSEric Joyner 	/* Configure RX settings */
3236c19c7afeSEric Joyner 	ixgbe_initialize_receive_units(ctx);
32378eb6488eSEric Joyner 
3238b2c1e8e6SEric Joyner 	/*
3239b2c1e8e6SEric Joyner 	 * Initialize variable holding task enqueue requests
3240b2c1e8e6SEric Joyner 	 * from MSI-X interrupts
3241b2c1e8e6SEric Joyner 	 */
3242b1d5caf3SKevin Bowling 	sc->task_requests = 0;
3243b2c1e8e6SEric Joyner 
32448eb6488eSEric Joyner 	/* Enable SDP & MSI-X interrupts based on adapter */
3245b1d5caf3SKevin Bowling 	ixgbe_config_gpie(sc);
32468eb6488eSEric Joyner 
32478eb6488eSEric Joyner 	/* Set MTU size */
3248ff06a8dbSJustin Hibbits 	if (if_getmtu(ifp) > ETHERMTU) {
32498eb6488eSEric Joyner 		/* aka IXGBE_MAXFRS on 82599 and newer */
32508eb6488eSEric Joyner 		mhadd = IXGBE_READ_REG(hw, IXGBE_MHADD);
32518eb6488eSEric Joyner 		mhadd &= ~IXGBE_MHADD_MFS_MASK;
3252b1d5caf3SKevin Bowling 		mhadd |= sc->max_frame_size << IXGBE_MHADD_MFS_SHIFT;
32538eb6488eSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_MHADD, mhadd);
32548eb6488eSEric Joyner 	}
32558eb6488eSEric Joyner 
32568eb6488eSEric Joyner 	/* Now enable all the queues */
3257c58d34ddSKevin Bowling 	for (i = 0, tx_que = sc->tx_queues; i < sc->num_tx_queues;
3258c58d34ddSKevin Bowling 	    i++, tx_que++) {
3259c19c7afeSEric Joyner 		struct tx_ring *txr = &tx_que->txr;
3260c19c7afeSEric Joyner 
32618eb6488eSEric Joyner 		txdctl = IXGBE_READ_REG(hw, IXGBE_TXDCTL(txr->me));
32628eb6488eSEric Joyner 		txdctl |= IXGBE_TXDCTL_ENABLE;
32638eb6488eSEric Joyner 		/* Set WTHRESH to 8, burst writeback */
32648eb6488eSEric Joyner 		txdctl |= (8 << 16);
32658eb6488eSEric Joyner 		/*
32668eb6488eSEric Joyner 		 * When the internal queue falls below PTHRESH (32),
32678eb6488eSEric Joyner 		 * start prefetching as long as there are at least
32688eb6488eSEric Joyner 		 * HTHRESH (1) buffers ready. The values are taken
32698eb6488eSEric Joyner 		 * from the Intel linux driver 3.8.21.
32708eb6488eSEric Joyner 		 * Prefetching enables tx line rate even with 1 queue.
32718eb6488eSEric Joyner 		 */
32728eb6488eSEric Joyner 		txdctl |= (32 << 0) | (1 << 8);
32738eb6488eSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_TXDCTL(txr->me), txdctl);
32748eb6488eSEric Joyner 	}
32758eb6488eSEric Joyner 
3276c58d34ddSKevin Bowling 	for (i = 0, rx_que = sc->rx_queues; i < sc->num_rx_queues;
3277c58d34ddSKevin Bowling 	    i++, rx_que++) {
3278c19c7afeSEric Joyner 		struct rx_ring *rxr = &rx_que->rxr;
3279c19c7afeSEric Joyner 
32808eb6488eSEric Joyner 		rxdctl = IXGBE_READ_REG(hw, IXGBE_RXDCTL(rxr->me));
32818eb6488eSEric Joyner 		if (hw->mac.type == ixgbe_mac_82598EB) {
32828eb6488eSEric Joyner 			/*
32838eb6488eSEric Joyner 			 * PTHRESH = 21
32848eb6488eSEric Joyner 			 * HTHRESH = 4
32858eb6488eSEric Joyner 			 * WTHRESH = 8
32868eb6488eSEric Joyner 			 */
32878eb6488eSEric Joyner 			rxdctl &= ~0x3FFFFF;
32888eb6488eSEric Joyner 			rxdctl |= 0x080420;
32898eb6488eSEric Joyner 		}
32908eb6488eSEric Joyner 		rxdctl |= IXGBE_RXDCTL_ENABLE;
32918eb6488eSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_RXDCTL(rxr->me), rxdctl);
3292c19c7afeSEric Joyner 		for (j = 0; j < 10; j++) {
32938eb6488eSEric Joyner 			if (IXGBE_READ_REG(hw, IXGBE_RXDCTL(rxr->me)) &
32948eb6488eSEric Joyner 			    IXGBE_RXDCTL_ENABLE)
32958eb6488eSEric Joyner 				break;
32968eb6488eSEric Joyner 			else
32978eb6488eSEric Joyner 				msec_delay(1);
32988eb6488eSEric Joyner 		}
32998eb6488eSEric Joyner 		wmb();
33008eb6488eSEric Joyner 	}
33018eb6488eSEric Joyner 
33028eb6488eSEric Joyner 	/* Enable Receive engine */
33038eb6488eSEric Joyner 	rxctrl = IXGBE_READ_REG(hw, IXGBE_RXCTRL);
33048eb6488eSEric Joyner 	if (hw->mac.type == ixgbe_mac_82598EB)
33058eb6488eSEric Joyner 		rxctrl |= IXGBE_RXCTRL_DMBYPS;
33068eb6488eSEric Joyner 	rxctrl |= IXGBE_RXCTRL_RXEN;
33078eb6488eSEric Joyner 	ixgbe_enable_rx_dma(hw, rxctrl);
33088eb6488eSEric Joyner 
3309c19c7afeSEric Joyner 	/* Set up MSI/MSI-X routing */
3310c19c7afeSEric Joyner 	if (ixgbe_enable_msix)  {
3311b1d5caf3SKevin Bowling 		ixgbe_configure_ivars(sc);
33128eb6488eSEric Joyner 		/* Set up auto-mask */
33138eb6488eSEric Joyner 		if (hw->mac.type == ixgbe_mac_82598EB)
33148eb6488eSEric Joyner 			IXGBE_WRITE_REG(hw, IXGBE_EIAM, IXGBE_EICS_RTX_QUEUE);
33158eb6488eSEric Joyner 		else {
33168eb6488eSEric Joyner 			IXGBE_WRITE_REG(hw, IXGBE_EIAM_EX(0), 0xFFFFFFFF);
33178eb6488eSEric Joyner 			IXGBE_WRITE_REG(hw, IXGBE_EIAM_EX(1), 0xFFFFFFFF);
33188eb6488eSEric Joyner 		}
33198eb6488eSEric Joyner 	} else {  /* Simple settings for Legacy/MSI */
3320b1d5caf3SKevin Bowling 		ixgbe_set_ivar(sc, 0, 0, 0);
3321b1d5caf3SKevin Bowling 		ixgbe_set_ivar(sc, 0, 0, 1);
33228eb6488eSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_EIAM, IXGBE_EICS_RTX_QUEUE);
33238eb6488eSEric Joyner 	}
33248eb6488eSEric Joyner 
3325b1d5caf3SKevin Bowling 	ixgbe_init_fdir(sc);
33268eb6488eSEric Joyner 
33278eb6488eSEric Joyner 	/*
33288eb6488eSEric Joyner 	 * Check on any SFP devices that
33298eb6488eSEric Joyner 	 * need to be kick-started
33308eb6488eSEric Joyner 	 */
33318eb6488eSEric Joyner 	if (hw->phy.type == ixgbe_phy_none) {
33328eb6488eSEric Joyner 		err = hw->phy.ops.identify(hw);
33338eb6488eSEric Joyner 		if (err == IXGBE_ERR_SFP_NOT_SUPPORTED) {
33348eb6488eSEric Joyner 			device_printf(dev,
33358eb6488eSEric Joyner 			    "Unsupported SFP+ module type was detected.\n");
33368eb6488eSEric Joyner 			return;
33378eb6488eSEric Joyner 		}
33388eb6488eSEric Joyner 	}
33398eb6488eSEric Joyner 
33408eb6488eSEric Joyner 	/* Set moderation on the Link interrupt */
3341b1d5caf3SKevin Bowling 	IXGBE_WRITE_REG(hw, IXGBE_EITR(sc->vector), IXGBE_LINK_ITR);
33428eb6488eSEric Joyner 
3343c19c7afeSEric Joyner 	/* Enable power to the phy. */
334479b36ec9SKevin Bowling 	ixgbe_set_phy_power(hw, true);
3345c19c7afeSEric Joyner 
33468eb6488eSEric Joyner 	/* Config/Enable Link */
3347b2c1e8e6SEric Joyner 	ixgbe_config_link(ctx);
33488eb6488eSEric Joyner 
33498eb6488eSEric Joyner 	/* Hardware Packet Buffer & Flow Control setup */
3350b1d5caf3SKevin Bowling 	ixgbe_config_delay_values(sc);
33518eb6488eSEric Joyner 
33528eb6488eSEric Joyner 	/* Initialize the FC settings */
33538eb6488eSEric Joyner 	ixgbe_start_hw(hw);
33548eb6488eSEric Joyner 
33558eb6488eSEric Joyner 	/* Set up VLAN support and filter */
3356c19c7afeSEric Joyner 	ixgbe_setup_vlan_hw_support(ctx);
33578eb6488eSEric Joyner 
33588eb6488eSEric Joyner 	/* Setup DMA Coalescing */
3359b1d5caf3SKevin Bowling 	ixgbe_config_dmac(sc);
33608eb6488eSEric Joyner 
33618eb6488eSEric Joyner 	/* And now turn on interrupts */
3362c19c7afeSEric Joyner 	ixgbe_if_enable_intr(ctx);
33638eb6488eSEric Joyner 
33648eb6488eSEric Joyner 	/* Enable the use of the MBX by the VF's */
3365b1d5caf3SKevin Bowling 	if (sc->feat_en & IXGBE_FEATURE_SRIOV) {
33668eb6488eSEric Joyner 		ctrl_ext = IXGBE_READ_REG(hw, IXGBE_CTRL_EXT);
33678eb6488eSEric Joyner 		ctrl_ext |= IXGBE_CTRL_EXT_PFRSTD;
33688eb6488eSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_CTRL_EXT, ctrl_ext);
33698eb6488eSEric Joyner 	}
33708eb6488eSEric Joyner 
33718eb6488eSEric Joyner } /* ixgbe_init_locked */
33728eb6488eSEric Joyner 
33738eb6488eSEric Joyner /************************************************************************
33748eb6488eSEric Joyner  * ixgbe_set_ivar
33758eb6488eSEric Joyner  *
33768eb6488eSEric Joyner  *   Setup the correct IVAR register for a particular MSI-X interrupt
33778eb6488eSEric Joyner  *     (yes this is all very magic and confusing :)
33788eb6488eSEric Joyner  *    - entry is the register array entry
33798eb6488eSEric Joyner  *    - vector is the MSI-X vector for this queue
33808eb6488eSEric Joyner  *    - type is RX/TX/MISC
33818eb6488eSEric Joyner  ************************************************************************/
33828eb6488eSEric Joyner static void
ixgbe_set_ivar(struct ixgbe_softc * sc,u8 entry,u8 vector,s8 type)3383b1d5caf3SKevin Bowling ixgbe_set_ivar(struct ixgbe_softc *sc, u8 entry, u8 vector, s8 type)
33848eb6488eSEric Joyner {
3385b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
33868eb6488eSEric Joyner 	u32 ivar, index;
33878eb6488eSEric Joyner 
33888eb6488eSEric Joyner 	vector |= IXGBE_IVAR_ALLOC_VAL;
33898eb6488eSEric Joyner 
33908eb6488eSEric Joyner 	switch (hw->mac.type) {
33918eb6488eSEric Joyner 	case ixgbe_mac_82598EB:
33928eb6488eSEric Joyner 		if (type == -1)
33938eb6488eSEric Joyner 			entry = IXGBE_IVAR_OTHER_CAUSES_INDEX;
33948eb6488eSEric Joyner 		else
33958eb6488eSEric Joyner 			entry += (type * 64);
33968eb6488eSEric Joyner 		index = (entry >> 2) & 0x1F;
33978eb6488eSEric Joyner 		ivar = IXGBE_READ_REG(hw, IXGBE_IVAR(index));
33988eb6488eSEric Joyner 		ivar &= ~(0xFF << (8 * (entry & 0x3)));
33998eb6488eSEric Joyner 		ivar |= (vector << (8 * (entry & 0x3)));
3400b1d5caf3SKevin Bowling 		IXGBE_WRITE_REG(&sc->hw, IXGBE_IVAR(index), ivar);
34018eb6488eSEric Joyner 		break;
34028eb6488eSEric Joyner 	case ixgbe_mac_82599EB:
34038eb6488eSEric Joyner 	case ixgbe_mac_X540:
34048eb6488eSEric Joyner 	case ixgbe_mac_X550:
34058eb6488eSEric Joyner 	case ixgbe_mac_X550EM_x:
34068eb6488eSEric Joyner 	case ixgbe_mac_X550EM_a:
34078eb6488eSEric Joyner 		if (type == -1) { /* MISC IVAR */
34088eb6488eSEric Joyner 			index = (entry & 1) * 8;
34098eb6488eSEric Joyner 			ivar = IXGBE_READ_REG(hw, IXGBE_IVAR_MISC);
34108eb6488eSEric Joyner 			ivar &= ~(0xFF << index);
34118eb6488eSEric Joyner 			ivar |= (vector << index);
34128eb6488eSEric Joyner 			IXGBE_WRITE_REG(hw, IXGBE_IVAR_MISC, ivar);
34138eb6488eSEric Joyner 		} else {          /* RX/TX IVARS */
34148eb6488eSEric Joyner 			index = (16 * (entry & 1)) + (8 * type);
34158eb6488eSEric Joyner 			ivar = IXGBE_READ_REG(hw, IXGBE_IVAR(entry >> 1));
34168eb6488eSEric Joyner 			ivar &= ~(0xFF << index);
34178eb6488eSEric Joyner 			ivar |= (vector << index);
34188eb6488eSEric Joyner 			IXGBE_WRITE_REG(hw, IXGBE_IVAR(entry >> 1), ivar);
34198eb6488eSEric Joyner 		}
34208eb6488eSEric Joyner 	default:
34218eb6488eSEric Joyner 		break;
34228eb6488eSEric Joyner 	}
34238eb6488eSEric Joyner } /* ixgbe_set_ivar */
34248eb6488eSEric Joyner 
34258eb6488eSEric Joyner /************************************************************************
34268eb6488eSEric Joyner  * ixgbe_configure_ivars
34278eb6488eSEric Joyner  ************************************************************************/
34288eb6488eSEric Joyner static void
ixgbe_configure_ivars(struct ixgbe_softc * sc)3429b1d5caf3SKevin Bowling ixgbe_configure_ivars(struct ixgbe_softc *sc)
34308eb6488eSEric Joyner {
3431b1d5caf3SKevin Bowling 	struct ix_rx_queue *rx_que = sc->rx_queues;
3432b1d5caf3SKevin Bowling 	struct ix_tx_queue *tx_que = sc->tx_queues;
34338eb6488eSEric Joyner 	u32 newitr;
34348eb6488eSEric Joyner 
34358eb6488eSEric Joyner 	if (ixgbe_max_interrupt_rate > 0)
34368eb6488eSEric Joyner 		newitr = (4000000 / ixgbe_max_interrupt_rate) & 0x0FF8;
34378eb6488eSEric Joyner 	else {
34388eb6488eSEric Joyner 		/*
34398eb6488eSEric Joyner 		 * Disable DMA coalescing if interrupt moderation is
34408eb6488eSEric Joyner 		 * disabled.
34418eb6488eSEric Joyner 		 */
3442b1d5caf3SKevin Bowling 		sc->dmac = 0;
34438eb6488eSEric Joyner 		newitr = 0;
34448eb6488eSEric Joyner 	}
34458eb6488eSEric Joyner 
3446b1d5caf3SKevin Bowling 	for (int i = 0; i < sc->num_rx_queues; i++, rx_que++) {
3447c19c7afeSEric Joyner 		struct rx_ring *rxr = &rx_que->rxr;
34488eb6488eSEric Joyner 
3449c19c7afeSEric Joyner 		/* First the RX queue entry */
3450b1d5caf3SKevin Bowling 		ixgbe_set_ivar(sc, rxr->me, rx_que->msix, 0);
3451c19c7afeSEric Joyner 
3452c19c7afeSEric Joyner 		/* Set an Initial EITR value */
3453b1d5caf3SKevin Bowling 		IXGBE_WRITE_REG(&sc->hw, IXGBE_EITR(rx_que->msix), newitr);
3454c19c7afeSEric Joyner 	}
3455b1d5caf3SKevin Bowling 	for (int i = 0; i < sc->num_tx_queues; i++, tx_que++) {
3456c19c7afeSEric Joyner 		struct tx_ring *txr = &tx_que->txr;
3457c19c7afeSEric Joyner 
3458c19c7afeSEric Joyner 		/* ... and the TX */
3459b1d5caf3SKevin Bowling 		ixgbe_set_ivar(sc, txr->me, tx_que->msix, 1);
3460c19c7afeSEric Joyner 	}
34618eb6488eSEric Joyner 	/* For the Link interrupt */
3462b1d5caf3SKevin Bowling 	ixgbe_set_ivar(sc, 1, sc->vector, -1);
34638eb6488eSEric Joyner } /* ixgbe_configure_ivars */
34648eb6488eSEric Joyner 
34658eb6488eSEric Joyner /************************************************************************
34668eb6488eSEric Joyner  * ixgbe_config_gpie
34678eb6488eSEric Joyner  ************************************************************************/
34688eb6488eSEric Joyner static void
ixgbe_config_gpie(struct ixgbe_softc * sc)3469b1d5caf3SKevin Bowling ixgbe_config_gpie(struct ixgbe_softc *sc)
34708eb6488eSEric Joyner {
3471b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
34728eb6488eSEric Joyner 	u32 gpie;
34738eb6488eSEric Joyner 
34748eb6488eSEric Joyner 	gpie = IXGBE_READ_REG(hw, IXGBE_GPIE);
34758eb6488eSEric Joyner 
3476b1d5caf3SKevin Bowling 	if (sc->intr_type == IFLIB_INTR_MSIX) {
34778eb6488eSEric Joyner 		/* Enable Enhanced MSI-X mode */
3478c58d34ddSKevin Bowling 		gpie |= IXGBE_GPIE_MSIX_MODE |
3479c58d34ddSKevin Bowling 		    IXGBE_GPIE_EIAME |
3480c58d34ddSKevin Bowling 		    IXGBE_GPIE_PBA_SUPPORT |
3481c58d34ddSKevin Bowling 		    IXGBE_GPIE_OCD;
34828eb6488eSEric Joyner 	}
34838eb6488eSEric Joyner 
34848eb6488eSEric Joyner 	/* Fan Failure Interrupt */
3485b1d5caf3SKevin Bowling 	if (sc->feat_en & IXGBE_FEATURE_FAN_FAIL)
34868eb6488eSEric Joyner 		gpie |= IXGBE_SDP1_GPIEN;
34878eb6488eSEric Joyner 
34888eb6488eSEric Joyner 	/* Thermal Sensor Interrupt */
3489b1d5caf3SKevin Bowling 	if (sc->feat_en & IXGBE_FEATURE_TEMP_SENSOR)
34908eb6488eSEric Joyner 		gpie |= IXGBE_SDP0_GPIEN_X540;
34918eb6488eSEric Joyner 
34928eb6488eSEric Joyner 	/* Link detection */
34938eb6488eSEric Joyner 	switch (hw->mac.type) {
34948eb6488eSEric Joyner 	case ixgbe_mac_82599EB:
34958eb6488eSEric Joyner 		gpie |= IXGBE_SDP1_GPIEN | IXGBE_SDP2_GPIEN;
34968eb6488eSEric Joyner 		break;
34978eb6488eSEric Joyner 	case ixgbe_mac_X550EM_x:
34988eb6488eSEric Joyner 	case ixgbe_mac_X550EM_a:
34998eb6488eSEric Joyner 		gpie |= IXGBE_SDP0_GPIEN_X540;
35008eb6488eSEric Joyner 		break;
35018eb6488eSEric Joyner 	default:
35028eb6488eSEric Joyner 		break;
35038eb6488eSEric Joyner 	}
35048eb6488eSEric Joyner 
35058eb6488eSEric Joyner 	IXGBE_WRITE_REG(hw, IXGBE_GPIE, gpie);
35068eb6488eSEric Joyner 
35078eb6488eSEric Joyner } /* ixgbe_config_gpie */
35088eb6488eSEric Joyner 
35098eb6488eSEric Joyner /************************************************************************
35108eb6488eSEric Joyner  * ixgbe_config_delay_values
35118eb6488eSEric Joyner  *
3512b1d5caf3SKevin Bowling  *   Requires sc->max_frame_size to be set.
35138eb6488eSEric Joyner  ************************************************************************/
35148eb6488eSEric Joyner static void
ixgbe_config_delay_values(struct ixgbe_softc * sc)3515b1d5caf3SKevin Bowling ixgbe_config_delay_values(struct ixgbe_softc *sc)
35168eb6488eSEric Joyner {
3517b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
35188eb6488eSEric Joyner 	u32 rxpb, frame, size, tmp;
35198eb6488eSEric Joyner 
3520b1d5caf3SKevin Bowling 	frame = sc->max_frame_size;
35218eb6488eSEric Joyner 
35228eb6488eSEric Joyner 	/* Calculate High Water */
35238eb6488eSEric Joyner 	switch (hw->mac.type) {
35248eb6488eSEric Joyner 	case ixgbe_mac_X540:
35258eb6488eSEric Joyner 	case ixgbe_mac_X550:
35268eb6488eSEric Joyner 	case ixgbe_mac_X550EM_x:
35278eb6488eSEric Joyner 	case ixgbe_mac_X550EM_a:
35288eb6488eSEric Joyner 		tmp = IXGBE_DV_X540(frame, frame);
35298eb6488eSEric Joyner 		break;
35308eb6488eSEric Joyner 	default:
35318eb6488eSEric Joyner 		tmp = IXGBE_DV(frame, frame);
35328eb6488eSEric Joyner 		break;
35338eb6488eSEric Joyner 	}
35348eb6488eSEric Joyner 	size = IXGBE_BT2KB(tmp);
35358eb6488eSEric Joyner 	rxpb = IXGBE_READ_REG(hw, IXGBE_RXPBSIZE(0)) >> 10;
35368eb6488eSEric Joyner 	hw->fc.high_water[0] = rxpb - size;
35378eb6488eSEric Joyner 
35388eb6488eSEric Joyner 	/* Now calculate Low Water */
35398eb6488eSEric Joyner 	switch (hw->mac.type) {
35408eb6488eSEric Joyner 	case ixgbe_mac_X540:
35418eb6488eSEric Joyner 	case ixgbe_mac_X550:
35428eb6488eSEric Joyner 	case ixgbe_mac_X550EM_x:
35438eb6488eSEric Joyner 	case ixgbe_mac_X550EM_a:
35448eb6488eSEric Joyner 		tmp = IXGBE_LOW_DV_X540(frame);
35458eb6488eSEric Joyner 		break;
35468eb6488eSEric Joyner 	default:
35478eb6488eSEric Joyner 		tmp = IXGBE_LOW_DV(frame);
35488eb6488eSEric Joyner 		break;
35498eb6488eSEric Joyner 	}
35508eb6488eSEric Joyner 	hw->fc.low_water[0] = IXGBE_BT2KB(tmp);
35518eb6488eSEric Joyner 
35528eb6488eSEric Joyner 	hw->fc.pause_time = IXGBE_FC_PAUSE;
355379b36ec9SKevin Bowling 	hw->fc.send_xon = true;
35548eb6488eSEric Joyner } /* ixgbe_config_delay_values */
35558eb6488eSEric Joyner 
35568eb6488eSEric Joyner /************************************************************************
35578eb6488eSEric Joyner  * ixgbe_set_multi - Multicast Update
35588eb6488eSEric Joyner  *
35598eb6488eSEric Joyner  *   Called whenever multicast address list is updated.
35608eb6488eSEric Joyner  ************************************************************************/
3561ba76aa63SGleb Smirnoff static u_int
ixgbe_mc_filter_apply(void * arg,struct sockaddr_dl * sdl,u_int idx)356221afed4bSKevin Bowling ixgbe_mc_filter_apply(void *arg, struct sockaddr_dl *sdl, u_int idx)
35638eb6488eSEric Joyner {
3564b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = arg;
3565b1d5caf3SKevin Bowling 	struct ixgbe_mc_addr *mta = sc->mta;
3566c19c7afeSEric Joyner 
356721afed4bSKevin Bowling 	if (idx == MAX_NUM_MULTICAST_ADDRESSES)
3568c19c7afeSEric Joyner 		return (0);
356921afed4bSKevin Bowling 	bcopy(LLADDR(sdl), mta[idx].addr, IXGBE_ETH_LENGTH_OF_ADDRESS);
3570b1d5caf3SKevin Bowling 	mta[idx].vmdq = sc->pool;
3571c19c7afeSEric Joyner 
3572c19c7afeSEric Joyner 	return (1);
3573c19c7afeSEric Joyner } /* ixgbe_mc_filter_apply */
3574c19c7afeSEric Joyner 
3575c19c7afeSEric Joyner static void
ixgbe_if_multi_set(if_ctx_t ctx)3576c19c7afeSEric Joyner ixgbe_if_multi_set(if_ctx_t ctx)
3577c19c7afeSEric Joyner {
3578b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
35798eb6488eSEric Joyner 	struct ixgbe_mc_addr *mta;
3580ff06a8dbSJustin Hibbits 	if_t ifp = iflib_get_ifp(ctx);
35818eb6488eSEric Joyner 	u8 *update_ptr;
35828eb6488eSEric Joyner 	u32 fctrl;
3583ba76aa63SGleb Smirnoff 	u_int mcnt;
35848eb6488eSEric Joyner 
3585c19c7afeSEric Joyner 	IOCTL_DEBUGOUT("ixgbe_if_multi_set: begin");
35868eb6488eSEric Joyner 
3587b1d5caf3SKevin Bowling 	mta = sc->mta;
35888eb6488eSEric Joyner 	bzero(mta, sizeof(*mta) * MAX_NUM_MULTICAST_ADDRESSES);
35898eb6488eSEric Joyner 
3590c58d34ddSKevin Bowling 	mcnt = if_foreach_llmaddr(iflib_get_ifp(ctx), ixgbe_mc_filter_apply,
3591c58d34ddSKevin Bowling 	    sc);
35928eb6488eSEric Joyner 
3593395cc55dSKevin Bowling 	if (mcnt < MAX_NUM_MULTICAST_ADDRESSES) {
3594395cc55dSKevin Bowling 		update_ptr = (u8 *)mta;
3595395cc55dSKevin Bowling 		ixgbe_update_mc_addr_list(&sc->hw, update_ptr, mcnt,
3596395cc55dSKevin Bowling 		    ixgbe_mc_array_itr, true);
3597395cc55dSKevin Bowling 	}
3598395cc55dSKevin Bowling 
3599b1d5caf3SKevin Bowling 	fctrl = IXGBE_READ_REG(&sc->hw, IXGBE_FCTRL);
3600deecaa14SKevin Bowling 
3601ff06a8dbSJustin Hibbits 	if (if_getflags(ifp) & IFF_PROMISC)
36028eb6488eSEric Joyner 		fctrl |= (IXGBE_FCTRL_UPE | IXGBE_FCTRL_MPE);
36038eb6488eSEric Joyner 	else if (mcnt >= MAX_NUM_MULTICAST_ADDRESSES ||
3604ff06a8dbSJustin Hibbits 	    if_getflags(ifp) & IFF_ALLMULTI) {
36058eb6488eSEric Joyner 		fctrl |= IXGBE_FCTRL_MPE;
36068eb6488eSEric Joyner 		fctrl &= ~IXGBE_FCTRL_UPE;
36078eb6488eSEric Joyner 	} else
36088eb6488eSEric Joyner 		fctrl &= ~(IXGBE_FCTRL_UPE | IXGBE_FCTRL_MPE);
36098eb6488eSEric Joyner 
3610b1d5caf3SKevin Bowling 	IXGBE_WRITE_REG(&sc->hw, IXGBE_FCTRL, fctrl);
3611c19c7afeSEric Joyner } /* ixgbe_if_multi_set */
36128eb6488eSEric Joyner 
36138eb6488eSEric Joyner /************************************************************************
36148eb6488eSEric Joyner  * ixgbe_mc_array_itr
36158eb6488eSEric Joyner  *
36168eb6488eSEric Joyner  *   An iterator function needed by the multicast shared code.
36178eb6488eSEric Joyner  *   It feeds the shared code routine the addresses in the
36188eb6488eSEric Joyner  *   array of ixgbe_set_multi() one by one.
36198eb6488eSEric Joyner  ************************************************************************/
36208eb6488eSEric Joyner static u8 *
ixgbe_mc_array_itr(struct ixgbe_hw * hw,u8 ** update_ptr,u32 * vmdq)36218eb6488eSEric Joyner ixgbe_mc_array_itr(struct ixgbe_hw *hw, u8 **update_ptr, u32 *vmdq)
36228eb6488eSEric Joyner {
36238eb6488eSEric Joyner 	struct ixgbe_mc_addr *mta;
36248eb6488eSEric Joyner 
36258eb6488eSEric Joyner 	mta = (struct ixgbe_mc_addr *)*update_ptr;
36268eb6488eSEric Joyner 	*vmdq = mta->vmdq;
36278eb6488eSEric Joyner 
36288eb6488eSEric Joyner 	*update_ptr = (u8*)(mta + 1);
36298eb6488eSEric Joyner 
36308eb6488eSEric Joyner 	return (mta->addr);
36318eb6488eSEric Joyner } /* ixgbe_mc_array_itr */
36328eb6488eSEric Joyner 
36338eb6488eSEric Joyner /************************************************************************
36348eb6488eSEric Joyner  * ixgbe_local_timer - Timer routine
36358eb6488eSEric Joyner  *
36368eb6488eSEric Joyner  *   Checks for link status, updates statistics,
36378eb6488eSEric Joyner  *   and runs the watchdog check.
36388eb6488eSEric Joyner  ************************************************************************/
36398eb6488eSEric Joyner static void
ixgbe_if_timer(if_ctx_t ctx,uint16_t qid)3640c19c7afeSEric Joyner ixgbe_if_timer(if_ctx_t ctx, uint16_t qid)
36418eb6488eSEric Joyner {
3642b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
36438eb6488eSEric Joyner 
3644c19c7afeSEric Joyner 	if (qid != 0)
3645c19c7afeSEric Joyner 		return;
36468eb6488eSEric Joyner 
36478eb6488eSEric Joyner 	/* Check for pluggable optics */
3648b1d5caf3SKevin Bowling 	if (sc->sfp_probe)
3649c19c7afeSEric Joyner 		if (!ixgbe_sfp_probe(ctx))
3650c19c7afeSEric Joyner 			return; /* Nothing to do */
36518eb6488eSEric Joyner 
3652b1d5caf3SKevin Bowling 	ixgbe_check_link(&sc->hw, &sc->link_speed, &sc->link_up, 0);
36538eb6488eSEric Joyner 
3654c19c7afeSEric Joyner 	/* Fire off the adminq task */
3655c19c7afeSEric Joyner 	iflib_admin_intr_deferred(ctx);
36568eb6488eSEric Joyner 
3657c19c7afeSEric Joyner } /* ixgbe_if_timer */
36588eb6488eSEric Joyner 
36598eb6488eSEric Joyner /************************************************************************
3660f72de14eSKevin Bowling  * ixgbe_fw_mode_timer - FW mode timer routine
3661f72de14eSKevin Bowling  ************************************************************************/
3662f72de14eSKevin Bowling static void
ixgbe_fw_mode_timer(void * arg)3663f72de14eSKevin Bowling ixgbe_fw_mode_timer(void *arg)
3664f72de14eSKevin Bowling {
3665f72de14eSKevin Bowling 	struct ixgbe_softc *sc = arg;
3666f72de14eSKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
3667f72de14eSKevin Bowling 
3668f72de14eSKevin Bowling 	if (ixgbe_fw_recovery_mode(hw)) {
3669f72de14eSKevin Bowling 		if (atomic_cmpset_acq_int(&sc->recovery_mode, 0, 1)) {
3670f72de14eSKevin Bowling 			/* Firmware error detected, entering recovery mode */
3671c58d34ddSKevin Bowling 			device_printf(sc->dev,
3672c58d34ddSKevin Bowling 			    "Firmware recovery mode detected. Limiting"
3673c58d34ddSKevin Bowling 			    " functionality. Refer to the Intel(R) Ethernet"
3674c58d34ddSKevin Bowling 			    " Adapters and Devices User Guide for details on"
3675c58d34ddSKevin Bowling 			    " firmware recovery mode.\n");
3676f72de14eSKevin Bowling 
3677f72de14eSKevin Bowling 			if (hw->adapter_stopped == FALSE)
3678f72de14eSKevin Bowling 				ixgbe_if_stop(sc->ctx);
3679f72de14eSKevin Bowling 		}
3680f72de14eSKevin Bowling 	} else
3681f72de14eSKevin Bowling 		atomic_cmpset_acq_int(&sc->recovery_mode, 1, 0);
3682f72de14eSKevin Bowling 
3683f72de14eSKevin Bowling 
3684f72de14eSKevin Bowling 	callout_reset(&sc->fw_mode_timer, hz,
3685f72de14eSKevin Bowling 	    ixgbe_fw_mode_timer, sc);
3686f72de14eSKevin Bowling } /* ixgbe_fw_mode_timer */
3687f72de14eSKevin Bowling 
3688f72de14eSKevin Bowling /************************************************************************
36898eb6488eSEric Joyner  * ixgbe_sfp_probe
36908eb6488eSEric Joyner  *
36918eb6488eSEric Joyner  *   Determine if a port had optics inserted.
36928eb6488eSEric Joyner  ************************************************************************/
36938eb6488eSEric Joyner static bool
ixgbe_sfp_probe(if_ctx_t ctx)3694c19c7afeSEric Joyner ixgbe_sfp_probe(if_ctx_t ctx)
36958eb6488eSEric Joyner {
3696b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
3697b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
3698c19c7afeSEric Joyner 	device_t dev = iflib_get_dev(ctx);
369979b36ec9SKevin Bowling 	bool result = false;
37008eb6488eSEric Joyner 
37018eb6488eSEric Joyner 	if ((hw->phy.type == ixgbe_phy_nl) &&
37028eb6488eSEric Joyner 	    (hw->phy.sfp_type == ixgbe_sfp_type_not_present)) {
37038eb6488eSEric Joyner 		s32 ret = hw->phy.ops.identify_sfp(hw);
37048eb6488eSEric Joyner 		if (ret)
37058eb6488eSEric Joyner 			goto out;
37068eb6488eSEric Joyner 		ret = hw->phy.ops.reset(hw);
3707b1d5caf3SKevin Bowling 		sc->sfp_probe = false;
37088eb6488eSEric Joyner 		if (ret == IXGBE_ERR_SFP_NOT_SUPPORTED) {
3709c58d34ddSKevin Bowling 			device_printf(dev,
3710c58d34ddSKevin Bowling 			    "Unsupported SFP+ module detected!");
37118eb6488eSEric Joyner 			device_printf(dev,
37128eb6488eSEric Joyner 			    "Reload driver with supported module.\n");
37138eb6488eSEric Joyner 			goto out;
37148eb6488eSEric Joyner 		} else
37158eb6488eSEric Joyner 			device_printf(dev, "SFP+ module detected!\n");
37168eb6488eSEric Joyner 		/* We now have supported optics */
371779b36ec9SKevin Bowling 		result = true;
37188eb6488eSEric Joyner 	}
37198eb6488eSEric Joyner out:
37208eb6488eSEric Joyner 
37218eb6488eSEric Joyner 	return (result);
37228eb6488eSEric Joyner } /* ixgbe_sfp_probe */
37238eb6488eSEric Joyner 
37248eb6488eSEric Joyner /************************************************************************
37258eb6488eSEric Joyner  * ixgbe_handle_mod - Tasklet for SFP module interrupts
37268eb6488eSEric Joyner  ************************************************************************/
37278eb6488eSEric Joyner static void
ixgbe_handle_mod(void * context)3728c19c7afeSEric Joyner ixgbe_handle_mod(void *context)
37298eb6488eSEric Joyner {
3730c19c7afeSEric Joyner 	if_ctx_t ctx = context;
3731b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
3732b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
3733c19c7afeSEric Joyner 	device_t dev = iflib_get_dev(ctx);
37348eb6488eSEric Joyner 	u32 err, cage_full = 0;
37358eb6488eSEric Joyner 
3736b1d5caf3SKevin Bowling 	if (sc->hw.need_crosstalk_fix) {
37378eb6488eSEric Joyner 		switch (hw->mac.type) {
37388eb6488eSEric Joyner 		case ixgbe_mac_82599EB:
37398eb6488eSEric Joyner 			cage_full = IXGBE_READ_REG(hw, IXGBE_ESDP) &
37408eb6488eSEric Joyner 			    IXGBE_ESDP_SDP2;
37418eb6488eSEric Joyner 			break;
37428eb6488eSEric Joyner 		case ixgbe_mac_X550EM_x:
37438eb6488eSEric Joyner 		case ixgbe_mac_X550EM_a:
37448eb6488eSEric Joyner 			cage_full = IXGBE_READ_REG(hw, IXGBE_ESDP) &
37458eb6488eSEric Joyner 			    IXGBE_ESDP_SDP0;
37468eb6488eSEric Joyner 			break;
37478eb6488eSEric Joyner 		default:
37488eb6488eSEric Joyner 			break;
37498eb6488eSEric Joyner 		}
37508eb6488eSEric Joyner 
37518eb6488eSEric Joyner 		if (!cage_full)
3752c19c7afeSEric Joyner 			goto handle_mod_out;
37538eb6488eSEric Joyner 	}
37548eb6488eSEric Joyner 
37558eb6488eSEric Joyner 	err = hw->phy.ops.identify_sfp(hw);
37568eb6488eSEric Joyner 	if (err == IXGBE_ERR_SFP_NOT_SUPPORTED) {
37578eb6488eSEric Joyner 		device_printf(dev,
37588eb6488eSEric Joyner 		    "Unsupported SFP+ module type was detected.\n");
3759c19c7afeSEric Joyner 		goto handle_mod_out;
37608eb6488eSEric Joyner 	}
37618eb6488eSEric Joyner 
3762c19c7afeSEric Joyner 	if (hw->mac.type == ixgbe_mac_82598EB)
3763c19c7afeSEric Joyner 		err = hw->phy.ops.reset(hw);
3764c19c7afeSEric Joyner 	else
37658eb6488eSEric Joyner 		err = hw->mac.ops.setup_sfp(hw);
3766c19c7afeSEric Joyner 
37678eb6488eSEric Joyner 	if (err == IXGBE_ERR_SFP_NOT_SUPPORTED) {
37688eb6488eSEric Joyner 		device_printf(dev,
37698eb6488eSEric Joyner 		    "Setup failure - unsupported SFP+ module type.\n");
3770c19c7afeSEric Joyner 		goto handle_mod_out;
37718eb6488eSEric Joyner 	}
3772b1d5caf3SKevin Bowling 	sc->task_requests |= IXGBE_REQUEST_TASK_MSF;
3773c19c7afeSEric Joyner 	return;
3774c19c7afeSEric Joyner 
3775c19c7afeSEric Joyner handle_mod_out:
3776b1d5caf3SKevin Bowling 	sc->task_requests &= ~(IXGBE_REQUEST_TASK_MSF);
37778eb6488eSEric Joyner } /* ixgbe_handle_mod */
37788eb6488eSEric Joyner 
37798eb6488eSEric Joyner 
37808eb6488eSEric Joyner /************************************************************************
37818eb6488eSEric Joyner  * ixgbe_handle_msf - Tasklet for MSF (multispeed fiber) interrupts
37828eb6488eSEric Joyner  ************************************************************************/
37838eb6488eSEric Joyner static void
ixgbe_handle_msf(void * context)3784c19c7afeSEric Joyner ixgbe_handle_msf(void *context)
37858eb6488eSEric Joyner {
3786c19c7afeSEric Joyner 	if_ctx_t ctx = context;
3787b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
3788b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
37898eb6488eSEric Joyner 	u32 autoneg;
37908eb6488eSEric Joyner 	bool negotiate;
37918eb6488eSEric Joyner 
37928eb6488eSEric Joyner 	/* get_supported_phy_layer will call hw->phy.ops.identify_sfp() */
3793b1d5caf3SKevin Bowling 	sc->phy_layer = ixgbe_get_supported_physical_layer(hw);
37948eb6488eSEric Joyner 
37958eb6488eSEric Joyner 	autoneg = hw->phy.autoneg_advertised;
37968eb6488eSEric Joyner 	if ((!autoneg) && (hw->mac.ops.get_link_capabilities))
37978eb6488eSEric Joyner 		hw->mac.ops.get_link_capabilities(hw, &autoneg, &negotiate);
37988eb6488eSEric Joyner 	if (hw->mac.ops.setup_link)
379979b36ec9SKevin Bowling 		hw->mac.ops.setup_link(hw, autoneg, true);
38008eb6488eSEric Joyner 
38018eb6488eSEric Joyner 	/* Adjust media types shown in ifconfig */
3802b1d5caf3SKevin Bowling 	ifmedia_removeall(sc->media);
3803b1d5caf3SKevin Bowling 	ixgbe_add_media_types(sc->ctx);
3804b1d5caf3SKevin Bowling 	ifmedia_set(sc->media, IFM_ETHER | IFM_AUTO);
38058eb6488eSEric Joyner } /* ixgbe_handle_msf */
38068eb6488eSEric Joyner 
38078eb6488eSEric Joyner /************************************************************************
38088eb6488eSEric Joyner  * ixgbe_handle_phy - Tasklet for external PHY interrupts
38098eb6488eSEric Joyner  ************************************************************************/
38108eb6488eSEric Joyner static void
ixgbe_handle_phy(void * context)3811c19c7afeSEric Joyner ixgbe_handle_phy(void *context)
38128eb6488eSEric Joyner {
3813c19c7afeSEric Joyner 	if_ctx_t ctx = context;
3814b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
3815b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
38168eb6488eSEric Joyner 	int error;
38178eb6488eSEric Joyner 
38188eb6488eSEric Joyner 	error = hw->phy.ops.handle_lasi(hw);
38198eb6488eSEric Joyner 	if (error == IXGBE_ERR_OVERTEMP)
3820c58d34ddSKevin Bowling 		device_printf(sc->dev,
3821c58d34ddSKevin Bowling 		    "CRITICAL: EXTERNAL PHY OVER TEMP!!"
3822c58d34ddSKevin Bowling 		    "  PHY will downshift to lower power state!\n");
38238eb6488eSEric Joyner 	else if (error)
3824b1d5caf3SKevin Bowling 		device_printf(sc->dev,
38258eb6488eSEric Joyner 		    "Error handling LASI interrupt: %d\n", error);
38268eb6488eSEric Joyner } /* ixgbe_handle_phy */
38278eb6488eSEric Joyner 
38288eb6488eSEric Joyner /************************************************************************
3829c19c7afeSEric Joyner  * ixgbe_if_stop - Stop the hardware
38308eb6488eSEric Joyner  *
38318eb6488eSEric Joyner  *   Disables all traffic on the adapter by issuing a
38328eb6488eSEric Joyner  *   global reset on the MAC and deallocates TX/RX buffers.
38338eb6488eSEric Joyner  ************************************************************************/
38348eb6488eSEric Joyner static void
ixgbe_if_stop(if_ctx_t ctx)3835c19c7afeSEric Joyner ixgbe_if_stop(if_ctx_t ctx)
38368eb6488eSEric Joyner {
3837b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
3838b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
38398eb6488eSEric Joyner 
3840c19c7afeSEric Joyner 	INIT_DEBUGOUT("ixgbe_if_stop: begin\n");
38418eb6488eSEric Joyner 
38428eb6488eSEric Joyner 	ixgbe_reset_hw(hw);
384379b36ec9SKevin Bowling 	hw->adapter_stopped = false;
38448eb6488eSEric Joyner 	ixgbe_stop_adapter(hw);
38458eb6488eSEric Joyner 	if (hw->mac.type == ixgbe_mac_82599EB)
38468eb6488eSEric Joyner 		ixgbe_stop_mac_link_on_d3_82599(hw);
38478eb6488eSEric Joyner 	/* Turn off the laser - noop with no optics */
38488eb6488eSEric Joyner 	ixgbe_disable_tx_laser(hw);
38498eb6488eSEric Joyner 
38508eb6488eSEric Joyner 	/* Update the stack */
3851b1d5caf3SKevin Bowling 	sc->link_up = false;
3852c19c7afeSEric Joyner 	ixgbe_if_update_admin_status(ctx);
38538eb6488eSEric Joyner 
38548eb6488eSEric Joyner 	/* reprogram the RAR[0] in case user changed it. */
3855b1d5caf3SKevin Bowling 	ixgbe_set_rar(&sc->hw, 0, sc->hw.mac.addr, 0, IXGBE_RAH_AV);
38568eb6488eSEric Joyner 
38578eb6488eSEric Joyner 	return;
3858c19c7afeSEric Joyner } /* ixgbe_if_stop */
38598eb6488eSEric Joyner 
38608eb6488eSEric Joyner /************************************************************************
38618eb6488eSEric Joyner  * ixgbe_update_link_status - Update OS on link state
38628eb6488eSEric Joyner  *
38638eb6488eSEric Joyner  * Note: Only updates the OS on the cached link state.
38648eb6488eSEric Joyner  *       The real check of the hardware only happens with
38658eb6488eSEric Joyner  *       a link interrupt.
38668eb6488eSEric Joyner  ************************************************************************/
38678eb6488eSEric Joyner static void
ixgbe_if_update_admin_status(if_ctx_t ctx)3868c19c7afeSEric Joyner ixgbe_if_update_admin_status(if_ctx_t ctx)
38698eb6488eSEric Joyner {
3870b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
3871c19c7afeSEric Joyner 	device_t dev = iflib_get_dev(ctx);
38728eb6488eSEric Joyner 
3873b1d5caf3SKevin Bowling 	if (sc->link_up) {
3874b1d5caf3SKevin Bowling 		if (sc->link_active == false) {
38758eb6488eSEric Joyner 			if (bootverbose)
38768eb6488eSEric Joyner 				device_printf(dev, "Link is up %d Gbps %s \n",
3877b1d5caf3SKevin Bowling 				    ((sc->link_speed == 128) ? 10 : 1),
38788eb6488eSEric Joyner 				    "Full Duplex");
3879b1d5caf3SKevin Bowling 			sc->link_active = true;
38808eb6488eSEric Joyner 			/* Update any Flow Control changes */
3881b1d5caf3SKevin Bowling 			ixgbe_fc_enable(&sc->hw);
38828eb6488eSEric Joyner 			/* Update DMA coalescing config */
3883b1d5caf3SKevin Bowling 			ixgbe_config_dmac(sc);
3884a0302c92SPiotr Pietruszewski 			iflib_link_state_change(ctx, LINK_STATE_UP,
38855ddb1aa3SPiotr Kubaj 			    ixgbe_link_speed_to_baudrate(sc->link_speed));
3886c19c7afeSEric Joyner 
3887b1d5caf3SKevin Bowling 			if (sc->feat_en & IXGBE_FEATURE_SRIOV)
3888b1d5caf3SKevin Bowling 				ixgbe_ping_all_vfs(sc);
38898eb6488eSEric Joyner 		}
38908eb6488eSEric Joyner 	} else { /* Link down */
3891b1d5caf3SKevin Bowling 		if (sc->link_active == true) {
38928eb6488eSEric Joyner 			if (bootverbose)
38938eb6488eSEric Joyner 				device_printf(dev, "Link is Down\n");
3894c19c7afeSEric Joyner 			iflib_link_state_change(ctx, LINK_STATE_DOWN, 0);
3895b1d5caf3SKevin Bowling 			sc->link_active = false;
3896b1d5caf3SKevin Bowling 			if (sc->feat_en & IXGBE_FEATURE_SRIOV)
3897b1d5caf3SKevin Bowling 				ixgbe_ping_all_vfs(sc);
38988eb6488eSEric Joyner 		}
38998eb6488eSEric Joyner 	}
39008eb6488eSEric Joyner 
3901b2c1e8e6SEric Joyner 	/* Handle task requests from msix_link() */
3902b1d5caf3SKevin Bowling 	if (sc->task_requests & IXGBE_REQUEST_TASK_MOD)
3903b2c1e8e6SEric Joyner 		ixgbe_handle_mod(ctx);
3904b1d5caf3SKevin Bowling 	if (sc->task_requests & IXGBE_REQUEST_TASK_MSF)
3905b2c1e8e6SEric Joyner 		ixgbe_handle_msf(ctx);
3906b1d5caf3SKevin Bowling 	if (sc->task_requests & IXGBE_REQUEST_TASK_MBX)
3907b2c1e8e6SEric Joyner 		ixgbe_handle_mbx(ctx);
3908b1d5caf3SKevin Bowling 	if (sc->task_requests & IXGBE_REQUEST_TASK_FDIR)
3909b2c1e8e6SEric Joyner 		ixgbe_reinit_fdir(ctx);
3910b1d5caf3SKevin Bowling 	if (sc->task_requests & IXGBE_REQUEST_TASK_PHY)
3911b2c1e8e6SEric Joyner 		ixgbe_handle_phy(ctx);
3912b1d5caf3SKevin Bowling 	sc->task_requests = 0;
3913c19c7afeSEric Joyner 
3914b1d5caf3SKevin Bowling 	ixgbe_update_stats_counters(sc);
3915c19c7afeSEric Joyner } /* ixgbe_if_update_admin_status */
39168eb6488eSEric Joyner 
39178eb6488eSEric Joyner /************************************************************************
39188eb6488eSEric Joyner  * ixgbe_config_dmac - Configure DMA Coalescing
39198eb6488eSEric Joyner  ************************************************************************/
39208eb6488eSEric Joyner static void
ixgbe_config_dmac(struct ixgbe_softc * sc)3921b1d5caf3SKevin Bowling ixgbe_config_dmac(struct ixgbe_softc *sc)
39228eb6488eSEric Joyner {
3923b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
39248eb6488eSEric Joyner 	struct ixgbe_dmac_config *dcfg = &hw->mac.dmac_config;
39258eb6488eSEric Joyner 
39268eb6488eSEric Joyner 	if (hw->mac.type < ixgbe_mac_X550 || !hw->mac.ops.dmac_config)
39278eb6488eSEric Joyner 		return;
39288eb6488eSEric Joyner 
3929b1d5caf3SKevin Bowling 	if (dcfg->watchdog_timer ^ sc->dmac ||
3930b1d5caf3SKevin Bowling 	    dcfg->link_speed ^ sc->link_speed) {
3931b1d5caf3SKevin Bowling 		dcfg->watchdog_timer = sc->dmac;
393279b36ec9SKevin Bowling 		dcfg->fcoe_en = false;
3933b1d5caf3SKevin Bowling 		dcfg->link_speed = sc->link_speed;
39348eb6488eSEric Joyner 		dcfg->num_tcs = 1;
39358eb6488eSEric Joyner 
39368eb6488eSEric Joyner 		INIT_DEBUGOUT2("dmac settings: watchdog %d, link speed %d\n",
39378eb6488eSEric Joyner 		    dcfg->watchdog_timer, dcfg->link_speed);
39388eb6488eSEric Joyner 
39398eb6488eSEric Joyner 		hw->mac.ops.dmac_config(hw);
39408eb6488eSEric Joyner 	}
39418eb6488eSEric Joyner } /* ixgbe_config_dmac */
39428eb6488eSEric Joyner 
39438eb6488eSEric Joyner /************************************************************************
3944c19c7afeSEric Joyner  * ixgbe_if_enable_intr
39458eb6488eSEric Joyner  ************************************************************************/
3946c19c7afeSEric Joyner void
ixgbe_if_enable_intr(if_ctx_t ctx)3947c19c7afeSEric Joyner ixgbe_if_enable_intr(if_ctx_t ctx)
39488eb6488eSEric Joyner {
3949b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
3950b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
3951b1d5caf3SKevin Bowling 	struct ix_rx_queue *que = sc->rx_queues;
39528eb6488eSEric Joyner 	u32 mask, fwsm;
39538eb6488eSEric Joyner 
39548eb6488eSEric Joyner 	mask = (IXGBE_EIMS_ENABLE_MASK & ~IXGBE_EIMS_RTX_QUEUE);
39558eb6488eSEric Joyner 
3956b1d5caf3SKevin Bowling 	switch (sc->hw.mac.type) {
39578eb6488eSEric Joyner 	case ixgbe_mac_82599EB:
39588eb6488eSEric Joyner 		mask |= IXGBE_EIMS_ECC;
3959b1d5caf3SKevin Bowling 		/* Temperature sensor on some scs */
39608eb6488eSEric Joyner 		mask |= IXGBE_EIMS_GPI_SDP0;
39618eb6488eSEric Joyner 		/* SFP+ (RX_LOS_N & MOD_ABS_N) */
39628eb6488eSEric Joyner 		mask |= IXGBE_EIMS_GPI_SDP1;
39638eb6488eSEric Joyner 		mask |= IXGBE_EIMS_GPI_SDP2;
39648eb6488eSEric Joyner 		break;
39658eb6488eSEric Joyner 	case ixgbe_mac_X540:
39668eb6488eSEric Joyner 		/* Detect if Thermal Sensor is enabled */
39678eb6488eSEric Joyner 		fwsm = IXGBE_READ_REG(hw, IXGBE_FWSM);
39688eb6488eSEric Joyner 		if (fwsm & IXGBE_FWSM_TS_ENABLED)
39698eb6488eSEric Joyner 			mask |= IXGBE_EIMS_TS;
39708eb6488eSEric Joyner 		mask |= IXGBE_EIMS_ECC;
39718eb6488eSEric Joyner 		break;
39728eb6488eSEric Joyner 	case ixgbe_mac_X550:
39738eb6488eSEric Joyner 		/* MAC thermal sensor is automatically enabled */
39748eb6488eSEric Joyner 		mask |= IXGBE_EIMS_TS;
39758eb6488eSEric Joyner 		mask |= IXGBE_EIMS_ECC;
39768eb6488eSEric Joyner 		break;
39778eb6488eSEric Joyner 	case ixgbe_mac_X550EM_x:
39788eb6488eSEric Joyner 	case ixgbe_mac_X550EM_a:
39798eb6488eSEric Joyner 		/* Some devices use SDP0 for important information */
39808eb6488eSEric Joyner 		if (hw->device_id == IXGBE_DEV_ID_X550EM_X_SFP ||
39818eb6488eSEric Joyner 		    hw->device_id == IXGBE_DEV_ID_X550EM_A_SFP ||
39828eb6488eSEric Joyner 		    hw->device_id == IXGBE_DEV_ID_X550EM_A_SFP_N ||
39838eb6488eSEric Joyner 		    hw->device_id == IXGBE_DEV_ID_X550EM_X_10G_T)
39848eb6488eSEric Joyner 			mask |= IXGBE_EIMS_GPI_SDP0_BY_MAC(hw);
39858eb6488eSEric Joyner 		if (hw->phy.type == ixgbe_phy_x550em_ext_t)
39868eb6488eSEric Joyner 			mask |= IXGBE_EICR_GPI_SDP0_X540;
39878eb6488eSEric Joyner 		mask |= IXGBE_EIMS_ECC;
39888eb6488eSEric Joyner 		break;
39898eb6488eSEric Joyner 	default:
39908eb6488eSEric Joyner 		break;
39918eb6488eSEric Joyner 	}
39928eb6488eSEric Joyner 
39938eb6488eSEric Joyner 	/* Enable Fan Failure detection */
3994b1d5caf3SKevin Bowling 	if (sc->feat_en & IXGBE_FEATURE_FAN_FAIL)
39958eb6488eSEric Joyner 		mask |= IXGBE_EIMS_GPI_SDP1;
39968eb6488eSEric Joyner 	/* Enable SR-IOV */
3997b1d5caf3SKevin Bowling 	if (sc->feat_en & IXGBE_FEATURE_SRIOV)
39988eb6488eSEric Joyner 		mask |= IXGBE_EIMS_MAILBOX;
39998eb6488eSEric Joyner 	/* Enable Flow Director */
4000b1d5caf3SKevin Bowling 	if (sc->feat_en & IXGBE_FEATURE_FDIR)
40018eb6488eSEric Joyner 		mask |= IXGBE_EIMS_FLOW_DIR;
40028eb6488eSEric Joyner 
40038eb6488eSEric Joyner 	IXGBE_WRITE_REG(hw, IXGBE_EIMS, mask);
40048eb6488eSEric Joyner 
40058eb6488eSEric Joyner 	/* With MSI-X we use auto clear */
4006b1d5caf3SKevin Bowling 	if (sc->intr_type == IFLIB_INTR_MSIX) {
40078eb6488eSEric Joyner 		mask = IXGBE_EIMS_ENABLE_MASK;
40088eb6488eSEric Joyner 		/* Don't autoclear Link */
40098eb6488eSEric Joyner 		mask &= ~IXGBE_EIMS_OTHER;
40108eb6488eSEric Joyner 		mask &= ~IXGBE_EIMS_LSC;
4011b1d5caf3SKevin Bowling 		if (sc->feat_cap & IXGBE_FEATURE_SRIOV)
40128eb6488eSEric Joyner 			mask &= ~IXGBE_EIMS_MAILBOX;
40138eb6488eSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_EIAC, mask);
40146f37f232SEric Joyner 	}
40156f37f232SEric Joyner 
4016758cc3dcSJack F Vogel 	/*
40178eb6488eSEric Joyner 	 * Now enable all queues, this is done separately to
40188eb6488eSEric Joyner 	 * allow for handling the extended (beyond 32) MSI-X
40198eb6488eSEric Joyner 	 * vectors that can be used by 82599
4020758cc3dcSJack F Vogel 	 */
4021b1d5caf3SKevin Bowling 	for (int i = 0; i < sc->num_rx_queues; i++, que++)
4022b1d5caf3SKevin Bowling 		ixgbe_enable_queue(sc, que->msix);
40238eb6488eSEric Joyner 
40248eb6488eSEric Joyner 	IXGBE_WRITE_FLUSH(hw);
40258eb6488eSEric Joyner 
4026c19c7afeSEric Joyner } /* ixgbe_if_enable_intr */
40278eb6488eSEric Joyner 
40288eb6488eSEric Joyner /************************************************************************
40298eb6488eSEric Joyner  * ixgbe_disable_intr
40308eb6488eSEric Joyner  ************************************************************************/
4031758cc3dcSJack F Vogel static void
ixgbe_if_disable_intr(if_ctx_t ctx)4032c19c7afeSEric Joyner ixgbe_if_disable_intr(if_ctx_t ctx)
4033758cc3dcSJack F Vogel {
4034b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
4035c19c7afeSEric Joyner 
4036b1d5caf3SKevin Bowling 	if (sc->intr_type == IFLIB_INTR_MSIX)
4037b1d5caf3SKevin Bowling 		IXGBE_WRITE_REG(&sc->hw, IXGBE_EIAC, 0);
4038b1d5caf3SKevin Bowling 	if (sc->hw.mac.type == ixgbe_mac_82598EB) {
4039b1d5caf3SKevin Bowling 		IXGBE_WRITE_REG(&sc->hw, IXGBE_EIMC, ~0);
40408eb6488eSEric Joyner 	} else {
4041b1d5caf3SKevin Bowling 		IXGBE_WRITE_REG(&sc->hw, IXGBE_EIMC, 0xFFFF0000);
4042b1d5caf3SKevin Bowling 		IXGBE_WRITE_REG(&sc->hw, IXGBE_EIMC_EX(0), ~0);
4043b1d5caf3SKevin Bowling 		IXGBE_WRITE_REG(&sc->hw, IXGBE_EIMC_EX(1), ~0);
40448eb6488eSEric Joyner 	}
4045b1d5caf3SKevin Bowling 	IXGBE_WRITE_FLUSH(&sc->hw);
4046758cc3dcSJack F Vogel 
4047c19c7afeSEric Joyner } /* ixgbe_if_disable_intr */
40488eb6488eSEric Joyner 
40498eb6488eSEric Joyner /************************************************************************
4050b2c1e8e6SEric Joyner  * ixgbe_link_intr_enable
4051b2c1e8e6SEric Joyner  ************************************************************************/
4052b2c1e8e6SEric Joyner static void
ixgbe_link_intr_enable(if_ctx_t ctx)4053b2c1e8e6SEric Joyner ixgbe_link_intr_enable(if_ctx_t ctx)
4054b2c1e8e6SEric Joyner {
4055c58d34ddSKevin Bowling 	struct ixgbe_hw *hw =
4056c58d34ddSKevin Bowling 	    &((struct ixgbe_softc *)iflib_get_softc(ctx))->hw;
4057b2c1e8e6SEric Joyner 
4058b2c1e8e6SEric Joyner 	/* Re-enable other interrupts */
4059b2c1e8e6SEric Joyner 	IXGBE_WRITE_REG(hw, IXGBE_EIMS, IXGBE_EIMS_OTHER | IXGBE_EIMS_LSC);
4060b2c1e8e6SEric Joyner } /* ixgbe_link_intr_enable */
4061b2c1e8e6SEric Joyner 
4062b2c1e8e6SEric Joyner /************************************************************************
4063c19c7afeSEric Joyner  * ixgbe_if_rx_queue_intr_enable
4064c19c7afeSEric Joyner  ************************************************************************/
4065c19c7afeSEric Joyner static int
ixgbe_if_rx_queue_intr_enable(if_ctx_t ctx,uint16_t rxqid)4066c19c7afeSEric Joyner ixgbe_if_rx_queue_intr_enable(if_ctx_t ctx, uint16_t rxqid)
4067c19c7afeSEric Joyner {
4068b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
4069b1d5caf3SKevin Bowling 	struct ix_rx_queue *que = &sc->rx_queues[rxqid];
4070c19c7afeSEric Joyner 
4071b1d5caf3SKevin Bowling 	ixgbe_enable_queue(sc, que->msix);
4072c19c7afeSEric Joyner 
4073c19c7afeSEric Joyner 	return (0);
4074c19c7afeSEric Joyner } /* ixgbe_if_rx_queue_intr_enable */
4075c19c7afeSEric Joyner 
4076c19c7afeSEric Joyner /************************************************************************
4077c19c7afeSEric Joyner  * ixgbe_enable_queue
40788eb6488eSEric Joyner  ************************************************************************/
40798eb6488eSEric Joyner static void
ixgbe_enable_queue(struct ixgbe_softc * sc,u32 vector)4080b1d5caf3SKevin Bowling ixgbe_enable_queue(struct ixgbe_softc *sc, u32 vector)
40818eb6488eSEric Joyner {
4082b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
40832dc2d580SEric Joyner 	u64 queue = 1ULL << vector;
4084c19c7afeSEric Joyner 	u32 mask;
4085758cc3dcSJack F Vogel 
4086c19c7afeSEric Joyner 	if (hw->mac.type == ixgbe_mac_82598EB) {
4087c19c7afeSEric Joyner 		mask = (IXGBE_EIMS_RTX_QUEUE & queue);
4088c19c7afeSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_EIMS, mask);
4089c19c7afeSEric Joyner 	} else {
4090c19c7afeSEric Joyner 		mask = (queue & 0xFFFFFFFF);
4091c19c7afeSEric Joyner 		if (mask)
4092c19c7afeSEric Joyner 			IXGBE_WRITE_REG(hw, IXGBE_EIMS_EX(0), mask);
4093c19c7afeSEric Joyner 		mask = (queue >> 32);
4094c19c7afeSEric Joyner 		if (mask)
4095c19c7afeSEric Joyner 			IXGBE_WRITE_REG(hw, IXGBE_EIMS_EX(1), mask);
4096c19c7afeSEric Joyner 	}
4097c19c7afeSEric Joyner } /* ixgbe_enable_queue */
4098c19c7afeSEric Joyner 
4099c19c7afeSEric Joyner /************************************************************************
4100c19c7afeSEric Joyner  * ixgbe_disable_queue
4101c19c7afeSEric Joyner  ************************************************************************/
4102c19c7afeSEric Joyner static void
ixgbe_disable_queue(struct ixgbe_softc * sc,u32 vector)4103b1d5caf3SKevin Bowling ixgbe_disable_queue(struct ixgbe_softc *sc, u32 vector)
4104c19c7afeSEric Joyner {
4105b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
41062dc2d580SEric Joyner 	u64 queue = 1ULL << vector;
4107c19c7afeSEric Joyner 	u32 mask;
4108c19c7afeSEric Joyner 
4109c19c7afeSEric Joyner 	if (hw->mac.type == ixgbe_mac_82598EB) {
4110c19c7afeSEric Joyner 		mask = (IXGBE_EIMS_RTX_QUEUE & queue);
4111c19c7afeSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_EIMC, mask);
4112c19c7afeSEric Joyner 	} else {
4113c19c7afeSEric Joyner 		mask = (queue & 0xFFFFFFFF);
4114c19c7afeSEric Joyner 		if (mask)
4115c19c7afeSEric Joyner 			IXGBE_WRITE_REG(hw, IXGBE_EIMC_EX(0), mask);
4116c19c7afeSEric Joyner 		mask = (queue >> 32);
4117c19c7afeSEric Joyner 		if (mask)
4118c19c7afeSEric Joyner 			IXGBE_WRITE_REG(hw, IXGBE_EIMC_EX(1), mask);
4119c19c7afeSEric Joyner 	}
4120c19c7afeSEric Joyner } /* ixgbe_disable_queue */
4121c19c7afeSEric Joyner 
4122c19c7afeSEric Joyner /************************************************************************
4123c19c7afeSEric Joyner  * ixgbe_intr - Legacy Interrupt Service Routine
4124c19c7afeSEric Joyner  ************************************************************************/
4125c19c7afeSEric Joyner int
ixgbe_intr(void * arg)4126c19c7afeSEric Joyner ixgbe_intr(void *arg)
4127c19c7afeSEric Joyner {
4128b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = arg;
4129b1d5caf3SKevin Bowling 	struct ix_rx_queue *que = sc->rx_queues;
4130b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
4131b1d5caf3SKevin Bowling 	if_ctx_t ctx = sc->ctx;
4132c19c7afeSEric Joyner 	u32 eicr, eicr_mask;
4133758cc3dcSJack F Vogel 
41348eb6488eSEric Joyner 	eicr = IXGBE_READ_REG(hw, IXGBE_EICR);
4135758cc3dcSJack F Vogel 
41368eb6488eSEric Joyner 	++que->irqs;
41378eb6488eSEric Joyner 	if (eicr == 0) {
4138c19c7afeSEric Joyner 		ixgbe_if_enable_intr(ctx);
4139c19c7afeSEric Joyner 		return (FILTER_HANDLED);
4140758cc3dcSJack F Vogel 	}
4141758cc3dcSJack F Vogel 
41428eb6488eSEric Joyner 	/* Check for fan failure */
4143f72de14eSKevin Bowling 	if ((sc->feat_en & IXGBE_FEATURE_FAN_FAIL) &&
4144c19c7afeSEric Joyner 	    (eicr & IXGBE_EICR_GPI_SDP1)) {
4145b1d5caf3SKevin Bowling 		device_printf(sc->dev,
4146c19c7afeSEric Joyner 		    "\nCRITICAL: FAN FAILURE!! REPLACE IMMEDIATELY!!\n");
4147c58d34ddSKevin Bowling 		IXGBE_WRITE_REG(hw, IXGBE_EIMS,
4148c58d34ddSKevin Bowling 		    IXGBE_EICR_GPI_SDP1_BY_MAC(hw));
4149758cc3dcSJack F Vogel 	}
4150758cc3dcSJack F Vogel 
41518eb6488eSEric Joyner 	/* Link status change */
4152c19c7afeSEric Joyner 	if (eicr & IXGBE_EICR_LSC) {
4153c19c7afeSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_EIMC, IXGBE_EIMC_LSC);
4154c19c7afeSEric Joyner 		iflib_admin_intr_deferred(ctx);
4155c19c7afeSEric Joyner 	}
41568eb6488eSEric Joyner 
41578eb6488eSEric Joyner 	if (ixgbe_is_sfp(hw)) {
41588eb6488eSEric Joyner 		/* Pluggable optics-related interrupt */
41598eb6488eSEric Joyner 		if (hw->mac.type >= ixgbe_mac_X540)
41608eb6488eSEric Joyner 			eicr_mask = IXGBE_EICR_GPI_SDP0_X540;
41618eb6488eSEric Joyner 		else
41628eb6488eSEric Joyner 			eicr_mask = IXGBE_EICR_GPI_SDP2_BY_MAC(hw);
41638eb6488eSEric Joyner 
41648eb6488eSEric Joyner 		if (eicr & eicr_mask) {
41658eb6488eSEric Joyner 			IXGBE_WRITE_REG(hw, IXGBE_EICR, eicr_mask);
4166b1d5caf3SKevin Bowling 			sc->task_requests |= IXGBE_REQUEST_TASK_MOD;
41678eb6488eSEric Joyner 		}
41688eb6488eSEric Joyner 
41698eb6488eSEric Joyner 		if ((hw->mac.type == ixgbe_mac_82599EB) &&
41708eb6488eSEric Joyner 		    (eicr & IXGBE_EICR_GPI_SDP1_BY_MAC(hw))) {
41718eb6488eSEric Joyner 			IXGBE_WRITE_REG(hw, IXGBE_EICR,
41728eb6488eSEric Joyner 			    IXGBE_EICR_GPI_SDP1_BY_MAC(hw));
4173b1d5caf3SKevin Bowling 			sc->task_requests |= IXGBE_REQUEST_TASK_MSF;
41748eb6488eSEric Joyner 		}
41758eb6488eSEric Joyner 	}
41768eb6488eSEric Joyner 
41778eb6488eSEric Joyner 	/* External PHY interrupt */
41788eb6488eSEric Joyner 	if ((hw->phy.type == ixgbe_phy_x550em_ext_t) &&
41798eb6488eSEric Joyner 	    (eicr & IXGBE_EICR_GPI_SDP0_X540))
4180b1d5caf3SKevin Bowling 		sc->task_requests |= IXGBE_REQUEST_TASK_PHY;
41818eb6488eSEric Joyner 
4182c19c7afeSEric Joyner 	return (FILTER_SCHEDULE_THREAD);
4183c19c7afeSEric Joyner } /* ixgbe_intr */
41848eb6488eSEric Joyner 
41858eb6488eSEric Joyner /************************************************************************
41868eb6488eSEric Joyner  * ixgbe_free_pci_resources
41878eb6488eSEric Joyner  ************************************************************************/
41888eb6488eSEric Joyner static void
ixgbe_free_pci_resources(if_ctx_t ctx)4189c19c7afeSEric Joyner ixgbe_free_pci_resources(if_ctx_t ctx)
41908eb6488eSEric Joyner {
4191b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
4192b1d5caf3SKevin Bowling 	struct ix_rx_queue *que = sc->rx_queues;
4193c19c7afeSEric Joyner 	device_t dev = iflib_get_dev(ctx);
41948eb6488eSEric Joyner 
4195b97de13aSMarius Strobl 	/* Release all MSI-X queue resources */
4196b1d5caf3SKevin Bowling 	if (sc->intr_type == IFLIB_INTR_MSIX)
4197b1d5caf3SKevin Bowling 		iflib_irq_free(ctx, &sc->irq);
4198c19c7afeSEric Joyner 
4199c19c7afeSEric Joyner 	if (que != NULL) {
4200b1d5caf3SKevin Bowling 		for (int i = 0; i < sc->num_rx_queues; i++, que++) {
4201c19c7afeSEric Joyner 			iflib_irq_free(ctx, &que->que_irq);
4202c19c7afeSEric Joyner 		}
4203c19c7afeSEric Joyner 	}
42048eb6488eSEric Joyner 
4205b1d5caf3SKevin Bowling 	if (sc->pci_mem != NULL)
4206c19c7afeSEric Joyner 		bus_release_resource(dev, SYS_RES_MEMORY,
4207b1d5caf3SKevin Bowling 		    rman_get_rid(sc->pci_mem), sc->pci_mem);
42088eb6488eSEric Joyner } /* ixgbe_free_pci_resources */
42098eb6488eSEric Joyner 
42108eb6488eSEric Joyner /************************************************************************
42118eb6488eSEric Joyner  * ixgbe_sysctl_flowcntl
42128eb6488eSEric Joyner  *
42138eb6488eSEric Joyner  *   SYSCTL wrapper around setting Flow Control
42148eb6488eSEric Joyner  ************************************************************************/
4215758cc3dcSJack F Vogel static int
ixgbe_sysctl_flowcntl(SYSCTL_HANDLER_ARGS)4216f2c4db54SSteven Hartland ixgbe_sysctl_flowcntl(SYSCTL_HANDLER_ARGS)
4217758cc3dcSJack F Vogel {
4218b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc;
42198eb6488eSEric Joyner 	int error, fc;
4220758cc3dcSJack F Vogel 
4221b1d5caf3SKevin Bowling 	sc = (struct ixgbe_softc *)arg1;
4222b1d5caf3SKevin Bowling 	fc = sc->hw.fc.current_mode;
4223f2c4db54SSteven Hartland 
4224f2c4db54SSteven Hartland 	error = sysctl_handle_int(oidp, &fc, 0, req);
4225758cc3dcSJack F Vogel 	if ((error) || (req->newptr == NULL))
4226758cc3dcSJack F Vogel 		return (error);
4227758cc3dcSJack F Vogel 
4228758cc3dcSJack F Vogel 	/* Don't bother if it's not changed */
4229b1d5caf3SKevin Bowling 	if (fc == sc->hw.fc.current_mode)
4230758cc3dcSJack F Vogel 		return (0);
4231758cc3dcSJack F Vogel 
4232b1d5caf3SKevin Bowling 	return ixgbe_set_flowcntl(sc, fc);
42338eb6488eSEric Joyner } /* ixgbe_sysctl_flowcntl */
4234f2c4db54SSteven Hartland 
42358eb6488eSEric Joyner /************************************************************************
42368eb6488eSEric Joyner  * ixgbe_set_flowcntl - Set flow control
42378eb6488eSEric Joyner  *
42388eb6488eSEric Joyner  *   Flow control values:
42398eb6488eSEric Joyner  *     0 - off
42408eb6488eSEric Joyner  *     1 - rx pause
42418eb6488eSEric Joyner  *     2 - tx pause
42428eb6488eSEric Joyner  *     3 - full
42438eb6488eSEric Joyner  ************************************************************************/
4244f2c4db54SSteven Hartland static int
ixgbe_set_flowcntl(struct ixgbe_softc * sc,int fc)4245b1d5caf3SKevin Bowling ixgbe_set_flowcntl(struct ixgbe_softc *sc, int fc)
4246f2c4db54SSteven Hartland {
4247f2c4db54SSteven Hartland 	switch (fc) {
4248758cc3dcSJack F Vogel 	case ixgbe_fc_rx_pause:
4249758cc3dcSJack F Vogel 	case ixgbe_fc_tx_pause:
4250758cc3dcSJack F Vogel 	case ixgbe_fc_full:
4251b1d5caf3SKevin Bowling 		sc->hw.fc.requested_mode = fc;
4252b1d5caf3SKevin Bowling 		if (sc->num_rx_queues > 1)
4253b1d5caf3SKevin Bowling 			ixgbe_disable_rx_drop(sc);
4254758cc3dcSJack F Vogel 		break;
4255758cc3dcSJack F Vogel 	case ixgbe_fc_none:
4256b1d5caf3SKevin Bowling 		sc->hw.fc.requested_mode = ixgbe_fc_none;
4257b1d5caf3SKevin Bowling 		if (sc->num_rx_queues > 1)
4258b1d5caf3SKevin Bowling 			ixgbe_enable_rx_drop(sc);
4259758cc3dcSJack F Vogel 		break;
4260758cc3dcSJack F Vogel 	default:
4261758cc3dcSJack F Vogel 		return (EINVAL);
4262758cc3dcSJack F Vogel 	}
42638eb6488eSEric Joyner 
4264758cc3dcSJack F Vogel 	/* Don't autoneg if forcing a value */
4265b1d5caf3SKevin Bowling 	sc->hw.fc.disable_fc_autoneg = true;
4266b1d5caf3SKevin Bowling 	ixgbe_fc_enable(&sc->hw);
42678eb6488eSEric Joyner 
4268f2c4db54SSteven Hartland 	return (0);
42698eb6488eSEric Joyner } /* ixgbe_set_flowcntl */
42708eb6488eSEric Joyner 
42718eb6488eSEric Joyner /************************************************************************
42728eb6488eSEric Joyner  * ixgbe_enable_rx_drop
42738eb6488eSEric Joyner  *
42748eb6488eSEric Joyner  *   Enable the hardware to drop packets when the buffer is
42758eb6488eSEric Joyner  *   full. This is useful with multiqueue, so that no single
42768eb6488eSEric Joyner  *   queue being full stalls the entire RX engine. We only
42778eb6488eSEric Joyner  *   enable this when Multiqueue is enabled AND Flow Control
42788eb6488eSEric Joyner  *   is disabled.
42798eb6488eSEric Joyner  ************************************************************************/
42808eb6488eSEric Joyner static void
ixgbe_enable_rx_drop(struct ixgbe_softc * sc)4281b1d5caf3SKevin Bowling ixgbe_enable_rx_drop(struct ixgbe_softc *sc)
42828eb6488eSEric Joyner {
4283b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
42848eb6488eSEric Joyner 	struct rx_ring *rxr;
42858eb6488eSEric Joyner 	u32 srrctl;
42868eb6488eSEric Joyner 
4287b1d5caf3SKevin Bowling 	for (int i = 0; i < sc->num_rx_queues; i++) {
4288b1d5caf3SKevin Bowling 		rxr = &sc->rx_queues[i].rxr;
42898eb6488eSEric Joyner 		srrctl = IXGBE_READ_REG(hw, IXGBE_SRRCTL(rxr->me));
42908eb6488eSEric Joyner 		srrctl |= IXGBE_SRRCTL_DROP_EN;
42918eb6488eSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_SRRCTL(rxr->me), srrctl);
4292758cc3dcSJack F Vogel 	}
4293758cc3dcSJack F Vogel 
42948eb6488eSEric Joyner 	/* enable drop for each vf */
4295b1d5caf3SKevin Bowling 	for (int i = 0; i < sc->num_vfs; i++) {
42968eb6488eSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_QDE,
4297c58d34ddSKevin Bowling 		    (IXGBE_QDE_WRITE |
4298c58d34ddSKevin Bowling 		    (i << IXGBE_QDE_IDX_SHIFT) |
42998eb6488eSEric Joyner 		    IXGBE_QDE_ENABLE));
43008eb6488eSEric Joyner 	}
43018eb6488eSEric Joyner } /* ixgbe_enable_rx_drop */
43028eb6488eSEric Joyner 
43038eb6488eSEric Joyner /************************************************************************
43048eb6488eSEric Joyner  * ixgbe_disable_rx_drop
43058eb6488eSEric Joyner  ************************************************************************/
43068eb6488eSEric Joyner static void
ixgbe_disable_rx_drop(struct ixgbe_softc * sc)4307b1d5caf3SKevin Bowling ixgbe_disable_rx_drop(struct ixgbe_softc *sc)
43088eb6488eSEric Joyner {
4309b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
43108eb6488eSEric Joyner 	struct rx_ring *rxr;
43118eb6488eSEric Joyner 	u32 srrctl;
43128eb6488eSEric Joyner 
4313b1d5caf3SKevin Bowling 	for (int i = 0; i < sc->num_rx_queues; i++) {
4314b1d5caf3SKevin Bowling 		rxr = &sc->rx_queues[i].rxr;
43158eb6488eSEric Joyner 		srrctl = IXGBE_READ_REG(hw, IXGBE_SRRCTL(rxr->me));
43168eb6488eSEric Joyner 		srrctl &= ~IXGBE_SRRCTL_DROP_EN;
43178eb6488eSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_SRRCTL(rxr->me), srrctl);
43188eb6488eSEric Joyner 	}
43198eb6488eSEric Joyner 
43208eb6488eSEric Joyner 	/* disable drop for each vf */
4321b1d5caf3SKevin Bowling 	for (int i = 0; i < sc->num_vfs; i++) {
43228eb6488eSEric Joyner 		IXGBE_WRITE_REG(hw, IXGBE_QDE,
43238eb6488eSEric Joyner 		    (IXGBE_QDE_WRITE | (i << IXGBE_QDE_IDX_SHIFT)));
43248eb6488eSEric Joyner 	}
43258eb6488eSEric Joyner } /* ixgbe_disable_rx_drop */
43268eb6488eSEric Joyner 
43278eb6488eSEric Joyner /************************************************************************
43288eb6488eSEric Joyner  * ixgbe_sysctl_advertise
43298eb6488eSEric Joyner  *
43308eb6488eSEric Joyner  *   SYSCTL wrapper around setting advertised speed
43318eb6488eSEric Joyner  ************************************************************************/
4332758cc3dcSJack F Vogel static int
ixgbe_sysctl_advertise(SYSCTL_HANDLER_ARGS)4333f2c4db54SSteven Hartland ixgbe_sysctl_advertise(SYSCTL_HANDLER_ARGS)
4334758cc3dcSJack F Vogel {
4335b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc;
43368eb6488eSEric Joyner 	int error, advertise;
4337758cc3dcSJack F Vogel 
4338b1d5caf3SKevin Bowling 	sc = (struct ixgbe_softc *)arg1;
4339f72de14eSKevin Bowling 	if (atomic_load_acq_int(&sc->recovery_mode))
4340f72de14eSKevin Bowling 		return (EPERM);
4341f72de14eSKevin Bowling 
4342b1d5caf3SKevin Bowling 	advertise = sc->advertise;
4343758cc3dcSJack F Vogel 
4344f2c4db54SSteven Hartland 	error = sysctl_handle_int(oidp, &advertise, 0, req);
4345758cc3dcSJack F Vogel 	if ((error) || (req->newptr == NULL))
4346758cc3dcSJack F Vogel 		return (error);
4347758cc3dcSJack F Vogel 
4348b1d5caf3SKevin Bowling 	return ixgbe_set_advertise(sc, advertise);
43498eb6488eSEric Joyner } /* ixgbe_sysctl_advertise */
4350f2c4db54SSteven Hartland 
43518eb6488eSEric Joyner /************************************************************************
43528eb6488eSEric Joyner  * ixgbe_set_advertise - Control advertised link speed
43538eb6488eSEric Joyner  *
43548eb6488eSEric Joyner  *   Flags:
43558eb6488eSEric Joyner  *     0x1  - advertise 100 Mb
43568eb6488eSEric Joyner  *     0x2  - advertise 1G
43578eb6488eSEric Joyner  *     0x4  - advertise 10G
43588eb6488eSEric Joyner  *     0x8  - advertise 10 Mb (yes, Mb)
4359d381c807SPiotr Pietruszewski  *     0x10 - advertise 2.5G (disabled by default)
4360d381c807SPiotr Pietruszewski  *     0x20 - advertise 5G (disabled by default)
4361d381c807SPiotr Pietruszewski  *
43628eb6488eSEric Joyner  ************************************************************************/
4363f2c4db54SSteven Hartland static int
ixgbe_set_advertise(struct ixgbe_softc * sc,int advertise)4364b1d5caf3SKevin Bowling ixgbe_set_advertise(struct ixgbe_softc *sc, int advertise)
4365f2c4db54SSteven Hartland {
4366b1d5caf3SKevin Bowling 	device_t dev = iflib_get_dev(sc->ctx);
4367f2c4db54SSteven Hartland 	struct ixgbe_hw *hw;
43688eb6488eSEric Joyner 	ixgbe_link_speed speed = 0;
43698eb6488eSEric Joyner 	ixgbe_link_speed link_caps = 0;
43708eb6488eSEric Joyner 	s32 err = IXGBE_NOT_IMPLEMENTED;
437179b36ec9SKevin Bowling 	bool negotiate = false;
4372f2c4db54SSteven Hartland 
4373d775d23aSSteven Hartland 	/* Checks to validate new value */
4374b1d5caf3SKevin Bowling 	if (sc->advertise == advertise) /* no change */
4375d775d23aSSteven Hartland 		return (0);
4376d775d23aSSteven Hartland 
4377b1d5caf3SKevin Bowling 	hw = &sc->hw;
4378f2c4db54SSteven Hartland 
4379a9ca1c79SSean Bruno 	/* No speed changes for backplane media */
4380a9ca1c79SSean Bruno 	if (hw->phy.media_type == ixgbe_media_type_backplane)
4381a9ca1c79SSean Bruno 		return (ENODEV);
4382a9ca1c79SSean Bruno 
4383758cc3dcSJack F Vogel 	if (!((hw->phy.media_type == ixgbe_media_type_copper) ||
4384758cc3dcSJack F Vogel 	      (hw->phy.multispeed_fiber))) {
4385c58d34ddSKevin Bowling 		device_printf(dev,
4386c58d34ddSKevin Bowling 		    "Advertised speed can only be set on copper or multispeed"
4387c58d34ddSKevin Bowling 		    " fiber media types.\n");
4388758cc3dcSJack F Vogel 		return (EINVAL);
4389758cc3dcSJack F Vogel 	}
4390758cc3dcSJack F Vogel 
4391d381c807SPiotr Pietruszewski 	if (advertise < 0x1 || advertise > 0x3F) {
4392c58d34ddSKevin Bowling 		device_printf(dev,
4393c58d34ddSKevin Bowling 		    "Invalid advertised speed; valid modes are 0x1 through"
4394c58d34ddSKevin Bowling 		    " 0x3F\n");
4395758cc3dcSJack F Vogel 		return (EINVAL);
4396758cc3dcSJack F Vogel 	}
4397758cc3dcSJack F Vogel 
43988eb6488eSEric Joyner 	if (hw->mac.ops.get_link_capabilities) {
43998eb6488eSEric Joyner 		err = hw->mac.ops.get_link_capabilities(hw, &link_caps,
44008eb6488eSEric Joyner 		    &negotiate);
44018eb6488eSEric Joyner 		if (err != IXGBE_SUCCESS) {
4402c58d34ddSKevin Bowling 			device_printf(dev,
4403c58d34ddSKevin Bowling 			    "Unable to determine supported advertise speeds"
4404c58d34ddSKevin Bowling 			    "\n");
44058eb6488eSEric Joyner 			return (ENODEV);
44068eb6488eSEric Joyner 		}
4407758cc3dcSJack F Vogel 	}
4408758cc3dcSJack F Vogel 
4409758cc3dcSJack F Vogel 	/* Set new value and report new advertised mode */
44108eb6488eSEric Joyner 	if (advertise & 0x1) {
44118eb6488eSEric Joyner 		if (!(link_caps & IXGBE_LINK_SPEED_100_FULL)) {
4412c58d34ddSKevin Bowling 			device_printf(dev,
4413c58d34ddSKevin Bowling 			    "Interface does not support 100Mb advertised"
4414c58d34ddSKevin Bowling 			    " speed\n");
44158eb6488eSEric Joyner 			return (EINVAL);
44168eb6488eSEric Joyner 		}
4417758cc3dcSJack F Vogel 		speed |= IXGBE_LINK_SPEED_100_FULL;
44188eb6488eSEric Joyner 	}
44198eb6488eSEric Joyner 	if (advertise & 0x2) {
44208eb6488eSEric Joyner 		if (!(link_caps & IXGBE_LINK_SPEED_1GB_FULL)) {
4421c58d34ddSKevin Bowling 			device_printf(dev,
4422c58d34ddSKevin Bowling 			    "Interface does not support 1Gb advertised speed"
4423c58d34ddSKevin Bowling 			    "\n");
44248eb6488eSEric Joyner 			return (EINVAL);
44258eb6488eSEric Joyner 		}
4426758cc3dcSJack F Vogel 		speed |= IXGBE_LINK_SPEED_1GB_FULL;
44278eb6488eSEric Joyner 	}
44288eb6488eSEric Joyner 	if (advertise & 0x4) {
44298eb6488eSEric Joyner 		if (!(link_caps & IXGBE_LINK_SPEED_10GB_FULL)) {
4430c58d34ddSKevin Bowling 			device_printf(dev,
4431c58d34ddSKevin Bowling 			    "Interface does not support 10Gb advertised speed"
4432c58d34ddSKevin Bowling 			    "\n");
44338eb6488eSEric Joyner 			return (EINVAL);
44348eb6488eSEric Joyner 		}
4435758cc3dcSJack F Vogel 		speed |= IXGBE_LINK_SPEED_10GB_FULL;
44368eb6488eSEric Joyner 	}
44378eb6488eSEric Joyner 	if (advertise & 0x8) {
44388eb6488eSEric Joyner 		if (!(link_caps & IXGBE_LINK_SPEED_10_FULL)) {
4439c58d34ddSKevin Bowling 			device_printf(dev,
4440c58d34ddSKevin Bowling 			    "Interface does not support 10Mb advertised speed"
4441c58d34ddSKevin Bowling 			    "\n");
44428eb6488eSEric Joyner 			return (EINVAL);
44438eb6488eSEric Joyner 		}
44448eb6488eSEric Joyner 		speed |= IXGBE_LINK_SPEED_10_FULL;
44458eb6488eSEric Joyner 	}
4446d381c807SPiotr Pietruszewski 	if (advertise & 0x10) {
4447d381c807SPiotr Pietruszewski 		if (!(link_caps & IXGBE_LINK_SPEED_2_5GB_FULL)) {
4448c58d34ddSKevin Bowling 			device_printf(dev,
4449c58d34ddSKevin Bowling 			    "Interface does not support 2.5G advertised speed"
4450c58d34ddSKevin Bowling 			    "\n");
4451d381c807SPiotr Pietruszewski 			return (EINVAL);
4452d381c807SPiotr Pietruszewski 		}
4453d381c807SPiotr Pietruszewski 		speed |= IXGBE_LINK_SPEED_2_5GB_FULL;
4454d381c807SPiotr Pietruszewski 	}
4455d381c807SPiotr Pietruszewski 	if (advertise & 0x20) {
4456d381c807SPiotr Pietruszewski 		if (!(link_caps & IXGBE_LINK_SPEED_5GB_FULL)) {
4457c58d34ddSKevin Bowling 			device_printf(dev,
4458c58d34ddSKevin Bowling 			    "Interface does not support 5G advertised speed"
4459c58d34ddSKevin Bowling 			    "\n");
4460d381c807SPiotr Pietruszewski 			return (EINVAL);
4461d381c807SPiotr Pietruszewski 		}
4462d381c807SPiotr Pietruszewski 		speed |= IXGBE_LINK_SPEED_5GB_FULL;
4463d381c807SPiotr Pietruszewski 	}
4464758cc3dcSJack F Vogel 
446579b36ec9SKevin Bowling 	hw->mac.autotry_restart = true;
446679b36ec9SKevin Bowling 	hw->mac.ops.setup_link(hw, speed, true);
4467b1d5caf3SKevin Bowling 	sc->advertise = advertise;
4468758cc3dcSJack F Vogel 
4469f2c4db54SSteven Hartland 	return (0);
44708eb6488eSEric Joyner } /* ixgbe_set_advertise */
44718eb6488eSEric Joyner 
44728eb6488eSEric Joyner /************************************************************************
4473d381c807SPiotr Pietruszewski  * ixgbe_get_default_advertise - Get default advertised speed settings
44748eb6488eSEric Joyner  *
44758eb6488eSEric Joyner  *   Formatted for sysctl usage.
44768eb6488eSEric Joyner  *   Flags:
44778eb6488eSEric Joyner  *     0x1 - advertise 100 Mb
44788eb6488eSEric Joyner  *     0x2 - advertise 1G
44798eb6488eSEric Joyner  *     0x4 - advertise 10G
44808eb6488eSEric Joyner  *     0x8 - advertise 10 Mb (yes, Mb)
4481d381c807SPiotr Pietruszewski  *     0x10 - advertise 2.5G (disabled by default)
4482d381c807SPiotr Pietruszewski  *     0x20 - advertise 5G (disabled by default)
44838eb6488eSEric Joyner  ************************************************************************/
44848eb6488eSEric Joyner static int
ixgbe_get_default_advertise(struct ixgbe_softc * sc)4485d381c807SPiotr Pietruszewski ixgbe_get_default_advertise(struct ixgbe_softc *sc)
44868eb6488eSEric Joyner {
4487b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
44888eb6488eSEric Joyner 	int speed;
44898eb6488eSEric Joyner 	ixgbe_link_speed link_caps = 0;
44908eb6488eSEric Joyner 	s32 err;
449179b36ec9SKevin Bowling 	bool negotiate = false;
4492758cc3dcSJack F Vogel 
4493758cc3dcSJack F Vogel 	/*
44948eb6488eSEric Joyner 	 * Advertised speed means nothing unless it's copper or
44958eb6488eSEric Joyner 	 * multi-speed fiber
4496758cc3dcSJack F Vogel 	 */
44978eb6488eSEric Joyner 	if (!(hw->phy.media_type == ixgbe_media_type_copper) &&
44988eb6488eSEric Joyner 	    !(hw->phy.multispeed_fiber))
4499758cc3dcSJack F Vogel 		return (0);
4500758cc3dcSJack F Vogel 
45018eb6488eSEric Joyner 	err = hw->mac.ops.get_link_capabilities(hw, &link_caps, &negotiate);
45028eb6488eSEric Joyner 	if (err != IXGBE_SUCCESS)
45038eb6488eSEric Joyner 		return (0);
45048eb6488eSEric Joyner 
4505d381c807SPiotr Pietruszewski 	if (hw->mac.type == ixgbe_mac_X550) {
4506d381c807SPiotr Pietruszewski 		/*
4507d381c807SPiotr Pietruszewski 		 * 2.5G and 5G autonegotiation speeds on X550
4508d381c807SPiotr Pietruszewski 		 * are disabled by default due to reported
4509d381c807SPiotr Pietruszewski 		 * interoperability issues with some switches.
4510d381c807SPiotr Pietruszewski 		 */
4511d381c807SPiotr Pietruszewski 		link_caps &= ~(IXGBE_LINK_SPEED_2_5GB_FULL |
4512d381c807SPiotr Pietruszewski 		    IXGBE_LINK_SPEED_5GB_FULL);
4513d381c807SPiotr Pietruszewski 	}
4514d381c807SPiotr Pietruszewski 
45158eb6488eSEric Joyner 	speed =
4516d381c807SPiotr Pietruszewski 	    ((link_caps & IXGBE_LINK_SPEED_10GB_FULL)  ? 0x4  : 0) |
4517d381c807SPiotr Pietruszewski 	    ((link_caps & IXGBE_LINK_SPEED_5GB_FULL)   ? 0x20 : 0) |
4518d381c807SPiotr Pietruszewski 	    ((link_caps & IXGBE_LINK_SPEED_2_5GB_FULL) ? 0x10 : 0) |
4519d381c807SPiotr Pietruszewski 	    ((link_caps & IXGBE_LINK_SPEED_1GB_FULL)   ? 0x2  : 0) |
4520d381c807SPiotr Pietruszewski 	    ((link_caps & IXGBE_LINK_SPEED_100_FULL)   ? 0x1  : 0) |
4521d381c807SPiotr Pietruszewski 	    ((link_caps & IXGBE_LINK_SPEED_10_FULL)    ? 0x8  : 0);
45228eb6488eSEric Joyner 
45238eb6488eSEric Joyner 	return speed;
4524d381c807SPiotr Pietruszewski } /* ixgbe_get_default_advertise */
45258eb6488eSEric Joyner 
45268eb6488eSEric Joyner /************************************************************************
45278eb6488eSEric Joyner  * ixgbe_sysctl_dmac - Manage DMA Coalescing
45288eb6488eSEric Joyner  *
45298eb6488eSEric Joyner  *   Control values:
45308eb6488eSEric Joyner  *     0/1 - off / on (use default value of 1000)
45318eb6488eSEric Joyner  *
45328eb6488eSEric Joyner  *     Legal timer values are:
45338eb6488eSEric Joyner  *     50,100,250,500,1000,2000,5000,10000
45348eb6488eSEric Joyner  *
45358eb6488eSEric Joyner  *     Turning off interrupt moderation will also turn this off.
45368eb6488eSEric Joyner  ************************************************************************/
45376f37f232SEric Joyner static int
ixgbe_sysctl_dmac(SYSCTL_HANDLER_ARGS)45386f37f232SEric Joyner ixgbe_sysctl_dmac(SYSCTL_HANDLER_ARGS)
45396f37f232SEric Joyner {
4540b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = (struct ixgbe_softc *)arg1;
4541ff06a8dbSJustin Hibbits 	if_t ifp = iflib_get_ifp(sc->ctx);
45426f37f232SEric Joyner 	int error;
4543c19c7afeSEric Joyner 	u16 newval;
45446f37f232SEric Joyner 
4545b1d5caf3SKevin Bowling 	newval = sc->dmac;
4546c19c7afeSEric Joyner 	error = sysctl_handle_16(oidp, &newval, 0, req);
45476f37f232SEric Joyner 	if ((error) || (req->newptr == NULL))
45486f37f232SEric Joyner 		return (error);
45496f37f232SEric Joyner 
4550a9ca1c79SSean Bruno 	switch (newval) {
45516f37f232SEric Joyner 	case 0:
45526f37f232SEric Joyner 		/* Disabled */
4553b1d5caf3SKevin Bowling 		sc->dmac = 0;
45546f37f232SEric Joyner 		break;
4555a9ca1c79SSean Bruno 	case 1:
4556a9ca1c79SSean Bruno 		/* Enable and use default */
4557b1d5caf3SKevin Bowling 		sc->dmac = 1000;
45586f37f232SEric Joyner 		break;
45596f37f232SEric Joyner 	case 50:
45606f37f232SEric Joyner 	case 100:
45616f37f232SEric Joyner 	case 250:
45626f37f232SEric Joyner 	case 500:
45636f37f232SEric Joyner 	case 1000:
45646f37f232SEric Joyner 	case 2000:
45656f37f232SEric Joyner 	case 5000:
45666f37f232SEric Joyner 	case 10000:
45676f37f232SEric Joyner 		/* Legal values - allow */
4568b1d5caf3SKevin Bowling 		sc->dmac = newval;
45696f37f232SEric Joyner 		break;
45706f37f232SEric Joyner 	default:
45716f37f232SEric Joyner 		/* Do nothing, illegal value */
45726f37f232SEric Joyner 		return (EINVAL);
45736f37f232SEric Joyner 	}
45746f37f232SEric Joyner 
45756f37f232SEric Joyner 	/* Re-initialize hardware if it's already running */
4576ff06a8dbSJustin Hibbits 	if (if_getdrvflags(ifp) & IFF_DRV_RUNNING)
4577ff06a8dbSJustin Hibbits 		if_init(ifp, ifp);
45786f37f232SEric Joyner 
45796f37f232SEric Joyner 	return (0);
45808eb6488eSEric Joyner } /* ixgbe_sysctl_dmac */
45816f37f232SEric Joyner 
4582a9ca1c79SSean Bruno #ifdef IXGBE_DEBUG
45838eb6488eSEric Joyner /************************************************************************
45848eb6488eSEric Joyner  * ixgbe_sysctl_power_state
45858eb6488eSEric Joyner  *
4586a9ca1c79SSean Bruno  *   Sysctl to test power states
4587a9ca1c79SSean Bruno  *   Values:
4588a9ca1c79SSean Bruno  *     0      - set device to D0
4589a9ca1c79SSean Bruno  *     3      - set device to D3
4590a9ca1c79SSean Bruno  *     (none) - get current device power state
45918eb6488eSEric Joyner  ************************************************************************/
4592a9ca1c79SSean Bruno static int
ixgbe_sysctl_power_state(SYSCTL_HANDLER_ARGS)4593a9ca1c79SSean Bruno ixgbe_sysctl_power_state(SYSCTL_HANDLER_ARGS)
4594a9ca1c79SSean Bruno {
4595b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = (struct ixgbe_softc *)arg1;
4596b1d5caf3SKevin Bowling 	device_t dev = sc->dev;
4597a9ca1c79SSean Bruno 	int curr_ps, new_ps, error = 0;
4598a9ca1c79SSean Bruno 
4599a9ca1c79SSean Bruno 	curr_ps = new_ps = pci_get_powerstate(dev);
4600a9ca1c79SSean Bruno 
4601a9ca1c79SSean Bruno 	error = sysctl_handle_int(oidp, &new_ps, 0, req);
4602a9ca1c79SSean Bruno 	if ((error) || (req->newptr == NULL))
4603a9ca1c79SSean Bruno 		return (error);
4604a9ca1c79SSean Bruno 
4605a9ca1c79SSean Bruno 	if (new_ps == curr_ps)
4606a9ca1c79SSean Bruno 		return (0);
4607a9ca1c79SSean Bruno 
4608a9ca1c79SSean Bruno 	if (new_ps == 3 && curr_ps == 0)
4609a9ca1c79SSean Bruno 		error = DEVICE_SUSPEND(dev);
4610a9ca1c79SSean Bruno 	else if (new_ps == 0 && curr_ps == 3)
4611a9ca1c79SSean Bruno 		error = DEVICE_RESUME(dev);
4612a9ca1c79SSean Bruno 	else
4613a9ca1c79SSean Bruno 		return (EINVAL);
4614a9ca1c79SSean Bruno 
4615a9ca1c79SSean Bruno 	device_printf(dev, "New state: %d\n", pci_get_powerstate(dev));
4616a9ca1c79SSean Bruno 
4617a9ca1c79SSean Bruno 	return (error);
46188eb6488eSEric Joyner } /* ixgbe_sysctl_power_state */
4619a9ca1c79SSean Bruno #endif
46208eb6488eSEric Joyner 
46218eb6488eSEric Joyner /************************************************************************
46228eb6488eSEric Joyner  * ixgbe_sysctl_wol_enable
46238eb6488eSEric Joyner  *
46248eb6488eSEric Joyner  *   Sysctl to enable/disable the WoL capability,
46258eb6488eSEric Joyner  *   if supported by the adapter.
46268eb6488eSEric Joyner  *
46276f37f232SEric Joyner  *   Values:
46286f37f232SEric Joyner  *     0 - disabled
46296f37f232SEric Joyner  *     1 - enabled
46308eb6488eSEric Joyner  ************************************************************************/
46316f37f232SEric Joyner static int
ixgbe_sysctl_wol_enable(SYSCTL_HANDLER_ARGS)46326f37f232SEric Joyner ixgbe_sysctl_wol_enable(SYSCTL_HANDLER_ARGS)
46336f37f232SEric Joyner {
4634b1d5caf3SKevin Bowling 	struct ixgbe_softc  *sc = (struct ixgbe_softc *)arg1;
4635b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
46366f37f232SEric Joyner 	int new_wol_enabled;
46376f37f232SEric Joyner 	int error = 0;
46386f37f232SEric Joyner 
46396f37f232SEric Joyner 	new_wol_enabled = hw->wol_enabled;
46406f37f232SEric Joyner 	error = sysctl_handle_int(oidp, &new_wol_enabled, 0, req);
46416f37f232SEric Joyner 	if ((error) || (req->newptr == NULL))
46426f37f232SEric Joyner 		return (error);
4643a9ca1c79SSean Bruno 	new_wol_enabled = !!(new_wol_enabled);
46446f37f232SEric Joyner 	if (new_wol_enabled == hw->wol_enabled)
46456f37f232SEric Joyner 		return (0);
46466f37f232SEric Joyner 
4647b1d5caf3SKevin Bowling 	if (new_wol_enabled > 0 && !sc->wol_support)
46486f37f232SEric Joyner 		return (ENODEV);
46496f37f232SEric Joyner 	else
4650a9ca1c79SSean Bruno 		hw->wol_enabled = new_wol_enabled;
46516f37f232SEric Joyner 
46526f37f232SEric Joyner 	return (0);
46538eb6488eSEric Joyner } /* ixgbe_sysctl_wol_enable */
46546f37f232SEric Joyner 
46558eb6488eSEric Joyner /************************************************************************
46568eb6488eSEric Joyner  * ixgbe_sysctl_wufc - Wake Up Filter Control
46578eb6488eSEric Joyner  *
46586f37f232SEric Joyner  *   Sysctl to enable/disable the types of packets that the
46596f37f232SEric Joyner  *   adapter will wake up on upon receipt.
46606f37f232SEric Joyner  *   Flags:
46616f37f232SEric Joyner  *     0x1  - Link Status Change
46626f37f232SEric Joyner  *     0x2  - Magic Packet
46636f37f232SEric Joyner  *     0x4  - Direct Exact
46646f37f232SEric Joyner  *     0x8  - Directed Multicast
46656f37f232SEric Joyner  *     0x10 - Broadcast
46666f37f232SEric Joyner  *     0x20 - ARP/IPv4 Request Packet
46676f37f232SEric Joyner  *     0x40 - Direct IPv4 Packet
46686f37f232SEric Joyner  *     0x80 - Direct IPv6 Packet
46696f37f232SEric Joyner  *
46708eb6488eSEric Joyner  *   Settings not listed above will cause the sysctl to return an error.
46718eb6488eSEric Joyner  ************************************************************************/
46726f37f232SEric Joyner static int
ixgbe_sysctl_wufc(SYSCTL_HANDLER_ARGS)46736f37f232SEric Joyner ixgbe_sysctl_wufc(SYSCTL_HANDLER_ARGS)
46746f37f232SEric Joyner {
4675b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = (struct ixgbe_softc *)arg1;
46766f37f232SEric Joyner 	int error = 0;
46776f37f232SEric Joyner 	u32 new_wufc;
46786f37f232SEric Joyner 
4679b1d5caf3SKevin Bowling 	new_wufc = sc->wufc;
46806f37f232SEric Joyner 
4681c19c7afeSEric Joyner 	error = sysctl_handle_32(oidp, &new_wufc, 0, req);
46826f37f232SEric Joyner 	if ((error) || (req->newptr == NULL))
46836f37f232SEric Joyner 		return (error);
4684b1d5caf3SKevin Bowling 	if (new_wufc == sc->wufc)
46856f37f232SEric Joyner 		return (0);
46866f37f232SEric Joyner 
46876f37f232SEric Joyner 	if (new_wufc & 0xffffff00)
46886f37f232SEric Joyner 		return (EINVAL);
46898eb6488eSEric Joyner 
46906f37f232SEric Joyner 	new_wufc &= 0xff;
4691b1d5caf3SKevin Bowling 	new_wufc |= (0xffffff & sc->wufc);
4692b1d5caf3SKevin Bowling 	sc->wufc = new_wufc;
46936f37f232SEric Joyner 
46946f37f232SEric Joyner 	return (0);
46958eb6488eSEric Joyner } /* ixgbe_sysctl_wufc */
46966f37f232SEric Joyner 
4697a9ca1c79SSean Bruno #ifdef IXGBE_DEBUG
46988eb6488eSEric Joyner /************************************************************************
46998eb6488eSEric Joyner  * ixgbe_sysctl_print_rss_config
47008eb6488eSEric Joyner  ************************************************************************/
4701a9ca1c79SSean Bruno static int
ixgbe_sysctl_print_rss_config(SYSCTL_HANDLER_ARGS)4702a9ca1c79SSean Bruno ixgbe_sysctl_print_rss_config(SYSCTL_HANDLER_ARGS)
4703a9ca1c79SSean Bruno {
4704b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = (struct ixgbe_softc *)arg1;
4705b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
4706b1d5caf3SKevin Bowling 	device_t dev = sc->dev;
4707a9ca1c79SSean Bruno 	struct sbuf *buf;
47088eb6488eSEric Joyner 	int error = 0, reta_size;
4709a9ca1c79SSean Bruno 	u32 reg;
4710a9ca1c79SSean Bruno 
4711f72de14eSKevin Bowling 	if (atomic_load_acq_int(&sc->recovery_mode))
4712f72de14eSKevin Bowling 		return (EPERM);
4713f72de14eSKevin Bowling 
4714a9ca1c79SSean Bruno 	buf = sbuf_new_for_sysctl(NULL, NULL, 128, req);
4715a9ca1c79SSean Bruno 	if (!buf) {
4716a9ca1c79SSean Bruno 		device_printf(dev, "Could not allocate sbuf for output.\n");
4717a9ca1c79SSean Bruno 		return (ENOMEM);
4718a9ca1c79SSean Bruno 	}
4719a9ca1c79SSean Bruno 
4720a9ca1c79SSean Bruno 	// TODO: use sbufs to make a string to print out
4721a9ca1c79SSean Bruno 	/* Set multiplier for RETA setup and table size based on MAC */
4722b1d5caf3SKevin Bowling 	switch (sc->hw.mac.type) {
4723a9ca1c79SSean Bruno 	case ixgbe_mac_X550:
4724a9ca1c79SSean Bruno 	case ixgbe_mac_X550EM_x:
47258eb6488eSEric Joyner 	case ixgbe_mac_X550EM_a:
4726a9ca1c79SSean Bruno 		reta_size = 128;
4727a9ca1c79SSean Bruno 		break;
4728a9ca1c79SSean Bruno 	default:
4729a9ca1c79SSean Bruno 		reta_size = 32;
4730a9ca1c79SSean Bruno 		break;
4731a9ca1c79SSean Bruno 	}
4732a9ca1c79SSean Bruno 
4733a9ca1c79SSean Bruno 	/* Print out the redirection table */
4734a9ca1c79SSean Bruno 	sbuf_cat(buf, "\n");
4735a9ca1c79SSean Bruno 	for (int i = 0; i < reta_size; i++) {
4736a9ca1c79SSean Bruno 		if (i < 32) {
4737a9ca1c79SSean Bruno 			reg = IXGBE_READ_REG(hw, IXGBE_RETA(i));
4738a9ca1c79SSean Bruno 			sbuf_printf(buf, "RETA(%2d): 0x%08x\n", i, reg);
4739a9ca1c79SSean Bruno 		} else {
4740a9ca1c79SSean Bruno 			reg = IXGBE_READ_REG(hw, IXGBE_ERETA(i - 32));
4741a9ca1c79SSean Bruno 			sbuf_printf(buf, "ERETA(%2d): 0x%08x\n", i - 32, reg);
4742a9ca1c79SSean Bruno 		}
4743a9ca1c79SSean Bruno 	}
4744a9ca1c79SSean Bruno 
4745a9ca1c79SSean Bruno 	// TODO: print more config
4746a9ca1c79SSean Bruno 
4747a9ca1c79SSean Bruno 	error = sbuf_finish(buf);
4748a9ca1c79SSean Bruno 	if (error)
4749a9ca1c79SSean Bruno 		device_printf(dev, "Error finishing sbuf: %d\n", error);
4750a9ca1c79SSean Bruno 
4751a9ca1c79SSean Bruno 	sbuf_delete(buf);
47528eb6488eSEric Joyner 
4753a9ca1c79SSean Bruno 	return (0);
47548eb6488eSEric Joyner } /* ixgbe_sysctl_print_rss_config */
4755a9ca1c79SSean Bruno #endif /* IXGBE_DEBUG */
4756a9ca1c79SSean Bruno 
47578eb6488eSEric Joyner /************************************************************************
47588eb6488eSEric Joyner  * ixgbe_sysctl_phy_temp - Retrieve temperature of PHY
47598eb6488eSEric Joyner  *
47608eb6488eSEric Joyner  *   For X552/X557-AT devices using an external PHY
47618eb6488eSEric Joyner  ************************************************************************/
47628eb6488eSEric Joyner static int
ixgbe_sysctl_phy_temp(SYSCTL_HANDLER_ARGS)47638eb6488eSEric Joyner ixgbe_sysctl_phy_temp(SYSCTL_HANDLER_ARGS)
47648eb6488eSEric Joyner {
4765b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = (struct ixgbe_softc *)arg1;
4766b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
47678eb6488eSEric Joyner 	u16 reg;
47688eb6488eSEric Joyner 
4769f72de14eSKevin Bowling 	if (atomic_load_acq_int(&sc->recovery_mode))
4770f72de14eSKevin Bowling 		return (EPERM);
4771f72de14eSKevin Bowling 
47728eb6488eSEric Joyner 	if (hw->device_id != IXGBE_DEV_ID_X550EM_X_10G_T) {
4773b1d5caf3SKevin Bowling 		device_printf(iflib_get_dev(sc->ctx),
47748eb6488eSEric Joyner 		    "Device has no supported external thermal sensor.\n");
47758eb6488eSEric Joyner 		return (ENODEV);
47768eb6488eSEric Joyner 	}
47778eb6488eSEric Joyner 
47788eb6488eSEric Joyner 	if (hw->phy.ops.read_reg(hw, IXGBE_PHY_CURRENT_TEMP,
47798eb6488eSEric Joyner 	    IXGBE_MDIO_VENDOR_SPECIFIC_1_DEV_TYPE, &reg)) {
4780b1d5caf3SKevin Bowling 		device_printf(iflib_get_dev(sc->ctx),
4781c58d34ddSKevin Bowling 		    "Error reading from PHY's current temperature register"
4782c58d34ddSKevin Bowling 		    "\n");
47838eb6488eSEric Joyner 		return (EAGAIN);
47848eb6488eSEric Joyner 	}
47858eb6488eSEric Joyner 
47868eb6488eSEric Joyner 	/* Shift temp for output */
47878eb6488eSEric Joyner 	reg = reg >> 8;
47888eb6488eSEric Joyner 
4789c19c7afeSEric Joyner 	return (sysctl_handle_16(oidp, NULL, reg, req));
47908eb6488eSEric Joyner } /* ixgbe_sysctl_phy_temp */
47918eb6488eSEric Joyner 
47928eb6488eSEric Joyner /************************************************************************
47938eb6488eSEric Joyner  * ixgbe_sysctl_phy_overtemp_occurred
47948eb6488eSEric Joyner  *
47958eb6488eSEric Joyner  *   Reports (directly from the PHY) whether the current PHY
47968eb6488eSEric Joyner  *   temperature is over the overtemp threshold.
47978eb6488eSEric Joyner  ************************************************************************/
47988eb6488eSEric Joyner static int
ixgbe_sysctl_phy_overtemp_occurred(SYSCTL_HANDLER_ARGS)47998eb6488eSEric Joyner ixgbe_sysctl_phy_overtemp_occurred(SYSCTL_HANDLER_ARGS)
48008eb6488eSEric Joyner {
4801b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = (struct ixgbe_softc *)arg1;
4802b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
48038eb6488eSEric Joyner 	u16 reg;
48048eb6488eSEric Joyner 
4805f72de14eSKevin Bowling 	if (atomic_load_acq_int(&sc->recovery_mode))
4806f72de14eSKevin Bowling 		return (EPERM);
4807f72de14eSKevin Bowling 
48088eb6488eSEric Joyner 	if (hw->device_id != IXGBE_DEV_ID_X550EM_X_10G_T) {
4809b1d5caf3SKevin Bowling 		device_printf(iflib_get_dev(sc->ctx),
48108eb6488eSEric Joyner 		    "Device has no supported external thermal sensor.\n");
48118eb6488eSEric Joyner 		return (ENODEV);
48128eb6488eSEric Joyner 	}
48138eb6488eSEric Joyner 
48148eb6488eSEric Joyner 	if (hw->phy.ops.read_reg(hw, IXGBE_PHY_OVERTEMP_STATUS,
48158eb6488eSEric Joyner 	    IXGBE_MDIO_VENDOR_SPECIFIC_1_DEV_TYPE, &reg)) {
4816b1d5caf3SKevin Bowling 		device_printf(iflib_get_dev(sc->ctx),
48178eb6488eSEric Joyner 		    "Error reading from PHY's temperature status register\n");
48188eb6488eSEric Joyner 		return (EAGAIN);
48198eb6488eSEric Joyner 	}
48208eb6488eSEric Joyner 
48218eb6488eSEric Joyner 	/* Get occurrence bit */
48228eb6488eSEric Joyner 	reg = !!(reg & 0x4000);
48238eb6488eSEric Joyner 
4824c19c7afeSEric Joyner 	return (sysctl_handle_16(oidp, 0, reg, req));
48258eb6488eSEric Joyner } /* ixgbe_sysctl_phy_overtemp_occurred */
48268eb6488eSEric Joyner 
48278eb6488eSEric Joyner /************************************************************************
48288eb6488eSEric Joyner  * ixgbe_sysctl_eee_state
48298eb6488eSEric Joyner  *
48308eb6488eSEric Joyner  *   Sysctl to set EEE power saving feature
48318eb6488eSEric Joyner  *   Values:
48328eb6488eSEric Joyner  *     0      - disable EEE
48338eb6488eSEric Joyner  *     1      - enable EEE
48348eb6488eSEric Joyner  *     (none) - get current device EEE state
48358eb6488eSEric Joyner  ************************************************************************/
48368eb6488eSEric Joyner static int
ixgbe_sysctl_eee_state(SYSCTL_HANDLER_ARGS)48378eb6488eSEric Joyner ixgbe_sysctl_eee_state(SYSCTL_HANDLER_ARGS)
48388eb6488eSEric Joyner {
4839b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = (struct ixgbe_softc *)arg1;
4840b1d5caf3SKevin Bowling 	device_t dev = sc->dev;
4841ff06a8dbSJustin Hibbits 	if_t ifp = iflib_get_ifp(sc->ctx);
48428eb6488eSEric Joyner 	int curr_eee, new_eee, error = 0;
48438eb6488eSEric Joyner 	s32 retval;
48448eb6488eSEric Joyner 
4845f72de14eSKevin Bowling 	if (atomic_load_acq_int(&sc->recovery_mode))
4846f72de14eSKevin Bowling 		return (EPERM);
4847f72de14eSKevin Bowling 
4848b1d5caf3SKevin Bowling 	curr_eee = new_eee = !!(sc->feat_en & IXGBE_FEATURE_EEE);
48498eb6488eSEric Joyner 
48508eb6488eSEric Joyner 	error = sysctl_handle_int(oidp, &new_eee, 0, req);
48518eb6488eSEric Joyner 	if ((error) || (req->newptr == NULL))
48528eb6488eSEric Joyner 		return (error);
48538eb6488eSEric Joyner 
48548eb6488eSEric Joyner 	/* Nothing to do */
48558eb6488eSEric Joyner 	if (new_eee == curr_eee)
48568eb6488eSEric Joyner 		return (0);
48578eb6488eSEric Joyner 
48588eb6488eSEric Joyner 	/* Not supported */
4859b1d5caf3SKevin Bowling 	if (!(sc->feat_cap & IXGBE_FEATURE_EEE))
48608eb6488eSEric Joyner 		return (EINVAL);
48618eb6488eSEric Joyner 
48628eb6488eSEric Joyner 	/* Bounds checking */
48638eb6488eSEric Joyner 	if ((new_eee < 0) || (new_eee > 1))
48648eb6488eSEric Joyner 		return (EINVAL);
48658eb6488eSEric Joyner 
4866b1d5caf3SKevin Bowling 	retval = ixgbe_setup_eee(&sc->hw, new_eee);
48678eb6488eSEric Joyner 	if (retval) {
48688eb6488eSEric Joyner 		device_printf(dev, "Error in EEE setup: 0x%08X\n", retval);
48698eb6488eSEric Joyner 		return (EINVAL);
48708eb6488eSEric Joyner 	}
48718eb6488eSEric Joyner 
48728eb6488eSEric Joyner 	/* Restart auto-neg */
4873ff06a8dbSJustin Hibbits 	if_init(ifp, ifp);
48748eb6488eSEric Joyner 
48758eb6488eSEric Joyner 	device_printf(dev, "New EEE state: %d\n", new_eee);
48768eb6488eSEric Joyner 
48778eb6488eSEric Joyner 	/* Cache new value */
48788eb6488eSEric Joyner 	if (new_eee)
4879b1d5caf3SKevin Bowling 		sc->feat_en |= IXGBE_FEATURE_EEE;
48808eb6488eSEric Joyner 	else
4881b1d5caf3SKevin Bowling 		sc->feat_en &= ~IXGBE_FEATURE_EEE;
48828eb6488eSEric Joyner 
48838eb6488eSEric Joyner 	return (error);
48848eb6488eSEric Joyner } /* ixgbe_sysctl_eee_state */
48858eb6488eSEric Joyner 
4886eea2e089SMichael Tuexen static int
ixgbe_sysctl_tso_tcp_flags_mask(SYSCTL_HANDLER_ARGS)4887eea2e089SMichael Tuexen ixgbe_sysctl_tso_tcp_flags_mask(SYSCTL_HANDLER_ARGS)
4888eea2e089SMichael Tuexen {
4889eea2e089SMichael Tuexen 	struct ixgbe_softc *sc;
4890eea2e089SMichael Tuexen 	u32 reg, val, shift;
4891eea2e089SMichael Tuexen 	int error, mask;
4892eea2e089SMichael Tuexen 
4893eea2e089SMichael Tuexen 	sc = oidp->oid_arg1;
4894eea2e089SMichael Tuexen 	switch (oidp->oid_arg2) {
4895eea2e089SMichael Tuexen 	case 0:
4896eea2e089SMichael Tuexen 		reg = IXGBE_DTXTCPFLGL;
4897eea2e089SMichael Tuexen 		shift = 0;
4898eea2e089SMichael Tuexen 		break;
4899eea2e089SMichael Tuexen 	case 1:
4900eea2e089SMichael Tuexen 		reg = IXGBE_DTXTCPFLGL;
4901eea2e089SMichael Tuexen 		shift = 16;
4902eea2e089SMichael Tuexen 		break;
4903eea2e089SMichael Tuexen 	case 2:
4904eea2e089SMichael Tuexen 		reg = IXGBE_DTXTCPFLGH;
4905eea2e089SMichael Tuexen 		shift = 0;
4906eea2e089SMichael Tuexen 		break;
4907eea2e089SMichael Tuexen 	default:
4908eea2e089SMichael Tuexen 		return (EINVAL);
4909eea2e089SMichael Tuexen 		break;
4910eea2e089SMichael Tuexen 	}
4911eea2e089SMichael Tuexen 	val = IXGBE_READ_REG(&sc->hw, reg);
4912eea2e089SMichael Tuexen 	mask = (val >> shift) & 0xfff;
4913eea2e089SMichael Tuexen 	error = sysctl_handle_int(oidp, &mask, 0, req);
4914eea2e089SMichael Tuexen 	if (error != 0 || req->newptr == NULL)
4915eea2e089SMichael Tuexen 		return (error);
4916eea2e089SMichael Tuexen 	if (mask < 0 || mask > 0xfff)
4917eea2e089SMichael Tuexen 		return (EINVAL);
4918eea2e089SMichael Tuexen 	val = (val & ~(0xfff << shift)) | (mask << shift);
4919eea2e089SMichael Tuexen 	IXGBE_WRITE_REG(&sc->hw, reg, val);
4920eea2e089SMichael Tuexen 	return (0);
4921eea2e089SMichael Tuexen }
4922eea2e089SMichael Tuexen 
49238eb6488eSEric Joyner /************************************************************************
49248eb6488eSEric Joyner  * ixgbe_init_device_features
49258eb6488eSEric Joyner  ************************************************************************/
49268eb6488eSEric Joyner static void
ixgbe_init_device_features(struct ixgbe_softc * sc)4927b1d5caf3SKevin Bowling ixgbe_init_device_features(struct ixgbe_softc *sc)
49288eb6488eSEric Joyner {
4929c58d34ddSKevin Bowling 	sc->feat_cap = IXGBE_FEATURE_NETMAP |
4930c58d34ddSKevin Bowling 	    IXGBE_FEATURE_RSS |
4931c58d34ddSKevin Bowling 	    IXGBE_FEATURE_MSI |
4932c58d34ddSKevin Bowling 	    IXGBE_FEATURE_MSIX |
4933c58d34ddSKevin Bowling 	    IXGBE_FEATURE_LEGACY_IRQ;
49348eb6488eSEric Joyner 
49358eb6488eSEric Joyner 	/* Set capabilities first... */
4936b1d5caf3SKevin Bowling 	switch (sc->hw.mac.type) {
49378eb6488eSEric Joyner 	case ixgbe_mac_82598EB:
4938b1d5caf3SKevin Bowling 		if (sc->hw.device_id == IXGBE_DEV_ID_82598AT)
4939b1d5caf3SKevin Bowling 			sc->feat_cap |= IXGBE_FEATURE_FAN_FAIL;
49408eb6488eSEric Joyner 		break;
49418eb6488eSEric Joyner 	case ixgbe_mac_X540:
4942b1d5caf3SKevin Bowling 		sc->feat_cap |= IXGBE_FEATURE_SRIOV;
4943b1d5caf3SKevin Bowling 		sc->feat_cap |= IXGBE_FEATURE_FDIR;
4944b1d5caf3SKevin Bowling 		if ((sc->hw.device_id == IXGBE_DEV_ID_X540_BYPASS) &&
4945b1d5caf3SKevin Bowling 		    (sc->hw.bus.func == 0))
4946b1d5caf3SKevin Bowling 			sc->feat_cap |= IXGBE_FEATURE_BYPASS;
49478eb6488eSEric Joyner 		break;
49488eb6488eSEric Joyner 	case ixgbe_mac_X550:
4949f72de14eSKevin Bowling 		sc->feat_cap |= IXGBE_FEATURE_RECOVERY_MODE;
4950b1d5caf3SKevin Bowling 		sc->feat_cap |= IXGBE_FEATURE_TEMP_SENSOR;
4951b1d5caf3SKevin Bowling 		sc->feat_cap |= IXGBE_FEATURE_SRIOV;
4952b1d5caf3SKevin Bowling 		sc->feat_cap |= IXGBE_FEATURE_FDIR;
49538eb6488eSEric Joyner 		break;
49548eb6488eSEric Joyner 	case ixgbe_mac_X550EM_x:
4955f72de14eSKevin Bowling 		sc->feat_cap |= IXGBE_FEATURE_RECOVERY_MODE;
4956b1d5caf3SKevin Bowling 		sc->feat_cap |= IXGBE_FEATURE_SRIOV;
4957b1d5caf3SKevin Bowling 		sc->feat_cap |= IXGBE_FEATURE_FDIR;
4958f72de14eSKevin Bowling 		if (sc->hw.device_id == IXGBE_DEV_ID_X550EM_X_KR)
4959f72de14eSKevin Bowling 			sc->feat_cap |= IXGBE_FEATURE_EEE;
49608eb6488eSEric Joyner 		break;
49618eb6488eSEric Joyner 	case ixgbe_mac_X550EM_a:
4962f72de14eSKevin Bowling 		sc->feat_cap |= IXGBE_FEATURE_RECOVERY_MODE;
4963b1d5caf3SKevin Bowling 		sc->feat_cap |= IXGBE_FEATURE_SRIOV;
4964b1d5caf3SKevin Bowling 		sc->feat_cap |= IXGBE_FEATURE_FDIR;
4965b1d5caf3SKevin Bowling 		sc->feat_cap &= ~IXGBE_FEATURE_LEGACY_IRQ;
4966b1d5caf3SKevin Bowling 		if ((sc->hw.device_id == IXGBE_DEV_ID_X550EM_A_1G_T) ||
4967b1d5caf3SKevin Bowling 		    (sc->hw.device_id == IXGBE_DEV_ID_X550EM_A_1G_T_L)) {
4968b1d5caf3SKevin Bowling 			sc->feat_cap |= IXGBE_FEATURE_TEMP_SENSOR;
4969b1d5caf3SKevin Bowling 			sc->feat_cap |= IXGBE_FEATURE_EEE;
49708eb6488eSEric Joyner 		}
49718eb6488eSEric Joyner 		break;
49728eb6488eSEric Joyner 	case ixgbe_mac_82599EB:
4973b1d5caf3SKevin Bowling 		sc->feat_cap |= IXGBE_FEATURE_SRIOV;
4974b1d5caf3SKevin Bowling 		sc->feat_cap |= IXGBE_FEATURE_FDIR;
4975b1d5caf3SKevin Bowling 		if ((sc->hw.device_id == IXGBE_DEV_ID_82599_BYPASS) &&
4976b1d5caf3SKevin Bowling 		    (sc->hw.bus.func == 0))
4977b1d5caf3SKevin Bowling 			sc->feat_cap |= IXGBE_FEATURE_BYPASS;
4978b1d5caf3SKevin Bowling 		if (sc->hw.device_id == IXGBE_DEV_ID_82599_QSFP_SF_QP)
4979b1d5caf3SKevin Bowling 			sc->feat_cap &= ~IXGBE_FEATURE_LEGACY_IRQ;
49808eb6488eSEric Joyner 		break;
49818eb6488eSEric Joyner 	default:
49828eb6488eSEric Joyner 		break;
49838eb6488eSEric Joyner 	}
49848eb6488eSEric Joyner 
49858eb6488eSEric Joyner 	/* Enabled by default... */
49868eb6488eSEric Joyner 	/* Fan failure detection */
4987b1d5caf3SKevin Bowling 	if (sc->feat_cap & IXGBE_FEATURE_FAN_FAIL)
4988b1d5caf3SKevin Bowling 		sc->feat_en |= IXGBE_FEATURE_FAN_FAIL;
49898eb6488eSEric Joyner 	/* Netmap */
4990b1d5caf3SKevin Bowling 	if (sc->feat_cap & IXGBE_FEATURE_NETMAP)
4991b1d5caf3SKevin Bowling 		sc->feat_en |= IXGBE_FEATURE_NETMAP;
49928eb6488eSEric Joyner 	/* EEE */
4993b1d5caf3SKevin Bowling 	if (sc->feat_cap & IXGBE_FEATURE_EEE)
4994b1d5caf3SKevin Bowling 		sc->feat_en |= IXGBE_FEATURE_EEE;
49958eb6488eSEric Joyner 	/* Thermal Sensor */
4996b1d5caf3SKevin Bowling 	if (sc->feat_cap & IXGBE_FEATURE_TEMP_SENSOR)
4997b1d5caf3SKevin Bowling 		sc->feat_en |= IXGBE_FEATURE_TEMP_SENSOR;
4998f72de14eSKevin Bowling 	/* Recovery mode */
4999f72de14eSKevin Bowling 	if (sc->feat_cap & IXGBE_FEATURE_RECOVERY_MODE)
5000f72de14eSKevin Bowling 		sc->feat_en |= IXGBE_FEATURE_RECOVERY_MODE;
50018eb6488eSEric Joyner 
50028eb6488eSEric Joyner 	/* Enabled via global sysctl... */
50038eb6488eSEric Joyner 	/* Flow Director */
50048eb6488eSEric Joyner 	if (ixgbe_enable_fdir) {
5005b1d5caf3SKevin Bowling 		if (sc->feat_cap & IXGBE_FEATURE_FDIR)
5006b1d5caf3SKevin Bowling 			sc->feat_en |= IXGBE_FEATURE_FDIR;
50078eb6488eSEric Joyner 		else
5008c58d34ddSKevin Bowling 			device_printf(sc->dev,
5009c58d34ddSKevin Bowling 			    "Device does not support Flow Director."
5010c58d34ddSKevin Bowling 			    " Leaving disabled.");
50118eb6488eSEric Joyner 	}
50126f37f232SEric Joyner 	/*
50138eb6488eSEric Joyner 	 * Message Signal Interrupts - Extended (MSI-X)
50148eb6488eSEric Joyner 	 * Normal MSI is only enabled if MSI-X calls fail.
5015758cc3dcSJack F Vogel 	 */
50168eb6488eSEric Joyner 	if (!ixgbe_enable_msix)
5017b1d5caf3SKevin Bowling 		sc->feat_cap &= ~IXGBE_FEATURE_MSIX;
50188eb6488eSEric Joyner 	/* Receive-Side Scaling (RSS) */
5019b1d5caf3SKevin Bowling 	if ((sc->feat_cap & IXGBE_FEATURE_RSS) && ixgbe_enable_rss)
5020b1d5caf3SKevin Bowling 		sc->feat_en |= IXGBE_FEATURE_RSS;
50218eb6488eSEric Joyner 
50228eb6488eSEric Joyner 	/* Disable features with unmet dependencies... */
50238eb6488eSEric Joyner 	/* No MSI-X */
5024b1d5caf3SKevin Bowling 	if (!(sc->feat_cap & IXGBE_FEATURE_MSIX)) {
5025b1d5caf3SKevin Bowling 		sc->feat_cap &= ~IXGBE_FEATURE_RSS;
5026b1d5caf3SKevin Bowling 		sc->feat_cap &= ~IXGBE_FEATURE_SRIOV;
5027b1d5caf3SKevin Bowling 		sc->feat_en &= ~IXGBE_FEATURE_RSS;
5028b1d5caf3SKevin Bowling 		sc->feat_en &= ~IXGBE_FEATURE_SRIOV;
50298eb6488eSEric Joyner 	}
50308eb6488eSEric Joyner } /* ixgbe_init_device_features */
50318eb6488eSEric Joyner 
50328eb6488eSEric Joyner /************************************************************************
50338eb6488eSEric Joyner  * ixgbe_check_fan_failure
50348eb6488eSEric Joyner  ************************************************************************/
50358eb6488eSEric Joyner static void
ixgbe_check_fan_failure(struct ixgbe_softc * sc,u32 reg,bool in_interrupt)5036b1d5caf3SKevin Bowling ixgbe_check_fan_failure(struct ixgbe_softc *sc, u32 reg, bool in_interrupt)
50378eb6488eSEric Joyner {
50388eb6488eSEric Joyner 	u32 mask;
50398eb6488eSEric Joyner 
5040b1d5caf3SKevin Bowling 	mask = (in_interrupt) ? IXGBE_EICR_GPI_SDP1_BY_MAC(&sc->hw) :
50418eb6488eSEric Joyner 	    IXGBE_ESDP_SDP1;
50428eb6488eSEric Joyner 
50438eb6488eSEric Joyner 	if (reg & mask)
5044c58d34ddSKevin Bowling 		device_printf(sc->dev,
5045c58d34ddSKevin Bowling 		    "\nCRITICAL: FAN FAILURE!! REPLACE IMMEDIATELY!!\n");
50468eb6488eSEric Joyner } /* ixgbe_check_fan_failure */
50477660e4eaSKevin Bowling 
50487660e4eaSKevin Bowling /************************************************************************
50497660e4eaSKevin Bowling  * ixgbe_sbuf_fw_version
50507660e4eaSKevin Bowling  ************************************************************************/
50517660e4eaSKevin Bowling static void
ixgbe_sbuf_fw_version(struct ixgbe_hw * hw,struct sbuf * buf)50527660e4eaSKevin Bowling ixgbe_sbuf_fw_version(struct ixgbe_hw *hw, struct sbuf *buf)
50537660e4eaSKevin Bowling {
50547660e4eaSKevin Bowling 	struct ixgbe_nvm_version nvm_ver = {0};
50555de5419bSKevin Bowling 	const char *space = "";
50567660e4eaSKevin Bowling 
5057f72de14eSKevin Bowling 	ixgbe_get_nvm_version(hw, &nvm_ver); /* NVM version */
50587660e4eaSKevin Bowling 	ixgbe_get_oem_prod_version(hw, &nvm_ver); /* OEM's NVM version */
5059c58d34ddSKevin Bowling 	ixgbe_get_etk_id(hw, &nvm_ver); /* eTrack a build ID in Intel's SCM */
5060f72de14eSKevin Bowling 	ixgbe_get_orom_version(hw, &nvm_ver); /* Option ROM */
5061f72de14eSKevin Bowling 
5062f72de14eSKevin Bowling 	/* FW version */
5063f72de14eSKevin Bowling 	if ((nvm_ver.phy_fw_maj == 0x0 &&
5064f72de14eSKevin Bowling 	    nvm_ver.phy_fw_min == 0x0 &&
5065f72de14eSKevin Bowling 	    nvm_ver.phy_fw_id == 0x0) ||
5066f72de14eSKevin Bowling 		(nvm_ver.phy_fw_maj == 0xF &&
5067f72de14eSKevin Bowling 	    nvm_ver.phy_fw_min == 0xFF &&
5068f72de14eSKevin Bowling 	    nvm_ver.phy_fw_id == 0xF)) {
5069f72de14eSKevin Bowling 		/* If major, minor and id numbers are set to 0,
5070f72de14eSKevin Bowling 		 * reading FW version is unsupported. If major number
5071f72de14eSKevin Bowling 		 * is set to 0xF, minor is set to 0xFF and id is set
5072f72de14eSKevin Bowling 		 * to 0xF, this means that number read is invalid. */
5073f72de14eSKevin Bowling 	} else
5074f72de14eSKevin Bowling 		sbuf_printf(buf, "fw %d.%d.%d ",
5075c58d34ddSKevin Bowling 		    nvm_ver.phy_fw_maj, nvm_ver.phy_fw_min,
5076c58d34ddSKevin Bowling 		    nvm_ver.phy_fw_id);
5077f72de14eSKevin Bowling 
5078f72de14eSKevin Bowling 	/* NVM version */
5079f72de14eSKevin Bowling 	if ((nvm_ver.nvm_major == 0x0 &&
5080f72de14eSKevin Bowling 	    nvm_ver.nvm_minor == 0x0 &&
5081f72de14eSKevin Bowling 	    nvm_ver.nvm_id == 0x0) ||
5082f72de14eSKevin Bowling 		(nvm_ver.nvm_major == 0xF &&
5083f72de14eSKevin Bowling 	    nvm_ver.nvm_minor == 0xFF &&
5084f72de14eSKevin Bowling 	    nvm_ver.nvm_id == 0xF)) {
5085f72de14eSKevin Bowling 		/* If major, minor and id numbers are set to 0,
5086f72de14eSKevin Bowling 		 * reading NVM version is unsupported. If major number
5087f72de14eSKevin Bowling 		 * is set to 0xF, minor is set to 0xFF and id is set
5088f72de14eSKevin Bowling 		 * to 0xF, this means that number read is invalid. */
5089f72de14eSKevin Bowling 	} else
5090f72de14eSKevin Bowling 		sbuf_printf(buf, "nvm %x.%02x.%x ",
5091f72de14eSKevin Bowling 		    nvm_ver.nvm_major, nvm_ver.nvm_minor, nvm_ver.nvm_id);
50927660e4eaSKevin Bowling 
50935de5419bSKevin Bowling 	if (nvm_ver.oem_valid) {
50947660e4eaSKevin Bowling 		sbuf_printf(buf, "NVM OEM V%d.%d R%d", nvm_ver.oem_major,
50957660e4eaSKevin Bowling 		    nvm_ver.oem_minor, nvm_ver.oem_release);
50965de5419bSKevin Bowling 		space = " ";
50975de5419bSKevin Bowling 	}
50987660e4eaSKevin Bowling 
50995de5419bSKevin Bowling 	if (nvm_ver.or_valid) {
51005de5419bSKevin Bowling 		sbuf_printf(buf, "%sOption ROM V%d-b%d-p%d",
5101c58d34ddSKevin Bowling 		    space, nvm_ver.or_major, nvm_ver.or_build,
5102c58d34ddSKevin Bowling 		    nvm_ver.or_patch);
51035de5419bSKevin Bowling 		space = " ";
51045de5419bSKevin Bowling 	}
51057660e4eaSKevin Bowling 
51065de5419bSKevin Bowling 	if (nvm_ver.etk_id != ((NVM_VER_INVALID << NVM_ETK_SHIFT) |
5107f72de14eSKevin Bowling 	    NVM_VER_INVALID | 0xFFFFFFFF)) {
51085de5419bSKevin Bowling 		sbuf_printf(buf, "%seTrack 0x%08x", space, nvm_ver.etk_id);
51095de5419bSKevin Bowling 	}
51107660e4eaSKevin Bowling } /* ixgbe_sbuf_fw_version */
51117660e4eaSKevin Bowling 
51127660e4eaSKevin Bowling /************************************************************************
51137660e4eaSKevin Bowling  * ixgbe_print_fw_version
51147660e4eaSKevin Bowling  ************************************************************************/
51157660e4eaSKevin Bowling static void
ixgbe_print_fw_version(if_ctx_t ctx)51167660e4eaSKevin Bowling ixgbe_print_fw_version(if_ctx_t ctx)
51177660e4eaSKevin Bowling {
5118b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = iflib_get_softc(ctx);
5119b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
5120b1d5caf3SKevin Bowling 	device_t dev = sc->dev;
51217660e4eaSKevin Bowling 	struct sbuf *buf;
51227660e4eaSKevin Bowling 	int error = 0;
51237660e4eaSKevin Bowling 
51247660e4eaSKevin Bowling 	buf = sbuf_new_auto();
51257660e4eaSKevin Bowling 	if (!buf) {
51267660e4eaSKevin Bowling 		device_printf(dev, "Could not allocate sbuf for output.\n");
51277660e4eaSKevin Bowling 		return;
51287660e4eaSKevin Bowling 	}
51297660e4eaSKevin Bowling 
51307660e4eaSKevin Bowling 	ixgbe_sbuf_fw_version(hw, buf);
51317660e4eaSKevin Bowling 
51327660e4eaSKevin Bowling 	error = sbuf_finish(buf);
51337660e4eaSKevin Bowling 	if (error)
51347660e4eaSKevin Bowling 		device_printf(dev, "Error finishing sbuf: %d\n", error);
51357660e4eaSKevin Bowling 	else if (sbuf_len(buf))
51367660e4eaSKevin Bowling 		device_printf(dev, "%s\n", sbuf_data(buf));
51377660e4eaSKevin Bowling 
51387660e4eaSKevin Bowling 	sbuf_delete(buf);
51397660e4eaSKevin Bowling } /* ixgbe_print_fw_version */
51407660e4eaSKevin Bowling 
51417660e4eaSKevin Bowling /************************************************************************
51427660e4eaSKevin Bowling  * ixgbe_sysctl_print_fw_version
51437660e4eaSKevin Bowling  ************************************************************************/
51447660e4eaSKevin Bowling static int
ixgbe_sysctl_print_fw_version(SYSCTL_HANDLER_ARGS)51457660e4eaSKevin Bowling ixgbe_sysctl_print_fw_version(SYSCTL_HANDLER_ARGS)
51467660e4eaSKevin Bowling {
5147b1d5caf3SKevin Bowling 	struct ixgbe_softc *sc = (struct ixgbe_softc *)arg1;
5148b1d5caf3SKevin Bowling 	struct ixgbe_hw *hw = &sc->hw;
5149b1d5caf3SKevin Bowling 	device_t dev = sc->dev;
51507660e4eaSKevin Bowling 	struct sbuf *buf;
51517660e4eaSKevin Bowling 	int error = 0;
51527660e4eaSKevin Bowling 
51537660e4eaSKevin Bowling 	buf = sbuf_new_for_sysctl(NULL, NULL, 128, req);
51547660e4eaSKevin Bowling 	if (!buf) {
51557660e4eaSKevin Bowling 		device_printf(dev, "Could not allocate sbuf for output.\n");
51567660e4eaSKevin Bowling 		return (ENOMEM);
51577660e4eaSKevin Bowling 	}
51587660e4eaSKevin Bowling 
51597660e4eaSKevin Bowling 	ixgbe_sbuf_fw_version(hw, buf);
51607660e4eaSKevin Bowling 
51617660e4eaSKevin Bowling 	error = sbuf_finish(buf);
51627660e4eaSKevin Bowling 	if (error)
51637660e4eaSKevin Bowling 		device_printf(dev, "Error finishing sbuf: %d\n", error);
51647660e4eaSKevin Bowling 
51657660e4eaSKevin Bowling 	sbuf_delete(buf);
51667660e4eaSKevin Bowling 
51677660e4eaSKevin Bowling 	return (0);
51687660e4eaSKevin Bowling } /* ixgbe_sysctl_print_fw_version */
5169