networkd: manager/link - only serialize once per event-loop iteration
Every time the state is written out we may trigger third-party apps, so let's be a bit more careful about writing this out unnecessarily.
This commit is contained in:
parent
e7780c8d44
commit
84de38c569
|
@ -98,7 +98,6 @@ void address_free(Address *address) {
|
|||
if (address->link) {
|
||||
set_remove(address->link->addresses, address);
|
||||
set_remove(address->link->addresses_foreign, address);
|
||||
link_save(address->link);
|
||||
}
|
||||
|
||||
free(address);
|
||||
|
@ -277,7 +276,8 @@ static int address_add(Link *link, int family, const union in_addr_union *in_add
|
|||
if (r < 0)
|
||||
return r;
|
||||
|
||||
link_save(link);
|
||||
link_update_operstate(link);
|
||||
link_dirty(link);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
@ -334,6 +334,8 @@ int address_drop(Address *address) {
|
|||
address_release(address);
|
||||
address_free(address);
|
||||
|
||||
link_update_operstate(link);
|
||||
|
||||
if (link && !ready)
|
||||
link_check_ready(link);
|
||||
|
||||
|
|
|
@ -130,6 +130,57 @@ static IPv6PrivacyExtensions link_ipv6_privacy_extensions(Link *link) {
|
|||
return link->network->ipv6_privacy_extensions;
|
||||
}
|
||||
|
||||
void link_update_operstate(Link *link) {
|
||||
LinkOperationalState operstate;
|
||||
assert(link);
|
||||
|
||||
if (link->kernel_operstate == IF_OPER_DORMANT)
|
||||
operstate = LINK_OPERSTATE_DORMANT;
|
||||
else if (link_has_carrier(link)) {
|
||||
Address *address;
|
||||
uint8_t scope = RT_SCOPE_NOWHERE;
|
||||
Iterator i;
|
||||
|
||||
/* if we have carrier, check what addresses we have */
|
||||
SET_FOREACH(address, link->addresses, i) {
|
||||
if (!address_is_ready(address))
|
||||
continue;
|
||||
|
||||
if (address->scope < scope)
|
||||
scope = address->scope;
|
||||
}
|
||||
|
||||
/* for operstate we also take foreign addresses into account */
|
||||
SET_FOREACH(address, link->addresses_foreign, i) {
|
||||
if (!address_is_ready(address))
|
||||
continue;
|
||||
|
||||
if (address->scope < scope)
|
||||
scope = address->scope;
|
||||
}
|
||||
|
||||
if (scope < RT_SCOPE_SITE)
|
||||
/* universally accessible addresses found */
|
||||
operstate = LINK_OPERSTATE_ROUTABLE;
|
||||
else if (scope < RT_SCOPE_HOST)
|
||||
/* only link or site local addresses found */
|
||||
operstate = LINK_OPERSTATE_DEGRADED;
|
||||
else
|
||||
/* no useful addresses found */
|
||||
operstate = LINK_OPERSTATE_CARRIER;
|
||||
} else if (link->flags & IFF_UP)
|
||||
operstate = LINK_OPERSTATE_NO_CARRIER;
|
||||
else
|
||||
operstate = LINK_OPERSTATE_OFF;
|
||||
|
||||
if (link->operstate != operstate) {
|
||||
link->operstate = operstate;
|
||||
link_send_changed(link, "OperationalState", NULL);
|
||||
link_dirty(link);
|
||||
manager_dirty(link->manager);
|
||||
}
|
||||
}
|
||||
|
||||
#define FLAG_STRING(string, flag, old, new) \
|
||||
(((old ^ new) & flag) \
|
||||
? ((old & flag) ? (" -" string) : (" +" string)) \
|
||||
|
@ -202,7 +253,7 @@ static int link_update_flags(Link *link, sd_netlink_message *m) {
|
|||
link->flags = flags;
|
||||
link->kernel_operstate = operstate;
|
||||
|
||||
link_save(link);
|
||||
link_update_operstate(link);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
@ -326,6 +377,7 @@ static void link_free(Link *link) {
|
|||
|
||||
free(link->ifname);
|
||||
|
||||
(void)unlink(link->state_file);
|
||||
free(link->state_file);
|
||||
|
||||
udev_device_unref(link->udev_device);
|
||||
|
@ -404,7 +456,7 @@ static void link_enter_unmanaged(Link *link) {
|
|||
|
||||
link_set_state(link, LINK_STATE_UNMANAGED);
|
||||
|
||||
link_save(link);
|
||||
link_dirty(link);
|
||||
}
|
||||
|
||||
static int link_stop_clients(Link *link) {
|
||||
|
@ -462,7 +514,7 @@ void link_enter_failed(Link *link) {
|
|||
|
||||
link_stop_clients(link);
|
||||
|
||||
link_save(link);
|
||||
link_dirty(link);
|
||||
}
|
||||
|
||||
static Address* link_find_dhcp_server_address(Link *link) {
|
||||
|
@ -503,7 +555,7 @@ static int link_enter_configured(Link *link) {
|
|||
|
||||
link_set_state(link, LINK_STATE_CONFIGURED);
|
||||
|
||||
link_save(link);
|
||||
link_dirty(link);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
@ -1460,14 +1512,14 @@ static int link_new_bound_by_list(Link *link) {
|
|||
}
|
||||
|
||||
if (list_updated)
|
||||
link_save(link);
|
||||
link_dirty(link);
|
||||
|
||||
HASHMAP_FOREACH (carrier, link->bound_by_links, i) {
|
||||
r = link_put_carrier(carrier, link, &carrier->bound_to_links);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
link_save(carrier);
|
||||
link_dirty(carrier);
|
||||
}
|
||||
|
||||
return 0;
|
||||
|
@ -1502,14 +1554,14 @@ static int link_new_bound_to_list(Link *link) {
|
|||
}
|
||||
|
||||
if (list_updated)
|
||||
link_save(link);
|
||||
link_dirty(link);
|
||||
|
||||
HASHMAP_FOREACH (carrier, link->bound_to_links, i) {
|
||||
r = link_put_carrier(carrier, link, &carrier->bound_by_links);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
link_save(carrier);
|
||||
link_dirty(carrier);
|
||||
}
|
||||
|
||||
return 0;
|
||||
|
@ -1545,7 +1597,7 @@ static void link_free_bound_to_list(Link *link) {
|
|||
hashmap_remove(link->bound_to_links, INT_TO_PTR(bound_to->ifindex));
|
||||
|
||||
if (hashmap_remove(bound_to->bound_by_links, INT_TO_PTR(link->ifindex)))
|
||||
link_save(bound_to);
|
||||
link_dirty(bound_to);
|
||||
}
|
||||
|
||||
return;
|
||||
|
@ -1559,7 +1611,7 @@ static void link_free_bound_by_list(Link *link) {
|
|||
hashmap_remove(link->bound_by_links, INT_TO_PTR(bound_by->ifindex));
|
||||
|
||||
if (hashmap_remove(bound_by->bound_to_links, INT_TO_PTR(link->ifindex))) {
|
||||
link_save(bound_by);
|
||||
link_dirty(bound_by);
|
||||
link_handle_bound_to_list(bound_by);
|
||||
}
|
||||
}
|
||||
|
@ -1583,7 +1635,7 @@ static void link_free_carrier_maps(Link *link) {
|
|||
}
|
||||
|
||||
if (list_updated)
|
||||
link_save(link);
|
||||
link_dirty(link);
|
||||
|
||||
return;
|
||||
}
|
||||
|
@ -1598,6 +1650,7 @@ void link_drop(Link *link) {
|
|||
|
||||
log_link_debug(link, "Link removed");
|
||||
|
||||
(void)unlink(link->state_file);
|
||||
link_unref(link);
|
||||
|
||||
return;
|
||||
|
@ -1667,7 +1720,7 @@ static int link_enter_join_netdev(Link *link) {
|
|||
|
||||
link_set_state(link, LINK_STATE_ENSLAVING);
|
||||
|
||||
link_save(link);
|
||||
link_dirty(link);
|
||||
|
||||
if (!link->network->bridge &&
|
||||
!link->network->bond &&
|
||||
|
@ -2311,55 +2364,6 @@ int link_update(Link *link, sd_netlink_message *m) {
|
|||
return 0;
|
||||
}
|
||||
|
||||
static void link_update_operstate(Link *link) {
|
||||
LinkOperationalState operstate;
|
||||
assert(link);
|
||||
|
||||
if (link->kernel_operstate == IF_OPER_DORMANT)
|
||||
operstate = LINK_OPERSTATE_DORMANT;
|
||||
else if (link_has_carrier(link)) {
|
||||
Address *address;
|
||||
uint8_t scope = RT_SCOPE_NOWHERE;
|
||||
Iterator i;
|
||||
|
||||
/* if we have carrier, check what addresses we have */
|
||||
SET_FOREACH(address, link->addresses, i) {
|
||||
if (!address_is_ready(address))
|
||||
continue;
|
||||
|
||||
if (address->scope < scope)
|
||||
scope = address->scope;
|
||||
}
|
||||
|
||||
/* for operstate we also take foreign addresses into account */
|
||||
SET_FOREACH(address, link->addresses_foreign, i) {
|
||||
if (!address_is_ready(address))
|
||||
continue;
|
||||
|
||||
if (address->scope < scope)
|
||||
scope = address->scope;
|
||||
}
|
||||
|
||||
if (scope < RT_SCOPE_SITE)
|
||||
/* universally accessible addresses found */
|
||||
operstate = LINK_OPERSTATE_ROUTABLE;
|
||||
else if (scope < RT_SCOPE_HOST)
|
||||
/* only link or site local addresses found */
|
||||
operstate = LINK_OPERSTATE_DEGRADED;
|
||||
else
|
||||
/* no useful addresses found */
|
||||
operstate = LINK_OPERSTATE_CARRIER;
|
||||
} else if (link->flags & IFF_UP)
|
||||
operstate = LINK_OPERSTATE_NO_CARRIER;
|
||||
else
|
||||
operstate = LINK_OPERSTATE_OFF;
|
||||
|
||||
if (link->operstate != operstate) {
|
||||
link->operstate = operstate;
|
||||
link_send_changed(link, "OperationalState", NULL);
|
||||
}
|
||||
}
|
||||
|
||||
int link_save(Link *link) {
|
||||
_cleanup_free_ char *temp_path = NULL;
|
||||
_cleanup_fclose_ FILE *f = NULL;
|
||||
|
@ -2373,12 +2377,6 @@ int link_save(Link *link) {
|
|||
assert(link->lease_file);
|
||||
assert(link->manager);
|
||||
|
||||
link_update_operstate(link);
|
||||
|
||||
r = manager_save(link->manager);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
if (link->state == LINK_STATE_LINGER) {
|
||||
unlink(link->state_file);
|
||||
return 0;
|
||||
|
@ -2553,8 +2551,6 @@ int link_save(Link *link) {
|
|||
if (r < 0)
|
||||
goto fail;
|
||||
|
||||
if (space)
|
||||
fputc(' ', f);
|
||||
fprintf(f, "%s%s/%u", space ? " " : "", address_str, a->prefixlen);
|
||||
space = true;
|
||||
}
|
||||
|
@ -2645,6 +2641,34 @@ fail:
|
|||
return log_link_error_errno(link, r, "Failed to save link data to %s: %m", link->state_file);
|
||||
}
|
||||
|
||||
/* The serialized state in /run is no longer up-to-date. */
|
||||
void link_dirty(Link *link) {
|
||||
int r;
|
||||
|
||||
assert(link);
|
||||
|
||||
r = set_ensure_allocated(&link->manager->dirty_links, NULL);
|
||||
if (r < 0)
|
||||
/* allocation errors are ignored */
|
||||
return;
|
||||
|
||||
r = set_put(link->manager->dirty_links, link);
|
||||
if (r < 0)
|
||||
/* allocation errors are ignored */
|
||||
return;
|
||||
|
||||
link_ref(link);
|
||||
}
|
||||
|
||||
/* The serialized state in /run is up-to-date */
|
||||
void link_clean(Link *link) {
|
||||
assert(link);
|
||||
assert(link->manager);
|
||||
|
||||
set_remove(link->manager->dirty_links, link);
|
||||
link_unref(link);
|
||||
}
|
||||
|
||||
static const char* const link_state_table[_LINK_STATE_MAX] = {
|
||||
[LINK_STATE_PENDING] = "pending",
|
||||
[LINK_STATE_ENSLAVING] = "configuring",
|
||||
|
|
|
@ -129,8 +129,11 @@ int link_initialized(Link *link, struct udev_device *device);
|
|||
|
||||
void link_check_ready(Link *link);
|
||||
|
||||
void link_update_operstate(Link *link);
|
||||
int link_update(Link *link, sd_netlink_message *message);
|
||||
|
||||
void link_dirty(Link *link);
|
||||
void link_clean(Link *link);
|
||||
int link_save(Link *link);
|
||||
|
||||
int link_carrier_reset(Link *link);
|
||||
|
|
|
@ -573,228 +573,6 @@ static int manager_connect_rtnl(Manager *m) {
|
|||
return 0;
|
||||
}
|
||||
|
||||
int manager_new(Manager **ret) {
|
||||
_cleanup_manager_free_ Manager *m = NULL;
|
||||
int r;
|
||||
|
||||
m = new0(Manager, 1);
|
||||
if (!m)
|
||||
return -ENOMEM;
|
||||
|
||||
m->state_file = strdup("/run/systemd/netif/state");
|
||||
if (!m->state_file)
|
||||
return -ENOMEM;
|
||||
|
||||
r = sd_event_default(&m->event);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
sd_event_set_watchdog(m->event, true);
|
||||
|
||||
sd_event_add_signal(m->event, NULL, SIGTERM, NULL, NULL);
|
||||
sd_event_add_signal(m->event, NULL, SIGINT, NULL, NULL);
|
||||
|
||||
r = manager_connect_rtnl(m);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
r = manager_connect_udev(m);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
m->netdevs = hashmap_new(&string_hash_ops);
|
||||
if (!m->netdevs)
|
||||
return -ENOMEM;
|
||||
|
||||
LIST_HEAD_INIT(m->networks);
|
||||
|
||||
r = setup_default_address_pool(m);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
*ret = m;
|
||||
m = NULL;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
void manager_free(Manager *m) {
|
||||
Network *network;
|
||||
NetDev *netdev;
|
||||
Link *link;
|
||||
AddressPool *pool;
|
||||
|
||||
if (!m)
|
||||
return;
|
||||
|
||||
free(m->state_file);
|
||||
|
||||
while ((link = hashmap_first(m->links)))
|
||||
link_unref(link);
|
||||
hashmap_free(m->links);
|
||||
|
||||
while ((network = m->networks))
|
||||
network_free(network);
|
||||
|
||||
hashmap_free(m->networks_by_name);
|
||||
|
||||
while ((netdev = hashmap_first(m->netdevs)))
|
||||
netdev_unref(netdev);
|
||||
hashmap_free(m->netdevs);
|
||||
|
||||
while ((pool = m->address_pools))
|
||||
address_pool_free(pool);
|
||||
|
||||
sd_netlink_unref(m->rtnl);
|
||||
sd_event_unref(m->event);
|
||||
|
||||
sd_event_source_unref(m->udev_event_source);
|
||||
udev_monitor_unref(m->udev_monitor);
|
||||
udev_unref(m->udev);
|
||||
|
||||
sd_bus_unref(m->bus);
|
||||
sd_bus_slot_unref(m->prepare_for_sleep_slot);
|
||||
sd_event_source_unref(m->bus_retry_event_source);
|
||||
|
||||
free(m);
|
||||
}
|
||||
|
||||
static bool manager_check_idle(void *userdata) {
|
||||
Manager *m = userdata;
|
||||
Link *link;
|
||||
Iterator i;
|
||||
|
||||
assert(m);
|
||||
|
||||
HASHMAP_FOREACH(link, m->links, i) {
|
||||
/* we are not woken on udev activity, so let's just wait for the
|
||||
* pending udev event */
|
||||
if (link->state == LINK_STATE_PENDING)
|
||||
return false;
|
||||
|
||||
if (!link->network)
|
||||
continue;
|
||||
|
||||
/* we are not woken on netork activity, so let's stay around */
|
||||
if (link_lldp_enabled(link) ||
|
||||
link_ipv4ll_enabled(link) ||
|
||||
link_dhcp4_server_enabled(link) ||
|
||||
link_dhcp4_enabled(link) ||
|
||||
link_dhcp6_enabled(link))
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
int manager_run(Manager *m) {
|
||||
assert(m);
|
||||
|
||||
if (m->bus)
|
||||
return bus_event_loop_with_idle(
|
||||
m->event,
|
||||
m->bus,
|
||||
"org.freedesktop.network1",
|
||||
DEFAULT_EXIT_USEC,
|
||||
manager_check_idle,
|
||||
m);
|
||||
else
|
||||
/* failed to connect to the bus, so we lose exit-on-idle logic,
|
||||
this should not happen except if dbus is not around at all */
|
||||
return sd_event_loop(m->event);
|
||||
}
|
||||
|
||||
int manager_load_config(Manager *m) {
|
||||
int r;
|
||||
|
||||
/* update timestamp */
|
||||
paths_check_timestamp(network_dirs, &m->network_dirs_ts_usec, true);
|
||||
|
||||
r = netdev_load(m);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
r = network_load(m);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
bool manager_should_reload(Manager *m) {
|
||||
return paths_check_timestamp(network_dirs, &m->network_dirs_ts_usec, false);
|
||||
}
|
||||
|
||||
int manager_rtnl_enumerate_links(Manager *m) {
|
||||
_cleanup_netlink_message_unref_ sd_netlink_message *req = NULL, *reply = NULL;
|
||||
sd_netlink_message *link;
|
||||
int r;
|
||||
|
||||
assert(m);
|
||||
assert(m->rtnl);
|
||||
|
||||
r = sd_rtnl_message_new_link(m->rtnl, &req, RTM_GETLINK, 0);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
r = sd_netlink_message_request_dump(req, true);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
r = sd_netlink_call(m->rtnl, req, 0, &reply);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
for (link = reply; link; link = sd_netlink_message_next(link)) {
|
||||
int k;
|
||||
|
||||
m->enumerating = true;
|
||||
|
||||
k = manager_rtnl_process_link(m->rtnl, link, m);
|
||||
if (k < 0)
|
||||
r = k;
|
||||
|
||||
m->enumerating = false;
|
||||
}
|
||||
|
||||
return r;
|
||||
}
|
||||
|
||||
int manager_rtnl_enumerate_addresses(Manager *m) {
|
||||
_cleanup_netlink_message_unref_ sd_netlink_message *req = NULL, *reply = NULL;
|
||||
sd_netlink_message *addr;
|
||||
int r;
|
||||
|
||||
assert(m);
|
||||
assert(m->rtnl);
|
||||
|
||||
r = sd_rtnl_message_new_addr(m->rtnl, &req, RTM_GETADDR, 0, 0);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
r = sd_netlink_message_request_dump(req, true);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
r = sd_netlink_call(m->rtnl, req, 0, &reply);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
for (addr = reply; addr; addr = sd_netlink_message_next(addr)) {
|
||||
int k;
|
||||
|
||||
m->enumerating = true;
|
||||
|
||||
k = manager_rtnl_process_address(m->rtnl, addr, m);
|
||||
if (k < 0)
|
||||
r = k;
|
||||
|
||||
m->enumerating = false;
|
||||
}
|
||||
|
||||
return r;
|
||||
}
|
||||
|
||||
static int set_put_in_addr(Set *s, const struct in_addr *address) {
|
||||
char *p;
|
||||
int r;
|
||||
|
@ -848,7 +626,7 @@ static void print_string_set(FILE *f, const char *field, Set *s) {
|
|||
fputc('\n', f);
|
||||
}
|
||||
|
||||
int manager_save(Manager *m) {
|
||||
static int manager_save(Manager *m) {
|
||||
_cleanup_set_free_free_ Set *dns = NULL, *ntp = NULL, *domains = NULL;
|
||||
Link *link;
|
||||
Iterator i;
|
||||
|
@ -971,6 +749,8 @@ int manager_save(Manager *m) {
|
|||
log_error_errno(r, "Could not emit changed OperationalState: %m");
|
||||
}
|
||||
|
||||
m->dirty = false;
|
||||
|
||||
return 0;
|
||||
|
||||
fail:
|
||||
|
@ -980,6 +760,263 @@ fail:
|
|||
return log_error_errno(r, "Failed to save network state to %s: %m", m->state_file);
|
||||
}
|
||||
|
||||
static int manager_dirty_handler(sd_event_source *s, void *userdata) {
|
||||
Manager *m = userdata;
|
||||
Link *link;
|
||||
Iterator i;
|
||||
int r;
|
||||
|
||||
assert(m);
|
||||
|
||||
if (m->dirty)
|
||||
manager_save(m);
|
||||
|
||||
SET_FOREACH(link, m->dirty_links, i) {
|
||||
r = link_save(link);
|
||||
if (r >= 0)
|
||||
link_clean(link);
|
||||
}
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
int manager_new(Manager **ret) {
|
||||
_cleanup_manager_free_ Manager *m = NULL;
|
||||
int r;
|
||||
|
||||
m = new0(Manager, 1);
|
||||
if (!m)
|
||||
return -ENOMEM;
|
||||
|
||||
m->state_file = strdup("/run/systemd/netif/state");
|
||||
if (!m->state_file)
|
||||
return -ENOMEM;
|
||||
|
||||
r = sd_event_default(&m->event);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
sd_event_set_watchdog(m->event, true);
|
||||
|
||||
sd_event_add_signal(m->event, NULL, SIGTERM, NULL, NULL);
|
||||
sd_event_add_signal(m->event, NULL, SIGINT, NULL, NULL);
|
||||
|
||||
r = sd_event_add_post(m->event, NULL, manager_dirty_handler, m);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
r = manager_connect_rtnl(m);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
r = manager_connect_udev(m);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
m->netdevs = hashmap_new(&string_hash_ops);
|
||||
if (!m->netdevs)
|
||||
return -ENOMEM;
|
||||
|
||||
LIST_HEAD_INIT(m->networks);
|
||||
|
||||
r = setup_default_address_pool(m);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
*ret = m;
|
||||
m = NULL;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
void manager_free(Manager *m) {
|
||||
Network *network;
|
||||
NetDev *netdev;
|
||||
Link *link;
|
||||
AddressPool *pool;
|
||||
|
||||
if (!m)
|
||||
return;
|
||||
|
||||
free(m->state_file);
|
||||
|
||||
while ((link = hashmap_first(m->links)))
|
||||
link_unref(link);
|
||||
hashmap_free(m->links);
|
||||
|
||||
while ((network = m->networks))
|
||||
network_free(network);
|
||||
|
||||
hashmap_free(m->networks_by_name);
|
||||
|
||||
while ((netdev = hashmap_first(m->netdevs)))
|
||||
netdev_unref(netdev);
|
||||
hashmap_free(m->netdevs);
|
||||
|
||||
while ((pool = m->address_pools))
|
||||
address_pool_free(pool);
|
||||
|
||||
sd_netlink_unref(m->rtnl);
|
||||
sd_event_unref(m->event);
|
||||
|
||||
sd_event_source_unref(m->udev_event_source);
|
||||
udev_monitor_unref(m->udev_monitor);
|
||||
udev_unref(m->udev);
|
||||
|
||||
sd_bus_unref(m->bus);
|
||||
sd_bus_slot_unref(m->prepare_for_sleep_slot);
|
||||
sd_event_source_unref(m->bus_retry_event_source);
|
||||
|
||||
free(m);
|
||||
}
|
||||
|
||||
static bool manager_check_idle(void *userdata) {
|
||||
Manager *m = userdata;
|
||||
Link *link;
|
||||
Iterator i;
|
||||
|
||||
assert(m);
|
||||
|
||||
HASHMAP_FOREACH(link, m->links, i) {
|
||||
/* we are not woken on udev activity, so let's just wait for the
|
||||
* pending udev event */
|
||||
if (link->state == LINK_STATE_PENDING)
|
||||
return false;
|
||||
|
||||
if (!link->network)
|
||||
continue;
|
||||
|
||||
/* we are not woken on netork activity, so let's stay around */
|
||||
if (link_lldp_enabled(link) ||
|
||||
link_ipv4ll_enabled(link) ||
|
||||
link_dhcp4_server_enabled(link) ||
|
||||
link_dhcp4_enabled(link) ||
|
||||
link_dhcp6_enabled(link))
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
int manager_run(Manager *m) {
|
||||
Link *link;
|
||||
Iterator i;
|
||||
|
||||
assert(m);
|
||||
|
||||
/* The dirty handler will deal with future serialization, but the first one
|
||||
must be done explicitly. */
|
||||
|
||||
manager_save(m);
|
||||
|
||||
HASHMAP_FOREACH(link, m->links, i)
|
||||
link_save(link);
|
||||
|
||||
if (m->bus)
|
||||
return bus_event_loop_with_idle(
|
||||
m->event,
|
||||
m->bus,
|
||||
"org.freedesktop.network1",
|
||||
DEFAULT_EXIT_USEC,
|
||||
manager_check_idle,
|
||||
m);
|
||||
else
|
||||
/* failed to connect to the bus, so we lose exit-on-idle logic,
|
||||
this should not happen except if dbus is not around at all */
|
||||
return sd_event_loop(m->event);
|
||||
}
|
||||
|
||||
int manager_load_config(Manager *m) {
|
||||
int r;
|
||||
|
||||
/* update timestamp */
|
||||
paths_check_timestamp(network_dirs, &m->network_dirs_ts_usec, true);
|
||||
|
||||
r = netdev_load(m);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
r = network_load(m);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
bool manager_should_reload(Manager *m) {
|
||||
return paths_check_timestamp(network_dirs, &m->network_dirs_ts_usec, false);
|
||||
}
|
||||
|
||||
int manager_rtnl_enumerate_links(Manager *m) {
|
||||
_cleanup_netlink_message_unref_ sd_netlink_message *req = NULL, *reply = NULL;
|
||||
sd_netlink_message *link;
|
||||
int r;
|
||||
|
||||
assert(m);
|
||||
assert(m->rtnl);
|
||||
|
||||
r = sd_rtnl_message_new_link(m->rtnl, &req, RTM_GETLINK, 0);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
r = sd_netlink_message_request_dump(req, true);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
r = sd_netlink_call(m->rtnl, req, 0, &reply);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
for (link = reply; link; link = sd_netlink_message_next(link)) {
|
||||
int k;
|
||||
|
||||
m->enumerating = true;
|
||||
|
||||
k = manager_rtnl_process_link(m->rtnl, link, m);
|
||||
if (k < 0)
|
||||
r = k;
|
||||
|
||||
m->enumerating = false;
|
||||
}
|
||||
|
||||
return r;
|
||||
}
|
||||
|
||||
int manager_rtnl_enumerate_addresses(Manager *m) {
|
||||
_cleanup_netlink_message_unref_ sd_netlink_message *req = NULL, *reply = NULL;
|
||||
sd_netlink_message *addr;
|
||||
int r;
|
||||
|
||||
assert(m);
|
||||
assert(m->rtnl);
|
||||
|
||||
r = sd_rtnl_message_new_addr(m->rtnl, &req, RTM_GETADDR, 0, 0);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
r = sd_netlink_message_request_dump(req, true);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
r = sd_netlink_call(m->rtnl, req, 0, &reply);
|
||||
if (r < 0)
|
||||
return r;
|
||||
|
||||
for (addr = reply; addr; addr = sd_netlink_message_next(addr)) {
|
||||
int k;
|
||||
|
||||
m->enumerating = true;
|
||||
|
||||
k = manager_rtnl_process_address(m->rtnl, addr, m);
|
||||
if (k < 0)
|
||||
r = k;
|
||||
|
||||
m->enumerating = false;
|
||||
}
|
||||
|
||||
return r;
|
||||
}
|
||||
|
||||
int manager_address_pool_acquire(Manager *m, int family, unsigned prefixlen, union in_addr_union *found) {
|
||||
AddressPool *p;
|
||||
int r;
|
||||
|
@ -1036,3 +1073,10 @@ Link* manager_find_uplink(Manager *m, Link *exclude) {
|
|||
|
||||
return NULL;
|
||||
}
|
||||
|
||||
void manager_dirty(Manager *manager) {
|
||||
assert(manager);
|
||||
|
||||
/* the serialized state in /run is no longer up-to-date */
|
||||
manager->dirty = true;
|
||||
}
|
||||
|
|
|
@ -369,10 +369,9 @@ int network_apply(Manager *manager, Network *network, Link *link) {
|
|||
route->protocol = RTPROT_STATIC;
|
||||
}
|
||||
|
||||
if (network->dns || network->ntp) {
|
||||
r = link_save(link);
|
||||
if (r < 0)
|
||||
return r;
|
||||
if (network->dns || network->ntp || network->domains) {
|
||||
manager_dirty(manager);
|
||||
link_dirty(link);
|
||||
}
|
||||
|
||||
return 0;
|
||||
|
|
|
@ -48,7 +48,10 @@ struct Manager {
|
|||
struct udev_monitor *udev_monitor;
|
||||
sd_event_source *udev_event_source;
|
||||
|
||||
bool enumerating;
|
||||
bool enumerating:1;
|
||||
bool dirty:1;
|
||||
|
||||
Set *dirty_links;
|
||||
|
||||
char *state_file;
|
||||
LinkOperationalState operational_state;
|
||||
|
@ -83,7 +86,7 @@ int manager_rtnl_enumerate_addresses(Manager *m);
|
|||
int manager_rtnl_process_address(sd_netlink *nl, sd_netlink_message *message, void *userdata);
|
||||
|
||||
int manager_send_changed(Manager *m, const char *property, ...) _sentinel_;
|
||||
int manager_save(Manager *m);
|
||||
void manager_dirty(Manager *m);
|
||||
|
||||
int manager_address_pool_acquire(Manager *m, int family, unsigned prefixlen, union in_addr_union *found);
|
||||
|
||||
|
|
Loading…
Reference in New Issue