bonding: add RCU for bond_3ad_state_machine_handler()

The bond_3ad_state_machine_handler() use the bond lock to protect
the bond slave list and slave port together, but it is not enough,
the bond slave list was link and unlink in RTNL, not bond lock,
so I add RCU to protect the slave list from leaving.

The bond lock is still used here, because when the slave has been
removed from the list by the time the state machine runs, it appears
to be possible for both function to manupulate the same aggregator->lag_ports
by finding the aggregator via two different ports that are both members of
that aggregator (i.e., port A of the agg is being unbound, and port B
of the agg is runing its state machine).

If I remove the bond lock, there are nothing to mutex changes
to aggregator->lag_ports between bond_3ad_state_machine_handler and
bond_3ad_unbind_slave, So the bond lock is the simplest way to protect
aggregator->lag_ports.

There was a lot of function need RCU protect, I have two choice
to make the function in RCU-safe, (1) create new similar functions
and make the bond slave list in RCU. (2) modify the existed functions
and make them in read-side critical section, because the RCU
read-side critical sections may be nested.

I choose (2) because it is no need to create more similar functions.

The nots in the function is still too old, clean up the nots.

Suggested-by: Nikolay Aleksandrov <nikolay@redhat.com>
Suggested-by: Jay Vosburgh <fubar@us.ibm.com>
Suggested-by: Veaceslav Falico <vfalico@redhat.com>
Signed-off-by: Ding Tianhong <dingtianhong@huawei.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
dingtianhong 2013-12-13 10:20:12 +08:00 committed by David S. Miller
commit be79bd048a
2 changed files with 35 additions and 26 deletions

View file

