aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--include/linux/netdevice.h14
-rw-r--r--net/core/dev.c36
2 files changed, 44 insertions, 6 deletions
diff --git a/include/linux/netdevice.h b/include/linux/netdevice.h
index 880d56565828..2861565a27d9 100644
--- a/include/linux/netdevice.h
+++ b/include/linux/netdevice.h
@@ -2248,9 +2248,17 @@ static inline int skb_gso_ok(struct sk_buff *skb, int features)
2248 2248
2249static inline int netif_needs_gso(struct net_device *dev, struct sk_buff *skb) 2249static inline int netif_needs_gso(struct net_device *dev, struct sk_buff *skb)
2250{ 2250{
2251 return skb_is_gso(skb) && 2251 if (skb_is_gso(skb)) {
2252 (!skb_gso_ok(skb, dev->features) || 2252 int features = dev->features;
2253 unlikely(skb->ip_summed != CHECKSUM_PARTIAL)); 2253
2254 if (skb->protocol == htons(ETH_P_8021Q) || skb->vlan_tci)
2255 features &= dev->vlan_features;
2256
2257 return (!skb_gso_ok(skb, features) ||
2258 unlikely(skb->ip_summed != CHECKSUM_PARTIAL));
2259 }
2260
2261 return 0;
2254} 2262}
2255 2263
2256static inline void netif_set_gso_max_size(struct net_device *dev, 2264static inline void netif_set_gso_max_size(struct net_device *dev,
diff --git a/net/core/dev.c b/net/core/dev.c
index 4c3ac53e4b16..1bfd96b1fbd4 100644
--- a/net/core/dev.c
+++ b/net/core/dev.c
@@ -1694,7 +1694,12 @@ static bool can_checksum_protocol(unsigned long features, __be16 protocol)
1694 1694
1695static bool dev_can_checksum(struct net_device *dev, struct sk_buff *skb) 1695static bool dev_can_checksum(struct net_device *dev, struct sk_buff *skb)
1696{ 1696{
1697 if (can_checksum_protocol(dev->features, skb->protocol)) 1697 int features = dev->features;
1698
1699 if (vlan_tx_tag_present(skb))
1700 features &= dev->vlan_features;
1701
1702 if (can_checksum_protocol(features, skb->protocol))
1698 return true; 1703 return true;
1699 1704
1700 if (skb->protocol == htons(ETH_P_8021Q)) { 1705 if (skb->protocol == htons(ETH_P_8021Q)) {
@@ -1793,6 +1798,16 @@ struct sk_buff *skb_gso_segment(struct sk_buff *skb, int features)
1793 __be16 type = skb->protocol; 1798 __be16 type = skb->protocol;
1794 int err; 1799 int err;
1795 1800
1801 if (type == htons(ETH_P_8021Q)) {
1802 struct vlan_ethhdr *veh;
1803
1804 if (unlikely(!pskb_may_pull(skb, VLAN_ETH_HLEN)))
1805 return ERR_PTR(-EINVAL);
1806
1807 veh = (struct vlan_ethhdr *)skb->data;
1808 type = veh->h_vlan_encapsulated_proto;
1809 }
1810
1796 skb_reset_mac_header(skb); 1811 skb_reset_mac_header(skb);
1797 skb->mac_len = skb->network_header - skb->mac_header; 1812 skb->mac_len = skb->network_header - skb->mac_header;
1798 __skb_pull(skb, skb->mac_len); 1813 __skb_pull(skb, skb->mac_len);
@@ -1964,9 +1979,14 @@ static inline void skb_orphan_try(struct sk_buff *skb)
1964static inline int skb_needs_linearize(struct sk_buff *skb, 1979static inline int skb_needs_linearize(struct sk_buff *skb,
1965 struct net_device *dev) 1980 struct net_device *dev)
1966{ 1981{
1982 int features = dev->features;
1983
1984 if (skb->protocol == htons(ETH_P_8021Q) || vlan_tx_tag_present(skb))
1985 features &= dev->vlan_features;
1986
1967 return skb_is_nonlinear(skb) && 1987 return skb_is_nonlinear(skb) &&
1968 ((skb_has_frag_list(skb) && !(dev->features & NETIF_F_FRAGLIST)) || 1988 ((skb_has_frag_list(skb) && !(features & NETIF_F_FRAGLIST)) ||
1969 (skb_shinfo(skb)->nr_frags && (!(dev->features & NETIF_F_SG) || 1989 (skb_shinfo(skb)->nr_frags && (!(features & NETIF_F_SG) ||
1970 illegal_highdma(dev, skb)))); 1990 illegal_highdma(dev, skb))));
1971} 1991}
1972 1992
@@ -1989,6 +2009,15 @@ int dev_hard_start_xmit(struct sk_buff *skb, struct net_device *dev,
1989 2009
1990 skb_orphan_try(skb); 2010 skb_orphan_try(skb);
1991 2011
2012 if (vlan_tx_tag_present(skb) &&
2013 !(dev->features & NETIF_F_HW_VLAN_TX)) {
2014 skb = __vlan_put_tag(skb, vlan_tx_tag_get(skb));
2015 if (unlikely(!skb))
2016 goto out;
2017
2018 skb->vlan_tci = 0;
2019 }
2020
1992 if (netif_needs_gso(dev, skb)) { 2021 if (netif_needs_gso(dev, skb)) {
1993 if (unlikely(dev_gso_segment(skb))) 2022 if (unlikely(dev_gso_segment(skb)))
1994 goto out_kfree_skb; 2023 goto out_kfree_skb;
@@ -2050,6 +2079,7 @@ out_kfree_gso_skb:
2050 skb->destructor = DEV_GSO_CB(skb)->destructor; 2079 skb->destructor = DEV_GSO_CB(skb)->destructor;
2051out_kfree_skb: 2080out_kfree_skb:
2052 kfree_skb(skb); 2081 kfree_skb(skb);
2082out:
2053 return rc; 2083 return rc;
2054} 2084}
2055 2085