diff options
Diffstat (limited to 'drivers/net/tun.c')
-rw-r--r-- | drivers/net/tun.c | 53 |
1 files changed, 52 insertions, 1 deletions
diff --git a/drivers/net/tun.c b/drivers/net/tun.c index 5b5d87585d91..f359d6082c38 100644 --- a/drivers/net/tun.c +++ b/drivers/net/tun.c | |||
@@ -63,6 +63,7 @@ | |||
63 | #include <linux/if_tun.h> | 63 | #include <linux/if_tun.h> |
64 | #include <linux/crc32.h> | 64 | #include <linux/crc32.h> |
65 | #include <net/net_namespace.h> | 65 | #include <net/net_namespace.h> |
66 | #include <net/netns/generic.h> | ||
66 | 67 | ||
67 | #include <asm/system.h> | 68 | #include <asm/system.h> |
68 | #include <asm/uaccess.h> | 69 | #include <asm/uaccess.h> |
@@ -106,6 +107,11 @@ struct tun_struct { | |||
106 | 107 | ||
107 | /* Network device part of the driver */ | 108 | /* Network device part of the driver */ |
108 | 109 | ||
110 | static unsigned int tun_net_id; | ||
111 | struct tun_net { | ||
112 | struct list_head dev_list; | ||
113 | }; | ||
114 | |||
109 | static LIST_HEAD(tun_dev_list); | 115 | static LIST_HEAD(tun_dev_list); |
110 | static const struct ethtool_ops tun_ethtool_ops; | 116 | static const struct ethtool_ops tun_ethtool_ops; |
111 | 117 | ||
@@ -909,6 +915,37 @@ static const struct ethtool_ops tun_ethtool_ops = { | |||
909 | .set_rx_csum = tun_set_rx_csum | 915 | .set_rx_csum = tun_set_rx_csum |
910 | }; | 916 | }; |
911 | 917 | ||
918 | static int tun_init_net(struct net *net) | ||
919 | { | ||
920 | struct tun_net *tn; | ||
921 | |||
922 | tn = kmalloc(sizeof(*tn), GFP_KERNEL); | ||
923 | if (tn == NULL) | ||
924 | return -ENOMEM; | ||
925 | |||
926 | INIT_LIST_HEAD(&tn->dev_list); | ||
927 | |||
928 | if (net_assign_generic(net, tun_net_id, tn)) { | ||
929 | kfree(tn); | ||
930 | return -ENOMEM; | ||
931 | } | ||
932 | |||
933 | return 0; | ||
934 | } | ||
935 | |||
936 | static void tun_exit_net(struct net *net) | ||
937 | { | ||
938 | struct tun_net *tn; | ||
939 | |||
940 | tn = net_generic(net, tun_net_id); | ||
941 | kfree(tn); | ||
942 | } | ||
943 | |||
944 | static struct pernet_operations tun_net_ops = { | ||
945 | .init = tun_init_net, | ||
946 | .exit = tun_exit_net, | ||
947 | }; | ||
948 | |||
912 | static int __init tun_init(void) | 949 | static int __init tun_init(void) |
913 | { | 950 | { |
914 | int ret = 0; | 951 | int ret = 0; |
@@ -916,9 +953,22 @@ static int __init tun_init(void) | |||
916 | printk(KERN_INFO "tun: %s, %s\n", DRV_DESCRIPTION, DRV_VERSION); | 953 | printk(KERN_INFO "tun: %s, %s\n", DRV_DESCRIPTION, DRV_VERSION); |
917 | printk(KERN_INFO "tun: %s\n", DRV_COPYRIGHT); | 954 | printk(KERN_INFO "tun: %s\n", DRV_COPYRIGHT); |
918 | 955 | ||
956 | ret = register_pernet_gen_device(&tun_net_id, &tun_net_ops); | ||
957 | if (ret) { | ||
958 | printk(KERN_ERR "tun: Can't register pernet ops\n"); | ||
959 | goto err_pernet; | ||
960 | } | ||
961 | |||
919 | ret = misc_register(&tun_miscdev); | 962 | ret = misc_register(&tun_miscdev); |
920 | if (ret) | 963 | if (ret) { |
921 | printk(KERN_ERR "tun: Can't register misc device %d\n", TUN_MINOR); | 964 | printk(KERN_ERR "tun: Can't register misc device %d\n", TUN_MINOR); |
965 | goto err_misc; | ||
966 | } | ||
967 | return 0; | ||
968 | |||
969 | err_misc: | ||
970 | unregister_pernet_gen_device(tun_net_id, &tun_net_ops); | ||
971 | err_pernet: | ||
922 | return ret; | 972 | return ret; |
923 | } | 973 | } |
924 | 974 | ||
@@ -935,6 +985,7 @@ static void tun_cleanup(void) | |||
935 | } | 985 | } |
936 | rtnl_unlock(); | 986 | rtnl_unlock(); |
937 | 987 | ||
988 | unregister_pernet_gen_device(tun_net_id, &tun_net_ops); | ||
938 | } | 989 | } |
939 | 990 | ||
940 | module_init(tun_init); | 991 | module_init(tun_init); |