Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

route: check sysctl_fib_multipath_use_neigh earlier than hash

Prior to this patch, when one packet is hashed into path [1]
(hash <= nh_upper_bound) and it's neigh is dead, it will try
path [2]. However, if path [2]'s neigh is alive but it's
hash > nh_upper_bound, it will not return this alive path.
This packet will never be sent even if path [2] is alive.

3.3.3.1/24:
nexthop via 1.1.1.254 dev eth1 weight 1 <--[1] (dead neigh)
nexthop via 2.2.2.254 dev eth2 weight 1 <--[2]

With sysctl_fib_multipath_use_neigh set is supposed to find an
available path respecting to the l3/l4 hash. But if there is
no available route with this hash, it should at least return
an alive route even with other hash.

This patch is to fix it by processing fib_multipath_use_neigh
earlier than the hash check, so that it will at least return
an alive route if there is when fib_multipath_use_neigh is
enabled. It's also compatible with before when there are alive
routes with the l3/l4 hash.

Fixes: a6db4494d218 ("net: ipv4: Consider failed nexthops in multipath routes")
Reported-by: Jianlin Shi <jishi@redhat.com>
Signed-off-by: Xin Long <lucien.xin@gmail.com>
Acked-by: David Ahern <dsa@cumulusnetworks.com>
Signed-off-by: David S. Miller <davem@davemloft.net>

authored by

Xin Long and committed by
David S. Miller
6174a30d 781492ab

+11 -9
+11 -9
net/ipv4/fib_semantics.c
··· 1746 1746 bool first = false; 1747 1747 1748 1748 for_nexthops(fi) { 1749 + if (net->ipv4.sysctl_fib_multipath_use_neigh) { 1750 + if (!fib_good_nh(nh)) 1751 + continue; 1752 + if (!first) { 1753 + res->nh_sel = nhsel; 1754 + first = true; 1755 + } 1756 + } 1757 + 1749 1758 if (hash > atomic_read(&nh->nh_upper_bound)) 1750 1759 continue; 1751 1760 1752 - if (!net->ipv4.sysctl_fib_multipath_use_neigh || 1753 - fib_good_nh(nh)) { 1754 - res->nh_sel = nhsel; 1755 - return; 1756 - } 1757 - if (!first) { 1758 - res->nh_sel = nhsel; 1759 - first = true; 1760 - } 1761 + res->nh_sel = nhsel; 1762 + return; 1761 1763 } endfor_nexthops(fi); 1762 1764 } 1763 1765 #endif