Blame src/core/ppp/nm-ppp-manager.c

Packit Service 5ffa24
/* SPDX-License-Identifier: GPL-2.0-or-later */
Packit Service 5ffa24
/*
Packit Service 5ffa24
 * Copyright (C) 2008 Novell, Inc.
Packit Service 5ffa24
 * Copyright (C) 2008 - 2012 Red Hat, Inc.
Packit Service 5ffa24
 */
Packit Service 5ffa24
Packit Service 5ffa24
#include "nm-default.h"
Packit Service 5ffa24
Packit Service 5ffa24
#include "nm-ppp-manager.h"
Packit Service 5ffa24
Packit Service 5ffa24
#include <sys/types.h>
Packit Service 5ffa24
#include <sys/wait.h>
Packit Service 5ffa24
#include <signal.h>
Packit Service 5ffa24
#include <unistd.h>
Packit Service 5ffa24
#include <arpa/inet.h>
Packit Service 5ffa24
#include <stdlib.h>
Packit Service 5ffa24
#include <sys/socket.h>
Packit Service 5ffa24
#include <sys/ioctl.h>
Packit Service 5ffa24
#include <asm/types.h>
Packit Service 5ffa24
#include <sys/stat.h>
Packit Service 5ffa24
Packit Service 5ffa24
#include <linux/ppp_defs.h>
Packit Service 5ffa24
#ifndef aligned_u64
Packit Service 5ffa24
    #define aligned_u64 unsigned long long __attribute__((aligned(8)))
Packit Service 5ffa24
#endif
Packit Service 5ffa24
#include <linux/if.h>
Packit Service 5ffa24
#include <linux/if_ppp.h>
Packit Service 5ffa24
#include <linux/rtnetlink.h>
Packit Service 5ffa24
Packit Service 5ffa24
#include "NetworkManagerUtils.h"
Packit Service 5ffa24
#include "platform/nm-platform.h"
Packit Service 5ffa24
#include "nm-core-internal.h"
Packit Service 5ffa24
#include "nm-act-request.h"
Packit Service 5ffa24
#include "nm-ip4-config.h"
Packit Service 5ffa24
#include "nm-ip6-config.h"
Packit Service 5ffa24
#include "nm-dbus-object.h"
Packit Service 5ffa24
Packit Service 5ffa24
#include "nm-pppd-plugin.h"
Packit Service 5ffa24
#include "nm-ppp-plugin-api.h"
Packit Service 5ffa24
#include "nm-ppp-status.h"
Packit Service 5ffa24
Packit Service 5ffa24
#define NM_PPPD_PLUGIN PPPD_PLUGIN_DIR "/nm-pppd-plugin.so"
Packit Service 5ffa24
Packit Service 5ffa24
static NM_CACHED_QUARK_FCN("ppp-manager-secret-tries", ppp_manager_secret_tries_quark);
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
#define NM_TYPE_PPP_MANAGER (nm_ppp_manager_get_type())
Packit Service 5ffa24
#define NM_PPP_MANAGER(obj) (G_TYPE_CHECK_INSTANCE_CAST((obj), NM_TYPE_PPP_MANAGER, NMPPPManager))
Packit Service 5ffa24
#define NM_PPP_MANAGER_CLASS(klass) \
Packit Service 5ffa24
    (G_TYPE_CHECK_CLASS_CAST((klass), NM_TYPE_PPP_MANAGER, NMPPPManagerClass))
Packit Service 5ffa24
#define NM_IS_PPP_MANAGER(obj)         (G_TYPE_CHECK_INSTANCE_TYPE((obj), NM_TYPE_PPP_MANAGER))
Packit Service 5ffa24
#define NM_IS_PPP_MANAGER_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE((klass), NM_TYPE_PPP_MANAGER))
Packit Service 5ffa24
#define NM_PPP_MANAGER_GET_CLASS(obj) \
Packit Service 5ffa24
    (G_TYPE_INSTANCE_GET_CLASS((obj), NM_TYPE_PPP_MANAGER, NMPPPManagerClass))
Packit Service 5ffa24
Packit Service 5ffa24
GType nm_ppp_manager_get_type(void);
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
enum {
Packit Service 5ffa24
    STATE_CHANGED,
Packit Service 5ffa24
    IFINDEX_SET,
Packit Service 5ffa24
    IP4_CONFIG,
Packit Service 5ffa24
    IP6_CONFIG,
Packit Service 5ffa24
    STATS,
Packit Service 5ffa24
Packit Service 5ffa24
    LAST_SIGNAL
Packit Service 5ffa24
};
Packit Service 5ffa24
Packit Service 5ffa24
static guint signals[LAST_SIGNAL] = {0};
Packit Service 5ffa24
Packit Service 5ffa24
NM_GOBJECT_PROPERTIES_DEFINE_BASE(PROP_PARENT_IFACE, );
Packit Service 5ffa24
Packit Service 5ffa24
typedef struct {
Packit Service 5ffa24
    GPid pid;
Packit Service 5ffa24
Packit Service 5ffa24
    char *parent_iface;
Packit Service 5ffa24
    char *ip_iface;
Packit Service 5ffa24
    int   ifindex;
Packit Service 5ffa24
Packit Service 5ffa24
    NMActRequest *                act_req;
Packit Service 5ffa24
    GDBusMethodInvocation *       pending_secrets_context;
Packit Service 5ffa24
    NMActRequestGetSecretsCallId *secrets_id;
Packit Service 5ffa24
    const char *                  secrets_setting_name;
Packit Service 5ffa24
Packit Service 5ffa24
    guint ppp_watch_id;
Packit Service 5ffa24
    guint ppp_timeout_handler;
Packit Service 5ffa24
Packit Service 5ffa24
    /* Monitoring */
Packit Service 5ffa24
    int   monitor_fd;
Packit Service 5ffa24
    guint monitor_id;
Packit Service 5ffa24
Packit Service 5ffa24
    guint32 ip4_route_table;
Packit Service 5ffa24
    guint32 ip4_route_metric;
Packit Service 5ffa24
    guint32 ip6_route_table;
Packit Service 5ffa24
    guint32 ip6_route_metric;
Packit Service 5ffa24
} NMPPPManagerPrivate;
Packit Service 5ffa24
Packit Service 5ffa24
struct _NMPPPManager {
Packit Service 5ffa24
    NMDBusObject        parent;
Packit Service 5ffa24
    NMPPPManagerPrivate _priv;
Packit Service 5ffa24
};
Packit Service 5ffa24
Packit Service 5ffa24
typedef struct {
Packit Service 5ffa24
    NMDBusObjectClass parent;
Packit Service 5ffa24
} NMPPPManagerClass;
Packit Service 5ffa24
Packit Service 5ffa24
G_DEFINE_TYPE(NMPPPManager, nm_ppp_manager, NM_TYPE_DBUS_OBJECT)
Packit Service 5ffa24
Packit Service 5ffa24
#define NM_PPP_MANAGER_GET_PRIVATE(self) \
Packit Service 5ffa24
    _NM_GET_PRIVATE(self, NMPPPManager, NM_IS_PPP_MANAGER, NMDBusObject)
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
#define _NMLOG_DOMAIN      LOGD_PPP
Packit Service 5ffa24
#define _NMLOG(level, ...) __NMLOG_DEFAULT(level, _NMLOG_DOMAIN, "ppp-manager", __VA_ARGS__)
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
static void _ppp_cleanup(NMPPPManager *self);
Packit Service 5ffa24
Packit Service 5ffa24
static NMPPPManagerStopHandle *_ppp_manager_stop(NMPPPManager *           self,
Packit Service 5ffa24
                                                 GCancellable *           cancellable,
Packit Service 5ffa24
                                                 NMPPPManagerStopCallback callback,
Packit Service 5ffa24
                                                 gpointer                 user_data);
Packit Service 5ffa24
Packit Service 5ffa24
static void _ppp_manager_stop_cancel(NMPPPManagerStopHandle *handle);
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
_ppp_manager_set_route_parameters(NMPPPManager *self,
Packit Service 5ffa24
                                  guint32       ip4_route_table,
Packit Service 5ffa24
                                  guint32       ip4_route_metric,
Packit Service 5ffa24
                                  guint32       ip6_route_table,
Packit Service 5ffa24
                                  guint32       ip6_route_metric)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMPPPManagerPrivate *priv;
Packit Service 5ffa24
Packit Service 5ffa24
    g_return_if_fail(NM_IS_PPP_MANAGER(self));
Packit Service 5ffa24
Packit Service 5ffa24
    priv = NM_PPP_MANAGER_GET_PRIVATE(self);
Packit Service 5ffa24
    if (priv->ip4_route_table != ip4_route_table || priv->ip4_route_metric != ip4_route_metric
Packit Service 5ffa24
        || priv->ip6_route_table != ip6_route_table || priv->ip6_route_metric != ip6_route_metric) {
Packit Service 5ffa24
        priv->ip4_route_table  = ip4_route_table;
Packit Service 5ffa24
        priv->ip4_route_metric = ip4_route_metric;
Packit Service 5ffa24
        priv->ip6_route_table  = ip6_route_table;
Packit Service 5ffa24
        priv->ip6_route_metric = ip6_route_metric;
Packit Service 5ffa24
Packit Service 5ffa24
        _LOGT("route-parameters: table-v4: %u, metric-v4: %u, table-v6: %u, metric-v6: %u",
Packit Service 5ffa24
              priv->ip4_route_table,
Packit Service 5ffa24
              priv->ip4_route_metric,
Packit Service 5ffa24
              priv->ip6_route_table,
Packit Service 5ffa24
              priv->ip6_route_metric);
Packit Service 5ffa24
    }
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
static gboolean
Packit Service 5ffa24
monitor_cb(gpointer user_data)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMPPPManager *       self = NM_PPP_MANAGER(user_data);
Packit Service 5ffa24
    NMPPPManagerPrivate *priv = NM_PPP_MANAGER_GET_PRIVATE(self);
Packit Service 5ffa24
    const char *         ifname;
Packit Service 5ffa24
    int                  errsv;
Packit Service 5ffa24
Packit Service 5ffa24
    ifname = nm_platform_link_get_name(NM_PLATFORM_GET, priv->ifindex);
