Skip to content

Commit a220445

Browse files
NicolasDichteldavem330
authored andcommitted
ipv6: correctly add local routes when lo goes up
The goal of the patch is to fix this scenario: ip link add dummy1 type dummy ip link set dummy1 up ip link set lo down ; ip link set lo up After that sequence, the local route to the link layer address of dummy1 is not there anymore. When the loopback is set down, all local routes are deleted by addrconf_ifdown()/rt6_ifdown(). At this time, the rt6_info entry still exists, because the corresponding idev has a reference on it. After the rcu grace period, dst_rcu_free() is called, and thus ___dst_free(), which will set obsolete to DST_OBSOLETE_DEAD. In this case, init_loopback() is called before dst_rcu_free(), thus obsolete is still sets to something <= 0. So, the function doesn't add the route again. To avoid that race, let's check the rt6 refcnt instead. Fixes: 25fb6ca ("net IPv6 : Fix broken IPv6 routing table after loopback down-up") Fixes: a881ae1 ("ipv6: don't call addrconf_dst_alloc again when enable lo") Fixes: 33d9911 ("ipv6: reallocate addrconf router for ipv6 address when lo device up") Reported-by: Francesco Santoro <francesco.santoro@6wind.com> Reported-by: Samuel Gauthier <samuel.gauthier@6wind.com> CC: Balakumaran Kannan <Balakumaran.Kannan@ap.sony.com> CC: Maruthi Thotad <Maruthi.Thotad@ap.sony.com> CC: Sabrina Dubroca <sd@queasysnail.net> CC: Hannes Frederic Sowa <hannes@stressinduktion.org> CC: Weilong Chen <chenweilong@huawei.com> CC: Gao feng <gaofeng@cn.fujitsu.com> Signed-off-by: Nicolas Dichtel <nicolas.dichtel@6wind.com> Signed-off-by: David S. Miller <davem@davemloft.net>
1 parent 68d00f3 commit a220445

File tree

1 file changed

+1
-1
lines changed

1 file changed

+1
-1
lines changed

net/ipv6/addrconf.c

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -3018,7 +3018,7 @@ static void init_loopback(struct net_device *dev)
30183018
* lo device down, release this obsolete dst and
30193019
* reallocate a new router for ifa.
30203020
*/
3021-
if (sp_ifa->rt->dst.obsolete > 0) {
3021+
if (!atomic_read(&sp_ifa->rt->rt6i_ref)) {
30223022
ip6_rt_put(sp_ifa->rt);
30233023
sp_ifa->rt = NULL;
30243024
} else {

0 commit comments

Comments
 (0)