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