diff options
author | Tom Herbert <therbert@google.com> | 2014-06-10 21:54:19 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2014-06-11 18:46:13 -0400 |
commit | 7e3cead5172927732f51fde77fef6f521e22f209 (patch) | |
tree | 7b7cad61aa68ba302c395b9a1ea3dafb69881d0d | |
parent | 5d0c2b95bc57cf8fdc0e7b3e9d7e751eb65ad052 (diff) |
net: Save software checksum complete
In skb_checksum complete, if we need to compute the checksum for the
packet (via skb_checksum) save the result as CHECKSUM_COMPLETE.
Subsequent checksum verification can use this.
Also, added csum_complete_sw flag to distinguish between software and
hardware generated checksum complete, we should always be able to trust
the software computation.
Signed-off-by: Tom Herbert <therbert@google.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
-rw-r--r-- | include/linux/skbuff.h | 3 | ||||
-rw-r--r-- | net/core/datagram.c | 14 | ||||
-rw-r--r-- | net/ipv4/gre_offload.c | 6 | ||||
-rw-r--r-- | net/sunrpc/socklib.c | 3 |
4 files changed, 17 insertions, 9 deletions
diff --git a/include/linux/skbuff.h b/include/linux/skbuff.h index 72a53805858a..5b5cd3189c98 100644 --- a/include/linux/skbuff.h +++ b/include/linux/skbuff.h | |||
@@ -573,7 +573,8 @@ struct sk_buff { | |||
573 | __u8 encapsulation:1; | 573 | __u8 encapsulation:1; |
574 | __u8 encap_hdr_csum:1; | 574 | __u8 encap_hdr_csum:1; |
575 | __u8 csum_valid:1; | 575 | __u8 csum_valid:1; |
576 | /* 4/6 bit hole (depending on ndisc_nodetype presence) */ | 576 | __u8 csum_complete_sw:1; |
577 | /* 3/5 bit hole (depending on ndisc_nodetype presence) */ | ||
577 | kmemcheck_bitfield_end(flags2); | 578 | kmemcheck_bitfield_end(flags2); |
578 | 579 | ||
579 | #if defined CONFIG_NET_DMA || defined CONFIG_NET_RX_BUSY_POLL | 580 | #if defined CONFIG_NET_DMA || defined CONFIG_NET_RX_BUSY_POLL |
diff --git a/net/core/datagram.c b/net/core/datagram.c index a16ed7bbe376..6b1c04ca1d50 100644 --- a/net/core/datagram.c +++ b/net/core/datagram.c | |||
@@ -739,11 +739,15 @@ __sum16 __skb_checksum_complete_head(struct sk_buff *skb, int len) | |||
739 | __sum16 sum; | 739 | __sum16 sum; |
740 | 740 | ||
741 | sum = csum_fold(skb_checksum(skb, 0, len, skb->csum)); | 741 | sum = csum_fold(skb_checksum(skb, 0, len, skb->csum)); |
742 | if (likely(!sum)) { | 742 | if (unlikely(skb->ip_summed == CHECKSUM_COMPLETE) && !sum && |
743 | if (unlikely(skb->ip_summed == CHECKSUM_COMPLETE)) | 743 | !skb->csum_complete_sw) |
744 | netdev_rx_csum_fault(skb->dev); | 744 | netdev_rx_csum_fault(skb->dev); |
745 | skb->ip_summed = CHECKSUM_UNNECESSARY; | 745 | |
746 | } | 746 | /* Save checksum complete for later use */ |
747 | skb->csum = sum; | ||
748 | skb->ip_summed = CHECKSUM_COMPLETE; | ||
749 | skb->csum_complete_sw = 1; | ||
750 | |||
747 | return sum; | 751 | return sum; |
748 | } | 752 | } |
749 | EXPORT_SYMBOL(__skb_checksum_complete_head); | 753 | EXPORT_SYMBOL(__skb_checksum_complete_head); |
diff --git a/net/ipv4/gre_offload.c b/net/ipv4/gre_offload.c index 24deb3928b9e..eb92deb12666 100644 --- a/net/ipv4/gre_offload.c +++ b/net/ipv4/gre_offload.c | |||
@@ -131,10 +131,12 @@ static __sum16 gro_skb_checksum(struct sk_buff *skb) | |||
131 | csum_partial(skb->data, skb_gro_offset(skb), 0)); | 131 | csum_partial(skb->data, skb_gro_offset(skb), 0)); |
132 | sum = csum_fold(NAPI_GRO_CB(skb)->csum); | 132 | sum = csum_fold(NAPI_GRO_CB(skb)->csum); |
133 | if (unlikely(skb->ip_summed == CHECKSUM_COMPLETE)) { | 133 | if (unlikely(skb->ip_summed == CHECKSUM_COMPLETE)) { |
134 | if (unlikely(!sum)) | 134 | if (unlikely(!sum) && !skb->csum_complete_sw) |
135 | netdev_rx_csum_fault(skb->dev); | 135 | netdev_rx_csum_fault(skb->dev); |
136 | } else | 136 | } else { |
137 | skb->ip_summed = CHECKSUM_COMPLETE; | 137 | skb->ip_summed = CHECKSUM_COMPLETE; |
138 | skb->csum_complete_sw = 1; | ||
139 | } | ||
138 | 140 | ||
139 | return sum; | 141 | return sum; |
140 | } | 142 | } |
diff --git a/net/sunrpc/socklib.c b/net/sunrpc/socklib.c index 0a648c502fc3..2df87f78e518 100644 --- a/net/sunrpc/socklib.c +++ b/net/sunrpc/socklib.c | |||
@@ -173,7 +173,8 @@ int csum_partial_copy_to_xdr(struct xdr_buf *xdr, struct sk_buff *skb) | |||
173 | return -1; | 173 | return -1; |
174 | if (csum_fold(desc.csum)) | 174 | if (csum_fold(desc.csum)) |
175 | return -1; | 175 | return -1; |
176 | if (unlikely(skb->ip_summed == CHECKSUM_COMPLETE)) | 176 | if (unlikely(skb->ip_summed == CHECKSUM_COMPLETE) && |
177 | !skb->csum_complete_sw) | ||
177 | netdev_rx_csum_fault(skb->dev); | 178 | netdev_rx_csum_fault(skb->dev); |
178 | return 0; | 179 | return 0; |
179 | no_checksum: | 180 | no_checksum: |