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