udp: add udp gso
authorWillem de Bruijn <willemb@google.com>
Thu, 26 Apr 2018 17:42:16 +0000 (13:42 -0400)
committerDavid S. Miller <davem@davemloft.net>
Thu, 26 Apr 2018 19:07:42 +0000 (15:07 -0400)
Implement generic segmentation offload support for udp datagrams. A
follow-up patch adds support to the protocol stack to generate such
packets.

UDP GSO is not UFO. UFO fragments a single large datagram. GSO splits
a large payload into a number of discrete UDP datagrams.

The implementation adds a GSO type SKB_UDP_GSO_L4 to differentiate it
from UFO (SKB_UDP_GSO).

IPPROTO_UDPLITE is excluded, as that protocol has no gso handler
registered.

[ Export __udp_gso_segment for ipv6. -DaveM ]

Signed-off-by: Willem de Bruijn <willemb@google.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
include/linux/skbuff.h
include/net/udp.h
net/core/skbuff.c
net/ipv4/udp_offload.c
net/ipv6/ip6_offload.c
net/ipv6/udp_offload.c

index d274059529eb5216d041dfdcad4a564a623c8ea0..a4a5c0c5cba89d1db792435cc230d774cfa6ad3f 100644 (file)
@@ -573,6 +573,8 @@ enum {
        SKB_GSO_ESP = 1 << 15,
 
        SKB_GSO_UDP = 1 << 16,
+
+       SKB_GSO_UDP_L4 = 1 << 17,
 };
 
 #if BITS_PER_LONG > 32
index 0676b272f6ac8bf91d07f18b7f75e39481a47eaa..741d888d0fdb4a62f899f74890d3d06a5fc5306f 100644 (file)
@@ -174,6 +174,10 @@ struct sk_buff **udp_gro_receive(struct sk_buff **head, struct sk_buff *skb,
                                 struct udphdr *uh, udp_lookup_t lookup);
 int udp_gro_complete(struct sk_buff *skb, int nhoff, udp_lookup_t lookup);
 
