aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--include/net/netns/ipv6.h2
-rw-r--r--net/ipv6/addrconf.c82
2 files changed, 72 insertions, 12 deletions
diff --git a/include/net/netns/ipv6.h b/include/net/netns/ipv6.h
index 10733a6f1bd4..06b4dc034bbd 100644
--- a/include/net/netns/ipv6.h
+++ b/include/net/netns/ipv6.h
@@ -28,5 +28,7 @@ struct netns_sysctl_ipv6 {
28 28
29struct netns_ipv6 { 29struct netns_ipv6 {
30 struct netns_sysctl_ipv6 sysctl; 30 struct netns_sysctl_ipv6 sysctl;
31 struct ipv6_devconf *devconf_all;
32 struct ipv6_devconf *devconf_dflt;
31}; 33};
32#endif 34#endif
diff --git a/net/ipv6/addrconf.c b/net/ipv6/addrconf.c
index bde50c686722..3ad081e9366b 100644
--- a/net/ipv6/addrconf.c
+++ b/net/ipv6/addrconf.c
@@ -4135,6 +4135,70 @@ static void addrconf_sysctl_unregister(struct inet6_dev *idev)
4135 4135
4136#endif 4136#endif
4137 4137
4138static int addrconf_init_net(struct net *net)
4139{
4140 int err;
4141 struct ipv6_devconf *all, *dflt;
4142
4143 err = -ENOMEM;
4144 all = &ipv6_devconf;
4145 dflt = &ipv6_devconf_dflt;
4146
4147 if (net != &init_net) {
4148 all = kmemdup(all, sizeof(ipv6_devconf), GFP_KERNEL);
4149 if (all == NULL)
4150 goto err_alloc_all;
4151
4152 dflt = kmemdup(dflt, sizeof(ipv6_devconf_dflt), GFP_KERNEL);
4153 if (dflt == NULL)
4154 goto err_alloc_dflt;
4155 }
4156
4157 net->ipv6.devconf_all = all;
4158 net->ipv6.devconf_dflt = dflt;
4159
4160#ifdef CONFIG_SYSCTL
4161 err = __addrconf_sysctl_register(net, "all", NET_PROTO_CONF_ALL,
4162 NULL, all);
4163 if (err < 0)
4164 goto err_reg_all;
4165
4166 err = __addrconf_sysctl_register(net, "default", NET_PROTO_CONF_DEFAULT,
4167 NULL, dflt);
4168 if (err < 0)
4169 goto err_reg_dflt;
4170#endif
4171 return 0;
4172
4173#ifdef CONFIG_SYSCTL
4174err_reg_dflt:
4175 __addrconf_sysctl_unregister(all);
4176err_reg_all:
4177 kfree(dflt);
4178#endif
4179err_alloc_dflt:
4180 kfree(all);
4181err_alloc_all:
4182 return err;
4183}
4184
4185static void addrconf_exit_net(struct net *net)
4186{
4187#ifdef CONFIG_SYSCTL
4188 __addrconf_sysctl_unregister(net->ipv6.devconf_dflt);
4189 __addrconf_sysctl_unregister(net->ipv6.devconf_all);
4190#endif
4191 if (net != &init_net) {
4192 kfree(net->ipv6.devconf_dflt);
4193 kfree(net->ipv6.devconf_all);
4194 }
4195}
4196
4197static struct pernet_operations addrconf_ops = {
4198 .init = addrconf_init_net,
4199 .exit = addrconf_exit_net,
4200};
4201
4138/* 4202/*
4139 * Device notifier 4203 * Device notifier
4140 */ 4204 */
@@ -4167,6 +4231,8 @@ int __init addrconf_init(void)
4167 return err; 4231 return err;
4168 } 4232 }
4169 4233
4234 register_pernet_subsys(&addrconf_ops);
4235
4170 /* The addrconf netdev notifier requires that loopback_dev 4236 /* The addrconf netdev notifier requires that loopback_dev
4171 * has it's ipv6 private information allocated and setup 4237 * has it's ipv6 private information allocated and setup
4172 * before it can bring up and give link-local addresses 4238 * before it can bring up and give link-local addresses
@@ -4190,7 +4256,7 @@ int __init addrconf_init(void)
4190 err = -ENOMEM; 4256 err = -ENOMEM;
4191 rtnl_unlock(); 4257 rtnl_unlock();
4192 if (err) 4258 if (err)
4193 return err; 4259 goto errlo;
4194 4260
4195 ip6_null_entry.u.dst.dev = init_net.loopback_dev; 4261 ip6_null_entry.u.dst.dev = init_net.loopback_dev;
4196 ip6_null_entry.rt6i_idev = in6_dev_get(init_net.loopback_dev); 4262 ip6_null_entry.rt6i_idev = in6_dev_get(init_net.loopback_dev);
@@ -4218,16 +4284,11 @@ int __init addrconf_init(void)
4218 4284
4219 ipv6_addr_label_rtnl_register(); 4285 ipv6_addr_label_rtnl_register();
4220 4286
4221#ifdef CONFIG_SYSCTL
4222 __addrconf_sysctl_register(&init_net, "all", NET_PROTO_CONF_ALL,
4223 NULL, &ipv6_devconf);
4224 __addrconf_sysctl_register(&init_net, "default", NET_PROTO_CONF_DEFAULT,
4225 NULL, &ipv6_devconf_dflt);
4226#endif
4227
4228 return 0; 4287 return 0;
4229errout: 4288errout:
4230 unregister_netdevice_notifier(&ipv6_dev_notf); 4289 unregister_netdevice_notifier(&ipv6_dev_notf);
4290errlo:
4291 unregister_pernet_subsys(&addrconf_ops);
4231 4292
4232 return err; 4293 return err;
4233} 4294}
@@ -4240,10 +4301,7 @@ void addrconf_cleanup(void)
4240 4301
4241 unregister_netdevice_notifier(&ipv6_dev_notf); 4302 unregister_netdevice_notifier(&ipv6_dev_notf);
4242 4303
4243#ifdef CONFIG_SYSCTL 4304 unregister_pernet_subsys(&addrconf_ops);
4244 __addrconf_sysctl_unregister(&ipv6_devconf_dflt);
4245 __addrconf_sysctl_unregister(&ipv6_devconf);
4246#endif
4247 4305
4248 rtnl_lock(); 4306 rtnl_lock();
4249 4307