mirror of
https://gitlab.freedesktop.org/NetworkManager/NetworkManager
synced 2024-10-06 16:21:50 +00:00
device: support creating generic devices via device-handler
If the device-handler of the generic connection is set, the connection is virtual and the device is created by invoking the device-handler via NetworkManager-dispatcher service. With this change, a generic device now represents two different device classes: - existing interfaces that are not natively supported or recognized by NetworkManager. Those devices have the `has_device_handler` property set to FALSE; - interfaces that are created by NM by invoking the device-handler; they have `has_device_handler` set to TRUE.
This commit is contained in:
parent
ae7ac3c8b7
commit
df6c35ec75
|
@ -396,6 +396,7 @@ nm_device_factory_manager_load_factories(NMDeviceFactoryManagerFactoryFunc callb
|
|||
_ADD_INTERNAL(nm_bridge_device_factory_get_type);
|
||||
_ADD_INTERNAL(nm_dummy_device_factory_get_type);
|
||||
_ADD_INTERNAL(nm_ethernet_device_factory_get_type);
|
||||
_ADD_INTERNAL(nm_generic_device_factory_get_type);
|
||||
_ADD_INTERNAL(nm_hsr_device_factory_get_type);
|
||||
_ADD_INTERNAL(nm_infiniband_device_factory_get_type);
|
||||
_ADD_INTERNAL(nm_ip_tunnel_device_factory_get_type);
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
||||
/*
|
||||
* Copyright (C) 2013 Red Hat, Inc.
|
||||
* Copyright (C) 2013-2023 Red Hat, Inc.
|
||||
*/
|
||||
|
||||
#include "src/core/nm-default-daemon.h"
|
||||
|
@ -10,13 +10,27 @@
|
|||
#include "nm-device-private.h"
|
||||
#include "libnm-platform/nm-platform.h"
|
||||
#include "libnm-core-intern/nm-core-internal.h"
|
||||
#include "nm-dispatcher.h"
|
||||
#include "nm-device-factory.h"
|
||||
|
||||
#define _NMLOG_DEVICE_TYPE NMDeviceGeneric
|
||||
#include "devices/nm-device-logging.h"
|
||||
|
||||
/*****************************************************************************/
|
||||
|
||||
NM_GOBJECT_PROPERTIES_DEFINE_BASE(PROP_TYPE_DESCRIPTION, );
|
||||
NM_GOBJECT_PROPERTIES_DEFINE(NMDeviceGeneric, PROP_TYPE_DESCRIPTION, PROP_HAS_DEVICE_HANDLER, );
|
||||
|
||||
typedef struct {
|
||||
const char *type_description;
|
||||
const char *type_description;
|
||||
bool prepare_done : 1;
|
||||
bool has_device_handler : 1;
|
||||
NMDispatcherCallId *dispatcher_call_id;
|
||||
struct {
|
||||
NMDeviceDeactivateCallback callback;
|
||||
gpointer callback_data;
|
||||
GCancellable *cancellable;
|
||||
gulong cancellable_id;
|
||||
} deactivate;
|
||||
} NMDeviceGenericPrivate;
|
||||
|
||||
struct _NMDeviceGeneric {
|
||||
|
@ -38,13 +52,151 @@ G_DEFINE_TYPE(NMDeviceGeneric, nm_device_generic, NM_TYPE_DEVICE)
|
|||
static NMDeviceCapabilities
|
||||
get_generic_capabilities(NMDevice *device)
|
||||
{
|
||||
int ifindex = nm_device_get_ifindex(device);
|
||||
NMDeviceGenericPrivate *priv = NM_DEVICE_GENERIC_GET_PRIVATE(device);
|
||||
int ifindex = nm_device_get_ifindex(device);
|
||||
NMDeviceCapabilities cap = NM_DEVICE_CAP_NONE;
|
||||
|
||||
if (priv->has_device_handler)
|
||||
cap |= NM_DEVICE_CAP_IS_SOFTWARE;
|
||||
|
||||
if (ifindex > 0
|
||||
&& nm_platform_link_supports_carrier_detect(nm_device_get_platform(device), ifindex))
|
||||
return NM_DEVICE_CAP_CARRIER_DETECT;
|
||||
else
|
||||
return NM_DEVICE_CAP_NONE;
|
||||
cap |= NM_DEVICE_CAP_CARRIER_DETECT;
|
||||
|
||||
return cap;
|
||||
}
|
||||
|
||||
static void
|
||||
device_add_dispatcher_cb(NMDispatcherCallId *call_id,
|
||||
gpointer user_data,
|
||||
gboolean success,
|
||||
const char *error,
|
||||
GHashTable *dict)
|
||||
{
|
||||
nm_auto_unref_object NMDeviceGeneric *self = NM_DEVICE_GENERIC(user_data);
|
||||
NMDeviceGenericPrivate *priv = NM_DEVICE_GENERIC_GET_PRIVATE(self);
|
||||
NMDevice *device = NM_DEVICE(self);
|
||||
NMPlatform *platform = nm_device_get_platform(device);
|
||||
const NMPlatformLink *link;
|
||||
int ifindex = -1;
|
||||
const char *ifindex_str;
|
||||
NMSettingConnection *s_con;
|
||||
|
||||
nm_assert(call_id == priv->dispatcher_call_id);
|
||||
priv->dispatcher_call_id = NULL;
|
||||
|
||||
if (!success) {
|
||||
_LOGW(LOGD_CORE, "device handler 'device-add' failed: %s", error);
|
||||
nm_device_state_changed(device,
|
||||
NM_DEVICE_STATE_FAILED,
|
||||
NM_DEVICE_STATE_REASON_DEVICE_HANDLER_FAILED);
|
||||
return;
|
||||
}
|
||||
|
||||
ifindex_str = g_hash_table_lookup(dict, "IFINDEX");
|
||||
if (!ifindex_str) {
|
||||
_LOGW(LOGD_CORE, "device handler 'device-add' didn't return a IFINDEX key");
|
||||
nm_device_state_changed(device,
|
||||
NM_DEVICE_STATE_FAILED,
|
||||
NM_DEVICE_STATE_REASON_DEVICE_HANDLER_FAILED);
|
||||
return;
|
||||
}
|
||||
|
||||
ifindex = _nm_utils_ascii_str_to_int64(ifindex_str, 10, 1, G_MAXINT32, -1);
|
||||
if (ifindex < 0) {
|
||||
_LOGW(LOGD_CORE, "device handler 'device-add' returned invalid ifindex '%s'", ifindex_str);
|
||||
nm_device_state_changed(device,
|
||||
NM_DEVICE_STATE_FAILED,
|
||||
NM_DEVICE_STATE_REASON_DEVICE_HANDLER_FAILED);
|
||||
return;
|
||||
}
|
||||
|
||||
_LOGD(LOGD_DEVICE, "device handler 'device-add' returned ifindex %d", ifindex);
|
||||
|
||||
/* Check that the ifindex is valid and matches the interface name. */
|
||||
nm_platform_process_events(platform);
|
||||
link = nm_platform_link_get(platform, ifindex);
|
||||
if (!link) {
|
||||
_LOGW(LOGD_DEVICE,
|
||||
"device handler 'device-add' didn't create link with ifindex %d",
|
||||
ifindex);
|
||||
nm_device_state_changed(device,
|
||||
NM_DEVICE_STATE_FAILED,
|
||||
NM_DEVICE_STATE_REASON_DEVICE_HANDLER_FAILED);
|
||||
return;
|
||||
}
|
||||
|
||||
s_con = nm_device_get_applied_setting(device, NM_TYPE_SETTING_CONNECTION);
|
||||
nm_assert(s_con);
|
||||
|
||||
if (!nm_streq(link->name, nm_setting_connection_get_interface_name(s_con))) {
|
||||
_LOGW(LOGD_DEVICE,
|
||||
"device handler 'device-add' created a kernel link with name '%s' instead of '%s'",
|
||||
link->name,
|
||||
nm_setting_connection_get_interface_name(s_con));
|
||||
nm_device_state_changed(device,
|
||||
NM_DEVICE_STATE_FAILED,
|
||||
NM_DEVICE_STATE_REASON_DEVICE_HANDLER_FAILED);
|
||||
return;
|
||||
}
|
||||
|
||||
priv->prepare_done = TRUE;
|
||||
nm_device_activate_schedule_stage1_device_prepare(device, FALSE);
|
||||
}
|
||||
|
||||
static NMActStageReturn
|
||||
act_stage1_prepare(NMDevice *self, NMDeviceStateReason *out_failure_reason)
|
||||
{
|
||||
NMDevice *device = NM_DEVICE(self);
|
||||
NMDeviceGenericPrivate *priv = NM_DEVICE_GENERIC_GET_PRIVATE(device);
|
||||
NMSettingGeneric *s_generic;
|
||||
const char *type_desc;
|
||||
int ifindex;
|
||||
|
||||
s_generic = nm_device_get_applied_setting(device, NM_TYPE_SETTING_GENERIC);
|
||||
g_return_val_if_fail(s_generic, NM_ACT_STAGE_RETURN_FAILURE);
|
||||
|
||||
if (!nm_setting_generic_get_device_handler(s_generic))
|
||||
return NM_ACT_STAGE_RETURN_SUCCESS;
|
||||
|
||||
if (priv->prepare_done) {
|
||||
/* after we create a new interface via a device-handler, update the
|
||||
* type description */
|
||||
ifindex = nm_device_get_ip_ifindex(NM_DEVICE(self));
|
||||
if (ifindex > 0) {
|
||||
type_desc = nm_platform_link_get_type_name(nm_device_get_platform(device), ifindex);
|
||||
if (!nm_streq0(priv->type_description, type_desc)) {
|
||||
priv->type_description = type_desc;
|
||||
_notify(NM_DEVICE_GENERIC(self), PROP_TYPE_DESCRIPTION);
|
||||
}
|
||||
}
|
||||
return NM_ACT_STAGE_RETURN_SUCCESS;
|
||||
}
|
||||
|
||||
if (priv->dispatcher_call_id) {
|
||||
nm_dispatcher_call_cancel(priv->dispatcher_call_id);
|
||||
priv->dispatcher_call_id = NULL;
|
||||
}
|
||||
|
||||
_LOGD(LOGD_CORE, "calling device handler 'device-add'");
|
||||
if (!nm_dispatcher_call_device_handler(NM_DISPATCHER_ACTION_DEVICE_ADD,
|
||||
device,
|
||||
NULL,
|
||||
device_add_dispatcher_cb,
|
||||
g_object_ref(self),
|
||||
&priv->dispatcher_call_id)) {
|
||||
_LOGW(LOGD_DEVICE, "failed to call device handler 'device-add'");
|
||||
NM_SET_OUT(out_failure_reason, NM_DEVICE_STATE_REASON_DEVICE_HANDLER_FAILED);
|
||||
return NM_ACT_STAGE_RETURN_FAILURE;
|
||||
}
|
||||
|
||||
return NM_ACT_STAGE_RETURN_POSTPONE;
|
||||
}
|
||||
|
||||
static void
|
||||
act_stage3_ip_config(NMDevice *device, int addr_family)
|
||||
{
|
||||
nm_device_devip_set_state(device, addr_family, NM_DEVICE_IP_STATE_READY, NULL);
|
||||
}
|
||||
|
||||
static const char *
|
||||
|
@ -110,6 +262,111 @@ update_connection(NMDevice *device, NMConnection *connection)
|
|||
NULL);
|
||||
}
|
||||
|
||||
static gboolean
|
||||
create_and_realize(NMDevice *device,
|
||||
NMConnection *connection,
|
||||
NMDevice *parent,
|
||||
const NMPlatformLink **out_plink,
|
||||
GError **error)
|
||||
{
|
||||
/* The actual interface is created during stage1 once the device
|
||||
* starts activating, as we need to call the dispatcher service
|
||||
* which returns asynchronously */
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
static void
|
||||
deactivate_clear_data(NMDeviceGeneric *self)
|
||||
{
|
||||
NMDeviceGenericPrivate *priv = NM_DEVICE_GENERIC_GET_PRIVATE(self);
|
||||
|
||||
if (priv->dispatcher_call_id) {
|
||||
nm_dispatcher_call_cancel(priv->dispatcher_call_id);
|
||||
priv->dispatcher_call_id = NULL;
|
||||
}
|
||||
|
||||
priv->deactivate.callback = NULL;
|
||||
priv->deactivate.callback_data = NULL;
|
||||
g_clear_object(&priv->deactivate.cancellable);
|
||||
}
|
||||
|
||||
static void
|
||||
device_delete_dispatcher_cb(NMDispatcherCallId *call_id,
|
||||
gpointer user_data,
|
||||
gboolean success,
|
||||
const char *error,
|
||||
GHashTable *dict)
|
||||
{
|
||||
NMDeviceGeneric *self = user_data;
|
||||
NMDeviceGenericPrivate *priv = NM_DEVICE_GENERIC_GET_PRIVATE(self);
|
||||
gs_free_error GError *local = NULL;
|
||||
|
||||
nm_assert(call_id == priv->dispatcher_call_id);
|
||||
priv->dispatcher_call_id = NULL;
|
||||
|
||||
if (success)
|
||||
_LOGT(LOGD_DEVICE, "deactivate: async callback");
|
||||
else {
|
||||
local = g_error_new(NM_DEVICE_ERROR,
|
||||
NM_DEVICE_ERROR_FAILED,
|
||||
"device handler 'device-delete' failed with error: %s",
|
||||
error);
|
||||
}
|
||||
|
||||
priv->deactivate.callback(NM_DEVICE(self), local, priv->deactivate.callback_data);
|
||||
nm_clear_g_cancellable_disconnect(priv->deactivate.cancellable,
|
||||
&priv->deactivate.cancellable_id);
|
||||
deactivate_clear_data(self);
|
||||
}
|
||||
|
||||
static void
|
||||
deactivate_cancellable_cancelled(GCancellable *cancellable, NMDeviceGeneric *self)
|
||||
{
|
||||
NMDeviceGenericPrivate *priv = NM_DEVICE_GENERIC_GET_PRIVATE(self);
|
||||
gs_free_error GError *error = NULL;
|
||||
|
||||
error = nm_utils_error_new_cancelled(FALSE, NULL);
|
||||
priv->deactivate.callback(NM_DEVICE(self), error, priv->deactivate.callback_data);
|
||||
|
||||
deactivate_clear_data(self);
|
||||
}
|
||||
|
||||
static void
|
||||
deactivate_async(NMDevice *device,
|
||||
GCancellable *cancellable,
|
||||
NMDeviceDeactivateCallback callback,
|
||||
gpointer callback_user_data)
|
||||
{
|
||||
NMDeviceGeneric *self = NM_DEVICE_GENERIC(device);
|
||||
NMDeviceGenericPrivate *priv = NM_DEVICE_GENERIC_GET_PRIVATE(self);
|
||||
|
||||
_LOGT(LOGD_CORE, "deactivate: start async");
|
||||
|
||||
priv->prepare_done = FALSE;
|
||||
|
||||
if (priv->dispatcher_call_id) {
|
||||
nm_dispatcher_call_cancel(priv->dispatcher_call_id);
|
||||
priv->dispatcher_call_id = NULL;
|
||||
}
|
||||
|
||||
g_object_ref(self);
|
||||
priv->deactivate.callback = callback;
|
||||
priv->deactivate.callback_data = callback_user_data;
|
||||
priv->deactivate.cancellable = g_object_ref(cancellable);
|
||||
priv->deactivate.cancellable_id =
|
||||
g_cancellable_connect(cancellable,
|
||||
G_CALLBACK(deactivate_cancellable_cancelled),
|
||||
self,
|
||||
NULL);
|
||||
|
||||
nm_dispatcher_call_device_handler(NM_DISPATCHER_ACTION_DEVICE_DELETE,
|
||||
device,
|
||||
NULL,
|
||||
device_delete_dispatcher_cb,
|
||||
self,
|
||||
&priv->dispatcher_call_id);
|
||||
}
|
||||
|
||||
/*****************************************************************************/
|
||||
|
||||
static void
|
||||
|
@ -122,6 +379,26 @@ get_property(GObject *object, guint prop_id, GValue *value, GParamSpec *pspec)
|
|||
case PROP_TYPE_DESCRIPTION:
|
||||
g_value_set_string(value, priv->type_description);
|
||||
break;
|
||||
case PROP_HAS_DEVICE_HANDLER:
|
||||
g_value_set_boolean(value, priv->has_device_handler);
|
||||
break;
|
||||
default:
|
||||
G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
set_property(GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec)
|
||||
{
|
||||
NMDeviceGeneric *self = (NMDeviceGeneric *) object;
|
||||
NMDeviceGenericPrivate *priv = NM_DEVICE_GENERIC_GET_PRIVATE(self);
|
||||
|
||||
switch (prop_id) {
|
||||
case PROP_HAS_DEVICE_HANDLER:
|
||||
/* construct-only */
|
||||
priv->has_device_handler = g_value_get_boolean(value);
|
||||
break;
|
||||
default:
|
||||
G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
|
||||
break;
|
||||
|
@ -137,16 +414,41 @@ nm_device_generic_init(NMDeviceGeneric *self)
|
|||
static GObject *
|
||||
constructor(GType type, guint n_construct_params, GObjectConstructParam *construct_params)
|
||||
{
|
||||
GObject *object;
|
||||
GObject *object;
|
||||
NMDeviceGenericPrivate *priv;
|
||||
|
||||
object = G_OBJECT_CLASS(nm_device_generic_parent_class)
|
||||
->constructor(type, n_construct_params, construct_params);
|
||||
|
||||
nm_device_set_unmanaged_flags((NMDevice *) object, NM_UNMANAGED_BY_DEFAULT, TRUE);
|
||||
priv = NM_DEVICE_GENERIC_GET_PRIVATE(object);
|
||||
/* If the device is software (has a device-handler), don't set
|
||||
* unmanaged-by-default so that the device can autoconnect if
|
||||
* necessary. */
|
||||
if (!priv->has_device_handler)
|
||||
nm_device_set_unmanaged_flags((NMDevice *) object, NM_UNMANAGED_BY_DEFAULT, TRUE);
|
||||
|
||||
return object;
|
||||
}
|
||||
|
||||
static NMDevice *
|
||||
create_device(NMDeviceFactory *factory,
|
||||
const char *iface,
|
||||
const NMPlatformLink *plink,
|
||||
NMConnection *connection,
|
||||
gboolean *out_ignore)
|
||||
{
|
||||
return g_object_new(NM_TYPE_DEVICE_GENERIC,
|
||||
NM_DEVICE_IFACE,
|
||||
iface,
|
||||
NM_DEVICE_TYPE_DESC,
|
||||
"Generic",
|
||||
NM_DEVICE_DEVICE_TYPE,
|
||||
NM_DEVICE_TYPE_GENERIC,
|
||||
NM_DEVICE_GENERIC_HAS_DEVICE_HANDLER,
|
||||
TRUE,
|
||||
NULL);
|
||||
}
|
||||
|
||||
NMDevice *
|
||||
nm_device_generic_new(const NMPlatformLink *plink, gboolean nm_plugin_missing)
|
||||
{
|
||||
|
@ -188,6 +490,7 @@ nm_device_generic_class_init(NMDeviceGenericClass *klass)
|
|||
|
||||
object_class->constructor = constructor;
|
||||
object_class->get_property = get_property;
|
||||
object_class->set_property = set_property;
|
||||
|
||||
dbus_object_class->interface_infos = NM_DBUS_INTERFACE_INFOS(&interface_info_device_generic);
|
||||
|
||||
|
@ -195,10 +498,14 @@ nm_device_generic_class_init(NMDeviceGenericClass *klass)
|
|||
device_class->connection_type_check_compatible = NM_SETTING_GENERIC_SETTING_NAME;
|
||||
device_class->link_types = NM_DEVICE_DEFINE_LINK_TYPES(NM_LINK_TYPE_ANY);
|
||||
|
||||
device_class->realize_start_notify = realize_start_notify;
|
||||
device_class->act_stage1_prepare = act_stage1_prepare;
|
||||
device_class->act_stage3_ip_config = act_stage3_ip_config;
|
||||
device_class->check_connection_compatible = check_connection_compatible;
|
||||
device_class->create_and_realize = create_and_realize;
|
||||
device_class->deactivate_async = deactivate_async;
|
||||
device_class->get_generic_capabilities = get_generic_capabilities;
|
||||
device_class->get_type_description = get_type_description;
|
||||
device_class->check_connection_compatible = check_connection_compatible;
|
||||
device_class->realize_start_notify = realize_start_notify;
|
||||
device_class->update_connection = update_connection;
|
||||
|
||||
obj_properties[PROP_TYPE_DESCRIPTION] =
|
||||
|
@ -207,6 +514,18 @@ nm_device_generic_class_init(NMDeviceGenericClass *klass)
|
|||
"",
|
||||
NULL,
|
||||
G_PARAM_READABLE | G_PARAM_STATIC_STRINGS);
|
||||
|
||||
obj_properties[PROP_HAS_DEVICE_HANDLER] = g_param_spec_boolean(
|
||||
NM_DEVICE_GENERIC_HAS_DEVICE_HANDLER,
|
||||
"",
|
||||
"",
|
||||
FALSE,
|
||||
G_PARAM_READABLE | G_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS);
|
||||
g_object_class_install_properties(object_class, _PROPERTY_ENUMS_LAST, obj_properties);
|
||||
}
|
||||
|
||||
NM_DEVICE_FACTORY_DEFINE_INTERNAL(
|
||||
GENERIC,
|
||||
Generic,
|
||||
generic,
|
||||
NM_DEVICE_FACTORY_DECLARE_SETTING_TYPES(NM_SETTING_GENERIC_SETTING_NAME),
|
||||
factory_class->create_device = create_device;);
|
||||
|
|
|
@ -18,7 +18,8 @@
|
|||
#define NM_DEVICE_GENERIC_GET_CLASS(obj) \
|
||||
(G_TYPE_INSTANCE_GET_CLASS((obj), NM_TYPE_DEVICE_GENERIC, NMDeviceGenericClass))
|
||||
|
||||
#define NM_DEVICE_GENERIC_TYPE_DESCRIPTION "type-description"
|
||||
#define NM_DEVICE_GENERIC_TYPE_DESCRIPTION "type-description"
|
||||
#define NM_DEVICE_GENERIC_HAS_DEVICE_HANDLER "has-device-handler"
|
||||
|
||||
typedef struct _NMDeviceGeneric NMDeviceGeneric;
|
||||
typedef struct _NMDeviceGenericClass NMDeviceGenericClass;
|
||||
|
|
|
@ -127,7 +127,9 @@ NM_UTILS_LOOKUP_STR_DEFINE(
|
|||
NM_UTILS_LOOKUP_STR_ITEM(NM_DEVICE_STATE_REASON_IP_METHOD_UNSUPPORTED, "ip-method-unsupported"),
|
||||
NM_UTILS_LOOKUP_STR_ITEM(NM_DEVICE_STATE_REASON_SRIOV_CONFIGURATION_FAILED,
|
||||
"sriov-configuration-failed"),
|
||||
NM_UTILS_LOOKUP_STR_ITEM(NM_DEVICE_STATE_REASON_PEER_NOT_FOUND, "peer-not-found"), );
|
||||
NM_UTILS_LOOKUP_STR_ITEM(NM_DEVICE_STATE_REASON_PEER_NOT_FOUND, "peer-not-found"),
|
||||
NM_UTILS_LOOKUP_STR_ITEM(NM_DEVICE_STATE_REASON_DEVICE_HANDLER_FAILED,
|
||||
"device-handler-failed"), );
|
||||
|
||||
NM_UTILS_LOOKUP_STR_DEFINE(nm_device_mtu_source_to_string,
|
||||
NMDeviceMtuSource,
|
||||
|
|
|
@ -529,7 +529,7 @@ dispatcher_results_process(NMDispatcherAction action,
|
|||
NM_SET_OUT(out_success, FALSE);
|
||||
NM_SET_OUT(out_dict, NULL);
|
||||
NM_SET_OUT(out_error_msg,
|
||||
err2 ? g_strdup_printf("%s: Error: %s", err, err2) : g_strdup(err));
|
||||
err2 ? g_strdup_printf("%s (Error: %s)", err, err2) : g_strdup(err));
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
|
|
@ -3207,6 +3207,13 @@ nm_connection_is_virtual(NMConnection *connection)
|
|||
return !!nm_setting_pppoe_get_parent(s_pppoe);
|
||||
}
|
||||
|
||||
if (nm_streq(type, NM_SETTING_GENERIC_SETTING_NAME)) {
|
||||
NMSettingGeneric *s_generic;
|
||||
|
||||
s_generic = nm_connection_get_setting_generic(connection);
|
||||
return !!nm_setting_generic_get_device_handler(s_generic);
|
||||
}
|
||||
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
|
|
|
@ -610,6 +610,8 @@ typedef enum {
|
|||
* @NM_DEVICE_STATE_REASON_IP_METHOD_UNSUPPORTED: The selected IP method is not supported
|
||||
* @NM_DEVICE_STATE_REASON_SRIOV_CONFIGURATION_FAILED: configuration of SR-IOV parameters failed
|
||||
* @NM_DEVICE_STATE_REASON_PEER_NOT_FOUND: The Wi-Fi P2P peer could not be found
|
||||
* @NM_DEVICE_STATE_REASON_DEVICE_HANDLER_FAILED: The device handler dispatcher returned an
|
||||
* error. Since: 1.46
|
||||
*
|
||||
* Device state change reason codes
|
||||
*/
|
||||
|
@ -682,6 +684,7 @@ typedef enum {
|
|||
NM_DEVICE_STATE_REASON_IP_METHOD_UNSUPPORTED = 65,
|
||||
NM_DEVICE_STATE_REASON_SRIOV_CONFIGURATION_FAILED = 66,
|
||||
NM_DEVICE_STATE_REASON_PEER_NOT_FOUND = 67,
|
||||
NM_DEVICE_STATE_REASON_DEVICE_HANDLER_FAILED = 68,
|
||||
} NMDeviceStateReason;
|
||||
|
||||
/**
|
||||
|
|
|
@ -464,7 +464,9 @@ NM_UTILS_LOOKUP_STR_DEFINE(
|
|||
NM_UTILS_LOOKUP_ITEM(NM_DEVICE_STATE_REASON_SRIOV_CONFIGURATION_FAILED,
|
||||
N_("Failed to configure SR-IOV parameters")),
|
||||
NM_UTILS_LOOKUP_ITEM(NM_DEVICE_STATE_REASON_PEER_NOT_FOUND,
|
||||
N_("The Wi-Fi P2P peer could not be found")), );
|
||||
N_("The Wi-Fi P2P peer could not be found")),
|
||||
NM_UTILS_LOOKUP_ITEM(NM_DEVICE_STATE_REASON_DEVICE_HANDLER_FAILED,
|
||||
N_("The device handler dispatcher returned an error")), );
|
||||
|
||||
NM_UTILS_LOOKUP_STR_DEFINE(
|
||||
nm_active_connection_state_reason_to_string,
|
||||
|
|
Loading…
Reference in a new issue