Blame src/devices/nm-device-ethernet.c

Packit Service b23acc
// SPDX-License-Identifier: GPL-2.0+
Packit Service b23acc
/*
Packit Service b23acc
 * Copyright (C) 2005 - 2014 Red Hat, Inc.
Packit Service b23acc
 * Copyright (C) 2006 - 2008 Novell, Inc.
Packit Service b23acc
 */
Packit Service b23acc
Packit Service b23acc
#include "nm-default.h"
Packit Service b23acc
Packit Service b23acc
#include "nm-device-ethernet.h"
Packit Service b23acc
Packit Service b23acc
#include <netinet/in.h>
Packit Service b23acc
#include <stdlib.h>
Packit Service b23acc
#include <unistd.h>
Packit Service b23acc
#include <libudev.h>
Packit Service b23acc
Packit Service b23acc
#include "nm-device-private.h"
Packit Service b23acc
#include "nm-act-request.h"
Packit Service b23acc
#include "nm-ip4-config.h"
Packit Service b23acc
#include "NetworkManagerUtils.h"
Packit Service b23acc
#include "supplicant/nm-supplicant-manager.h"
Packit Service b23acc
#include "supplicant/nm-supplicant-interface.h"
Packit Service b23acc
#include "supplicant/nm-supplicant-config.h"
Packit Service b23acc
#include "ppp/nm-ppp-manager.h"
Packit Service b23acc
#include "ppp/nm-ppp-manager-call.h"
Packit Service b23acc
#include "ppp/nm-ppp-status.h"
Packit Service b23acc
#include "platform/nm-platform.h"
Packit Service b23acc
#include "platform/nm-platform-utils.h"
Packit Service b23acc
#include "nm-dcb.h"
Packit Service b23acc
#include "settings/nm-settings-connection.h"
Packit Service b23acc
#include "nm-config.h"
Packit Service b23acc
#include "nm-device-ethernet-utils.h"
Packit Service b23acc
#include "settings/nm-settings.h"
Packit Service b23acc
#include "nm-device-factory.h"
Packit Service b23acc
#include "nm-core-internal.h"
Packit Service b23acc
#include "NetworkManagerUtils.h"
Packit Service b23acc
#include "nm-udev-aux/nm-udev-utils.h"
Packit Service b23acc
Packit Service b23acc
#include "nm-device-logging.h"
Packit Service b23acc
_LOG_DECLARE_SELF(NMDeviceEthernet);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
#define PPPOE_RECONNECT_DELAY 7
Packit Service b23acc
#define PPPOE_ENCAP_OVERHEAD  8 /* 2 bytes for PPP, 6 for PPPoE */
Packit Service b23acc
Packit Service b23acc
#define SUPPLICANT_LNK_TIMEOUT_SEC 15
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
typedef enum {
Packit Service b23acc
	DCB_WAIT_UNKNOWN = 0,
Packit Service b23acc
	/* Ensure carrier is up before enabling DCB */
Packit Service b23acc
	DCB_WAIT_CARRIER_PREENABLE_UP,
Packit Service b23acc
	/* Wait for carrier down when device starts enabling */
Packit Service b23acc
	DCB_WAIT_CARRIER_PRECONFIG_DOWN,
Packit Service b23acc
	/* Wait for carrier up when device has finished enabling */
Packit Service b23acc
	DCB_WAIT_CARRIER_PRECONFIG_UP,
Packit Service b23acc
	/* Wait carrier down when device starts configuring */
Packit Service b23acc
	DCB_WAIT_CARRIER_POSTCONFIG_DOWN,
Packit Service b23acc
	/* Wait carrier up when device has finished configuring */
Packit Service b23acc
	DCB_WAIT_CARRIER_POSTCONFIG_UP,
Packit Service b23acc
} DcbWait;
Packit Service b23acc
Packit Service b23acc
typedef struct _NMDeviceEthernetPrivate {
Packit Service b23acc
	/* s390 */
Packit Service b23acc
	char *              subchan1;
Packit Service b23acc
	char *              subchan2;
Packit Service b23acc
	char *              subchan3;
Packit Service b23acc
	char *              subchannels; /* Composite used for checking unmanaged specs */
Packit Service b23acc
	char **             subchannels_dbus; /* Array exported on D-Bus */
Packit Service b23acc
	char *              s390_nettype;
Packit Service b23acc
	GHashTable *        s390_options;
Packit Service b23acc
Packit Service b23acc
	guint32             speed;
Packit Service b23acc
	gulong              carrier_id;
Packit Service b23acc
Packit Service b23acc
	struct {
Packit Service b23acc
		NMSupplicantManager *mgr;
Packit Service b23acc
		NMSupplMgrCreateIfaceHandle *create_handle;
Packit Service b23acc
		NMSupplicantInterface *iface;
Packit Service b23acc
Packit Service b23acc
		gulong iface_state_id;
Packit Service b23acc
		gulong auth_state_id;
Packit Service b23acc
Packit Service b23acc
		guint con_timeout_id;
Packit Service b23acc
Packit Service b23acc
		guint lnk_timeout_id;
Packit Service b23acc
Packit Service b23acc
		bool is_associated:1;
Packit Service b23acc
	} supplicant;
Packit Service b23acc
Packit Service b23acc
	NMActRequestGetSecretsCallId *wired_secrets_id;
Packit Service b23acc
Packit Service b23acc
	/* PPPoE */
Packit Service b23acc
	NMPPPManager *ppp_manager;
Packit Service b23acc
	gint32        last_pppoe_time;
Packit Service b23acc
	guint         pppoe_wait_id;
Packit Service b23acc
Packit Service b23acc
	/* DCB */
Packit Service b23acc
	DcbWait       dcb_wait;
Packit Service b23acc
	guint         dcb_timeout_id;
Packit Service b23acc
Packit Service b23acc
	guint32       ethtool_prev_speed;
Packit Service b23acc
Packit Service b23acc
	NMPlatformLinkDuplexType ethtool_prev_duplex:3;
Packit Service b23acc
Packit Service b23acc
	bool          dcb_handle_carrier_changes:1;
Packit Service b23acc
Packit Service b23acc
	bool          ethtool_prev_set:1;
Packit Service b23acc
	bool          ethtool_prev_autoneg:1;
Packit Service b23acc
Packit Service b23acc
} NMDeviceEthernetPrivate;
Packit Service b23acc
Packit Service b23acc
NM_GOBJECT_PROPERTIES_DEFINE (NMDeviceEthernet,
Packit Service b23acc
	PROP_SPEED,
Packit Service b23acc
	PROP_S390_SUBCHANNELS,
Packit Service b23acc
);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
G_DEFINE_TYPE (NMDeviceEthernet, nm_device_ethernet, NM_TYPE_DEVICE)
Packit Service b23acc
Packit Service b23acc
#define NM_DEVICE_ETHERNET_GET_PRIVATE(self) _NM_GET_PRIVATE_PTR(self, NMDeviceEthernet, NM_IS_DEVICE_ETHERNET, NMDevice)
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
static void wired_secrets_cancel (NMDeviceEthernet *self);
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
static char *
Packit Service b23acc
get_link_basename (const char *parent_path, const char *name, GError **error)
Packit Service b23acc
{
Packit Service b23acc
	char *link_dest, *path;
Packit Service b23acc
	char *result = NULL;
Packit Service b23acc
Packit Service b23acc
	path = g_strdup_printf ("%s/%s", parent_path, name);
Packit Service b23acc
	link_dest = g_file_read_link (path, error);
Packit Service b23acc
	if (link_dest) {
Packit Service b23acc
		result = g_path_get_basename (link_dest);
Packit Service b23acc
		g_free (link_dest);
Packit Service b23acc
	}
Packit Service b23acc
	g_free (path);
Packit Service b23acc
	return result;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
_update_s390_subchannels (NMDeviceEthernet *self)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
	struct udev_device *dev = NULL;
Packit Service b23acc
	struct udev_device *parent = NULL;
Packit Service b23acc
	const char *parent_path, *item;
Packit Service b23acc
	int ifindex;
Packit Service b23acc
	GDir *dir;
Packit Service b23acc
	GError *error = NULL;
Packit Service b23acc
Packit Service b23acc
	if (priv->subchannels) {
Packit Service b23acc
		/* only read the subchannels once. For one, we don't expect them to change
Packit Service b23acc
		 * on multiple invocations. Second, we didn't implement proper reloading.
Packit Service b23acc
		 * Proper reloading might also be complicated, because the subchannels are
Packit Service b23acc
		 * used to match on devices based on a device-spec. Thus, it's not clear
Packit Service b23acc
		 * what it means to change afterwards. */
Packit Service b23acc
		return;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	ifindex = nm_device_get_ifindex ((NMDevice *) self);
Packit Service b23acc
	dev = nm_platform_link_get_udev_device (nm_device_get_platform (NM_DEVICE (self)), ifindex);
Packit Service b23acc
	if (!dev)
Packit Service b23acc
		return;
Packit Service b23acc
Packit Service b23acc
	/* Try for the "ccwgroup" parent */
Packit Service b23acc
	parent = udev_device_get_parent_with_subsystem_devtype (dev, "ccwgroup", NULL);
Packit Service b23acc
	if (!parent) {
Packit Service b23acc
		/* FIXME: whatever 'lcs' devices' subsystem is here... */
Packit Service b23acc
Packit Service b23acc
		/* Not an s390 device */
Packit Service b23acc
		return;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	parent_path = udev_device_get_syspath (parent);
Packit Service b23acc
	dir = g_dir_open (parent_path, 0, &error);
Packit Service b23acc
	if (!dir) {
Packit Service b23acc
		_LOGW (LOGD_DEVICE | LOGD_PLATFORM, "update-s390: failed to open directory '%s': %s",
Packit Service b23acc
		       parent_path, error->message);
Packit Service b23acc
		g_clear_error (&error);
Packit Service b23acc
		return;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	while ((item = g_dir_read_name (dir))) {
Packit Service b23acc
		if (!strcmp (item, "cdev0")) {
Packit Service b23acc
			priv->subchan1 = get_link_basename (parent_path, "cdev0", &error);
Packit Service b23acc
		} else if (!strcmp (item, "cdev1")) {
Packit Service b23acc
			priv->subchan2 = get_link_basename (parent_path, "cdev1", &error);
Packit Service b23acc
		} else if (!strcmp (item, "cdev2")) {
Packit Service b23acc
			priv->subchan3 = get_link_basename (parent_path, "cdev2", &error);
Packit Service b23acc
		} else if (!strcmp (item, "driver")) {
Packit Service b23acc
			priv->s390_nettype = get_link_basename (parent_path, "driver", &error);
Packit Service b23acc
		} else if (   !strcmp (item, "layer2")
Packit Service b23acc
		           || !strcmp (item, "portname")
Packit Service b23acc
		           || !strcmp (item, "portno")) {
Packit Service b23acc
			gs_free char *path = NULL, *value = NULL;
Packit Service b23acc
Packit Service b23acc
			path = g_strdup_printf ("%s/%s", parent_path, item);
Packit Service b23acc
			value = nm_platform_sysctl_get (nm_device_get_platform (NM_DEVICE (self)), NMP_SYSCTL_PATHID_ABSOLUTE (path));
Packit Service b23acc
Packit Service b23acc
			if (   !strcmp (item, "portname")
Packit Service b23acc
			    && !g_strcmp0 (value, "no portname required")) {
Packit Service b23acc
				/* Do nothing */
Packit Service b23acc
			} else if (value && *value) {
Packit Service b23acc
				g_hash_table_insert (priv->s390_options, g_strdup (item), value);
Packit Service b23acc
				value = NULL;
Packit Service b23acc
			} else
Packit Service b23acc
				_LOGW (LOGD_DEVICE | LOGD_PLATFORM, "update-s390: error reading %s", path);
Packit Service b23acc
		}
Packit Service b23acc
Packit Service b23acc
		if (error) {
Packit Service b23acc
			_LOGW (LOGD_DEVICE | LOGD_PLATFORM, "update-s390: failed reading sysfs for %s (%s)", item, error->message);
Packit Service b23acc
			g_clear_error (&error);
Packit Service b23acc
		}
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	g_dir_close (dir);
Packit Service b23acc
Packit Service b23acc
	if (priv->subchan3) {
Packit Service b23acc
		priv->subchannels = g_strdup_printf ("%s,%s,%s",
Packit Service b23acc
		                                     priv->subchan1,
Packit Service b23acc
		                                     priv->subchan2,
Packit Service b23acc
		                                     priv->subchan3);
Packit Service b23acc
	} else if (priv->subchan2) {
Packit Service b23acc
		priv->subchannels = g_strdup_printf ("%s,%s",
Packit Service b23acc
		                                     priv->subchan1,
Packit Service b23acc
		                                     priv->subchan2);
Packit Service b23acc
	} else
Packit Service b23acc
		priv->subchannels = g_strdup (priv->subchan1);
Packit Service b23acc
Packit Service b23acc
	priv->subchannels_dbus = g_new (char *, 3 + 1);
Packit Service b23acc
	priv->subchannels_dbus[0] = g_strdup (priv->subchan1);
Packit Service b23acc
	priv->subchannels_dbus[1] = g_strdup (priv->subchan2);
Packit Service b23acc
	priv->subchannels_dbus[2] = g_strdup (priv->subchan3);
Packit Service b23acc
	priv->subchannels_dbus[3] = NULL;
Packit Service b23acc
Packit Service b23acc
	_LOGI (LOGD_DEVICE | LOGD_PLATFORM, "update-s390: found s390 '%s' subchannels [%s]",
Packit Service b23acc
	       nm_device_get_driver ((NMDevice *) self) ?: "(unknown driver)",
Packit Service b23acc
	       priv->subchannels);
Packit Service b23acc
Packit Service b23acc
	_notify (self, PROP_S390_SUBCHANNELS);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
device_state_changed (NMDevice *device,
Packit Service b23acc
                      NMDeviceState new_state,
Packit Service b23acc
                      NMDeviceState old_state,
Packit Service b23acc
                      NMDeviceStateReason reason)
Packit Service b23acc
{
Packit Service b23acc
	if (new_state > NM_DEVICE_STATE_ACTIVATED)
Packit Service b23acc
		wired_secrets_cancel (NM_DEVICE_ETHERNET (device));
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
nm_device_ethernet_init (NMDeviceEthernet *self)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernetPrivate *priv;
Packit Service b23acc
Packit Service b23acc
	priv = G_TYPE_INSTANCE_GET_PRIVATE (self, NM_TYPE_DEVICE_ETHERNET, NMDeviceEthernetPrivate);
Packit Service b23acc
	self->_priv = priv;
Packit Service b23acc
Packit Service b23acc
	priv->s390_options = g_hash_table_new_full (nm_str_hash, g_str_equal, g_free, g_free);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static NMDeviceCapabilities
Packit Service b23acc
get_generic_capabilities (NMDevice *device)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernet *self = NM_DEVICE_ETHERNET (device);
Packit Service b23acc
	int ifindex = nm_device_get_ifindex (device);
Packit Service b23acc
Packit Service b23acc
	if (ifindex > 0) {
Packit Service b23acc
		if (nm_platform_link_supports_carrier_detect (nm_device_get_platform (device), ifindex))
Packit Service b23acc
			return NM_DEVICE_CAP_CARRIER_DETECT;
Packit Service b23acc
		else {
Packit Service b23acc
			_LOGI (LOGD_PLATFORM, "driver '%s' does not support carrier detection.",
Packit Service b23acc
			       nm_device_get_driver (device));
Packit Service b23acc
		}
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	return NM_DEVICE_CAP_NONE;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static guint32
Packit Service b23acc
_subchannels_count_num (const char * const *array)
Packit Service b23acc
{
Packit Service b23acc
	int i;
Packit Service b23acc
Packit Service b23acc
	if (!array)
Packit Service b23acc
		return 0;
Packit Service b23acc
	for (i = 0; array[i]; i++)
Packit Service b23acc
		/* NOP */;
Packit Service b23acc
	return i;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static gboolean
Packit Service b23acc
match_subchans (NMDeviceEthernet *self, NMSettingWired *s_wired, gboolean *try_mac)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
	const char * const *subchans;
Packit Service b23acc
	guint32 num1, num2;
Packit Service b23acc
	int i;
Packit Service b23acc
Packit Service b23acc
	*try_mac = TRUE;
Packit Service b23acc
Packit Service b23acc
	subchans = nm_setting_wired_get_s390_subchannels (s_wired);
Packit Service b23acc
	num1 = _subchannels_count_num (subchans);
Packit Service b23acc
	num2 = _subchannels_count_num ((const char * const *) priv->subchannels_dbus);
Packit Service b23acc
	/* connection has no subchannels */
Packit Service b23acc
	if (num1 == 0)
Packit Service b23acc
		return TRUE;
Packit Service b23acc
	/* connection requires subchannels but the device has none */
Packit Service b23acc
	if (num2 == 0)
Packit Service b23acc
		return FALSE;
Packit Service b23acc
	/* number of subchannels differ */
Packit Service b23acc
	if (num1 != num2)
Packit Service b23acc
		return FALSE;
Packit Service b23acc
Packit Service b23acc
	/* Make sure each subchannel in the connection is a subchannel of this device */
Packit Service b23acc
	for (i = 0; subchans[i]; i++) {
Packit Service b23acc
		const char *candidate = subchans[i];
Packit Service b23acc
Packit Service b23acc
		if (   (priv->subchan1 && !strcmp (priv->subchan1, candidate))
Packit Service b23acc
		    || (priv->subchan2 && !strcmp (priv->subchan2, candidate))
Packit Service b23acc
		    || (priv->subchan3 && !strcmp (priv->subchan3, candidate)))
Packit Service b23acc
			continue;
Packit Service b23acc
Packit Service b23acc
		return FALSE;  /* a subchannel was not found */
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	*try_mac = FALSE;
Packit Service b23acc
	return TRUE;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static gboolean
Packit Service b23acc
check_connection_compatible (NMDevice *device, NMConnection *connection, GError **error)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernet *self = NM_DEVICE_ETHERNET (device);
Packit Service b23acc
	NMSettingWired *s_wired;
Packit Service b23acc
Packit Service b23acc
	if (!NM_DEVICE_CLASS (nm_device_ethernet_parent_class)->check_connection_compatible (device, connection, error))
Packit Service b23acc
		return FALSE;
Packit Service b23acc
Packit Service b23acc
	if (nm_connection_is_type (connection, NM_SETTING_PPPOE_SETTING_NAME)) {
Packit Service b23acc
		s_wired = nm_connection_get_setting_wired (connection);
Packit Service b23acc
	} else {
Packit Service b23acc
		s_wired = _nm_connection_check_main_setting (connection, NM_SETTING_WIRED_SETTING_NAME, error);
Packit Service b23acc
		if (!s_wired)
Packit Service b23acc
			return FALSE;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	if (s_wired) {
Packit Service b23acc
		const char *mac, *perm_hw_addr;
Packit Service b23acc
		gboolean try_mac = TRUE;
Packit Service b23acc
		const char * const *mac_blacklist;
Packit Service b23acc
		int i;
Packit Service b23acc
Packit Service b23acc
		if (!match_subchans (self, s_wired, &try_mac)) {
Packit Service b23acc
			nm_utils_error_set_literal (error, NM_UTILS_ERROR_CONNECTION_AVAILABLE_TEMPORARY,
Packit Service b23acc
			                            "s390 subchannels don't match");
Packit Service b23acc
			return FALSE;
Packit Service b23acc
		}
Packit Service b23acc
Packit Service b23acc
		perm_hw_addr = nm_device_get_permanent_hw_address (device);
Packit Service b23acc
		mac = nm_setting_wired_get_mac_address (s_wired);
Packit Service b23acc
		if (perm_hw_addr) {
Packit Service b23acc
			if (   try_mac
Packit Service b23acc
			    && mac
Packit Service b23acc
			    && !nm_utils_hwaddr_matches (mac, -1, perm_hw_addr, -1)) {
Packit Service b23acc
				nm_utils_error_set_literal (error, NM_UTILS_ERROR_CONNECTION_AVAILABLE_TEMPORARY,
Packit Service b23acc
				                            "permanent MAC address doesn't match");
Packit Service b23acc
				return FALSE;
Packit Service b23acc
			}
Packit Service b23acc
Packit Service b23acc
			/* Check for MAC address blacklist */
Packit Service b23acc
			mac_blacklist = nm_setting_wired_get_mac_address_blacklist (s_wired);
Packit Service b23acc
			for (i = 0; mac_blacklist[i]; i++) {
Packit Service b23acc
				if (!nm_utils_hwaddr_valid (mac_blacklist[i], ETH_ALEN)) {
Packit Service b23acc
					nm_utils_error_set_literal (error, NM_UTILS_ERROR_CONNECTION_AVAILABLE_TEMPORARY,
Packit Service b23acc
					                            "invalid MAC in blacklist");
Packit Service b23acc
					return FALSE;
Packit Service b23acc
				}
Packit Service b23acc
Packit Service b23acc
				if (nm_utils_hwaddr_matches (mac_blacklist[i], -1, perm_hw_addr, -1)) {
Packit Service b23acc
					nm_utils_error_set_literal (error, NM_UTILS_ERROR_CONNECTION_AVAILABLE_TEMPORARY,
Packit Service b23acc
					                            "permanent MAC address of device blacklisted");
Packit Service b23acc
					return FALSE;
Packit Service b23acc
				}
Packit Service b23acc
			}
Packit Service b23acc
		} else if (mac) {
Packit Service b23acc
			nm_utils_error_set_literal (error, NM_UTILS_ERROR_CONNECTION_AVAILABLE_TEMPORARY,
Packit Service b23acc
			                            "device has no permanent MAC address to match");
Packit Service b23acc
			return FALSE;
Packit Service b23acc
		}
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	return TRUE;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
/* 802.1X */
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
supplicant_interface_release (NMDeviceEthernet *self)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
Packit Service b23acc
	nm_clear_pointer (&priv->supplicant.create_handle, nm_supplicant_manager_create_interface_cancel);
Packit Service b23acc
Packit Service b23acc
	nm_clear_g_source (&priv->supplicant.lnk_timeout_id);
Packit Service b23acc
	nm_clear_g_source (&priv->supplicant.con_timeout_id);
Packit Service b23acc
	nm_clear_g_signal_handler (priv->supplicant.iface, &priv->supplicant.iface_state_id);
Packit Service b23acc
	nm_clear_g_signal_handler (priv->supplicant.iface, &priv->supplicant.auth_state_id);
Packit Service b23acc
Packit Service b23acc
	if (priv->supplicant.iface) {
Packit Service b23acc
		nm_supplicant_interface_disconnect (priv->supplicant.iface);
Packit Service b23acc
		g_clear_object (&priv->supplicant.iface);
Packit Service b23acc
	}
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
supplicant_auth_state_changed (NMSupplicantInterface *iface,
Packit Service b23acc
                               GParamSpec *pspec,
Packit Service b23acc
                               NMDeviceEthernet *self)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
	NMSupplicantAuthState state;
Packit Service b23acc
Packit Service b23acc
	state = nm_supplicant_interface_get_auth_state (priv->supplicant.iface);
Packit Service b23acc
	_LOGD (LOGD_CORE, "supplicant auth state changed to %u", (unsigned) state);
Packit Service b23acc
Packit Service b23acc
	if (state == NM_SUPPLICANT_AUTH_STATE_SUCCESS) {
Packit Service b23acc
		nm_clear_g_signal_handler (priv->supplicant.iface, &priv->supplicant.iface_state_id);
Packit Service b23acc
		nm_device_update_dynamic_ip_setup (NM_DEVICE (self));
Packit Service b23acc
	}
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static gboolean
Packit Service b23acc
wired_auth_is_optional (NMDeviceEthernet *self)
Packit Service b23acc
{
Packit Service b23acc
	NMSetting8021x *s_8021x;
Packit Service b23acc
Packit Service b23acc
	s_8021x = nm_device_get_applied_setting (NM_DEVICE (self), NM_TYPE_SETTING_802_1X);
Packit Service b23acc
	g_return_val_if_fail (s_8021x, FALSE);
Packit Service b23acc
	return nm_setting_802_1x_get_optional (s_8021x);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
wired_auth_cond_fail (NMDeviceEthernet *self, NMDeviceStateReason reason)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
	NMDevice *device = NM_DEVICE (self);
Packit Service b23acc
Packit Service b23acc
	if (wired_auth_is_optional (self)) {
Packit Service b23acc
		_LOGI (LOGD_DEVICE | LOGD_ETHER,
Packit Service b23acc
		       "Activation: (ethernet) 802.1X authentication is optional, continuing after a failure");
Packit Service b23acc
		if (NM_IN_SET (nm_device_get_state (device),
Packit Service b23acc
		               NM_DEVICE_STATE_CONFIG,
Packit Service b23acc
		               NM_DEVICE_STATE_NEED_AUTH))
Packit Service b23acc
			nm_device_activate_schedule_stage3_ip_config_start (device);
Packit Service b23acc
Packit Service b23acc
		if (!priv->supplicant.auth_state_id) {
Packit Service b23acc
			priv->supplicant.auth_state_id = g_signal_connect (priv->supplicant.iface,
Packit Service b23acc
			                                                   "notify::" NM_SUPPLICANT_INTERFACE_AUTH_STATE,
Packit Service b23acc
			                                                   G_CALLBACK (supplicant_auth_state_changed),
Packit Service b23acc
			                                                   self);
Packit Service b23acc
		}
Packit Service b23acc
		return;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	supplicant_interface_release (self);
Packit Service b23acc
	nm_device_state_changed (NM_DEVICE (self),
Packit Service b23acc
	                         NM_DEVICE_STATE_FAILED,
Packit Service b23acc
	                         reason);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
wired_secrets_cb (NMActRequest *req,
Packit Service b23acc
                  NMActRequestGetSecretsCallId *call_id,
Packit Service b23acc
                  NMSettingsConnection *connection,
Packit Service b23acc
                  GError *error,
Packit Service b23acc
                  gpointer user_data)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernet *self = user_data;
Packit Service b23acc
	NMDevice *device = user_data;
Packit Service b23acc
	NMDeviceEthernetPrivate *priv;
Packit Service b23acc
Packit Service b23acc
	g_return_if_fail (NM_IS_DEVICE_ETHERNET (self));
Packit Service b23acc
	g_return_if_fail (NM_IS_ACT_REQUEST (req));
Packit Service b23acc
Packit Service b23acc
	priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
Packit Service b23acc
	g_return_if_fail (priv->wired_secrets_id == call_id);
Packit Service b23acc
Packit Service b23acc
	priv->wired_secrets_id = NULL;
Packit Service b23acc
Packit Service b23acc
	if (g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
Packit Service b23acc
		return;
Packit Service b23acc
Packit Service b23acc
	g_return_if_fail (req == nm_device_get_act_request (device));
Packit Service b23acc
	g_return_if_fail (nm_device_get_state (device) == NM_DEVICE_STATE_NEED_AUTH);
Packit Service b23acc
	g_return_if_fail (nm_act_request_get_settings_connection (req) == connection);
Packit Service b23acc
Packit Service b23acc
	if (error) {
Packit Service b23acc
		_LOGW (LOGD_ETHER, "%s", error->message);
Packit Service b23acc
		wired_auth_cond_fail (self, NM_DEVICE_STATE_REASON_NO_SECRETS);
Packit Service b23acc
		return;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	supplicant_interface_release (self);
Packit Service b23acc
	nm_device_activate_schedule_stage1_device_prepare (device, FALSE);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
wired_secrets_cancel (NMDeviceEthernet *self)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
Packit Service b23acc
	if (priv->wired_secrets_id)
Packit Service b23acc
		nm_act_request_cancel_secrets (NULL, priv->wired_secrets_id);
Packit Service b23acc
	nm_assert (!priv->wired_secrets_id);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
wired_secrets_get_secrets (NMDeviceEthernet *self,
Packit Service b23acc
                           const char *setting_name,
Packit Service b23acc
                           NMSecretAgentGetSecretsFlags flags)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
	NMActRequest *req;
Packit Service b23acc
Packit Service b23acc
	wired_secrets_cancel (self);
Packit Service b23acc
Packit Service b23acc
	req = nm_device_get_act_request (NM_DEVICE (self));
Packit Service b23acc
	g_return_if_fail (NM_IS_ACT_REQUEST (req));
Packit Service b23acc
Packit Service b23acc
	priv->wired_secrets_id = nm_act_request_get_secrets (req,
Packit Service b23acc
	                                                     TRUE,
Packit Service b23acc
	                                                     setting_name,
Packit Service b23acc
	                                                     flags,
Packit Service b23acc
	                                                     NULL,
Packit Service b23acc
	                                                     wired_secrets_cb,
Packit Service b23acc
	                                                     self);
Packit Service b23acc
	g_return_if_fail (priv->wired_secrets_id);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static gboolean
Packit Service b23acc
supplicant_lnk_timeout_cb (gpointer user_data)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernet *self = NM_DEVICE_ETHERNET (user_data);
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
	NMDevice *device = NM_DEVICE (self);
Packit Service b23acc
	NMActRequest *req;
Packit Service b23acc
	NMConnection *applied_connection;
Packit Service b23acc
	const char *setting_name;
Packit Service b23acc
Packit Service b23acc
	priv->supplicant.lnk_timeout_id = 0;
Packit Service b23acc
Packit Service b23acc
	req = nm_device_get_act_request (device);
Packit Service b23acc
Packit Service b23acc
	if (nm_device_get_state (device) == NM_DEVICE_STATE_ACTIVATED) {
Packit Service b23acc
		wired_auth_cond_fail (self, NM_DEVICE_STATE_REASON_SUPPLICANT_TIMEOUT);
Packit Service b23acc
		return G_SOURCE_REMOVE;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	/* Disconnect event during initial authentication and credentials
Packit Service b23acc
	 * ARE checked - we are likely to have wrong key.  Ask the user for
Packit Service b23acc
	 * another one.
Packit Service b23acc
	 */
Packit Service b23acc
	if (nm_device_get_state (device) != NM_DEVICE_STATE_CONFIG)
Packit Service b23acc
		goto time_out;
Packit Service b23acc
Packit Service b23acc
	nm_active_connection_clear_secrets (NM_ACTIVE_CONNECTION (req));
Packit Service b23acc
Packit Service b23acc
	applied_connection = nm_act_request_get_applied_connection (req);
Packit Service b23acc
	setting_name = nm_connection_need_secrets (applied_connection, NULL);
Packit Service b23acc
	if (!setting_name)
Packit Service b23acc
		goto time_out;
Packit Service b23acc
Packit Service b23acc
	_LOGI (LOGD_DEVICE | LOGD_ETHER,
Packit Service b23acc
	       "Activation: (ethernet) disconnected during authentication, asking for new key.");
Packit Service b23acc
	if (!wired_auth_is_optional (self))
Packit Service b23acc
		supplicant_interface_release (self);
Packit Service b23acc
Packit Service b23acc
	nm_device_state_changed (device, NM_DEVICE_STATE_NEED_AUTH, NM_DEVICE_STATE_REASON_SUPPLICANT_DISCONNECT);
Packit Service b23acc
	wired_secrets_get_secrets (self, setting_name, NM_SECRET_AGENT_GET_SECRETS_FLAG_REQUEST_NEW);
Packit Service b23acc
Packit Service b23acc
	return G_SOURCE_REMOVE;
Packit Service b23acc
Packit Service b23acc
time_out:
Packit Service b23acc
	_LOGW (LOGD_DEVICE | LOGD_ETHER, "link timed out.");
Packit Service b23acc
	wired_auth_cond_fail (self, NM_DEVICE_STATE_REASON_SUPPLICANT_DISCONNECT);
Packit Service b23acc
Packit Service b23acc
	return G_SOURCE_REMOVE;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static NMSupplicantConfig *
Packit Service b23acc
build_supplicant_config (NMDeviceEthernet *self,
Packit Service b23acc
                         GError **error)
Packit Service b23acc
{
Packit Service b23acc
	const char *con_uuid;
Packit Service b23acc
	NMSupplicantConfig *config = NULL;
Packit Service b23acc
	NMSetting8021x *security;
Packit Service b23acc
	NMConnection *connection;
Packit Service b23acc
	guint32 mtu;
Packit Service b23acc
Packit Service b23acc
	connection = nm_device_get_applied_connection (NM_DEVICE (self));
Packit Service b23acc
Packit Service b23acc
	g_return_val_if_fail (connection, NULL);
Packit Service b23acc
Packit Service b23acc
	con_uuid = nm_connection_get_uuid (connection);
Packit Service b23acc
	mtu = nm_platform_link_get_mtu (nm_device_get_platform (NM_DEVICE (self)),
Packit Service b23acc
	                                nm_device_get_ifindex (NM_DEVICE (self)));
Packit Service b23acc
Packit Service b23acc
	config = nm_supplicant_config_new (NM_SUPPL_CAP_MASK_NONE);
Packit Service b23acc
Packit Service b23acc
	security = nm_connection_get_setting_802_1x (connection);
Packit Service b23acc
	if (!nm_supplicant_config_add_setting_8021x (config, security, con_uuid, mtu, TRUE, error)) {
Packit Service b23acc
		g_prefix_error (error, "802-1x-setting: ");
Packit Service b23acc
		g_clear_object (&config);
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	return config;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
supplicant_iface_state_is_completed (NMDeviceEthernet *self,
Packit Service b23acc
                                     NMSupplicantInterfaceState state)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
Packit Service b23acc
	if (state == NM_SUPPLICANT_INTERFACE_STATE_COMPLETED) {
Packit Service b23acc
		nm_clear_g_source (&priv->supplicant.lnk_timeout_id);
Packit Service b23acc
		nm_clear_g_source (&priv->supplicant.con_timeout_id);
Packit Service b23acc
Packit Service b23acc
		/* If this is the initial association during device activation,
Packit Service b23acc
		 * schedule the next activation stage.
Packit Service b23acc
		 */
Packit Service b23acc
		if (nm_device_get_state (NM_DEVICE (self)) == NM_DEVICE_STATE_CONFIG) {
Packit Service b23acc
			_LOGI (LOGD_DEVICE | LOGD_ETHER,
Packit Service b23acc
			       "Activation: (ethernet) Stage 2 of 5 (Device Configure) successful.");
Packit Service b23acc
			nm_device_activate_schedule_stage3_ip_config_start (NM_DEVICE (self));
Packit Service b23acc
		}
Packit Service b23acc
		return;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	if (   !priv->supplicant.lnk_timeout_id
Packit Service b23acc
	    && !priv->supplicant.con_timeout_id)
Packit Service b23acc
		priv->supplicant.lnk_timeout_id = g_timeout_add_seconds (SUPPLICANT_LNK_TIMEOUT_SEC, supplicant_lnk_timeout_cb, self);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
supplicant_iface_assoc_cb (NMSupplicantInterface *iface,
Packit Service b23acc
                           GError *error,
Packit Service b23acc
                           gpointer user_data)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernet *self;
Packit Service b23acc
	NMDeviceEthernetPrivate *priv;
Packit Service b23acc
Packit Service b23acc
	if (nm_utils_error_is_cancelled_or_disposing (error))
Packit Service b23acc
		return;
Packit Service b23acc
Packit Service b23acc
	self = NM_DEVICE_ETHERNET (user_data);
Packit Service b23acc
	priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
Packit Service b23acc
	if (error) {
Packit Service b23acc
		supplicant_interface_release (self);
Packit Service b23acc
		nm_device_queue_state (NM_DEVICE (self),
Packit Service b23acc
		                       NM_DEVICE_STATE_FAILED,
Packit Service b23acc
		                       NM_DEVICE_STATE_REASON_SUPPLICANT_CONFIG_FAILED);
Packit Service b23acc
		return;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	nm_assert (!priv->supplicant.lnk_timeout_id);
Packit Service b23acc
	nm_assert (!priv->supplicant.is_associated);
Packit Service b23acc
Packit Service b23acc
	priv->supplicant.is_associated = TRUE;
Packit Service b23acc
	supplicant_iface_state_is_completed (self,
Packit Service b23acc
	                                     nm_supplicant_interface_get_state (priv->supplicant.iface));
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static gboolean
Packit Service b23acc
supplicant_iface_start (NMDeviceEthernet *self)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
	gs_unref_object NMSupplicantConfig *config = NULL;
Packit Service b23acc
	gs_free_error GError *error = NULL;
Packit Service b23acc
Packit Service b23acc
	config = build_supplicant_config (self, &error);
Packit Service b23acc
	if (!config) {
Packit Service b23acc
		_LOGE (LOGD_DEVICE | LOGD_ETHER,
Packit Service b23acc
		       "Activation: (ethernet) couldn't build security configuration: %s",
Packit Service b23acc
		       error->message);
Packit Service b23acc
		supplicant_interface_release (self);
Packit Service b23acc
		nm_device_state_changed (NM_DEVICE (self),
Packit Service b23acc
		                         NM_DEVICE_STATE_FAILED,
Packit Service b23acc
		                         NM_DEVICE_STATE_REASON_SUPPLICANT_CONFIG_FAILED);
Packit Service b23acc
		return FALSE;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	nm_supplicant_interface_disconnect (priv->supplicant.iface);
Packit Service b23acc
	nm_supplicant_interface_assoc (priv->supplicant.iface,
Packit Service b23acc
	                               config,
Packit Service b23acc
	                               supplicant_iface_assoc_cb,
Packit Service b23acc
	                               self);
Packit Service b23acc
	return TRUE;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
supplicant_iface_state_cb (NMSupplicantInterface *iface,
Packit Service b23acc
                           int new_state_i,
Packit Service b23acc
                           int old_state_i,
Packit Service b23acc
                           int disconnect_reason,
Packit Service b23acc
                           gpointer user_data)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernet *self = NM_DEVICE_ETHERNET (user_data);
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
	NMSupplicantInterfaceState new_state = new_state_i;
Packit Service b23acc
	NMSupplicantInterfaceState old_state = old_state_i;
Packit Service b23acc
Packit Service b23acc
	_LOGI (LOGD_DEVICE | LOGD_ETHER, "supplicant interface state: %s -> %s",
Packit Service b23acc
	       nm_supplicant_interface_state_to_string (old_state),
Packit Service b23acc
	       nm_supplicant_interface_state_to_string (new_state));
Packit Service b23acc
Packit Service b23acc
	if (new_state == NM_SUPPLICANT_INTERFACE_STATE_DOWN) {
Packit Service b23acc
		supplicant_interface_release (self);
Packit Service b23acc
		wired_auth_cond_fail (self, NM_DEVICE_STATE_REASON_SUPPLICANT_FAILED);
Packit Service b23acc
		return;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	if (old_state == NM_SUPPLICANT_INTERFACE_STATE_STARTING) {
Packit Service b23acc
		if (!supplicant_iface_start (self))
Packit Service b23acc
			return;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	if (priv->supplicant.is_associated)
Packit Service b23acc
		supplicant_iface_state_is_completed (self, new_state);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static gboolean
Packit Service b23acc
handle_auth_or_fail (NMDeviceEthernet *self,
Packit Service b23acc
                     NMActRequest *req,
Packit Service b23acc
                     gboolean new_secrets)
Packit Service b23acc
{
Packit Service b23acc
	const char *setting_name;
Packit Service b23acc
	NMConnection *applied_connection;
Packit Service b23acc
Packit Service b23acc
	if (!nm_device_auth_retries_try_next (NM_DEVICE (self)))
Packit Service b23acc
		return FALSE;
Packit Service b23acc
Packit Service b23acc
	nm_device_state_changed (NM_DEVICE (self), NM_DEVICE_STATE_NEED_AUTH, NM_DEVICE_STATE_REASON_NONE);
Packit Service b23acc
Packit Service b23acc
	nm_active_connection_clear_secrets (NM_ACTIVE_CONNECTION (req));
Packit Service b23acc
Packit Service b23acc
	applied_connection = nm_act_request_get_applied_connection (req);
Packit Service b23acc
	setting_name = nm_connection_need_secrets (applied_connection, NULL);
Packit Service b23acc
	if (!setting_name) {
Packit Service b23acc
		_LOGI (LOGD_DEVICE, "Cleared secrets, but setting didn't need any secrets.");
Packit Service b23acc
		return FALSE;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	_LOGI (LOGD_DEVICE | LOGD_ETHER, "Activation: (ethernet) asking for new secrets");
Packit Service b23acc
Packit Service b23acc
	/* Don't tear down supplicant if the authentication is optional
Packit Service b23acc
	 * because in case of a failure in getting new secrets we want to
Packit Service b23acc
	 * keep the supplicant alive.
Packit Service b23acc
	 */
Packit Service b23acc
	if (!wired_auth_is_optional (self))
Packit Service b23acc
		supplicant_interface_release (self);
Packit Service b23acc
Packit Service b23acc
	wired_secrets_get_secrets (self, setting_name,
Packit Service b23acc
	                             NM_SECRET_AGENT_GET_SECRETS_FLAG_ALLOW_INTERACTION
Packit Service b23acc
	                           | (new_secrets ? NM_SECRET_AGENT_GET_SECRETS_FLAG_REQUEST_NEW : 0));
Packit Service b23acc
	return TRUE;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static gboolean
Packit Service b23acc
supplicant_connection_timeout_cb (gpointer user_data)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernet *self = NM_DEVICE_ETHERNET (user_data);
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
	NMDevice *device = NM_DEVICE (self);
Packit Service b23acc
	NMActRequest *req;
Packit Service b23acc
	NMSettingsConnection *connection;
Packit Service b23acc
	guint64 timestamp = 0;
Packit Service b23acc
	gboolean new_secrets = TRUE;
Packit Service b23acc
Packit Service b23acc
	priv->supplicant.con_timeout_id = 0;
Packit Service b23acc
Packit Service b23acc
	/* Authentication failed; either driver problems, the encryption key is
Packit Service b23acc
	 * wrong, the passwords or certificates were wrong or the Ethernet switch's
Packit Service b23acc
	 * port is not configured for 802.1x. */
Packit Service b23acc
	_LOGW (LOGD_DEVICE | LOGD_ETHER,
Packit Service b23acc
	       "Activation: (ethernet) association took too long.");
Packit Service b23acc
Packit Service b23acc
	req = nm_device_get_act_request (device);
Packit Service b23acc
	connection = nm_act_request_get_settings_connection (req);
Packit Service b23acc
Packit Service b23acc
	/* Ask for new secrets only if we've never activated this connection
Packit Service b23acc
	 * before.  If we've connected before, don't bother the user with dialogs,
Packit Service b23acc
	 * just retry or fail, and if we never connect the user can fix the
Packit Service b23acc
	 * password somewhere else. */
Packit Service b23acc
	if (nm_settings_connection_get_timestamp (connection, &timestamp))
Packit Service b23acc
		new_secrets = !timestamp;
Packit Service b23acc
Packit Service b23acc
	if (!handle_auth_or_fail (self, req, new_secrets)) {
Packit Service b23acc
		wired_auth_cond_fail (self, NM_DEVICE_STATE_REASON_NO_SECRETS);
Packit Service b23acc
		return G_SOURCE_REMOVE;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	if (   !priv->supplicant.lnk_timeout_id
Packit Service b23acc
	    && priv->supplicant.iface) {
Packit Service b23acc
		NMSupplicantInterfaceState state;
Packit Service b23acc
Packit Service b23acc
		state = nm_supplicant_interface_get_state (priv->supplicant.iface);
Packit Service b23acc
		if (state != NM_SUPPLICANT_INTERFACE_STATE_COMPLETED
Packit Service b23acc
		    && nm_supplicant_interface_state_is_operational (state))
Packit Service b23acc
			priv->supplicant.lnk_timeout_id = g_timeout_add_seconds (SUPPLICANT_LNK_TIMEOUT_SEC, supplicant_lnk_timeout_cb, self);
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	return G_SOURCE_REMOVE;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
supplicant_interface_create_cb (NMSupplicantManager *supplicant_manager,
Packit Service b23acc
                                NMSupplMgrCreateIfaceHandle *handle,
Packit Service b23acc
                                NMSupplicantInterface *iface,
Packit Service b23acc
                                GError *error,
Packit Service b23acc
                                gpointer user_data)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernet *self;
Packit Service b23acc
	NMDeviceEthernetPrivate *priv;
Packit Service b23acc
	guint timeout;
Packit Service b23acc
Packit Service b23acc
	if (nm_utils_error_is_cancelled (error))
Packit Service b23acc
		return;
Packit Service b23acc
Packit Service b23acc
	self = user_data;
Packit Service b23acc
	priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
Packit Service b23acc
	nm_assert (priv->supplicant.create_handle == handle);
Packit Service b23acc
	priv->supplicant.create_handle = NULL;
Packit Service b23acc
Packit Service b23acc
	if (error) {
Packit Service b23acc
		_LOGE (LOGD_DEVICE | LOGD_ETHER,
Packit Service b23acc
		       "Couldn't initialize supplicant interface: %s",
Packit Service b23acc
		       error->message);
Packit Service b23acc
		supplicant_interface_release (self);
Packit Service b23acc
		nm_device_state_changed (NM_DEVICE (self),
Packit Service b23acc
		                         NM_DEVICE_STATE_FAILED,
Packit Service b23acc
		                         NM_DEVICE_STATE_REASON_SUPPLICANT_FAILED);
Packit Service b23acc
		return;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	priv->supplicant.iface = g_object_ref (iface);
Packit Service b23acc
	priv->supplicant.is_associated = FALSE;
Packit Service b23acc
Packit Service b23acc
	priv->supplicant.iface_state_id = g_signal_connect (priv->supplicant.iface,
Packit Service b23acc
	                                                    NM_SUPPLICANT_INTERFACE_STATE,
Packit Service b23acc
	                                                    G_CALLBACK (supplicant_iface_state_cb),
Packit Service b23acc
	                                                    self);
Packit Service b23acc
Packit Service b23acc
	timeout = nm_device_get_supplicant_timeout (NM_DEVICE (self));
Packit Service b23acc
	priv->supplicant.con_timeout_id = g_timeout_add_seconds (timeout,
Packit Service b23acc
	                                                         supplicant_connection_timeout_cb,
Packit Service b23acc
	                                                         self);
Packit Service b23acc
Packit Service b23acc
	if (nm_supplicant_interface_state_is_operational (nm_supplicant_interface_get_state (iface)))
Packit Service b23acc
		supplicant_iface_start (self);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static NMPlatformLinkDuplexType
Packit Service b23acc
link_duplex_to_platform (const char *duplex)
Packit Service b23acc
{
Packit Service b23acc
	if (!duplex)
Packit Service b23acc
		return NM_PLATFORM_LINK_DUPLEX_UNKNOWN;
Packit Service b23acc
	if (nm_streq (duplex, "full"))
Packit Service b23acc
		return NM_PLATFORM_LINK_DUPLEX_FULL;
Packit Service b23acc
	if (nm_streq (duplex, "half"))
Packit Service b23acc
		return NM_PLATFORM_LINK_DUPLEX_HALF;
Packit Service b23acc
	g_return_val_if_reached (NM_PLATFORM_LINK_DUPLEX_UNKNOWN);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
link_negotiation_set (NMDevice *device)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernet *self = NM_DEVICE_ETHERNET (device);
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
	NMSettingWired *s_wired;
Packit Service b23acc
	gboolean autoneg = TRUE;
Packit Service b23acc
	gboolean link_autoneg;
Packit Service b23acc
	NMPlatformLinkDuplexType duplex = NM_PLATFORM_LINK_DUPLEX_UNKNOWN;
Packit Service b23acc
	NMPlatformLinkDuplexType link_duplex;
Packit Service b23acc
	guint32 speed = 0;
Packit Service b23acc
	guint32 link_speed;
Packit Service b23acc
Packit Service b23acc
	s_wired = nm_device_get_applied_setting (device, NM_TYPE_SETTING_WIRED);
Packit Service b23acc
	if (s_wired) {
Packit Service b23acc
		autoneg = nm_setting_wired_get_auto_negotiate (s_wired);
Packit Service b23acc
		speed = nm_setting_wired_get_speed (s_wired);
Packit Service b23acc
		duplex = link_duplex_to_platform (nm_setting_wired_get_duplex (s_wired));
Packit Service b23acc
		if (   !autoneg
Packit Service b23acc
		    && !speed
Packit Service b23acc
		    && !duplex) {
Packit Service b23acc
			_LOGD (LOGD_DEVICE, "set-link: ignore link negotiation");
Packit Service b23acc
			return;
Packit Service b23acc
		}
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	if (!nm_platform_ethtool_get_link_settings (nm_device_get_platform (device),
Packit Service b23acc
	                                            nm_device_get_ifindex (device),
Packit Service b23acc
	                                            &link_autoneg,
Packit Service b23acc
	                                            &link_speed,
Packit Service b23acc
	                                            &link_duplex)) {
Packit Service b23acc
		_LOGW (LOGD_DEVICE, "set-link: unable to retrieve link negotiation");
Packit Service b23acc
		return;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	/* If link negotiation setting are already in place do nothing and return with success */
Packit Service b23acc
	if (   !!autoneg == !!link_autoneg
Packit Service b23acc
	    && speed == link_speed
Packit Service b23acc
	    && duplex == link_duplex) {
Packit Service b23acc
		_LOGD (LOGD_DEVICE, "set-link: link negotiation is already configured");
Packit Service b23acc
		return;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	if (   autoneg
Packit Service b23acc
	    && !speed
Packit Service b23acc
	    && !duplex)
Packit Service b23acc
		_LOGD (LOGD_DEVICE, "set-link: configure auto-negotiation");
Packit Service b23acc
	else {
Packit Service b23acc
		_LOGD (LOGD_DEVICE, "set-link: configure %snegotiation (%u Mbit%s, %s duplex%s)",
Packit Service b23acc
		       autoneg ? "auto-" : "static ",
Packit Service b23acc
		       speed ?: link_speed,
Packit Service b23acc
		       speed ? "" : "*",
Packit Service b23acc
		         duplex
Packit Service b23acc
		       ? nm_platform_link_duplex_type_to_string (duplex)
Packit Service b23acc
		       : nm_platform_link_duplex_type_to_string (link_duplex),
Packit Service b23acc
		       duplex ? "" : "*");
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	if (!priv->ethtool_prev_set) {
Packit Service b23acc
		/* remember the values we had before setting it. */
Packit Service b23acc
		priv->ethtool_prev_autoneg = link_autoneg;
Packit Service b23acc
		priv->ethtool_prev_speed = link_speed;
Packit Service b23acc
		priv->ethtool_prev_duplex = link_duplex;
Packit Service b23acc
		priv->ethtool_prev_set = TRUE;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	if (!nm_platform_ethtool_set_link_settings (nm_device_get_platform (device),
Packit Service b23acc
	                                            nm_device_get_ifindex (device),
Packit Service b23acc
	                                            autoneg,
Packit Service b23acc
	                                            speed,
Packit Service b23acc
	                                            duplex)) {
Packit Service b23acc
		_LOGW (LOGD_DEVICE, "set-link: failure to set link negotiation");
Packit Service b23acc
		return;
Packit Service b23acc
	}
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static gboolean
Packit Service b23acc
pppoe_reconnect_delay (gpointer user_data)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernet *self = NM_DEVICE_ETHERNET (user_data);
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
Packit Service b23acc
	priv->pppoe_wait_id = 0;
Packit Service b23acc
	priv->last_pppoe_time = 0;
Packit Service b23acc
	_LOGI (LOGD_DEVICE, "PPPoE reconnect delay complete, resuming connection...");
Packit Service b23acc
	nm_device_activate_schedule_stage1_device_prepare (NM_DEVICE (self), FALSE);
Packit Service b23acc
	return G_SOURCE_REMOVE;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static NMActStageReturn
Packit Service b23acc
act_stage1_prepare (NMDevice *device, NMDeviceStateReason *out_failure_reason)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernet *self = NM_DEVICE_ETHERNET (device);
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
Packit Service b23acc
	if (nm_device_sys_iface_state_is_external_or_assume (device)) {
Packit Service b23acc
		if (   !priv->ethtool_prev_set
Packit Service b23acc
		    && !nm_device_sys_iface_state_is_external (device)) {
Packit Service b23acc
			NMSettingWired *s_wired;
Packit Service b23acc
Packit Service b23acc
			/* During restart of NetworkManager service we forget the original auto
Packit Service b23acc
			 * negotiation settings. When taking over a device, remember to reset
Packit Service b23acc
			 * the "default" during deactivate. */
Packit Service b23acc
			s_wired = nm_device_get_applied_setting (device, NM_TYPE_SETTING_WIRED);
Packit Service b23acc
			if (   s_wired
Packit Service b23acc
			    && (   nm_setting_wired_get_auto_negotiate (s_wired)
Packit Service b23acc
			        || nm_setting_wired_get_speed (s_wired)
Packit Service b23acc
			        || nm_setting_wired_get_duplex (s_wired))) {
Packit Service b23acc
				priv->ethtool_prev_set = TRUE;
Packit Service b23acc
				priv->ethtool_prev_autoneg = TRUE;
Packit Service b23acc
				priv->ethtool_prev_speed = 0;
Packit Service b23acc
				priv->ethtool_prev_duplex = NM_PLATFORM_LINK_DUPLEX_UNKNOWN;
Packit Service b23acc
			}
Packit Service b23acc
		}
Packit Service b23acc
		return NM_ACT_STAGE_RETURN_SUCCESS;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	link_negotiation_set (device);
Packit Service b23acc
Packit Service b23acc
	/* If we're re-activating a PPPoE connection a short while after
Packit Service b23acc
	 * a previous PPPoE connection was torn down, wait a bit to allow the
Packit Service b23acc
	 * remote side to handle the disconnection.  Otherwise the peer may
Packit Service b23acc
	 * get confused and fail to negotiate the new connection. (rh #1023503)
Packit Service b23acc
	 *
Packit Service b23acc
	 * FIXME(shutdown): when exiting, we also need to wait before quiting,
Packit Service b23acc
	 * at least for additional NM_SHUTDOWN_TIMEOUT_MS seconds because
Packit Service b23acc
	 * otherwise after restart the device won't work for the first seconds.
Packit Service b23acc
	 */
Packit Service b23acc
	if (priv->last_pppoe_time != 0) {
Packit Service b23acc
		gint32 delay = nm_utils_get_monotonic_timestamp_sec () - priv->last_pppoe_time;
Packit Service b23acc
Packit Service b23acc
		if (   delay < PPPOE_RECONNECT_DELAY
Packit Service b23acc
		    && nm_device_get_applied_setting (device, NM_TYPE_SETTING_PPPOE)) {
Packit Service b23acc
			if (priv->pppoe_wait_id == 0) {
Packit Service b23acc
				_LOGI (LOGD_DEVICE, "delaying PPPoE reconnect for %d seconds to ensure peer is ready...",
Packit Service b23acc
				       delay);
Packit Service b23acc
				priv->pppoe_wait_id = g_timeout_add_seconds (delay,
Packit Service b23acc
				                                             pppoe_reconnect_delay,
Packit Service b23acc
				                                             self);
Packit Service b23acc
			}
Packit Service b23acc
			return NM_ACT_STAGE_RETURN_POSTPONE;
Packit Service b23acc
		}
Packit Service b23acc
		nm_clear_g_source (&priv->pppoe_wait_id);
Packit Service b23acc
		priv->last_pppoe_time = 0;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	return NM_ACT_STAGE_RETURN_SUCCESS;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static NMActStageReturn
Packit Service b23acc
supplicant_check_secrets_needed (NMDeviceEthernet *self, NMDeviceStateReason *out_failure_reason)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
	NMConnection *connection;
Packit Service b23acc
	NMSetting8021x *security;
Packit Service b23acc
	const char *setting_name;
Packit Service b23acc
Packit Service b23acc
	connection = nm_device_get_applied_connection (NM_DEVICE (self));
Packit Service b23acc
	g_return_val_if_fail (connection, NM_ACT_STAGE_RETURN_FAILURE);
Packit Service b23acc
Packit Service b23acc
	security = nm_connection_get_setting_802_1x (connection);
Packit Service b23acc
	if (!security) {
Packit Service b23acc
		_LOGE (LOGD_DEVICE, "Invalid or missing 802.1X security");
Packit Service b23acc
		NM_SET_OUT (out_failure_reason, NM_DEVICE_STATE_REASON_CONFIG_FAILED);
Packit Service b23acc
		return NM_ACT_STAGE_RETURN_FAILURE;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	if (!priv->supplicant.mgr)
Packit Service b23acc
		priv->supplicant.mgr = g_object_ref (nm_supplicant_manager_get ());
Packit Service b23acc
Packit Service b23acc
	/* If we need secrets, get them */
Packit Service b23acc
	setting_name = nm_connection_need_secrets (connection, NULL);
Packit Service b23acc
	if (setting_name) {
Packit Service b23acc
		NMActRequest *req = nm_device_get_act_request (NM_DEVICE (self));
Packit Service b23acc
Packit Service b23acc
		_LOGI (LOGD_DEVICE | LOGD_ETHER,
Packit Service b23acc
		       "Activation: (ethernet) connection '%s' has security, but secrets are required.",
Packit Service b23acc
		       nm_connection_get_id (connection));
Packit Service b23acc
Packit Service b23acc
		if (!handle_auth_or_fail (self, req, FALSE)) {
Packit Service b23acc
			NM_SET_OUT (out_failure_reason, NM_DEVICE_STATE_REASON_NO_SECRETS);
Packit Service b23acc
			return NM_ACT_STAGE_RETURN_FAILURE;
Packit Service b23acc
		}
Packit Service b23acc
		return NM_ACT_STAGE_RETURN_POSTPONE;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	_LOGI (LOGD_DEVICE | LOGD_ETHER,
Packit Service b23acc
	       "Activation: (ethernet) connection '%s' requires no security. No secrets needed.",
Packit Service b23acc
	       nm_connection_get_id (connection));
Packit Service b23acc
Packit Service b23acc
	supplicant_interface_release (self);
Packit Service b23acc
Packit Service b23acc
	priv->supplicant.create_handle = nm_supplicant_manager_create_interface (priv->supplicant.mgr,
Packit Service b23acc
	                                                                         nm_device_get_ifindex (NM_DEVICE (self)),
Packit Service b23acc
	                                                                         NM_SUPPLICANT_DRIVER_WIRED,
Packit Service b23acc
	                                                                         supplicant_interface_create_cb,
Packit Service b23acc
	                                                                         self);
Packit Service b23acc
	return NM_ACT_STAGE_RETURN_POSTPONE;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
carrier_changed (NMSupplicantInterface *iface,
Packit Service b23acc
                 GParamSpec *pspec,
Packit Service b23acc
                 NMDeviceEthernet *self)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
	NMDeviceStateReason reason;
Packit Service b23acc
	NMActStageReturn ret;
Packit Service b23acc
Packit Service b23acc
	if (!nm_device_has_carrier (NM_DEVICE (self)))
Packit Service b23acc
		return;
Packit Service b23acc
Packit Service b23acc
	_LOGD (LOGD_DEVICE | LOGD_ETHER, "got carrier, initializing supplicant");
Packit Service b23acc
	nm_clear_g_signal_handler (self, &priv->carrier_id);
Packit Service b23acc
	ret = supplicant_check_secrets_needed (self, &reason);
Packit Service b23acc
	if (ret == NM_ACT_STAGE_RETURN_FAILURE) {
Packit Service b23acc
		nm_device_state_changed (NM_DEVICE (self),
Packit Service b23acc
		                         NM_DEVICE_STATE_FAILED,
Packit Service b23acc
		                         reason);
Packit Service b23acc
	}
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
/* PPPoE */
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
ppp_state_changed (NMPPPManager *ppp_manager, NMPPPStatus status, gpointer user_data)
Packit Service b23acc
{
Packit Service b23acc
	NMDevice *device = NM_DEVICE (user_data);
Packit Service b23acc
Packit Service b23acc
	switch (status) {
Packit Service b23acc
	case NM_PPP_STATUS_DISCONNECT:
Packit Service b23acc
		nm_device_state_changed (device, NM_DEVICE_STATE_FAILED, NM_DEVICE_STATE_REASON_PPP_DISCONNECT);
Packit Service b23acc
		break;
Packit Service b23acc
	case NM_PPP_STATUS_DEAD:
Packit Service b23acc
		nm_device_state_changed (device, NM_DEVICE_STATE_FAILED, NM_DEVICE_STATE_REASON_PPP_FAILED);
Packit Service b23acc
		break;
Packit Service b23acc
	default:
Packit Service b23acc
		break;
Packit Service b23acc
	}
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
ppp_ifindex_set (NMPPPManager *ppp_manager,
Packit Service b23acc
                 int ifindex,
Packit Service b23acc
                 const char *iface,
Packit Service b23acc
                 gpointer user_data)
Packit Service b23acc
{
Packit Service b23acc
	NMDevice *device = NM_DEVICE (user_data);
Packit Service b23acc
Packit Service b23acc
	if (!nm_device_set_ip_ifindex (device, ifindex)) {
Packit Service b23acc
		nm_device_state_changed (device,
Packit Service b23acc
		                         NM_DEVICE_STATE_FAILED,
Packit Service b23acc
		                         NM_DEVICE_STATE_REASON_IP_CONFIG_UNAVAILABLE);
Packit Service b23acc
	}
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
ppp_ip4_config (NMPPPManager *ppp_manager,
Packit Service b23acc
                NMIP4Config *config,
Packit Service b23acc
                gpointer user_data)
Packit Service b23acc
{
Packit Service b23acc
	NMDevice *device = NM_DEVICE (user_data);
Packit Service b23acc
Packit Service b23acc
	/* Ignore PPP IP4 events that come in after initial configuration */
Packit Service b23acc
	if (nm_device_activate_ip4_state_in_conf (device))
Packit Service b23acc
		nm_device_activate_schedule_ip_config_result (device, AF_INET, NM_IP_CONFIG_CAST (config));
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static NMActStageReturn
Packit Service b23acc
pppoe_stage3_ip4_config_start (NMDeviceEthernet *self, NMDeviceStateReason *out_failure_reason)
Packit Service b23acc
{
Packit Service b23acc
	NMDevice *device = NM_DEVICE (self);
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
	NMSettingPppoe *s_pppoe;
Packit Service b23acc
	NMActRequest *req;
Packit Service b23acc
	GError *err = NULL;
Packit Service b23acc
Packit Service b23acc
	req = nm_device_get_act_request (device);
Packit Service b23acc
Packit Service b23acc
	g_return_val_if_fail (req, NM_ACT_STAGE_RETURN_FAILURE);
Packit Service b23acc
Packit Service b23acc
	s_pppoe = nm_device_get_applied_setting (device, NM_TYPE_SETTING_PPPOE);
Packit Service b23acc
Packit Service b23acc
	g_return_val_if_fail (s_pppoe, NM_ACT_STAGE_RETURN_FAILURE);
Packit Service b23acc
Packit Service b23acc
	priv->ppp_manager = nm_ppp_manager_create (nm_device_get_iface (device),
Packit Service b23acc
	                                           &err;;
Packit Service b23acc
Packit Service b23acc
	if (priv->ppp_manager) {
Packit Service b23acc
		nm_ppp_manager_set_route_parameters (priv->ppp_manager,
Packit Service b23acc
		                                     nm_device_get_route_table (device, AF_INET),
Packit Service b23acc
		                                     nm_device_get_route_metric (device, AF_INET),
Packit Service b23acc
		                                     nm_device_get_route_table (device, AF_INET6),
Packit Service b23acc
		                                     nm_device_get_route_metric (device, AF_INET6));
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	if (   !priv->ppp_manager
Packit Service b23acc
	    || !nm_ppp_manager_start (priv->ppp_manager, req,
Packit Service b23acc
	                              nm_setting_pppoe_get_username (s_pppoe),
Packit Service b23acc
	                              30, 0, &err)) {
Packit Service b23acc
		_LOGW (LOGD_DEVICE, "PPPoE failed to start: %s", err->message);
Packit Service b23acc
		g_error_free (err);
Packit Service b23acc
Packit Service b23acc
		g_clear_object (&priv->ppp_manager);
Packit Service b23acc
Packit Service b23acc
		NM_SET_OUT (out_failure_reason, NM_DEVICE_STATE_REASON_PPP_START_FAILED);
Packit Service b23acc
		return NM_ACT_STAGE_RETURN_FAILURE;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	g_signal_connect (priv->ppp_manager, NM_PPP_MANAGER_SIGNAL_STATE_CHANGED,
Packit Service b23acc
	                  G_CALLBACK (ppp_state_changed),
Packit Service b23acc
	                  self);
Packit Service b23acc
	g_signal_connect (priv->ppp_manager, NM_PPP_MANAGER_SIGNAL_IFINDEX_SET,
Packit Service b23acc
	                  G_CALLBACK (ppp_ifindex_set),
Packit Service b23acc
	                  self);
Packit Service b23acc
	g_signal_connect (priv->ppp_manager, NM_PPP_MANAGER_SIGNAL_IP4_CONFIG,
Packit Service b23acc
	                  G_CALLBACK (ppp_ip4_config),
Packit Service b23acc
	                  self);
Packit Service b23acc
	return NM_ACT_STAGE_RETURN_POSTPONE;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
static void dcb_state (NMDevice *device, gboolean timeout);
Packit Service b23acc
Packit Service b23acc
static gboolean
Packit Service b23acc
dcb_carrier_timeout (gpointer user_data)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernet *self = NM_DEVICE_ETHERNET (user_data);
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
	NMDevice *device = NM_DEVICE (user_data);
Packit Service b23acc
Packit Service b23acc
	g_return_val_if_fail (nm_device_get_state (device) == NM_DEVICE_STATE_CONFIG, G_SOURCE_REMOVE);
Packit Service b23acc
Packit Service b23acc
	priv->dcb_timeout_id = 0;
Packit Service b23acc
	if (priv->dcb_wait != DCB_WAIT_CARRIER_POSTCONFIG_DOWN) {
Packit Service b23acc
		_LOGW (LOGD_DCB, "DCB: timed out waiting for carrier (step %d)",
Packit Service b23acc
		       priv->dcb_wait);
Packit Service b23acc
	}
Packit Service b23acc
	dcb_state (device, TRUE);
Packit Service b23acc
	return G_SOURCE_REMOVE;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static gboolean
Packit Service b23acc
dcb_configure (NMDevice *device)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernet *self = (NMDeviceEthernet *) device;
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
	NMSettingDcb *s_dcb;
Packit Service b23acc
	GError *error = NULL;
Packit Service b23acc
Packit Service b23acc
	nm_clear_g_source (&priv->dcb_timeout_id);
Packit Service b23acc
Packit Service b23acc
	s_dcb = nm_device_get_applied_setting (device, NM_TYPE_SETTING_DCB);
Packit Service b23acc
Packit Service b23acc
	g_return_val_if_fail (s_dcb, FALSE);
Packit Service b23acc
Packit Service b23acc
	if (!nm_dcb_setup (nm_device_get_iface (device), s_dcb, &error)) {
Packit Service b23acc
		_LOGW (LOGD_DCB, "Activation: (ethernet) failed to enable DCB/FCoE: %s",
Packit Service b23acc
		       error->message);
Packit Service b23acc
		g_clear_error (&error);
Packit Service b23acc
		return FALSE;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	/* Pause again just in case the device takes the carrier down when
Packit Service b23acc
	 * setting specific DCB attributes.
Packit Service b23acc
	 */
Packit Service b23acc
	_LOGD (LOGD_DCB, "waiting for carrier (postconfig down)");
Packit Service b23acc
	priv->dcb_wait = DCB_WAIT_CARRIER_POSTCONFIG_DOWN;
Packit Service b23acc
	priv->dcb_timeout_id = g_timeout_add_seconds (3, dcb_carrier_timeout, device);
Packit Service b23acc
	return TRUE;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static gboolean
Packit Service b23acc
dcb_enable (NMDevice *device)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernet *self = NM_DEVICE_ETHERNET (device);
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
	GError *error = NULL;
Packit Service b23acc
Packit Service b23acc
	nm_clear_g_source (&priv->dcb_timeout_id);
Packit Service b23acc
	if (!nm_dcb_enable (nm_device_get_iface (device), TRUE, &error)) {
Packit Service b23acc
		_LOGW (LOGD_DCB, "Activation: (ethernet) failed to enable DCB/FCoE: %s",
Packit Service b23acc
		       error->message);
Packit Service b23acc
		g_clear_error (&error);
Packit Service b23acc
		return FALSE;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	/* Pause for 3 seconds after enabling DCB to let the card reconfigure
Packit Service b23acc
	 * itself.  Drivers will often re-initialize internal settings which
Packit Service b23acc
	 * takes the carrier down for 2 or more seconds.  During this time,
Packit Service b23acc
	 * lldpad will refuse to do anything else with the card since the carrier
Packit Service b23acc
	 * is down.  But NM might get the carrier-down signal long after calling
Packit Service b23acc
	 * "dcbtool dcb on", so we have to first wait for the carrier to go down.
Packit Service b23acc
	 */
Packit Service b23acc
	_LOGD (LOGD_DCB, "waiting for carrier (preconfig down)");
Packit Service b23acc
	priv->dcb_wait = DCB_WAIT_CARRIER_PRECONFIG_DOWN;
Packit Service b23acc
	priv->dcb_timeout_id = g_timeout_add_seconds (3, dcb_carrier_timeout, device);
Packit Service b23acc
	return TRUE;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
dcb_state (NMDevice *device, gboolean timeout)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernet *self = NM_DEVICE_ETHERNET (device);
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
	gboolean carrier;
Packit Service b23acc
Packit Service b23acc
	g_return_if_fail (nm_device_get_state (device) == NM_DEVICE_STATE_CONFIG);
Packit Service b23acc
Packit Service b23acc
	carrier = nm_platform_link_is_connected (nm_device_get_platform (device), nm_device_get_ifindex (device));
Packit Service b23acc
	_LOGD (LOGD_DCB, "dcb_state() wait %d carrier %d timeout %d", priv->dcb_wait, carrier, timeout);
Packit Service b23acc
Packit Service b23acc
	switch (priv->dcb_wait) {
Packit Service b23acc
	case DCB_WAIT_CARRIER_PREENABLE_UP:
Packit Service b23acc
		if (timeout || carrier) {
Packit Service b23acc
			_LOGD (LOGD_DCB, "dcb_state() enabling DCB");
Packit Service b23acc
			nm_clear_g_source (&priv->dcb_timeout_id);
Packit Service b23acc
			if (!dcb_enable (device)) {
Packit Service b23acc
				priv->dcb_handle_carrier_changes = FALSE;
Packit Service b23acc
				nm_device_state_changed (device,
Packit Service b23acc
				                         NM_DEVICE_STATE_FAILED,
Packit Service b23acc
				                         NM_DEVICE_STATE_REASON_DCB_FCOE_FAILED);
Packit Service b23acc
			}
Packit Service b23acc
		}
Packit Service b23acc
		break;
Packit Service b23acc
	case DCB_WAIT_CARRIER_PRECONFIG_DOWN:
Packit Service b23acc
		nm_clear_g_source (&priv->dcb_timeout_id);
Packit Service b23acc
		priv->dcb_wait = DCB_WAIT_CARRIER_PRECONFIG_UP;
Packit Service b23acc
Packit Service b23acc
		if (!carrier) {
Packit Service b23acc
			/* Wait for the carrier to come back up */
Packit Service b23acc
			_LOGD (LOGD_DCB, "waiting for carrier (preconfig up)");
Packit Service b23acc
			priv->dcb_timeout_id = g_timeout_add_seconds (5, dcb_carrier_timeout, device);
Packit Service b23acc
			break;
Packit Service b23acc
		}
Packit Service b23acc
		_LOGD (LOGD_DCB, "dcb_state() preconfig down falling through");
Packit Service b23acc
		/* fall-through */
Packit Service b23acc
	case DCB_WAIT_CARRIER_PRECONFIG_UP:
Packit Service b23acc
		if (timeout || carrier) {
Packit Service b23acc
			_LOGD (LOGD_DCB, "dcb_state() preconfig up configuring DCB");
Packit Service b23acc
			nm_clear_g_source (&priv->dcb_timeout_id);
Packit Service b23acc
			if (!dcb_configure (device)) {
Packit Service b23acc
				priv->dcb_handle_carrier_changes = FALSE;
Packit Service b23acc
				nm_device_state_changed (device,
Packit Service b23acc
				                         NM_DEVICE_STATE_FAILED,
Packit Service b23acc
				                         NM_DEVICE_STATE_REASON_DCB_FCOE_FAILED);
Packit Service b23acc
			}
Packit Service b23acc
		}
Packit Service b23acc
		break;
Packit Service b23acc
	case DCB_WAIT_CARRIER_POSTCONFIG_DOWN:
Packit Service b23acc
		nm_clear_g_source (&priv->dcb_timeout_id);
Packit Service b23acc
		priv->dcb_wait = DCB_WAIT_CARRIER_POSTCONFIG_UP;
Packit Service b23acc
Packit Service b23acc
		if (!carrier) {
Packit Service b23acc
			/* Wait for the carrier to come back up */
Packit Service b23acc
			_LOGD (LOGD_DCB, "waiting for carrier (postconfig up)");
Packit Service b23acc
			priv->dcb_timeout_id = g_timeout_add_seconds (5, dcb_carrier_timeout, device);
Packit Service b23acc
			break;
Packit Service b23acc
		}
Packit Service b23acc
		_LOGD (LOGD_DCB, "dcb_state() postconfig down falling through");
Packit Service b23acc
		/* fall-through */
Packit Service b23acc
	case DCB_WAIT_CARRIER_POSTCONFIG_UP:
Packit Service b23acc
		if (timeout || carrier) {
Packit Service b23acc
			_LOGD (LOGD_DCB, "dcb_state() postconfig up starting IP");
Packit Service b23acc
			nm_clear_g_source (&priv->dcb_timeout_id);
Packit Service b23acc
			priv->dcb_handle_carrier_changes = FALSE;
Packit Service b23acc
			priv->dcb_wait = DCB_WAIT_UNKNOWN;
Packit Service b23acc
			nm_device_activate_schedule_stage3_ip_config_start (device);
Packit Service b23acc
		}
Packit Service b23acc
		break;
Packit Service b23acc
	default:
Packit Service b23acc
		g_assert_not_reached ();
Packit Service b23acc
	}
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
static gboolean
Packit Service b23acc
wake_on_lan_enable (NMDevice *device)
Packit Service b23acc
{
Packit Service b23acc
	NMSettingWiredWakeOnLan wol;
Packit Service b23acc
	NMSettingWired *s_wired;
Packit Service b23acc
	const char *password = NULL;
Packit Service b23acc
Packit Service b23acc
	s_wired = nm_device_get_applied_setting (device, NM_TYPE_SETTING_WIRED);
Packit Service b23acc
Packit Service b23acc
	if (s_wired) {
Packit Service b23acc
		wol = nm_setting_wired_get_wake_on_lan (s_wired);
Packit Service b23acc
		password = nm_setting_wired_get_wake_on_lan_password (s_wired);
Packit Service b23acc
		if (wol != NM_SETTING_WIRED_WAKE_ON_LAN_DEFAULT)
Packit Service b23acc
			goto found;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	wol = nm_config_data_get_connection_default_int64 (NM_CONFIG_GET_DATA,
Packit Service b23acc
	                                                   NM_CON_DEFAULT ("ethernet.wake-on-lan"),
Packit Service b23acc
	                                                   device,
Packit Service b23acc
	                                                   NM_SETTING_WIRED_WAKE_ON_LAN_NONE,
Packit Service b23acc
	                                                   G_MAXINT32,
Packit Service b23acc
	                                                   NM_SETTING_WIRED_WAKE_ON_LAN_DEFAULT);
Packit Service b23acc
Packit Service b23acc
	if (   NM_FLAGS_ANY (wol, NM_SETTING_WIRED_WAKE_ON_LAN_EXCLUSIVE_FLAGS)
Packit Service b23acc
	    && !nm_utils_is_power_of_two (wol)) {
Packit Service b23acc
		nm_log_dbg (LOGD_ETHER, "invalid default value %u for wake-on-lan", (guint) wol);
Packit Service b23acc
		wol = NM_SETTING_WIRED_WAKE_ON_LAN_DEFAULT;
Packit Service b23acc
	}
Packit Service b23acc
	if (wol != NM_SETTING_WIRED_WAKE_ON_LAN_DEFAULT)
Packit Service b23acc
		goto found;
Packit Service b23acc
	wol = NM_SETTING_WIRED_WAKE_ON_LAN_IGNORE;
Packit Service b23acc
found:
Packit Service b23acc
	return nm_platform_ethtool_set_wake_on_lan (nm_device_get_platform (device),
Packit Service b23acc
	                                            nm_device_get_ifindex (device),
Packit Service b23acc
	                                            wol, password);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
static NMActStageReturn
Packit Service b23acc
act_stage2_config (NMDevice *device, NMDeviceStateReason *out_failure_reason)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernet *self = (NMDeviceEthernet *) device;
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
	NMSettingConnection *s_con;
Packit Service b23acc
	const char *connection_type;
Packit Service b23acc
	gboolean do_postpone = FALSE;
Packit Service b23acc
	NMSettingDcb *s_dcb;
Packit Service b23acc
Packit Service b23acc
	s_con = nm_device_get_applied_setting (device, NM_TYPE_SETTING_CONNECTION);
Packit Service b23acc
Packit Service b23acc
	g_return_val_if_fail (s_con, NM_ACT_STAGE_RETURN_FAILURE);
Packit Service b23acc
Packit Service b23acc
	nm_clear_g_source (&priv->dcb_timeout_id);
Packit Service b23acc
	priv->dcb_handle_carrier_changes = FALSE;
Packit Service b23acc
Packit Service b23acc
	/* 802.1x has to run before any IP configuration since the 802.1x auth
Packit Service b23acc
	 * process opens the port up for normal traffic.
Packit Service b23acc
	 */
Packit Service b23acc
	connection_type = nm_setting_connection_get_connection_type (s_con);
Packit Service b23acc
	if (nm_streq (connection_type, NM_SETTING_WIRED_SETTING_NAME)) {
Packit Service b23acc
		NMSetting8021x *security;
Packit Service b23acc
Packit Service b23acc
		security = nm_device_get_applied_setting (device, NM_TYPE_SETTING_802_1X);
Packit Service b23acc
Packit Service b23acc
		if (security) {
Packit Service b23acc
			/* FIXME: for now 802.1x is mutually exclusive with DCB */
Packit Service b23acc
			if (!nm_device_has_carrier (NM_DEVICE (self))) {
Packit Service b23acc
				_LOGD (LOGD_DEVICE | LOGD_ETHER, "delay supplicant initialization until carrier goes up");
Packit Service b23acc
				priv->carrier_id = g_signal_connect (self,
Packit Service b23acc
				                                     "notify::" NM_DEVICE_CARRIER,
Packit Service b23acc
				                                     G_CALLBACK (carrier_changed),
Packit Service b23acc
				                                     self);
Packit Service b23acc
				return NM_ACT_STAGE_RETURN_POSTPONE;
Packit Service b23acc
			}
Packit Service b23acc
Packit Service b23acc
			return supplicant_check_secrets_needed (self, out_failure_reason);
Packit Service b23acc
		}
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	wake_on_lan_enable (device);
Packit Service b23acc
Packit Service b23acc
	/* DCB and FCoE setup */
Packit Service b23acc
	s_dcb = nm_device_get_applied_setting (device, NM_TYPE_SETTING_DCB);
Packit Service b23acc
	if (s_dcb) {
Packit Service b23acc
		/* lldpad really really wants the carrier to be up */
Packit Service b23acc
		if (nm_platform_link_is_connected (nm_device_get_platform (device), nm_device_get_ifindex (device))) {
Packit Service b23acc
			if (!dcb_enable (device)) {
Packit Service b23acc
				NM_SET_OUT (out_failure_reason, NM_DEVICE_STATE_REASON_DCB_FCOE_FAILED);
Packit Service b23acc
				return NM_ACT_STAGE_RETURN_FAILURE;
Packit Service b23acc
			}
Packit Service b23acc
		} else {
Packit Service b23acc
			_LOGD (LOGD_DCB, "waiting for carrier (preenable up)");
Packit Service b23acc
			priv->dcb_wait = DCB_WAIT_CARRIER_PREENABLE_UP;
Packit Service b23acc
			priv->dcb_timeout_id = g_timeout_add_seconds (4, dcb_carrier_timeout, device);
Packit Service b23acc
		}
Packit Service b23acc
Packit Service b23acc
		priv->dcb_handle_carrier_changes = TRUE;
Packit Service b23acc
		do_postpone = TRUE;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	/* PPPoE setup */
Packit Service b23acc
	if (nm_connection_is_type (nm_device_get_applied_connection (device),
Packit Service b23acc
	                           NM_SETTING_PPPOE_SETTING_NAME)) {
Packit Service b23acc
		NMSettingPpp *s_ppp;
Packit Service b23acc
Packit Service b23acc
		s_ppp = nm_device_get_applied_setting (device, NM_TYPE_SETTING_PPP);
Packit Service b23acc
		if (s_ppp) {
Packit Service b23acc
			guint32 mtu;
Packit Service b23acc
			guint32 mru;
Packit Service b23acc
			guint32 mxu;
Packit Service b23acc
Packit Service b23acc
			mtu = nm_setting_ppp_get_mtu (s_ppp);
Packit Service b23acc
			mru = nm_setting_ppp_get_mru (s_ppp);
Packit Service b23acc
			mxu = MAX (mru, mtu);
Packit Service b23acc
			if (mxu) {
Packit Service b23acc
				_LOGD (LOGD_PPP, "set MTU to %u (PPP interface MRU %u, MTU %u)",
Packit Service b23acc
				       mxu + PPPOE_ENCAP_OVERHEAD, mru, mtu);
Packit Service b23acc
				nm_platform_link_set_mtu (nm_device_get_platform (device),
Packit Service b23acc
				                          nm_device_get_ifindex (device),
Packit Service b23acc
				                          mxu + PPPOE_ENCAP_OVERHEAD);
Packit Service b23acc
			}
Packit Service b23acc
		}
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	return   do_postpone
Packit Service b23acc
	       ? NM_ACT_STAGE_RETURN_POSTPONE
Packit Service b23acc
	       : NM_ACT_STAGE_RETURN_SUCCESS;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static NMActStageReturn
Packit Service b23acc
act_stage3_ip_config_start (NMDevice *device,
Packit Service b23acc
                            int addr_family,
Packit Service b23acc
                            gpointer *out_config,
Packit Service b23acc
                            NMDeviceStateReason *out_failure_reason)
Packit Service b23acc
{
Packit Service b23acc
	NMSettingConnection *s_con;
Packit Service b23acc
	const char *connection_type;
Packit Service b23acc
Packit Service b23acc
	if (addr_family == AF_INET) {
Packit Service b23acc
		s_con = nm_device_get_applied_setting (device, NM_TYPE_SETTING_CONNECTION);
Packit Service b23acc
Packit Service b23acc
		g_return_val_if_fail (s_con, NM_ACT_STAGE_RETURN_FAILURE);
Packit Service b23acc
Packit Service b23acc
		connection_type = nm_setting_connection_get_connection_type (s_con);
Packit Service b23acc
		if (!strcmp (connection_type, NM_SETTING_PPPOE_SETTING_NAME))
Packit Service b23acc
			return pppoe_stage3_ip4_config_start (NM_DEVICE_ETHERNET (device), out_failure_reason);
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	return NM_DEVICE_CLASS (nm_device_ethernet_parent_class)->act_stage3_ip_config_start (device, addr_family, out_config, out_failure_reason);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static guint32
Packit Service b23acc
get_configured_mtu (NMDevice *device,
Packit Service b23acc
                    NMDeviceMtuSource *out_source,
Packit Service b23acc
                    gboolean *out_force)
Packit Service b23acc
{
Packit Service b23acc
	/* MTU only set for plain ethernet */
Packit Service b23acc
	if (NM_DEVICE_ETHERNET_GET_PRIVATE (device)->ppp_manager)
Packit Service b23acc
		return 0;
Packit Service b23acc
Packit Service b23acc
	return nm_device_get_configured_mtu_for_wired (device, out_source, out_force);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
deactivate (NMDevice *device)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernet *self = NM_DEVICE_ETHERNET (device);
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
	NMSettingDcb *s_dcb;
Packit Service b23acc
	GError *error = NULL;
Packit Service b23acc
Packit Service b23acc
	nm_clear_g_source (&priv->pppoe_wait_id);
Packit Service b23acc
	nm_clear_g_signal_handler (self, &priv->carrier_id);
Packit Service b23acc
Packit Service b23acc
	if (priv->ppp_manager) {
Packit Service b23acc
		nm_ppp_manager_stop (priv->ppp_manager, NULL, NULL, NULL);
Packit Service b23acc
		g_clear_object (&priv->ppp_manager);
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	supplicant_interface_release (self);
Packit Service b23acc
Packit Service b23acc
	priv->dcb_wait = DCB_WAIT_UNKNOWN;
Packit Service b23acc
	nm_clear_g_source (&priv->dcb_timeout_id);
Packit Service b23acc
	priv->dcb_handle_carrier_changes = FALSE;
Packit Service b23acc
Packit Service b23acc
	/* Tear down DCB/FCoE if it was enabled */
Packit Service b23acc
	s_dcb = nm_device_get_applied_setting (device, NM_TYPE_SETTING_DCB);
Packit Service b23acc
	if (s_dcb) {
Packit Service b23acc
		if (!nm_dcb_cleanup (nm_device_get_iface (device), &error)) {
Packit Service b23acc
			_LOGW (LOGD_DEVICE | LOGD_PLATFORM, "failed to disable DCB/FCoE: %s",
Packit Service b23acc
			       error->message);
Packit Service b23acc
			g_clear_error (&error);
Packit Service b23acc
		}
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	/* Set last PPPoE connection time */
Packit Service b23acc
	if (nm_device_get_applied_setting (device, NM_TYPE_SETTING_PPPOE))
Packit Service b23acc
		priv->last_pppoe_time = nm_utils_get_monotonic_timestamp_sec ();
Packit Service b23acc
Packit Service b23acc
	if (priv->ethtool_prev_set) {
Packit Service b23acc
		priv->ethtool_prev_set = FALSE;
Packit Service b23acc
Packit Service b23acc
		_LOGD (LOGD_DEVICE, "set-link: reset %snegotiation (%u Mbit, %s duplex)",
Packit Service b23acc
		       priv->ethtool_prev_autoneg ? "auto-" : "static ",
Packit Service b23acc
		       priv->ethtool_prev_speed,
Packit Service b23acc
		       nm_platform_link_duplex_type_to_string (priv->ethtool_prev_duplex));
Packit Service b23acc
		if (!nm_platform_ethtool_set_link_settings (nm_device_get_platform (device),
Packit Service b23acc
		                                            nm_device_get_ifindex (device),
Packit Service b23acc
		                                            priv->ethtool_prev_autoneg,
Packit Service b23acc
		                                            priv->ethtool_prev_speed,
Packit Service b23acc
		                                            priv->ethtool_prev_duplex)) {
Packit Service b23acc
			_LOGW (LOGD_DEVICE, "set-link: failure to reset link negotiation");
Packit Service b23acc
			return;
Packit Service b23acc
		}
Packit Service b23acc
	}
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static gboolean
Packit Service b23acc
complete_connection (NMDevice *device,
Packit Service b23acc
                     NMConnection *connection,
Packit Service b23acc
                     const char *specific_object,
Packit Service b23acc
                     NMConnection *const*existing_connections,
Packit Service b23acc
                     GError **error)
Packit Service b23acc
{
Packit Service b23acc
	NMSettingWired *s_wired;
Packit Service b23acc
	NMSettingPppoe *s_pppoe;
Packit Service b23acc
Packit Service b23acc
	s_pppoe = nm_connection_get_setting_pppoe (connection);
Packit Service b23acc
Packit Service b23acc
	/* We can't telepathically figure out the service name or username, so if
Packit Service b23acc
	 * those weren't given, we can't complete the connection.
Packit Service b23acc
	 */
Packit Service b23acc
	if (s_pppoe && !nm_setting_verify (NM_SETTING (s_pppoe), NULL, error))
Packit Service b23acc
		return FALSE;
Packit Service b23acc
Packit Service b23acc
	s_wired = nm_connection_get_setting_wired (connection);
Packit Service b23acc
	if (!s_wired) {
Packit Service b23acc
		s_wired = (NMSettingWired *) nm_setting_wired_new ();
Packit Service b23acc
		nm_connection_add_setting (connection, NM_SETTING (s_wired));
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	/* Default to an ethernet-only connection, but if a PPPoE setting was given
Packit Service b23acc
	 * then PPPoE should be our connection type.
Packit Service b23acc
	 */
Packit Service b23acc
	nm_utils_complete_generic (nm_device_get_platform (device),
Packit Service b23acc
	                           connection,
Packit Service b23acc
	                           s_pppoe ? NM_SETTING_PPPOE_SETTING_NAME : NM_SETTING_WIRED_SETTING_NAME,
Packit Service b23acc
	                           existing_connections,
Packit Service b23acc
	                           NULL,
Packit Service b23acc
	                           s_pppoe ? _("PPPoE connection") : _("Wired connection"),
Packit Service b23acc
	                           NULL,
Packit Service b23acc
	                           nm_setting_wired_get_mac_address (s_wired) ? NULL : nm_device_get_iface (device),
Packit Service b23acc
	                           s_pppoe ? FALSE : TRUE); /* No IPv6 by default yet for PPPoE */
Packit Service b23acc
Packit Service b23acc
	return TRUE;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static NMConnection *
Packit Service b23acc
new_default_connection (NMDevice *self)
Packit Service b23acc
{
Packit Service b23acc
	NMConnection *connection;
Packit Service b23acc
	NMSettingsConnection *const*connections;
Packit Service b23acc
	NMSetting *setting;
Packit Service b23acc
	gs_unref_hashtable GHashTable *existing_ids = NULL;
Packit Service b23acc
	struct udev_device *dev;
Packit Service b23acc
	const char *perm_hw_addr;
Packit Service b23acc
	const char *iface;
Packit Service b23acc
	const char *uprop = "0";
Packit Service b23acc
	gs_free char *defname = NULL;
Packit Service b23acc
	gs_free char *uuid = NULL;
Packit Service b23acc
	guint i, n_connections;
Packit Service b23acc
Packit Service b23acc
	perm_hw_addr = nm_device_get_permanent_hw_address (self);
Packit Service b23acc
	iface = nm_device_get_iface (self);
Packit Service b23acc
Packit Service b23acc
	connection = nm_simple_connection_new ();
Packit Service b23acc
	setting = nm_setting_connection_new ();
Packit Service b23acc
	nm_connection_add_setting (connection, setting);
Packit Service b23acc
Packit Service b23acc
	connections = nm_settings_get_connections (nm_device_get_settings (self), &n_connections);
Packit Service b23acc
	if (n_connections > 0) {
Packit Service b23acc
		existing_ids = g_hash_table_new (nm_str_hash, g_str_equal);
Packit Service b23acc
		for (i = 0; i < n_connections; i++)
Packit Service b23acc
			g_hash_table_add (existing_ids, (char *) nm_settings_connection_get_id (connections[i]));
Packit Service b23acc
	}
Packit Service b23acc
	defname = nm_device_ethernet_utils_get_default_wired_name (existing_ids);
Packit Service b23acc
	if (!defname)
Packit Service b23acc
		return NULL;
Packit Service b23acc
Packit Service b23acc
	/* Create a stable UUID. The UUID is also the Network_ID for stable-privacy addr-gen-mode,
Packit Service b23acc
	 * thus when it changes we will also generate different IPv6 addresses. */
Packit Service b23acc
	uuid = _nm_utils_uuid_generate_from_strings ("default-wired",
Packit Service b23acc
	                                             nm_utils_machine_id_str (),
Packit Service b23acc
	                                             defname,
Packit Service b23acc
	                                             perm_hw_addr ?: iface,
Packit Service b23acc
	                                             NULL);
Packit Service b23acc
Packit Service b23acc
	g_object_set (setting,
Packit Service b23acc
	              NM_SETTING_CONNECTION_ID, defname,
Packit Service b23acc
	              NM_SETTING_CONNECTION_TYPE, NM_SETTING_WIRED_SETTING_NAME,
Packit Service b23acc
	              NM_SETTING_CONNECTION_AUTOCONNECT, TRUE,
Packit Service b23acc
	              NM_SETTING_CONNECTION_AUTOCONNECT_PRIORITY, NM_SETTING_CONNECTION_AUTOCONNECT_PRIORITY_MIN,
Packit Service b23acc
	              NM_SETTING_CONNECTION_UUID, uuid,
Packit Service b23acc
	              NM_SETTING_CONNECTION_TIMESTAMP, (guint64) time (NULL),
Packit Service b23acc
	              NM_SETTING_CONNECTION_INTERFACE_NAME, iface,
Packit Service b23acc
	              NULL);
Packit Service b23acc
Packit Service b23acc
	/* Check if we should create a Link-Local only connection */
Packit Service b23acc
	dev = nm_platform_link_get_udev_device (nm_device_get_platform (NM_DEVICE (self)), nm_device_get_ip_ifindex (self));
Packit Service b23acc
	if (dev)
Packit Service b23acc
		uprop = udev_device_get_property_value (dev, "NM_AUTO_DEFAULT_LINK_LOCAL_ONLY");
Packit Service b23acc
Packit Service b23acc
	if (nm_udev_utils_property_as_boolean (uprop)) {
Packit Service b23acc
		setting = nm_setting_ip4_config_new ();
Packit Service b23acc
		g_object_set (setting,
Packit Service b23acc
		              NM_SETTING_IP_CONFIG_METHOD, NM_SETTING_IP4_CONFIG_METHOD_LINK_LOCAL,
Packit Service b23acc
		              NULL);
Packit Service b23acc
		nm_connection_add_setting (connection, setting);
Packit Service b23acc
Packit Service b23acc
		setting = nm_setting_ip6_config_new ();
Packit Service b23acc
		g_object_set (setting,
Packit Service b23acc
		              NM_SETTING_IP_CONFIG_METHOD,  NM_SETTING_IP6_CONFIG_METHOD_LINK_LOCAL,
Packit Service b23acc
		              NM_SETTING_IP_CONFIG_MAY_FAIL, TRUE,
Packit Service b23acc
		              NULL);
Packit Service b23acc
		nm_connection_add_setting (connection, setting);
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	return connection;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static const char *
Packit Service b23acc
get_s390_subchannels (NMDevice *device)
Packit Service b23acc
{
Packit Service b23acc
	nm_assert (NM_IS_DEVICE_ETHERNET (device));
Packit Service b23acc
Packit Service b23acc
	return NM_DEVICE_ETHERNET_GET_PRIVATE (device)->subchannels;
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
update_connection (NMDevice *device, NMConnection *connection)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (device);
Packit Service b23acc
	NMSettingWired *s_wired = nm_connection_get_setting_wired (connection);
Packit Service b23acc
	gboolean perm_hw_addr_is_fake;
Packit Service b23acc
	const char *perm_hw_addr;
Packit Service b23acc
	const char *mac = nm_device_get_hw_address (device);
Packit Service b23acc
	const char *mac_prop = NM_SETTING_WIRED_MAC_ADDRESS;
Packit Service b23acc
	GHashTableIter iter;
Packit Service b23acc
	gpointer key, value;
Packit Service b23acc
Packit Service b23acc
	if (!s_wired) {
Packit Service b23acc
		s_wired = (NMSettingWired *) nm_setting_wired_new ();
Packit Service b23acc
		nm_connection_add_setting (connection, (NMSetting *) s_wired);
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	g_object_set (nm_connection_get_setting_connection (connection),
Packit Service b23acc
	              NM_SETTING_CONNECTION_TYPE, nm_connection_get_setting_pppoe (connection)
Packit Service b23acc
	                                          ? NM_SETTING_PPPOE_SETTING_NAME
Packit Service b23acc
	                                          : NM_SETTING_WIRED_SETTING_NAME, NULL);
Packit Service b23acc
Packit Service b23acc
	/* If the device reports a permanent address, use that for the MAC address
Packit Service b23acc
	 * and the current MAC, if different, is the cloned MAC.
Packit Service b23acc
	 */
Packit Service b23acc
	perm_hw_addr = nm_device_get_permanent_hw_address_full (device, TRUE, &perm_hw_addr_is_fake);
Packit Service b23acc
	if (perm_hw_addr && !perm_hw_addr_is_fake) {
Packit Service b23acc
		g_object_set (s_wired, NM_SETTING_WIRED_MAC_ADDRESS, perm_hw_addr, NULL);
Packit Service b23acc
Packit Service b23acc
		mac_prop = NULL;
Packit Service b23acc
		if (mac && !nm_utils_hwaddr_matches (perm_hw_addr, -1, mac, -1))
Packit Service b23acc
			mac_prop = NM_SETTING_WIRED_CLONED_MAC_ADDRESS;
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	if (mac_prop && mac && nm_utils_hwaddr_valid (mac, ETH_ALEN))
Packit Service b23acc
		g_object_set (s_wired, mac_prop, mac, NULL);
Packit Service b23acc
Packit Service b23acc
	/* We don't set the MTU as we don't know whether it was set explicitly */
Packit Service b23acc
Packit Service b23acc
	/* s390 */
Packit Service b23acc
	if (priv->subchannels_dbus)
Packit Service b23acc
		g_object_set (s_wired, NM_SETTING_WIRED_S390_SUBCHANNELS, priv->subchannels_dbus, NULL);
Packit Service b23acc
	if (priv->s390_nettype)
Packit Service b23acc
		g_object_set (s_wired, NM_SETTING_WIRED_S390_NETTYPE, priv->s390_nettype, NULL);
Packit Service b23acc
Packit Service b23acc
	_nm_setting_wired_clear_s390_options (s_wired);
Packit Service b23acc
	g_hash_table_iter_init (&iter, priv->s390_options);
Packit Service b23acc
	while (g_hash_table_iter_next (&iter, &key, &value))
Packit Service b23acc
		nm_setting_wired_add_s390_option (s_wired, (const char *) key, (const char *) value);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
link_speed_update (NMDevice *device)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernet *self = NM_DEVICE_ETHERNET (device);
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
	guint32 speed;
Packit Service b23acc
Packit Service b23acc
	if (!nm_platform_ethtool_get_link_settings (nm_device_get_platform (device), nm_device_get_ifindex (device), NULL, &speed, NULL))
Packit Service b23acc
		return;
Packit Service b23acc
	if (priv->speed == speed)
Packit Service b23acc
		return;
Packit Service b23acc
Packit Service b23acc
	priv->speed = speed;
Packit Service b23acc
	_LOGD (LOGD_PLATFORM | LOGD_ETHER, "speed is now %d Mb/s", speed);
Packit Service b23acc
	_notify (self, PROP_SPEED);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
carrier_changed_notify (NMDevice *device, gboolean carrier)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernet *self = NM_DEVICE_ETHERNET (device);
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
Packit Service b23acc
	if (priv->dcb_handle_carrier_changes) {
Packit Service b23acc
		nm_assert (nm_device_get_state (device) == NM_DEVICE_STATE_CONFIG);
Packit Service b23acc
Packit Service b23acc
		if (priv->dcb_timeout_id) {
Packit Service b23acc
			_LOGD (LOGD_DCB, "carrier_changed() calling dcb_state()");
Packit Service b23acc
			dcb_state (device, FALSE);
Packit Service b23acc
		}
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	if (carrier)
Packit Service b23acc
		link_speed_update (device);
Packit Service b23acc
Packit Service b23acc
	NM_DEVICE_CLASS (nm_device_ethernet_parent_class)->carrier_changed_notify (device, carrier);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
link_changed (NMDevice *device,
Packit Service b23acc
              const NMPlatformLink *pllink)
Packit Service b23acc
{
Packit Service b23acc
	NM_DEVICE_CLASS (nm_device_ethernet_parent_class)->link_changed (device, pllink);
Packit Service b23acc
	if (pllink->initialized)
Packit Service b23acc
		_update_s390_subchannels ((NMDeviceEthernet *) device);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static gboolean
Packit Service b23acc
is_available (NMDevice *device, NMDeviceCheckDevAvailableFlags flags)
Packit Service b23acc
{
Packit Service b23acc
	if (!NM_DEVICE_CLASS (nm_device_ethernet_parent_class)->is_available (device, flags))
Packit Service b23acc
		return FALSE;
Packit Service b23acc
Packit Service b23acc
	return !!nm_device_get_initial_hw_address (device);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static gboolean
Packit Service b23acc
can_reapply_change (NMDevice *device,
Packit Service b23acc
                    const char *setting_name,
Packit Service b23acc
                    NMSetting *s_old,
Packit Service b23acc
                    NMSetting *s_new,
Packit Service b23acc
                    GHashTable *diffs,
Packit Service b23acc
                    GError **error)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceClass *device_class;
Packit Service b23acc
Packit Service b23acc
	/* Only handle wired setting here, delegate other settings to parent class */
Packit Service b23acc
	if (nm_streq (setting_name, NM_SETTING_WIRED_SETTING_NAME)) {
Packit Service b23acc
		return nm_device_hash_check_invalid_keys (diffs,
Packit Service b23acc
		                                          NM_SETTING_WIRED_SETTING_NAME,
Packit Service b23acc
		                                          error,
Packit Service b23acc
		                                          NM_SETTING_WIRED_MTU, /* reapplied with IP config */
Packit Service b23acc
		                                          NM_SETTING_WIRED_SPEED,
Packit Service b23acc
		                                          NM_SETTING_WIRED_DUPLEX,
Packit Service b23acc
		                                          NM_SETTING_WIRED_AUTO_NEGOTIATE,
Packit Service b23acc
		                                          NM_SETTING_WIRED_WAKE_ON_LAN,
Packit Service b23acc
		                                          NM_SETTING_WIRED_WAKE_ON_LAN_PASSWORD);
Packit Service b23acc
	}
Packit Service b23acc
Packit Service b23acc
	device_class = NM_DEVICE_CLASS (nm_device_ethernet_parent_class);
Packit Service b23acc
	return device_class->can_reapply_change (device,
Packit Service b23acc
	                                         setting_name,
Packit Service b23acc
	                                         s_old,
Packit Service b23acc
	                                         s_new,
Packit Service b23acc
	                                         diffs,
Packit Service b23acc
	                                         error);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
reapply_connection (NMDevice *device, NMConnection *con_old, NMConnection *con_new)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernet *self = NM_DEVICE_ETHERNET (device);
Packit Service b23acc
	NMDeviceState state = nm_device_get_state (device);
Packit Service b23acc
Packit Service b23acc
	NM_DEVICE_CLASS (nm_device_ethernet_parent_class)->reapply_connection (device,
Packit Service b23acc
	                                                                       con_old,
Packit Service b23acc
	                                                                       con_new);
Packit Service b23acc
Packit Service b23acc
	_LOGD (LOGD_DEVICE, "reapplying wired settings");
Packit Service b23acc
Packit Service b23acc
	if (state >= NM_DEVICE_STATE_PREPARE)
Packit Service b23acc
		link_negotiation_set (device);
Packit Service b23acc
	if (state >= NM_DEVICE_STATE_CONFIG)
Packit Service b23acc
		wake_on_lan_enable (device);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
dispose (GObject *object)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernet *self = NM_DEVICE_ETHERNET (object);
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
Packit Service b23acc
	wired_secrets_cancel (self);
Packit Service b23acc
Packit Service b23acc
	supplicant_interface_release (self);
Packit Service b23acc
Packit Service b23acc
	nm_clear_g_source (&priv->pppoe_wait_id);
Packit Service b23acc
Packit Service b23acc
	nm_clear_g_source (&priv->dcb_timeout_id);
Packit Service b23acc
Packit Service b23acc
	nm_clear_g_signal_handler (self, &priv->carrier_id);
Packit Service b23acc
Packit Service b23acc
	G_OBJECT_CLASS (nm_device_ethernet_parent_class)->dispose (object);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
finalize (GObject *object)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernet *self = NM_DEVICE_ETHERNET (object);
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
Packit Service b23acc
	g_clear_object (&priv->supplicant.mgr);
Packit Service b23acc
	g_free (priv->subchan1);
Packit Service b23acc
	g_free (priv->subchan2);
Packit Service b23acc
	g_free (priv->subchan3);
Packit Service b23acc
	g_free (priv->subchannels);
Packit Service b23acc
	g_strfreev (priv->subchannels_dbus);
Packit Service b23acc
	g_free (priv->s390_nettype);
Packit Service b23acc
	g_hash_table_destroy (priv->s390_options);
Packit Service b23acc
Packit Service b23acc
	G_OBJECT_CLASS (nm_device_ethernet_parent_class)->finalize (object);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
get_property (GObject *object, guint prop_id,
Packit Service b23acc
              GValue *value, GParamSpec *pspec)
Packit Service b23acc
{
Packit Service b23acc
	NMDeviceEthernet *self = NM_DEVICE_ETHERNET (object);
Packit Service b23acc
	NMDeviceEthernetPrivate *priv = NM_DEVICE_ETHERNET_GET_PRIVATE (self);
Packit Service b23acc
Packit Service b23acc
	switch (prop_id) {
Packit Service b23acc
	case PROP_SPEED:
Packit Service b23acc
		g_value_set_uint (value, priv->speed);
Packit Service b23acc
		break;
Packit Service b23acc
	case PROP_S390_SUBCHANNELS:
Packit Service b23acc
		g_value_set_boxed (value, priv->subchannels_dbus);
Packit Service b23acc
		break;
Packit Service b23acc
	default:
Packit Service b23acc
		G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
Packit Service b23acc
		break;
Packit Service b23acc
	}
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
set_property (GObject *object, guint prop_id,
Packit Service b23acc
              const GValue *value, GParamSpec *pspec)
Packit Service b23acc
{
Packit Service b23acc
	switch (prop_id) {
Packit Service b23acc
	default:
Packit Service b23acc
		G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
Packit Service b23acc
		break;
Packit Service b23acc
	}
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static const NMDBusInterfaceInfoExtended interface_info_device_wired = {
Packit Service b23acc
	.parent = NM_DEFINE_GDBUS_INTERFACE_INFO_INIT (
Packit Service b23acc
		NM_DBUS_INTERFACE_DEVICE_WIRED,
Packit Service b23acc
		.signals = NM_DEFINE_GDBUS_SIGNAL_INFOS (
Packit Service b23acc
			&nm_signal_info_property_changed_legacy,
Packit Service b23acc
		),
Packit Service b23acc
		.properties = NM_DEFINE_GDBUS_PROPERTY_INFOS (
Packit Service b23acc
			NM_DEFINE_DBUS_PROPERTY_INFO_EXTENDED_READABLE_L ("HwAddress",       "s",  NM_DEVICE_HW_ADDRESS),
Packit Service b23acc
			NM_DEFINE_DBUS_PROPERTY_INFO_EXTENDED_READABLE_L ("PermHwAddress",   "s",  NM_DEVICE_PERM_HW_ADDRESS),
Packit Service b23acc
			NM_DEFINE_DBUS_PROPERTY_INFO_EXTENDED_READABLE_L ("Speed",           "u",  NM_DEVICE_ETHERNET_SPEED),
Packit Service b23acc
			NM_DEFINE_DBUS_PROPERTY_INFO_EXTENDED_READABLE_L ("S390Subchannels", "as", NM_DEVICE_ETHERNET_S390_SUBCHANNELS),
Packit Service b23acc
			NM_DEFINE_DBUS_PROPERTY_INFO_EXTENDED_READABLE_L ("Carrier",         "b",  NM_DEVICE_CARRIER),
Packit Service b23acc
		),
Packit Service b23acc
	),
Packit Service b23acc
	.legacy_property_changed = TRUE,
Packit Service b23acc
};
Packit Service b23acc
Packit Service b23acc
static void
Packit Service b23acc
nm_device_ethernet_class_init (NMDeviceEthernetClass *klass)
Packit Service b23acc
{
Packit Service b23acc
	GObjectClass *object_class = G_OBJECT_CLASS (klass);
Packit Service b23acc
	NMDBusObjectClass *dbus_object_class = NM_DBUS_OBJECT_CLASS (klass);
Packit Service b23acc
	NMDeviceClass *device_class = NM_DEVICE_CLASS (klass);
Packit Service b23acc
Packit Service b23acc
	g_type_class_add_private (object_class, sizeof (NMDeviceEthernetPrivate));
Packit Service b23acc
Packit Service b23acc
	object_class->dispose = dispose;
Packit Service b23acc
	object_class->finalize = finalize;
Packit Service b23acc
	object_class->get_property = get_property;
Packit Service b23acc
	object_class->set_property = set_property;
Packit Service b23acc
Packit Service b23acc
	dbus_object_class->interface_infos = NM_DBUS_INTERFACE_INFOS (&interface_info_device_wired);
Packit Service b23acc
Packit Service b23acc
	device_class->connection_type_supported = NM_SETTING_WIRED_SETTING_NAME;
Packit Service b23acc
	device_class->link_types = NM_DEVICE_DEFINE_LINK_TYPES (NM_LINK_TYPE_ETHERNET);
Packit Service b23acc
Packit Service b23acc
	device_class->get_generic_capabilities = get_generic_capabilities;
Packit Service b23acc
	device_class->check_connection_compatible = check_connection_compatible;
Packit Service b23acc
	device_class->complete_connection = complete_connection;
Packit Service b23acc
	device_class->new_default_connection = new_default_connection;
Packit Service b23acc
Packit Service b23acc
	device_class->act_stage1_prepare_also_for_external_or_assume = TRUE;
Packit Service b23acc
	device_class->act_stage1_prepare = act_stage1_prepare;
Packit Service b23acc
	device_class->act_stage1_prepare_set_hwaddr_ethernet = TRUE;
Packit Service b23acc
	device_class->act_stage2_config = act_stage2_config;
Packit Service b23acc
	device_class->act_stage3_ip_config_start = act_stage3_ip_config_start;
Packit Service b23acc
	device_class->get_configured_mtu = get_configured_mtu;
Packit Service b23acc
	device_class->deactivate = deactivate;
Packit Service b23acc
	device_class->get_s390_subchannels = get_s390_subchannels;
Packit Service b23acc
	device_class->update_connection = update_connection;
Packit Service b23acc
	device_class->carrier_changed_notify = carrier_changed_notify;
Packit Service b23acc
	device_class->link_changed = link_changed;
Packit Service b23acc
	device_class->is_available = is_available;
Packit Service b23acc
	device_class->can_reapply_change = can_reapply_change;
Packit Service b23acc
	device_class->reapply_connection = reapply_connection;
Packit Service b23acc
Packit Service b23acc
	device_class->state_changed = device_state_changed;
Packit Service b23acc
Packit Service b23acc
	obj_properties[PROP_SPEED] =
Packit Service b23acc
	    g_param_spec_uint (NM_DEVICE_ETHERNET_SPEED, "", "",
Packit Service b23acc
	                       0, G_MAXUINT32, 0,
Packit Service b23acc
	                       G_PARAM_READABLE |
Packit Service b23acc
	                       G_PARAM_STATIC_STRINGS);
Packit Service b23acc
Packit Service b23acc
	obj_properties[PROP_S390_SUBCHANNELS] =
Packit Service b23acc
	    g_param_spec_boxed (NM_DEVICE_ETHERNET_S390_SUBCHANNELS, "", "",
Packit Service b23acc
	                        G_TYPE_STRV,
Packit Service b23acc
	                        G_PARAM_READABLE |
Packit Service b23acc
	                        G_PARAM_STATIC_STRINGS);
Packit Service b23acc
Packit Service b23acc
	g_object_class_install_properties (object_class, _PROPERTY_ENUMS_LAST, obj_properties);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
/*****************************************************************************/
Packit Service b23acc
Packit Service b23acc
#define NM_TYPE_ETHERNET_DEVICE_FACTORY (nm_ethernet_device_factory_get_type ())
Packit Service b23acc
#define NM_ETHERNET_DEVICE_FACTORY(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), NM_TYPE_ETHERNET_DEVICE_FACTORY, NMEthernetDeviceFactory))
Packit Service b23acc
Packit Service b23acc
static NMDevice *
Packit Service b23acc
create_device (NMDeviceFactory *factory,
Packit Service b23acc
               const char *iface,
Packit Service b23acc
               const NMPlatformLink *plink,
Packit Service b23acc
               NMConnection *connection,
Packit Service b23acc
               gboolean *out_ignore)
Packit Service b23acc
{
Packit Service b23acc
	return (NMDevice *) g_object_new (NM_TYPE_DEVICE_ETHERNET,
Packit Service b23acc
	                                  NM_DEVICE_IFACE, iface,
Packit Service b23acc
	                                  NM_DEVICE_TYPE_DESC, "Ethernet",
Packit Service b23acc
	                                  NM_DEVICE_DEVICE_TYPE, NM_DEVICE_TYPE_ETHERNET,
Packit Service b23acc
	                                  NM_DEVICE_LINK_TYPE, NM_LINK_TYPE_ETHERNET,
Packit Service b23acc
	                                  NULL);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
static gboolean
Packit Service b23acc
match_connection (NMDeviceFactory *factory, NMConnection *connection)
Packit Service b23acc
{
Packit Service b23acc
	const char *type = nm_connection_get_connection_type (connection);
Packit Service b23acc
	NMSettingPppoe *s_pppoe;
Packit Service b23acc
Packit Service b23acc
	if (nm_streq (type, NM_SETTING_WIRED_SETTING_NAME))
Packit Service b23acc
		return TRUE;
Packit Service b23acc
Packit Service b23acc
	nm_assert (nm_streq (type, NM_SETTING_PPPOE_SETTING_NAME));
Packit Service b23acc
	s_pppoe = nm_connection_get_setting_pppoe (connection);
Packit Service b23acc
Packit Service b23acc
	return !nm_setting_pppoe_get_parent (s_pppoe);
Packit Service b23acc
}
Packit Service b23acc
Packit Service b23acc
NM_DEVICE_FACTORY_DEFINE_INTERNAL (ETHERNET, Ethernet, ethernet,
Packit Service b23acc
	NM_DEVICE_FACTORY_DECLARE_LINK_TYPES    (NM_LINK_TYPE_ETHERNET)
Packit Service b23acc
	NM_DEVICE_FACTORY_DECLARE_SETTING_TYPES (NM_SETTING_WIRED_SETTING_NAME, NM_SETTING_PPPOE_SETTING_NAME),
Packit Service b23acc
	factory_class->create_device = create_device;
Packit Service b23acc
	factory_class->match_connection = match_connection;
Packit Service b23acc
);