aboutsummaryrefslogtreecommitdiffstats
path: root/net
diff options
context:
space:
mode:
authorToshiaki Makita <makita.toshiaki@lab.ntt.co.jp>2014-12-22 05:04:14 -0500
committerDavid S. Miller <davem@davemloft.net>2014-12-24 00:08:33 -0500
commit796f2da81bead71ffc91ef70912cd8d1827bf756 (patch)
treeb002f4fc70040e8763d53857582ae6cce5c145da /net
parente51a08b8df461919451b1a2bf4a730da0b994425 (diff)
net: Fix stacked vlan offload features computation
When vlan tags are stacked, it is very likely that the outer tag is stored in skb->vlan_tci and skb->protocol shows the inner tag's vlan_proto. Currently netif_skb_features() first looks at skb->protocol even if there is the outer tag in vlan_tci, thus it incorrectly retrieves the protocol encapsulated by the inner vlan instead of the inner vlan protocol. This allows GSO packets to be passed to HW and they end up being corrupted. Fixes: 58e998c6d239 ("offloading: Force software GSO for multiple vlan tags.") Signed-off-by: Toshiaki Makita <makita.toshiaki@lab.ntt.co.jp> Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net')
-rw-r--r--net/core/dev.c13
1 files changed, 8 insertions, 5 deletions
diff --git a/net/core/dev.c b/net/core/dev.c
index 67b6210a589a..bd44e28c735e 100644
--- a/net/core/dev.c
+++ b/net/core/dev.c
@@ -2570,11 +2570,14 @@ netdev_features_t netif_skb_features(struct sk_buff *skb)
2570 if (gso_segs > dev->gso_max_segs || gso_segs < dev->gso_min_segs) 2570 if (gso_segs > dev->gso_max_segs || gso_segs < dev->gso_min_segs)
2571 features &= ~NETIF_F_GSO_MASK; 2571 features &= ~NETIF_F_GSO_MASK;
2572 2572
2573 if (protocol == htons(ETH_P_8021Q) || protocol == htons(ETH_P_8021AD)) { 2573 if (!vlan_tx_tag_present(skb)) {
2574 struct vlan_ethhdr *veh = (struct vlan_ethhdr *)skb->data; 2574 if (unlikely(protocol == htons(ETH_P_8021Q) ||
2575 protocol = veh->h_vlan_encapsulated_proto; 2575 protocol == htons(ETH_P_8021AD))) {
2576 } else if (!vlan_tx_tag_present(skb)) { 2576 struct vlan_ethhdr *veh = (struct vlan_ethhdr *)skb->data;
2577 return harmonize_features(skb, features); 2577 protocol = veh->h_vlan_encapsulated_proto;
2578 } else {
2579 return harmonize_features(skb, features);
2580 }
2578 } 2581 }
2579 2582
2580 features = netdev_intersect_features(features, 2583 features = netdev_intersect_features(features,