提交 7d397214 编写于 作者: D David S. Miller

Merge branch '10GbE' of git://git.kernel.org/pub/scm/linux/kernel/git/jkirsher/next-queue

Jeff Kirsher says:

====================
10GbE Intel Wired LAN Driver Updates 2016-01-08

This series contains updates to ixgbe only.

Vasu provides three fixes for ixgbe, first assigns a minimum credit to
a traffic class to resolve a Tx hang for CEE mode configuration.  Second
fix changes the driver to use netdev->fcoe_ddp_xid instead of our local
IXGBE_FCOE_DDP_MAX, since it is correctly set for our different devices
and avoids a DDP skip error on X550.  Lastly fix the PFC configuration
to include X550 devices.

Emil provides a fix for reporting the speed in ethtool by using the
stored value in out adapter structure.  This is due to external drivers
may end up with unknown speed when calling ethtool_get_settings().

Mark fixes the handling of any outer UDP checksum, by passing the
skb up with CHECKSUM_NONE when an outer UDP checksum is set.  This
will cause the stack to check the checksum, also do not increment an
error counter because we do not really know if there is an actual error.
Ixgbe ATR was not handling IPv6 extended headers, so ATR is not being
performed on such packets.  Fix this by skipping extended headers
when they are present.

Usha fixes an issue with X550 and getting FDMI HBA attributes when
FCoE support is enabled.

