/* SPDX-License-Identifier: LGPL-2.1-or-later */
/*
* Copyright (C) 2018 Red Hat, Inc.
*/
#include "src/core/nm-default-daemon.h"
#include "nm-settings-storage.h"
#include "nm-utils.h"
#include "nm-settings-plugin.h"
#include "settings/plugins/keyfile/nms-keyfile-storage.h"
/*****************************************************************************/
int
nm_settings_storage_cmp(NMSettingsStorage *a, NMSettingsStorage *b, const GSList *plugin_list)
{
NMSettingsStorageClass *klass;
/* Sort by priority.
*
* If a > b (by priority), we return a positive number (as one
* would expect by a cmp() function). */
nm_assert(NM_IS_SETTINGS_STORAGE(a));
nm_assert(NM_IS_SETTINGS_STORAGE(b));
nm_assert(a != b);
nm_assert(nm_streq(nm_settings_storage_get_uuid(a), nm_settings_storage_get_uuid(b)));
/* in-memory has always higher priority */
NM_CMP_DIRECT(nm_settings_storage_is_keyfile_run(a), nm_settings_storage_is_keyfile_run(b));
NM_CMP_RETURN(nm_settings_plugin_cmp_by_priority(nm_settings_storage_get_plugin(a),
nm_settings_storage_get_plugin(b),
plugin_list));
klass = NM_SETTINGS_STORAGE_GET_CLASS(a);
if (klass != NM_SETTINGS_STORAGE_GET_CLASS(b)) {
/* one plugin must return storages of the same type. Otherwise, it's
* unclear how cmp_fcn() should compare them. */
nm_assert_not_reached();
return 0;
}
if (klass->cmp_fcn)
NM_CMP_RETURN(klass->cmp_fcn(a, b));
return 0;
}
/*****************************************************************************/
NM_GOBJECT_PROPERTIES_DEFINE_BASE(PROP_PLUGIN, PROP_UUID, PROP_FILENAME, );
G_DEFINE_TYPE(NMSettingsStorage, nm_settings_storage, G_TYPE_OBJECT)
/*****************************************************************************/
static void
set_property(GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec)
{
NMSettingsStorage *self = NM_SETTINGS_STORAGE(object);
switch (prop_id) {
case PROP_PLUGIN:
/* construct-only */
self->_plugin = g_object_ref(g_value_get_object(value));
nm_assert(NM_IS_SETTINGS_PLUGIN(self->_plugin));
break;
case PROP_UUID:
/* construct-only */
self->_uuid = g_value_dup_string(value);
nm_assert(!self->_uuid || nm_utils_is_uuid(self->_uuid));
break;
case PROP_FILENAME:
/* construct-only */
self->_filename = g_value_dup_string(value);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
break;
}
}
/*****************************************************************************/
static void
nm_settings_storage_init(NMSettingsStorage *self)
{
c_list_init(&self->_storage_lst);
c_list_init(&self->_storage_by_uuid_lst);
}
NMSettingsStorage *
nm_settings_storage_new(NMSettingsPlugin *plugin, const char *uuid, const char *filename)
{
nm_assert(NM_IS_SETTINGS_PLUGIN(plugin));
nm_assert(nm_utils_is_uuid(uuid));
return g_object_new(NM_TYPE_SETTINGS_STORAGE,
NM_SETTINGS_STORAGE_PLUGIN,
plugin,
NM_SETTINGS_STORAGE_UUID,
uuid,
NM_SETTINGS_STORAGE_FILENAME,
filename,
NULL);
}
static void
finalize(GObject *object)
{
NMSettingsStorage *self = NM_SETTINGS_STORAGE(object);
c_list_unlink_stale(&self->_storage_lst);
c_list_unlink_stale(&self->_storage_by_uuid_lst);
g_object_unref(self->_plugin);
g_free(self->_uuid);
g_free(self->_filename);
G_OBJECT_CLASS(nm_settings_storage_parent_class)->finalize(object);
}
static void
nm_settings_storage_class_init(NMSettingsStorageClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS(klass);
object_class->set_property = set_property;
object_class->finalize = finalize;
obj_properties[PROP_PLUGIN] =
g_param_spec_object(NM_SETTINGS_STORAGE_PLUGIN,
"",
"",
NM_TYPE_SETTINGS_PLUGIN,
G_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS);
obj_properties[PROP_UUID] =
g_param_spec_string(NM_SETTINGS_STORAGE_UUID,
"",
"",
NULL,
G_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS);
obj_properties[PROP_FILENAME] =
g_param_spec_string(NM_SETTINGS_STORAGE_FILENAME,
"",
"",
NULL,
G_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS);
g_object_class_install_properties(object_class, _PROPERTY_ENUMS_LAST, obj_properties);
}