* [PATCH net v2] rtnetlink: wait for unregistering devices in rtnl_link_unregister()
@ 2014-05-12 22:11 Cong Wang
2014-05-15 18:32 ` David Miller
0 siblings, 1 reply; 5+ messages in thread
From: Cong Wang @ 2014-05-12 22:11 UTC (permalink / raw)
To: netdev; +Cc: David S. Miller, stephen, Cong Wang, Eric W. Biederman, Cong Wang
From: Cong Wang <cwang@twopensource.com>
From: Cong Wang <cwang@twopensource.com>
commit 50624c934db18ab90 (net: Delay default_device_exit_batch until no
devices are unregistering) introduced rtnl_lock_unregistering() for
default_device_exit_batch(). Same race could happen we when rmmod a driver
which calls rtnl_link_unregister() as we call dev->destructor without rtnl
lock.
For long term, I think we should clean up the mess of netdev_run_todo()
and net namespce exit code.
Cc: Eric W. Biederman <ebiederm@xmission.com>
Cc: David S. Miller <davem@davemloft.net>
Signed-off-by: Cong Wang <xiyou.wangcong@gmail.com>
Signed-off-by: Cong Wang <cwang@twopensource.com>
---
v2: avoid using macro and hold net_mutex too
include/linux/rtnetlink.h | 5 +++++
net/core/dev.c | 2 +-
net/core/net_namespace.c | 2 +-
net/core/rtnetlink.c | 33 ++++++++++++++++++++++++++++++++-
4 files changed, 39 insertions(+), 3 deletions(-)
diff --git a/include/linux/rtnetlink.h b/include/linux/rtnetlink.h
index 8e3e66a..953937e 100644
--- a/include/linux/rtnetlink.h
+++ b/include/linux/rtnetlink.h
@@ -4,6 +4,7 @@
#include <linux/mutex.h>
#include <linux/netdevice.h>
+#include <linux/wait.h>
#include <uapi/linux/rtnetlink.h>
extern int rtnetlink_send(struct sk_buff *skb, struct net *net, u32 pid, u32 group, int echo);
@@ -22,6 +23,10 @@ extern void rtnl_lock(void);
extern void rtnl_unlock(void);
extern int rtnl_trylock(void);
extern int rtnl_is_locked(void);
+
+extern wait_queue_head_t netdev_unregistering_wq;
+extern struct mutex net_mutex;
+
#ifdef CONFIG_PROVE_LOCKING
extern int lockdep_rtnl_is_held(void);
#else
diff --git a/net/core/dev.c b/net/core/dev.c
index c619b86..6da649b 100644
--- a/net/core/dev.c
+++ b/net/core/dev.c
@@ -5541,7 +5541,7 @@ static int dev_new_index(struct net *net)
/* Delayed registration/unregisteration */
static LIST_HEAD(net_todo_list);
-static DECLARE_WAIT_QUEUE_HEAD(netdev_unregistering_wq);
+DECLARE_WAIT_QUEUE_HEAD(netdev_unregistering_wq);
static void net_set_todo(struct net_device *dev)
{
diff --git a/net/core/net_namespace.c b/net/core/net_namespace.c
index 81d3a9a..7c8ffd9 100644
--- a/net/core/net_namespace.c
+++ b/net/core/net_namespace.c
@@ -24,7 +24,7 @@
static LIST_HEAD(pernet_list);
static struct list_head *first_device = &pernet_list;
-static DEFINE_MUTEX(net_mutex);
+DEFINE_MUTEX(net_mutex);
LIST_HEAD(net_namespace_list);
EXPORT_SYMBOL_GPL(net_namespace_list);
diff --git a/net/core/rtnetlink.c b/net/core/rtnetlink.c
index 9837beb..2d8d8fc 100644
--- a/net/core/rtnetlink.c
+++ b/net/core/rtnetlink.c
@@ -353,15 +353,46 @@ void __rtnl_link_unregister(struct rtnl_link_ops *ops)
}
EXPORT_SYMBOL_GPL(__rtnl_link_unregister);
+/* Return with the rtnl_lock held when there are no network
+ * devices unregistering in any network namespace.
+ */
+static void rtnl_lock_unregistering_all(void)
+{
+ struct net *net;
+ bool unregistering;
+ DEFINE_WAIT(wait);
+
+ for (;;) {
+ prepare_to_wait(&netdev_unregistering_wq, &wait,
+ TASK_UNINTERRUPTIBLE);
+ unregistering = false;
+ rtnl_lock();
+ for_each_net(net) {
+ if (net->dev_unreg_count > 0) {
+ unregistering = true;
+ break;
+ }
+ }
+ if (!unregistering)
+ break;
+ __rtnl_unlock();
+ schedule();
+ }
+ finish_wait(&netdev_unregistering_wq, &wait);
+}
+
/**
* rtnl_link_unregister - Unregister rtnl_link_ops from rtnetlink.
* @ops: struct rtnl_link_ops * to unregister
*/
void rtnl_link_unregister(struct rtnl_link_ops *ops)
{
- rtnl_lock();
+ /* Close the race with cleanup_net() */
+ mutex_lock(&net_mutex);
+ rtnl_lock_unregistering_all();
__rtnl_link_unregister(ops);
rtnl_unlock();
+ mutex_unlock(&net_mutex);
}
EXPORT_SYMBOL_GPL(rtnl_link_unregister);
^ permalink raw reply related [flat|nested] 5+ messages in thread
* Re: [PATCH net v2] rtnetlink: wait for unregistering devices in rtnl_link_unregister()
2014-05-12 22:11 [PATCH net v2] rtnetlink: wait for unregistering devices in rtnl_link_unregister() Cong Wang
@ 2014-05-15 18:32 ` David Miller
2014-05-15 18:43 ` Cong Wang
0 siblings, 1 reply; 5+ messages in thread
From: David Miller @ 2014-05-15 18:32 UTC (permalink / raw)
To: xiyou.wangcong; +Cc: netdev, stephen, cwang, ebiederm
From: Cong Wang <xiyou.wangcong@gmail.com>
Date: Mon, 12 May 2014 15:11:20 -0700
> +static void rtnl_lock_unregistering_all(void)
This is just:
rtnl_lock_unregistering(&net_namespace_list);
There is no need to reimplement the entire thing.
^ permalink raw reply [flat|nested] 5+ messages in thread
* Re: [PATCH net v2] rtnetlink: wait for unregistering devices in rtnl_link_unregister()
2014-05-15 18:32 ` David Miller
@ 2014-05-15 18:43 ` Cong Wang
2014-05-15 18:50 ` David Miller
0 siblings, 1 reply; 5+ messages in thread
From: Cong Wang @ 2014-05-15 18:43 UTC (permalink / raw)
To: David Miller
Cc: Linux Kernel Network Developers, Stephen Hemminger, Cong Wang,
Eric W. Biederman
On Thu, May 15, 2014 at 11:32 AM, David Miller <davem@davemloft.net> wrote:
> From: Cong Wang <xiyou.wangcong@gmail.com>
> Date: Mon, 12 May 2014 15:11:20 -0700
>
>> +static void rtnl_lock_unregistering_all(void)
>
> This is just:
>
> rtnl_lock_unregistering(&net_namespace_list);
>
> There is no need to reimplement the entire thing.
This is exactly what I did in v0 (never sent out) locally,
I spent few hours to debug the problem. The problem
is net_namespace_list links netns by net->list,
while rtnl_lock_unregistering() uses net->exit_list.
Thus I used an ugly macro in v1.
There seems no other choice than between
an ugly macro and duplicated code.
^ permalink raw reply [flat|nested] 5+ messages in thread
* Re: [PATCH net v2] rtnetlink: wait for unregistering devices in rtnl_link_unregister()
2014-05-15 18:43 ` Cong Wang
@ 2014-05-15 18:50 ` David Miller
2014-05-15 19:31 ` David Miller
0 siblings, 1 reply; 5+ messages in thread
From: David Miller @ 2014-05-15 18:50 UTC (permalink / raw)
To: xiyou.wangcong; +Cc: netdev, stephen, cwang, ebiederm
From: Cong Wang <xiyou.wangcong@gmail.com>
Date: Thu, 15 May 2014 11:43:26 -0700
> On Thu, May 15, 2014 at 11:32 AM, David Miller <davem@davemloft.net> wrote:
>> From: Cong Wang <xiyou.wangcong@gmail.com>
>> Date: Mon, 12 May 2014 15:11:20 -0700
>>
>>> +static void rtnl_lock_unregistering_all(void)
>>
>> This is just:
>>
>> rtnl_lock_unregistering(&net_namespace_list);
>>
>> There is no need to reimplement the entire thing.
>
> This is exactly what I did in v0 (never sent out) locally,
> I spent few hours to debug the problem. The problem
> is net_namespace_list links netns by net->list,
> while rtnl_lock_unregistering() uses net->exit_list.
> Thus I used an ugly macro in v1.
>
> There seems no other choice than between
> an ugly macro and duplicated code.
Thanks for explaining (again), I'll look at this some more.
^ permalink raw reply [flat|nested] 5+ messages in thread
* Re: [PATCH net v2] rtnetlink: wait for unregistering devices in rtnl_link_unregister()
2014-05-15 18:50 ` David Miller
@ 2014-05-15 19:31 ` David Miller
0 siblings, 0 replies; 5+ messages in thread
From: David Miller @ 2014-05-15 19:31 UTC (permalink / raw)
To: xiyou.wangcong; +Cc: netdev, stephen, cwang, ebiederm
From: David Miller <davem@davemloft.net>
Date: Thu, 15 May 2014 14:50:43 -0400 (EDT)
> From: Cong Wang <xiyou.wangcong@gmail.com>
> Date: Thu, 15 May 2014 11:43:26 -0700
>
>> On Thu, May 15, 2014 at 11:32 AM, David Miller <davem@davemloft.net> wrote:
>>> From: Cong Wang <xiyou.wangcong@gmail.com>
>>> Date: Mon, 12 May 2014 15:11:20 -0700
>>>
>>>> +static void rtnl_lock_unregistering_all(void)
>>>
>>> This is just:
>>>
>>> rtnl_lock_unregistering(&net_namespace_list);
>>>
>>> There is no need to reimplement the entire thing.
>>
>> This is exactly what I did in v0 (never sent out) locally,
>> I spent few hours to debug the problem. The problem
>> is net_namespace_list links netns by net->list,
>> while rtnl_lock_unregistering() uses net->exit_list.
>> Thus I used an ugly macro in v1.
>>
>> There seems no other choice than between
>> an ugly macro and duplicated code.
>
> Thanks for explaining (again), I'll look at this some more.
Ok, applied and queued up for -stable, thanks.
^ permalink raw reply [flat|nested] 5+ messages in thread
end of thread, other threads:[~2014-05-15 19:31 UTC | newest]
Thread overview: 5+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2014-05-12 22:11 [PATCH net v2] rtnetlink: wait for unregistering devices in rtnl_link_unregister() Cong Wang
2014-05-15 18:32 ` David Miller
2014-05-15 18:43 ` Cong Wang
2014-05-15 18:50 ` David Miller
2014-05-15 19:31 ` David Miller
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).