1
0
mirror of https://github.com/systemd/systemd.git synced 2025-03-19 22:50:17 +03:00

network: process queued remove requests before networkd is stopped

This makes networkd process all queued remove requests when a
terminating or restarting signal is received. Otherwise, e.g. DHCPv4
address will not be removed on stop, especially when
KeepConfiguration=no.

Fixes a bug introduced by 85a6f300c14d75d161cbfdb3eaf5af9594400ecd and
its subsequent commits.

Fixes #34837.

Co-authored-by: Will Fancher <elvishjerricco@gmail.com>
(cherry picked from commit db68e990467ffea2b6655e4726856ca7eb32b38d)
This commit is contained in:
Yu Watanabe 2024-10-24 04:40:45 +09:00 committed by Luca Boccassi
parent bf681fcdf4
commit 96147d0104
4 changed files with 76 additions and 24 deletions

View File

@ -222,7 +222,7 @@ void link_dns_settings_clear(Link *link) {
link->dnssec_negative_trust_anchors = set_free_free(link->dnssec_negative_trust_anchors);
}
static void link_free_engines(Link *link) {
void link_free_engines(Link *link) {
if (!link)
return;
@ -376,7 +376,7 @@ int link_stop_engines(Link *link, bool may_keep_dhcp) {
bool keep_dhcp = may_keep_dhcp &&
link->network &&
!link->network->dhcp_send_decline && /* IPv4 ACD for the DHCPv4 address is running. */
(link->manager->restarting ||
(link->manager->state == MANAGER_RESTARTING ||
FLAGS_SET(link->network->keep_configuration, KEEP_CONFIGURATION_DHCP_ON_STOP));
if (!keep_dhcp) {
@ -1321,6 +1321,10 @@ int link_reconfigure_impl(Link *link, bool force) {
int r;
assert(link);
assert(link->manager);
if (link->manager->state != MANAGER_RUNNING)
return 0;
if (IN_SET(link->state, LINK_STATE_PENDING, LINK_STATE_LINGER))
return 0;

View File

@ -248,6 +248,7 @@ int link_ipv6ll_gained(Link *link);
bool link_has_ipv6_connectivity(Link *link);
int link_stop_engines(Link *link, bool may_keep_dhcp);
void link_free_engines(Link *link);
const char* link_state_to_string(LinkState s) _const_;
LinkState link_state_from_string(const char *s) _pure_;

View File

@ -425,30 +425,73 @@ static int manager_connect_rtnl(Manager *m, int fd) {
static int manager_post_handler(sd_event_source *s, void *userdata) {
Manager *manager = ASSERT_PTR(userdata);
/* To release dynamic leases, we need to process queued remove requests before stopping networkd.
* This is especially important when KeepConfiguration=no. See issue #34837. */
(void) manager_process_remove_requests(manager);
(void) manager_process_requests(manager);
(void) manager_clean_all(manager);
switch (manager->state) {
case MANAGER_RUNNING:
(void) manager_process_requests(manager);
(void) manager_clean_all(manager);
return 0;
case MANAGER_TERMINATING:
case MANAGER_RESTARTING:
if (!ordered_set_isempty(manager->remove_request_queue))
return 0; /* There are some unissued remove requests. */
if (netlink_get_reply_callback_count(manager->rtnl) > 0 ||
netlink_get_reply_callback_count(manager->genl) > 0 ||
fw_ctx_get_reply_callback_count(manager->fw_ctx) > 0)
return 0; /* There are some message calls waiting for their replies. */
manager->state = MANAGER_STOPPED;
return sd_event_exit(sd_event_source_get_event(s), 0);
default:
assert_not_reached();
}
return 0;
}
static int manager_stop(Manager *manager, ManagerState state) {
assert(manager);
assert(IN_SET(state, MANAGER_TERMINATING, MANAGER_RESTARTING));
if (manager->state != MANAGER_RUNNING) {
log_debug("Already terminating or restarting systemd-networkd, refusing further operation request.");
return 0;
}
switch (state) {
case MANAGER_TERMINATING:
log_debug("Terminate operation initiated.");
break;
case MANAGER_RESTARTING:
log_debug("Restart operation initiated.");
break;
default:
assert_not_reached();
}
manager->state = state;
Link *link;
HASHMAP_FOREACH(link, manager->links_by_index) {
(void) link_stop_engines(link, /* may_keep_dhcp = */ true);
link_free_engines(link);
}
return 0;
}
static int signal_terminate_callback(sd_event_source *s, const struct signalfd_siginfo *si, void *userdata) {
Manager *m = ASSERT_PTR(userdata);
m->restarting = false;
log_debug("Terminate operation initiated.");
return sd_event_exit(sd_event_source_get_event(s), 0);
return manager_stop(userdata, MANAGER_TERMINATING);
}
static int signal_restart_callback(sd_event_source *s, const struct signalfd_siginfo *si, void *userdata) {
Manager *m = ASSERT_PTR(userdata);
m->restarting = true;
log_debug("Restart operation initiated.");
return sd_event_exit(sd_event_source_get_event(s), 0);
return manager_stop(userdata, MANAGER_RESTARTING);
}
static int signal_reload_callback(sd_event_source *s, const struct signalfd_siginfo *si, void *userdata) {
@ -614,16 +657,11 @@ int manager_new(Manager **ret, bool test_mode) {
}
Manager* manager_free(Manager *m) {
Link *link;
if (!m)
return NULL;
free(m->state_file);
HASHMAP_FOREACH(link, m->links_by_index)
(void) link_stop_engines(link, true);
m->request_queue = ordered_set_free(m->request_queue);
m->remove_request_queue = ordered_set_free(m->remove_request_queue);

View File

@ -19,6 +19,15 @@
#include "time-util.h"
#include "varlink.h"
typedef enum ManagerState {
MANAGER_RUNNING,
MANAGER_TERMINATING,
MANAGER_RESTARTING,
MANAGER_STOPPED,
_MANAGER_STATE_MAX,
_MANAGER_STATE_INVALID = -EINVAL,
} ManagerState;
struct Manager {
sd_netlink *rtnl;
/* lazy initialized */
@ -35,10 +44,10 @@ struct Manager {
KeepConfiguration keep_configuration;
IPv6PrivacyExtensions ipv6_privacy_extensions;
ManagerState state;
bool test_mode;
bool enumerating;
bool dirty;
bool restarting;
bool manage_foreign_routes;
bool manage_foreign_rules;
bool manage_foreign_nexthops;