Blame src/core/dhcp/nm-dhcp-systemd.c

Packit Service 5ffa24
/* SPDX-License-Identifier: LGPL-2.1-or-later */
Packit Service 5ffa24
/*
Packit Service 5ffa24
 * Copyright (C) 2014 Red Hat, Inc.
Packit Service 5ffa24
 */
Packit Service 5ffa24
Packit Service 5ffa24
#include "nm-default.h"
Packit Service 5ffa24
Packit Service 5ffa24
#include <stdlib.h>
Packit Service 5ffa24
#include <unistd.h>
Packit Service 5ffa24
#include <stdio.h>
Packit Service 5ffa24
#include <netinet/in.h>
Packit Service 5ffa24
#include <arpa/inet.h>
Packit Service 5ffa24
#include <ctype.h>
Packit Service 5ffa24
#include <net/if_arp.h>
Packit Service 5ffa24
Packit Service 5ffa24
#include "nm-glib-aux/nm-dedup-multi.h"
Packit Service 5ffa24
#include "nm-std-aux/unaligned.h"
Packit Service 5ffa24
Packit Service 5ffa24
#include "nm-utils.h"
Packit Service 5ffa24
#include "nm-dhcp-utils.h"
Packit Service 5ffa24
#include "nm-dhcp-options.h"
Packit Service 5ffa24
#include "nm-core-utils.h"
Packit Service 5ffa24
#include "NetworkManagerUtils.h"
Packit Service 5ffa24
#include "platform/nm-platform.h"
Packit Service 5ffa24
#include "nm-dhcp-client-logging.h"
Packit Service 5ffa24
#include "systemd/nm-sd.h"
Packit Service 5ffa24
#include "systemd/nm-sd-utils-dhcp.h"
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
#define NM_TYPE_DHCP_SYSTEMD (nm_dhcp_systemd_get_type())
Packit Service 5ffa24
#define NM_DHCP_SYSTEMD(obj) \
Packit Service 5ffa24
    (G_TYPE_CHECK_INSTANCE_CAST((obj), NM_TYPE_DHCP_SYSTEMD, NMDhcpSystemd))
Packit Service 5ffa24
#define NM_DHCP_SYSTEMD_CLASS(klass) \
Packit Service 5ffa24
    (G_TYPE_CHECK_CLASS_CAST((klass), NM_TYPE_DHCP_SYSTEMD, NMDhcpSystemdClass))
Packit Service 5ffa24
#define NM_IS_DHCP_SYSTEMD(obj)         (G_TYPE_CHECK_INSTANCE_TYPE((obj), NM_TYPE_DHCP_SYSTEMD))
Packit Service 5ffa24
#define NM_IS_DHCP_SYSTEMD_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE((klass), NM_TYPE_DHCP_SYSTEMD))
Packit Service 5ffa24
#define NM_DHCP_SYSTEMD_GET_CLASS(obj) \
Packit Service 5ffa24
    (G_TYPE_INSTANCE_GET_CLASS((obj), NM_TYPE_DHCP_SYSTEMD, NMDhcpSystemdClass))
Packit Service 5ffa24
Packit Service 5ffa24
typedef struct _NMDhcpSystemd      NMDhcpSystemd;
Packit Service 5ffa24
typedef struct _NMDhcpSystemdClass NMDhcpSystemdClass;
Packit Service 5ffa24
Packit Service 5ffa24
static GType nm_dhcp_systemd_get_type(void);
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
typedef struct {
Packit Service 5ffa24
    sd_dhcp_client * client4;
Packit Service 5ffa24
    sd_dhcp6_client *client6;
Packit Service 5ffa24
    char *           lease_file;
Packit Service 5ffa24
Packit Service 5ffa24
    guint request_count;
Packit Service 5ffa24
Packit Service 5ffa24
    bool privacy : 1;
Packit Service 5ffa24
} NMDhcpSystemdPrivate;
Packit Service 5ffa24
Packit Service 5ffa24
struct _NMDhcpSystemd {
Packit Service 5ffa24
    NMDhcpClient         parent;
Packit Service 5ffa24
    NMDhcpSystemdPrivate _priv;
Packit Service 5ffa24
};
Packit Service 5ffa24
Packit Service 5ffa24
struct _NMDhcpSystemdClass {
Packit Service 5ffa24
    NMDhcpClientClass parent;
Packit Service 5ffa24
};
Packit Service 5ffa24
Packit Service 5ffa24
G_DEFINE_TYPE(NMDhcpSystemd, nm_dhcp_systemd, NM_TYPE_DHCP_CLIENT)
Packit Service 5ffa24
Packit Service 5ffa24
#define NM_DHCP_SYSTEMD_GET_PRIVATE(self) _NM_GET_PRIVATE(self, NMDhcpSystemd, NM_IS_DHCP_SYSTEMD)
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
static NMIP4Config *
Packit Service 5ffa24
lease_to_ip4_config(NMDedupMultiIndex *multi_idx,
Packit Service 5ffa24
                    const char *       iface,
Packit Service 5ffa24
                    int                ifindex,
Packit Service 5ffa24
                    sd_dhcp_lease *    lease,
Packit Service 5ffa24
                    guint32            route_table,
Packit Service 5ffa24
                    guint32            route_metric,
Packit Service 5ffa24
                    GHashTable **      out_options,
Packit Service 5ffa24
                    GError **          error)
