Commit ed83da12 authored by Alexander Duyck's avatar Alexander Duyck Committed by Jeff Kirsher

ixgbe: Do not parse past IP header on fragments beyond the first

This change makes it so that only the first fragment in a series of fragments
will have the L4 header pulled.  Previously we were always pulling the L4
header as well and in the case of UDP this can harm performance since only the
first fragment will have the header, the rest just contain data which should
be left in the paged portion of the packet.
Signed-off-by: default avatarAlexander Duyck <alexander.h.duyck@intel.com>
Tested-by: default avatarStephen Ko <stephen.s.ko@intel.com>
Tested-by: default avatarMarcus Dennis <marcusx.e.dennis@intel.com>
Signed-off-by: default avatarJeff Kirsher <jeffrey.t.kirsher@intel.com>
parent e921eb1a
...@@ -1338,26 +1338,29 @@ static unsigned int ixgbe_get_headlen(unsigned char *data, ...@@ -1338,26 +1338,29 @@ static unsigned int ixgbe_get_headlen(unsigned char *data,
if (hlen < sizeof(struct iphdr)) if (hlen < sizeof(struct iphdr))
return hdr.network - data; return hdr.network - data;
/* record next protocol */ /* record next protocol if header is present */
if (!hdr.ipv4->frag_off)
nexthdr = hdr.ipv4->protocol; nexthdr = hdr.ipv4->protocol;
hdr.network += hlen;
} else if (protocol == __constant_htons(ETH_P_IPV6)) { } else if (protocol == __constant_htons(ETH_P_IPV6)) {
if ((hdr.network - data) > (max_len - sizeof(struct ipv6hdr))) if ((hdr.network - data) > (max_len - sizeof(struct ipv6hdr)))
return max_len; return max_len;
/* record next protocol */ /* record next protocol */
nexthdr = hdr.ipv6->nexthdr; nexthdr = hdr.ipv6->nexthdr;
hdr.network += sizeof(struct ipv6hdr); hlen = sizeof(struct ipv6hdr);
#ifdef IXGBE_FCOE #ifdef IXGBE_FCOE
} else if (protocol == __constant_htons(ETH_P_FCOE)) { } else if (protocol == __constant_htons(ETH_P_FCOE)) {
if ((hdr.network - data) > (max_len - FCOE_HEADER_LEN)) if ((hdr.network - data) > (max_len - FCOE_HEADER_LEN))
return max_len; return max_len;
hdr.network += FCOE_HEADER_LEN; hlen = FCOE_HEADER_LEN;
#endif #endif
} else { } else {
return hdr.network - data; return hdr.network - data;
} }
/* relocate pointer to start of L4 header */
hdr.network += hlen;
/* finally sort out TCP/UDP */ /* finally sort out TCP/UDP */
if (nexthdr == IPPROTO_TCP) { if (nexthdr == IPPROTO_TCP) {
if ((hdr.network - data) > (max_len - sizeof(struct tcphdr))) if ((hdr.network - data) > (max_len - sizeof(struct tcphdr)))
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment