Blame src/core/devices/nm-device-vrf.c

Packit Service 5ffa24
/* SPDX-License-Identifier: GPL-2.0-or-later */
Packit Service 5ffa24
Packit Service 2bceb2
#include "src/core/nm-default-daemon.h"
Packit Service 5ffa24
Packit Service 5ffa24
#include "nm-device-vrf.h"
Packit Service 5ffa24
Packit Service 5ffa24
#include "nm-core-internal.h"
Packit Service 5ffa24
#include "nm-device-factory.h"
Packit Service 5ffa24
#include "nm-device-private.h"
Packit Service 5ffa24
#include "nm-manager.h"
Packit Service 5ffa24
#include "nm-setting-vrf.h"
Packit Service 5ffa24
#include "platform/nm-platform.h"
Packit Service 5ffa24
#include "settings/nm-settings.h"
Packit Service 5ffa24
Packit Service 5ffa24
#define _NMLOG_DEVICE_TYPE NMDeviceVrf
Packit Service 5ffa24
#include "nm-device-logging.h"
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
NM_GOBJECT_PROPERTIES_DEFINE(NMDeviceVrf, PROP_TABLE, );
Packit Service 5ffa24
Packit Service 5ffa24
typedef struct {
Packit Service 5ffa24
    NMPlatformLnkVrf props;
Packit Service 5ffa24
} NMDeviceVrfPrivate;
Packit Service 5ffa24
Packit Service 5ffa24
struct _NMDeviceVrf {
Packit Service 5ffa24
    NMDevice           parent;
Packit Service 5ffa24
    NMDeviceVrfPrivate _priv;
Packit Service 5ffa24
};
Packit Service 5ffa24
Packit Service 5ffa24
struct _NMDeviceVrfClass {
Packit Service 5ffa24
    NMDeviceClass parent;
Packit Service 5ffa24
};
Packit Service 5ffa24
Packit Service 5ffa24
G_DEFINE_TYPE(NMDeviceVrf, nm_device_vrf, NM_TYPE_DEVICE)
Packit Service 5ffa24
Packit Service 5ffa24
#define NM_DEVICE_VRF_GET_PRIVATE(self) \
Packit Service 5ffa24
    _NM_GET_PRIVATE(self, NMDeviceVrf, NM_IS_DEVICE_VRF, NMDevice)
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
do_update_properties(NMDeviceVrf *self, const NMPlatformLnkVrf *props)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMDeviceVrfPrivate *priv   = NM_DEVICE_VRF_GET_PRIVATE(self);
Packit Service 5ffa24
    GObject *           object = G_OBJECT(self);
Packit Service 5ffa24
    NMPlatformLnkVrf    props_null;
