diff options
author | Vlad Yasevich <vyasevic@redhat.com> | 2013-08-16 15:25:00 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2013-08-20 16:09:11 -0400 |
commit | a567dd6252263c8147b7269df5d03d9e31463e11 (patch) | |
tree | 862bea2a1641455e210e5d51cc3a187b6f62b124 /drivers/net/macvtap.c | |
parent | 7ed5c5ae96d23da22de95e1c7a239537acd378b1 (diff) |
macvtap: simplify usage of tap_features
In macvtap, tap_features specific the features of that the user
has specified via ioctl(). If we treat macvtap as a macvlan+tap
then we could all the tap a pseudo-device and give it other features
like SG and GSO. Then we can stop using the features of lower
device (macvlan) when forwarding the traffic the tap.
This solves the issue of possible checksum offload mismatch between
tap feature and macvlan features.
Signed-off-by: Vlad Yasevich <vyasevic@redhat.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'drivers/net/macvtap.c')
-rw-r--r-- | drivers/net/macvtap.c | 10 |
1 files changed, 6 insertions, 4 deletions
diff --git a/drivers/net/macvtap.c b/drivers/net/macvtap.c index b51db2abfe44..448f8a505cd9 100644 --- a/drivers/net/macvtap.c +++ b/drivers/net/macvtap.c | |||
@@ -68,6 +68,8 @@ static const struct proto_ops macvtap_socket_ops; | |||
68 | #define TUN_OFFLOADS (NETIF_F_HW_CSUM | NETIF_F_TSO_ECN | NETIF_F_TSO | \ | 68 | #define TUN_OFFLOADS (NETIF_F_HW_CSUM | NETIF_F_TSO_ECN | NETIF_F_TSO | \ |
69 | NETIF_F_TSO6 | NETIF_F_UFO) | 69 | NETIF_F_TSO6 | NETIF_F_UFO) |
70 | #define RX_OFFLOADS (NETIF_F_GRO | NETIF_F_LRO) | 70 | #define RX_OFFLOADS (NETIF_F_GRO | NETIF_F_LRO) |
71 | #define TAP_FEATURES (NETIF_F_GSO | NETIF_F_SG) | ||
72 | |||
71 | /* | 73 | /* |
72 | * RCU usage: | 74 | * RCU usage: |
73 | * The macvtap_queue and the macvlan_dev are loosely coupled, the | 75 | * The macvtap_queue and the macvlan_dev are loosely coupled, the |
@@ -278,7 +280,8 @@ static int macvtap_forward(struct net_device *dev, struct sk_buff *skb) | |||
278 | { | 280 | { |
279 | struct macvlan_dev *vlan = netdev_priv(dev); | 281 | struct macvlan_dev *vlan = netdev_priv(dev); |
280 | struct macvtap_queue *q = macvtap_get_queue(dev, skb); | 282 | struct macvtap_queue *q = macvtap_get_queue(dev, skb); |
281 | netdev_features_t features; | 283 | netdev_features_t features = TAP_FEATURES; |
284 | |||
282 | if (!q) | 285 | if (!q) |
283 | goto drop; | 286 | goto drop; |
284 | 287 | ||
@@ -289,7 +292,7 @@ static int macvtap_forward(struct net_device *dev, struct sk_buff *skb) | |||
289 | /* Apply the forward feature mask so that we perform segmentation | 292 | /* Apply the forward feature mask so that we perform segmentation |
290 | * according to users wishes. | 293 | * according to users wishes. |
291 | */ | 294 | */ |
292 | features = netif_skb_features(skb) & vlan->tap_features; | 295 | features |= vlan->tap_features; |
293 | if (netif_needs_gso(skb, features)) { | 296 | if (netif_needs_gso(skb, features)) { |
294 | struct sk_buff *segs = __skb_gso_segment(skb, features, false); | 297 | struct sk_buff *segs = __skb_gso_segment(skb, features, false); |
295 | 298 | ||
@@ -1064,8 +1067,7 @@ static int set_offload(struct macvtap_queue *q, unsigned long arg) | |||
1064 | /* tap_features are the same as features on tun/tap and | 1067 | /* tap_features are the same as features on tun/tap and |
1065 | * reflect user expectations. | 1068 | * reflect user expectations. |
1066 | */ | 1069 | */ |
1067 | vlan->tap_features = vlan->dev->features & | 1070 | vlan->tap_features = feature_mask; |
1068 | (feature_mask | ~TUN_OFFLOADS); | ||
1069 | vlan->set_features = features; | 1071 | vlan->set_features = features; |
1070 | netdev_update_features(vlan->dev); | 1072 | netdev_update_features(vlan->dev); |
1071 | 1073 | ||