From: Yu Watanabe Date: Sun, 18 Feb 2024 06:23:09 +0000 (+0900) Subject: sd-radv: set only basic information on stop X-Git-Tag: v256-rc1~126^2~1 X-Git-Url: http://git.ipfire.org/cgi-bin/gitweb.cgi?a=commitdiff_plain;h=19f3cc86a04ff7870dea3ac97fcbf624db3330fe;p=thirdparty%2Fsystemd.git sd-radv: set only basic information on stop There are many possible reasons to stop the service; tentative reboot of the service or the system, the router may be revoked, and so on. And, each situations, the availability of the previously announced options e.g. prefixes, DNSSL and so on is not clear. So, let's announce only the clear information, that is, the router lifetime is zero. which indicates that the router will be invalidated (regardless tentative or not). --- diff --git a/src/libsystemd-network/sd-radv.c b/src/libsystemd-network/sd-radv.c index 79210f9e2ca..69ebd94fb16 100644 --- a/src/libsystemd-network/sd-radv.c +++ b/src/libsystemd-network/sd-radv.c @@ -129,6 +129,30 @@ static bool router_lifetime_is_valid(usec_t lifetime_usec) { lifetime_usec <= RADV_MAX_ROUTER_LIFETIME_USEC); } +static int radv_send_router_on_stop(sd_radv *ra) { + static const struct nd_router_advert adv = { + .nd_ra_type = ND_ROUTER_ADVERT, + }; + + _cleanup_set_free_ Set *options = NULL; + usec_t time_now; + int r; + + assert(ra); + + r = sd_event_now(ra->event, CLOCK_BOOTTIME, &time_now); + if (r < 0) + return r; + + if (!ether_addr_is_null(&ra->mac_addr)) { + r = ndisc_option_set_link_layer_address(&options, SD_NDISC_OPTION_SOURCE_LL_ADDRESS, &ra->mac_addr); + if (r < 0) + return r; + } + + return ndisc_send(ra->fd, &IN6_ADDR_ALL_NODES_MULTICAST, &adv.nd_ra_hdr, options, time_now); +} + static int radv_send_router(sd_radv *ra, const struct in6_addr *dst, usec_t lifetime_usec) { assert(ra); assert(router_lifetime_is_valid(lifetime_usec)); @@ -362,7 +386,7 @@ int sd_radv_stop(sd_radv *ra) { /* RFC 4861, Section 6.2.5: * the router SHOULD transmit one or more (but not more than MAX_FINAL_RTR_ADVERTISEMENTS) final * multicast Router Advertisements on the interface with a Router Lifetime field of zero. */ - r = radv_send_router(ra, NULL, 0); + r = radv_send_router_on_stop(ra); if (r < 0) log_radv_errno(ra, r, "Unable to send last Router Advertisement with router lifetime set to zero, ignoring: %m"); diff --git a/src/libsystemd-network/test-ndisc-ra.c b/src/libsystemd-network/test-ndisc-ra.c index c5f10cc7955..76f4d7ba3f2 100644 --- a/src/libsystemd-network/test-ndisc-ra.c +++ b/src/libsystemd-network/test-ndisc-ra.c @@ -294,8 +294,13 @@ static void verify_message(const uint8_t *buf, size_t len) { /* verify only up to known options, rest is not yet implemented */ for (size_t i = 0, m = MIN(len, sizeof(advertisement)); i < m; i++) { if (test_stopped) + /* on stop, many header fields are zero */ switch (i) { - case 6 ... 7: /* router lifetime must be zero on stop. */ + case 4: /* hop limit */ + case 5: /* flags */ + case 6 ... 7: /* router lifetime */ + case 8 ... 11: /* reachable time */ + case 12 ... 15: /* retrans timer */ assert_se(buf[i] == 0); continue; }