aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/net/macvtap.c
diff options
context:
space:
mode:
authorPravin B Shelar <pshelar@nicira.com>2013-02-11 04:27:41 -0500
committerDavid S. Miller <davem@davemloft.net>2013-02-13 13:30:10 -0500
commitc9af6db4c11ccc6c3e7f19bbc15d54023956f97c (patch)
treec596e747d8940b848931ac31701e245a6c0efaf6 /drivers/net/macvtap.c
parentb8fa4100350432504df438014e2e5e9c1bbb6325 (diff)
net: Fix possible wrong checksum generation.
Patch cef401de7be8c4e (net: fix possible wrong checksum generation) fixed wrong checksum calculation but it broke TSO by defining new GSO type but not a netdev feature for that type. net_gso_ok() would not allow hardware checksum/segmentation offload of such packets without the feature. Following patch fixes TSO and wrong checksum. This patch uses same logic that Eric Dumazet used. Patch introduces new flag SKBTX_SHARED_FRAG if at least one frag can be modified by the user. but SKBTX_SHARED_FRAG flag is kept in skb shared info tx_flags rather than gso_type. tx_flags is better compared to gso_type since we can have skb with shared frag without gso packet. It does not link SHARED_FRAG to GSO, So there is no need to define netdev feature for this. Signed-off-by: Pravin B Shelar <pshelar@nicira.com> Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'drivers/net/macvtap.c')
-rw-r--r--drivers/net/macvtap.c4
1 files changed, 2 insertions, 2 deletions
diff --git a/drivers/net/macvtap.c b/drivers/net/macvtap.c
index b181dfb3d6d6..97243011d319 100644
--- a/drivers/net/macvtap.c
+++ b/drivers/net/macvtap.c
@@ -543,7 +543,6 @@ static int zerocopy_sg_from_iovec(struct sk_buff *skb, const struct iovec *from,
543 skb->data_len += len; 543 skb->data_len += len;
544 skb->len += len; 544 skb->len += len;
545 skb->truesize += truesize; 545 skb->truesize += truesize;
546 skb_shinfo(skb)->gso_type |= SKB_GSO_SHARED_FRAG;
547 atomic_add(truesize, &skb->sk->sk_wmem_alloc); 546 atomic_add(truesize, &skb->sk->sk_wmem_alloc);
548 while (len) { 547 while (len) {
549 int off = base & ~PAGE_MASK; 548 int off = base & ~PAGE_MASK;
@@ -599,7 +598,7 @@ static int macvtap_skb_from_vnet_hdr(struct sk_buff *skb,
599 598
600 if (vnet_hdr->gso_type != VIRTIO_NET_HDR_GSO_NONE) { 599 if (vnet_hdr->gso_type != VIRTIO_NET_HDR_GSO_NONE) {
601 skb_shinfo(skb)->gso_size = vnet_hdr->gso_size; 600 skb_shinfo(skb)->gso_size = vnet_hdr->gso_size;
602 skb_shinfo(skb)->gso_type |= gso_type; 601 skb_shinfo(skb)->gso_type = gso_type;
603 602
604 /* Header must be checked, and gso_segs computed. */ 603 /* Header must be checked, and gso_segs computed. */
605 skb_shinfo(skb)->gso_type |= SKB_GSO_DODGY; 604 skb_shinfo(skb)->gso_type |= SKB_GSO_DODGY;
@@ -743,6 +742,7 @@ static ssize_t macvtap_get_user(struct macvtap_queue *q, struct msghdr *m,
743 if (zerocopy) { 742 if (zerocopy) {
744 skb_shinfo(skb)->destructor_arg = m->msg_control; 743 skb_shinfo(skb)->destructor_arg = m->msg_control;
745 skb_shinfo(skb)->tx_flags |= SKBTX_DEV_ZEROCOPY; 744 skb_shinfo(skb)->tx_flags |= SKBTX_DEV_ZEROCOPY;
745 skb_shinfo(skb)->tx_flags |= SKBTX_SHARED_FRAG;
746 } 746 }
747 if (vlan) 747 if (vlan)
748 macvlan_start_xmit(skb, vlan->dev); 748 macvlan_start_xmit(skb, vlan->dev);