-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
In the case where a non-MPLS packet is received and an MPLS stack is added it may well be the case that the original skb is GSO but the NIC used for transmit does not support GSO of MPLS packets. The aim of this code is to provide GSO in software for MPLS packets whose skbs are GSO. SKB Usage: When an implementation adds an MPLS stack to a non-MPLS packet it should do the following to skb metadata: * Set skb->inner_protocol to the old non-MPLS ethertype of the packet. skb->inner_protocol is added by this patch. * Set skb->protocol to the new MPLS ethertype of the packet. * Set skb->network_header to correspond to the end of the L3 header, including the MPLS label stack. I have posted a patch, "[PATCH v3.29] datapath: Add basic MPLS support to kernel" which adds MPLS support to the kernel datapath of Open vSwtich. That patch sets the above requirements in datapath/actions.c:push_mpls() and was used to exercise this code. The datapath patch is against the Open vSwtich tree but it is intended that it be added to the Open vSwtich code present in the mainline Linux kernel at some point. Features: I believe that the approach that I have taken is at least partially consistent with the handling of other protocols. Jesse, I understand that you have some ideas here. I am more than happy to change my implementation. This patch adds dev->mpls_features which may be used by devices to advertise features supported for MPLS packets. A new NETIF_F_MPLS_GSO feature is added for devices which support hardware MPLS GSO offload. Currently no devices support this and MPLS GSO always falls back to software. Alternate Implementation: One possible alternate implementation is to teach netif_skb_features() and skb_network_protocol() about MPLS, in a similar way to their understanding of VLANs. I believe this would avoid the need for net/mpls/mpls_gso.c and in particular the calls to __skb_push() and __skb_push() in mpls_gso_segment(). I have decided on the implementation in this patch as it should not introduce any overhead in the case where mpls_gso is not compiled into the kernel or inserted as a module. MPLS GSO suggested by Jesse Gross. Based in part on "v4 GRE: Add TCP segmentation offload for GRE" by Pravin B Shelar. Cc: Jesse Gross <jesse@nicira.com> Cc: Pravin B Shelar <pshelar@nicira.com> Signed-off-by: Simon Horman <horms@verge.net.au> Signed-off-by: David S. Miller <davem@davemloft.net>
- Loading branch information
Simon Horman
authored and
David S. Miller
committed
May 28, 2013
1 parent
1a37e41
commit 0d89d20
Showing
15 changed files
with
143 additions
and
3 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,9 @@ | ||
# | ||
# MPLS configuration | ||
# | ||
config NET_MPLS_GSO | ||
tristate "MPLS: GSO support" | ||
help | ||
This is helper module to allow segmentation of non-MPLS GSO packets | ||
that have had MPLS stack entries pushed onto them and thus | ||
become MPLS GSO packets. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,4 @@ | ||
# | ||
# Makefile for MPLS. | ||
# | ||
obj-y += mpls_gso.o |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,108 @@ | ||
/* | ||
* MPLS GSO Support | ||
* | ||
* Authors: Simon Horman (horms@verge.net.au) | ||
* | ||
* This program is free software; you can redistribute it and/or | ||
* modify it under the terms of the GNU General Public License | ||
* as published by the Free Software Foundation; either version | ||
* 2 of the License, or (at your option) any later version. | ||
* | ||
* Based on: GSO portions of net/ipv4/gre.c | ||
*/ | ||
|
||
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt | ||
|
||
#include <linux/err.h> | ||
#include <linux/module.h> | ||
#include <linux/netdev_features.h> | ||
#include <linux/netdevice.h> | ||
#include <linux/skbuff.h> | ||
|
||
static struct sk_buff *mpls_gso_segment(struct sk_buff *skb, | ||
netdev_features_t features) | ||
{ | ||
struct sk_buff *segs = ERR_PTR(-EINVAL); | ||
netdev_features_t mpls_features; | ||
__be16 mpls_protocol; | ||
|
||
if (unlikely(skb_shinfo(skb)->gso_type & | ||
~(SKB_GSO_TCPV4 | | ||
SKB_GSO_TCPV6 | | ||
SKB_GSO_UDP | | ||
SKB_GSO_DODGY | | ||
SKB_GSO_TCP_ECN | | ||
SKB_GSO_GRE | | ||
SKB_GSO_MPLS))) | ||
goto out; | ||
|
||
/* Setup inner SKB. */ | ||
mpls_protocol = skb->protocol; | ||
skb->protocol = skb->inner_protocol; | ||
|
||
/* Push back the mac header that skb_mac_gso_segment() has pulled. | ||
* It will be re-pulled by the call to skb_mac_gso_segment() below | ||
*/ | ||
__skb_push(skb, skb->mac_len); | ||
|
||
/* Segment inner packet. */ | ||
mpls_features = skb->dev->mpls_features & netif_skb_features(skb); | ||
segs = skb_mac_gso_segment(skb, mpls_features); | ||
|
||
|
||
/* Restore outer protocol. */ | ||
skb->protocol = mpls_protocol; | ||
|
||
/* Re-pull the mac header that the call to skb_mac_gso_segment() | ||
* above pulled. It will be re-pushed after returning | ||
* skb_mac_gso_segment(), an indirect caller of this function. | ||
*/ | ||
__skb_push(skb, skb->data - skb_mac_header(skb)); | ||
|
||
out: | ||
return segs; | ||
} | ||
|
||
static int mpls_gso_send_check(struct sk_buff *skb) | ||
{ | ||
return 0; | ||
} | ||
|
||
static struct packet_offload mpls_mc_offload = { | ||
.type = cpu_to_be16(ETH_P_MPLS_MC), | ||
.callbacks = { | ||
.gso_send_check = mpls_gso_send_check, | ||
.gso_segment = mpls_gso_segment, | ||
}, | ||
}; | ||
|
||
static struct packet_offload mpls_uc_offload = { | ||
.type = cpu_to_be16(ETH_P_MPLS_UC), | ||
.callbacks = { | ||
.gso_send_check = mpls_gso_send_check, | ||
.gso_segment = mpls_gso_segment, | ||
}, | ||
}; | ||
|
||
static int __init mpls_gso_init(void) | ||
{ | ||
pr_info("MPLS GSO support\n"); | ||
|
||
dev_add_offload(&mpls_uc_offload); | ||
dev_add_offload(&mpls_mc_offload); | ||
|
||
return 0; | ||
} | ||
|
||
static void __exit mpls_gso_exit(void) | ||
{ | ||
dev_remove_offload(&mpls_uc_offload); | ||
dev_remove_offload(&mpls_mc_offload); | ||
} | ||
|
||
module_init(mpls_gso_init); | ||
module_exit(mpls_gso_exit); | ||
|
||
MODULE_DESCRIPTION("MPLS GSO support"); | ||
MODULE_AUTHOR("Simon Horman (horms@verge.net.au)"); | ||
MODULE_LICENSE("GPL"); |