Commit 8a20e7f1 authored by Sabrina Dubroca's avatar Sabrina Dubroca Committed by Kelsey Skunberg

net: ipv4: don't let PMTU updates increase route MTU

BugLink: https://bugs.launchpad.net/bugs/1873852

[ Upstream commit 28d35bcd ]

When an MTU update with PMTU smaller than net.ipv4.route.min_pmtu is
received, we must clamp its value. However, we can receive a PMTU
exception with PMTU < old_mtu < ip_rt_min_pmtu, which would lead to an
increase in PMTU.

To fix this, take the smallest of the old MTU and ip_rt_min_pmtu.

Before this patch, in case of an update, the exception's MTU would
always change. Now, an exception can have only its lock flag updated,
but not the MTU, so we need to add a check on locking to the following
"is this exception getting updated, or close to expiring?" test.

Fixes: d52e5a7e ("ipv4: lock mtu in fnhe when received PMTU < net.ipv4.route.min_pmtu")
Signed-off-by: default avatarSabrina Dubroca <sd@queasysnail.net>
Reviewed-by: default avatarStefano Brivio <sbrivio@redhat.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
Signed-off-by: default avatarSasha Levin <sashal@kernel.org>
Signed-off-by: default avatarKleber Sacilotto de Souza <kleber.souza@canonical.com>
Signed-off-by: default avatarKelsey Skunberg <kelsey.skunberg@canonical.com>
parent 8e408a39
...@@ -1004,21 +1004,22 @@ out: kfree_skb(skb); ...@@ -1004,21 +1004,22 @@ out: kfree_skb(skb);
static void __ip_rt_update_pmtu(struct rtable *rt, struct flowi4 *fl4, u32 mtu) static void __ip_rt_update_pmtu(struct rtable *rt, struct flowi4 *fl4, u32 mtu)
{ {
struct dst_entry *dst = &rt->dst; struct dst_entry *dst = &rt->dst;
u32 old_mtu = ipv4_mtu(dst);
struct fib_result res; struct fib_result res;
bool lock = false; bool lock = false;
if (ip_mtu_locked(dst)) if (ip_mtu_locked(dst))
return; return;
if (ipv4_mtu(dst) < mtu) if (old_mtu < mtu)
return; return;
if (mtu < ip_rt_min_pmtu) { if (mtu < ip_rt_min_pmtu) {
lock = true; lock = true;
mtu = ip_rt_min_pmtu; mtu = min(old_mtu, ip_rt_min_pmtu);
} }
if (rt->rt_pmtu == mtu && if (rt->rt_pmtu == mtu && !lock &&
time_before(jiffies, dst->expires - ip_rt_mtu_expires / 2)) time_before(jiffies, dst->expires - ip_rt_mtu_expires / 2))
return; return;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment