NetworkManager/src/core/devices/nm-device-loopback.c
Lubomir Rintel be034a1f3f device: simplify the nm_utils_complete_generic() machinery
The point is to get rid of device/connection type specific arguments, to
eventually be able to complete the connection on AddAndActivate before knowing
which factory is going to take care of creating the device.

Aside from that, the whole thing is pretty awful -- with complicated
macros and variadic argument (ugh). Let's get rid of that.
2025-01-20 06:13:59 +01:00

166 lines
6 KiB
C

/* SPDX-License-Identifier: LGPL-2.1-or-later */
/*
* Copyright (C) 2022 Red Hat, Inc.
*/
#include "src/core/nm-default-daemon.h"
#include "nm-device-loopback.h"
#include "nm-device-private.h"
#include "libnm-platform/nm-platform.h"
#include "nm-device-factory.h"
#include "nm-setting-loopback.h"
#include "libnm-core-aux-intern/nm-libnm-core-utils.h"
#define _NMLOG_DEVICE_TYPE NMDeviceLoopback
#include "nm-device-logging.h"
/*****************************************************************************/
struct _NMDeviceLoopback {
NMDevice parent;
};
struct _NMDeviceLoopbackClass {
NMDeviceClass parent;
};
G_DEFINE_TYPE(NMDeviceLoopback, nm_device_loopback, NM_TYPE_DEVICE)
/*****************************************************************************/
static NMDeviceCapabilities
get_generic_capabilities(NMDevice *dev)
{
/* In many aspects, loopback is a software device. Still, don't return
* NM_DEVICE_CAP_IS_SOFTWARE here, because we cannot delete nor create
* such an interface. */
return NM_DEVICE_CAP_NONE;
}
static guint32
get_configured_mtu(NMDevice *device, NMDeviceMtuSource *out_source, gboolean *out_force)
{
return nm_device_get_configured_mtu_from_connection(device,
NM_TYPE_SETTING_LOOPBACK,
out_source);
}
static const char *
get_auto_ip_config_method(NMDevice *device, int addr_family)
{
return NM_IS_IPv4(addr_family) ? NM_SETTING_IP4_CONFIG_METHOD_MANUAL
: NM_SETTING_IP6_CONFIG_METHOD_MANUAL;
}
static gboolean
complete_connection(NMDevice *device,
NMConnection *connection,
const char *specific_object,
NMConnection *const *existing_connections,
GError **error)
{
nm_utils_complete_generic(nm_device_get_platform(device),
connection,
NM_SETTING_LOOPBACK_SETTING_NAME,
existing_connections,
NULL,
_("Loopback connection"),
NULL,
nm_device_get_ip_iface(device));
_nm_connection_ensure_setting(connection, NM_TYPE_SETTING_LOOPBACK);
return TRUE;
}
static void
update_connection(NMDevice *device, NMConnection *connection)
{
_nm_connection_ensure_setting(connection, NM_TYPE_SETTING_LOOPBACK);
}
static gboolean
can_reapply_change(NMDevice *device,
const char *setting_name,
NMSetting *s_old,
NMSetting *s_new,
GHashTable *diffs,
GError **error)
{
if (!nm_streq(setting_name, NM_SETTING_LOOPBACK_SETTING_NAME)) {
return NM_DEVICE_CLASS(nm_device_loopback_parent_class)
->can_reapply_change(device, setting_name, s_old, s_new, diffs, error);
}
return nm_device_hash_check_invalid_keys(diffs,
NM_SETTING_LOOPBACK_SETTING_NAME,
error,
NM_SETTING_LOOPBACK_MTU);
}
/*****************************************************************************/
static const NMDBusInterfaceInfoExtended interface_info_device_loopback = {
.parent = NM_DEFINE_GDBUS_INTERFACE_INFO_INIT(NM_DBUS_INTERFACE_DEVICE_LOOPBACK, ),
};
static void
nm_device_loopback_init(NMDeviceLoopback *self)
{}
static void
nm_device_loopback_class_init(NMDeviceLoopbackClass *klass)
{
NMDeviceClass *device_class = NM_DEVICE_CLASS(klass);
NMDBusObjectClass *dbus_object_class = NM_DBUS_OBJECT_CLASS(klass);
dbus_object_class->interface_infos = NM_DBUS_INTERFACE_INFOS(&interface_info_device_loopback);
device_class->connection_type_supported = NM_SETTING_LOOPBACK_SETTING_NAME;
device_class->connection_type_check_compatible = NM_SETTING_LOOPBACK_SETTING_NAME;
device_class->link_types = NM_DEVICE_DEFINE_LINK_TYPES(NM_LINK_TYPE_LOOPBACK);
device_class->complete_connection = complete_connection;
device_class->get_generic_capabilities = get_generic_capabilities;
device_class->update_connection = update_connection;
device_class->act_stage1_prepare_set_hwaddr_ethernet = TRUE;
device_class->get_auto_ip_config_method = get_auto_ip_config_method;
device_class->get_configured_mtu = get_configured_mtu;
device_class->allow_autoconnect_on_external = TRUE;
device_class->can_reapply_change = can_reapply_change;
}
/*****************************************************************************/
#define NM_TYPE_LOOPBACK_DEVICE_FACTORY (nm_loopback_device_factory_get_type())
#define NM_LOOPBACK_DEVICE_FACTORY(obj) \
(_NM_G_TYPE_CHECK_INSTANCE_CAST((obj), \
NM_TYPE_LOOPBACK_DEVICE_FACTORY, \
NMLoopbackDeviceFactory))
static NMDevice *
create_device(NMDeviceFactory *factory,
const char *iface,
const NMPlatformLink *plink,
NMConnection *connection,
gboolean *out_ignore)
{
return g_object_new(NM_TYPE_DEVICE_LOOPBACK,
NM_DEVICE_IFACE,
iface,
NM_DEVICE_TYPE_DESC,
"Loopback",
NM_DEVICE_DEVICE_TYPE,
NM_DEVICE_TYPE_LOOPBACK,
NM_DEVICE_LINK_TYPE,
NM_LINK_TYPE_LOOPBACK,
NULL);
}
NM_DEVICE_FACTORY_DEFINE_INTERNAL(
LOOPBACK,
Loopback,
loopback,
NM_DEVICE_FACTORY_DECLARE_LINK_TYPES(NM_LINK_TYPE_LOOPBACK)
NM_DEVICE_FACTORY_DECLARE_SETTING_TYPES(NM_SETTING_LOOPBACK_SETTING_NAME),
factory_class->create_device = create_device;);