Packit Service 5ffa24
Packit Service 5ffa24
    if (ifname) {
Packit Service 5ffa24
        struct ppp_stats stats = {};
Packit Service 5ffa24
        struct ifreq     req   = {
Packit Service 5ffa24
            .ifr_data = (caddr_t) &stats,
Packit Service 5ffa24
        };
Packit Service 5ffa24
Packit Service 5ffa24
        nm_utils_ifname_cpy(req.ifr_name, ifname);
Packit Service 5ffa24
        if (ioctl(priv->monitor_fd, SIOCGPPPSTATS, &req) < 0) {
Packit Service 5ffa24
            errsv = errno;
Packit Service 5ffa24
            if (errsv != ENODEV)
Packit Service 5ffa24
                _LOGW("could not read ppp stats: %s", nm_strerror_native(errsv));
Packit Service 5ffa24
        } else {
Packit Service 5ffa24
            g_signal_emit(self,
Packit Service 5ffa24
                          signals[STATS],
Packit Service 5ffa24
                          0,
Packit Service 5ffa24
                          (guint) stats.p.ppp_ibytes,
Packit Service 5ffa24
                          (guint) stats.p.ppp_obytes);
Packit Service 5ffa24
        }
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    return G_SOURCE_CONTINUE;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
monitor_stats(NMPPPManager *self)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMPPPManagerPrivate *priv = NM_PPP_MANAGER_GET_PRIVATE(self);
Packit Service 5ffa24
    int                  errsv;
Packit Service 5ffa24
Packit Service 5ffa24
    /* already monitoring */
Packit Service 5ffa24
    if (priv->monitor_fd >= 0)
Packit Service 5ffa24
        return;
Packit Service 5ffa24
Packit Service 5ffa24
    priv->monitor_fd = socket(AF_INET, SOCK_DGRAM | SOCK_CLOEXEC, 0);
Packit Service 5ffa24
    if (priv->monitor_fd < 0) {
Packit Service 5ffa24
        errsv = errno;
Packit Service 5ffa24
        _LOGW("could not monitor PPP stats: %s", nm_strerror_native(errsv));
Packit Service 5ffa24
        return;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    g_warn_if_fail(priv->monitor_id == 0);
Packit Service 5ffa24
    if (priv->monitor_id)
Packit Service 5ffa24
        g_source_remove(priv->monitor_id);
Packit Service 5ffa24
    priv->monitor_id = g_timeout_add_seconds(5, monitor_cb, self);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
cancel_get_secrets(NMPPPManager *self)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMPPPManagerPrivate *priv = NM_PPP_MANAGER_GET_PRIVATE(self);
Packit Service 5ffa24
Packit Service 5ffa24
    if (priv->secrets_id)
Packit Service 5ffa24
        nm_act_request_cancel_secrets(priv->act_req, priv->secrets_id);
Packit Service 5ffa24
Packit Service 5ffa24
    g_return_if_fail(!priv->secrets_id && !priv->secrets_setting_name);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static gboolean
Packit Service 5ffa24
extract_details_from_connection(NMConnection *connection,
Packit Service 5ffa24
                                const char *  secrets_setting_name,
Packit Service 5ffa24
                                const char ** username,
Packit Service 5ffa24
                                const char ** password,
Packit Service 5ffa24
                                GError **     error)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMSettingConnection *s_con;
Packit Service 5ffa24
    NMSetting *          setting;
Packit Service 5ffa24
    const char *         setting_name;
Packit Service 5ffa24
Packit Service 5ffa24
    g_return_val_if_fail(connection != NULL, FALSE);
Packit Service 5ffa24
    g_return_val_if_fail(username != NULL, FALSE);
Packit Service 5ffa24
    g_return_val_if_fail(password != NULL, FALSE);
Packit Service 5ffa24
Packit Service 5ffa24
    if (secrets_setting_name)
Packit Service 5ffa24
        setting_name = secrets_setting_name;
Packit Service 5ffa24
    else {
Packit Service 5ffa24
        /* Get the setting matching the connection type */
Packit Service 5ffa24
        s_con = nm_connection_get_setting_connection(connection);
Packit Service 5ffa24
        g_assert(s_con);
Packit Service 5ffa24
Packit Service 5ffa24
        setting_name = nm_setting_connection_get_connection_type(s_con);
Packit Service 5ffa24
        g_assert(setting_name);
Packit Service 5ffa24
Packit Service 5ffa24
        /* In case of bluetooth connection, use GSM or CDMA setting */
Packit Service 5ffa24
        if (strcmp(setting_name, NM_SETTING_BLUETOOTH_SETTING_NAME) == 0) {
Packit Service 5ffa24
            if (nm_connection_get_setting_gsm(connection))
Packit Service 5ffa24
                setting_name = NM_SETTING_GSM_SETTING_NAME;
Packit Service 5ffa24
            else
Packit Service 5ffa24
                setting_name = NM_SETTING_CDMA_SETTING_NAME;
Packit Service 5ffa24
        }
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    setting = nm_connection_get_setting_by_name(connection, setting_name);
Packit Service 5ffa24
    if (!setting) {
Packit Service 5ffa24
        /* This shouldn't ever happen */
Packit Service 5ffa24
        g_set_error_literal(error,
Packit Service 5ffa24
                            NM_MANAGER_ERROR,
Packit Service 5ffa24
                            NM_MANAGER_ERROR_FAILED,
Packit Service 5ffa24
                            "Missing type-specific setting; no secrets could be found.");
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (NM_IS_SETTING_PPPOE(setting)) {
Packit Service 5ffa24
        *username = nm_setting_pppoe_get_username(NM_SETTING_PPPOE(setting));
Packit Service 5ffa24
        *password = nm_setting_pppoe_get_password(NM_SETTING_PPPOE(setting));
Packit Service 5ffa24
    } else if (NM_IS_SETTING_ADSL(setting)) {
Packit Service 5ffa24
        *username = nm_setting_adsl_get_username(NM_SETTING_ADSL(setting));
Packit Service 5ffa24
        *password = nm_setting_adsl_get_password(NM_SETTING_ADSL(setting));
Packit Service 5ffa24
    } else if (NM_IS_SETTING_GSM(setting)) {
Packit Service 5ffa24
        *username = nm_setting_gsm_get_username(NM_SETTING_GSM(setting));
Packit Service 5ffa24
        *password = nm_setting_gsm_get_password(NM_SETTING_GSM(setting));
Packit Service 5ffa24
    } else if (NM_IS_SETTING_CDMA(setting)) {
Packit Service 5ffa24
        *username = nm_setting_cdma_get_username(NM_SETTING_CDMA(setting));
Packit Service 5ffa24
        *password = nm_setting_cdma_get_password(NM_SETTING_CDMA(setting));
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    return TRUE;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
ppp_secrets_cb(NMActRequest *                req,
Packit Service 5ffa24
               NMActRequestGetSecretsCallId *call_id,
Packit Service 5ffa24
               NMSettingsConnection *        settings_connection, /* unused (we pass NULL here) */
Packit Service 5ffa24
               GError *                      error,
Packit Service 5ffa24
               gpointer                      user_data)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMPPPManager *       self     = NM_PPP_MANAGER(user_data);
Packit Service 5ffa24
    NMPPPManagerPrivate *priv     = NM_PPP_MANAGER_GET_PRIVATE(self);
Packit Service 5ffa24
    const char *         username = NULL;
Packit Service 5ffa24
    const char *         password = NULL;
Packit Service 5ffa24
    GError *             local    = NULL;
Packit Service 5ffa24
    NMConnection *       applied_connection;
Packit Service 5ffa24
Packit Service 5ffa24
    g_return_if_fail(priv->pending_secrets_context != NULL);
Packit Service 5ffa24
    g_return_if_fail(req == priv->act_req);
Packit Service 5ffa24
    g_return_if_fail(call_id == priv->secrets_id);
Packit Service 5ffa24
Packit Service 5ffa24
    if (g_error_matches(error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
Packit Service 5ffa24
        goto out;
Packit Service 5ffa24
Packit Service 5ffa24
    if (error) {
Packit Service 5ffa24
        _LOGW("%s", error->message);
Packit Service 5ffa24
        g_dbus_method_invocation_return_gerror(priv->pending_secrets_context, error);
Packit Service 5ffa24
        goto out;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    applied_connection = nm_act_request_get_applied_connection(req);
Packit Service 5ffa24
Packit Service 5ffa24
    if (!extract_details_from_connection(applied_connection,
Packit Service 5ffa24
                                         priv->secrets_setting_name,
Packit Service 5ffa24
                                         &username,
Packit Service 5ffa24
                                         &password,
Packit Service 5ffa24
                                         &local)) {
Packit Service 5ffa24
        _LOGW("%s", local->message);
Packit Service 5ffa24
        g_dbus_method_invocation_take_error(priv->pending_secrets_context, local);
Packit Service 5ffa24
        goto out;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    /* This is sort of a hack but...
Packit Service 5ffa24
     * pppd plugin only ever needs username and password. Passing the full
Packit Service 5ffa24
     * connection there would mean some bloat: the plugin would need to link
Packit Service 5ffa24
     * against libnm just to parse this. So instead, let's just send what
Packit Service 5ffa24
     * it needs.
Packit Service 5ffa24
     */
Packit Service 5ffa24
    g_dbus_method_invocation_return_value(priv->pending_secrets_context,
Packit Service 5ffa24
                                          g_variant_new("(ss)", username ?: "", password ?: ""));
Packit Service 5ffa24
Packit Service 5ffa24
out:
Packit Service 5ffa24
    priv->pending_secrets_context = NULL;
Packit Service 5ffa24
    priv->secrets_id              = NULL;
Packit Service 5ffa24
    priv->secrets_setting_name    = NULL;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
impl_ppp_manager_need_secrets(NMDBusObject *                     obj,
Packit Service 5ffa24
                              const NMDBusInterfaceInfoExtended *interface_info,
Packit Service 5ffa24
                              const NMDBusMethodInfoExtended *   method_info,
Packit Service 5ffa24
                              GDBusConnection *                  connection,
Packit Service 5ffa24
                              const char *                       sender,
Packit Service 5ffa24
                              GDBusMethodInvocation *            invocation,
Packit Service 5ffa24
                              GVariant *                         parameters)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMPPPManager *       self = NM_PPP_MANAGER(obj);
Packit Service 5ffa24
    NMPPPManagerPrivate *priv = NM_PPP_MANAGER_GET_PRIVATE(self);
Packit Service 5ffa24
    NMConnection *       applied_connection;
Packit Service 5ffa24
    const char *         username = NULL;
Packit Service 5ffa24
    const char *         password = NULL;
Packit Service 5ffa24
    guint32              tries;
Packit Service 5ffa24
    gs_unref_ptrarray GPtrArray *hints = NULL;
Packit Service 5ffa24
    GError *                     error = NULL;
Packit Service 5ffa24
    NMSecretAgentGetSecretsFlags flags = NM_SECRET_AGENT_GET_SECRETS_FLAG_ALLOW_INTERACTION;
Packit Service 5ffa24
Packit Service 5ffa24
    nm_active_connection_clear_secrets(NM_ACTIVE_CONNECTION(priv->act_req));
Packit Service 5ffa24
Packit Service 5ffa24
    applied_connection = nm_act_request_get_applied_connection(priv->act_req);
Packit Service 5ffa24
Packit Service 5ffa24
    priv->secrets_setting_name = nm_connection_need_secrets(applied_connection, &hints);
Packit Service 5ffa24
    if (!priv->secrets_setting_name) {
Packit Service 5ffa24
        /* Use existing secrets from the connection */
Packit Service 5ffa24
        if (extract_details_from_connection(applied_connection,
Packit Service 5ffa24
                                            NULL,
Packit Service 5ffa24
                                            &username,
Packit Service 5ffa24
                                            &password,
Packit Service 5ffa24
                                            &error)) {
Packit Service 5ffa24
            /* Send existing secrets to the PPP plugin */
Packit Service 5ffa24
            priv->pending_secrets_context = invocation;
Packit Service 5ffa24
            ppp_secrets_cb(priv->act_req, priv->secrets_id, NULL, NULL, self);
Packit Service 5ffa24
        } else {
Packit Service 5ffa24
            _LOGW("%s", error->message);
Packit Service 5ffa24
            g_dbus_method_invocation_take_error(priv->pending_secrets_context, error);
Packit Service 5ffa24
        }
Packit Service 5ffa24
        return;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    /* Only ask for completely new secrets after retrying them once; some devices
Packit Service 5ffa24
     * appear to ask a few times when they actually don't even care what you
Packit Service 5ffa24
     * pass back.
Packit Service 5ffa24
     */
Packit Service 5ffa24
    tries = GPOINTER_TO_UINT(
Packit Service 5ffa24
        g_object_get_qdata(G_OBJECT(applied_connection), ppp_manager_secret_tries_quark()));
Packit Service 5ffa24
    if (tries > 1)
Packit Service 5ffa24
        flags |= NM_SECRET_AGENT_GET_SECRETS_FLAG_REQUEST_NEW;
Packit Service 5ffa24
Packit Service 5ffa24
    if (hints)
Packit Service 5ffa24
        g_ptr_array_add(hints, NULL);
Packit Service 5ffa24
Packit Service 5ffa24
    priv->secrets_id = nm_act_request_get_secrets(priv->act_req,
Packit Service 5ffa24
                                                  FALSE,
Packit Service 5ffa24
                                                  priv->secrets_setting_name,
Packit Service 5ffa24
                                                  flags,
Packit Service 5ffa24
                                                  hints ? (const char *const *) hints->pdata : NULL,
Packit Service 5ffa24
                                                  ppp_secrets_cb,
Packit Service 5ffa24
                                                  self);
Packit Service 5ffa24
    g_object_set_qdata(G_OBJECT(applied_connection),
Packit Service 5ffa24
                       ppp_manager_secret_tries_quark(),
Packit Service 5ffa24
                       GUINT_TO_POINTER(++tries));
Packit Service 5ffa24
    priv->pending_secrets_context = invocation;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
impl_ppp_manager_set_state(NMDBusObject *                     obj,
Packit Service 5ffa24
                           const NMDBusInterfaceInfoExtended *interface_info,
Packit Service 5ffa24
                           const NMDBusMethodInfoExtended *   method_info,
Packit Service 5ffa24
                           GDBusConnection *                  connection,
Packit Service 5ffa24
                           const char *                       sender,
Packit Service 5ffa24
                           GDBusMethodInvocation *            invocation,
Packit Service 5ffa24
                           GVariant *                         parameters)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMPPPManager *self = NM_PPP_MANAGER(obj);
Packit Service 5ffa24
    guint32       state;
Packit Service 5ffa24
Packit Service 5ffa24
    g_variant_get(parameters, "(u)", &state);
Packit Service 5ffa24
    g_signal_emit(self, signals[STATE_CHANGED], 0, (guint) state);
Packit Service 5ffa24
    g_dbus_method_invocation_return_value(invocation, NULL);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
impl_ppp_manager_set_ifindex(NMDBusObject *                     obj,
Packit Service 5ffa24
                             const NMDBusInterfaceInfoExtended *interface_info,
Packit Service 5ffa24
                             const NMDBusMethodInfoExtended *   method_info,
Packit Service 5ffa24
                             GDBusConnection *                  connection,
Packit Service 5ffa24
                             const char *                       sender,
Packit Service 5ffa24
                             GDBusMethodInvocation *            invocation,
Packit Service 5ffa24
                             GVariant *                         parameters)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMPPPManager *        self                     = NM_PPP_MANAGER(obj);
Packit Service 5ffa24
    NMPPPManagerPrivate * priv                     = NM_PPP_MANAGER_GET_PRIVATE(self);
Packit Service 5ffa24
    const NMPlatformLink *plink                    = NULL;
Packit Service 5ffa24
    nm_auto_nmpobj const NMPObject *obj_keep_alive = NULL;
Packit Service 5ffa24
    gint32                          ifindex;
Packit Service 5ffa24
Packit Service 5ffa24
    g_variant_get(parameters, "(i)", &ifindex);
Packit Service 5ffa24
Packit Service 5ffa24
    if (priv->ifindex >= 0) {
Packit Service 5ffa24
        if (priv->ifindex == ifindex)
Packit Service 5ffa24
            _LOGD("set-ifindex: ignore repeated calls setting ifindex to %d", (int) ifindex);
Packit Service 5ffa24
        else
Packit Service 5ffa24
            _LOGW("set-ifindex: can't change the ifindex from %d to %d",
Packit Service 5ffa24
                  priv->ifindex,
Packit Service 5ffa24
                  (int) ifindex);
Packit Service 5ffa24
        goto out;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (ifindex > 0) {
Packit Service 5ffa24
        plink = nm_platform_link_get(NM_PLATFORM_GET, ifindex);
Packit Service 5ffa24
        if (!plink) {
Packit Service 5ffa24
            nm_platform_process_events(NM_PLATFORM_GET);
Packit Service 5ffa24
            plink = nm_platform_link_get(NM_PLATFORM_GET, ifindex);
Packit Service 5ffa24
        }
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (!plink) {
Packit Service 5ffa24
        _LOGW("set-ifindex: unknown interface with ifindex %d", ifindex);
Packit Service 5ffa24
        ifindex = 0;
Packit Service 5ffa24
    } else {
Packit Service 5ffa24
        obj_keep_alive = nmp_object_ref(NMP_OBJECT_UP_CAST(plink));
Packit Service 5ffa24
        _LOGD("set-ifindex: %d, name \"%s\"", (int) ifindex, plink->name);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    priv->ifindex = ifindex;
Packit Service 5ffa24
Packit Service 5ffa24
    g_signal_emit(self, signals[IFINDEX_SET], 0, ifindex, plink ? plink->name : NULL);
Packit Service 5ffa24
Packit Service 5ffa24
out:
Packit Service 5ffa24
    g_dbus_method_invocation_return_value(invocation, NULL);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static gboolean
Packit Service 5ffa24
set_ip_config_common(NMPPPManager *self, GVariant *config_dict, guint32 *out_mtu)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMPPPManagerPrivate *priv = NM_PPP_MANAGER_GET_PRIVATE(self);
Packit Service 5ffa24
    NMConnection *       applied_connection;
Packit Service 5ffa24
    NMSettingPpp *       s_ppp;
Packit Service 5ffa24
Packit Service 5ffa24
    if (priv->ifindex <= 0)
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
Packit Service 5ffa24
    /* Got successful IP config; obviously the secrets worked */
Packit Service 5ffa24
    applied_connection = nm_act_request_get_applied_connection(priv->act_req);
Packit Service 5ffa24
    g_object_set_qdata(G_OBJECT(applied_connection), ppp_manager_secret_tries_quark(), NULL);
Packit Service 5ffa24
Packit Service 5ffa24
    if (out_mtu) {
Packit Service 5ffa24
        /* Get any custom MTU */
Packit Service 5ffa24
        s_ppp    = nm_connection_get_setting_ppp(applied_connection);
Packit Service 5ffa24
        *out_mtu = s_ppp ? nm_setting_ppp_get_mtu(s_ppp) : 0;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    monitor_stats(self);
Packit Service 5ffa24
    return TRUE;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
impl_ppp_manager_set_ip4_config(NMDBusObject *                     obj,
Packit Service 5ffa24
                                const NMDBusInterfaceInfoExtended *interface_info,
Packit Service 5ffa24
                                const NMDBusMethodInfoExtended *   method_info,
Packit Service 5ffa24
                                GDBusConnection *                  connection,
Packit Service 5ffa24
                                const char *                       sender,
Packit Service 5ffa24
                                GDBusMethodInvocation *            invocation,
Packit Service 5ffa24
                                GVariant *                         parameters)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMPPPManager *       self           = NM_PPP_MANAGER(obj);
Packit Service 5ffa24
    NMPPPManagerPrivate *priv           = NM_PPP_MANAGER_GET_PRIVATE(self);
Packit Service 5ffa24
    gs_unref_object NMIP4Config *config = NULL;
Packit Service 5ffa24
    NMPlatformIP4Address         address;
Packit Service 5ffa24
    guint32                      u32, mtu;
Packit Service 5ffa24
    GVariantIter *               iter;
Packit Service 5ffa24
    gs_unref_variant GVariant *config_dict = NULL;
Packit Service 5ffa24
Packit Service 5ffa24
    _LOGI("(IPv4 Config Get) reply received.");
Packit Service 5ffa24
Packit Service 5ffa24
    g_variant_get(parameters, "(@a{sv})", &config_dict);
Packit Service 5ffa24
Packit Service 5ffa24
    nm_clear_g_source(&priv->ppp_timeout_handler);
Packit Service 5ffa24
Packit Service 5ffa24
    if (!set_ip_config_common(self, config_dict, &mtu))
Packit Service 5ffa24
        goto out;
Packit Service 5ffa24
Packit Service 5ffa24
    config = nm_ip4_config_new(nm_platform_get_multi_idx(NM_PLATFORM_GET), priv->ifindex);
Packit Service 5ffa24
Packit Service 5ffa24
    if (mtu)
Packit Service 5ffa24
        nm_ip4_config_set_mtu(config, mtu, NM_IP_CONFIG_SOURCE_PPP);
Packit Service 5ffa24
Packit Service 5ffa24
    memset(&address, 0, sizeof(address));
Packit Service 5ffa24
    address.plen = 32;
Packit Service 5ffa24
Packit Service 5ffa24
    if (g_variant_lookup(config_dict, NM_PPP_IP4_CONFIG_ADDRESS, "u", &u32))
Packit Service 5ffa24
        address.address = u32;
Packit Service 5ffa24
Packit Service 5ffa24
    if (g_variant_lookup(config_dict, NM_PPP_IP4_CONFIG_GATEWAY, "u", &u32)) {
Packit Service 5ffa24
        const NMPlatformIP4Route r = {
Packit Service 5ffa24
            .ifindex       = priv->ifindex,
Packit Service 5ffa24
            .rt_source     = NM_IP_CONFIG_SOURCE_PPP,
Packit Service 5ffa24
            .gateway       = u32,
Packit Service 5ffa24
            .table_coerced = nm_platform_route_table_coerce(priv->ip4_route_table),
Packit Service 5ffa24
            .metric        = priv->ip4_route_metric,
Packit Service 5ffa24
        };
Packit Service 5ffa24
Packit Service 5ffa24
        nm_ip4_config_add_route(config, &r, NULL);
Packit Service 5ffa24
        address.peer_address = u32;
Packit Service 5ffa24
    } else
Packit Service 5ffa24
        address.peer_address = address.address;
Packit Service 5ffa24
Packit Service 5ffa24
    if (g_variant_lookup(config_dict, NM_PPP_IP4_CONFIG_PREFIX, "u", &u32))
Packit Service 5ffa24
        address.plen = u32;
Packit Service 5ffa24
Packit Service 5ffa24
    if (address.address && address.plen && address.plen <= 32) {
Packit Service 5ffa24
        address.addr_source = NM_IP_CONFIG_SOURCE_PPP;
Packit Service 5ffa24
        nm_ip4_config_add_address(config, &address);
Packit Service 5ffa24
    } else {
Packit Service 5ffa24
        _LOGE("invalid IPv4 address received!");
Packit Service 5ffa24
        goto out;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (g_variant_lookup(config_dict, NM_PPP_IP4_CONFIG_DNS, "au", &iter)) {
Packit Service 5ffa24
        while (g_variant_iter_next(iter, "u", &u32))
Packit Service 5ffa24
            nm_ip4_config_add_nameserver(config, u32);
Packit Service 5ffa24
        g_variant_iter_free(iter);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (g_variant_lookup(config_dict, NM_PPP_IP4_CONFIG_WINS, "au", &iter)) {
Packit Service 5ffa24
        while (g_variant_iter_next(iter, "u", &u32))
Packit Service 5ffa24
            nm_ip4_config_add_wins(config, u32);
Packit Service 5ffa24
        g_variant_iter_free(iter);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    /* Push the IP4 config up to the device */
Packit Service 5ffa24
    g_signal_emit(self, signals[IP4_CONFIG], 0, config);
Packit Service 5ffa24
Packit Service 5ffa24
out:
Packit Service 5ffa24
    g_dbus_method_invocation_return_value(invocation, NULL);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
/* Converts the named Interface Identifier item to an IPv6 LL address and
Packit Service 5ffa24
 * returns the IID.
Packit Service 5ffa24
 */
Packit Service 5ffa24
static gboolean
Packit Service 5ffa24
iid_value_to_ll6_addr(GVariant *          dict,
Packit Service 5ffa24
                      const char *        prop,
Packit Service 5ffa24
                      struct in6_addr *   out_addr,
Packit Service 5ffa24
                      NMUtilsIPv6IfaceId *out_iid)
Packit Service 5ffa24
{
Packit Service 5ffa24
    guint64 iid;
Packit Service 5ffa24
Packit Service 5ffa24
    if (!g_variant_lookup(dict, prop, "t", &iid)) {
Packit Service 5ffa24
        _LOGD("pppd plugin property '%s' missing or not a uint64", prop);
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
    g_return_val_if_fail(iid != 0, FALSE);
Packit Service 5ffa24
Packit Service 5ffa24
    /* Construct an IPv6 LL address from the interface identifier.  See
Packit Service 5ffa24
     * http://tools.ietf.org/html/rfc4291#section-2.5.1 (IPv6) and
Packit Service 5ffa24
     * http://tools.ietf.org/html/rfc5072#section-4.1 (IPv6 over PPP).
Packit Service 5ffa24
     */
Packit Service 5ffa24
    memset(out_addr->s6_addr, 0, sizeof(out_addr->s6_addr));
Packit Service 5ffa24
    out_addr->s6_addr16[0] = htons(0xfe80);
Packit Service 5ffa24
    memcpy(out_addr->s6_addr + 8, &iid, sizeof(iid));
Packit Service 5ffa24
    if (out_iid)
Packit Service 5ffa24
        nm_utils_ipv6_interface_identifier_get_from_addr(out_iid, out_addr);
Packit Service 5ffa24
    return TRUE;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
impl_ppp_manager_set_ip6_config(NMDBusObject *                     obj,
Packit Service 5ffa24
                                const NMDBusInterfaceInfoExtended *interface_info,
Packit Service 5ffa24
                                const NMDBusMethodInfoExtended *   method_info,
Packit Service 5ffa24
                                GDBusConnection *                  connection,
Packit Service 5ffa24
                                const char *                       sender,
Packit Service 5ffa24
                                GDBusMethodInvocation *            invocation,
Packit Service 5ffa24
                                GVariant *                         parameters)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMPPPManager *       self           = NM_PPP_MANAGER(obj);
Packit Service 5ffa24
    NMPPPManagerPrivate *priv           = NM_PPP_MANAGER_GET_PRIVATE(self);
Packit Service 5ffa24
    gs_unref_object NMIP6Config *config = NULL;
Packit Service 5ffa24
    NMPlatformIP6Address         addr;
Packit Service 5ffa24
    struct in6_addr              a;
Packit Service 5ffa24
    NMUtilsIPv6IfaceId           iid       = NM_UTILS_IPV6_IFACE_ID_INIT;
Packit Service 5ffa24
    gboolean                     has_peer  = FALSE;
Packit Service 5ffa24
    gs_unref_variant GVariant *config_dict = NULL;
Packit Service 5ffa24
Packit Service 5ffa24
    _LOGI("(IPv6 Config Get) reply received.");
Packit Service 5ffa24
Packit Service 5ffa24
    g_variant_get(parameters, "(@a{sv})", &config_dict);
Packit Service 5ffa24
Packit Service 5ffa24
    nm_clear_g_source(&priv->ppp_timeout_handler);
Packit Service 5ffa24
Packit Service 5ffa24
    if (!set_ip_config_common(self, config_dict, NULL))
Packit Service 5ffa24
        goto out;
Packit Service 5ffa24
Packit Service 5ffa24
    config = nm_ip6_config_new(nm_platform_get_multi_idx(NM_PLATFORM_GET), priv->ifindex);
Packit Service 5ffa24
Packit Service 5ffa24
    memset(&addr, 0, sizeof(addr));
Packit Service 5ffa24
    addr.plen = 64;
Packit Service 5ffa24
Packit Service 5ffa24
    if (iid_value_to_ll6_addr(config_dict, NM_PPP_IP6_CONFIG_PEER_IID, &a, NULL)) {
Packit Service 5ffa24
        const NMPlatformIP6Route r = {
Packit Service 5ffa24
            .ifindex       = priv->ifindex,
Packit Service 5ffa24
            .rt_source     = NM_IP_CONFIG_SOURCE_PPP,
Packit Service 5ffa24
            .gateway       = a,
Packit Service 5ffa24
            .table_coerced = nm_platform_route_table_coerce(priv->ip6_route_table),
Packit Service 5ffa24
            .metric        = priv->ip6_route_metric,
Packit Service 5ffa24
        };
Packit Service 5ffa24
Packit Service 5ffa24
        nm_ip6_config_add_route(config, &r, NULL);
Packit Service 5ffa24
        addr.peer_address = a;
Packit Service 5ffa24
        has_peer          = TRUE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (iid_value_to_ll6_addr(config_dict, NM_PPP_IP6_CONFIG_OUR_IID, &addr.address, &iid)) {
Packit Service 5ffa24
        if (!has_peer)
Packit Service 5ffa24
            addr.peer_address = addr.address;
Packit Service 5ffa24
        nm_ip6_config_add_address(config, &addr);
Packit Service 5ffa24
Packit Service 5ffa24
        /* Push the IPv6 config and interface identifier up to the device */
Packit Service 5ffa24
        g_signal_emit(self, signals[IP6_CONFIG], 0, &iid, config);
Packit Service 5ffa24
    } else
Packit Service 5ffa24
        _LOGE("invalid IPv6 address received!");
Packit Service 5ffa24
Packit Service 5ffa24
out:
Packit Service 5ffa24
    g_dbus_method_invocation_return_value(invocation, NULL);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
static NM_UTILS_LOOKUP_STR_DEFINE(
Packit Service 5ffa24
    pppd_exit_code_to_str,
Packit Service 5ffa24
    int,
Packit Service 5ffa24
    NM_UTILS_LOOKUP_DEFAULT("Unknown error"),
Packit Service 5ffa24
    NM_UTILS_LOOKUP_STR_ITEM(1, "Fatal pppd error");
Packit Service 5ffa24
    NM_UTILS_LOOKUP_STR_ITEM(2, "pppd options error"),
Packit Service 5ffa24
    NM_UTILS_LOOKUP_STR_ITEM(3, "No root priv error"),
Packit Service 5ffa24
    NM_UTILS_LOOKUP_STR_ITEM(4, "No ppp module error"),
Packit Service 5ffa24
    NM_UTILS_LOOKUP_STR_ITEM(5, "pppd received a signal"),
Packit Service 5ffa24
    NM_UTILS_LOOKUP_STR_ITEM(6, "Serial port lock failed"),
Packit Service 5ffa24
    NM_UTILS_LOOKUP_STR_ITEM(7, "Serial port open failed"),
Packit Service 5ffa24
    NM_UTILS_LOOKUP_STR_ITEM(8, "Connect script failed"),
Packit Service 5ffa24
    NM_UTILS_LOOKUP_STR_ITEM(9, "Pty program error"),
Packit Service 5ffa24
    NM_UTILS_LOOKUP_STR_ITEM(10, "PPP negotiation failed"),
Packit Service 5ffa24
    NM_UTILS_LOOKUP_STR_ITEM(11, "Peer didn't authenticatie itself"),
Packit Service 5ffa24
    NM_UTILS_LOOKUP_STR_ITEM(12, "Link idle: Idle Seconds reached."),
Packit Service 5ffa24
    NM_UTILS_LOOKUP_STR_ITEM(13, "Connect time limit reached."),
Packit Service 5ffa24
    NM_UTILS_LOOKUP_STR_ITEM(14, "Callback negotiated, call should come back."),
Packit Service 5ffa24
    NM_UTILS_LOOKUP_STR_ITEM(15, "Lack of LCP echo responses"),
Packit Service 5ffa24
    NM_UTILS_LOOKUP_STR_ITEM(16, "A modem hung up the phone"),
Packit Service 5ffa24
    NM_UTILS_LOOKUP_STR_ITEM(17, "Loopback detected"),
Packit Service 5ffa24
    NM_UTILS_LOOKUP_STR_ITEM(18, "The init script failed"),
Packit Service 5ffa24
    NM_UTILS_LOOKUP_STR_ITEM(19,
Packit Service 5ffa24
                             "Authentication error. "
Packit Service 5ffa24
                             "We failed to authenticate ourselves to the peer. "
Packit Service 5ffa24
                             "Maybe bad account or password?"), );
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
ppp_watch_cb(GPid pid, int status, gpointer user_data)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMPPPManager *       self = NM_PPP_MANAGER(user_data);
Packit Service 5ffa24
    NMPPPManagerPrivate *priv = NM_PPP_MANAGER_GET_PRIVATE(self);
Packit Service 5ffa24
    int                  err;
Packit Service 5ffa24
    const long long      lpid = (long long) pid;
Packit Service 5ffa24
Packit Service 5ffa24
    g_return_if_fail(pid == priv->pid);
Packit Service 5ffa24
Packit Service 5ffa24
    if (WIFEXITED(status)) {
Packit Service 5ffa24
        err = WEXITSTATUS(status);
Packit Service 5ffa24
        if (err) {
Packit Service 5ffa24
            _LOGW("pppd pid %lld exited with error %d: %s", lpid, err, pppd_exit_code_to_str(err));
Packit Service 5ffa24
        } else
Packit Service 5ffa24
            _LOGD("pppd pid %lld exited with success", lpid);
Packit Service 5ffa24
    } else if (WIFSTOPPED(status)) {
Packit Service 5ffa24
        _LOGW("pppd pid %lld stopped unexpectedly with signal %d", lpid, WSTOPSIG(status));
Packit Service 5ffa24
    } else if (WIFSIGNALED(status)) {
Packit Service 5ffa24
        _LOGW("pppd pid %lld died with signal %d", lpid, WTERMSIG(status));
Packit Service 5ffa24
    } else
Packit Service 5ffa24
        _LOGW("pppd pid %lld died from an unknown cause", lpid);
Packit Service 5ffa24
Packit Service 5ffa24
    priv->pid          = 0;
Packit Service 5ffa24
    priv->ppp_watch_id = 0;
Packit Service 5ffa24
    _ppp_cleanup(self);
Packit Service 5ffa24
    g_signal_emit(self, signals[STATE_CHANGED], 0, (guint) NM_PPP_STATUS_DEAD);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static gboolean
Packit Service 5ffa24
pppd_timed_out(gpointer data)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMPPPManager *self = NM_PPP_MANAGER(data);
Packit Service 5ffa24
Packit Service 5ffa24
    _LOGW("pppd timed out or didn't initialize our dbus module");
Packit Service 5ffa24
    _ppp_manager_stop(self, NULL, NULL, NULL);
Packit Service 5ffa24
Packit Service 5ffa24
    g_signal_emit(self, signals[STATE_CHANGED], 0, (guint) NM_PPP_STATUS_DEAD);
Packit Service 5ffa24
Packit Service 5ffa24
    return FALSE;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static GPtrArray *
Packit Service 5ffa24
create_pppd_cmd_line(NMPPPManager *  self,
Packit Service 5ffa24
                     NMSettingPpp *  setting,
Packit Service 5ffa24
                     NMSettingPppoe *pppoe,
Packit Service 5ffa24
                     NMSettingAdsl * adsl,
Packit Service 5ffa24
                     const char *    ppp_name,
Packit Service 5ffa24
                     guint           baud_override,
Packit Service 5ffa24
                     gboolean        ip4_enabled,
Packit Service 5ffa24
                     gboolean        ip6_enabled,
Packit Service 5ffa24
                     GError **       err)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMPPPManagerPrivate *priv        = NM_PPP_MANAGER_GET_PRIVATE(self);
Packit Service 5ffa24
    const char *         pppd_binary = NULL;
Packit Service 5ffa24
    gs_unref_ptrarray GPtrArray *cmd = NULL;
Packit Service 5ffa24
    gboolean                     ppp_debug;
Packit Service 5ffa24
Packit Service 5ffa24
    g_return_val_if_fail(setting != NULL, NULL);
Packit Service 5ffa24
Packit Service 5ffa24
#ifndef PPPD_PATH
Packit Service 5ffa24
    #define PPPD_PATH NULL
Packit Service 5ffa24
#endif
Packit Service 5ffa24
Packit Service 5ffa24
    pppd_binary = nm_utils_find_helper("pppd", PPPD_PATH, err);
Packit Service 5ffa24
    if (!pppd_binary)
Packit Service 5ffa24
        return NULL;
Packit Service 5ffa24
Packit Service 5ffa24
    if (!ip4_enabled && !ip6_enabled) {
Packit Service 5ffa24
        g_set_error_literal(err,
Packit Service 5ffa24
                            NM_MANAGER_ERROR,
Packit Service 5ffa24
                            NM_MANAGER_ERROR_FAILED,
Packit Service 5ffa24
                            "Neither IPv4 or IPv6 allowed.");
Packit Service 5ffa24
        return NULL;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    cmd = g_ptr_array_new_with_free_func(g_free);
Packit Service 5ffa24
Packit Service 5ffa24
    nm_strv_ptrarray_add_string_dup(cmd, pppd_binary);
Packit Service 5ffa24
Packit Service 5ffa24
    nm_strv_ptrarray_add_string_dup(cmd, "nodetach");
Packit Service 5ffa24
    nm_strv_ptrarray_add_string_dup(cmd, "lock");
Packit Service 5ffa24
Packit Service 5ffa24
    /* NM handles setting the default route */
Packit Service 5ffa24
    nm_strv_ptrarray_add_string_dup(cmd, "nodefaultroute");
Packit Service 5ffa24
Packit Service 5ffa24
    if (!ip4_enabled)
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, "noip");
Packit Service 5ffa24
Packit Service 5ffa24
    if (ip6_enabled) {
Packit Service 5ffa24
        /* Allow IPv6 to be configured by IPV6CP */
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, "ipv6");
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, ",");
Packit Service 5ffa24
    } else
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, "noipv6");
Packit Service 5ffa24
Packit Service 5ffa24
    ppp_debug = !!getenv("NM_PPP_DEBUG");
Packit Service 5ffa24
    if (nm_logging_enabled(LOGL_DEBUG, LOGD_PPP))
Packit Service 5ffa24
        ppp_debug = TRUE;
Packit Service 5ffa24
Packit Service 5ffa24
    if (ppp_debug)
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, "debug");
Packit Service 5ffa24
Packit Service 5ffa24
    if (ppp_name) {
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, "user");
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, ppp_name);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (pppoe) {
Packit Service 5ffa24
        const char *pppoe_service;
Packit Service 5ffa24
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, "plugin");
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, "rp-pppoe.so");
Packit Service 5ffa24
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_concat(cmd, "nic-", priv->parent_iface);
Packit Service 5ffa24
Packit Service 5ffa24
        pppoe_service = nm_setting_pppoe_get_service(pppoe);
Packit Service 5ffa24
        if (pppoe_service) {
Packit Service 5ffa24
            nm_strv_ptrarray_add_string_dup(cmd, "rp_pppoe_service");
Packit Service 5ffa24
            nm_strv_ptrarray_add_string_dup(cmd, pppoe_service);
Packit Service 5ffa24
        }
Packit Service 5ffa24
    } else if (adsl) {
Packit Service 5ffa24
        const char *protocol = nm_setting_adsl_get_protocol(adsl);
Packit Service 5ffa24
Packit Service 5ffa24
        if (!strcmp(protocol, NM_SETTING_ADSL_PROTOCOL_PPPOA)) {
Packit Service 5ffa24
            guint32     vpi    = nm_setting_adsl_get_vpi(adsl);
Packit Service 5ffa24
            guint32     vci    = nm_setting_adsl_get_vci(adsl);
Packit Service 5ffa24
            const char *encaps = nm_setting_adsl_get_encapsulation(adsl);
Packit Service 5ffa24
Packit Service 5ffa24
            nm_strv_ptrarray_add_string_dup(cmd, "plugin");
Packit Service 5ffa24
            nm_strv_ptrarray_add_string_dup(cmd, "pppoatm.so");
Packit Service 5ffa24
Packit Service 5ffa24
            nm_strv_ptrarray_add_string_printf(cmd, "%d.%d", vpi, vci);
Packit Service 5ffa24
Packit Service 5ffa24
            if (g_strcmp0(encaps, NM_SETTING_ADSL_ENCAPSULATION_LLC) == 0)
Packit Service 5ffa24
                nm_strv_ptrarray_add_string_dup(cmd, "llc-encaps");
Packit Service 5ffa24
            else /*if (g_strcmp0 (encaps, NM_SETTING_ADSL_ENCAPSULATION_VCMUX) == 0)*/
Packit Service 5ffa24
                nm_strv_ptrarray_add_string_dup(cmd, "vc-encaps");
Packit Service 5ffa24
Packit Service 5ffa24
        } else if (!strcmp(protocol, NM_SETTING_ADSL_PROTOCOL_PPPOE)) {
Packit Service 5ffa24
            nm_strv_ptrarray_add_string_dup(cmd, "plugin");
Packit Service 5ffa24
            nm_strv_ptrarray_add_string_dup(cmd, "rp-pppoe.so");
Packit Service 5ffa24
            nm_strv_ptrarray_add_string_dup(cmd, priv->parent_iface);
Packit Service 5ffa24
        }
Packit Service 5ffa24
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, "noipdefault");
Packit Service 5ffa24
    } else {
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, priv->parent_iface);
Packit Service 5ffa24
        /* Don't send some random address as the local address */
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, "noipdefault");
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (nm_setting_ppp_get_baud(setting))
Packit Service 5ffa24
        nm_strv_ptrarray_add_int(cmd, nm_setting_ppp_get_baud(setting));
Packit Service 5ffa24
    else if (baud_override)
Packit Service 5ffa24
        nm_strv_ptrarray_add_int(cmd, baud_override);
Packit Service 5ffa24
Packit Service 5ffa24
    /* noauth by default, because we certainly don't have any information
Packit Service 5ffa24
     * with which to verify anything the peer gives us if we ask it to
Packit Service 5ffa24
     * authenticate itself, which is what 'auth' really means.
Packit Service 5ffa24
     */
Packit Service 5ffa24
    nm_strv_ptrarray_add_string_dup(cmd, "noauth");
Packit Service 5ffa24
Packit Service 5ffa24
    if (nm_setting_ppp_get_refuse_eap(setting))
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, "refuse-eap");
Packit Service 5ffa24
    if (nm_setting_ppp_get_refuse_pap(setting))
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, "refuse-pap");
Packit Service 5ffa24
    if (nm_setting_ppp_get_refuse_chap(setting))
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, "refuse-chap");
Packit Service 5ffa24
    if (nm_setting_ppp_get_refuse_mschap(setting))
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, "refuse-mschap");
Packit Service 5ffa24
    if (nm_setting_ppp_get_refuse_mschapv2(setting))
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, "refuse-mschap-v2");
Packit Service 5ffa24
    if (nm_setting_ppp_get_nobsdcomp(setting))
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, "nobsdcomp");
Packit Service 5ffa24
    if (nm_setting_ppp_get_no_vj_comp(setting))
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, "novj");
Packit Service 5ffa24
    if (nm_setting_ppp_get_nodeflate(setting))
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, "nodeflate");
Packit Service 5ffa24
    if (nm_setting_ppp_get_require_mppe(setting))
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, "require-mppe");
Packit Service 5ffa24
    if (nm_setting_ppp_get_require_mppe_128(setting))
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, "require-mppe-128");
Packit Service 5ffa24
    if (nm_setting_ppp_get_mppe_stateful(setting))
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, "mppe-stateful");
Packit Service 5ffa24
    if (nm_setting_ppp_get_crtscts(setting))
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, "crtscts");
Packit Service 5ffa24
Packit Service 5ffa24
    /* Always ask for DNS, we don't have to use them if the connection
Packit Service 5ffa24
     * overrides the returned servers.
Packit Service 5ffa24
     */