@ -147,11 +147,12 @@ static inline struct aggregator *__get_first_agg(struct port *port)
struct bonding *bond = __get_bond_by_port(port); struct bonding *bond = __get_bond_by_port(port);
struct slave *first_slave; struct slave *first_slave;
// If there's no bond for this port, or bond has no slaves /* If there's no bond for this port, or bond has no slaves */
if (bond == NULL) if (bond == NULL)
return NULL; return NULL;
first_slave = bond_first_slave(bond); rcu_read_lock();
first_slave = bond_first_slave_rcu(bond);
rcu_read_unlock();
return first_slave ? &(SLAVE_AD_INFO(first_slave).aggregator) : NULL; return first_slave ? &(SLAVE_AD_INFO(first_slave).aggregator) : NULL;
} }
@ -702,9 +703,13 @@ static struct aggregator *__get_active_agg(struct aggregator *aggregator)
struct list_head *iter; struct list_head *iter;
struct slave *slave; struct slave *slave;
bond_for_each_slave(bond, slave, iter) rcu_read_lock();
if (SLAVE_AD_INFO(slave).aggregator.is_active) bond_for_each_slave_rcu(bond, slave, iter)
if (SLAVE_AD_INFO(slave).aggregator.is_active) {
rcu_read_unlock();
return &(SLAVE_AD_INFO(slave).aggregator); return &(SLAVE_AD_INFO(slave).aggregator);
}
rcu_read_unlock();
return NULL; return NULL;
} }
@ -1471,7 +1476,8 @@ static void ad_agg_selection_logic(struct aggregator *agg)
active = __get_active_agg(agg); active = __get_active_agg(agg);
best = (active && agg_device_up(active)) ? active : NULL; best = (active && agg_device_up(active)) ? active : NULL;
bond_for_each_slave(bond, slave, iter) { rcu_read_lock();
bond_for_each_slave_rcu(bond, slave, iter) {
agg = &(SLAVE_AD_INFO(slave).aggregator); agg = &(SLAVE_AD_INFO(slave).aggregator);
agg->is_active = 0; agg->is_active = 0;
@ -1505,7 +1511,7 @@ static void ad_agg_selection_logic(struct aggregator *agg)
active->is_active = 1; active->is_active = 1;
} }
// if there is new best aggregator, activate it /* if there is new best aggregator, activate it */
if (best) { if (best) {
pr_debug("best Agg=%d; P=%d; a k=%d; p k=%d; Ind=%d; Act=%d\n", pr_debug("best Agg=%d; P=%d; a k=%d; p k=%d; Ind=%d; Act=%d\n",
best->aggregator_identifier, best->num_of_ports, best->aggregator_identifier, best->num_of_ports,
@ -1516,7 +1522,7 @@ static void ad_agg_selection_logic(struct aggregator *agg)
best->lag_ports, best->slave, best->lag_ports, best->slave,
best->slave ? best->slave->dev->name : "NULL"); best->slave ? best->slave->dev->name : "NULL");
bond_for_each_slave(bond, slave, iter) { bond_for_each_slave_rcu(bond, slave, iter) {
agg = &(SLAVE_AD_INFO(slave).aggregator); agg = &(SLAVE_AD_INFO(slave).aggregator);
pr_debug("Agg=%d; P=%d; a k=%d; p k=%d; Ind=%d; Act=%d\n", pr_debug("Agg=%d; P=%d; a k=%d; p k=%d; Ind=%d; Act=%d\n",
@ -1526,10 +1532,11 @@ static void ad_agg_selection_logic(struct aggregator *agg)
agg->is_individual, agg->is_active); agg->is_individual, agg->is_active);
} }
// check if any partner replys /* check if any partner replys */
if (best->is_individual) { if (best->is_individual) {
pr_warning("%s: Warning: No 802.3ad response from the link partner for any adapters in the bond\n", pr_warning("%s: Warning: No 802.3ad response from the link partner for any adapters in the bond\n",
best->slave ? best->slave->bond->dev->name : "NULL"); best->slave ?
best->slave->bond->dev->name : "NULL");
} }
best->is_active = 1; best->is_active = 1;
@ -1541,7 +1548,7 @@ static void ad_agg_selection_logic(struct aggregator *agg)
best->partner_oper_aggregator_key, best->partner_oper_aggregator_key,
best->is_individual, best->is_active); best->is_individual, best->is_active);
// disable the ports that were related to the former active_aggregator /* disable the ports that were related to the former active_aggregator */
if (active) { if (active) {
for (port = active->lag_ports; port; for (port = active->lag_ports; port;
port = port->next_port_in_aggregator) { port = port->next_port_in_aggregator) {
@ -1565,6 +1572,8 @@ static void ad_agg_selection_logic(struct aggregator *agg)
} }
} }
rcu_read_unlock();
bond_3ad_set_carrier(bond); bond_3ad_set_carrier(bond);
} }
@ -2069,17 +2078,18 @@ void bond_3ad_state_machine_handler(struct work_struct *work)
struct port *port; struct port *port;
read_lock(&bond->lock); read_lock(&bond->lock);
rcu_read_lock();
//check if there are any slaves /* check if there are any slaves */
if (!bond_has_slaves(bond)) if (!bond_has_slaves(bond))
goto re_arm; goto re_arm;
// check if agg_select_timer timer after initialize is timed out /* check if agg_select_timer timer after initialize is timed out */
if (BOND_AD_INFO(bond).agg_select_timer && !(--BOND_AD_INFO(bond).agg_select_timer)) { if (BOND_AD_INFO(bond).agg_select_timer && !(--BOND_AD_INFO(bond).agg_select_timer)) {
slave = bond_first_slave(bond); slave = bond_first_slave_rcu(bond);
port = slave ? &(SLAVE_AD_INFO(slave).port) : NULL; port = slave ? &(SLAVE_AD_INFO(slave).port) : NULL;
// select the active aggregator for the bond /* select the active aggregator for the bond */
if (port) { if (port) {
if (!port->slave) { if (!port->slave) {
pr_warning("%s: Warning: bond's first port is uninitialized\n", pr_warning("%s: Warning: bond's first port is uninitialized\n",
@ -2093,8 +2103,8 @@ void bond_3ad_state_machine_handler(struct work_struct *work)
bond_3ad_set_carrier(bond); bond_3ad_set_carrier(bond);
} }
// for each port run the state machines /* for each port run the state machines */
bond_for_each_slave(bond, slave, iter) { bond_for_each_slave_rcu(bond, slave, iter) {
port = &(SLAVE_AD_INFO(slave).port); port = &(SLAVE_AD_INFO(slave).port);
if (!port->slave) { if (!port->slave) {
pr_warning("%s: Warning: Found an uninitialized port\n", pr_warning("%s: Warning: Found an uninitialized port\n",
@ -2114,7 +2124,7 @@ void bond_3ad_state_machine_handler(struct work_struct *work)
ad_mux_machine(port); ad_mux_machine(port);
ad_tx_machine(port); ad_tx_machine(port);
// turn off the BEGIN bit, since we already handled it /* turn off the BEGIN bit, since we already handled it */
if (port->sm_vars & AD_PORT_BEGIN) if (port->sm_vars & AD_PORT_BEGIN)
port->sm_vars &= ~AD_PORT_BEGIN; port->sm_vars &= ~AD_PORT_BEGIN;
@ -2122,9 +2132,9 @@ void bond_3ad_state_machine_handler(struct work_struct *work)
} }
re_arm: re_arm:
queue_delayed_work(bond->wq, &bond->ad_work, ad_delta_in_ticks); rcu_read_unlock();
read_unlock(&bond->lock); read_unlock(&bond->lock);
queue_delayed_work(bond->wq, &bond->ad_work, ad_delta_in_ticks);
} }
/** /**
@ -2303,7 +2313,9 @@ int bond_3ad_set_carrier(struct bonding *bond)
struct aggregator *active; struct aggregator *active;
struct slave *first_slave; struct slave *first_slave;
first_slave = bond_first_slave(bond); rcu_read_lock();
first_slave = bond_first_slave_rcu(bond);
rcu_read_unlock();
if (!first_slave) if (!first_slave)
return 0; return 0;
active = __get_active_agg(&(SLAVE_AD_INFO(first_slave).aggregator)); active = __get_active_agg(&(SLAVE_AD_INFO(first_slave).aggregator));

View file

@ -1703,12 +1703,9 @@ static int __bond_release_one(struct net_device *bond_dev,
write_lock_bh(&bond->lock); write_lock_bh(&bond->lock);
/* Inform AD package of unbinding of slave. */ /* Inform AD package of unbinding of slave. */
if (bond->params.mode == BOND_MODE_8023AD) { if (bond->params.mode == BOND_MODE_8023AD)
/* must be called before the slave is
* detached from the list
*/
bond_3ad_unbind_slave(slave); bond_3ad_unbind_slave(slave);
}
write_unlock_bh(&bond->lock); write_unlock_bh(&bond->lock);
pr_info("%s: releasing %s interface %s\n", pr_info("%s: releasing %s interface %s\n",