Skip to content

Commit 51223bd

Browse files
edumazetgregkh
authored andcommitted
mptcp: fix race in mptcp_pm_nl_flush_addrs_doit()
commit e2a9eeb upstream. syzbot and Eulgyu Kim reported crashes in mptcp_pm_nl_get_local_id() and/or mptcp_pm_nl_is_backup() Root cause is list_splice_init() in mptcp_pm_nl_flush_addrs_doit() which is not RCU ready. list_splice_init_rcu() can not be called here while holding pernet->lock spinlock. Many thanks to Eulgyu Kim for providing a repro and testing our patches. Fixes: 141694d ("mptcp: remove address when netlink flushes addrs") Signed-off-by: Eric Dumazet <edumazet@google.com> Reported-by: syzbot+5498a510ff9de39d37da@syzkaller.appspotmail.com Closes: https://lore.kernel.org/all/6970a46d.a00a0220.3ad28e.5cf0.GAE@google.com/T/ Reported-by: Eulgyu Kim <eulgyukim@snu.ac.kr> Closes: multipath-tcp/mptcp_net-next#611 Reviewed-by: Mat Martineau <martineau@kernel.org> Signed-off-by: Matthieu Baerts (NGI0) <matttbe@kernel.org> Link: https://patch.msgid.link/20260124-net-mptcp-race_nl_flush_addrs-v3-1-b2dc1b613e9d@kernel.org Signed-off-by: Jakub Kicinski <kuba@kernel.org> [ Conflicts because the code has been moved from pm_netlink.c to pm_kernel.c later on in commit 8617e85 ("mptcp: pm: split in-kernel PM specific code"). The same modifications can be applied in pm_netlink.c with one exception, because 'pernet->local_addr_list' has been renamed to 'pernet->endp_list' in commit 35e71e4 ("mptcp: pm: in-kernel: rename 'local_addr_list' to 'endp_list'"). The previous name is then still being used in this version. ] Signed-off-by: Matthieu Baerts (NGI0) <matttbe@kernel.org> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
1 parent 9bef8be commit 51223bd

1 file changed

Lines changed: 13 additions & 3 deletions

File tree

net/mptcp/pm_netlink.c

Lines changed: 13 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -1811,16 +1811,26 @@ static void __reset_counters(struct pm_nl_pernet *pernet)
18111811
int mptcp_pm_nl_flush_addrs_doit(struct sk_buff *skb, struct genl_info *info)
18121812
{
18131813
struct pm_nl_pernet *pernet = genl_info_pm_nl(info);
1814-
LIST_HEAD(free_list);
1814+
struct list_head free_list;
18151815

18161816
spin_lock_bh(&pernet->lock);
1817-
list_splice_init(&pernet->local_addr_list, &free_list);
1817+
free_list = pernet->local_addr_list;
1818+
INIT_LIST_HEAD_RCU(&pernet->local_addr_list);
18181819
__reset_counters(pernet);
18191820
pernet->next_id = 1;
18201821
bitmap_zero(pernet->id_bitmap, MPTCP_PM_MAX_ADDR_ID + 1);
18211822
spin_unlock_bh(&pernet->lock);
1822-
mptcp_nl_flush_addrs_list(sock_net(skb->sk), &free_list);
1823+
1824+
if (free_list.next == &pernet->local_addr_list)
1825+
return 0;
1826+
18231827
synchronize_rcu();
1828+
1829+
/* Adjust the pointers to free_list instead of pernet->local_addr_list */
1830+
free_list.prev->next = &free_list;
1831+
free_list.next->prev = &free_list;
1832+
1833+
mptcp_nl_flush_addrs_list(sock_net(skb->sk), &free_list);
18241834
__flush_addrs(&free_list);
18251835
return 0;
18261836
}

0 commit comments

Comments
 (0)