From mboxrd@z Thu Jan 1 00:00:00 1970 From: Alex Gartrell Subject: Re: [RFC] GSO: Questions for IPVS and GSO for IPv6 Date: Tue, 29 Jul 2014 20:15:37 -0700 Message-ID: <53D86359.2000604@fb.com> References: Mime-Version: 1.0 Content-Type: text/plain; charset="windows-1252"; format=flowed Content-Transfer-Encoding: 7bit Cc: , kernel-team To: Julian Anastasov , Return-path: In-Reply-To: Sender: lvs-devel-owner@vger.kernel.org List-Id: netdev.vger.kernel.org Thanks for putting this up Julian. As a meta-point, working with upstream isn't necessarily something that has come easily or obviously to us corporate shills at Internet companies, but it's pretty obvious at this point that it was foolish for us to go it alone :) > [PATCH net] ipvs: properly declare tunnel encapsulation > > The tunneling method should properly use tunnel encapsulation. > Fixes problem with CHECKSUM_PARTIAL packets when TCP/UDP csum > offload is supported. > > Thanks to Alex Gartrell for reporting the problem, providing > solution and for all suggestions. > > Reported-by: Alex Gartrell > TODO: waiting for final ack from Alex before adding Signed-off-by line nit-picky comments below, but this looks good to me. > Signed-off-by: Julian Anastasov > --- > net/netfilter/ipvs/ip_vs_xmit.c | 20 ++++++++++++++++---- > 1 file changed, 16 insertions(+), 4 deletions(-) > > diff --git a/net/netfilter/ipvs/ip_vs_xmit.c b/net/netfilter/ipvs/ip_vs_xmit.c > index 73ba1cc..8c1172c 100644 > --- a/net/netfilter/ipvs/ip_vs_xmit.c > +++ b/net/netfilter/ipvs/ip_vs_xmit.c > @@ -38,6 +38,7 @@ > #include /* for ip_route_output */ > #include > #include > +#include > #include > #include > #include > @@ -862,14 +863,19 @@ ip_vs_tunnel_xmit(struct sk_buff *skb, struct ip_vs_conn *cp, > old_iph = ip_hdr(skb); > } > > - skb->transport_header = skb->network_header; > - > /* fix old IP header checksum */ > ip_send_check(old_iph); > > + skb = iptunnel_handle_offloads(skb, false, SKB_GSO_IPIP); > + if (IS_ERR(skb)) > + goto tx_error; > + > + skb->transport_header = skb->network_header; > + > skb_push(skb, sizeof(struct iphdr)); > skb_reset_network_header(skb); > memset(&(IPCB(skb)->opt), 0, sizeof(IPCB(skb)->opt)); > + skb_clear_hash(skb); We need this in ipv4 but not in ipv6? > > /* > * Push down and install the IPIP header. > @@ -900,7 +906,8 @@ ip_vs_tunnel_xmit(struct sk_buff *skb, struct ip_vs_conn *cp, > return NF_STOLEN; > > tx_error: > - kfree_skb(skb); > + if (!IS_ERR(skb)) > + kfree_skb(skb); > rcu_read_unlock(); > LeaveFunction(10); > return NF_STOLEN; > @@ -953,6 +960,10 @@ ip_vs_tunnel_xmit_v6(struct sk_buff *skb, struct ip_vs_conn *cp, > old_iph = ipv6_hdr(skb); > } > > + skb = iptunnel_handle_offloads(skb, false, SKB_GSO_SIT); It sounds like this is going to take a fair amount of work. Can we set gso_type_mask to 0 with a scary comment and push it as a fix and then do the right thing as a feature for a future release? > + if (IS_ERR(skb)) > + goto tx_error; > + > skb->transport_header = skb->network_header; > > skb_push(skb, sizeof(struct ipv6hdr)); > @@ -988,7 +999,8 @@ ip_vs_tunnel_xmit_v6(struct sk_buff *skb, struct ip_vs_conn *cp, > return NF_STOLEN; > > tx_error: > - kfree_skb(skb); > + if (!IS_ERR(skb)) > + kfree_skb(skb); > rcu_read_unlock(); > LeaveFunction(10); > return NF_STOLEN; >