Commit f4bcbf36 authored by Alex Henrie's avatar Alex Henrie Committed by Paolo Abeni

net: ipv6/addrconf: clamp preferred_lft to the minimum required

If the preferred lifetime was less than the minimum required lifetime,
ipv6_create_tempaddr would error out without creating any new address.
On my machine and network, this error happened immediately with the
preferred lifetime set to 5 seconds or less, after a few minutes with
the preferred lifetime set to 6 seconds, and not at all with the
preferred lifetime set to 7 seconds. During my investigation, I found a
Stack Exchange post from another person who seems to have had the same
problem: They stopped getting new addresses if they lowered the
preferred lifetime below 3 seconds, and they didn't really know why.

The preferred lifetime is a preference, not a hard requirement. The
kernel does not strictly forbid new connections on a deprecated address,
nor does it guarantee that the address will be disposed of the instant
its total valid lifetime expires. So rather than disable IPv6 privacy
extensions altogether if the minimum required lifetime swells above the
preferred lifetime, it is more in keeping with the user's intent to
increase the temporary address's lifetime to the minimum necessary for
the current network conditions.

With these fixes, setting the preferred lifetime to 5 or 6 seconds "just
works" because the extra fraction of a second is practically
unnoticeable. It's even possible to reduce the time before deprecation
to 1 or 2 seconds by setting /proc/sys/net/ipv6/conf/*/regen_min_advance
and /proc/sys/net/ipv6/conf/*/dad_transmits to 0. I realize that that is
a pretty niche use case, but I know at least one person who would gladly
sacrifice performance and convenience to be sure that they are getting
the maximum possible level of privacy.

Link: https://serverfault.com/a/1031168/310447Signed-off-by: default avatarAlex Henrie <alexhenrie24@gmail.com>
Reviewed-by: default avatarDavid Ahern <dsahern@kernel.org>
Signed-off-by: default avatarPaolo Abeni <pabeni@redhat.com>
parent a5fcea2d
...@@ -2511,7 +2511,7 @@ temp_valid_lft - INTEGER ...@@ -2511,7 +2511,7 @@ temp_valid_lft - INTEGER
temp_prefered_lft - INTEGER temp_prefered_lft - INTEGER
Preferred lifetime (in seconds) for temporary addresses. If Preferred lifetime (in seconds) for temporary addresses. If
temp_prefered_lft is less than the minimum required lifetime (typically temp_prefered_lft is less than the minimum required lifetime (typically
5-7 seconds), temporary addresses will not be created. If 5-7 seconds), the preferred lifetime is the minimum required. If
temp_prefered_lft is greater than temp_valid_lft, the preferred lifetime temp_prefered_lft is greater than temp_valid_lft, the preferred lifetime
is temp_valid_lft. is temp_valid_lft.
......
...@@ -1354,6 +1354,7 @@ static int ipv6_create_tempaddr(struct inet6_ifaddr *ifp, bool block) ...@@ -1354,6 +1354,7 @@ static int ipv6_create_tempaddr(struct inet6_ifaddr *ifp, bool block)
unsigned long tmp_tstamp, age; unsigned long tmp_tstamp, age;
unsigned long regen_advance; unsigned long regen_advance;
unsigned long now = jiffies; unsigned long now = jiffies;
u32 if_public_preferred_lft;
s32 cnf_temp_preferred_lft; s32 cnf_temp_preferred_lft;
struct inet6_ifaddr *ift; struct inet6_ifaddr *ift;
struct ifa6_config cfg; struct ifa6_config cfg;
...@@ -1409,11 +1410,13 @@ static int ipv6_create_tempaddr(struct inet6_ifaddr *ifp, bool block) ...@@ -1409,11 +1410,13 @@ static int ipv6_create_tempaddr(struct inet6_ifaddr *ifp, bool block)
} }
} }
if_public_preferred_lft = ifp->prefered_lft;
memset(&cfg, 0, sizeof(cfg)); memset(&cfg, 0, sizeof(cfg));
cfg.valid_lft = min_t(__u32, ifp->valid_lft, cfg.valid_lft = min_t(__u32, ifp->valid_lft,
idev->cnf.temp_valid_lft + age); idev->cnf.temp_valid_lft + age);
cfg.preferred_lft = cnf_temp_preferred_lft + age - idev->desync_factor; cfg.preferred_lft = cnf_temp_preferred_lft + age - idev->desync_factor;
cfg.preferred_lft = min_t(__u32, ifp->prefered_lft, cfg.preferred_lft); cfg.preferred_lft = min_t(__u32, if_public_preferred_lft, cfg.preferred_lft);
cfg.preferred_lft = min_t(__u32, cfg.valid_lft, cfg.preferred_lft); cfg.preferred_lft = min_t(__u32, cfg.valid_lft, cfg.preferred_lft);
cfg.plen = ifp->prefix_len; cfg.plen = ifp->prefix_len;
...@@ -1422,19 +1425,41 @@ static int ipv6_create_tempaddr(struct inet6_ifaddr *ifp, bool block) ...@@ -1422,19 +1425,41 @@ static int ipv6_create_tempaddr(struct inet6_ifaddr *ifp, bool block)
write_unlock_bh(&idev->lock); write_unlock_bh(&idev->lock);
/* A temporary address is created only if this calculated Preferred /* From RFC 4941:
* Lifetime is greater than REGEN_ADVANCE time units. In particular, *
* an implementation must not create a temporary address with a zero * A temporary address is created only if this calculated Preferred
* Preferred Lifetime. * Lifetime is greater than REGEN_ADVANCE time units. In
* particular, an implementation must not create a temporary address
* with a zero Preferred Lifetime.
*
* ...
*
* When creating a temporary address, the lifetime values MUST be
* derived from the corresponding prefix as follows:
*
* ...
*
* * Its Preferred Lifetime is the lower of the Preferred Lifetime
* of the public address or TEMP_PREFERRED_LIFETIME -
* DESYNC_FACTOR.
*
* To comply with the RFC's requirements, clamp the preferred lifetime
* to a minimum of regen_advance, unless that would exceed valid_lft or
* ifp->prefered_lft.
*
* Use age calculation as in addrconf_verify to avoid unnecessary * Use age calculation as in addrconf_verify to avoid unnecessary
* temporary addresses being generated. * temporary addresses being generated.
*/ */
age = (now - tmp_tstamp + ADDRCONF_TIMER_FUZZ_MINUS) / HZ; age = (now - tmp_tstamp + ADDRCONF_TIMER_FUZZ_MINUS) / HZ;
if (cfg.preferred_lft <= regen_advance + age) { if (cfg.preferred_lft <= regen_advance + age) {
in6_ifa_put(ifp); cfg.preferred_lft = regen_advance + age + 1;
in6_dev_put(idev); if (cfg.preferred_lft > cfg.valid_lft ||
ret = -1; cfg.preferred_lft > if_public_preferred_lft) {
goto out; in6_ifa_put(ifp);
in6_dev_put(idev);
ret = -1;
goto out;
}
} }
cfg.ifa_flags = IFA_F_TEMPORARY; cfg.ifa_flags = IFA_F_TEMPORARY;
......
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