Commit 79c441ae authored by Guillaume Nault's avatar Guillaume Nault Committed by David S. Miller

ppp: implement x-netns support

Let packets move from one netns to the other at PPP encapsulation and
decapsulation time.

PPP units and channels remain in the netns in which they were
originally created. Only the net_device may move to a different
namespace. Cross netns handling is thus transparent to lower PPP
layers (PPPoE, L2TP, etc.).

PPP devices are automatically unregistered when their netns gets
removed. So read() and poll() on the unit file descriptor will
respectively receive EOF and POLLHUP. Channels aren't affected.
Signed-off-by: default avatarGuillaume Nault <g.nault@alphalink.fr>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 542a64c7
...@@ -283,6 +283,8 @@ static int unit_set(struct idr *p, void *ptr, int n); ...@@ -283,6 +283,8 @@ static int unit_set(struct idr *p, void *ptr, int n);
static void unit_put(struct idr *p, int n); static void unit_put(struct idr *p, int n);
static void *unit_find(struct idr *p, int n); static void *unit_find(struct idr *p, int n);
static const struct net_device_ops ppp_netdev_ops;
static struct class *ppp_class; static struct class *ppp_class;
/* per net-namespace data */ /* per net-namespace data */
...@@ -919,12 +921,21 @@ static __net_init int ppp_init_net(struct net *net) ...@@ -919,12 +921,21 @@ static __net_init int ppp_init_net(struct net *net)
static __net_exit void ppp_exit_net(struct net *net) static __net_exit void ppp_exit_net(struct net *net)
{ {
struct ppp_net *pn = net_generic(net, ppp_net_id); struct ppp_net *pn = net_generic(net, ppp_net_id);
struct net_device *dev;
struct net_device *aux;
struct ppp *ppp; struct ppp *ppp;
LIST_HEAD(list); LIST_HEAD(list);
int id; int id;
rtnl_lock(); rtnl_lock();
for_each_netdev_safe(net, dev, aux) {
if (dev->netdev_ops == &ppp_netdev_ops)
unregister_netdevice_queue(dev, &list);
}
idr_for_each_entry(&pn->units_idr, ppp, id) idr_for_each_entry(&pn->units_idr, ppp, id)
/* Skip devices already unregistered by previous loop */
if (!net_eq(dev_net(ppp->dev), net))
unregister_netdevice_queue(ppp->dev, &list); unregister_netdevice_queue(ppp->dev, &list);
unregister_netdevice_many(&list); unregister_netdevice_many(&list);
...@@ -1017,6 +1028,7 @@ ppp_start_xmit(struct sk_buff *skb, struct net_device *dev) ...@@ -1017,6 +1028,7 @@ ppp_start_xmit(struct sk_buff *skb, struct net_device *dev)
proto = npindex_to_proto[npi]; proto = npindex_to_proto[npi];
put_unaligned_be16(proto, pp); put_unaligned_be16(proto, pp);
skb_scrub_packet(skb, !net_eq(ppp->ppp_net, dev_net(dev)));
skb_queue_tail(&ppp->file.xq, skb); skb_queue_tail(&ppp->file.xq, skb);
ppp_xmit_process(ppp); ppp_xmit_process(ppp);
return NETDEV_TX_OK; return NETDEV_TX_OK;
...@@ -1137,7 +1149,6 @@ static void ppp_setup(struct net_device *dev) ...@@ -1137,7 +1149,6 @@ static void ppp_setup(struct net_device *dev)
dev->tx_queue_len = 3; dev->tx_queue_len = 3;
dev->type = ARPHRD_PPP; dev->type = ARPHRD_PPP;
dev->flags = IFF_POINTOPOINT | IFF_NOARP | IFF_MULTICAST; dev->flags = IFF_POINTOPOINT | IFF_NOARP | IFF_MULTICAST;
dev->features |= NETIF_F_NETNS_LOCAL;
netif_keep_dst(dev); netif_keep_dst(dev);
} }
...@@ -1900,6 +1911,8 @@ ppp_receive_nonmp_frame(struct ppp *ppp, struct sk_buff *skb) ...@@ -1900,6 +1911,8 @@ ppp_receive_nonmp_frame(struct ppp *ppp, struct sk_buff *skb)
skb->dev = ppp->dev; skb->dev = ppp->dev;
skb->protocol = htons(npindex_to_ethertype[npi]); skb->protocol = htons(npindex_to_ethertype[npi]);
skb_reset_mac_header(skb); skb_reset_mac_header(skb);
skb_scrub_packet(skb, !net_eq(ppp->ppp_net,
dev_net(ppp->dev)));
netif_rx(skb); netif_rx(skb);
} }
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment