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

Packit Service 5ffa24
/* SPDX-License-Identifier: GPL-2.0-or-later */
Packit Service 5ffa24
/*
Packit Service 5ffa24
 * Copyright (C) 2005 - 2012 Red Hat, Inc.
Packit Service 5ffa24
 */
Packit Service 5ffa24
Packit Service 5ffa24
#include <config.h>
Packit Service 5ffa24
#define __CONFIG_H__
Packit Service 5ffa24
Packit Service 5ffa24
#define _XOPEN_SOURCE
Packit Service 5ffa24
#include <time.h>
Packit Service 5ffa24
#undef _XOPEN_SOURCE
Packit Service 5ffa24
Packit Service 2bceb2
#include "src/core/nm-default-daemon.h"
Packit Service 5ffa24
Packit Service 5ffa24
#if WITH_DHCLIENT
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
Packit Service dff8e4
    #include "libnm-glib-aux/nm-dedup-multi.h"
Packit Service 5ffa24
Packit Service 5ffa24
    #include "nm-utils.h"
Packit Service 5ffa24
    #include "nm-dhcp-dhclient-utils.h"
Packit Service 5ffa24
    #include "nm-dhcp-manager.h"
Packit Service 5ffa24
    #include "NetworkManagerUtils.h"
Packit Service 5ffa24
    #include "nm-dhcp-listener.h"
