Commit 1c72cfdc authored by Nikolay Aleksandrov's avatar Nikolay Aleksandrov Committed by David S. Miller

bonding: clean curr_slave_lock use

Mostly all users of curr_slave_lock already have RTNL as we've discussed
previously so there's no point in using it, the one case where the lock
must stay is the 3ad code, in fact it's the only one.
It's okay to remove it from bond_do_fail_over_mac() as it's called with
RTNL and drops the curr_slave_lock anyway.
bond_change_active_slave() is one of the main places where
curr_slave_lock was used, it's okay to remove it as all callers use RTNL
these days before calling it, that's why we move the ASSERT_RTNL() in
the beginning to catch any potential offenders to this rule.
The RTNL argument actually applies to all of the places where
curr_slave_lock has been removed from in this patch.
Also remove the unnecessary bond_deref_active_protected() macro and use
rtnl_dereference() instead.
Signed-off-by: default avatarNikolay Aleksandrov <nikolay@redhat.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 62c5f518
...@@ -451,7 +451,7 @@ static struct slave *__rlb_next_rx_slave(struct bonding *bond) ...@@ -451,7 +451,7 @@ static struct slave *__rlb_next_rx_slave(struct bonding *bond)
*/ */
static void rlb_teach_disabled_mac_on_primary(struct bonding *bond, u8 addr[]) static void rlb_teach_disabled_mac_on_primary(struct bonding *bond, u8 addr[])
{ {
struct slave *curr_active = bond_deref_active_protected(bond); struct slave *curr_active = rtnl_dereference(bond->curr_active_slave);
if (!curr_active) if (!curr_active)
return; return;
...@@ -512,7 +512,7 @@ static void rlb_clear_slave(struct bonding *bond, struct slave *slave) ...@@ -512,7 +512,7 @@ static void rlb_clear_slave(struct bonding *bond, struct slave *slave)
_unlock_rx_hashtbl_bh(bond); _unlock_rx_hashtbl_bh(bond);
if (slave != bond_deref_active_protected(bond)) if (slave != rtnl_dereference(bond->curr_active_slave))
rlb_teach_disabled_mac_on_primary(bond, slave->dev->dev_addr); rlb_teach_disabled_mac_on_primary(bond, slave->dev->dev_addr);
} }
......
...@@ -637,13 +637,11 @@ static void bond_set_dev_addr(struct net_device *bond_dev, ...@@ -637,13 +637,11 @@ static void bond_set_dev_addr(struct net_device *bond_dev,
* *
* Perform special MAC address swapping for fail_over_mac settings * Perform special MAC address swapping for fail_over_mac settings
* *
* Called with RTNL, curr_slave_lock for write_bh. * Called with RTNL
*/ */
static void bond_do_fail_over_mac(struct bonding *bond, static void bond_do_fail_over_mac(struct bonding *bond,
struct slave *new_active, struct slave *new_active,
struct slave *old_active) struct slave *old_active)
__releases(&bond->curr_slave_lock)
__acquires(&bond->curr_slave_lock)
{ {
u8 tmp_mac[ETH_ALEN]; u8 tmp_mac[ETH_ALEN];
struct sockaddr saddr; struct sockaddr saddr;
...@@ -651,11 +649,8 @@ static void bond_do_fail_over_mac(struct bonding *bond, ...@@ -651,11 +649,8 @@ static void bond_do_fail_over_mac(struct bonding *bond,
switch (bond->params.fail_over_mac) { switch (bond->params.fail_over_mac) {
case BOND_FOM_ACTIVE: case BOND_FOM_ACTIVE:
if (new_active) { if (new_active)
write_unlock_bh(&bond->curr_slave_lock);
bond_set_dev_addr(bond->dev, new_active->dev); bond_set_dev_addr(bond->dev, new_active->dev);
write_lock_bh(&bond->curr_slave_lock);
}
break; break;
case BOND_FOM_FOLLOW: case BOND_FOM_FOLLOW:
/* /*
...@@ -666,8 +661,6 @@ static void bond_do_fail_over_mac(struct bonding *bond, ...@@ -666,8 +661,6 @@ static void bond_do_fail_over_mac(struct bonding *bond,
if (!new_active) if (!new_active)
return; return;
write_unlock_bh(&bond->curr_slave_lock);
if (old_active) { if (old_active) {
ether_addr_copy(tmp_mac, new_active->dev->dev_addr); ether_addr_copy(tmp_mac, new_active->dev->dev_addr);
ether_addr_copy(saddr.sa_data, ether_addr_copy(saddr.sa_data,
...@@ -696,7 +689,6 @@ static void bond_do_fail_over_mac(struct bonding *bond, ...@@ -696,7 +689,6 @@ static void bond_do_fail_over_mac(struct bonding *bond,
netdev_err(bond->dev, "Error %d setting MAC of slave %s\n", netdev_err(bond->dev, "Error %d setting MAC of slave %s\n",
-rv, new_active->dev->name); -rv, new_active->dev->name);
out: out:
write_lock_bh(&bond->curr_slave_lock);
break; break;
default: default:
netdev_err(bond->dev, "bond_do_fail_over_mac impossible: bad policy %d\n", netdev_err(bond->dev, "bond_do_fail_over_mac impossible: bad policy %d\n",
...@@ -709,7 +701,7 @@ static void bond_do_fail_over_mac(struct bonding *bond, ...@@ -709,7 +701,7 @@ static void bond_do_fail_over_mac(struct bonding *bond,
static bool bond_should_change_active(struct bonding *bond) static bool bond_should_change_active(struct bonding *bond)
{ {
struct slave *prim = rtnl_dereference(bond->primary_slave); struct slave *prim = rtnl_dereference(bond->primary_slave);
struct slave *curr = bond_deref_active_protected(bond); struct slave *curr = rtnl_dereference(bond->curr_active_slave);
if (!prim || !curr || curr->link != BOND_LINK_UP) if (!prim || !curr || curr->link != BOND_LINK_UP)
return true; return true;
...@@ -785,15 +777,15 @@ static bool bond_should_notify_peers(struct bonding *bond) ...@@ -785,15 +777,15 @@ static bool bond_should_notify_peers(struct bonding *bond)
* because it is apparently the best available slave we have, even though its * because it is apparently the best available slave we have, even though its
* updelay hasn't timed out yet. * updelay hasn't timed out yet.
* *
* If new_active is not NULL, caller must hold curr_slave_lock for write_bh. * Caller must hold RTNL.
*/ */
void bond_change_active_slave(struct bonding *bond, struct slave *new_active) void bond_change_active_slave(struct bonding *bond, struct slave *new_active)
{ {
struct slave *old_active; struct slave *old_active;
old_active = rcu_dereference_protected(bond->curr_active_slave, ASSERT_RTNL();
!new_active ||
lockdep_is_held(&bond->curr_slave_lock)); old_active = rtnl_dereference(bond->curr_active_slave);
if (old_active == new_active) if (old_active == new_active)
return; return;
...@@ -861,14 +853,10 @@ void bond_change_active_slave(struct bonding *bond, struct slave *new_active) ...@@ -861,14 +853,10 @@ void bond_change_active_slave(struct bonding *bond, struct slave *new_active)
bond_should_notify_peers(bond); bond_should_notify_peers(bond);
} }
write_unlock_bh(&bond->curr_slave_lock);
call_netdevice_notifiers(NETDEV_BONDING_FAILOVER, bond->dev); call_netdevice_notifiers(NETDEV_BONDING_FAILOVER, bond->dev);
if (should_notify_peers) if (should_notify_peers)
call_netdevice_notifiers(NETDEV_NOTIFY_PEERS, call_netdevice_notifiers(NETDEV_NOTIFY_PEERS,
bond->dev); bond->dev);
write_lock_bh(&bond->curr_slave_lock);
} }
} }
...@@ -893,7 +881,7 @@ void bond_change_active_slave(struct bonding *bond, struct slave *new_active) ...@@ -893,7 +881,7 @@ void bond_change_active_slave(struct bonding *bond, struct slave *new_active)
* - The primary_slave has got its link back. * - The primary_slave has got its link back.
* - A slave has got its link back and there's no old curr_active_slave. * - A slave has got its link back and there's no old curr_active_slave.
* *
* Caller must hold curr_slave_lock for write_bh. * Caller must hold RTNL.
*/ */
void bond_select_active_slave(struct bonding *bond) void bond_select_active_slave(struct bonding *bond)
{ {
...@@ -901,7 +889,7 @@ void bond_select_active_slave(struct bonding *bond) ...@@ -901,7 +889,7 @@ void bond_select_active_slave(struct bonding *bond)
int rv; int rv;
best_slave = bond_find_best_slave(bond); best_slave = bond_find_best_slave(bond);
if (best_slave != bond_deref_active_protected(bond)) { if (best_slave != rtnl_dereference(bond->curr_active_slave)) {
bond_change_active_slave(bond, best_slave); bond_change_active_slave(bond, best_slave);
rv = bond_set_carrier(bond); rv = bond_set_carrier(bond);
if (!rv) if (!rv)
...@@ -1571,9 +1559,7 @@ int bond_enslave(struct net_device *bond_dev, struct net_device *slave_dev) ...@@ -1571,9 +1559,7 @@ int bond_enslave(struct net_device *bond_dev, struct net_device *slave_dev)
if (bond_uses_primary(bond)) { if (bond_uses_primary(bond)) {
block_netpoll_tx(); block_netpoll_tx();
write_lock_bh(&bond->curr_slave_lock);
bond_select_active_slave(bond); bond_select_active_slave(bond);
write_unlock_bh(&bond->curr_slave_lock);
unblock_netpoll_tx(); unblock_netpoll_tx();
} }
...@@ -1601,10 +1587,8 @@ int bond_enslave(struct net_device *bond_dev, struct net_device *slave_dev) ...@@ -1601,10 +1587,8 @@ int bond_enslave(struct net_device *bond_dev, struct net_device *slave_dev)
RCU_INIT_POINTER(bond->primary_slave, NULL); RCU_INIT_POINTER(bond->primary_slave, NULL);
if (rcu_access_pointer(bond->curr_active_slave) == new_slave) { if (rcu_access_pointer(bond->curr_active_slave) == new_slave) {
block_netpoll_tx(); block_netpoll_tx();
write_lock_bh(&bond->curr_slave_lock);
bond_change_active_slave(bond, NULL); bond_change_active_slave(bond, NULL);
bond_select_active_slave(bond); bond_select_active_slave(bond);
write_unlock_bh(&bond->curr_slave_lock);
unblock_netpoll_tx(); unblock_netpoll_tx();
} }
/* either primary_slave or curr_active_slave might've changed */ /* either primary_slave or curr_active_slave might've changed */
...@@ -1720,11 +1704,8 @@ static int __bond_release_one(struct net_device *bond_dev, ...@@ -1720,11 +1704,8 @@ static int __bond_release_one(struct net_device *bond_dev,
if (rtnl_dereference(bond->primary_slave) == slave) if (rtnl_dereference(bond->primary_slave) == slave)
RCU_INIT_POINTER(bond->primary_slave, NULL); RCU_INIT_POINTER(bond->primary_slave, NULL);
if (oldcurrent == slave) { if (oldcurrent == slave)
write_lock_bh(&bond->curr_slave_lock);
bond_change_active_slave(bond, NULL); bond_change_active_slave(bond, NULL);
write_unlock_bh(&bond->curr_slave_lock);
}
if (bond_is_lb(bond)) { if (bond_is_lb(bond)) {
/* Must be called only after the slave has been /* Must be called only after the slave has been
...@@ -1743,11 +1724,7 @@ static int __bond_release_one(struct net_device *bond_dev, ...@@ -1743,11 +1724,7 @@ static int __bond_release_one(struct net_device *bond_dev,
* is no concern that another slave add/remove event * is no concern that another slave add/remove event
* will interfere. * will interfere.
*/ */
write_lock_bh(&bond->curr_slave_lock);
bond_select_active_slave(bond); bond_select_active_slave(bond);
write_unlock_bh(&bond->curr_slave_lock);
} }
if (!bond_has_slaves(bond)) { if (!bond_has_slaves(bond)) {
...@@ -2058,9 +2035,7 @@ static void bond_miimon_commit(struct bonding *bond) ...@@ -2058,9 +2035,7 @@ static void bond_miimon_commit(struct bonding *bond)
do_failover: do_failover:
ASSERT_RTNL(); ASSERT_RTNL();
block_netpoll_tx(); block_netpoll_tx();
write_lock_bh(&bond->curr_slave_lock);
bond_select_active_slave(bond); bond_select_active_slave(bond);
write_unlock_bh(&bond->curr_slave_lock);
unblock_netpoll_tx(); unblock_netpoll_tx();
} }
...@@ -2506,15 +2481,8 @@ static void bond_loadbalance_arp_mon(struct work_struct *work) ...@@ -2506,15 +2481,8 @@ static void bond_loadbalance_arp_mon(struct work_struct *work)
if (slave_state_changed) { if (slave_state_changed) {
bond_slave_state_change(bond); bond_slave_state_change(bond);
} else if (do_failover) { } else if (do_failover) {
/* the bond_select_active_slave must hold RTNL
* and curr_slave_lock for write.
*/
block_netpoll_tx(); block_netpoll_tx();
write_lock_bh(&bond->curr_slave_lock);
bond_select_active_slave(bond); bond_select_active_slave(bond);
write_unlock_bh(&bond->curr_slave_lock);
unblock_netpoll_tx(); unblock_netpoll_tx();
} }
rtnl_unlock(); rtnl_unlock();
...@@ -2670,9 +2638,7 @@ static void bond_ab_arp_commit(struct bonding *bond) ...@@ -2670,9 +2638,7 @@ static void bond_ab_arp_commit(struct bonding *bond)
do_failover: do_failover:
ASSERT_RTNL(); ASSERT_RTNL();
block_netpoll_tx(); block_netpoll_tx();
write_lock_bh(&bond->curr_slave_lock);
bond_select_active_slave(bond); bond_select_active_slave(bond);
write_unlock_bh(&bond->curr_slave_lock);
unblock_netpoll_tx(); unblock_netpoll_tx();
} }
...@@ -2939,9 +2905,7 @@ static int bond_slave_netdev_event(unsigned long event, ...@@ -2939,9 +2905,7 @@ static int bond_slave_netdev_event(unsigned long event,
primary ? slave_dev->name : "none"); primary ? slave_dev->name : "none");
block_netpoll_tx(); block_netpoll_tx();
write_lock_bh(&bond->curr_slave_lock);
bond_select_active_slave(bond); bond_select_active_slave(bond);
write_unlock_bh(&bond->curr_slave_lock);
unblock_netpoll_tx(); unblock_netpoll_tx();
break; break;
case NETDEV_FEAT_CHANGE: case NETDEV_FEAT_CHANGE:
...@@ -3106,7 +3070,6 @@ static int bond_open(struct net_device *bond_dev) ...@@ -3106,7 +3070,6 @@ static int bond_open(struct net_device *bond_dev)
/* reset slave->backup and slave->inactive */ /* reset slave->backup and slave->inactive */
if (bond_has_slaves(bond)) { if (bond_has_slaves(bond)) {
read_lock(&bond->curr_slave_lock);
bond_for_each_slave(bond, slave, iter) { bond_for_each_slave(bond, slave, iter) {
if (bond_uses_primary(bond) && if (bond_uses_primary(bond) &&
slave != rcu_access_pointer(bond->curr_active_slave)) { slave != rcu_access_pointer(bond->curr_active_slave)) {
...@@ -3117,7 +3080,6 @@ static int bond_open(struct net_device *bond_dev) ...@@ -3117,7 +3080,6 @@ static int bond_open(struct net_device *bond_dev)
BOND_SLAVE_NOTIFY_NOW); BOND_SLAVE_NOTIFY_NOW);
} }
} }
read_unlock(&bond->curr_slave_lock);
} }
bond_work_init_all(bond); bond_work_init_all(bond);
...@@ -3239,14 +3201,10 @@ static int bond_do_ioctl(struct net_device *bond_dev, struct ifreq *ifr, int cmd ...@@ -3239,14 +3201,10 @@ static int bond_do_ioctl(struct net_device *bond_dev, struct ifreq *ifr, int cmd
if (!mii) if (!mii)
return -EINVAL; return -EINVAL;
if (mii->reg_num == 1) { if (mii->reg_num == 1) {
mii->val_out = 0; mii->val_out = 0;
read_lock(&bond->curr_slave_lock);
if (netif_carrier_ok(bond->dev)) if (netif_carrier_ok(bond->dev))
mii->val_out = BMSR_LSTATUS; mii->val_out = BMSR_LSTATUS;
read_unlock(&bond->curr_slave_lock);
} }
return 0; return 0;
......
...@@ -734,15 +734,13 @@ static int bond_option_active_slave_set(struct bonding *bond, ...@@ -734,15 +734,13 @@ static int bond_option_active_slave_set(struct bonding *bond,
} }
block_netpoll_tx(); block_netpoll_tx();
write_lock_bh(&bond->curr_slave_lock);
/* check to see if we are clearing active */ /* check to see if we are clearing active */
if (!slave_dev) { if (!slave_dev) {
netdev_info(bond->dev, "Clearing current active slave\n"); netdev_info(bond->dev, "Clearing current active slave\n");
RCU_INIT_POINTER(bond->curr_active_slave, NULL); RCU_INIT_POINTER(bond->curr_active_slave, NULL);
bond_select_active_slave(bond); bond_select_active_slave(bond);
} else { } else {
struct slave *old_active = bond_deref_active_protected(bond); struct slave *old_active = rtnl_dereference(bond->curr_active_slave);
struct slave *new_active = bond_slave_get_rtnl(slave_dev); struct slave *new_active = bond_slave_get_rtnl(slave_dev);
BUG_ON(!new_active); BUG_ON(!new_active);
...@@ -765,8 +763,6 @@ static int bond_option_active_slave_set(struct bonding *bond, ...@@ -765,8 +763,6 @@ static int bond_option_active_slave_set(struct bonding *bond,
} }
} }
} }
write_unlock_bh(&bond->curr_slave_lock);
unblock_netpoll_tx(); unblock_netpoll_tx();
return ret; return ret;
...@@ -1066,7 +1062,6 @@ static int bond_option_primary_set(struct bonding *bond, ...@@ -1066,7 +1062,6 @@ static int bond_option_primary_set(struct bonding *bond,
struct slave *slave; struct slave *slave;
block_netpoll_tx(); block_netpoll_tx();
write_lock_bh(&bond->curr_slave_lock);
p = strchr(primary, '\n'); p = strchr(primary, '\n');
if (p) if (p)
...@@ -1103,7 +1098,6 @@ static int bond_option_primary_set(struct bonding *bond, ...@@ -1103,7 +1098,6 @@ static int bond_option_primary_set(struct bonding *bond,
primary, bond->dev->name); primary, bond->dev->name);
out: out:
write_unlock_bh(&bond->curr_slave_lock);
unblock_netpoll_tx(); unblock_netpoll_tx();
return 0; return 0;
...@@ -1117,9 +1111,7 @@ static int bond_option_primary_reselect_set(struct bonding *bond, ...@@ -1117,9 +1111,7 @@ static int bond_option_primary_reselect_set(struct bonding *bond,
bond->params.primary_reselect = newval->value; bond->params.primary_reselect = newval->value;
block_netpoll_tx(); block_netpoll_tx();
write_lock_bh(&bond->curr_slave_lock);
bond_select_active_slave(bond); bond_select_active_slave(bond);
write_unlock_bh(&bond->curr_slave_lock);
unblock_netpoll_tx(); unblock_netpoll_tx();
return 0; return 0;
......
...@@ -184,9 +184,7 @@ struct slave { ...@@ -184,9 +184,7 @@ struct slave {
/* /*
* Here are the locking policies for the two bonding locks: * Here are the locking policies for the two bonding locks:
* * Get rcu_read_lock when reading or RTNL when writing slave list.
* 1) Get rcu_read_lock when reading or RTNL when writing slave list.
* 2) Get bond->curr_slave_lock when reading/writing bond->curr_active_slave.
*/ */
struct bonding { struct bonding {
struct net_device *dev; /* first - useful for panic debug */ struct net_device *dev; /* first - useful for panic debug */
...@@ -227,10 +225,6 @@ struct bonding { ...@@ -227,10 +225,6 @@ struct bonding {
#define bond_slave_get_rtnl(dev) \ #define bond_slave_get_rtnl(dev) \
((struct slave *) rtnl_dereference(dev->rx_handler_data)) ((struct slave *) rtnl_dereference(dev->rx_handler_data))
#define bond_deref_active_protected(bond) \
rcu_dereference_protected(bond->curr_active_slave, \
lockdep_is_held(&bond->curr_slave_lock))
struct bond_vlan_tag { struct bond_vlan_tag {
__be16 vlan_proto; __be16 vlan_proto;
unsigned short vlan_id; unsigned short vlan_id;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment