diff options
author | Eric Dumazet <eric.dumazet@gmail.com> | 2011-05-19 08:24:16 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2011-05-20 00:33:18 -0400 |
commit | 449f4544267e73d5db372971da63634707c32299 (patch) | |
tree | acee6de13f34c179ec46ef7f8a49e66b229f4460 | |
parent | 034cfe48d0efc248ba4b725e3a94b95e76fbc5d3 (diff) |
macvlan: remove one synchronize_rcu() call
When one macvlan device is dismantled, we can avoid one
synchronize_rcu() call done after deletion from hash list, since caller
will perform a synchronize_net() call after its ndo_stop() call.
Add a new netdev->dismantle field to signal this dismantle intent.
Reduces RTNL hold time.
Signed-off-by: Eric Dumazet <eric.dumazet@gmail.com>
CC: Patrick McHardy <kaber@trash.net>
CC: Ben Greear <greearb@candelatech.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
-rw-r--r-- | drivers/net/macvlan.c | 9 | ||||
-rw-r--r-- | include/linux/netdevice.h | 4 | ||||
-rw-r--r-- | net/core/dev.c | 2 |
3 files changed, 9 insertions, 6 deletions
diff --git a/drivers/net/macvlan.c b/drivers/net/macvlan.c index d7c0bc62da7f..07bcb8084d78 100644 --- a/drivers/net/macvlan.c +++ b/drivers/net/macvlan.c | |||
@@ -70,16 +70,17 @@ static void macvlan_hash_add(struct macvlan_dev *vlan) | |||
70 | hlist_add_head_rcu(&vlan->hlist, &port->vlan_hash[addr[5]]); | 70 | hlist_add_head_rcu(&vlan->hlist, &port->vlan_hash[addr[5]]); |
71 | } | 71 | } |
72 | 72 | ||
73 | static void macvlan_hash_del(struct macvlan_dev *vlan) | 73 | static void macvlan_hash_del(struct macvlan_dev *vlan, bool sync) |
74 | { | 74 | { |
75 | hlist_del_rcu(&vlan->hlist); | 75 | hlist_del_rcu(&vlan->hlist); |
76 | synchronize_rcu(); | 76 | if (sync) |
77 | synchronize_rcu(); | ||
77 | } | 78 | } |
78 | 79 | ||
79 | static void macvlan_hash_change_addr(struct macvlan_dev *vlan, | 80 | static void macvlan_hash_change_addr(struct macvlan_dev *vlan, |
80 | const unsigned char *addr) | 81 | const unsigned char *addr) |
81 | { | 82 | { |
82 | macvlan_hash_del(vlan); | 83 | macvlan_hash_del(vlan, true); |
83 | /* Now that we are unhashed it is safe to change the device | 84 | /* Now that we are unhashed it is safe to change the device |
84 | * address without confusing packet delivery. | 85 | * address without confusing packet delivery. |
85 | */ | 86 | */ |
@@ -345,7 +346,7 @@ static int macvlan_stop(struct net_device *dev) | |||
345 | dev_uc_del(lowerdev, dev->dev_addr); | 346 | dev_uc_del(lowerdev, dev->dev_addr); |
346 | 347 | ||
347 | hash_del: | 348 | hash_del: |
348 | macvlan_hash_del(vlan); | 349 | macvlan_hash_del(vlan, !dev->dismantle); |
349 | return 0; | 350 | return 0; |
350 | } | 351 | } |
351 | 352 | ||
diff --git a/include/linux/netdevice.h b/include/linux/netdevice.h index a134d809125b..ca333e79e10f 100644 --- a/include/linux/netdevice.h +++ b/include/linux/netdevice.h | |||
@@ -1293,7 +1293,9 @@ struct net_device { | |||
1293 | NETREG_UNREGISTERED, /* completed unregister todo */ | 1293 | NETREG_UNREGISTERED, /* completed unregister todo */ |
1294 | NETREG_RELEASED, /* called free_netdev */ | 1294 | NETREG_RELEASED, /* called free_netdev */ |
1295 | NETREG_DUMMY, /* dummy device for NAPI poll */ | 1295 | NETREG_DUMMY, /* dummy device for NAPI poll */ |
1296 | } reg_state:16; | 1296 | } reg_state:8; |
1297 | |||
1298 | bool dismantle; /* device is going do be freed */ | ||
1297 | 1299 | ||
1298 | enum { | 1300 | enum { |
1299 | RTNL_LINK_INITIALIZED, | 1301 | RTNL_LINK_INITIALIZED, |
diff --git a/net/core/dev.c b/net/core/dev.c index 155de2094e71..d94537914a71 100644 --- a/net/core/dev.c +++ b/net/core/dev.c | |||
@@ -5126,7 +5126,7 @@ static void rollback_registered_many(struct list_head *head) | |||
5126 | list_del(&dev->unreg_list); | 5126 | list_del(&dev->unreg_list); |
5127 | continue; | 5127 | continue; |
5128 | } | 5128 | } |
5129 | 5129 | dev->dismantle = true; | |
5130 | BUG_ON(dev->reg_state != NETREG_REGISTERED); | 5130 | BUG_ON(dev->reg_state != NETREG_REGISTERED); |
5131 | } | 5131 | } |
5132 | 5132 | ||