Packit Service 5ffa24
    nm_strv_ptrarray_add_string_dup(cmd, "usepeerdns");
Packit Service 5ffa24
Packit Service 5ffa24
    if (nm_setting_ppp_get_mru(setting)) {
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, "mru");
Packit Service 5ffa24
        nm_strv_ptrarray_add_int(cmd, nm_setting_ppp_get_mru(setting));
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (nm_setting_ppp_get_mtu(setting)) {
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, "mtu");
Packit Service 5ffa24
        nm_strv_ptrarray_add_int(cmd, nm_setting_ppp_get_mtu(setting));
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    nm_strv_ptrarray_add_string_dup(cmd, "lcp-echo-failure");
Packit Service 5ffa24
    nm_strv_ptrarray_add_int(cmd, nm_setting_ppp_get_lcp_echo_failure(setting));
Packit Service 5ffa24
Packit Service 5ffa24
    nm_strv_ptrarray_add_string_dup(cmd, "lcp-echo-interval");
Packit Service 5ffa24
    nm_strv_ptrarray_add_int(cmd, nm_setting_ppp_get_lcp_echo_interval(setting));
Packit Service 5ffa24
Packit Service 5ffa24
    /* Avoid pppd to exit if no traffic going through */
Packit Service 5ffa24
    nm_strv_ptrarray_add_string_dup(cmd, "idle");