Neerav fixes an issue for X550 when FCoE and SR-IOV are enabled, which
the hardware generates MDD events.  Resolve this by setting the expected
values in the transmit context descriptors for FCoE/FIP frames and
adding a flush after writing the RDLEN register.
====================
Signed-off-by: NDavid S. Miller <davem@davemloft.net>
...@@ -139,6 +139,11 @@ s32 ixgbe_dcb_calculate_tc_credits(struct ixgbe_hw *hw, ...@@ -139,6 +139,11 @@ s32 ixgbe_dcb_calculate_tc_credits(struct ixgbe_hw *hw,
/* Calculate credit refill ratio using multiplier */ /* Calculate credit refill ratio using multiplier */
credit_refill = min(link_percentage * min_multiplier, credit_refill = min(link_percentage * min_multiplier,
MAX_CREDIT_REFILL); MAX_CREDIT_REFILL);
/* Refill at least minimum credit */
if (credit_refill < min_credit)
credit_refill = min_credit;
p->data_credits_refill = (u16)credit_refill; p->data_credits_refill = (u16)credit_refill;
/* Calculate maximum credit for the TC */ /* Calculate maximum credit for the TC */
...@@ -149,7 +154,7 @@ s32 ixgbe_dcb_calculate_tc_credits(struct ixgbe_hw *hw, ...@@ -149,7 +154,7 @@ s32 ixgbe_dcb_calculate_tc_credits(struct ixgbe_hw *hw,
* of a TC is too small, the maximum credit may not be * of a TC is too small, the maximum credit may not be
* enough to send out a jumbo frame in data plane arbitration. * enough to send out a jumbo frame in data plane arbitration.
*/ */
if (credit_max && (credit_max < min_credit)) if (credit_max < min_credit)
credit_max = min_credit; credit_max = min_credit;
if (direction == DCB_TX_CONFIG) { if (direction == DCB_TX_CONFIG) {
......
...@@ -223,13 +223,13 @@ s32 ixgbe_dcb_config_pfc_82599(struct ixgbe_hw *hw, u8 pfc_en, u8 *prio_tc) ...@@ -223,13 +223,13 @@ s32 ixgbe_dcb_config_pfc_82599(struct ixgbe_hw *hw, u8 pfc_en, u8 *prio_tc)
reg |= IXGBE_MFLCN_DPF; reg |= IXGBE_MFLCN_DPF;
/* /*
* X540 supports per TC Rx priority flow control. So * X540 & X550 supports per TC Rx priority flow control.
* clear all TCs and only enable those that should be * So clear all TCs and only enable those that should be
* enabled. * enabled.
*/ */
reg &= ~(IXGBE_MFLCN_RPFCE_MASK | IXGBE_MFLCN_RFCE); reg &= ~(IXGBE_MFLCN_RPFCE_MASK | IXGBE_MFLCN_RFCE);
if (hw->mac.type == ixgbe_mac_X540) if (hw->mac.type >= ixgbe_mac_X540)
reg |= pfc_en << IXGBE_MFLCN_RPFCE_SHIFT; reg |= pfc_en << IXGBE_MFLCN_RPFCE_SHIFT;
if (pfc_en) if (pfc_en)
......
...@@ -185,9 +185,7 @@ static int ixgbe_get_settings(struct net_device *netdev, ...@@ -185,9 +185,7 @@ static int ixgbe_get_settings(struct net_device *netdev,
struct ixgbe_adapter *adapter = netdev_priv(netdev); struct ixgbe_adapter *adapter = netdev_priv(netdev);
struct ixgbe_hw *hw = &adapter->hw; struct ixgbe_hw *hw = &adapter->hw;
ixgbe_link_speed supported_link; ixgbe_link_speed supported_link;
u32 link_speed = 0;
bool autoneg = false; bool autoneg = false;
bool link_up;
hw->mac.ops.get_link_capabilities(hw, &supported_link, &autoneg); hw->mac.ops.get_link_capabilities(hw, &supported_link, &autoneg);
...@@ -313,9 +311,8 @@ static int ixgbe_get_settings(struct net_device *netdev, ...@@ -313,9 +311,8 @@ static int ixgbe_get_settings(struct net_device *netdev,
break; break;
} }
hw->mac.ops.check_link(hw, &link_speed, &link_up, false); if (netif_carrier_ok(netdev)) {
if (link_up) { switch (adapter->link_speed) {
switch (link_speed) {
case IXGBE_LINK_SPEED_10GB_FULL: case IXGBE_LINK_SPEED_10GB_FULL:
ethtool_cmd_speed_set(ecmd, SPEED_10000); ethtool_cmd_speed_set(ecmd, SPEED_10000);
break; break;
......
...@@ -77,7 +77,7 @@ int ixgbe_fcoe_ddp_put(struct net_device *netdev, u16 xid) ...@@ -77,7 +77,7 @@ int ixgbe_fcoe_ddp_put(struct net_device *netdev, u16 xid)
if (!netdev) if (!netdev)
return 0; return 0;
if (xid >= IXGBE_FCOE_DDP_MAX) if (xid >= netdev->fcoe_ddp_xid)
return 0; return 0;
adapter = netdev_priv(netdev); adapter = netdev_priv(netdev);
...@@ -177,7 +177,7 @@ static int ixgbe_fcoe_ddp_setup(struct net_device *netdev, u16 xid, ...@@ -177,7 +177,7 @@ static int ixgbe_fcoe_ddp_setup(struct net_device *netdev, u16 xid,
return 0; return 0;
adapter = netdev_priv(netdev); adapter = netdev_priv(netdev);
if (xid >= IXGBE_FCOE_DDP_MAX) { if (xid >= netdev->fcoe_ddp_xid) {
e_warn(drv, "xid=0x%x out-of-range\n", xid); e_warn(drv, "xid=0x%x out-of-range\n", xid);
return 0; return 0;
} }
...@@ -517,6 +517,7 @@ int ixgbe_fso(struct ixgbe_ring *tx_ring, ...@@ -517,6 +517,7 @@ int ixgbe_fso(struct ixgbe_ring *tx_ring,
u32 vlan_macip_lens; u32 vlan_macip_lens;
u32 fcoe_sof_eof = 0; u32 fcoe_sof_eof = 0;
u32 mss_l4len_idx; u32 mss_l4len_idx;
u32 type_tucmd = IXGBE_ADVTXT_TUCMD_FCOE;
u8 sof, eof; u8 sof, eof;
if (skb_is_gso(skb) && (skb_shinfo(skb)->gso_type != SKB_GSO_FCOE)) { if (skb_is_gso(skb) && (skb_shinfo(skb)->gso_type != SKB_GSO_FCOE)) {
...@@ -593,6 +594,8 @@ int ixgbe_fso(struct ixgbe_ring *tx_ring, ...@@ -593,6 +594,8 @@ int ixgbe_fso(struct ixgbe_ring *tx_ring,
skb_shinfo(skb)->gso_size); skb_shinfo(skb)->gso_size);
first->bytecount += (first->gso_segs - 1) * *hdr_len; first->bytecount += (first->gso_segs - 1) * *hdr_len;
first->tx_flags |= IXGBE_TX_FLAGS_TSO; first->tx_flags |= IXGBE_TX_FLAGS_TSO;
/* Hardware expects L4T to be RSV for FCoE TSO */
type_tucmd |= IXGBE_ADVTXD_TUCMD_L4T_RSV;
} }
/* set flag indicating FCOE to ixgbe_tx_map call */ /* set flag indicating FCOE to ixgbe_tx_map call */
...@@ -610,7 +613,7 @@ int ixgbe_fso(struct ixgbe_ring *tx_ring, ...@@ -610,7 +613,7 @@ int ixgbe_fso(struct ixgbe_ring *tx_ring,
/* write context desc */ /* write context desc */
ixgbe_tx_ctxtdesc(tx_ring, vlan_macip_lens, fcoe_sof_eof, ixgbe_tx_ctxtdesc(tx_ring, vlan_macip_lens, fcoe_sof_eof,
IXGBE_ADVTXT_TUCMD_FCOE, mss_l4len_idx); type_tucmd, mss_l4len_idx);
return 0; return 0;
} }
...@@ -996,8 +999,7 @@ int ixgbe_fcoe_get_hbainfo(struct net_device *netdev, ...@@ -996,8 +999,7 @@ int ixgbe_fcoe_get_hbainfo(struct net_device *netdev,
return -EINVAL; return -EINVAL;
/* Don't return information on unsupported devices */ /* Don't return information on unsupported devices */
if (hw->mac.type != ixgbe_mac_82599EB && if (!(adapter->flags & IXGBE_FLAG_FCOE_ENABLED))
hw->mac.type != ixgbe_mac_X540)
return -EINVAL; return -EINVAL;
/* Manufacturer */ /* Manufacturer */
...@@ -1043,6 +1045,10 @@ int ixgbe_fcoe_get_hbainfo(struct net_device *netdev, ...@@ -1043,6 +1045,10 @@ int ixgbe_fcoe_get_hbainfo(struct net_device *netdev,
snprintf(info->model, snprintf(info->model,
sizeof(info->model), sizeof(info->model),
"Intel 82599"); "Intel 82599");
} else if (hw->mac.type == ixgbe_mac_X550) {
snprintf(info->model,
sizeof(info->model),
"Intel X550");
} else { } else {
snprintf(info->model, snprintf(info->model,
sizeof(info->model), sizeof(info->model),
......
...@@ -1483,7 +1483,7 @@ static inline void ixgbe_rx_checksum(struct ixgbe_ring *ring, ...@@ -1483,7 +1483,7 @@ static inline void ixgbe_rx_checksum(struct ixgbe_ring *ring,
return; return;
if (ixgbe_test_staterr(rx_desc, IXGBE_RXDADV_ERR_OUTERIPER)) { if (ixgbe_test_staterr(rx_desc, IXGBE_RXDADV_ERR_OUTERIPER)) {
ring->rx_stats.csum_err++; skb->ip_summed = CHECKSUM_NONE;
return; return;
} }
/* If we checked the outer header let the stack know */ /* If we checked the outer header let the stack know */
...@@ -3619,6 +3619,9 @@ void ixgbe_configure_rx_ring(struct ixgbe_adapter *adapter, ...@@ -3619,6 +3619,9 @@ void ixgbe_configure_rx_ring(struct ixgbe_adapter *adapter,
IXGBE_WRITE_REG(hw, IXGBE_RDBAH(reg_idx), (rdba >> 32)); IXGBE_WRITE_REG(hw, IXGBE_RDBAH(reg_idx), (rdba >> 32));
IXGBE_WRITE_REG(hw, IXGBE_RDLEN(reg_idx), IXGBE_WRITE_REG(hw, IXGBE_RDLEN(reg_idx),
ring->count * sizeof(union ixgbe_adv_rx_desc)); ring->count * sizeof(union ixgbe_adv_rx_desc));
/* Force flushing of IXGBE_RDLEN to prevent MDD */
IXGBE_WRITE_FLUSH(hw);
IXGBE_WRITE_REG(hw, IXGBE_RDH(reg_idx), 0); IXGBE_WRITE_REG(hw, IXGBE_RDH(reg_idx), 0);
IXGBE_WRITE_REG(hw, IXGBE_RDT(reg_idx), 0); IXGBE_WRITE_REG(hw, IXGBE_RDT(reg_idx), 0);
ring->tail = adapter->io_addr + IXGBE_RDT(reg_idx); ring->tail = adapter->io_addr + IXGBE_RDT(reg_idx);
...@@ -7570,7 +7573,9 @@ static void ixgbe_atr(struct ixgbe_ring *ring, ...@@ -7570,7 +7573,9 @@ static void ixgbe_atr(struct ixgbe_ring *ring,
/* snag network header to get L4 type and address */ /* snag network header to get L4 type and address */
skb = first->skb; skb = first->skb;
hdr.network = skb_network_header(skb); hdr.network = skb_network_header(skb);
if (skb->encapsulation) { if (!skb->encapsulation) {
th = tcp_hdr(skb);
} else {
#ifdef CONFIG_IXGBE_VXLAN #ifdef CONFIG_IXGBE_VXLAN
struct ixgbe_adapter *adapter = q_vector->adapter; struct ixgbe_adapter *adapter = q_vector->adapter;
...@@ -7589,14 +7594,34 @@ static void ixgbe_atr(struct ixgbe_ring *ring, ...@@ -7589,14 +7594,34 @@ static void ixgbe_atr(struct ixgbe_ring *ring,
#else #else
return; return;
#endif /* CONFIG_IXGBE_VXLAN */ #endif /* CONFIG_IXGBE_VXLAN */
} else { }
/* Currently only IPv4/IPv6 with TCP is supported */
if ((first->protocol != htons(ETH_P_IPV6) || /* Currently only IPv4/IPv6 with TCP is supported */
hdr.ipv6->nexthdr != IPPROTO_TCP) && switch (hdr.ipv4->version) {
(first->protocol != htons(ETH_P_IP) || case IPVERSION:
hdr.ipv4->protocol != IPPROTO_TCP)) if (hdr.ipv4->protocol != IPPROTO_TCP)
return; return;
th = tcp_hdr(skb); break;
case 6:
if (likely((unsigned char *)th - hdr.network ==
sizeof(struct ipv6hdr))) {
if (hdr.ipv6->nexthdr != IPPROTO_TCP)
return;
} else {
__be16 frag_off;
u8 l4_hdr;
ipv6_skip_exthdr(skb, hdr.network - skb->data +
sizeof(struct ipv6hdr),
&l4_hdr, &frag_off);
if (unlikely(frag_off))
return;
if (l4_hdr != IPPROTO_TCP)
return;
}
break;
default:
return;
} }
/* skip this packet since it is invalid or the socket is closing */ /* skip this packet since it is invalid or the socket is closing */
...@@ -7631,10 +7656,12 @@ static void ixgbe_atr(struct ixgbe_ring *ring, ...@@ -7631,10 +7656,12 @@ static void ixgbe_atr(struct ixgbe_ring *ring,
common.port.src ^= th->dest ^ first->protocol; common.port.src ^= th->dest ^ first->protocol;
common.port.dst ^= th->source; common.port.dst ^= th->source;
if (first->protocol == htons(ETH_P_IP)) { switch (hdr.ipv4->version) {
case IPVERSION:
input.formatted.flow_type = IXGBE_ATR_FLOW_TYPE_TCPV4; input.formatted.flow_type = IXGBE_ATR_FLOW_TYPE_TCPV4;
common.ip ^= hdr.ipv4->saddr ^ hdr.ipv4->daddr; common.ip ^= hdr.ipv4->saddr ^ hdr.ipv4->daddr;
} else { break;
case 6:
input.formatted.flow_type = IXGBE_ATR_FLOW_TYPE_TCPV6; input.formatted.flow_type = IXGBE_ATR_FLOW_TYPE_TCPV6;
common.ip ^= hdr.ipv6->saddr.s6_addr32[0] ^ common.ip ^= hdr.ipv6->saddr.s6_addr32[0] ^
hdr.ipv6->saddr.s6_addr32[1] ^ hdr.ipv6->saddr.s6_addr32[1] ^
...@@ -7644,6 +7671,9 @@ static void ixgbe_atr(struct ixgbe_ring *ring, ...@@ -7644,6 +7671,9 @@ static void ixgbe_atr(struct ixgbe_ring *ring,
hdr.ipv6->daddr.s6_addr32[1] ^ hdr.ipv6->daddr.s6_addr32[1] ^
hdr.ipv6->daddr.s6_addr32[2] ^ hdr.ipv6->daddr.s6_addr32[2] ^
hdr.ipv6->daddr.s6_addr32[3]; hdr.ipv6->daddr.s6_addr32[3];
break;
default:
break;
} }
#ifdef CONFIG_IXGBE_VXLAN #ifdef CONFIG_IXGBE_VXLAN
......
...@@ -2780,6 +2780,7 @@ struct ixgbe_adv_tx_context_desc { ...@@ -2780,6 +2780,7 @@ struct ixgbe_adv_tx_context_desc {
#define IXGBE_ADVTXD_TUCMD_L4T_UDP 0x00000000 /* L4 Packet TYPE of UDP */ #define IXGBE_ADVTXD_TUCMD_L4T_UDP 0x00000000 /* L4 Packet TYPE of UDP */
#define IXGBE_ADVTXD_TUCMD_L4T_TCP 0x00000800 /* L4 Packet TYPE of TCP */ #define IXGBE_ADVTXD_TUCMD_L4T_TCP 0x00000800 /* L4 Packet TYPE of TCP */
#define IXGBE_ADVTXD_TUCMD_L4T_SCTP 0x00001000 /* L4 Packet TYPE of SCTP */ #define IXGBE_ADVTXD_TUCMD_L4T_SCTP 0x00001000 /* L4 Packet TYPE of SCTP */
#define IXGBE_ADVTXD_TUCMD_L4T_RSV 0x00001800 /* RSV L4 Packet TYPE */
#define IXGBE_ADVTXD_TUCMD_MKRREQ 0x00002000 /*Req requires Markers and CRC*/ #define IXGBE_ADVTXD_TUCMD_MKRREQ 0x00002000 /*Req requires Markers and CRC*/
#define IXGBE_ADVTXD_POPTS_IPSEC 0x00000400 /* IPSec offload request */ #define IXGBE_ADVTXD_POPTS_IPSEC 0x00000400 /* IPSec offload request */
#define IXGBE_ADVTXD_TUCMD_IPSEC_TYPE_ESP 0x00002000 /* IPSec Type ESP */ #define IXGBE_ADVTXD_TUCMD_IPSEC_TYPE_ESP 0x00002000 /* IPSec Type ESP */
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册