Skip to content

Commit

Permalink
batman-adv: Make vis info stack traversal threadsafe
Browse files Browse the repository at this point in the history
The batman-adv vis server has to a stack which stores all information
about packets which should be send later. This stack is protected
with a spinlock that is used to prevent concurrent write access to it.

The send_vis_packets function has to take all elements from the stack
and send them to other hosts over the primary interface. The send will
be initiated without the lock which protects the stack.

The implementation using list_for_each_entry_safe has the problem that
it stores the next element as "safe ptr" to allow the deletion of the
current element in the list. The list may be modified during the
unlock/lock pair in the loop body which may make the safe pointer
not pointing to correct next element.

It is safer to remove and use the first element from the stack until no
elements are available. This does not need reduntant information which
would have to be validated each time the lock was removed.

Reported-by: Russell Senior <russell@personaltelco.net>
Signed-off-by: Sven Eckelmann <sven@narfation.org>
  • Loading branch information
Sven Eckelmann committed Jan 30, 2011
1 parent dda9fc6 commit 1181e1d
Showing 1 changed file with 4 additions and 3 deletions.
7 changes: 4 additions & 3 deletions net/batman-adv/vis.c
Original file line number Diff line number Diff line change
Expand Up @@ -816,7 +816,7 @@ static void send_vis_packets(struct work_struct *work)
container_of(work, struct delayed_work, work);
struct bat_priv *bat_priv =
container_of(delayed_work, struct bat_priv, vis_work);
struct vis_info *info, *temp;
struct vis_info *info;

spin_lock_bh(&bat_priv->vis_hash_lock);
purge_vis_packets(bat_priv);
Expand All @@ -826,8 +826,9 @@ static void send_vis_packets(struct work_struct *work)
send_list_add(bat_priv, bat_priv->my_vis_info);
}

list_for_each_entry_safe(info, temp, &bat_priv->vis_send_list,
send_list) {
while (!list_empty(&bat_priv->vis_send_list)) {
info = list_first_entry(&bat_priv->vis_send_list,
typeof(*info), send_list);

kref_get(&info->refcount);
spin_unlock_bh(&bat_priv->vis_hash_lock);
Expand Down

0 comments on commit 1181e1d

Please sign in to comment.