Packit Service 5ffa24
{
Packit Service 5ffa24
    gs_unref_object NMIP4Config *ip4_config = NULL;
Packit Service 5ffa24
    gs_unref_hashtable GHashTable *options  = NULL;
Packit Service 5ffa24
    const struct in_addr *         addr_list;
Packit Service 5ffa24
    char                           addr_str[NM_UTILS_INET_ADDRSTRLEN];
Packit Service 5ffa24
    const char *                   s;
Packit Service 5ffa24
    nm_auto_free_gstring GString *str      = NULL;
Packit Service 5ffa24
    gs_free sd_dhcp_route **routes         = NULL;
Packit Service 5ffa24
    const char *const *     search_domains = NULL;
Packit Service 5ffa24
    guint16                 mtu;
Packit Service 5ffa24
    int                     i, num;
Packit Service 5ffa24
    const void *            data;
Packit Service 5ffa24
    gsize                   data_len;
Packit Service 5ffa24
    gboolean                metered                   = FALSE;
Packit Service 5ffa24
    gboolean                has_router_from_classless = FALSE;
Packit Service 5ffa24
    gboolean                has_classless_route       = FALSE;
Packit Service 5ffa24
    gboolean                has_static_route          = FALSE;
Packit Service 5ffa24
    const gint32            ts                        = nm_utils_get_monotonic_timestamp_sec();
Packit Service 5ffa24
    gint64                  ts_time                   = time(NULL);
Packit Service 5ffa24
    struct in_addr          a_address;
Packit Service 5ffa24
    struct in_addr          a_netmask;
Packit Service 5ffa24
    struct in_addr          a_next_server;
Packit Service 5ffa24
    struct in_addr          server_id;
Packit Service 5ffa24
    struct in_addr          broadcast;
Packit Service 5ffa24
    const struct in_addr *  a_router;
Packit Service 5ffa24
    guint32                 a_plen;
Packit Service 5ffa24
    guint32                 a_lifetime;
Packit Service 5ffa24
    guint32                 renewal;
Packit Service 5ffa24
    guint32                 rebinding;
Packit Service 5ffa24
    gs_free nm_sd_dhcp_option *private_options = NULL;
Packit Service 5ffa24
Packit Service 5ffa24
    nm_assert(lease != NULL);
Packit Service 5ffa24
Packit Service 5ffa24
    if (sd_dhcp_lease_get_address(lease, &a_address) < 0) {
Packit Service 5ffa24
        nm_utils_error_set_literal(error,
Packit Service 5ffa24
                                   NM_UTILS_ERROR_UNKNOWN,
Packit Service 5ffa24
                                   "could not get address from lease");
Packit Service 5ffa24
        return NULL;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (sd_dhcp_lease_get_netmask(lease, &a_netmask) < 0) {
Packit Service 5ffa24
        nm_utils_error_set_literal(error,
Packit Service 5ffa24
                                   NM_UTILS_ERROR_UNKNOWN,
Packit Service 5ffa24
                                   "could not get netmask from lease");
Packit Service 5ffa24
        return NULL;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (sd_dhcp_lease_get_lifetime(lease, &a_lifetime) < 0) {
Packit Service 5ffa24
        nm_utils_error_set_literal(error,
Packit Service 5ffa24
                                   NM_UTILS_ERROR_UNKNOWN,
Packit Service 5ffa24
                                   "could not get lifetime from lease");
Packit Service 5ffa24
        return NULL;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    ip4_config = nm_ip4_config_new(multi_idx, ifindex);
Packit Service 5ffa24
Packit Service 5ffa24
    options = out_options ? nm_dhcp_option_create_options_dict() : NULL;
Packit Service 5ffa24
Packit Service 5ffa24
    _nm_utils_inet4_ntop(a_address.s_addr, addr_str);
Packit Service 5ffa24
    nm_dhcp_option_add_option(options,
Packit Service 5ffa24
                              _nm_dhcp_option_dhcp4_options,
Packit Service 5ffa24
                              NM_DHCP_OPTION_DHCP4_NM_IP_ADDRESS,
Packit Service 5ffa24
                              addr_str);
Packit Service 5ffa24
Packit Service 5ffa24
    a_plen = nm_utils_ip4_netmask_to_prefix(a_netmask.s_addr);
Packit Service 5ffa24
    nm_dhcp_option_add_option(options,
Packit Service 5ffa24
                              _nm_dhcp_option_dhcp4_options,
Packit Service 5ffa24
                              NM_DHCP_OPTION_DHCP4_SUBNET_MASK,
Packit Service 5ffa24
                              _nm_utils_inet4_ntop(a_netmask.s_addr, addr_str));
Packit Service 5ffa24
Packit Service 5ffa24
    nm_dhcp_option_add_option_u64(options,
Packit Service 5ffa24
                                  _nm_dhcp_option_dhcp4_options,
Packit Service 5ffa24
                                  NM_DHCP_OPTION_DHCP4_IP_ADDRESS_LEASE_TIME,
Packit Service 5ffa24
                                  a_lifetime);
Packit Service 5ffa24
    nm_dhcp_option_add_option_u64(options,
Packit Service 5ffa24
                                  _nm_dhcp_option_dhcp4_options,
Packit Service 5ffa24
                                  NM_DHCP_OPTION_DHCP4_NM_EXPIRY,
Packit Service 5ffa24
                                  (guint64)(ts_time + a_lifetime));
Packit Service 5ffa24
Packit Service 5ffa24
    if (sd_dhcp_lease_get_next_server(lease, &a_next_server) == 0) {
Packit Service 5ffa24
        _nm_utils_inet4_ntop(a_next_server.s_addr, addr_str);
Packit Service 5ffa24
        nm_dhcp_option_add_option(options,
Packit Service 5ffa24
                                  _nm_dhcp_option_dhcp4_options,
Packit Service 5ffa24
                                  NM_DHCP_OPTION_DHCP4_NM_NEXT_SERVER,
Packit Service 5ffa24
                                  addr_str);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    nm_ip4_config_add_address(ip4_config,
Packit Service 5ffa24
                              &((const NMPlatformIP4Address){
Packit Service 5ffa24
                                  .address      = a_address.s_addr,
Packit Service 5ffa24
                                  .peer_address = a_address.s_addr,
Packit Service 5ffa24
                                  .plen         = a_plen,
Packit Service 5ffa24
                                  .addr_source  = NM_IP_CONFIG_SOURCE_DHCP,
Packit Service 5ffa24
                                  .timestamp    = ts,
Packit Service 5ffa24
                                  .lifetime     = a_lifetime,
Packit Service 5ffa24
                                  .preferred    = a_lifetime,
Packit Service 5ffa24
                              }));
Packit Service 5ffa24
Packit Service 5ffa24
    if (sd_dhcp_lease_get_server_identifier(lease, &server_id) >= 0) {
Packit Service 5ffa24
        _nm_utils_inet4_ntop(server_id.s_addr, addr_str);
Packit Service 5ffa24
        nm_dhcp_option_add_option(options,
Packit Service 5ffa24
                                  _nm_dhcp_option_dhcp4_options,
Packit Service 5ffa24
                                  NM_DHCP_OPTION_DHCP4_SERVER_ID,
Packit Service 5ffa24
                                  addr_str);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (sd_dhcp_lease_get_broadcast(lease, &broadcast) >= 0) {
Packit Service 5ffa24
        _nm_utils_inet4_ntop(broadcast.s_addr, addr_str);
Packit Service 5ffa24
        nm_dhcp_option_add_option(options,
Packit Service 5ffa24
                                  _nm_dhcp_option_dhcp4_options,
Packit Service 5ffa24
                                  NM_DHCP_OPTION_DHCP4_BROADCAST,
Packit Service 5ffa24
                                  addr_str);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    num = sd_dhcp_lease_get_dns(lease, &addr_list);
Packit Service 5ffa24
    if (num > 0) {
Packit Service 5ffa24
        nm_gstring_prepare(&str);
Packit Service 5ffa24
        for (i = 0; i < num; i++) {
Packit Service 5ffa24
            _nm_utils_inet4_ntop(addr_list[i].s_addr, addr_str);
Packit Service 5ffa24
            g_string_append(nm_gstring_add_space_delimiter(str), addr_str);
Packit Service 5ffa24
Packit Service 5ffa24
            if (addr_list[i].s_addr == 0 || nm_ip4_addr_is_localhost(addr_list[i].s_addr)) {
Packit Service 5ffa24
                /* Skip localhost addresses, like also networkd does.
Packit Service 5ffa24
                 * See https://github.com/systemd/systemd/issues/4524. */
Packit Service 5ffa24
                continue;
Packit Service 5ffa24
            }
Packit Service 5ffa24
            nm_ip4_config_add_nameserver(ip4_config, addr_list[i].s_addr);
Packit Service 5ffa24
        }
Packit Service 5ffa24
        nm_dhcp_option_add_option(options,
Packit Service 5ffa24
                                  _nm_dhcp_option_dhcp4_options,
Packit Service 5ffa24
                                  NM_DHCP_OPTION_DHCP4_DOMAIN_NAME_SERVER,
Packit Service 5ffa24
                                  str->str);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    num = sd_dhcp_lease_get_search_domains(lease, (char ***) &search_domains);
Packit Service 5ffa24
    if (num > 0) {
Packit Service 5ffa24
        nm_gstring_prepare(&str);
Packit Service 5ffa24
        for (i = 0; i < num; i++) {
Packit Service 5ffa24
            g_string_append(nm_gstring_add_space_delimiter(str), search_domains[i]);
Packit Service 5ffa24
            nm_ip4_config_add_search(ip4_config, search_domains[i]);
Packit Service 5ffa24
        }
Packit Service 5ffa24
        nm_dhcp_option_add_option(options,
Packit Service 5ffa24
                                  _nm_dhcp_option_dhcp4_options,
Packit Service 5ffa24
                                  NM_DHCP_OPTION_DHCP4_DOMAIN_SEARCH_LIST,
Packit Service 5ffa24
                                  str->str);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (sd_dhcp_lease_get_domainname(lease, &s) >= 0) {
Packit Service 5ffa24
        gs_strfreev char **domains = NULL;
Packit Service 5ffa24
        char **            d;
Packit Service 5ffa24
Packit Service 5ffa24
        nm_dhcp_option_add_option(options,
Packit Service 5ffa24
                                  _nm_dhcp_option_dhcp4_options,
Packit Service 5ffa24
                                  NM_DHCP_OPTION_DHCP4_DOMAIN_NAME,
Packit Service 5ffa24
                                  s);
Packit Service 5ffa24
Packit Service 5ffa24
        /* Multiple domains sometimes stuffed into option 15 "Domain Name".
Packit Service 5ffa24
         * As systemd escapes such characters, split them at \\032. */
Packit Service 5ffa24
        domains = g_strsplit(s, "\\032", 0);
Packit Service 5ffa24
        for (d = domains; *d; d++)
Packit Service 5ffa24
            nm_ip4_config_add_domain(ip4_config, *d);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (sd_dhcp_lease_get_hostname(lease, &s) >= 0) {
Packit Service 5ffa24
        nm_dhcp_option_add_option(options,
Packit Service 5ffa24
                                  _nm_dhcp_option_dhcp4_options,
Packit Service 5ffa24
                                  NM_DHCP_OPTION_DHCP4_HOST_NAME,
Packit Service 5ffa24
                                  s);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    num = sd_dhcp_lease_get_routes(lease, &routes);
Packit Service 5ffa24
    if (num > 0) {
Packit Service 5ffa24
        nm_auto_free_gstring GString *str_classless        = NULL;
Packit Service 5ffa24
        nm_auto_free_gstring GString *str_static           = NULL;
Packit Service 5ffa24
        guint32                       default_route_metric = route_metric;
Packit Service 5ffa24
Packit Service 5ffa24
        for (i = 0; i < num; i++) {
Packit Service 5ffa24
            switch (sd_dhcp_route_get_option(routes[i])) {
Packit Service 5ffa24
            case NM_DHCP_OPTION_DHCP4_CLASSLESS_STATIC_ROUTE:
Packit Service 5ffa24
                has_classless_route = TRUE;
Packit Service 5ffa24
                break;
Packit Service 5ffa24
            case NM_DHCP_OPTION_DHCP4_STATIC_ROUTE:
Packit Service 5ffa24
                has_static_route = TRUE;
Packit Service 5ffa24
                break;
Packit Service 5ffa24
            }
Packit Service 5ffa24
        }
Packit Service 5ffa24
Packit Service 5ffa24
        if (has_classless_route)
Packit Service 5ffa24
            str_classless = g_string_sized_new(30);
Packit Service 5ffa24
        if (has_static_route)
Packit Service 5ffa24
            str_static = g_string_sized_new(30);
Packit Service 5ffa24
Packit Service 5ffa24
        for (i = 0; i < num; i++) {
Packit Service 5ffa24
            char           network_net_str[NM_UTILS_INET_ADDRSTRLEN];
Packit Service 5ffa24
            char           gateway_str[NM_UTILS_INET_ADDRSTRLEN];
Packit Service 5ffa24
            guint8         r_plen;
Packit Service 5ffa24
            struct in_addr r_network;
Packit Service 5ffa24
            struct in_addr r_gateway;
Packit Service 5ffa24
            in_addr_t      network_net;
Packit Service 5ffa24
            int            option;
Packit Service 5ffa24
            guint32        m;
Packit Service 5ffa24
Packit Service 5ffa24
            option = sd_dhcp_route_get_option(routes[i]);
Packit Service 5ffa24
            if (!NM_IN_SET(option,
Packit Service 5ffa24
                           NM_DHCP_OPTION_DHCP4_CLASSLESS_STATIC_ROUTE,
Packit Service 5ffa24
                           NM_DHCP_OPTION_DHCP4_STATIC_ROUTE))
Packit Service 5ffa24
                continue;
Packit Service 5ffa24
Packit Service 5ffa24
            if (sd_dhcp_route_get_destination(routes[i], &r_network) < 0)
Packit Service 5ffa24
                continue;
Packit Service 5ffa24
            if (sd_dhcp_route_get_destination_prefix_length(routes[i], &r_plen) < 0 || r_plen > 32)
Packit Service 5ffa24
                continue;
Packit Service 5ffa24
            if (sd_dhcp_route_get_gateway(routes[i], &r_gateway) < 0)
Packit Service 5ffa24
                continue;
Packit Service 5ffa24
Packit Service 5ffa24
            network_net = nm_utils_ip4_address_clear_host_address(r_network.s_addr, r_plen);
Packit Service 5ffa24
            _nm_utils_inet4_ntop(network_net, network_net_str);
Packit Service 5ffa24
            _nm_utils_inet4_ntop(r_gateway.s_addr, gateway_str);
Packit Service 5ffa24
Packit Service 5ffa24
            g_string_append_printf(
Packit Service 5ffa24
                nm_gstring_add_space_delimiter(option == NM_DHCP_OPTION_DHCP4_CLASSLESS_STATIC_ROUTE
Packit Service 5ffa24
                                                   ? str_classless
Packit Service 5ffa24
                                                   : str_static),
Packit Service 5ffa24
                "%s/%d %s",
Packit Service 5ffa24
                network_net_str,
Packit Service 5ffa24
                (int) r_plen,
Packit Service 5ffa24
                gateway_str);
Packit Service 5ffa24
Packit Service 5ffa24
            if (option == NM_DHCP_OPTION_DHCP4_STATIC_ROUTE && has_classless_route) {
Packit Service 5ffa24
                /* RFC 3443: if the DHCP server returns both a Classless Static Routes
Packit Service 5ffa24
                 * option and a Static Routes option, the DHCP client MUST ignore the
Packit Service 5ffa24
                 * Static Routes option. */
Packit Service 5ffa24
                continue;
Packit Service 5ffa24
            }
Packit Service 5ffa24
Packit Service 5ffa24
            if (r_plen == 0 && option == NM_DHCP_OPTION_DHCP4_STATIC_ROUTE) {
Packit Service 5ffa24
                /* for option 33 (static route), RFC 2132 says:
Packit Service 5ffa24
                 *
Packit Service 5ffa24
                 * The default route (0.0.0.0) is an illegal destination for a static
Packit Service 5ffa24
                 * route. */
Packit Service 5ffa24
                continue;
Packit Service 5ffa24
            }
Packit Service 5ffa24
Packit Service 5ffa24
            if (r_plen == 0) {
Packit Service 5ffa24
                /* if there are multiple default routes, we add them with differing
Packit Service 5ffa24
                 * metrics. */
Packit Service 5ffa24
                m = default_route_metric;
Packit Service 5ffa24
                if (default_route_metric < G_MAXUINT32)
Packit Service 5ffa24
                    default_route_metric++;
Packit Service 5ffa24
Packit Service 5ffa24
                has_router_from_classless = TRUE;
Packit Service 5ffa24
            } else
Packit Service 5ffa24
                m = route_metric;
Packit Service 5ffa24
Packit Service 5ffa24
            nm_ip4_config_add_route(
Packit Service 5ffa24
                ip4_config,
Packit Service 5ffa24
                &((const NMPlatformIP4Route){
Packit Service 5ffa24
                    .network       = network_net,
Packit Service 5ffa24
                    .plen          = r_plen,
Packit Service 5ffa24
                    .gateway       = r_gateway.s_addr,
Packit Service 5ffa24
                    .rt_source     = NM_IP_CONFIG_SOURCE_DHCP,
Packit Service 5ffa24
                    .metric        = m,
Packit Service 5ffa24
                    .table_coerced = nm_platform_route_table_coerce(route_table),
Packit Service 5ffa24
                }),
Packit Service 5ffa24
                NULL);
Packit Service 5ffa24
        }
Packit Service 5ffa24
Packit Service 5ffa24
        if (str_classless && str_classless->len > 0)
Packit Service 5ffa24
            nm_dhcp_option_add_option(options,
Packit Service 5ffa24
                                      _nm_dhcp_option_dhcp4_options,
Packit Service 5ffa24
                                      NM_DHCP_OPTION_DHCP4_CLASSLESS_STATIC_ROUTE,
Packit Service 5ffa24
                                      str_classless->str);
Packit Service 5ffa24
        if (str_static && str_static->len > 0)
Packit Service 5ffa24
            nm_dhcp_option_add_option(options,
Packit Service 5ffa24
                                      _nm_dhcp_option_dhcp4_options,
Packit Service 5ffa24
                                      NM_DHCP_OPTION_DHCP4_STATIC_ROUTE,
Packit Service 5ffa24
                                      str_static->str);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    num = sd_dhcp_lease_get_router(lease, &a_router);
Packit Service 5ffa24
    if (num > 0) {
Packit Service 5ffa24
        guint32 default_route_metric = route_metric;
Packit Service 5ffa24
Packit Service 5ffa24
        nm_gstring_prepare(&str);
Packit Service 5ffa24
        for (i = 0; i < num; i++) {
Packit Service 5ffa24
            guint32 m;
Packit Service 5ffa24
Packit Service 5ffa24
            s = _nm_utils_inet4_ntop(a_router[i].s_addr, addr_str);
Packit Service 5ffa24
            g_string_append(nm_gstring_add_space_delimiter(str), s);
Packit Service 5ffa24
Packit Service 5ffa24
            if (a_router[i].s_addr == 0) {
Packit Service 5ffa24
                /* silently skip 0.0.0.0 */
Packit Service 5ffa24
                continue;
Packit Service 5ffa24
            }
Packit Service 5ffa24
Packit Service 5ffa24
            if (has_router_from_classless) {
Packit Service 5ffa24
                /* If the DHCP server returns both a Classless Static Routes option and a
Packit Service 5ffa24
                 * Router option, the DHCP client MUST ignore the Router option [RFC 3442].
Packit Service 5ffa24
                 *
Packit Service 5ffa24
                 * Be more lenient and ignore the Router option only if Classless Static
Packit Service 5ffa24
                 * Routes contain a default gateway (as other DHCP backends do).
Packit Service 5ffa24
                 */
Packit Service 5ffa24
                continue;
Packit Service 5ffa24
            }
Packit Service 5ffa24
Packit Service 5ffa24
            /* if there are multiple default routes, we add them with differing
Packit Service 5ffa24
             * metrics. */
Packit Service 5ffa24
            m = default_route_metric;
Packit Service 5ffa24
            if (default_route_metric < G_MAXUINT32)
Packit Service 5ffa24
                default_route_metric++;
Packit Service 5ffa24
Packit Service 5ffa24
            nm_ip4_config_add_route(
Packit Service 5ffa24
                ip4_config,
Packit Service 5ffa24
                &((const NMPlatformIP4Route){
Packit Service 5ffa24
                    .rt_source     = NM_IP_CONFIG_SOURCE_DHCP,
Packit Service 5ffa24
                    .gateway       = a_router[i].s_addr,
Packit Service 5ffa24
                    .table_coerced = nm_platform_route_table_coerce(route_table),
Packit Service 5ffa24
                    .metric        = m,
Packit Service 5ffa24
                }),
Packit Service 5ffa24
                NULL);
Packit Service 5ffa24
        }
Packit Service 5ffa24
        nm_dhcp_option_add_option(options,
Packit Service 5ffa24
                                  _nm_dhcp_option_dhcp4_options,
Packit Service 5ffa24
                                  NM_DHCP_OPTION_DHCP4_ROUTER,
Packit Service 5ffa24
                                  str->str);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (sd_dhcp_lease_get_mtu(lease, &mtu) >= 0 && mtu) {
Packit Service 5ffa24
        nm_dhcp_option_add_option_u64(options,
Packit Service 5ffa24
                                      _nm_dhcp_option_dhcp4_options,
Packit Service 5ffa24
                                      NM_DHCP_OPTION_DHCP4_INTERFACE_MTU,
Packit Service 5ffa24
                                      mtu);
Packit Service 5ffa24
        nm_ip4_config_set_mtu(ip4_config, mtu, NM_IP_CONFIG_SOURCE_DHCP);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    num = sd_dhcp_lease_get_ntp(lease, &addr_list);
Packit Service 5ffa24
    if (num > 0) {
Packit Service 5ffa24
        nm_gstring_prepare(&str);
Packit Service 5ffa24
        for (i = 0; i < num; i++) {
Packit Service 5ffa24
            _nm_utils_inet4_ntop(addr_list[i].s_addr, addr_str);
Packit Service 5ffa24
            g_string_append(nm_gstring_add_space_delimiter(str), addr_str);
Packit Service 5ffa24
        }
Packit Service 5ffa24
        nm_dhcp_option_add_option(options,
Packit Service 5ffa24
                                  _nm_dhcp_option_dhcp4_options,
Packit Service 5ffa24
                                  NM_DHCP_OPTION_DHCP4_NTP_SERVER,
Packit Service 5ffa24
                                  str->str);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (sd_dhcp_lease_get_root_path(lease, &s) >= 0) {
Packit Service 5ffa24
        nm_dhcp_option_add_option(options,
Packit Service 5ffa24
                                  _nm_dhcp_option_dhcp4_options,
Packit Service 5ffa24
                                  NM_DHCP_OPTION_DHCP4_ROOT_PATH,
Packit Service 5ffa24
                                  s);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (sd_dhcp_lease_get_t1(lease, &renewal) >= 0) {
Packit Service 5ffa24
        nm_dhcp_option_add_option_u64(options,
Packit Service 5ffa24
                                      _nm_dhcp_option_dhcp4_options,
Packit Service 5ffa24
                                      NM_DHCP_OPTION_DHCP4_RENEWAL_T1_TIME,
Packit Service 5ffa24
                                      renewal);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (sd_dhcp_lease_get_t2(lease, &rebinding) >= 0) {
Packit Service 5ffa24
        nm_dhcp_option_add_option_u64(options,
Packit Service 5ffa24
                                      _nm_dhcp_option_dhcp4_options,
Packit Service 5ffa24
                                      NM_DHCP_OPTION_DHCP4_REBINDING_T2_TIME,
Packit Service 5ffa24
                                      rebinding);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (sd_dhcp_lease_get_timezone(lease, &s) >= 0) {
Packit Service 5ffa24
        nm_dhcp_option_add_option(options,
Packit Service 5ffa24
                                  _nm_dhcp_option_dhcp4_options,
Packit Service 5ffa24
                                  NM_DHCP_OPTION_DHCP4_NEW_TZDB_TIMEZONE,
Packit Service 5ffa24
                                  s);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (sd_dhcp_lease_get_vendor_specific(lease, &data, &data_len) >= 0)
Packit Service 5ffa24
        metered = !!memmem(data, data_len, "ANDROID_METERED", NM_STRLEN("ANDROID_METERED"));
Packit Service 5ffa24
    nm_ip4_config_set_metered(ip4_config, metered);
Packit Service 5ffa24
Packit Service 5ffa24
    num = nm_sd_dhcp_lease_get_private_options(lease, &private_options);
Packit Service 5ffa24
    if (num > 0) {
Packit Service 5ffa24
        for (i = 0; i < num; i++) {
Packit Service 5ffa24
            char *option_string;
Packit Service 5ffa24
Packit Service 5ffa24
            option_string = nm_utils_bin2hexstr_full(private_options[i].data,
Packit Service 5ffa24
                                                     private_options[i].data_len,
Packit Service 5ffa24
                                                     ':',
Packit Service 5ffa24
                                                     FALSE,
Packit Service 5ffa24
                                                     NULL);
Packit Service 5ffa24
            if (!options) {
Packit Service 5ffa24
                g_free(option_string);
Packit Service 5ffa24
                continue;
Packit Service 5ffa24
            }
Packit Service 5ffa24
            nm_dhcp_option_take_option(options,
Packit Service 5ffa24
                                       _nm_dhcp_option_dhcp4_options,
Packit Service 5ffa24
                                       private_options[i].code,
Packit Service 5ffa24
                                       option_string);
Packit Service 5ffa24
        }
Packit Service 5ffa24
    }
Packit Service 5ffa24
    NM_SET_OUT(out_options, g_steal_pointer(&options));
Packit Service 5ffa24
    return g_steal_pointer(&ip4_config);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
bound4_handle(NMDhcpSystemd *self, gboolean extended)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMDhcpSystemdPrivate *priv              = NM_DHCP_SYSTEMD_GET_PRIVATE(self);
Packit Service 5ffa24
    const char *          iface             = nm_dhcp_client_get_iface(NM_DHCP_CLIENT(self));
Packit Service 5ffa24
    gs_unref_object NMIP4Config *ip4_config = NULL;
Packit Service 5ffa24
    gs_unref_hashtable GHashTable *options  = NULL;
Packit Service 5ffa24
    sd_dhcp_lease *                lease    = NULL;
Packit Service 5ffa24
    GError *                       error    = NULL;
Packit Service 5ffa24
Packit Service 5ffa24
    if (sd_dhcp_client_get_lease(priv->client4, &lease) < 0 || !lease) {
Packit Service 5ffa24
        _LOGW("no lease!");
Packit Service 5ffa24
        nm_dhcp_client_set_state(NM_DHCP_CLIENT(self), NM_DHCP_STATE_FAIL, NULL, NULL);
Packit Service 5ffa24
        return;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    _LOGD("lease available");
Packit Service 5ffa24
Packit Service 5ffa24
    ip4_config = lease_to_ip4_config(nm_dhcp_client_get_multi_idx(NM_DHCP_CLIENT(self)),
Packit Service 5ffa24
                                     iface,
Packit Service 5ffa24
                                     nm_dhcp_client_get_ifindex(NM_DHCP_CLIENT(self)),
Packit Service 5ffa24
                                     lease,
Packit Service 5ffa24
                                     nm_dhcp_client_get_route_table(NM_DHCP_CLIENT(self)),
Packit Service 5ffa24
                                     nm_dhcp_client_get_route_metric(NM_DHCP_CLIENT(self)),
Packit Service 5ffa24
                                     &options,
Packit Service 5ffa24
                                     &error);
Packit Service 5ffa24
    if (!ip4_config) {
Packit Service 5ffa24
        _LOGW("%s", error->message);
Packit Service 5ffa24
        g_clear_error(&error);
Packit Service 5ffa24
        nm_dhcp_client_set_state(NM_DHCP_CLIENT(self), NM_DHCP_STATE_FAIL, NULL, NULL);
Packit Service 5ffa24
        return;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    nm_dhcp_option_add_requests_to_options(options, _nm_dhcp_option_dhcp4_options);
Packit Service 5ffa24
    dhcp_lease_save(lease, priv->lease_file);
Packit Service 5ffa24
Packit Service 5ffa24
    nm_dhcp_client_set_state(NM_DHCP_CLIENT(self),
Packit Service 5ffa24
                             extended ? NM_DHCP_STATE_EXTENDED : NM_DHCP_STATE_BOUND,
Packit Service 5ffa24
                             NM_IP_CONFIG_CAST(ip4_config),
Packit Service 5ffa24
                             options);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static int
Packit Service 5ffa24
dhcp_event_cb(sd_dhcp_client *client, int event, gpointer user_data)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMDhcpSystemd *       self = NM_DHCP_SYSTEMD(user_data);
Packit Service 5ffa24
    NMDhcpSystemdPrivate *priv = NM_DHCP_SYSTEMD_GET_PRIVATE(self);
Packit Service 5ffa24
    char                  addr_str[INET_ADDRSTRLEN];
Packit Service 5ffa24
    sd_dhcp_lease *       lease = NULL;
Packit Service 5ffa24
    struct in_addr        addr;
Packit Service 5ffa24
    int                   r;
Packit Service 5ffa24
Packit Service 5ffa24
    nm_assert(priv->client4 == client);
Packit Service 5ffa24
Packit Service 5ffa24
    _LOGD("client event %d", event);
Packit Service 5ffa24
Packit Service 5ffa24
    switch (event) {
Packit Service 5ffa24
    case SD_DHCP_CLIENT_EVENT_EXPIRED:
Packit Service 5ffa24
        nm_dhcp_client_set_state(NM_DHCP_CLIENT(user_data), NM_DHCP_STATE_EXPIRE, NULL, NULL);
Packit Service 5ffa24
        break;
Packit Service 5ffa24
    case SD_DHCP_CLIENT_EVENT_STOP:
Packit Service 5ffa24
        nm_dhcp_client_set_state(NM_DHCP_CLIENT(user_data), NM_DHCP_STATE_FAIL, NULL, NULL);
Packit Service 5ffa24
        break;
Packit Service 5ffa24
    case SD_DHCP_CLIENT_EVENT_RENEW:
Packit Service 5ffa24
    case SD_DHCP_CLIENT_EVENT_IP_CHANGE:
Packit Service 5ffa24
        bound4_handle(self, TRUE);
Packit Service 5ffa24
        break;
Packit Service 5ffa24
    case SD_DHCP_CLIENT_EVENT_IP_ACQUIRE:
Packit Service 5ffa24
        bound4_handle(self, FALSE);
Packit Service 5ffa24
        break;
Packit Service 5ffa24
    case SD_DHCP_CLIENT_EVENT_SELECTING:
Packit Service 5ffa24
        r = sd_dhcp_client_get_lease(priv->client4, &lease);
Packit Service 5ffa24
        if (r < 0)
Packit Service 5ffa24
            return r;
Packit Service 5ffa24
        r = sd_dhcp_lease_get_server_identifier(lease, &addr);
Packit Service 5ffa24
        if (r < 0)
Packit Service 5ffa24
            return r;
Packit Service 5ffa24
        if (nm_dhcp_client_server_id_is_rejected(NM_DHCP_CLIENT(user_data), &addr)) {
Packit Service 5ffa24
            _LOGD("server-id %s is in the reject-list, ignoring",
Packit Service 5ffa24
                  nm_utils_inet_ntop(AF_INET, &addr, addr_str));
Packit Service 5ffa24
            return -ENOMSG;
Packit Service 5ffa24
        }
Packit Service 5ffa24
        break;
Packit Service 5ffa24
    case SD_DHCP_CLIENT_EVENT_TRANSIENT_FAILURE:
Packit Service 5ffa24
        break;
Packit Service 5ffa24
    default:
Packit Service 5ffa24
        _LOGW("unhandled DHCP event %d", event);
Packit Service 5ffa24
        break;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    return 0;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static gboolean
Packit Service 5ffa24
ip4_start(NMDhcpClient *client,
Packit Service 5ffa24
          const char *  dhcp_anycast_addr,
Packit Service 5ffa24
          const char *  last_ip4_address,
Packit Service 5ffa24
          GError **     error)
Packit Service 5ffa24
{
Packit Service 5ffa24
    nm_auto(sd_dhcp_client_unrefp) sd_dhcp_client *sd_client  = NULL;
Packit Service 5ffa24
    NMDhcpSystemd *                                self       = NM_DHCP_SYSTEMD(client);
Packit Service 5ffa24
    NMDhcpSystemdPrivate *                         priv       = NM_DHCP_SYSTEMD_GET_PRIVATE(self);
Packit Service 5ffa24
    gs_free char *                                 lease_file = NULL;
Packit Service 5ffa24
    GBytes *                                       hwaddr;
Packit Service 5ffa24
    const uint8_t *                                hwaddr_arr;
Packit Service 5ffa24
    gsize                                          hwaddr_len;
Packit Service 5ffa24
    int                                            arp_type;
Packit Service 5ffa24
    GBytes *                                       client_id;
Packit Service 5ffa24
    gs_unref_bytes GBytes *client_id_new = NULL;
Packit Service 5ffa24
    GBytes *               vendor_class_identifier;
Packit Service 5ffa24
    const uint8_t *        client_id_arr;
Packit Service 5ffa24
    size_t                 client_id_len;
Packit Service 5ffa24
    struct in_addr         last_addr = {0};
Packit Service 5ffa24
    const char *           hostname;
Packit Service 5ffa24
    const char *           mud_url;
Packit Service 5ffa24
    int                    r, i;
Packit Service 5ffa24
    GBytes *               bcast_hwaddr;
Packit Service 5ffa24
    const uint8_t *        bcast_hwaddr_arr;
Packit Service 5ffa24
    gsize                  bcast_hwaddr_len;
Packit Service 5ffa24
Packit Service 5ffa24
    g_return_val_if_fail(!priv->client4, FALSE);
Packit Service 5ffa24
    g_return_val_if_fail(!priv->client6, FALSE);
Packit Service 5ffa24
Packit Service 5ffa24
    r = sd_dhcp_client_new(&sd_client, FALSE);
Packit Service 5ffa24
    if (r < 0) {
Packit Service 5ffa24
        nm_utils_error_set_errno(error, r, "failed to create dhcp-client: %s");
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    _LOGT("dhcp-client4: set %p", sd_client);
Packit Service 5ffa24
Packit Service 5ffa24
    r = sd_dhcp_client_attach_event(sd_client, NULL, 0);
Packit Service 5ffa24
    if (r < 0) {
Packit Service 5ffa24
        nm_utils_error_set_errno(error, r, "failed to attach event: %s");
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    hwaddr = nm_dhcp_client_get_hw_addr(client);
Packit Service 5ffa24
    if (!hwaddr || !(hwaddr_arr = g_bytes_get_data(hwaddr, &hwaddr_len))
Packit Service 5ffa24
        || (arp_type = nm_utils_arp_type_detect_from_hwaddrlen(hwaddr_len)) < 0) {
Packit Service 5ffa24
        nm_utils_error_set_literal(error, NM_UTILS_ERROR_UNKNOWN, "invalid MAC address");
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    bcast_hwaddr_arr = NULL;
Packit Service 5ffa24
    if ((bcast_hwaddr = nm_dhcp_client_get_broadcast_hw_addr(NM_DHCP_CLIENT(self)))) {
Packit Service 5ffa24
        bcast_hwaddr_arr = g_bytes_get_data(bcast_hwaddr, &bcast_hwaddr_len);
Packit Service 5ffa24
        if (bcast_hwaddr_len != hwaddr_len)
Packit Service 5ffa24
            bcast_hwaddr_arr = NULL;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    r = sd_dhcp_client_set_mac(sd_client,
Packit Service 5ffa24
                               hwaddr_arr,
Packit Service 5ffa24
                               bcast_hwaddr_arr,
Packit Service 5ffa24
                               hwaddr_len,
Packit Service 5ffa24
                               (guint16) arp_type);
Packit Service 5ffa24
    if (r < 0) {
Packit Service 5ffa24
        nm_utils_error_set_errno(error, r, "failed to set MAC address: %s");
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    r = sd_dhcp_client_set_ifindex(sd_client, nm_dhcp_client_get_ifindex(client));
Packit Service 5ffa24
    if (r < 0) {
Packit Service 5ffa24
        nm_utils_error_set_errno(error, r, "failed to set ifindex: %s");
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    nm_dhcp_utils_get_leasefile_path(AF_INET,
Packit Service 5ffa24
                                     "internal",
Packit Service 5ffa24
                                     nm_dhcp_client_get_iface(client),
Packit Service 5ffa24
                                     nm_dhcp_client_get_uuid(client),
Packit Service 5ffa24
                                     &lease_file);
Packit Service 5ffa24
Packit Service 5ffa24
    if (last_ip4_address)
Packit Service 5ffa24
        inet_pton(AF_INET, last_ip4_address, &last_addr);
Packit Service 5ffa24
    else {
Packit Service 5ffa24
        nm_auto(sd_dhcp_lease_unrefp) sd_dhcp_lease *lease = NULL;
Packit Service 5ffa24
Packit Service 5ffa24
        dhcp_lease_load(&lease, lease_file);
Packit Service 5ffa24
        if (lease)
Packit Service 5ffa24
            sd_dhcp_lease_get_address(lease, &last_addr);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (last_addr.s_addr) {
Packit Service 5ffa24
        r = sd_dhcp_client_set_request_address(sd_client, &last_addr);
Packit Service 5ffa24
        if (r < 0) {
Packit Service 5ffa24
            nm_utils_error_set_errno(error, r, "failed to set last IPv4 address: %s");
Packit Service 5ffa24
            return FALSE;
Packit Service 5ffa24
        }
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    client_id = nm_dhcp_client_get_client_id(client);
Packit Service 5ffa24
    if (!client_id) {
Packit Service 5ffa24
        client_id_new = nm_utils_dhcp_client_id_mac(arp_type, hwaddr_arr, hwaddr_len);
Packit Service 5ffa24
        client_id     = client_id_new;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (!(client_id_arr = g_bytes_get_data(client_id, &client_id_len)) || client_id_len < 2) {
Packit Service 5ffa24
        /* invalid client-ids are not expected. */
Packit Service 5ffa24
        nm_assert_not_reached();
Packit Service 5ffa24
Packit Service 5ffa24
        nm_utils_error_set_literal(error, NM_UTILS_ERROR_UNKNOWN, "no valid IPv4 client-id");
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    /* Note that we always set a client-id. In particular for infiniband that is necessary,
Packit Service 5ffa24
     * see https://tools.ietf.org/html/rfc4390#section-2.1 . */
Packit Service 5ffa24
    r = sd_dhcp_client_set_client_id(sd_client,
Packit Service 5ffa24
                                     client_id_arr[0],
Packit Service 5ffa24
                                     client_id_arr + 1,
Packit Service 5ffa24
                                     NM_MIN(client_id_len - 1, _NM_SD_MAX_CLIENT_ID_LEN));
Packit Service 5ffa24
    if (r < 0) {
Packit Service 5ffa24
        nm_utils_error_set_errno(error, r, "failed to set IPv4 client-id: %s");
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    /* Add requested options */
Packit Service 5ffa24
    for (i = 0; _nm_dhcp_option_dhcp4_options[i].name; i++) {
Packit Service 5ffa24
        if (_nm_dhcp_option_dhcp4_options[i].include) {
Packit Service 5ffa24
            nm_assert(_nm_dhcp_option_dhcp4_options[i].option_num <= 255);
Packit Service 5ffa24
            r = sd_dhcp_client_set_request_option(sd_client,
Packit Service 5ffa24
                                                  _nm_dhcp_option_dhcp4_options[i].option_num);
Packit Service 5ffa24
            nm_assert(r >= 0 || r == -EEXIST);
Packit Service 5ffa24
        }
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    hostname = nm_dhcp_client_get_hostname(client);
Packit Service 5ffa24
    if (hostname) {
Packit Service 5ffa24
        /* FIXME: sd-dhcp decides which hostname/FQDN option to send (12 or 81)
Packit Service 5ffa24
         * only based on whether the hostname has a domain part or not. At the
Packit Service 5ffa24
         * moment there is no way to force one or another.
Packit Service 5ffa24
         */
Packit Service 5ffa24
        r = sd_dhcp_client_set_hostname(sd_client, hostname);
Packit Service 5ffa24
        if (r < 0) {
Packit Service 5ffa24
            nm_utils_error_set_errno(error, r, "failed to set DHCP hostname: %s");
Packit Service 5ffa24
            return FALSE;
Packit Service 5ffa24
        }
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    mud_url = nm_dhcp_client_get_mud_url(client);
Packit Service 5ffa24
    if (mud_url) {
Packit Service 5ffa24
        r = sd_dhcp_client_set_mud_url(sd_client, mud_url);
Packit Service 5ffa24
        if (r < 0) {
Packit Service 5ffa24
            nm_utils_error_set_errno(error, r, "failed to set DHCP MUDURL: %s");
Packit Service 5ffa24
            return FALSE;
Packit Service 5ffa24
        }
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    vendor_class_identifier = nm_dhcp_client_get_vendor_class_identifier(client);
Packit Service 5ffa24
    if (vendor_class_identifier) {
Packit Service 5ffa24
        const char *option_data;
Packit Service 5ffa24
        gsize       len;
Packit Service 5ffa24
Packit Service 5ffa24
        option_data = g_bytes_get_data(vendor_class_identifier, &len;;
Packit Service 5ffa24
        nm_assert(option_data);
Packit Service 5ffa24
        nm_assert(len <= 255);
Packit Service 5ffa24
Packit Service 5ffa24
        option_data = nm_strndup_a(300, option_data, len, NULL);
Packit Service 5ffa24
Packit Service 5ffa24
        r = sd_dhcp_client_set_vendor_class_identifier(sd_client, option_data);
Packit Service 5ffa24
        if (r < 0) {
Packit Service 5ffa24
            nm_utils_error_set_errno(error, r, "failed to set DHCP vendor class identifier: %s");
Packit Service 5ffa24
            return FALSE;
Packit Service 5ffa24
        }
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    r = sd_dhcp_client_set_callback(sd_client, dhcp_event_cb, client);
Packit Service 5ffa24
    if (r < 0) {
Packit Service 5ffa24
        nm_utils_error_set_errno(error, r, "failed to set callback: %s");
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    priv->client4 = g_steal_pointer(&sd_client);
Packit Service 5ffa24
Packit Service 5ffa24
    g_free(priv->lease_file);
Packit Service 5ffa24
    priv->lease_file = g_steal_pointer(&lease_file);
Packit Service 5ffa24
Packit Service 5ffa24
    nm_dhcp_client_set_client_id(client, client_id);
Packit Service 5ffa24
Packit Service 5ffa24
    r = sd_dhcp_client_start(priv->client4);
Packit Service 5ffa24
    if (r < 0) {
Packit Service 5ffa24
        sd_dhcp_client_set_callback(priv->client4, NULL, NULL);
Packit Service 5ffa24
        nm_clear_pointer(&priv->client4, sd_dhcp_client_unref);
Packit Service 5ffa24
        nm_utils_error_set_errno(error, r, "failed to start DHCP client: %s");
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    nm_dhcp_client_start_timeout(client);
Packit Service 5ffa24
    return TRUE;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static NMIP6Config *
Packit Service 5ffa24
lease_to_ip6_config(NMDedupMultiIndex *multi_idx,
Packit Service 5ffa24
                    const char *       iface,
Packit Service 5ffa24
                    int                ifindex,
Packit Service 5ffa24
                    sd_dhcp6_lease *   lease,
Packit Service 5ffa24
                    gboolean           info_only,
Packit Service 5ffa24
                    GHashTable **      out_options,
Packit Service 5ffa24
                    gint32             ts,
Packit Service 5ffa24
                    GError **          error)
Packit Service 5ffa24
{
Packit Service 5ffa24
    gs_unref_object NMIP6Config *ip6_config = NULL;
Packit Service 5ffa24
    gs_unref_hashtable GHashTable *options  = NULL;
Packit Service 5ffa24
    struct in6_addr                tmp_addr;
Packit Service 5ffa24
    const struct in6_addr *        dns;
Packit Service 5ffa24
    uint32_t                       lft_pref, lft_valid;
Packit Service 5ffa24
    char                           addr_str[NM_UTILS_INET_ADDRSTRLEN];
Packit Service 5ffa24
    char **                        domains;
Packit Service 5ffa24
    const char *                   s;
Packit Service 5ffa24
    nm_auto_free_gstring GString *str = NULL;
Packit Service 5ffa24
    int                           num, i;
Packit Service 5ffa24
Packit Service 5ffa24
    nm_assert(lease);
Packit Service 5ffa24
Packit Service 5ffa24
    ip6_config = nm_ip6_config_new(multi_idx, ifindex);
Packit Service 5ffa24
Packit Service 5ffa24
    options = out_options ? nm_dhcp_option_create_options_dict() : NULL;
Packit Service 5ffa24
Packit Service 5ffa24
    sd_dhcp6_lease_reset_address_iter(lease);
Packit Service 5ffa24
    nm_gstring_prepare(&str);
Packit Service 5ffa24
    while (sd_dhcp6_lease_get_address(lease, &tmp_addr, &lft_pref, &lft_valid) >= 0) {
Packit Service 5ffa24
        const NMPlatformIP6Address address = {
Packit Service 5ffa24
            .plen        = 128,
Packit Service 5ffa24
            .address     = tmp_addr,
Packit Service 5ffa24
            .timestamp   = ts,
Packit Service 5ffa24
            .lifetime    = lft_valid,
Packit Service 5ffa24
            .preferred   = lft_pref,
Packit Service 5ffa24
            .addr_source = NM_IP_CONFIG_SOURCE_DHCP,
Packit Service 5ffa24
        };
Packit Service 5ffa24
Packit Service 5ffa24
        nm_ip6_config_add_address(ip6_config, &address);
Packit Service 5ffa24
Packit Service 5ffa24
        _nm_utils_inet6_ntop(&tmp_addr, addr_str);
Packit Service 5ffa24
        g_string_append(nm_gstring_add_space_delimiter(str), addr_str);
Packit Service 5ffa24
    };
Packit Service 5ffa24
    if (str->len)
Packit Service 5ffa24
        nm_dhcp_option_add_option(options,
Packit Service 5ffa24
                                  _nm_dhcp_option_dhcp6_options,
Packit Service 5ffa24
                                  NM_DHCP_OPTION_DHCP6_NM_IP_ADDRESS,
Packit Service 5ffa24
                                  str->str);
Packit Service 5ffa24
Packit Service 5ffa24
    if (!info_only && nm_ip6_config_get_num_addresses(ip6_config) == 0) {
Packit Service 5ffa24
        g_set_error_literal(error,
Packit Service 5ffa24
                            NM_MANAGER_ERROR,
Packit Service 5ffa24
                            NM_MANAGER_ERROR_FAILED,
Packit Service 5ffa24
                            "no address received in managed mode");
Packit Service 5ffa24
        return NULL;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    num = sd_dhcp6_lease_get_dns(lease, &dns);
Packit Service 5ffa24
    if (num > 0) {
Packit Service 5ffa24
        nm_gstring_prepare(&str);
Packit Service 5ffa24
        for (i = 0; i < num; i++) {
Packit Service 5ffa24
            _nm_utils_inet6_ntop(&dns[i], addr_str);
Packit Service 5ffa24
            g_string_append(nm_gstring_add_space_delimiter(str), addr_str);
Packit Service 5ffa24
            nm_ip6_config_add_nameserver(ip6_config, &dns[i]);
Packit Service 5ffa24
        }
Packit Service 5ffa24
        nm_dhcp_option_add_option(options,
Packit Service 5ffa24
                                  _nm_dhcp_option_dhcp6_options,
Packit Service 5ffa24
                                  NM_DHCP_OPTION_DHCP6_DNS_SERVERS,
Packit Service 5ffa24
                                  str->str);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    num = sd_dhcp6_lease_get_domains(lease, &domains);
Packit Service 5ffa24
    if (num > 0) {
Packit Service 5ffa24
        nm_gstring_prepare(&str);
Packit Service 5ffa24
        for (i = 0; i < num; i++) {
Packit Service 5ffa24
            g_string_append(nm_gstring_add_space_delimiter(str), domains[i]);
Packit Service 5ffa24
            nm_ip6_config_add_search(ip6_config, domains[i]);
Packit Service 5ffa24
        }
Packit Service 5ffa24
        nm_dhcp_option_add_option(options,
Packit Service 5ffa24
                                  _nm_dhcp_option_dhcp6_options,
Packit Service 5ffa24
                                  NM_DHCP_OPTION_DHCP6_DOMAIN_LIST,
Packit Service 5ffa24
                                  str->str);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (sd_dhcp6_lease_get_fqdn(lease, &s) >= 0) {
Packit Service 5ffa24
        nm_dhcp_option_add_option(options,
Packit Service 5ffa24
                                  _nm_dhcp_option_dhcp6_options,
Packit Service 5ffa24
                                  NM_DHCP_OPTION_DHCP6_FQDN,
Packit Service 5ffa24
                                  s);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    NM_SET_OUT(out_options, g_steal_pointer(&options));
Packit Service 5ffa24
    return g_steal_pointer(&ip6_config);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
bound6_handle(NMDhcpSystemd *self)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMDhcpSystemdPrivate *priv              = NM_DHCP_SYSTEMD_GET_PRIVATE(self);
Packit Service 5ffa24
    const gint32          ts                = nm_utils_get_monotonic_timestamp_sec();
Packit Service 5ffa24
    const char *          iface             = nm_dhcp_client_get_iface(NM_DHCP_CLIENT(self));
Packit Service 5ffa24
    gs_unref_object NMIP6Config *ip6_config = NULL;
Packit Service 5ffa24
    gs_unref_hashtable GHashTable *options  = NULL;
Packit Service 5ffa24
    gs_free_error GError *error             = NULL;
Packit Service 5ffa24
    NMPlatformIP6Address  prefix            = {0};
Packit Service 5ffa24
    sd_dhcp6_lease *      lease             = NULL;
Packit Service 5ffa24
Packit Service 5ffa24
    if (sd_dhcp6_client_get_lease(priv->client6, &lease) < 0 || !lease) {
Packit Service 5ffa24
        _LOGW(" no lease!");
Packit Service 5ffa24
        nm_dhcp_client_set_state(NM_DHCP_CLIENT(self), NM_DHCP_STATE_FAIL, NULL, NULL);
Packit Service 5ffa24
        return;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    _LOGD("lease available");
Packit Service 5ffa24
Packit Service 5ffa24
    ip6_config = lease_to_ip6_config(nm_dhcp_client_get_multi_idx(NM_DHCP_CLIENT(self)),
Packit Service 5ffa24
                                     iface,
Packit Service 5ffa24
                                     nm_dhcp_client_get_ifindex(NM_DHCP_CLIENT(self)),
Packit Service 5ffa24
                                     lease,
Packit Service 5ffa24
                                     nm_dhcp_client_get_info_only(NM_DHCP_CLIENT(self)),
Packit Service 5ffa24
                                     &options,
Packit Service 5ffa24
                                     ts,
Packit Service 5ffa24
                                     &error);
Packit Service 5ffa24
Packit Service 5ffa24
    if (!ip6_config) {
Packit Service 5ffa24
        _LOGW("%s", error->message);
Packit Service 5ffa24
        nm_dhcp_client_set_state(NM_DHCP_CLIENT(self), NM_DHCP_STATE_FAIL, NULL, NULL);
Packit Service 5ffa24
        return;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    nm_dhcp_client_set_state(NM_DHCP_CLIENT(self),
Packit Service 5ffa24
                             NM_DHCP_STATE_BOUND,
Packit Service 5ffa24
                             NM_IP_CONFIG_CAST(ip6_config),
Packit Service 5ffa24
                             options);
Packit Service 5ffa24
Packit Service 5ffa24
    sd_dhcp6_lease_reset_pd_prefix_iter(lease);
Packit Service 5ffa24
    while (!sd_dhcp6_lease_get_pd(lease,
Packit Service 5ffa24
                                  &prefix.address,
Packit Service 5ffa24
                                  &prefix.plen,
Packit Service 5ffa24
                                  &prefix.preferred,
Packit Service 5ffa24
                                  &prefix.lifetime)) {
Packit Service 5ffa24
        prefix.timestamp = ts;
Packit Service 5ffa24
        nm_dhcp_client_emit_ipv6_prefix_delegated(NM_DHCP_CLIENT(self), &prefix);
Packit Service 5ffa24
    }
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
dhcp6_event_cb(sd_dhcp6_client *client, int event, gpointer user_data)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMDhcpSystemd *       self = NM_DHCP_SYSTEMD(user_data);
Packit Service 5ffa24
    NMDhcpSystemdPrivate *priv = NM_DHCP_SYSTEMD_GET_PRIVATE(self);
Packit Service 5ffa24
Packit Service 5ffa24
    nm_assert(priv->client6 == client);
Packit Service 5ffa24
Packit Service 5ffa24
    _LOGD("client event %d", event);
Packit Service 5ffa24
Packit Service 5ffa24
    switch (event) {
Packit Service 5ffa24
    case SD_DHCP6_CLIENT_EVENT_RETRANS_MAX:
Packit Service 5ffa24
        nm_dhcp_client_set_state(NM_DHCP_CLIENT(user_data), NM_DHCP_STATE_TIMEOUT, NULL, NULL);
Packit Service 5ffa24
        break;
Packit Service 5ffa24
    case SD_DHCP6_CLIENT_EVENT_RESEND_EXPIRE:
Packit Service 5ffa24
    case SD_DHCP6_CLIENT_EVENT_STOP:
Packit Service 5ffa24
        nm_dhcp_client_set_state(NM_DHCP_CLIENT(user_data), NM_DHCP_STATE_FAIL, NULL, NULL);
Packit Service 5ffa24
        break;
Packit Service 5ffa24
    case SD_DHCP6_CLIENT_EVENT_IP_ACQUIRE:
Packit Service 5ffa24
    case SD_DHCP6_CLIENT_EVENT_INFORMATION_REQUEST:
Packit Service 5ffa24
        bound6_handle(self);
Packit Service 5ffa24
        break;
Packit Service 5ffa24
    default:
Packit Service 5ffa24
        _LOGW("unhandled event %d", event);
Packit Service 5ffa24
        break;
Packit Service 5ffa24
    }
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static gboolean
Packit Service 5ffa24
ip6_start(NMDhcpClient *            client,
Packit Service 5ffa24
          const char *              dhcp_anycast_addr,
Packit Service 5ffa24
          const struct in6_addr *   ll_addr,
Packit Service 5ffa24
          NMSettingIP6ConfigPrivacy privacy,
Packit Service 5ffa24
          guint                     needed_prefixes,
Packit Service 5ffa24
          GError **                 error)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMDhcpSystemd *                                  self      = NM_DHCP_SYSTEMD(client);
Packit Service 5ffa24
    NMDhcpSystemdPrivate *                           priv      = NM_DHCP_SYSTEMD_GET_PRIVATE(self);
Packit Service 5ffa24
    nm_auto(sd_dhcp6_client_unrefp) sd_dhcp6_client *sd_client = NULL;
Packit Service 5ffa24
    const char *                                     hostname;
Packit Service 5ffa24
    const char *                                     mud_url;
Packit Service 5ffa24
    int                                              r, i;
Packit Service 5ffa24
    const guint8 *                                   duid_arr;
Packit Service 5ffa24
    gsize                                            duid_len;
Packit Service 5ffa24
    GBytes *                                         duid;
Packit Service 5ffa24
Packit Service 5ffa24
    g_return_val_if_fail(!priv->client4, FALSE);
Packit Service 5ffa24
    g_return_val_if_fail(!priv->client6, FALSE);
Packit Service 5ffa24
Packit Service 5ffa24
    if (!(duid = nm_dhcp_client_get_client_id(client))
Packit Service 5ffa24
        || !(duid_arr = g_bytes_get_data(duid, &duid_len)) || duid_len < 2) {
Packit Service 5ffa24
        nm_utils_error_set_literal(error, NM_UTILS_ERROR_UNKNOWN, "missing DUID");
Packit Service 5ffa24
        g_return_val_if_reached(FALSE);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    r = sd_dhcp6_client_new(&sd_client);
Packit Service 5ffa24
    if (r < 0) {
Packit Service 5ffa24
        nm_utils_error_set_errno(error, r, "failed to create dhcp-client: %s");
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    _LOGT("dhcp-client6: set %p", sd_client);
Packit Service 5ffa24
Packit Service 5ffa24
    if (nm_dhcp_client_get_info_only(client))
Packit Service 5ffa24
        sd_dhcp6_client_set_information_request(sd_client, 1);
Packit Service 5ffa24
Packit Service 5ffa24
    r = sd_dhcp6_client_set_iaid(sd_client, nm_dhcp_client_get_iaid(client));
Packit Service 5ffa24
    if (r < 0) {
Packit Service 5ffa24
        nm_utils_error_set_errno(error, r, "failed to set IAID: %s");
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    r = sd_dhcp6_client_set_duid(sd_client,
Packit Service 5ffa24
                                 unaligned_read_be16(&duid_arr[0]),
Packit Service 5ffa24
                                 &duid_arr[2],
Packit Service 5ffa24
                                 duid_len - 2);
Packit Service 5ffa24
    if (r < 0) {
Packit Service 5ffa24
        nm_utils_error_set_errno(error, r, "failed to set DUID: %s");
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    r = sd_dhcp6_client_attach_event(sd_client, NULL, 0);
Packit Service 5ffa24
    if (r < 0) {
Packit Service 5ffa24
        nm_utils_error_set_errno(error, r, "failed to attach event: %s");
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    r = sd_dhcp6_client_set_ifindex(sd_client, nm_dhcp_client_get_ifindex(client));
Packit Service 5ffa24
    if (r < 0) {
Packit Service 5ffa24
        nm_utils_error_set_errno(error, r, "failed to set ifindex: %s");
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    /* Add requested options */
Packit Service 5ffa24
    for (i = 0; _nm_dhcp_option_dhcp6_options[i].name; i++) {
Packit Service 5ffa24
        if (_nm_dhcp_option_dhcp6_options[i].include) {
Packit Service 5ffa24
            r = sd_dhcp6_client_set_request_option(sd_client,
Packit Service 5ffa24
                                                   _nm_dhcp_option_dhcp6_options[i].option_num);
Packit Service 5ffa24
            nm_assert(r >= 0 || r == -EEXIST);
Packit Service 5ffa24
        }
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    mud_url = nm_dhcp_client_get_mud_url(client);
Packit Service 5ffa24
    if (mud_url) {
Packit Service 5ffa24
        r = sd_dhcp6_client_set_request_mud_url(sd_client, mud_url);
Packit Service 5ffa24
        if (r < 0) {
Packit Service 5ffa24
            nm_utils_error_set_errno(error, r, "failed to set mud-url: %s");
Packit Service 5ffa24
            return FALSE;
Packit Service 5ffa24
        }
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (needed_prefixes > 0) {
Packit Service 5ffa24
        if (needed_prefixes > 1)
Packit Service 5ffa24
            _LOGW("dhcp-client6: only one prefix request is supported");
Packit Service 5ffa24
        /* FIXME: systemd-networkd API only allows to request a
Packit Service 5ffa24
         * single prefix */
Packit Service 5ffa24
        r = sd_dhcp6_client_set_prefix_delegation(sd_client, TRUE);
Packit Service 5ffa24
        if (r < 0) {
Packit Service 5ffa24
            nm_utils_error_set_errno(error, r, "failed to enable prefix delegation: %s");
Packit Service 5ffa24
            return FALSE;
Packit Service 5ffa24
        }
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    r = sd_dhcp6_client_set_local_address(sd_client, ll_addr);
Packit Service 5ffa24
    if (r < 0) {
Packit Service 5ffa24
        nm_utils_error_set_errno(error, r, "failed to set local address: %s");
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    hostname = nm_dhcp_client_get_hostname(client);
Packit Service 5ffa24
    r        = sd_dhcp6_client_set_fqdn(sd_client, hostname);
Packit Service 5ffa24
    if (r < 0) {
Packit Service 5ffa24
        nm_utils_error_set_errno(error, r, "failed to set DHCP hostname: %s");
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    r = sd_dhcp6_client_set_callback(sd_client, dhcp6_event_cb, client);
Packit Service 5ffa24
    if (r < 0) {
Packit Service 5ffa24
        nm_utils_error_set_errno(error, r, "failed to set callback: %s");
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    priv->client6 = g_steal_pointer(&sd_client);
Packit Service 5ffa24
Packit Service 5ffa24
    r = sd_dhcp6_client_start(priv->client6);
Packit Service 5ffa24
    if (r < 0) {
Packit Service 5ffa24
        sd_dhcp6_client_set_callback(priv->client6, NULL, NULL);
Packit Service 5ffa24
        nm_clear_pointer(&priv->client6, sd_dhcp6_client_unref);
Packit Service 5ffa24
        nm_utils_error_set_errno(error, r, "failed to start client: %s");
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    nm_dhcp_client_start_timeout(client);
Packit Service 5ffa24
    return TRUE;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
stop(NMDhcpClient *client, gboolean release)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMDhcpSystemd *       self = NM_DHCP_SYSTEMD(client);
Packit Service 5ffa24
    NMDhcpSystemdPrivate *priv = NM_DHCP_SYSTEMD_GET_PRIVATE(self);
Packit Service 5ffa24
    int                   r    = 0;
Packit Service 5ffa24
Packit Service 5ffa24
    NM_DHCP_CLIENT_CLASS(nm_dhcp_systemd_parent_class)->stop(client, release);
Packit Service 5ffa24
Packit Service 5ffa24
    _LOGT("dhcp-client%d: stop %p",
Packit Service 5ffa24
          priv->client4 ? '4' : '6',
Packit Service 5ffa24
          priv->client4 ? (gpointer) priv->client4 : (gpointer) priv->client6);
Packit Service 5ffa24
Packit Service 5ffa24
    if (priv->client4) {
Packit Service 5ffa24
        sd_dhcp_client_set_callback(priv->client4, NULL, NULL);
Packit Service 5ffa24
        r = sd_dhcp_client_stop(priv->client4);
Packit Service 5ffa24
    } else if (priv->client6) {
Packit Service 5ffa24
        sd_dhcp6_client_set_callback(priv->client6, NULL, NULL);
Packit Service 5ffa24
        r = sd_dhcp6_client_stop(priv->client6);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (r)
Packit Service 5ffa24
        _LOGW("failed to stop client (%d)", r);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
nm_dhcp_systemd_init(NMDhcpSystemd *self)
Packit Service 5ffa24
{}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
dispose(GObject *object)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMDhcpSystemdPrivate *priv = NM_DHCP_SYSTEMD_GET_PRIVATE(object);
Packit Service 5ffa24
Packit Service 5ffa24
    nm_clear_g_free(&priv->lease_file);
Packit Service 5ffa24
Packit Service 5ffa24
    if (priv->client4) {
Packit Service 5ffa24
        sd_dhcp_client_stop(priv->client4);
Packit Service 5ffa24
        sd_dhcp_client_unref(priv->client4);
Packit Service 5ffa24
        priv->client4 = NULL;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (priv->client6) {
Packit Service 5ffa24
        sd_dhcp6_client_stop(priv->client6);
Packit Service 5ffa24
        sd_dhcp6_client_unref(priv->client6);
Packit Service 5ffa24
        priv->client6 = NULL;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    G_OBJECT_CLASS(nm_dhcp_systemd_parent_class)->dispose(object);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
nm_dhcp_systemd_class_init(NMDhcpSystemdClass *sdhcp_class)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMDhcpClientClass *client_class = NM_DHCP_CLIENT_CLASS(sdhcp_class);
Packit Service 5ffa24
    GObjectClass *     object_class = G_OBJECT_CLASS(sdhcp_class);
Packit Service 5ffa24
Packit Service 5ffa24
    object_class->dispose = dispose;
Packit Service 5ffa24
Packit Service 5ffa24
    client_class->ip4_start = ip4_start;
Packit Service 5ffa24
    client_class->ip6_start = ip6_start;
Packit Service 5ffa24
    client_class->stop      = stop;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
const NMDhcpClientFactory _nm_dhcp_client_factory_systemd = {
Packit Service 5ffa24
    .name         = "systemd",
Packit Service 5ffa24
    .get_type     = nm_dhcp_systemd_get_type,
Packit Service 5ffa24
    .experimental = TRUE,
Packit Service 5ffa24
};
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
static GType
Packit Service 5ffa24
_get_type_per_addr_family(int addr_family)
Packit Service 5ffa24
{
Packit Service 5ffa24
    nm_assert_addr_family(addr_family);
Packit Service 5ffa24
Packit Service 5ffa24
    if (addr_family == AF_INET)
Packit Service 5ffa24
        return nm_dhcp_nettools_get_type();
Packit Service 5ffa24
    return nm_dhcp_systemd_get_type();
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
const NMDhcpClientFactory _nm_dhcp_client_factory_internal = {
Packit Service 5ffa24
    .name                     = "internal",
Packit Service 5ffa24
    .get_type_per_addr_family = _get_type_per_addr_family,
Packit Service 5ffa24
};