Skip to content

Commit d1a65f1

Browse files
ecsvordex
authored andcommitted
batman-adv: Reduce refcnt of removed router when updating route
_batadv_update_route rcu_derefences orig_ifinfo->router outside of a spinlock protected region to print some information messages to the debug log. But this pointer is not checked again when the new pointer is assigned in the spinlock protected region. Thus is can happen that the value of orig_ifinfo->router changed in the meantime and thus the reference counter of the wrong router gets reduced after the spinlock protected region. Just rcu_dereferencing the value of orig_ifinfo->router inside the spinlock protected region (which also set the new pointer) is enough to get the correct old router object. Fixes: e1a5382 ("batman-adv: Make orig_node->router an rcu protected pointer") Signed-off-by: Sven Eckelmann <sven@narfation.org> Signed-off-by: Marek Lindner <mareklindner@neomailbox.ch> Signed-off-by: Antonio Quartulli <a@unstable.cc>
1 parent f2d2386 commit d1a65f1

File tree

1 file changed

+9
-0
lines changed

1 file changed

+9
-0
lines changed

net/batman-adv/routing.c

Lines changed: 9 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -105,6 +105,15 @@ static void _batadv_update_route(struct batadv_priv *bat_priv,
105105
neigh_node = NULL;
106106

107107
spin_lock_bh(&orig_node->neigh_list_lock);
108+
/* curr_router used earlier may not be the current orig_ifinfo->router
109+
* anymore because it was dereferenced outside of the neigh_list_lock
110+
* protected region. After the new best neighbor has replace the current
111+
* best neighbor the reference counter needs to decrease. Consequently,
112+
* the code needs to ensure the curr_router variable contains a pointer
113+
* to the replaced best neighbor.
114+
*/
115+
curr_router = rcu_dereference_protected(orig_ifinfo->router, true);
116+
108117
rcu_assign_pointer(orig_ifinfo->router, neigh_node);
109118
spin_unlock_bh(&orig_node->neigh_list_lock);
110119
batadv_orig_ifinfo_put(orig_ifinfo);

0 commit comments

Comments
 (0)