Packit Service 5ffa24
Packit Service 5ffa24
    if (!props) {
Packit Service 5ffa24
        props_null = (NMPlatformLnkVrf){};
Packit Service 5ffa24
        props      = &props_null;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    g_object_freeze_notify(object);
Packit Service 5ffa24
Packit Service 5ffa24
#define CHECK_PROPERTY_CHANGED(field, prop)      \
Packit Service 5ffa24
    G_STMT_START                                 \
Packit Service 5ffa24
    {                                            \
Packit Service 5ffa24
        if (priv->props.field != props->field) { \
Packit Service 5ffa24
            priv->props.field = props->field;    \
Packit Service 5ffa24
            _notify(self, prop);                 \
Packit Service 5ffa24
        }                                        \
Packit Service 5ffa24
    }                                            \
Packit Service 5ffa24
    G_STMT_END
Packit Service 5ffa24
Packit Service 5ffa24
    CHECK_PROPERTY_CHANGED(table, PROP_TABLE);
Packit Service 5ffa24
Packit Service 5ffa24
    g_object_thaw_notify(object);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
update_properties(NMDevice *device)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMDeviceVrf *           self = NM_DEVICE_VRF(device);
Packit Service 5ffa24
    const NMPlatformLnkVrf *props;
Packit Service 5ffa24
Packit Service 5ffa24
    props = nm_platform_link_get_lnk_vrf(nm_device_get_platform(device),
Packit Service 5ffa24
                                         nm_device_get_ifindex(device),
Packit Service 5ffa24
                                         NULL);
Packit Service 5ffa24
    if (!props) {
Packit Service 5ffa24
        _LOGW(LOGD_PLATFORM, "could not get vrf properties");
Packit Service 5ffa24
        return;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    do_update_properties(self, props);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static NMDeviceCapabilities
Packit Service 5ffa24
get_generic_capabilities(NMDevice *dev)
Packit Service 5ffa24
{
Packit Service 5ffa24
    return NM_DEVICE_CAP_IS_SOFTWARE;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
link_changed(NMDevice *device, const NMPlatformLink *pllink)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NM_DEVICE_CLASS(nm_device_vrf_parent_class)->link_changed(device, pllink);
Packit Service 5ffa24
    update_properties(device);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
unrealize_notify(NMDevice *device)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMDeviceVrf *self = NM_DEVICE_VRF(device);
Packit Service 5ffa24
Packit Service 5ffa24
    NM_DEVICE_CLASS(nm_device_vrf_parent_class)->unrealize_notify(device);
Packit Service 5ffa24
Packit Service 5ffa24
    do_update_properties(self, NULL);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static gboolean
Packit Service 5ffa24
create_and_realize(NMDevice *             device,
Packit Service 5ffa24
                   NMConnection *         connection,
Packit Service 5ffa24
                   NMDevice *             parent,
Packit Service 5ffa24
                   const NMPlatformLink **out_plink,
Packit Service 5ffa24
                   GError **              error)
Packit Service 5ffa24
{
Packit Service 5ffa24
    const char *     iface = nm_device_get_iface(device);
Packit Service 5ffa24
    NMPlatformLnkVrf props = {};
Packit Service 5ffa24
    NMSettingVrf *   s_vrf;
Packit Service 5ffa24
    int              r;
Packit Service 5ffa24
Packit Service 5ffa24
    s_vrf = _nm_connection_get_setting(connection, NM_TYPE_SETTING_VRF);
Packit Service 5ffa24
    nm_assert(s_vrf);
Packit Service 5ffa24
Packit Service 5ffa24
    props.table = nm_setting_vrf_get_table(s_vrf);
Packit Service 5ffa24
Packit Service 5ffa24
    r = nm_platform_link_vrf_add(nm_device_get_platform(device), iface, &props, out_plink);
Packit Service 5ffa24
    if (r < 0) {
Packit Service 5ffa24
        g_set_error(error,
Packit Service 5ffa24
                    NM_DEVICE_ERROR,
Packit Service 5ffa24
                    NM_DEVICE_ERROR_CREATION_FAILED,
Packit Service 5ffa24
                    "Failed to create VRF interface '%s' for '%s': %s",
Packit Service 5ffa24
                    iface,
Packit Service 5ffa24
                    nm_connection_get_id(connection),
Packit Service 5ffa24
                    nm_strerror(r));
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    return TRUE;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static gboolean
Packit Service 5ffa24
check_connection_compatible(NMDevice *device, NMConnection *connection, GError **error)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMDeviceVrfPrivate *priv = NM_DEVICE_VRF_GET_PRIVATE(device);
Packit Service 5ffa24
    NMSettingVrf *      s_vrf;
Packit Service 5ffa24
Packit Service 5ffa24
    if (!NM_DEVICE_CLASS(nm_device_vrf_parent_class)
Packit Service 5ffa24
             ->check_connection_compatible(device, connection, error))
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
Packit Service 5ffa24
    if (nm_device_is_real(device)) {
Packit Service 5ffa24
        s_vrf = _nm_connection_get_setting(connection, NM_TYPE_SETTING_VRF);
Packit Service 5ffa24
Packit Service 5ffa24
        if (priv->props.table != nm_setting_vrf_get_table(s_vrf)) {
Packit Service 5ffa24
            nm_utils_error_set_literal(error,
Packit Service 5ffa24
                                       NM_UTILS_ERROR_CONNECTION_AVAILABLE_TEMPORARY,
Packit Service 5ffa24
                                       "vrf table mismatches");
Packit Service 5ffa24
            return FALSE;
Packit Service 5ffa24
        }
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    return TRUE;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static gboolean
Packit Service 5ffa24
complete_connection(NMDevice *           device,
Packit Service 5ffa24
                    NMConnection *       connection,
Packit Service 5ffa24
                    const char *         specific_object,
Packit Service 5ffa24
                    NMConnection *const *existing_connections,
Packit Service 5ffa24
                    GError **            error)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMSettingVrf *s_vrf;
Packit Service 5ffa24
Packit Service 5ffa24
    nm_utils_complete_generic(nm_device_get_platform(device),
Packit Service 5ffa24
                              connection,
Packit Service 5ffa24
                              NM_SETTING_VRF_SETTING_NAME,
Packit Service 5ffa24
                              existing_connections,
Packit Service 5ffa24
                              NULL,
Packit Service 5ffa24
                              _("VRF connection"),
Packit Service 5ffa24
                              NULL,
Packit Service 5ffa24
                              NULL,
Packit Service 5ffa24
                              TRUE);
Packit Service 5ffa24
Packit Service 5ffa24
    s_vrf = _nm_connection_get_setting(connection, NM_TYPE_SETTING_VRF);
Packit Service 5ffa24
    if (!s_vrf) {
Packit Service 5ffa24
        g_set_error_literal(error,
Packit Service 5ffa24
                            NM_DEVICE_ERROR,
Packit Service 5ffa24
                            NM_DEVICE_ERROR_INVALID_CONNECTION,
Packit Service 5ffa24
                            "A 'vrf' setting is required.");
Packit Service 5ffa24
        return FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    return TRUE;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
update_connection(NMDevice *device, NMConnection *connection)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMDeviceVrfPrivate *priv  = NM_DEVICE_VRF_GET_PRIVATE(device);
Packit Service 5ffa24
    NMSettingVrf *      s_vrf = _nm_connection_get_setting(connection, NM_TYPE_SETTING_VRF);
Packit Service 5ffa24
Packit Service 5ffa24
    if (!s_vrf) {
Packit Service 5ffa24
        s_vrf = (NMSettingVrf *) nm_setting_vrf_new();
Packit Service 5ffa24
        nm_connection_add_setting(connection, (NMSetting *) s_vrf);
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    if (priv->props.table != nm_setting_vrf_get_table(s_vrf))
Packit Service 5ffa24
        g_object_set(G_OBJECT(s_vrf), NM_SETTING_VRF_TABLE, priv->props.table, NULL);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static gboolean
Packit Service 5ffa24
enslave_slave(NMDevice *device, NMDevice *slave, NMConnection *connection, gboolean configure)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMDeviceVrf *self        = NM_DEVICE_VRF(device);
Packit Service 5ffa24
    gboolean     success     = TRUE;
Packit Service 5ffa24
    const char * slave_iface = nm_device_get_ip_iface(slave);
Packit Service 5ffa24
Packit Service 5ffa24
    nm_device_master_check_slave_physical_port(device, slave, LOGD_DEVICE);
Packit Service 5ffa24
Packit Service 5ffa24
    if (configure) {
Packit Service 5ffa24
        nm_device_take_down(slave, TRUE);
Packit Service 5ffa24
        success = nm_platform_link_enslave(nm_device_get_platform(device),
Packit Service 5ffa24
                                           nm_device_get_ip_ifindex(device),
Packit Service 5ffa24
                                           nm_device_get_ip_ifindex(slave));
Packit Service 5ffa24
        nm_device_bring_up(slave, TRUE, NULL);
Packit Service 5ffa24
Packit Service 5ffa24
        if (!success)
Packit Service 5ffa24
            return FALSE;
Packit Service 5ffa24
Packit Service 5ffa24
        _LOGI(LOGD_DEVICE, "enslaved VRF slave %s", slave_iface);
Packit Service 5ffa24
    } else
Packit Service 5ffa24
        _LOGI(LOGD_BOND, "VRF slave %s was enslaved", slave_iface);
Packit Service 5ffa24
Packit Service 5ffa24
    return TRUE;
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
release_slave(NMDevice *device, NMDevice *slave, gboolean configure)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMDeviceVrf *self = NM_DEVICE_VRF(device);
Packit Service 5ffa24
    gboolean     success;
Packit Service 5ffa24
    int          ifindex_slave;
Packit Service 5ffa24
    int          ifindex;
Packit Service 5ffa24
Packit Service 5ffa24
    if (configure) {
Packit Service 5ffa24
        ifindex = nm_device_get_ifindex(device);
Packit Service 5ffa24
        if (ifindex <= 0 || !nm_platform_link_get(nm_device_get_platform(device), ifindex))
Packit Service 5ffa24
            configure = FALSE;
Packit Service 5ffa24
    }
Packit Service 5ffa24
Packit Service 5ffa24
    ifindex_slave = nm_device_get_ip_ifindex(slave);
Packit Service 5ffa24
Packit Service 5ffa24
    if (ifindex_slave <= 0)
Packit Service 5ffa24
        _LOGD(LOGD_DEVICE, "VRF slave %s is already released", nm_device_get_ip_iface(slave));
Packit Service 5ffa24
Packit Service 5ffa24
    if (configure) {
Packit Service 5ffa24
        if (ifindex_slave > 0) {
Packit Service 5ffa24
            success = nm_platform_link_release(nm_device_get_platform(device),
Packit Service 5ffa24
                                               nm_device_get_ip_ifindex(device),
Packit Service 5ffa24
                                               ifindex_slave);
Packit Service 5ffa24
Packit Service 5ffa24
            if (success) {
Packit Service 5ffa24
                _LOGI(LOGD_DEVICE, "released VRF slave %s", nm_device_get_ip_iface(slave));
Packit Service 5ffa24
            } else {
Packit Service 5ffa24
                _LOGW(LOGD_DEVICE, "failed to release VRF slave %s", nm_device_get_ip_iface(slave));
Packit Service 5ffa24
            }
Packit Service 5ffa24
        }
Packit Service 5ffa24
    } else {
Packit Service 5ffa24
        if (ifindex_slave > 0) {
Packit Service 5ffa24
            _LOGI(LOGD_DEVICE, "VRF slave %s was released", nm_device_get_ip_iface(slave));
Packit Service 5ffa24
        }
Packit Service 5ffa24
    }
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
get_property(GObject *object, guint prop_id, GValue *value, GParamSpec *pspec)
Packit Service 5ffa24
{
Packit Service 5ffa24
    NMDeviceVrfPrivate *priv = NM_DEVICE_VRF_GET_PRIVATE(object);
Packit Service 5ffa24
Packit Service 5ffa24
    switch (prop_id) {
Packit Service 5ffa24
    case PROP_TABLE:
Packit Service 5ffa24
        g_value_set_uint(value, priv->props.table);
Packit Service 5ffa24
        break;
Packit Service 5ffa24
    default:
Packit Service 5ffa24
        G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
Packit Service 5ffa24
        break;
Packit Service 5ffa24
    }
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
nm_device_vrf_init(NMDeviceVrf *self)
Packit Service 5ffa24
{}
Packit Service 5ffa24
Packit Service 5ffa24
static const NMDBusInterfaceInfoExtended interface_info_device_vrf = {
Packit Service 5ffa24
    .parent = NM_DEFINE_GDBUS_INTERFACE_INFO_INIT(
Packit Service 5ffa24
        NM_DBUS_INTERFACE_DEVICE_VRF,
Packit Service 5ffa24
        .properties = NM_DEFINE_GDBUS_PROPERTY_INFOS(
Packit Service 5ffa24
            NM_DEFINE_DBUS_PROPERTY_INFO_EXTENDED_READABLE("Table", "u", NM_DEVICE_VRF_TABLE), ), ),
Packit Service 5ffa24
};
Packit Service 5ffa24
Packit Service 5ffa24
static void
Packit Service 5ffa24
nm_device_vrf_class_init(NMDeviceVrfClass *klass)
Packit Service 5ffa24
{
Packit Service 5ffa24
    GObjectClass *     object_class      = G_OBJECT_CLASS(klass);
Packit Service 5ffa24
    NMDBusObjectClass *dbus_object_class = NM_DBUS_OBJECT_CLASS(klass);
Packit Service 5ffa24
    NMDeviceClass *    device_class      = NM_DEVICE_CLASS(klass);
Packit Service 5ffa24
Packit Service 5ffa24
    object_class->get_property = get_property;
Packit Service 5ffa24
Packit Service 5ffa24
    dbus_object_class->interface_infos = NM_DBUS_INTERFACE_INFOS(&interface_info_device_vrf);
Packit Service 5ffa24
Packit Service 5ffa24
    device_class->connection_type_supported        = NM_SETTING_VRF_SETTING_NAME;
Packit Service 5ffa24
    device_class->connection_type_check_compatible = NM_SETTING_VRF_SETTING_NAME;
Packit Service 5ffa24
    device_class->is_master                        = TRUE;
Packit Service 5ffa24
    device_class->link_types                       = NM_DEVICE_DEFINE_LINK_TYPES(NM_LINK_TYPE_VRF);
Packit Service 5ffa24
Packit Service 5ffa24
    device_class->enslave_slave               = enslave_slave;
Packit Service 5ffa24
    device_class->release_slave               = release_slave;
Packit Service 5ffa24
    device_class->link_changed                = link_changed;
Packit Service 5ffa24
    device_class->unrealize_notify            = unrealize_notify;
Packit Service 5ffa24
    device_class->create_and_realize          = create_and_realize;
Packit Service 5ffa24
    device_class->check_connection_compatible = check_connection_compatible;
Packit Service 5ffa24
    device_class->complete_connection         = complete_connection;
Packit Service 5ffa24
    device_class->get_generic_capabilities    = get_generic_capabilities;
Packit Service 5ffa24
    device_class->update_connection           = update_connection;
Packit Service 5ffa24
Packit Service 5ffa24
    obj_properties[PROP_TABLE] = g_param_spec_uint(NM_DEVICE_VRF_TABLE,
Packit Service 5ffa24
                                                   "",
Packit Service 5ffa24
                                                   "",
Packit Service 5ffa24
                                                   0,
Packit Service 5ffa24
                                                   G_MAXUINT32,
Packit Service 5ffa24
                                                   0,
Packit Service 5ffa24
                                                   G_PARAM_READABLE | G_PARAM_STATIC_STRINGS);
Packit Service 5ffa24
Packit Service 5ffa24
    g_object_class_install_properties(object_class, _PROPERTY_ENUMS_LAST, obj_properties);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
/*****************************************************************************/
Packit Service 5ffa24
Packit Service 5ffa24
#define NM_TYPE_VRF_DEVICE_FACTORY (nm_vrf_device_factory_get_type())
Packit Service 5ffa24
#define NM_VRF_DEVICE_FACTORY(obj) \
Packit Service 5ffa24
    (G_TYPE_CHECK_INSTANCE_CAST((obj), NM_TYPE_VRF_DEVICE_FACTORY, NMVrfDeviceFactory))
Packit Service 5ffa24
Packit Service 5ffa24
static NMDevice *
Packit Service 5ffa24
create_device(NMDeviceFactory *     factory,
Packit Service 5ffa24
              const char *          iface,
Packit Service 5ffa24
              const NMPlatformLink *plink,
Packit Service 5ffa24
              NMConnection *        connection,
Packit Service 5ffa24
              gboolean *            out_ignore)
Packit Service 5ffa24
{
Packit Service 5ffa24
    return g_object_new(NM_TYPE_DEVICE_VRF,
Packit Service 5ffa24
                        NM_DEVICE_IFACE,
Packit Service 5ffa24
                        iface,
Packit Service 5ffa24
                        NM_DEVICE_TYPE_DESC,
Packit Service 5ffa24
                        "Vrf",
Packit Service 5ffa24
                        NM_DEVICE_DEVICE_TYPE,
Packit Service 5ffa24
                        NM_DEVICE_TYPE_VRF,
Packit Service 5ffa24
                        NM_DEVICE_LINK_TYPE,
Packit Service 5ffa24
                        NM_LINK_TYPE_VRF,
Packit Service 5ffa24
                        NULL);
Packit Service 5ffa24
}
Packit Service 5ffa24
Packit Service 5ffa24
NM_DEVICE_FACTORY_DEFINE_INTERNAL(
Packit Service 5ffa24
    VRF,
Packit Service 5ffa24
    Vrf,
Packit Service 5ffa24
    vrf,
Packit Service 5ffa24
    NM_DEVICE_FACTORY_DECLARE_LINK_TYPES(NM_LINK_TYPE_VRF)
Packit Service 5ffa24
        NM_DEVICE_FACTORY_DECLARE_SETTING_TYPES(NM_SETTING_VRF_SETTING_NAME),
Packit Service 5ffa24
    factory_class->create_device = create_device;);