+struct sk_buff *__udp_gso_segment(struct sk_buff *gso_skb,
+                                 netdev_features_t features,
+                                 unsigned int mss, __sum16 check);
+
 static inline struct udphdr *udp_gro_udphdr(struct sk_buff *skb)
 {
        struct udphdr *uh;
index ff49e352deea1d07049f5c5ac425fbcdb4fd96a8..c647cfe114e02e537a39c2d124634fafaa2251e5 100644 (file)
@@ -4940,6 +4940,8 @@ static unsigned int skb_gso_transport_seglen(const struct sk_buff *skb)
                thlen = tcp_hdrlen(skb);
        } else if (unlikely(skb_is_gso_sctp(skb))) {
                thlen = sizeof(struct sctphdr);
+       } else if (shinfo->gso_type & SKB_GSO_UDP_L4) {
+               thlen = sizeof(struct udphdr);
        }
        /* UFO sets gso_size to the size of the fragmentation
         * payload, i.e. the size of the L4 (UDP) header is already
index ea6e6e7df0eec8e79f631851846cf6957014c162..6c2b7640f6f3fc5eac06b142a8924e46c644d0a7 100644 (file)
@@ -187,6 +187,54 @@ out_unlock:
 }
 EXPORT_SYMBOL(skb_udp_tunnel_segment);
 
+struct sk_buff *__udp_gso_segment(struct sk_buff *gso_skb,
+                                 netdev_features_t features,
+                                 unsigned int mss, __sum16 check)
+{
+       struct sk_buff *segs, *seg;
+       unsigned int hdrlen;
+       struct udphdr *uh;
+
+       if (gso_skb->len <= sizeof(*uh) + mss)
+               return ERR_PTR(-EINVAL);
+
+       hdrlen = gso_skb->data - skb_mac_header(gso_skb);
+       skb_pull(gso_skb, sizeof(*uh));
+
+       segs = skb_segment(gso_skb, features);
+       if (unlikely(IS_ERR_OR_NULL(segs)))
+               return segs;
+
+       for (seg = segs; seg; seg = seg->next) {
+               uh = udp_hdr(seg);
+               uh->len = htons(seg->len - hdrlen);
+               uh->check = check;
+
+               /* last packet can be partial gso_size */
+               if (!seg->next)
+                       csum_replace2(&uh->check, htons(mss),
+                                     htons(seg->len - hdrlen - sizeof(*uh)));
+       }
+
+       return segs;
+}
+EXPORT_SYMBOL_GPL(__udp_gso_segment);
+
+static struct sk_buff *__udp4_gso_segment(struct sk_buff *gso_skb,
+                                         netdev_features_t features)
+{
+       const struct iphdr *iph = ip_hdr(gso_skb);
+       unsigned int mss = skb_shinfo(gso_skb)->gso_size;
+
+       if (!can_checksum_protocol(features, htons(ETH_P_IP)))
+               return ERR_PTR(-EIO);
+
+       return __udp_gso_segment(gso_skb, features, mss,
+                                udp_v4_check(sizeof(struct udphdr) + mss,
+                                             iph->saddr, iph->daddr, 0));
+}
+EXPORT_SYMBOL_GPL(__udp4_gso_segment);
+
 static struct sk_buff *udp4_ufo_fragment(struct sk_buff *skb,
                                         netdev_features_t features)
 {
@@ -203,12 +251,15 @@ static struct sk_buff *udp4_ufo_fragment(struct sk_buff *skb,
                goto out;
        }
 
-       if (!(skb_shinfo(skb)->gso_type & SKB_GSO_UDP))
+       if (!(skb_shinfo(skb)->gso_type & (SKB_GSO_UDP | SKB_GSO_UDP_L4)))
                goto out;
 
        if (!pskb_may_pull(skb, sizeof(struct udphdr)))
                goto out;
 
+       if (skb_shinfo(skb)->gso_type & SKB_GSO_UDP_L4)
+               return __udp4_gso_segment(skb, features);
+
        mss = skb_shinfo(skb)->gso_size;
        if (unlikely(skb->len <= mss))
                goto out;
index 4a87f9428ca519b475f8feaceaaa3a225bcfe6d2..5b3f2f89ef41c3276ef4b478683bd9ab04a1d3da 100644 (file)
@@ -88,9 +88,11 @@ static struct sk_buff *ipv6_gso_segment(struct sk_buff *skb,
 
        if (skb->encapsulation &&
            skb_shinfo(skb)->gso_type & (SKB_GSO_IPXIP4 | SKB_GSO_IPXIP6))
-               udpfrag = proto == IPPROTO_UDP && encap;
+               udpfrag = proto == IPPROTO_UDP && encap &&
+                         (skb_shinfo(skb)->gso_type & SKB_GSO_UDP);
        else
-               udpfrag = proto == IPPROTO_UDP && !skb->encapsulation;
+               udpfrag = proto == IPPROTO_UDP && !skb->encapsulation &&
+                         (skb_shinfo(skb)->gso_type & SKB_GSO_UDP);
 
        ops = rcu_dereference(inet6_offloads[proto]);
        if (likely(ops && ops->callbacks.gso_segment)) {
index 2a04dc9c781b5f236fbe6dc3804f5eeed174ee44..f7b85b1e6b3e6d06bbeafd6ecfed025ef71143d7 100644 (file)
 #include <net/ip6_checksum.h>
 #include "ip6_offload.h"
 
+static struct sk_buff *__udp6_gso_segment(struct sk_buff *gso_skb,
+                                         netdev_features_t features)
+{
+       const struct ipv6hdr *ip6h = ipv6_hdr(gso_skb);
+       unsigned int mss = skb_shinfo(gso_skb)->gso_size;
+
+       if (!can_checksum_protocol(features, htons(ETH_P_IPV6)))
+               return ERR_PTR(-EIO);
+
+       return __udp_gso_segment(gso_skb, features, mss,
+                                udp_v6_check(sizeof(struct udphdr) + mss,
+                                             &ip6h->saddr, &ip6h->daddr, 0));
+}
+
 static struct sk_buff *udp6_ufo_fragment(struct sk_buff *skb,
                                         netdev_features_t features)
 {
@@ -42,12 +56,15 @@ static struct sk_buff *udp6_ufo_fragment(struct sk_buff *skb,
                const struct ipv6hdr *ipv6h;
                struct udphdr *uh;
 
-               if (!(skb_shinfo(skb)->gso_type & SKB_GSO_UDP))
+               if (!(skb_shinfo(skb)->gso_type & (SKB_GSO_UDP | SKB_GSO_UDP_L4)))
                        goto out;
 
                if (!pskb_may_pull(skb, sizeof(struct udphdr)))
                        goto out;
 
+               if (skb_shinfo(skb)->gso_type & SKB_GSO_UDP_L4)
+                       return __udp6_gso_segment(skb, features);
+
                /* Do software UFO. Complete and fill in the UDP checksum as HW cannot
                 * do checksum of UDP packets sent as multiple IP fragments.
                 */