Packit Service 5ffa24
    #include "nm-dhcp-client-logging.h"
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
static const char *
Packit Service 5ffa24
_addr_family_to_path_part(int addr_family)
Packit Service 5ffa24
{
Packit Service 5ffa24
    nm_assert(NM_IN_SET(addr_family, AF_INET, AF_INET6));
Packit Service 5ffa24
    return (addr_family == AF_INET6) ? "6" : "";
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
    #define NM_TYPE_DHCP_DHCLIENT (nm_dhcp_dhclient_get_type())
Packit Service 5ffa24
    #define NM_DHCP_DHCLIENT(obj) \
Packit Service 5ffa24
        (G_TYPE_CHECK_INSTANCE_CAST((obj), NM_TYPE_DHCP_DHCLIENT, NMDhcpDhclient))
Packit Service 5ffa24
    #define NM_DHCP_DHCLIENT_CLASS(klass) \
Packit Service 5ffa24
        (G_TYPE_CHECK_CLASS_CAST((klass), NM_TYPE_DHCP_DHCLIENT, NMDhcpDhclientClass))
Packit Service 5ffa24
    #define NM_IS_DHCP_DHCLIENT(obj) (G_TYPE_CHECK_INSTANCE_TYPE((obj), NM_TYPE_DHCP_DHCLIENT))
Packit Service 5ffa24
    #define NM_IS_DHCP_DHCLIENT_CLASS(klass) \
Packit Service 5ffa24
        (G_TYPE_CHECK_CLASS_TYPE((klass), NM_TYPE_DHCP_DHCLIENT))
Packit Service 5ffa24
    #define NM_DHCP_DHCLIENT_GET_CLASS(obj) \
Packit Service 5ffa24
        (G_TYPE_INSTANCE_GET_CLASS((obj), NM_TYPE_DHCP_DHCLIENT, NMDhcpDhclientClass))
Packit Service 5ffa24
Packit Service 5ffa24
typedef struct _NMDhcpDhclient      NMDhcpDhclient;
Packit Service 5ffa24
typedef struct _NMDhcpDhclientClass NMDhcpDhclientClass;
Packit Service 5ffa24
Packit Service 5ffa24
static GType nm_dhcp_dhclient_get_type(void);
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
typedef struct {
Packit Service 5ffa24
    char *          conf_file;
Packit Service 5ffa24
    const char *    def_leasefile;
Packit Service 5ffa24
    char *          lease_file;
Packit Service 5ffa24
    char *          pid_file;
Packit Service 5ffa24
    NMDhcpListener *dhcp_listener;
Packit Service 5ffa24
} NMDhcpDhclientPrivate;
Packit Service 5ffa24
Packit Service 5ffa24
struct _NMDhcpDhclient {
Packit Service 5ffa24
    NMDhcpClient          parent;
Packit Service 5ffa24
    NMDhcpDhclientPrivate _priv;
Packit Service 5ffa24
};
Packit Service 5ffa24
Packit Service 5ffa24
struct _NMDhcpDhclientClass {
Packit Service 5ffa24
    NMDhcpClientClass parent;
Packit Service 5ffa24
};
Packit Service 5ffa24
Packit Service 5ffa24
G_DEFINE_TYPE(NMDhcpDhclient, nm_dhcp_dhclient, NM_TYPE_DHCP_CLIENT)
Packit Service 5ffa24
Packit Service 5ffa24
    #define NM_DHCP_DHCLIENT_GET_PRIVATE(self) \
Packit Service 5ffa24
        _NM_GET_PRIVATE(self, NMDhcpDhclient, NM_IS_DHCP_DHCLIENT)
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
static const char *
Packit Service 5ffa24
nm_dhcp_dhclient_get_path(void)
Packit Service 5ffa24
{
Packit Service 5ffa24
    return nm_utils_find_helper("dhclient", DHCLIENT_PATH, NULL);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
/**
Packit Service 5ffa24
 * get_dhclient_leasefile():
Packit Service 5ffa24
 * @addr_family: AF_INET or AF_INET6
Packit Service 5ffa24
 * @iface: the interface name of the device on which DHCP will be done
Packit Service 5ffa24
 * @uuid: the connection UUID to which the returned lease should belong
Packit Service 5ffa24
 * @out_preferred_path: on return, the "most preferred" leasefile path
Packit Service 5ffa24
 *
Packit Service 5ffa24
 * Returns the path of an existing leasefile (if any) for this interface and
Packit Service 5ffa24
 * connection UUID.  Also returns the "most preferred" leasefile path, which
Packit Service 5ffa24
 * may be different than any found leasefile.
Packit Service 5ffa24
 *
Packit Service 5ffa24
 * Returns: an existing leasefile, or %NULL if no matching leasefile could be found
Packit Service 5ffa24
 */
Packit Service 5ffa24
static char *
Packit Service 5ffa24
get_dhclient_leasefile(int         addr_family,
Packit Service 5ffa24
                       const char *iface,
Packit Service 5ffa24
                       const char *uuid,
Packit Service 5ffa24
                       char **     out_preferred_path)
Packit Service 5ffa24
{
Packit Service 5ffa24
    gs_free char *path = NULL;
Packit Service 5ffa24
Packit Service 5ffa24
    if (nm_dhcp_utils_get_leasefile_path(addr_family, "dhclient", iface, uuid, &path)) {
Packit Service 5ffa24
        NM_SET_OUT(out_preferred_path, g_strdup(path));
Packit Service 5ffa24
        return g_steal_pointer(&path);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    NM_SET_OUT(out_preferred_path, g_steal_pointer(&path));
Packit Service 5ffa24
Packit Service 5ffa24
    /* If the leasefile we're looking for doesn't exist yet in the new location
Packit Service 5ffa24
     * (eg, /var/lib/NetworkManager) then look in old locations to maintain
Packit Service 5ffa24
     * backwards compatibility with external tools (like dracut) that put
Packit Service 5ffa24
     * leasefiles there.
Packit Service 5ffa24
     */
Packit Service 5ffa24
Packit Service 5ffa24
    /* Old Debian, SUSE, and Mandriva location */
Packit Service 5ffa24
    g_free(path);
Packit Service 5ffa24
    path = g_strdup_printf(LOCALSTATEDIR "/lib/dhcp/dhclient%s-%s-%s.lease",
Packit Service 5ffa24
                           _addr_family_to_path_part(addr_family),
Packit Service 5ffa24
                           uuid,
Packit Service 5ffa24
                           iface);
Packit Service 5ffa24
    if (g_file_test(path, G_FILE_TEST_EXISTS))
Packit Service 5ffa24
        return g_steal_pointer(&path);
Packit Service 5ffa24
Packit Service 5ffa24
    /* Old Red Hat and Fedora location */
Packit Service 5ffa24
    g_free(path);
Packit Service 5ffa24
    path = g_strdup_printf(LOCALSTATEDIR "/lib/dhclient/dhclient%s-%s-%s.lease",
Packit Service 5ffa24
                           _addr_family_to_path_part(addr_family),
Packit Service 5ffa24
                           uuid,
Packit Service 5ffa24
                           iface);
Packit Service 5ffa24
    if (g_file_test(path, G_FILE_TEST_EXISTS))
Packit Service 5ffa24
        return g_steal_pointer(&path);
Packit Service 5ffa24
Packit Service 5ffa24
    /* Fail */
Packit Service 5ffa24
    return NULL;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static gboolean
Packit Service 5ffa24
merge_dhclient_config(NMDhcpDhclient *    self,
Packit Service 5ffa24
                      int                 addr_family,
Packit Service 5ffa24
                      const char *        iface,
Packit Service 5ffa24
                      const char *        conf_file,
Packit Service 5ffa24
                      GBytes *            client_id,
Packit Service 5ffa24
                      const char *        anycast_addr,
Packit Service 5ffa24
                      const char *        hostname,
Packit Service 5ffa24
                      guint32             timeout,
Packit Service 5ffa24
                      gboolean            use_fqdn,
Packit Service 5ffa24
                      NMDhcpHostnameFlags hostname_flags,
Packit Service 5ffa24
                      const char *        mud_url,
Packit Service 5ffa24
                      const char *const * reject_servers,
Packit Service 5ffa24
                      const char *        orig_path,
Packit Service 5ffa24
                      GBytes **           out_new_client_id,
Packit Service 5ffa24
                      GError **           error)
Packit Service 5ffa24
{
Packit Service 5ffa24
    gs_free char *orig = NULL;
Packit Service 5ffa24
    gs_free char *new  = NULL;
Packit Service 5ffa24
Packit Service 5ffa24
    g_return_val_if_fail(iface, FALSE);
Packit Service 5ffa24
    g_return_val_if_fail(conf_file, FALSE);
Packit Service 5ffa24
Packit Service 5ffa24
    if (orig_path && g_file_test(orig_path, G_FILE_TEST_EXISTS)) {
Packit Service 5ffa24
        GError *read_error = NULL;
Packit Service 5ffa24
Packit Service 5ffa24
        if (!g_file_get_contents(orig_path, &orig, NULL, &read_error)) {
Packit Service 5ffa24
            _LOGW("error reading dhclient configuration %s: %s", orig_path, read_error->message);
Packit Service 5ffa24
            g_error_free(read_error);
Packit Service 5ffa24
        }
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    new = nm_dhcp_dhclient_create_config(iface,
Packit Service 5ffa24
                                         addr_family,
Packit Service 5ffa24
                                         client_id,
Packit Service 5ffa24
                                         anycast_addr,
Packit Service 5ffa24
                                         hostname,
Packit Service 5ffa24
                                         timeout,
Packit Service 5ffa24
                                         use_fqdn,
Packit Service 5ffa24
                                         hostname_flags,
Packit Service 5ffa24
                                         mud_url,
Packit Service 5ffa24
                                         reject_servers,
Packit Service 5ffa24
                                         orig_path,
Packit Service 5ffa24
                                         orig,
Packit Service 5ffa24
                                         out_new_client_id);
Packit Service 5ffa24
    nm_assert(new);
Packit Service 5ffa24
Packit Service 5ffa24
    return g_file_set_contents(conf_file, new, -1, error);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static char *
Packit Service 5ffa24
find_existing_config(NMDhcpDhclient *self, int addr_family, const char *iface, const char *uuid)
Packit Service 5ffa24
{
Packit Service 5ffa24
    char *path;
Packit Service 5ffa24
Packit Service 5ffa24
    /* NetworkManager-overridden configuration can be used to ship DHCP config
Packit Service 5ffa24
     * with NetworkManager itself. It can be uuid-specific, device-specific
Packit Service 5ffa24
     * or generic.
Packit Service 5ffa24
     */
Packit Service 5ffa24
    if (uuid) {
Packit Service 5ffa24
        path = g_strdup_printf(NMCONFDIR "/dhclient%s-%s.conf",
Packit Service 5ffa24
                               _addr_family_to_path_part(addr_family),
Packit Service 5ffa24
                               uuid);
Packit Service 5ffa24
        _LOGD("looking for existing config %s", path);
Packit Service 5ffa24
        if (g_file_test(path, G_FILE_TEST_EXISTS))
Packit Service 5ffa24
            return path;
Packit Service 5ffa24
        g_free(path);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    path = g_strdup_printf(NMCONFDIR "/dhclient%s-%s.conf",
Packit Service 5ffa24
                           _addr_family_to_path_part(addr_family),
Packit Service 5ffa24
                           iface);
Packit Service 5ffa24
    _LOGD("looking for existing config %s", path);
Packit Service 5ffa24
    if (g_file_test(path, G_FILE_TEST_EXISTS))
Packit Service 5ffa24
        return path;
Packit Service 5ffa24
    g_free(path);
Packit Service 5ffa24
Packit Service 5ffa24
    path = g_strdup_printf(NMCONFDIR "/dhclient%s.conf", _addr_family_to_path_part(addr_family));
Packit Service 5ffa24
    _LOGD("looking for existing config %s", path);
Packit Service 5ffa24
    if (g_file_test(path, G_FILE_TEST_EXISTS))
Packit Service 5ffa24
        return path;
Packit Service 5ffa24
    g_free(path);
Packit Service 5ffa24
Packit Service 5ffa24
    /* Distribution's dhclient configuration is used so that we can use
Packit Service 5ffa24
     * configuration shipped with dhclient (if any).
Packit Service 5ffa24
     *
Packit Service 5ffa24
     * This replaces conditional compilation based on distribution name. Fedora
Packit Service 5ffa24
     * and Debian store the configs in /etc/dhcp while upstream defaults to /etc
Packit Service 5ffa24
     * which is then used by many other distributions. Some distributions
Packit Service 5ffa24
     * (including Fedora) don't even provide a default configuration file.
Packit Service 5ffa24
     */
Packit Service 5ffa24
    path = g_strdup_printf(SYSCONFDIR "/dhcp/dhclient%s-%s.conf",
Packit Service 5ffa24
                           _addr_family_to_path_part(addr_family),
Packit Service 5ffa24
                           iface);
Packit Service 5ffa24
    _LOGD("looking for existing config %s", path);
Packit Service 5ffa24
    if (g_file_test(path, G_FILE_TEST_EXISTS))
Packit Service 5ffa24
        return path;
Packit Service 5ffa24
    g_free(path);
Packit Service 5ffa24
Packit Service 5ffa24
    path = g_strdup_printf(SYSCONFDIR "/dhclient%s-%s.conf",
Packit Service 5ffa24
                           _addr_family_to_path_part(addr_family),
Packit Service 5ffa24
                           iface);
Packit Service 5ffa24
    _LOGD("looking for existing config %s", path);
Packit Service 5ffa24
    if (g_file_test(path, G_FILE_TEST_EXISTS))
Packit Service 5ffa24
        return path;
Packit Service 5ffa24
    g_free(path);
Packit Service 5ffa24
Packit Service 5ffa24
    path =
Packit Service 5ffa24
        g_strdup_printf(SYSCONFDIR "/dhcp/dhclient%s.conf", _addr_family_to_path_part(addr_family));
Packit Service 5ffa24
    _LOGD("looking for existing config %s", path);
Packit Service 5ffa24
    if (g_file_test(path, G_FILE_TEST_EXISTS))
Packit Service 5ffa24
        return path;
Packit Service 5ffa24
    g_free(path);
Packit Service 5ffa24
Packit Service 5ffa24
    path = g_strdup_printf(SYSCONFDIR "/dhclient%s.conf", _addr_family_to_path_part(addr_family));
Packit Service 5ffa24
    _LOGD("looking for existing config %s", path);
Packit Service 5ffa24
    if (g_file_test(path, G_FILE_TEST_EXISTS))
Packit Service 5ffa24
        return path;
Packit Service 5ffa24
    g_free(path);
Packit Service 5ffa24
Packit Service 5ffa24
    return NULL;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
/* NM provides interface-specific options; thus the same dhclient config
Packit Service 5ffa24
 * file cannot be used since DHCP transactions can happen in parallel.
Packit Service 5ffa24
 * Since some distros don't have default per-interface dhclient config files,
Packit Service 5ffa24
 * read their single config file and merge that into a custom per-interface
Packit Service 5ffa24
 * config file along with the NM options.
Packit Service 5ffa24
 */
Packit Service 5ffa24
static char *
Packit Service 5ffa24
create_dhclient_config(NMDhcpDhclient *    self,
Packit Service 5ffa24
                       int                 addr_family,
Packit Service 5ffa24
                       const char *        iface,
Packit Service 5ffa24
                       const char *        uuid,
Packit Service 5ffa24
                       GBytes *            client_id,
Packit Service 5ffa24
                       const char *        dhcp_anycast_addr,
Packit Service 5ffa24
                       const char *        hostname,
Packit Service 5ffa24
                       guint32             timeout,
Packit Service 5ffa24
                       gboolean            use_fqdn,
Packit Service 5ffa24
                       NMDhcpHostnameFlags hostname_flags,
Packit Service 5ffa24
                       const char *        mud_url,
Packit Service 5ffa24
                       const char *const * reject_servers,
Packit Service 5ffa24
                       GBytes **           out_new_client_id)
Packit Service 5ffa24
{
Packit Service 5ffa24
    gs_free char *orig = NULL;
Packit Service 5ffa24
    char *new          = NULL;
Packit Service 5ffa24
    GError *error      = NULL;
Packit Service 5ffa24
Packit Service 5ffa24
    g_return_val_if_fail(iface != NULL, NULL);
Packit Service 5ffa24
Packit Service 5ffa24
    new = g_strdup_printf(NMSTATEDIR "/dhclient%s-%s.conf",
Packit Service 5ffa24
                          _addr_family_to_path_part(addr_family),
Packit Service 5ffa24
                          iface);
Packit Service 5ffa24
Packit Service 5ffa24
    _LOGD("creating composite dhclient config %s", new);
Packit Service 5ffa24
Packit Service 5ffa24
    orig = find_existing_config(self, addr_family, iface, uuid);
Packit Service 5ffa24
    if (orig)
Packit Service 5ffa24
        _LOGD("merging existing dhclient config %s", orig);
Packit Service 5ffa24
    else
Packit Service 5ffa24
        _LOGD("no existing dhclient configuration to merge");
Packit Service 5ffa24
Packit Service 5ffa24
    if (!merge_dhclient_config(self,
Packit Service 5ffa24
                               addr_family,
Packit Service 5ffa24
                               iface,
Packit Service 5ffa24
                               new,
Packit Service 5ffa24
                               client_id,
Packit Service 5ffa24
                               dhcp_anycast_addr,
Packit Service 5ffa24
                               hostname,
Packit Service 5ffa24
                               timeout,
Packit Service 5ffa24
                               use_fqdn,
Packit Service 5ffa24
                               hostname_flags,
Packit Service 5ffa24
                               mud_url,
Packit Service 5ffa24
                               reject_servers,
Packit Service 5ffa24
                               orig,
Packit Service 5ffa24
                               out_new_client_id,
Packit Service 5ffa24
                               &error)) {
Packit Service 5ffa24
        _LOGW("error creating dhclient configuration: %s", error->message);
Packit Service 5ffa24
        g_clear_error(&error);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    return new;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static gboolean
Packit Service 5ffa24
dhclient_start(NMDhcpClient *client,
Packit Service 5ffa24
               const char *  mode_opt,
Packit Service 5ffa24
               gboolean      release,
Packit Service 5ffa24
               pid_t *       out_pid,
Packit Service 5ffa24
               int           prefixes,
Packit Service 5ffa24
               GError **     error)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMDhcpDhclient *       self       = NM_DHCP_DHCLIENT(client);
Packit Service 5ffa24
    NMDhcpDhclientPrivate *priv       = NM_DHCP_DHCLIENT_GET_PRIVATE(self);
Packit Service 5ffa24
    gs_unref_ptrarray GPtrArray *argv = NULL;
Packit Service 5ffa24
    pid_t                        pid;
Packit Service 5ffa24
    gs_free_error GError *local = NULL;
Packit Service 5ffa24
    const char *          iface;
Packit Service 5ffa24
    const char *          uuid;
Packit Service 5ffa24
    const char *          system_bus_address;
Packit Service 5ffa24
    const char *          dhclient_path;
Packit Service 5ffa24
    char *                binary_name;
Packit Service 5ffa24
    gs_free char *        cmd_str                  = NULL;
Packit Service 5ffa24
    gs_free char *        pid_file                 = NULL;
Packit Service 5ffa24
    gs_free char *        system_bus_address_env   = NULL;
Packit Service 5ffa24
    gs_free char *        preferred_leasefile_path = NULL;
Packit Service 5ffa24
    const int             addr_family              = nm_dhcp_client_get_addr_family(client);
Packit Service 5ffa24
Packit Service 5ffa24
    g_return_val_if_fail(!priv->pid_file, FALSE);
Packit Service 5ffa24
Packit Service 5ffa24
    NM_SET_OUT(out_pid, 0);
Packit Service 5ffa24
Packit Service 5ffa24
    dhclient_path = nm_dhcp_dhclient_get_path();
Packit Service 5ffa24
    if (!dhclient_path) {
Packit Service 5ffa24
        nm_utils_error_set_literal(error, NM_UTILS_ERROR_UNKNOWN, "dhclient binary not found");
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    iface = nm_dhcp_client_get_iface(client);
Packit Service 5ffa24
    uuid  = nm_dhcp_client_get_uuid(client);
Packit Service 5ffa24
Packit Service 5ffa24
    pid_file = g_strdup_printf(NMRUNDIR "/dhclient%s-%s.pid",
Packit Service 5ffa24
                               _addr_family_to_path_part(addr_family),
Packit Service 5ffa24
                               iface);
Packit Service 5ffa24
Packit Service 5ffa24
    /* Kill any existing dhclient from the pidfile */
Packit Service 5ffa24
    binary_name = g_path_get_basename(dhclient_path);
Packit Service 5ffa24
    nm_dhcp_client_stop_existing(pid_file, binary_name);
Packit Service 5ffa24
    g_free(binary_name);
Packit Service 5ffa24
Packit Service 5ffa24
    if (release) {
Packit Service 5ffa24
        /* release doesn't use the pidfile after killing an old client */
Packit Service 5ffa24
        nm_clear_g_free(&pid_file);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    g_free(priv->lease_file);
Packit Service 5ffa24
    priv->lease_file = get_dhclient_leasefile(addr_family, iface, uuid, &preferred_leasefile_path);
Packit Service 5ffa24
    nm_assert(preferred_leasefile_path);
Packit Service 5ffa24
    if (!priv->lease_file) {
Packit Service 5ffa24
        /* No existing leasefile, dhclient will create one at the preferred path */
Packit Service 5ffa24
        priv->lease_file = g_steal_pointer(&preferred_leasefile_path);
Packit Service 5ffa24
    } else if (!nm_streq0(priv->lease_file, preferred_leasefile_path)) {
Packit Service 5ffa24
        gs_unref_object GFile *src = g_file_new_for_path(priv->lease_file);
Packit Service 5ffa24
        gs_unref_object GFile *dst = g_file_new_for_path(preferred_leasefile_path);
Packit Service 5ffa24
Packit Service 5ffa24
        /* Try to copy the existing leasefile to the preferred location */
Packit Service 5ffa24
        if (!g_file_copy(src, dst, G_FILE_COPY_OVERWRITE, NULL, NULL, NULL, &local)) {
Packit Service 5ffa24
            gs_free char *s_path = NULL;
Packit Service 5ffa24
            gs_free char *d_path = NULL;
Packit Service 5ffa24
Packit Service 5ffa24
            /* Failure; just use the existing leasefile */
Packit Service 5ffa24
            _LOGW("failed to copy leasefile %s to %s: %s",
Packit Service 5ffa24
                  (s_path = g_file_get_path(src)),
Packit Service 5ffa24
                  (d_path = g_file_get_path(dst)),
Packit Service 5ffa24
                  local->message);
Packit Service 5ffa24
            g_clear_error(&local);
Packit Service 5ffa24
        } else {
Packit Service 5ffa24
            /* Success; use the preferred leasefile path */
Packit Service 5ffa24
            g_free(priv->lease_file);
Packit Service 5ffa24
            priv->lease_file = g_file_get_path(dst);
Packit Service 5ffa24
        }
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    /* Save the DUID to the leasefile dhclient will actually use */
Packit Service 5ffa24
    if (addr_family == AF_INET6) {
Packit Service 5ffa24
        if (!nm_dhcp_dhclient_save_duid(priv->lease_file,
Packit Service 5ffa24
                                        nm_dhcp_client_get_client_id(client),
Packit Service 5ffa24
                                        &local)) {
Packit Service 5ffa24
            nm_utils_error_set(error,
Packit Service 5ffa24
                               NM_UTILS_ERROR_UNKNOWN,
Packit Service 5ffa24
                               "failed to save DUID to '%s': %s",
Packit Service 5ffa24
                               priv->lease_file,
Packit Service 5ffa24
                               local->message);
Packit Service 5ffa24
            return FALSE;
Packit Service 5ffa24
        }
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    argv = g_ptr_array_new();
Packit Service 5ffa24
    g_ptr_array_add(argv, (gpointer) dhclient_path);
Packit Service 5ffa24
Packit Service 5ffa24
    g_ptr_array_add(argv, (gpointer) "-d");
Packit Service 5ffa24
Packit Service 5ffa24
    /* Be quiet. dhclient logs to syslog anyway. And we duplicate the syslog
Packit Service 5ffa24
     * to stderr in case of NM running with --debug.
Packit Service 5ffa24
     */
Packit Service 5ffa24
    g_ptr_array_add(argv, (gpointer) "-q");
Packit Service 5ffa24
Packit Service 5ffa24
    if (release)
Packit Service 5ffa24
        g_ptr_array_add(argv, (gpointer) "-r");
Packit Service 5ffa24
Packit Service 5ffa24
    if (addr_family == AF_INET6) {
Packit Service 5ffa24
        g_ptr_array_add(argv, (gpointer) "-6");
Packit Service 2bceb2
Packit Service 2bceb2
        if (prefixes > 0 && nm_streq0(mode_opt, "-S")) {
Packit Service 2bceb2
            /* -S is incompatible with -P, only use the latter */
Packit Service 2bceb2
            mode_opt = NULL;
Packit Service 2bceb2
        }
Packit Service 2bceb2
Packit Service 5ffa24
        if (mode_opt)
Packit Service 5ffa24
            g_ptr_array_add(argv, (gpointer) mode_opt);
Packit Service 5ffa24
        while (prefixes--)
Packit Service 5ffa24
            g_ptr_array_add(argv, (gpointer) "-P");
Packit Service 5ffa24
    }
Packit Service 5ffa24
    g_ptr_array_add(argv, (gpointer) "-sf"); /* Set script file */
Packit Service 5ffa24
    g_ptr_array_add(argv, (gpointer) nm_dhcp_helper_path);
Packit Service 5ffa24
Packit Service 5ffa24
    if (pid_file) {
Packit Service 5ffa24
        g_ptr_array_add(argv, (gpointer) "-pf"); /* Set pid file */
Packit Service 5ffa24
        g_ptr_array_add(argv, (gpointer) pid_file);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    g_ptr_array_add(argv, (gpointer) "-lf"); /* Set lease file */
Packit Service 5ffa24
    g_ptr_array_add(argv, (gpointer) priv->lease_file);
Packit Service 5ffa24
Packit Service 5ffa24
    if (priv->conf_file) {
Packit Service 5ffa24
        g_ptr_array_add(argv, (gpointer) "-cf"); /* Set interface config file */
Packit Service 5ffa24
        g_ptr_array_add(argv, (gpointer) priv->conf_file);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    /* Usually the system bus address is well-known; but if it's supposed
Packit Service 5ffa24
     * to be something else, we need to push it to dhclient, since dhclient
Packit Service 5ffa24
     * sanitizes the environment it gives the action scripts.
Packit Service 5ffa24
     */
Packit Service 5ffa24
    system_bus_address = getenv("DBUS_SYSTEM_BUS_ADDRESS");
Packit Service 5ffa24
    if (system_bus_address) {
Packit Service 5ffa24
        system_bus_address_env = g_strdup_printf("DBUS_SYSTEM_BUS_ADDRESS=%s", system_bus_address);
Packit Service 5ffa24
        g_ptr_array_add(argv, (gpointer) "-e");
Packit Service 5ffa24
        g_ptr_array_add(argv, (gpointer) system_bus_address_env);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    g_ptr_array_add(argv, (gpointer) iface);
Packit Service 5ffa24
    g_ptr_array_add(argv, NULL);
Packit Service 5ffa24
Packit Service 5ffa24
    _LOGD("running: %s", (cmd_str = g_strjoinv(" ", (char **) argv->pdata)));
Packit Service 5ffa24
Packit Service 5ffa24
    if (!g_spawn_async(NULL,
Packit Service 5ffa24
                       (char **) argv->pdata,
Packit Service 5ffa24
                       NULL,
Packit Service 5ffa24
                       G_SPAWN_DO_NOT_REAP_CHILD | G_SPAWN_STDOUT_TO_DEV_NULL
Packit Service 5ffa24
                           | G_SPAWN_STDERR_TO_DEV_NULL,
Packit Service 5ffa24
                       nm_utils_setpgid,
Packit Service 5ffa24
                       NULL,
Packit Service 5ffa24
                       &pid,
Packit Service 5ffa24
                       &local)) {
Packit Service 5ffa24
        nm_utils_error_set(error,
Packit Service 5ffa24
                           NM_UTILS_ERROR_UNKNOWN,
Packit Service 5ffa24
                           "dhclient failed to start: %s",
Packit Service 5ffa24
                           local->message);
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    _LOGI("dhclient started with pid %lld", (long long int) pid);
Packit Service 5ffa24
Packit Service 5ffa24
    if (!release)
Packit Service 5ffa24
        nm_dhcp_client_watch_child(client, pid);
Packit Service 5ffa24
Packit Service 5ffa24
    priv->pid_file = g_steal_pointer(&pid_file);
Packit Service 5ffa24
Packit Service 5ffa24
    NM_SET_OUT(out_pid, pid);
Packit Service 5ffa24
    return TRUE;
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
    NMDhcpDhclient *       self = NM_DHCP_DHCLIENT(client);
Packit Service 5ffa24
    NMDhcpDhclientPrivate *priv = NM_DHCP_DHCLIENT_GET_PRIVATE(self);
Packit Service 5ffa24
    GBytes *               client_id;
Packit Service 5ffa24
    gs_unref_bytes GBytes *new_client_id = NULL;
Packit Service 5ffa24
Packit Service 5ffa24
    client_id = nm_dhcp_client_get_client_id(client);
Packit Service 5ffa24
Packit Service 5ffa24
    priv->conf_file = create_dhclient_config(self,
Packit Service 5ffa24
                                             AF_INET,
Packit Service 5ffa24
                                             nm_dhcp_client_get_iface(client),
Packit Service 5ffa24
                                             nm_dhcp_client_get_uuid(client),
Packit Service 5ffa24
                                             client_id,
Packit Service 5ffa24
                                             dhcp_anycast_addr,
Packit Service 5ffa24
                                             nm_dhcp_client_get_hostname(client),
Packit Service 5ffa24
                                             nm_dhcp_client_get_timeout(client),
Packit Service 5ffa24
                                             nm_dhcp_client_get_use_fqdn(client),
Packit Service 5ffa24
                                             nm_dhcp_client_get_hostname_flags(client),
Packit Service 5ffa24
                                             nm_dhcp_client_get_mud_url(client),
Packit Service 5ffa24
                                             nm_dhcp_client_get_reject_servers(client),
Packit Service 5ffa24
                                             &new_client_id);
Packit Service 5ffa24
    if (!priv->conf_file) {
Packit Service 5ffa24
        nm_utils_error_set_literal(error,
Packit Service 5ffa24
                                   NM_UTILS_ERROR_UNKNOWN,
Packit Service 5ffa24
                                   "error creating dhclient configuration file");
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (new_client_id) {
Packit Service 5ffa24
        nm_assert(!client_id);
Packit Service 5ffa24
        nm_dhcp_client_set_client_id(client, new_client_id);
Packit Service 5ffa24
    }
Packit Service 5ffa24
    return dhclient_start(client, NULL, FALSE, NULL, 0, error);
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
    NMDhcpDhclient *       self = NM_DHCP_DHCLIENT(client);
Packit Service 5ffa24
    NMDhcpDhclientPrivate *priv = NM_DHCP_DHCLIENT_GET_PRIVATE(self);
Packit Service 5ffa24
Packit Service 5ffa24
    if (nm_dhcp_client_get_iaid_explicit(client))
Packit Service 5ffa24
        _LOGW("dhclient does not support specifying an IAID for DHCPv6, it will be ignored");
Packit Service 5ffa24
Packit Service 5ffa24
    priv->conf_file = create_dhclient_config(self,
Packit Service 5ffa24
                                             AF_INET6,
Packit Service 5ffa24
                                             nm_dhcp_client_get_iface(client),
Packit Service 5ffa24
                                             nm_dhcp_client_get_uuid(client),
Packit Service 5ffa24
                                             NULL,
Packit Service 5ffa24
                                             dhcp_anycast_addr,
Packit Service 5ffa24
                                             nm_dhcp_client_get_hostname(client),
Packit Service 5ffa24
                                             nm_dhcp_client_get_timeout(client),
Packit Service 5ffa24
                                             TRUE,
Packit Service 5ffa24
                                             nm_dhcp_client_get_hostname_flags(client),
Packit Service 5ffa24
                                             nm_dhcp_client_get_mud_url(client),
Packit Service 5ffa24
                                             NULL,
Packit Service 5ffa24
                                             NULL);
Packit Service 5ffa24
    if (!priv->conf_file) {
Packit Service 5ffa24
        nm_utils_error_set_literal(error,
Packit Service 5ffa24
                                   NM_UTILS_ERROR_UNKNOWN,
Packit Service 5ffa24
                                   "error creating dhclient configuration file");
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    return dhclient_start(client,
Packit Service 5ffa24
                          nm_dhcp_client_get_info_only(NM_DHCP_CLIENT(self)) ? "-S" : "-N",
Packit Service 5ffa24
                          FALSE,
Packit Service 5ffa24
                          NULL,
Packit Service 5ffa24
                          needed_prefixes,
Packit Service 5ffa24
                          error);
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
    NMDhcpDhclient *       self = NM_DHCP_DHCLIENT(client);
Packit Service 5ffa24
    NMDhcpDhclientPrivate *priv = NM_DHCP_DHCLIENT_GET_PRIVATE(self);
Packit Service 5ffa24
    int                    errsv;
Packit Service 5ffa24
Packit Service 5ffa24
    NM_DHCP_CLIENT_CLASS(nm_dhcp_dhclient_parent_class)->stop(client, release);
Packit Service 5ffa24
Packit Service 5ffa24
    if (priv->conf_file)
Packit Service 5ffa24
        if (remove(priv->conf_file) == -1) {
Packit Service 5ffa24
            errsv = errno;
Packit Service 5ffa24
            _LOGD("could not remove dhcp config file \"%s\": %d (%s)",
Packit Service 5ffa24
                  priv->conf_file,
Packit Service 5ffa24
                  errsv,
Packit Service 5ffa24
                  nm_strerror_native(errsv));
Packit Service 5ffa24
        }
Packit Service 5ffa24
    if (priv->pid_file) {
Packit Service 5ffa24
        if (remove(priv->pid_file) == -1) {
Packit Service 5ffa24
            errsv = errno;
Packit Service 5ffa24
            _LOGD("could not remove dhcp pid file \"%s\": %s (%d)",
Packit Service 5ffa24
                  priv->pid_file,
Packit Service 5ffa24
                  nm_strerror_native(errsv),
Packit Service 5ffa24
                  errsv);
Packit Service 5ffa24
        }
Packit Service 5ffa24
        nm_clear_g_free(&priv->pid_file);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (release) {
Packit Service 5ffa24
        pid_t rpid = -1;
Packit Service 5ffa24
Packit Service 5ffa24
        if (dhclient_start(client, NULL, TRUE, &rpid, 0, NULL)) {
Packit Service 5ffa24
            /* Wait a few seconds for the release to happen */
Packit Service 5ffa24
            nm_dhcp_client_stop_pid(rpid, nm_dhcp_client_get_iface(client));
Packit Service 5ffa24
        }
Packit Service 5ffa24
    }
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static GBytes *
Packit Service 5ffa24
get_duid(NMDhcpClient *client)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMDhcpDhclient *       self      = NM_DHCP_DHCLIENT(client);
Packit Service 5ffa24
    NMDhcpDhclientPrivate *priv      = NM_DHCP_DHCLIENT_GET_PRIVATE(self);
Packit Service 5ffa24
    GBytes *               duid      = NULL;
Packit Service 5ffa24
    gs_free char *         leasefile = NULL;
Packit Service 5ffa24
    GError *               error     = NULL;
Packit Service 5ffa24
Packit Service 5ffa24
    /* Look in interface-specific leasefile first for backwards compat */
Packit Service 5ffa24
    leasefile = get_dhclient_leasefile(AF_INET6,
Packit Service 5ffa24
                                       nm_dhcp_client_get_iface(client),
Packit Service 5ffa24
                                       nm_dhcp_client_get_uuid(client),
Packit Service 5ffa24
                                       NULL);
Packit Service 5ffa24
    if (leasefile) {
Packit Service 5ffa24
        _LOGD("looking for DUID in '%s'", leasefile);
Packit Service 5ffa24
        duid = nm_dhcp_dhclient_read_duid(leasefile, &error);
Packit Service 5ffa24
        if (error) {
Packit Service 5ffa24
            _LOGW("failed to read leasefile '%s': %s", leasefile, error->message);
Packit Service 5ffa24
            g_clear_error(&error);
Packit Service 5ffa24
        }
Packit Service 5ffa24
        if (duid)
Packit Service 5ffa24
            return duid;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    /* Otherwise, read the default machine-wide DUID */
Packit Service 5ffa24
    _LOGD("looking for default DUID in '%s'", priv->def_leasefile);
Packit Service 5ffa24
    duid = nm_dhcp_dhclient_read_duid(priv->def_leasefile, &error);
Packit Service 5ffa24
    if (error) {
Packit Service 5ffa24
        _LOGW("failed to read leasefile '%s': %s", priv->def_leasefile, error->message);
Packit Service 5ffa24
        g_clear_error(&error);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    return duid;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
nm_dhcp_dhclient_init(NMDhcpDhclient *self)
Packit Service 5ffa24
{
Packit Service 5ffa24
    static const char *const FILES[] = {
Packit Service 5ffa24
        SYSCONFDIR "/dhclient6.leases", /* default */
Packit Service 5ffa24
        LOCALSTATEDIR "/lib/dhcp/dhclient6.leases",
Packit Service 5ffa24
        LOCALSTATEDIR "/lib/dhclient/dhclient6.leases",
Packit Service 5ffa24
    };
Packit Service 5ffa24
    NMDhcpDhclientPrivate *priv = NM_DHCP_DHCLIENT_GET_PRIVATE(self);
Packit Service 5ffa24
    int                    i;
Packit Service 5ffa24
Packit Service 5ffa24
    priv->def_leasefile = FILES[0];
Packit Service 5ffa24
    for (i = 0; i < G_N_ELEMENTS(FILES); i++) {
Packit Service 5ffa24
        if (g_file_test(FILES[i], G_FILE_TEST_EXISTS)) {
Packit Service 5ffa24
            priv->def_leasefile = FILES[i];
Packit Service 5ffa24
            break;
Packit Service 5ffa24
        }
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    priv->dhcp_listener = g_object_ref(nm_dhcp_listener_get());
Packit Service 5ffa24
    g_signal_connect(priv->dhcp_listener,
Packit Service 5ffa24
                     NM_DHCP_LISTENER_EVENT,
Packit Service 5ffa24
                     G_CALLBACK(nm_dhcp_client_handle_event),
Packit Service 5ffa24
                     self);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
dispose(GObject *object)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMDhcpDhclientPrivate *priv = NM_DHCP_DHCLIENT_GET_PRIVATE(object);
Packit Service 5ffa24
Packit Service 5ffa24
    if (priv->dhcp_listener) {
Packit Service 5ffa24
        g_signal_handlers_disconnect_by_func(priv->dhcp_listener,
Packit Service 5ffa24
                                             G_CALLBACK(nm_dhcp_client_handle_event),
Packit Service 5ffa24
                                             NM_DHCP_DHCLIENT(object));
Packit Service 5ffa24
        g_clear_object(&priv->dhcp_listener);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    nm_clear_g_free(&priv->pid_file);
Packit Service 5ffa24
    nm_clear_g_free(&priv->conf_file);
Packit Service 5ffa24
    nm_clear_g_free(&priv->lease_file);
Packit Service 5ffa24
Packit Service 5ffa24
    G_OBJECT_CLASS(nm_dhcp_dhclient_parent_class)->dispose(object);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
nm_dhcp_dhclient_class_init(NMDhcpDhclientClass *dhclient_class)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMDhcpClientClass *client_class = NM_DHCP_CLIENT_CLASS(dhclient_class);
Packit Service 5ffa24
    GObjectClass *     object_class = G_OBJECT_CLASS(dhclient_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
    client_class->get_duid  = get_duid;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
const NMDhcpClientFactory _nm_dhcp_client_factory_dhclient = {
Packit Service 5ffa24
    .name     = "dhclient",
Packit Service 5ffa24
    .get_type = nm_dhcp_dhclient_get_type,
Packit Service 5ffa24
    .get_path = nm_dhcp_dhclient_get_path,
Packit Service 5ffa24
};
Packit Service 5ffa24
Packit Service 5ffa24
#endif /* WITH_DHCLIENT */