mirror of
https://github.com/hardkernel/linux.git
synced 2026-06-06 19:08:57 +09:00
ipv6: prevent infinite loop in rt6_nlmsg_size()
[ Upstream commit 54e6fe9dd3b0e7c481c2228782c9494d653546da ]
While testing prior patch, I was able to trigger
an infinite loop in rt6_nlmsg_size() in the following place:
list_for_each_entry_rcu(sibling, &f6i->fib6_siblings,
fib6_siblings) {
rt6_nh_nlmsg_size(sibling->fib6_nh, &nexthop_len);
}
This is because fib6_del_route() and fib6_add_rt2node()
uses list_del_rcu(), which can confuse rcu readers,
because they might no longer see the head of the list.
Restart the loop if f6i->fib6_nsiblings is zero.
Fixes: d9ccb18f83ea ("ipv6: Fix soft lockups in fib6_select_path under high next hop churn")
Signed-off-by: Eric Dumazet <edumazet@google.com>
Link: https://patch.msgid.link/20250725140725.3626540-3-edumazet@google.com
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
Signed-off-by: Sasha Levin <sashal@kernel.org>
This commit is contained in:
committed by
Greg Kroah-Hartman
parent
f237664113
commit
e1b7932af4
@@ -1240,7 +1240,7 @@ add:
|
||||
&rt->fib6_siblings,
|
||||
fib6_siblings)
|
||||
sibling->fib6_nsiblings--;
|
||||
rt->fib6_nsiblings = 0;
|
||||
WRITE_ONCE(rt->fib6_nsiblings, 0);
|
||||
list_del_rcu(&rt->fib6_siblings);
|
||||
rt6_multipath_rebalance(next_sibling);
|
||||
return err;
|
||||
@@ -1953,7 +1953,7 @@ static void fib6_del_route(struct fib6_table *table, struct fib6_node *fn,
|
||||
list_for_each_entry_safe(sibling, next_sibling,
|
||||
&rt->fib6_siblings, fib6_siblings)
|
||||
sibling->fib6_nsiblings--;
|
||||
rt->fib6_nsiblings = 0;
|
||||
WRITE_ONCE(rt->fib6_nsiblings, 0);
|
||||
list_del_rcu(&rt->fib6_siblings);
|
||||
rt6_multipath_rebalance(next_sibling);
|
||||
}
|
||||
|
||||
@@ -5577,32 +5577,34 @@ static int rt6_nh_nlmsg_size(struct fib6_nh *nh, void *arg)
|
||||
|
||||
static size_t rt6_nlmsg_size(struct fib6_info *f6i)
|
||||
{
|
||||
struct fib6_info *sibling;
|
||||
struct fib6_nh *nh;
|
||||
int nexthop_len;
|
||||
|
||||
if (f6i->nh) {
|
||||
nexthop_len = nla_total_size(4); /* RTA_NH_ID */
|
||||
nexthop_for_each_fib6_nh(f6i->nh, rt6_nh_nlmsg_size,
|
||||
&nexthop_len);
|
||||
} else {
|
||||
struct fib6_nh *nh = f6i->fib6_nh;
|
||||
struct fib6_info *sibling;
|
||||
|
||||
nexthop_len = 0;
|
||||
if (f6i->fib6_nsiblings) {
|
||||
rt6_nh_nlmsg_size(nh, &nexthop_len);
|
||||
|
||||
rcu_read_lock();
|
||||
|
||||
list_for_each_entry_rcu(sibling, &f6i->fib6_siblings,
|
||||
fib6_siblings) {
|
||||
rt6_nh_nlmsg_size(sibling->fib6_nh, &nexthop_len);
|
||||
}
|
||||
|
||||
rcu_read_unlock();
|
||||
}
|
||||
nexthop_len += lwtunnel_get_encap_size(nh->fib_nh_lws);
|
||||
goto common;
|
||||
}
|
||||
|
||||
rcu_read_lock();
|
||||
retry:
|
||||
nh = f6i->fib6_nh;
|
||||
nexthop_len = 0;
|
||||
if (READ_ONCE(f6i->fib6_nsiblings)) {
|
||||
rt6_nh_nlmsg_size(nh, &nexthop_len);
|
||||
|
||||
list_for_each_entry_rcu(sibling, &f6i->fib6_siblings,
|
||||
fib6_siblings) {
|
||||
rt6_nh_nlmsg_size(sibling->fib6_nh, &nexthop_len);
|
||||
if (!READ_ONCE(f6i->fib6_nsiblings))
|
||||
goto retry;
|
||||
}
|
||||
}
|
||||
rcu_read_unlock();
|
||||
nexthop_len += lwtunnel_get_encap_size(nh->fib_nh_lws);
|
||||
common:
|
||||
return NLMSG_ALIGN(sizeof(struct rtmsg))
|
||||
+ nla_total_size(16) /* RTA_SRC */
|
||||
+ nla_total_size(16) /* RTA_DST */
|
||||
|
||||
Reference in New Issue
Block a user