aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorNicolas Dichtel <nicolas.dichtel@6wind.com>2015-03-26 12:56:38 -0400
committerDavid S. Miller <davem@davemloft.net>2015-03-29 15:58:21 -0400
commit4217291e592da0e4258b652e82e5428639d29acc (patch)
treed2471fed2efe37bae94315963de49a4fac93157e
parent17f5ddd5a3b31bdb3acc6f7a41785503c9d113ee (diff)
netns: don't clear nsid too early on removal
With the current code, ids are removed too early. Suppose you have an ipip interface that stands in the netns foo and its link part in the netns bar (so the netns bar has an nsid into the netns foo). Now, you remove the netns bar: - the bar nsid into the netns foo is removed - the netns exit method of ipip is called, thus our ipip iface is removed: => a netlink message is sent in the netns foo to advertise this deletion => this netlink message requests an nsid for bar, thus a new nsid is allocated for bar and never removed. We must remove nsids when we are sure that nobody will refer to netns currently cleaned. Fixes: 0c7aecd4bde4 ("netns: add rtnl cmd to add and get peer netns ids") Signed-off-by: Nicolas Dichtel <nicolas.dichtel@6wind.com> Signed-off-by: David S. Miller <davem@davemloft.net>
-rw-r--r--net/core/net_namespace.c24
1 files changed, 15 insertions, 9 deletions
diff --git a/net/core/net_namespace.c b/net/core/net_namespace.c
index cb5290b8c428..5221f975a4cc 100644
--- a/net/core/net_namespace.c
+++ b/net/core/net_namespace.c
@@ -349,7 +349,7 @@ static LIST_HEAD(cleanup_list); /* Must hold cleanup_list_lock to touch */
349static void cleanup_net(struct work_struct *work) 349static void cleanup_net(struct work_struct *work)
350{ 350{
351 const struct pernet_operations *ops; 351 const struct pernet_operations *ops;
352 struct net *net, *tmp; 352 struct net *net, *tmp, *peer;
353 struct list_head net_kill_list; 353 struct list_head net_kill_list;
354 LIST_HEAD(net_exit_list); 354 LIST_HEAD(net_exit_list);
355 355
@@ -365,14 +365,6 @@ static void cleanup_net(struct work_struct *work)
365 list_for_each_entry(net, &net_kill_list, cleanup_list) { 365 list_for_each_entry(net, &net_kill_list, cleanup_list) {
366 list_del_rcu(&net->list); 366 list_del_rcu(&net->list);
367 list_add_tail(&net->exit_list, &net_exit_list); 367 list_add_tail(&net->exit_list, &net_exit_list);
368 for_each_net(tmp) {
369 int id = __peernet2id(tmp, net, false);
370
371 if (id >= 0)
372 idr_remove(&tmp->netns_ids, id);
373 }
374 idr_destroy(&net->netns_ids);
375
376 } 368 }
377 rtnl_unlock(); 369 rtnl_unlock();
378 370
@@ -398,12 +390,26 @@ static void cleanup_net(struct work_struct *work)
398 */ 390 */
399 rcu_barrier(); 391 rcu_barrier();
400 392
393 rtnl_lock();
401 /* Finally it is safe to free my network namespace structure */ 394 /* Finally it is safe to free my network namespace structure */
402 list_for_each_entry_safe(net, tmp, &net_exit_list, exit_list) { 395 list_for_each_entry_safe(net, tmp, &net_exit_list, exit_list) {
396 /* Unreference net from all peers (no need to loop over
397 * net_exit_list because idr_destroy() will be called for each
398 * element of this list.
399 */
400 for_each_net(peer) {
401 int id = __peernet2id(peer, net, false);
402
403 if (id >= 0)
404 idr_remove(&peer->netns_ids, id);
405 }
406 idr_destroy(&net->netns_ids);
407
403 list_del_init(&net->exit_list); 408 list_del_init(&net->exit_list);
404 put_user_ns(net->user_ns); 409 put_user_ns(net->user_ns);
405 net_drop_ns(net); 410 net_drop_ns(net);
406 } 411 }
412 rtnl_unlock();
407} 413}
408static DECLARE_WORK(net_cleanup_work, cleanup_net); 414static DECLARE_WORK(net_cleanup_work, cleanup_net);
409 415