Packit Service 5ffa24
    nm_strv_ptrarray_add_string_dup(cmd, "0");
Packit Service 5ffa24
Packit Service 5ffa24
    nm_strv_ptrarray_add_string_dup(cmd, "ipparam");
Packit Service 5ffa24
    nm_strv_ptrarray_add_string_dup(cmd, nm_dbus_object_get_path(NM_DBUS_OBJECT(self)));
Packit Service 5ffa24
Packit Service 5ffa24
    nm_strv_ptrarray_add_string_dup(cmd, "plugin");
Packit Service 5ffa24
    nm_strv_ptrarray_add_string_dup(cmd, NM_PPPD_PLUGIN);
Packit Service 5ffa24
Packit Service 5ffa24
    if (pppoe && nm_setting_pppoe_get_parent(pppoe)) {
Packit Service 5ffa24
        static int unit;
Packit Service 5ffa24
Packit Service 5ffa24
        /* The PPP interface is going to be renamed, so pass a
Packit Service 5ffa24
         * different unit each time so that activations don't
Packit Service 5ffa24
         * race with each others. */
Packit Service 5ffa24
        nm_strv_ptrarray_add_string_dup(cmd, "unit");
Packit Service 5ffa24
        nm_strv_ptrarray_add_int(cmd, unit);
Packit Service 5ffa24
        unit = unit < G_MAXINT ? unit + 1 : 0;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    g_ptr_array_add(cmd, NULL);
Packit Service 5ffa24
    return g_steal_pointer(&cmd);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
pppoe_fill_defaults(NMSettingPpp *setting)
Packit Service 5ffa24
{
Packit Service 5ffa24
    if (!nm_setting_ppp_get_mtu(setting))
Packit Service 5ffa24
        g_object_set(setting, NM_SETTING_PPP_MTU, (guint32) 1492, NULL);
Packit Service 5ffa24
Packit Service 5ffa24
    if (!nm_setting_ppp_get_mru(setting))
Packit Service 5ffa24
        g_object_set(setting, NM_SETTING_PPP_MRU, (guint32) 1492, NULL);
Packit Service 5ffa24
Packit Service 5ffa24
    g_object_set(setting, NM_SETTING_PPP_NOAUTH, TRUE, NM_SETTING_PPP_NODEFLATE, TRUE, NULL);
Packit Service 5ffa24
Packit Service 5ffa24
    /* FIXME: These commented settings should be set as well, update NMSettingPpp first. */
Packit Service 5ffa24
#if 0
Packit Service 5ffa24
    setting->noipdefault = TRUE;
Packit Service 5ffa24
    setting->default_asyncmap = TRUE;
Packit Service 5ffa24
    setting->defaultroute = TRUE;
Packit Service 5ffa24
    setting->hide_password = TRUE;
Packit Service 5ffa24
    setting->noaccomp = TRUE;
Packit Service 5ffa24
    setting->nopcomp = TRUE;
Packit Service 5ffa24
    setting->novj = TRUE;
Packit Service 5ffa24
    setting->novjccomp = TRUE;
Packit Service 5ffa24
#endif
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static gboolean
Packit Service 5ffa24
_ppp_manager_start(NMPPPManager *self,
Packit Service 5ffa24
                   NMActRequest *req,
Packit Service 5ffa24
                   const char *  ppp_name,
Packit Service 5ffa24
                   guint32       timeout_secs,
Packit Service 5ffa24
                   guint         baud_override,
Packit Service 5ffa24
                   GError **     err)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMPPPManagerPrivate *priv;
Packit Service 5ffa24
    NMConnection *       connection;
Packit Service 5ffa24
    NMSettingPpp *       s_ppp;
Packit Service 5ffa24
    gs_unref_object NMSettingPpp *s_ppp_free = NULL;
Packit Service 5ffa24
    NMSettingPppoe *              pppoe_setting;
Packit Service 5ffa24
    NMSettingAdsl *               adsl_setting;
Packit Service 5ffa24
    gs_unref_ptrarray GPtrArray *ppp_cmd = NULL;
Packit Service 5ffa24
    gs_free char *               cmd_str = NULL;
Packit Service 5ffa24
    struct stat                  st;
Packit Service 5ffa24
    const char *                 ip6_method, *ip4_method;
Packit Service 5ffa24
    gboolean                     ip6_enabled = FALSE;
Packit Service 5ffa24
    gboolean                     ip4_enabled = FALSE;
Packit Service 5ffa24
Packit Service 5ffa24
    g_return_val_if_fail(NM_IS_PPP_MANAGER(self), FALSE);
Packit Service 5ffa24
    g_return_val_if_fail(NM_IS_ACT_REQUEST(req), FALSE);
Packit Service 5ffa24
Packit Service 5ffa24
    priv = NM_PPP_MANAGER_GET_PRIVATE(self);
Packit Service 5ffa24
Packit Service 5ffa24
#if !WITH_PPP
Packit Service 5ffa24
    /* PPP support disabled */
Packit Service 5ffa24
    g_set_error_literal(err,
Packit Service 5ffa24
                        NM_MANAGER_ERROR,
Packit Service 5ffa24
                        NM_MANAGER_ERROR_FAILED,
Packit Service 5ffa24
                        "PPP support is not enabled.");
Packit Service 5ffa24
    return FALSE;
Packit Service 5ffa24
#endif
Packit Service 5ffa24
Packit Service 5ffa24
    nm_dbus_object_export(NM_DBUS_OBJECT(self));
Packit Service 5ffa24
Packit Service 5ffa24
    priv->pid = 0;
Packit Service 5ffa24
Packit Service 5ffa24
    /* Make sure /dev/ppp exists (bgo #533064) */
Packit Service 5ffa24
    if (stat("/dev/ppp", &st) || !S_ISCHR(st.st_mode))
Packit Service 5ffa24
        nm_utils_modprobe(NULL, FALSE, "ppp_generic", NULL);
Packit Service 5ffa24
Packit Service 5ffa24
    connection = nm_act_request_get_applied_connection(req);
Packit Service 5ffa24
    g_return_val_if_fail(connection, FALSE);
Packit Service 5ffa24
Packit Service 5ffa24
    s_ppp = nm_connection_get_setting_ppp(connection);
Packit Service 5ffa24
    if (!s_ppp) {
Packit Service 5ffa24
        /* If the PPP settings are all default we may not have a PPP setting yet,
Packit Service 5ffa24
         * so just make a default one here.
Packit Service 5ffa24
         */
Packit Service 5ffa24
        s_ppp = s_ppp_free = NM_SETTING_PPP(nm_setting_ppp_new());
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    pppoe_setting = nm_connection_get_setting_pppoe(connection);
Packit Service 5ffa24
    if (pppoe_setting) {
Packit Service 5ffa24
        /* We can't modify the applied connection's setting, make a copy */
Packit Service 5ffa24
        if (!s_ppp_free)
Packit Service 5ffa24
            s_ppp = s_ppp_free = NM_SETTING_PPP(nm_setting_duplicate((NMSetting *) s_ppp));
Packit Service 5ffa24
        pppoe_fill_defaults(s_ppp);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    adsl_setting = (NMSettingAdsl *) nm_connection_get_setting(connection, NM_TYPE_SETTING_ADSL);
Packit Service 5ffa24
Packit Service 5ffa24
    /* Figure out what address methods should be enabled */
Packit Service 5ffa24
    ip4_method  = nm_utils_get_ip_config_method(connection, AF_INET);
Packit Service 5ffa24
    ip4_enabled = nm_streq(ip4_method, NM_SETTING_IP4_CONFIG_METHOD_AUTO);
Packit Service 5ffa24
    ip6_method  = nm_utils_get_ip_config_method(connection, AF_INET6);
Packit Service 5ffa24
    ip6_enabled = nm_streq(ip6_method, NM_SETTING_IP6_CONFIG_METHOD_AUTO);
Packit Service 5ffa24
Packit Service 5ffa24
    ppp_cmd = create_pppd_cmd_line(self,
Packit Service 5ffa24
                                   s_ppp,
Packit Service 5ffa24
                                   pppoe_setting,
Packit Service 5ffa24
                                   adsl_setting,
Packit Service 5ffa24
                                   ppp_name,
Packit Service 5ffa24
                                   baud_override,
Packit Service 5ffa24
                                   ip4_enabled,
Packit Service 5ffa24
                                   ip6_enabled,
Packit Service 5ffa24
                                   err);
Packit Service 5ffa24
    if (!ppp_cmd)
Packit Service 5ffa24
        goto fail;
Packit Service 5ffa24
Packit Service 5ffa24
    _LOGI("starting PPP connection");
Packit Service 5ffa24
Packit Service 5ffa24
    _LOGD("command line: %s", (cmd_str = g_strjoinv(" ", (char **) ppp_cmd->pdata)));
Packit Service 5ffa24
Packit Service 5ffa24
    priv->pid = 0;
Packit Service 5ffa24
    if (!g_spawn_async(NULL,
Packit Service 5ffa24
                       (char **) ppp_cmd->pdata,
Packit Service 5ffa24
                       NULL,
Packit Service 5ffa24
                       G_SPAWN_DO_NOT_REAP_CHILD,
Packit Service 5ffa24
                       nm_utils_setpgid,
Packit Service 5ffa24
                       NULL,
Packit Service 5ffa24
                       &priv->pid,
Packit Service 5ffa24
                       err))
Packit Service 5ffa24
        goto fail;
Packit Service 5ffa24
Packit Service 5ffa24
    nm_assert(priv->pid > 0);
Packit Service 5ffa24
Packit Service 5ffa24
    _LOGI("pppd started with pid %lld", (long long) priv->pid);
Packit Service 5ffa24
Packit Service 5ffa24
    priv->ppp_watch_id        = g_child_watch_add(priv->pid, (GChildWatchFunc) ppp_watch_cb, self);
Packit Service 5ffa24
    priv->ppp_timeout_handler = g_timeout_add_seconds(timeout_secs, pppd_timed_out, self);
Packit Service 5ffa24
    priv->act_req             = g_object_ref(req);
Packit Service 5ffa24
Packit Service 5ffa24
    return TRUE;
Packit Service 5ffa24
fail:
Packit Service 5ffa24
    nm_dbus_object_unexport(NM_DBUS_OBJECT(self));
Packit Service 5ffa24
    return FALSE;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
_ppp_cleanup(NMPPPManager *self)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMPPPManagerPrivate *priv;
Packit Service 5ffa24
Packit Service 5ffa24
    g_return_if_fail(NM_IS_PPP_MANAGER(self));
Packit Service 5ffa24
Packit Service 5ffa24
    priv = NM_PPP_MANAGER_GET_PRIVATE(self);
Packit Service 5ffa24
Packit Service 5ffa24
    cancel_get_secrets(self);
Packit Service 5ffa24
Packit Service 5ffa24
    nm_clear_g_source(&priv->monitor_id);
Packit Service 5ffa24
Packit Service 5ffa24
    if (priv->monitor_fd >= 0) {
Packit Service 5ffa24
        /* Get the stats one last time */
Packit Service 5ffa24
        monitor_cb(self);
Packit Service 5ffa24
        nm_close(priv->monitor_fd);
Packit Service 5ffa24
        priv->monitor_fd = -1;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    nm_clear_g_source(&priv->ppp_timeout_handler);
Packit Service 5ffa24
    nm_clear_g_source(&priv->ppp_watch_id);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
struct _NMPPPManagerStopHandle {
Packit Service 5ffa24
    NMPPPManager *           self;
Packit Service 5ffa24
    NMPPPManagerStopCallback callback;
Packit Service 5ffa24
    gpointer                 user_data;
Packit Service 5ffa24
Packit Service 5ffa24
    /* this object delays shutdown, because we still need to wait until
Packit Service 5ffa24
     * pppd process terminated. */
Packit Service 5ffa24
    GObject *shutdown_waitobj;
Packit Service 5ffa24
Packit Service 5ffa24
    GCancellable *cancellable;
Packit Service 5ffa24
Packit Service 5ffa24
    gulong cancellable_id;
Packit Service 5ffa24
Packit Service 5ffa24
    guint idle_id;
Packit Service 5ffa24
};
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
_stop_handle_complete(NMPPPManagerStopHandle *handle, gboolean was_cancelled)
Packit Service 5ffa24
{
Packit Service 5ffa24
    gs_unref_object NMPPPManager *self = NULL;
Packit Service 5ffa24
    NMPPPManagerStopCallback      callback;
Packit Service 5ffa24
Packit Service 5ffa24
    if (handle->cancellable_id) {
Packit Service 5ffa24
        g_cancellable_disconnect(handle->cancellable, nm_steal_int(&handle->cancellable_id));
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    g_clear_object(&handle->cancellable);
Packit Service 5ffa24
Packit Service 5ffa24
    self = g_steal_pointer(&handle->self);
Packit Service 5ffa24
    if (!self)
Packit Service 5ffa24
        return;
Packit Service 5ffa24
Packit Service 5ffa24
    if (!handle->callback)
Packit Service 5ffa24
        return;
Packit Service 5ffa24
Packit Service 5ffa24
    callback         = handle->callback;
Packit Service 5ffa24
    handle->callback = NULL;
Packit Service 5ffa24
    callback(self, handle, was_cancelled, handle->user_data);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
_stop_handle_destroy(NMPPPManagerStopHandle *handle, gboolean was_cancelled)
Packit Service 5ffa24
{
Packit Service 5ffa24
    _stop_handle_complete(handle, was_cancelled);
Packit Service 5ffa24
    nm_clear_g_source(&handle->idle_id);
Packit Service 5ffa24
    g_clear_object(&handle->shutdown_waitobj);
Packit Service 5ffa24
    g_slice_free(NMPPPManagerStopHandle, handle);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
_stop_child_cb(pid_t pid, gboolean success, int child_status, gpointer user_data)
Packit Service 5ffa24
{
Packit Service 5ffa24
    _stop_handle_destroy(user_data, FALSE);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static gboolean
Packit Service 5ffa24
_stop_idle_cb(gpointer user_data)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMPPPManagerStopHandle *handle = user_data;
Packit Service 5ffa24
Packit Service 5ffa24
    handle->idle_id = 0;
Packit Service 5ffa24
    _stop_handle_destroy(handle, FALSE);
Packit Service 5ffa24
    return G_SOURCE_REMOVE;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
_stop_cancelled_cb(GCancellable *cancellable, gpointer user_data)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMPPPManagerStopHandle *handle = user_data;
Packit Service 5ffa24
Packit Service 5ffa24
    nm_clear_g_signal_handler(handle->cancellable, &handle->cancellable_id);
Packit Service 5ffa24
    _ppp_manager_stop_cancel(handle);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static NMPPPManagerStopHandle *
Packit Service 5ffa24
_ppp_manager_stop(NMPPPManager *           self,
Packit Service 5ffa24
                  GCancellable *           cancellable,
Packit Service 5ffa24
                  NMPPPManagerStopCallback callback,
Packit Service 5ffa24
                  gpointer                 user_data)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMPPPManagerPrivate *   priv = NM_PPP_MANAGER_GET_PRIVATE(self);
Packit Service 5ffa24
    NMDBusObject *          dbus = NM_DBUS_OBJECT(self);
Packit Service 5ffa24
    NMPPPManagerStopHandle *handle;
Packit Service 5ffa24
Packit Service 5ffa24
    if (nm_dbus_object_is_exported(dbus))
Packit Service 5ffa24
        nm_dbus_object_unexport(dbus);
Packit Service 5ffa24
Packit Service 5ffa24
    _ppp_cleanup(self);
Packit Service 5ffa24
Packit Service 5ffa24
    if (!priv->pid && !callback) {
Packit Service 5ffa24
        /* nothing to do further...
Packit Service 5ffa24
         *
Packit Service 5ffa24
         * In this case, we return a %NULL handle. The caller cannot cancel this
Packit Service 5ffa24
         * event, but clearly he is not waiting for a callback anyway. */
Packit Service 5ffa24
        return NULL;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    handle            = g_slice_new0(NMPPPManagerStopHandle);
Packit Service 5ffa24
    handle->self      = g_object_ref(self);
Packit Service 5ffa24
    handle->callback  = callback;
Packit Service 5ffa24
    handle->user_data = user_data;
Packit Service 5ffa24
    if (cancellable) {
Packit Service 5ffa24
        handle->cancellable = g_object_ref(cancellable);
Packit Service 5ffa24
        handle->cancellable_id =
Packit Service 5ffa24
            g_cancellable_connect(cancellable, G_CALLBACK(_stop_cancelled_cb), handle, NULL);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (!priv->pid) {
Packit Service 5ffa24
        /* No PID. There is nothing to kill, however, invoke the callback in
Packit Service 5ffa24
         * an idle handler.
Packit Service 5ffa24
         *
Packit Service 5ffa24
         * Note that we don't register nm_shutdown_wait_obj_register_object().
Packit Service 5ffa24
         * In order for shutdown to work properly, the caller must always
Packit Service 5ffa24
         * explicitly cancel the action to go down. With the idle-handler,
Packit Service 5ffa24
         * cancelling the handle completes the request. */
Packit Service 5ffa24
        handle->idle_id = g_idle_add(_stop_idle_cb, handle);
Packit Service 5ffa24
        return handle;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    /* we really want to kill the process and delay shutdown of NetworkManager
Packit Service 5ffa24
     * until the process terminated. We do that, by registering an object
Packit Service 5ffa24
     * that delays shutdown. */
Packit Service 5ffa24
    handle->shutdown_waitobj = g_object_new(G_TYPE_OBJECT, NULL);
Packit Service 5ffa24
    nm_shutdown_wait_obj_register_object(handle->shutdown_waitobj, "ppp-manager-wait-kill-pppd");
Packit Service 5ffa24
    nm_utils_kill_child_async(nm_steal_int(&priv->pid),
Packit Service 5ffa24
                              SIGTERM,
Packit Service 5ffa24
                              LOGD_PPP,
Packit Service 5ffa24
                              "pppd",
Packit Service 5ffa24
                              NM_SHUTDOWN_TIMEOUT_MS,
Packit Service 5ffa24
                              _stop_child_cb,
Packit Service 5ffa24
                              handle);
Packit Service 5ffa24
Packit Service 5ffa24
    return handle;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
_ppp_manager_stop_cancel(NMPPPManagerStopHandle *handle)
Packit Service 5ffa24
{
Packit Service 5ffa24
    g_return_if_fail(handle);
Packit Service 5ffa24
    g_return_if_fail(NM_IS_PPP_MANAGER(handle->self));
Packit Service 5ffa24
Packit Service 5ffa24
    if (handle->idle_id) {
Packit Service 5ffa24
        /* we can complete this fake handle right away. */
Packit Service 5ffa24
        _stop_handle_destroy(handle, TRUE);
Packit Service 5ffa24
        return;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    /* a real handle. Only invoke the callback (synchronously). This marks
Packit Service 5ffa24
     * the handle as handled, but it keeps shutdown_waitobj around, until
Packit Service 5ffa24
     * nm_utils_kill_child_async() returns. */
Packit Service 5ffa24
    _stop_handle_complete(handle, TRUE);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
get_property(GObject *object, guint prop_id, GValue *value, GParamSpec *pspec)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMPPPManagerPrivate *priv = NM_PPP_MANAGER_GET_PRIVATE(object);
Packit Service 5ffa24
Packit Service 5ffa24
    switch (prop_id) {
Packit Service 5ffa24
    case PROP_PARENT_IFACE:
Packit Service 5ffa24
        g_value_set_string(value, priv->parent_iface);
Packit Service 5ffa24
        break;
Packit Service 5ffa24
    default:
Packit Service 5ffa24
        G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
Packit Service 5ffa24
        break;
Packit Service 5ffa24
    }
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
set_property(GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMPPPManagerPrivate *priv = NM_PPP_MANAGER_GET_PRIVATE(object);
Packit Service 5ffa24
Packit Service 5ffa24
    switch (prop_id) {
Packit Service 5ffa24
    case PROP_PARENT_IFACE:
Packit Service 5ffa24
        /* construct-only */
Packit Service 5ffa24
        priv->parent_iface = g_value_dup_string(value);
Packit Service 5ffa24
        break;
Packit Service 5ffa24
    default:
Packit Service 5ffa24
        G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
Packit Service 5ffa24
        break;
Packit Service 5ffa24
    }
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
nm_ppp_manager_init(NMPPPManager *self)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMPPPManagerPrivate *priv = NM_PPP_MANAGER_GET_PRIVATE(self);
Packit Service 5ffa24
Packit Service 5ffa24
    priv->ifindex          = -1;
Packit Service 5ffa24
    priv->monitor_fd       = -1;
Packit Service 5ffa24
    priv->ip4_route_table  = RT_TABLE_MAIN;
Packit Service 5ffa24
    priv->ip4_route_metric = 460;
Packit Service 5ffa24
    priv->ip6_route_table  = RT_TABLE_MAIN;
Packit Service 5ffa24
    priv->ip6_route_metric = 460;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static NMPPPManager *
Packit Service 5ffa24
_ppp_manager_new(const char *iface)
Packit Service 5ffa24
{
Packit Service 5ffa24
    g_return_val_if_fail(iface != NULL, NULL);
Packit Service 5ffa24
Packit Service 5ffa24
    return g_object_new(NM_TYPE_PPP_MANAGER, NM_PPP_MANAGER_PARENT_IFACE, iface, NULL);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
dispose(GObject *object)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMPPPManager *       self = (NMPPPManager *) object;
Packit Service 5ffa24
    NMPPPManagerPrivate *priv = NM_PPP_MANAGER_GET_PRIVATE(self);
Packit Service 5ffa24
Packit Service 5ffa24
    /* we expect the user to first stop the manager. As fallback,
Packit Service 5ffa24
     * still stop. */
Packit Service 5ffa24
    g_warn_if_fail(!priv->pid);
Packit Service 5ffa24
    g_warn_if_fail(!nm_dbus_object_is_exported(NM_DBUS_OBJECT(self)));
Packit Service 5ffa24
    _ppp_manager_stop(self, NULL, NULL, NULL);
Packit Service 5ffa24
Packit Service 5ffa24
    g_clear_object(&priv->act_req);
Packit Service 5ffa24
Packit Service 5ffa24
    G_OBJECT_CLASS(nm_ppp_manager_parent_class)->dispose(object);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
finalize(GObject *object)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMPPPManagerPrivate *priv = NM_PPP_MANAGER_GET_PRIVATE(object);
Packit Service 5ffa24
Packit Service 5ffa24
    g_free(priv->parent_iface);
Packit Service 5ffa24
Packit Service 5ffa24
    G_OBJECT_CLASS(nm_ppp_manager_parent_class)->finalize(object);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static const NMDBusInterfaceInfoExtended interface_info_ppp = {
Packit Service 5ffa24
    .parent = NM_DEFINE_GDBUS_INTERFACE_INFO_INIT(
Packit Service 5ffa24
        NM_DBUS_INTERFACE_PPP,
Packit Service 5ffa24
        .methods = NM_DEFINE_GDBUS_METHOD_INFOS(
Packit Service 5ffa24
            NM_DEFINE_DBUS_METHOD_INFO_EXTENDED(
Packit Service 5ffa24
                NM_DEFINE_GDBUS_METHOD_INFO_INIT(
Packit Service 5ffa24
                    "NeedSecrets",
Packit Service 5ffa24
                    .out_args =
Packit Service 5ffa24
                        NM_DEFINE_GDBUS_ARG_INFOS(NM_DEFINE_GDBUS_ARG_INFO("username", "s"),
Packit Service 5ffa24
                                                  NM_DEFINE_GDBUS_ARG_INFO("password", "s"), ), ),
Packit Service 5ffa24
                .handle = impl_ppp_manager_need_secrets, ),
Packit Service 5ffa24
            NM_DEFINE_DBUS_METHOD_INFO_EXTENDED(
Packit Service 5ffa24
                NM_DEFINE_GDBUS_METHOD_INFO_INIT(
Packit Service 5ffa24
                    "SetIp4Config",
Packit Service 5ffa24
                    .in_args =
Packit Service 5ffa24
                        NM_DEFINE_GDBUS_ARG_INFOS(NM_DEFINE_GDBUS_ARG_INFO("config", "a{sv}"), ), ),
Packit Service 5ffa24
                .handle = impl_ppp_manager_set_ip4_config, ),
Packit Service 5ffa24
            NM_DEFINE_DBUS_METHOD_INFO_EXTENDED(
Packit Service 5ffa24
                NM_DEFINE_GDBUS_METHOD_INFO_INIT(
Packit Service 5ffa24
                    "SetIp6Config",
Packit Service 5ffa24
                    .in_args =
Packit Service 5ffa24
                        NM_DEFINE_GDBUS_ARG_INFOS(NM_DEFINE_GDBUS_ARG_INFO("config", "a{sv}"), ), ),
Packit Service 5ffa24
                .handle = impl_ppp_manager_set_ip6_config, ),
Packit Service 5ffa24
            NM_DEFINE_DBUS_METHOD_INFO_EXTENDED(
Packit Service 5ffa24
                NM_DEFINE_GDBUS_METHOD_INFO_INIT("SetState",
Packit Service 5ffa24
                                                 .in_args = NM_DEFINE_GDBUS_ARG_INFOS(
Packit Service 5ffa24
                                                     NM_DEFINE_GDBUS_ARG_INFO("state", "u"), ), ),
Packit Service 5ffa24
                .handle = impl_ppp_manager_set_state, ),
Packit Service 5ffa24
            NM_DEFINE_DBUS_METHOD_INFO_EXTENDED(
Packit Service 5ffa24
                NM_DEFINE_GDBUS_METHOD_INFO_INIT("SetIfindex",
Packit Service 5ffa24
                                                 .in_args = NM_DEFINE_GDBUS_ARG_INFOS(
Packit Service 5ffa24
                                                     NM_DEFINE_GDBUS_ARG_INFO("ifindex", "i"), ), ),
Packit Service 5ffa24
                .handle = impl_ppp_manager_set_ifindex, ), ), ),
Packit Service 5ffa24
};
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
nm_ppp_manager_class_init(NMPPPManagerClass *manager_class)
Packit Service 5ffa24
{
Packit Service 5ffa24
    GObjectClass *     object_class      = G_OBJECT_CLASS(manager_class);
Packit Service 5ffa24
    NMDBusObjectClass *dbus_object_class = NM_DBUS_OBJECT_CLASS(manager_class);
Packit Service 5ffa24
Packit Service 5ffa24
    object_class->dispose      = dispose;
Packit Service 5ffa24
    object_class->finalize     = finalize;
Packit Service 5ffa24
    object_class->get_property = get_property;
Packit Service 5ffa24
    object_class->set_property = set_property;
Packit Service 5ffa24
Packit Service 5ffa24
    dbus_object_class->export_path     = NM_DBUS_EXPORT_PATH_NUMBERED(NM_DBUS_PATH "/PPP");
Packit Service 5ffa24
    dbus_object_class->interface_infos = NM_DBUS_INTERFACE_INFOS(&interface_info_ppp);
Packit Service 5ffa24
Packit Service 5ffa24
    obj_properties[PROP_PARENT_IFACE] =
Packit Service 5ffa24
        g_param_spec_string(NM_PPP_MANAGER_PARENT_IFACE,
Packit Service 5ffa24
                            "",
Packit Service 5ffa24
                            "",
Packit Service 5ffa24
                            NULL,
Packit Service 5ffa24
                            G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS);
Packit Service 5ffa24
Packit Service 5ffa24
    g_object_class_install_properties(object_class, _PROPERTY_ENUMS_LAST, obj_properties);
Packit Service 5ffa24
Packit Service 5ffa24
    signals[STATE_CHANGED] = g_signal_new(NM_PPP_MANAGER_SIGNAL_STATE_CHANGED,
Packit Service 5ffa24
                                          G_OBJECT_CLASS_TYPE(object_class),
Packit Service 5ffa24
                                          G_SIGNAL_RUN_FIRST,
Packit Service 5ffa24
                                          0,
Packit Service 5ffa24
                                          NULL,
Packit Service 5ffa24
                                          NULL,
Packit Service 5ffa24
                                          NULL,
Packit Service 5ffa24
                                          G_TYPE_NONE,
Packit Service 5ffa24
                                          1,
Packit Service 5ffa24
                                          G_TYPE_UINT);
Packit Service 5ffa24
Packit Service 5ffa24
    signals[IFINDEX_SET] = g_signal_new(NM_PPP_MANAGER_SIGNAL_IFINDEX_SET,
Packit Service 5ffa24
                                        G_OBJECT_CLASS_TYPE(object_class),
Packit Service 5ffa24
                                        G_SIGNAL_RUN_FIRST,
Packit Service 5ffa24
                                        0,
Packit Service 5ffa24
                                        NULL,
Packit Service 5ffa24
                                        NULL,
Packit Service 5ffa24
                                        NULL,
Packit Service 5ffa24
                                        G_TYPE_NONE,
Packit Service 5ffa24
                                        2,
Packit Service 5ffa24
                                        G_TYPE_INT,
Packit Service 5ffa24
                                        G_TYPE_STRING);
Packit Service 5ffa24
Packit Service 5ffa24
    signals[IP4_CONFIG] = g_signal_new(NM_PPP_MANAGER_SIGNAL_IP4_CONFIG,
Packit Service 5ffa24
                                       G_OBJECT_CLASS_TYPE(object_class),
Packit Service 5ffa24
                                       G_SIGNAL_RUN_FIRST,
Packit Service 5ffa24
                                       0,
Packit Service 5ffa24
                                       NULL,
Packit Service 5ffa24
                                       NULL,
Packit Service 5ffa24
                                       NULL,
Packit Service 5ffa24
                                       G_TYPE_NONE,
Packit Service 5ffa24
                                       1,
Packit Service 5ffa24
                                       G_TYPE_OBJECT);
Packit Service 5ffa24
Packit Service 5ffa24
    signals[IP6_CONFIG] = g_signal_new(NM_PPP_MANAGER_SIGNAL_IP6_CONFIG,
Packit Service 5ffa24
                                       G_OBJECT_CLASS_TYPE(object_class),
Packit Service 5ffa24
                                       G_SIGNAL_RUN_FIRST,
Packit Service 5ffa24
                                       0,
Packit Service 5ffa24
                                       NULL,
Packit Service 5ffa24
                                       NULL,
Packit Service 5ffa24
                                       NULL,
Packit Service 5ffa24
                                       G_TYPE_NONE,
Packit Service 5ffa24
                                       2,
Packit Service 5ffa24
                                       G_TYPE_POINTER,
Packit Service 5ffa24
                                       G_TYPE_OBJECT);
Packit Service 5ffa24
Packit Service 5ffa24
    signals[STATS] = g_signal_new(NM_PPP_MANAGER_SIGNAL_STATS,
Packit Service 5ffa24
                                  G_OBJECT_CLASS_TYPE(object_class),
Packit Service 5ffa24
                                  G_SIGNAL_RUN_FIRST,
Packit Service 5ffa24
                                  0,
Packit Service 5ffa24
                                  NULL,
Packit Service 5ffa24
                                  NULL,
Packit Service 5ffa24
                                  NULL,
Packit Service 5ffa24
                                  G_TYPE_NONE,
Packit Service 5ffa24
                                  2,
Packit Service 5ffa24
                                  G_TYPE_UINT /*guint32 in_bytes*/,
Packit Service 5ffa24
                                  G_TYPE_UINT /*guint32 out_bytes*/);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
NMPPPOps ppp_ops = {
Packit Service 5ffa24
    .create               = _ppp_manager_new,
Packit Service 5ffa24
    .set_route_parameters = _ppp_manager_set_route_parameters,
Packit Service 5ffa24
    .start                = _ppp_manager_start,
Packit Service 5ffa24
    .stop                 = _ppp_manager_stop,
Packit Service 5ffa24
    .stop_cancel          = _ppp_manager_stop_cancel,
Packit Service 5ffa24
};