// SPDX-License-Identifier: LGPL-2.1+ /* * Copyright 2017 Red Hat, Inc. */ #include "nm-default.h" #include "nm-setting-ovs-interface.h" #include "nm-connection-private.h" #include "nm-setting-connection.h" #include "nm-setting-private.h" /** * SECTION:nm-setting-ovs-interface * @short_description: Describes connection properties for Open vSwitch interfaces. * * The #NMSettingOvsInterface object is a #NMSetting subclass that describes properties * necessary for Open vSwitch interfaces. **/ /*****************************************************************************/ NM_GOBJECT_PROPERTIES_DEFINE_BASE ( PROP_TYPE, ); /** * NMSettingOvsInterface: * * Open vSwitch Interface Settings */ struct _NMSettingOvsInterface { NMSetting parent; char *type; }; struct _NMSettingOvsInterfaceClass { NMSettingClass parent; }; G_DEFINE_TYPE (NMSettingOvsInterface, nm_setting_ovs_interface, NM_TYPE_SETTING) /*****************************************************************************/ /** * nm_setting_ovs_interface_get_interface_type: * @self: the #NMSettingOvsInterface * * Returns: the #NMSettingOvsInterface:type property of the setting * * Since: 1.10 **/ const char * nm_setting_ovs_interface_get_interface_type (NMSettingOvsInterface *self) { g_return_val_if_fail (NM_IS_SETTING_OVS_INTERFACE (self), NULL); return self->type; } /*****************************************************************************/ int _nm_setting_ovs_interface_verify_interface_type (NMSettingOvsInterface *self, NMConnection *connection, gboolean normalize, gboolean *out_modified, GError **error) { const char *type; const char *type_from_setting = NULL; const char *type_setting = NULL; const char *connection_type; gboolean is_ovs_connection_type; g_return_val_if_fail (NM_IS_SETTING_OVS_INTERFACE (self), FALSE); if (normalize) { g_return_val_if_fail (NM_IS_CONNECTION (connection), FALSE); nm_assert (self == nm_connection_get_setting_ovs_interface (connection)); } else g_return_val_if_fail (!connection || NM_IS_CONNECTION (connection), FALSE); NM_SET_OUT (out_modified, FALSE); type = self ? self->type : NULL; if ( type && !NM_IN_STRSET (type, "internal", "system", "patch", "dpdk")) { g_set_error (error, NM_CONNECTION_ERROR, NM_CONNECTION_ERROR_INVALID_PROPERTY, _("'%s' is not a valid interface type"), type); g_prefix_error (error, "%s.%s: ", NM_SETTING_OVS_INTERFACE_SETTING_NAME, NM_SETTING_OVS_INTERFACE_TYPE); return FALSE; } if (!connection) return TRUE; connection_type = nm_connection_get_connection_type (connection); if (!connection_type) { /* if we have an ovs-interface, then the connection type must be either * "ovs-interface" (for non "system" type) or anything else (for "system" type). * * The connection type usually can be normalized based on the presence of a * base setting. However, in this case, if the connection type is missing, * that is too complicate to guess what the user wanted. * * Require the use to be explicit and fail. */ g_set_error (error, NM_CONNECTION_ERROR, NM_CONNECTION_ERROR_INVALID_PROPERTY, _("A connection with a '%s' setting needs connection.type explicitly set"), NM_SETTING_OVS_INTERFACE_SETTING_NAME); g_prefix_error (error, "%s.%s: ", NM_SETTING_CONNECTION_SETTING_NAME, NM_SETTING_CONNECTION_TYPE); return FALSE; } if (nm_streq (connection_type, NM_SETTING_OVS_INTERFACE_SETTING_NAME)) { if ( type && nm_streq (type, "system")) { g_set_error (error, NM_CONNECTION_ERROR, NM_CONNECTION_ERROR_INVALID_PROPERTY, _("A connection of type '%s' cannot have ovs-interface.type \"system\""), NM_SETTING_OVS_INTERFACE_SETTING_NAME); g_prefix_error (error, "%s.%s: ", NM_SETTING_OVS_INTERFACE_SETTING_NAME, NM_SETTING_OVS_INTERFACE_TYPE); return FALSE; } is_ovs_connection_type = TRUE; } else { if ( type && !nm_streq (type, "system")) { g_set_error (error, NM_CONNECTION_ERROR, NM_CONNECTION_ERROR_INVALID_PROPERTY, _("A connection of type '%s' cannot have an ovs-interface.type \"%s\""), connection_type, type); g_prefix_error (error, "%s.%s: ", NM_SETTING_OVS_INTERFACE_SETTING_NAME, NM_SETTING_OVS_INTERFACE_TYPE); return FALSE; } is_ovs_connection_type = FALSE; } if (nm_connection_get_setting_by_name (connection, NM_SETTING_OVS_PATCH_SETTING_NAME)) { type_from_setting = "patch"; type_setting = NM_SETTING_OVS_PATCH_SETTING_NAME; } if (nm_connection_get_setting_by_name (connection, NM_SETTING_OVS_DPDK_SETTING_NAME)) { if (type_from_setting) { g_set_error (error, NM_CONNECTION_ERROR, NM_CONNECTION_ERROR_INVALID_PROPERTY, _("A connection can not have both '%s' and '%s' settings at the same time"), NM_SETTING_OVS_DPDK_SETTING_NAME, type_setting); return FALSE; } type_from_setting = "dpdk"; type_setting = NM_SETTING_OVS_DPDK_SETTING_NAME; } if (type_from_setting) { if (!is_ovs_connection_type) { g_set_error (error, NM_CONNECTION_ERROR, NM_CONNECTION_ERROR_INVALID_PROPERTY, _("A connection with '%s' setting must be of connection.type \"ovs-interface\" but is \"%s\""), NM_SETTING_OVS_PATCH_SETTING_NAME, connection_type); g_prefix_error (error, "%s.%s: ", NM_SETTING_OVS_INTERFACE_SETTING_NAME, NM_SETTING_OVS_INTERFACE_TYPE); return FALSE; } if (type) { if (!nm_streq (type, type_from_setting)) { g_set_error (error, NM_CONNECTION_ERROR, NM_CONNECTION_ERROR_INVALID_PROPERTY, _("A connection with '%s' setting needs to be of '%s' interface type, not '%s'"), type_setting, type_from_setting, type); g_prefix_error (error, "%s.%s: ", NM_SETTING_OVS_INTERFACE_SETTING_NAME, NM_SETTING_OVS_INTERFACE_TYPE); return FALSE; } return TRUE; } type = type_from_setting; goto normalize; } else { if (nm_streq0 (type, "patch")) { g_set_error (error, NM_CONNECTION_ERROR, NM_CONNECTION_ERROR_MISSING_SETTING, _("A connection with ovs-interface.type '%s' setting a 'ovs-patch' setting"), type); g_prefix_error (error, "%s.%s: ", NM_SETTING_OVS_INTERFACE_SETTING_NAME, NM_SETTING_OVS_INTERFACE_TYPE); return FALSE; } } if (type) return TRUE; if (is_ovs_connection_type) type = "internal"; else type = "system"; normalize: if (!normalize) { if (!self) { g_set_error (error, NM_CONNECTION_ERROR, NM_CONNECTION_ERROR_MISSING_SETTING, _("Missing ovs interface setting")); g_prefix_error (error, "%s: ", NM_SETTING_OVS_INTERFACE_SETTING_NAME); } else { g_set_error (error, NM_CONNECTION_ERROR, NM_CONNECTION_ERROR_MISSING_PROPERTY, _("Missing ovs interface type")); g_prefix_error (error, "%s.%s: ", NM_SETTING_OVS_INTERFACE_SETTING_NAME, NM_SETTING_OVS_INTERFACE_TYPE); } return NM_SETTING_VERIFY_NORMALIZABLE_ERROR; } if (!self) { self = NM_SETTING_OVS_INTERFACE (nm_setting_ovs_interface_new ()); nm_connection_add_setting (connection, NM_SETTING (self)); } g_object_set (self, NM_SETTING_OVS_INTERFACE_TYPE, type, NULL); NM_SET_OUT (out_modified, TRUE); return TRUE; } static int verify (NMSetting *setting, NMConnection *connection, GError **error) { NMSettingOvsInterface *self = NM_SETTING_OVS_INTERFACE (setting); if (connection) { NMSettingConnection *s_con; const char *slave_type; s_con = nm_connection_get_setting_connection (connection); if (!s_con) { g_set_error (error, NM_CONNECTION_ERROR, NM_CONNECTION_ERROR_MISSING_SETTING, _("missing setting")); g_prefix_error (error, "%s: ", NM_SETTING_CONNECTION_SETTING_NAME); return FALSE; } if (!nm_setting_connection_get_master (s_con)) { g_set_error (error, NM_CONNECTION_ERROR, NM_CONNECTION_ERROR_INVALID_PROPERTY, _("A connection with a '%s' setting must have a master."), NM_SETTING_OVS_INTERFACE_SETTING_NAME); g_prefix_error (error, "%s.%s: ", NM_SETTING_CONNECTION_SETTING_NAME, NM_SETTING_CONNECTION_MASTER); return FALSE; } slave_type = nm_setting_connection_get_slave_type (s_con); if ( slave_type && !nm_streq (slave_type, NM_SETTING_OVS_PORT_SETTING_NAME)) { g_set_error (error, NM_CONNECTION_ERROR, NM_CONNECTION_ERROR_INVALID_PROPERTY, _("A connection with a '%s' setting must have the slave-type set to '%s'. Instead it is '%s'"), NM_SETTING_OVS_INTERFACE_SETTING_NAME, NM_SETTING_OVS_PORT_SETTING_NAME, slave_type); g_prefix_error (error, "%s.%s: ", NM_SETTING_CONNECTION_SETTING_NAME, NM_SETTING_CONNECTION_SLAVE_TYPE); return FALSE; } } return _nm_setting_ovs_interface_verify_interface_type (self, connection, FALSE, NULL, error); } /*****************************************************************************/ static void get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { NMSettingOvsInterface *self = NM_SETTING_OVS_INTERFACE (object); switch (prop_id) { case PROP_TYPE: g_value_set_string (value, self->type); 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) { NMSettingOvsInterface *self = NM_SETTING_OVS_INTERFACE (object); switch (prop_id) { case PROP_TYPE: g_free (self->type); self->type = g_value_dup_string (value); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } /*****************************************************************************/ static void nm_setting_ovs_interface_init (NMSettingOvsInterface *self) { } /** * nm_setting_ovs_interface_new: * * Creates a new #NMSettingOvsInterface object with default values. * * Returns: (transfer full): the new empty #NMSettingOvsInterface object * * Since: 1.10 **/ NMSetting * nm_setting_ovs_interface_new (void) { return (NMSetting *) g_object_new (NM_TYPE_SETTING_OVS_INTERFACE, NULL); } static void finalize (GObject *object) { NMSettingOvsInterface *self = NM_SETTING_OVS_INTERFACE (object); g_free (self->type); G_OBJECT_CLASS (nm_setting_ovs_interface_parent_class)->finalize (object); } static void nm_setting_ovs_interface_class_init (NMSettingOvsInterfaceClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); NMSettingClass *setting_class = NM_SETTING_CLASS (klass); object_class->get_property = get_property; object_class->set_property = set_property; object_class->finalize = finalize; setting_class->verify = verify; /** * NMSettingOvsInterface:type: * * The interface type. Either "internal", "system", "patch", "dpdk", or empty. * * Since: 1.10 **/ obj_properties[PROP_TYPE] = g_param_spec_string (NM_SETTING_OVS_INTERFACE_TYPE, "", "", NULL, G_PARAM_READWRITE | G_PARAM_CONSTRUCT | NM_SETTING_PARAM_INFERRABLE | G_PARAM_STATIC_STRINGS); g_object_class_install_properties (object_class, _PROPERTY_ENUMS_LAST, obj_properties); _nm_setting_class_commit (setting_class, NM_META_SETTING_TYPE_OVS_INTERFACE); }