diff options
author | Tom Herbert <therbert@google.com> | 2014-06-04 20:20:23 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2014-06-05 01:46:38 -0400 |
commit | 4749c09c37030ccdc44aecebe0f71b02a377fc14 (patch) | |
tree | d8bef1d48e73e2e5492765ae5348f1f0dffbad1c | |
parent | 0f4f4ffa7b7c3d29d0537a126145c9f8d8ed5dbc (diff) |
gre: Call gso_make_checksum
Call gso_make_checksum. This should have the benefit of using a
checksum that may have been previously computed for the packet.
This also adds NETIF_F_GSO_GRE_CSUM to differentiate devices that
offload GRE GSO with and without the GRE checksum offloaed.
Signed-off-by: Tom Herbert <therbert@google.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
-rw-r--r-- | include/linux/netdev_features.h | 2 | ||||
-rw-r--r-- | include/linux/skbuff.h | 2 | ||||
-rw-r--r-- | include/net/gre.h | 5 | ||||
-rw-r--r-- | net/ipv4/af_inet.c | 1 | ||||
-rw-r--r-- | net/ipv4/gre_demux.c | 3 | ||||
-rw-r--r-- | net/ipv4/gre_offload.c | 10 | ||||
-rw-r--r-- | net/ipv4/tcp_offload.c | 1 | ||||
-rw-r--r-- | net/ipv4/udp_offload.c | 3 | ||||
-rw-r--r-- | net/ipv6/ip6_offload.c | 1 | ||||
-rw-r--r-- | net/ipv6/udp_offload.c | 1 | ||||
-rw-r--r-- | net/mpls/mpls_gso.c | 1 |
11 files changed, 24 insertions, 6 deletions
diff --git a/include/linux/netdev_features.h b/include/linux/netdev_features.h index f1338e0f9866..e5a589435e2b 100644 --- a/include/linux/netdev_features.h +++ b/include/linux/netdev_features.h | |||
@@ -42,6 +42,7 @@ enum { | |||
42 | NETIF_F_TSO6_BIT, /* ... TCPv6 segmentation */ | 42 | NETIF_F_TSO6_BIT, /* ... TCPv6 segmentation */ |
43 | NETIF_F_FSO_BIT, /* ... FCoE segmentation */ | 43 | NETIF_F_FSO_BIT, /* ... FCoE segmentation */ |
44 | NETIF_F_GSO_GRE_BIT, /* ... GRE with TSO */ | 44 | NETIF_F_GSO_GRE_BIT, /* ... GRE with TSO */ |
45 | NETIF_F_GSO_GRE_CSUM_BIT, /* ... GRE with csum with TSO */ | ||
45 | NETIF_F_GSO_IPIP_BIT, /* ... IPIP tunnel with TSO */ | 46 | NETIF_F_GSO_IPIP_BIT, /* ... IPIP tunnel with TSO */ |
46 | NETIF_F_GSO_SIT_BIT, /* ... SIT tunnel with TSO */ | 47 | NETIF_F_GSO_SIT_BIT, /* ... SIT tunnel with TSO */ |
47 | NETIF_F_GSO_UDP_TUNNEL_BIT, /* ... UDP TUNNEL with TSO */ | 48 | NETIF_F_GSO_UDP_TUNNEL_BIT, /* ... UDP TUNNEL with TSO */ |
@@ -112,6 +113,7 @@ enum { | |||
112 | #define NETIF_F_RXFCS __NETIF_F(RXFCS) | 113 | #define NETIF_F_RXFCS __NETIF_F(RXFCS) |
113 | #define NETIF_F_RXALL __NETIF_F(RXALL) | 114 | #define NETIF_F_RXALL __NETIF_F(RXALL) |
114 | #define NETIF_F_GSO_GRE __NETIF_F(GSO_GRE) | 115 | #define NETIF_F_GSO_GRE __NETIF_F(GSO_GRE) |
116 | #define NETIF_F_GSO_GRE_CSUM __NETIF_F(GSO_GRE_CSUM) | ||
115 | #define NETIF_F_GSO_IPIP __NETIF_F(GSO_IPIP) | 117 | #define NETIF_F_GSO_IPIP __NETIF_F(GSO_IPIP) |
116 | #define NETIF_F_GSO_SIT __NETIF_F(GSO_SIT) | 118 | #define NETIF_F_GSO_SIT __NETIF_F(GSO_SIT) |
117 | #define NETIF_F_GSO_UDP_TUNNEL __NETIF_F(GSO_UDP_TUNNEL) | 119 | #define NETIF_F_GSO_UDP_TUNNEL __NETIF_F(GSO_UDP_TUNNEL) |
diff --git a/include/linux/skbuff.h b/include/linux/skbuff.h index 5a6d10a538f5..c705808bef9c 100644 --- a/include/linux/skbuff.h +++ b/include/linux/skbuff.h | |||
@@ -347,6 +347,8 @@ enum { | |||
347 | SKB_GSO_MPLS = 1 << 10, | 347 | SKB_GSO_MPLS = 1 << 10, |
348 | 348 | ||
349 | SKB_GSO_UDP_TUNNEL_CSUM = 1 << 11, | 349 | SKB_GSO_UDP_TUNNEL_CSUM = 1 << 11, |
350 | |||
351 | SKB_GSO_GRE_CSUM = 1 << 12, | ||
350 | }; | 352 | }; |
351 | 353 | ||
352 | #if BITS_PER_LONG > 32 | 354 | #if BITS_PER_LONG > 32 |
diff --git a/include/net/gre.h b/include/net/gre.h index 70046a0b0b89..b53182018743 100644 --- a/include/net/gre.h +++ b/include/net/gre.h | |||
@@ -37,9 +37,10 @@ void gre_build_header(struct sk_buff *skb, const struct tnl_ptk_info *tpi, | |||
37 | int hdr_len); | 37 | int hdr_len); |
38 | 38 | ||
39 | static inline struct sk_buff *gre_handle_offloads(struct sk_buff *skb, | 39 | static inline struct sk_buff *gre_handle_offloads(struct sk_buff *skb, |
40 | bool gre_csum) | 40 | bool csum) |
41 | { | 41 | { |
42 | return iptunnel_handle_offloads(skb, gre_csum, SKB_GSO_GRE); | 42 | return iptunnel_handle_offloads(skb, csum, |
43 | csum ? SKB_GSO_GRE_CSUM : SKB_GSO_GRE); | ||
43 | } | 44 | } |
44 | 45 | ||
45 | 46 | ||
diff --git a/net/ipv4/af_inet.c b/net/ipv4/af_inet.c index 0070ab87109b..d5e6836cf772 100644 --- a/net/ipv4/af_inet.c +++ b/net/ipv4/af_inet.c | |||
@@ -1254,6 +1254,7 @@ static struct sk_buff *inet_gso_segment(struct sk_buff *skb, | |||
1254 | SKB_GSO_DODGY | | 1254 | SKB_GSO_DODGY | |
1255 | SKB_GSO_TCP_ECN | | 1255 | SKB_GSO_TCP_ECN | |
1256 | SKB_GSO_GRE | | 1256 | SKB_GSO_GRE | |
1257 | SKB_GSO_GRE_CSUM | | ||
1257 | SKB_GSO_IPIP | | 1258 | SKB_GSO_IPIP | |
1258 | SKB_GSO_SIT | | 1259 | SKB_GSO_SIT | |
1259 | SKB_GSO_TCPV6 | | 1260 | SKB_GSO_TCPV6 | |
diff --git a/net/ipv4/gre_demux.c b/net/ipv4/gre_demux.c index fbfd829f4049..4e9619bca732 100644 --- a/net/ipv4/gre_demux.c +++ b/net/ipv4/gre_demux.c | |||
@@ -84,7 +84,8 @@ void gre_build_header(struct sk_buff *skb, const struct tnl_ptk_info *tpi, | |||
84 | ptr--; | 84 | ptr--; |
85 | } | 85 | } |
86 | if (tpi->flags&TUNNEL_CSUM && | 86 | if (tpi->flags&TUNNEL_CSUM && |
87 | !(skb_shinfo(skb)->gso_type & SKB_GSO_GRE)) { | 87 | !(skb_shinfo(skb)->gso_type & |
88 | (SKB_GSO_GRE|SKB_GSO_GRE_CSUM))) { | ||
88 | *ptr = 0; | 89 | *ptr = 0; |
89 | *(__sum16 *)ptr = csum_fold(skb_checksum(skb, 0, | 90 | *(__sum16 *)ptr = csum_fold(skb_checksum(skb, 0, |
90 | skb->len, 0)); | 91 | skb->len, 0)); |
diff --git a/net/ipv4/gre_offload.c b/net/ipv4/gre_offload.c index f1d32280cb54..24deb3928b9e 100644 --- a/net/ipv4/gre_offload.c +++ b/net/ipv4/gre_offload.c | |||
@@ -42,6 +42,7 @@ static struct sk_buff *gre_gso_segment(struct sk_buff *skb, | |||
42 | SKB_GSO_DODGY | | 42 | SKB_GSO_DODGY | |
43 | SKB_GSO_TCP_ECN | | 43 | SKB_GSO_TCP_ECN | |
44 | SKB_GSO_GRE | | 44 | SKB_GSO_GRE | |
45 | SKB_GSO_GRE_CSUM | | ||
45 | SKB_GSO_IPIP))) | 46 | SKB_GSO_IPIP))) |
46 | goto out; | 47 | goto out; |
47 | 48 | ||
@@ -55,6 +56,8 @@ static struct sk_buff *gre_gso_segment(struct sk_buff *skb, | |||
55 | goto out; | 56 | goto out; |
56 | 57 | ||
57 | csum = !!(greh->flags & GRE_CSUM); | 58 | csum = !!(greh->flags & GRE_CSUM); |
59 | if (csum) | ||
60 | skb->encap_hdr_csum = 1; | ||
58 | 61 | ||
59 | if (unlikely(!pskb_may_pull(skb, ghl))) | 62 | if (unlikely(!pskb_may_pull(skb, ghl))) |
60 | goto out; | 63 | goto out; |
@@ -94,10 +97,13 @@ static struct sk_buff *gre_gso_segment(struct sk_buff *skb, | |||
94 | } | 97 | } |
95 | } | 98 | } |
96 | 99 | ||
97 | greh = (struct gre_base_hdr *)(skb->data); | 100 | skb_reset_transport_header(skb); |
101 | |||
102 | greh = (struct gre_base_hdr *) | ||
103 | skb_transport_header(skb); | ||
98 | pcsum = (__be32 *)(greh + 1); | 104 | pcsum = (__be32 *)(greh + 1); |
99 | *pcsum = 0; | 105 | *pcsum = 0; |
100 | *(__sum16 *)pcsum = csum_fold(skb_checksum(skb, 0, skb->len, 0)); | 106 | *(__sum16 *)pcsum = gso_make_checksum(skb, 0); |
101 | } | 107 | } |
102 | __skb_push(skb, tnl_hlen - ghl); | 108 | __skb_push(skb, tnl_hlen - ghl); |
103 | 109 | ||
diff --git a/net/ipv4/tcp_offload.c b/net/ipv4/tcp_offload.c index c02f2d2e7bab..4e86c59ec7f7 100644 --- a/net/ipv4/tcp_offload.c +++ b/net/ipv4/tcp_offload.c | |||
@@ -57,6 +57,7 @@ struct sk_buff *tcp_gso_segment(struct sk_buff *skb, | |||
57 | SKB_GSO_TCP_ECN | | 57 | SKB_GSO_TCP_ECN | |
58 | SKB_GSO_TCPV6 | | 58 | SKB_GSO_TCPV6 | |
59 | SKB_GSO_GRE | | 59 | SKB_GSO_GRE | |
60 | SKB_GSO_GRE_CSUM | | ||
60 | SKB_GSO_IPIP | | 61 | SKB_GSO_IPIP | |
61 | SKB_GSO_SIT | | 62 | SKB_GSO_SIT | |
62 | SKB_GSO_MPLS | | 63 | SKB_GSO_MPLS | |
diff --git a/net/ipv4/udp_offload.c b/net/ipv4/udp_offload.c index 5c23f4765af9..7b1840110173 100644 --- a/net/ipv4/udp_offload.c +++ b/net/ipv4/udp_offload.c | |||
@@ -74,7 +74,8 @@ static struct sk_buff *udp4_ufo_fragment(struct sk_buff *skb, | |||
74 | SKB_GSO_UDP_TUNNEL | | 74 | SKB_GSO_UDP_TUNNEL | |
75 | SKB_GSO_UDP_TUNNEL_CSUM | | 75 | SKB_GSO_UDP_TUNNEL_CSUM | |
76 | SKB_GSO_IPIP | | 76 | SKB_GSO_IPIP | |
77 | SKB_GSO_GRE | SKB_GSO_MPLS) || | 77 | SKB_GSO_GRE | SKB_GSO_GRE_CSUM | |
78 | SKB_GSO_MPLS) || | ||
78 | !(type & (SKB_GSO_UDP)))) | 79 | !(type & (SKB_GSO_UDP)))) |
79 | goto out; | 80 | goto out; |
80 | 81 | ||
diff --git a/net/ipv6/ip6_offload.c b/net/ipv6/ip6_offload.c index d54c5744e3db..65eda2a8af48 100644 --- a/net/ipv6/ip6_offload.c +++ b/net/ipv6/ip6_offload.c | |||
@@ -97,6 +97,7 @@ static struct sk_buff *ipv6_gso_segment(struct sk_buff *skb, | |||
97 | SKB_GSO_DODGY | | 97 | SKB_GSO_DODGY | |
98 | SKB_GSO_TCP_ECN | | 98 | SKB_GSO_TCP_ECN | |
99 | SKB_GSO_GRE | | 99 | SKB_GSO_GRE | |
100 | SKB_GSO_GRE_CSUM | | ||
100 | SKB_GSO_IPIP | | 101 | SKB_GSO_IPIP | |
101 | SKB_GSO_SIT | | 102 | SKB_GSO_SIT | |
102 | SKB_GSO_UDP_TUNNEL | | 103 | SKB_GSO_UDP_TUNNEL | |
diff --git a/net/ipv6/udp_offload.c b/net/ipv6/udp_offload.c index 79da8b305ced..0ae3d98f83e0 100644 --- a/net/ipv6/udp_offload.c +++ b/net/ipv6/udp_offload.c | |||
@@ -65,6 +65,7 @@ static struct sk_buff *udp6_ufo_fragment(struct sk_buff *skb, | |||
65 | SKB_GSO_UDP_TUNNEL | | 65 | SKB_GSO_UDP_TUNNEL | |
66 | SKB_GSO_UDP_TUNNEL_CSUM | | 66 | SKB_GSO_UDP_TUNNEL_CSUM | |
67 | SKB_GSO_GRE | | 67 | SKB_GSO_GRE | |
68 | SKB_GSO_GRE_CSUM | | ||
68 | SKB_GSO_IPIP | | 69 | SKB_GSO_IPIP | |
69 | SKB_GSO_SIT | | 70 | SKB_GSO_SIT | |
70 | SKB_GSO_MPLS) || | 71 | SKB_GSO_MPLS) || |
diff --git a/net/mpls/mpls_gso.c b/net/mpls/mpls_gso.c index 851cd880b0c0..6b38d083e1c9 100644 --- a/net/mpls/mpls_gso.c +++ b/net/mpls/mpls_gso.c | |||
@@ -33,6 +33,7 @@ static struct sk_buff *mpls_gso_segment(struct sk_buff *skb, | |||
33 | SKB_GSO_DODGY | | 33 | SKB_GSO_DODGY | |
34 | SKB_GSO_TCP_ECN | | 34 | SKB_GSO_TCP_ECN | |
35 | SKB_GSO_GRE | | 35 | SKB_GSO_GRE | |
36 | SKB_GSO_GRE_CSUM | | ||
36 | SKB_GSO_IPIP | | 37 | SKB_GSO_IPIP | |
37 | SKB_GSO_MPLS))) | 38 | SKB_GSO_MPLS))) |
38 | goto out; | 39 | goto out; |