diff options
author | Robert Mustacchi <rm@joyent.com> | 2018-03-21 22:42:01 +0000 |
---|---|---|
committer | Robert Mustacchi <rm@joyent.com> | 2018-03-21 22:53:35 +0000 |
commit | fee89948732fb03b583e92a1103cdedefed8da9b (patch) | |
tree | 6d960d8c45697ce55c16dc1b7c625a0791d207fe /usr | |
parent | 104a8d59880bf4fd87a6d37e78339ea14447d78e (diff) | |
download | illumos-joyent-fee89948732fb03b583e92a1103cdedefed8da9b.tar.gz |
Fix i40e rx checksum logic
Diffstat (limited to 'usr')
-rw-r--r-- | usr/src/uts/common/io/i40e/i40e_transceiver.c | 104 |
1 files changed, 71 insertions, 33 deletions
diff --git a/usr/src/uts/common/io/i40e/i40e_transceiver.c b/usr/src/uts/common/io/i40e/i40e_transceiver.c index bedefab8c6..02dcc41b25 100644 --- a/usr/src/uts/common/io/i40e/i40e_transceiver.c +++ b/usr/src/uts/common/io/i40e/i40e_transceiver.c @@ -1262,6 +1262,33 @@ i40e_rx_copy(i40e_trqpair_t *itrq, i40e_rx_data_t *rxd, uint32_t index, } /* + * Determine if this pinfo is valid for L4 outer checksum offload for a + * non-tunneled packet. This is the case for an IP packe + */ +static inline int +i40e_rx_ptype_nontunnel_ol4(struct i40e_rx_ptype_decoded *pinfo) +{ + return (pinfo->outer_ip == I40E_RX_PTYPE_OUTER_IP && + pinfo->tunnel_type == I40E_RX_PTYPE_TUNNEL_NONE && + (pinfo->inner_prot == I40E_RX_PTYPE_INNER_PROT_UDP || + pinfo->inner_prot == I40E_RX_PTYPE_INNER_PROT_TCP || + pinfo->inner_prot == I40E_RX_PTYPE_INNER_PROT_SCTP)); +} + +static inline int +i40e_rx_ptype_tunnel_il4(struct i40e_rx_ptype_decoded *pinfo) +{ + return (pinfo->outer_ip == I40E_RX_PTYPE_OUTER_IP && + (pinfo->tunnel_type == I40E_RX_PTYPE_TUNNEL_IP_GRENAT_MAC || + pinfo->tunnel_type == I40E_RX_PTYPE_TUNNEL_IP_GRENAT_MAC_VLAN) && + pinfo->tunnel_end_frag == I40E_RX_PTYPE_NOT_FRAG && + pinfo->tunnel_end_prot != I40E_RX_PTYPE_TUNNEL_END_NONE && + (pinfo->inner_prot == I40E_RX_PTYPE_INNER_PROT_UDP || + pinfo->inner_prot == I40E_RX_PTYPE_INNER_PROT_TCP || + pinfo->inner_prot == I40E_RX_PTYPE_INNER_PROT_SCTP)); +} + +/* * Determine if the device has enabled any checksum flags for us. The level of * checksum computed will depend on the type packet that we have, which is * contained in ptype. For example, the checksum logic it does will vary @@ -1319,29 +1346,20 @@ i40e_rx_hcksum(i40e_trqpair_t *itrq, mblk_t *mp, uint64_t status, uint32_t err, * The hardware denotes three kinds of possible errors. Two are used * for inner and outer IP checksum errors (IPE and EIPE) and the third * is for L4 checksum errors (L4E). If there is only one IP header, then - * the only thing that we care about is IPE. For both checksums on - * tunneled packets we handle IPE and EIPE with two different status - * flags. + * the only thing that we care about is IPE. However, if this is a + * tunnel packet, then we care about EPIE. Note, none of this controls + * whether or not we have an inner IPv4 checksum. */ if (pinfo.outer_ip == I40E_RX_PTYPE_OUTER_IP && pinfo.outer_ip_ver == I40E_RX_PTYPE_OUTER_IPV4) { - /* - * The IPE check applies to either no tunnel, or to the inner - * HW checksum. - */ - if ((err & (1 << I40E_RX_DESC_ERROR_IPE_SHIFT)) != 0) { - itrq->itrq_rxstat.irxs_hck_iperr.value.ui64++; - } else { - itrq->itrq_rxstat.irxs_hck_v4hdrok.value.ui64++; - if (pinfo.tunnel_type == I40E_RX_PTYPE_TUNNEL_NONE) { - cksum |= HCK_IPV4_HDRCKSUM_OK; + if (pinfo.tunnel_type == I40E_RX_PTYPE_TUNNEL_NONE) { + if ((err & (1 << I40E_RX_DESC_ERROR_IPE_SHIFT)) != 0) { + itrq->itrq_rxstat.irxs_hck_iperr.value.ui64++; } else { - cksum |= HCK_INNER_IPV4_HDRCKSUM_OK; + itrq->itrq_rxstat.irxs_hck_v4hdrok.value.ui64++; + cksum |= HCK_IPV4_HDRCKSUM_OK; } - } - - if (pinfo.tunnel_type != I40E_RX_PTYPE_TUNNEL_NONE) { - /* Handle outer HW checksum */ + } else { if ((err & (1 << I40E_RX_DESC_ERROR_EIPE_SHIFT)) != 0) { itrq->itrq_rxstat.irxs_hck_eiperr.value.ui64++; } else { @@ -1352,22 +1370,32 @@ i40e_rx_hcksum(i40e_trqpair_t *itrq, mblk_t *mp, uint64_t status, uint32_t err, } /* - * We only have meaningful L4 checksums in the case of IP->L4 and - * IP->IP->L4. There is no outer L4 checksum data available in any - * other case. - * - * XXX - * Older hardware does not support an outer L4 checksum, although newer - * hardware does. For now we assume there is no outer L4 support, so - * we do not attempt to handle a tunnel and HCK_INNER_FULLCKSUM_OK. + * If we have a fragmented packet in any form, we're done. */ - if (pinfo.outer_ip == I40E_RX_PTYPE_OUTER_IP && - pinfo.tunnel_type == I40E_RX_PTYPE_TUNNEL_NONE && - (pinfo.inner_prot == I40E_RX_PTYPE_INNER_PROT_UDP || - pinfo.inner_prot == I40E_RX_PTYPE_INNER_PROT_TCP || - pinfo.inner_prot == I40E_RX_PTYPE_INNER_PROT_ICMP || - pinfo.inner_prot == I40E_RX_PTYPE_INNER_PROT_SCTP)) { - ASSERT(pinfo.payload_layer == I40E_RX_PTYPE_PAYLOAD_LAYER_PAY4); + if (pinfo.outer_frag != I40E_RX_PTYPE_NOT_FRAG) + goto done; + + /* + * If we have a tunneled packet and the inner IP header is IPv4, check + * IPE to see if we have a valid L4 checksum. + */ + if (pinfo.tunnel_type != I40E_RX_PTYPE_TUNNEL_NONE && + pinfo.tunnel_end_prot == I40E_RX_PTYPE_TUNNEL_END_IPV4) { + if ((err & (1 << I40E_RX_DESC_ERROR_IPE_SHIFT)) != 0) { + itrq->itrq_rxstat.irxs_hck_iperr.value.ui64++; + } else { + itrq->itrq_rxstat.irxs_hck_v4hdrok.value.ui64++; + cksum |= HCK_INNER_IPV4_HDRCKSUM_OK; + } + } + + /* + * Determine if we have a valid outer L4 checksum. The only supported L4 + * checksums are TCP, SCTP, and UDP. If this is a UDP tunneled packet, + * then there is no support for the outer L4 unless we are on the X722 + * MAC. However, we do not support that at this time. + */ + if (i40e_rx_ptype_nontunnel_ol4(&pinfo)) { if ((err & (1 << I40E_RX_DESC_ERROR_L4E_SHIFT)) != 0) { itrq->itrq_rxstat.irxs_hck_l4err.value.ui64++; } else { @@ -1376,6 +1404,16 @@ i40e_rx_hcksum(i40e_trqpair_t *itrq, mblk_t *mp, uint64_t status, uint32_t err, } } + if (i40e_rx_ptype_tunnel_il4(&pinfo)) { + if ((err & (1 << I40E_RX_DESC_ERROR_L4E_SHIFT)) != 0) { + itrq->itrq_rxstat.irxs_hck_l4err.value.ui64++; + } else { + itrq->itrq_rxstat.irxs_hck_l4hdrok.value.ui64++; + cksum |= HCK_INNER_FULLCKSUM_OK; + } + } + +done: if (cksum != 0) { itrq->itrq_rxstat.irxs_hck_set.value.ui64++; mac_hcksum_set(mp, 0, 0, 0, 0, cksum); |