tun: Implement ip link del tunXXX

This greatly simplifies testing to verify I have fixed the problems
with a tun device disappearing when the tun file descriptor is still
held open.

Further it allows removal network namespace operations for the tun
driver.  Reducing the network namespace handling in the driver to the
minimum.  i.e. When we are creating a tun device.

Signed-off-by: Eric W. Biederman <ebiederm@aristanetworks.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
Eric W. Biederman 2009-01-21 16:02:16 -08:00 committed by David S. Miller
parent aec191aa2a
commit f019a7a594

View File

@ -63,6 +63,7 @@
#include <linux/virtio_net.h> #include <linux/virtio_net.h>
#include <net/net_namespace.h> #include <net/net_namespace.h>
#include <net/netns/generic.h> #include <net/netns/generic.h>
#include <net/rtnetlink.h>
#include <asm/system.h> #include <asm/system.h>
#include <asm/uaccess.h> #include <asm/uaccess.h>
@ -812,6 +813,22 @@ static void tun_setup(struct net_device *dev)
dev->destructor = free_netdev; dev->destructor = free_netdev;
} }
/* Trivial set of netlink ops to allow deleting tun or tap
* device with netlink.
*/
static int tun_validate(struct nlattr *tb[], struct nlattr *data[])
{
return -EINVAL;
}
static struct rtnl_link_ops tun_link_ops __read_mostly = {
.kind = DRV_NAME,
.priv_size = sizeof(struct tun_struct),
.setup = tun_setup,
.validate = tun_validate,
};
static int tun_set_iff(struct net *net, struct file *file, struct ifreq *ifr) static int tun_set_iff(struct net *net, struct file *file, struct ifreq *ifr)
{ {
struct tun_struct *tun; struct tun_struct *tun;
@ -861,6 +878,7 @@ static int tun_set_iff(struct net *net, struct file *file, struct ifreq *ifr)
return -ENOMEM; return -ENOMEM;
dev_net_set(dev, net); dev_net_set(dev, net);
dev->rtnl_link_ops = &tun_link_ops;
tun = netdev_priv(dev); tun = netdev_priv(dev);
tun->dev = dev; tun->dev = dev;
@ -1317,29 +1335,6 @@ static const struct ethtool_ops tun_ethtool_ops = {
.set_rx_csum = tun_set_rx_csum .set_rx_csum = tun_set_rx_csum
}; };
static int tun_init_net(struct net *net)
{
return 0;
}
static void tun_exit_net(struct net *net)
{
struct net_device *dev, *next;
rtnl_lock();
for_each_netdev_safe(net, dev, next) {
if (dev->ethtool_ops != &tun_ethtool_ops)
continue;
DBG(KERN_INFO "%s cleaned up\n", dev->name);
unregister_netdevice(dev);
}
rtnl_unlock();
}
static struct pernet_operations tun_net_ops = {
.init = tun_init_net,
.exit = tun_exit_net,
};
static int __init tun_init(void) static int __init tun_init(void)
{ {
@ -1348,10 +1343,10 @@ static int __init tun_init(void)
printk(KERN_INFO "tun: %s, %s\n", DRV_DESCRIPTION, DRV_VERSION); printk(KERN_INFO "tun: %s, %s\n", DRV_DESCRIPTION, DRV_VERSION);
printk(KERN_INFO "tun: %s\n", DRV_COPYRIGHT); printk(KERN_INFO "tun: %s\n", DRV_COPYRIGHT);
ret = register_pernet_device(&tun_net_ops); ret = rtnl_link_register(&tun_link_ops);
if (ret) { if (ret) {
printk(KERN_ERR "tun: Can't register pernet ops\n"); printk(KERN_ERR "tun: Can't register link_ops\n");
goto err_pernet; goto err_linkops;
} }
ret = misc_register(&tun_miscdev); ret = misc_register(&tun_miscdev);
@ -1359,18 +1354,17 @@ static int __init tun_init(void)
printk(KERN_ERR "tun: Can't register misc device %d\n", TUN_MINOR); printk(KERN_ERR "tun: Can't register misc device %d\n", TUN_MINOR);
goto err_misc; goto err_misc;
} }
return 0; return 0;
err_misc: err_misc:
unregister_pernet_device(&tun_net_ops); rtnl_link_unregister(&tun_link_ops);
err_pernet: err_linkops:
return ret; return ret;
} }
static void tun_cleanup(void) static void tun_cleanup(void)
{ {
misc_deregister(&tun_miscdev); misc_deregister(&tun_miscdev);
unregister_pernet_device(&tun_net_ops); rtnl_link_unregister(&tun_link_ops);
} }
module_init(tun_init); module_init(tun_init);