mirror of
https://gitlab.freedesktop.org/NetworkManager/NetworkManager.git
synced 2025-12-25 06:00:08 +01:00
Keep the include paths clean and separate. We use directories to group source files together. That makes sense (I guess), but then we should use this grouping also when including files. Thus require to #include files with their path relative to "src/". Also, we build various artifacts from the "src/" tree. Instead of having individual CFLAGS for each artifact in Makefile.am, the CFLAGS should be unified. Previously, the CFLAGS for each artifact differ and are inconsistent in which paths they add to the search path. Fix the inconsistency by just don't add the paths at all.
302 lines
8.9 KiB
C
302 lines
8.9 KiB
C
/* -*- Mode: C; tab-width: 4; indent-tabs-mode: t; c-basic-offset: 4 -*- */
|
|
/* NetworkManager -- Network link manager
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*
|
|
* Copyright (C) 2016 Red Hat, Inc.
|
|
*/
|
|
|
|
#include "nm-default.h"
|
|
|
|
#include "nm-checkpoint-manager.h"
|
|
|
|
#include "nm-checkpoint.h"
|
|
#include "nm-connection.h"
|
|
#include "nm-core-utils.h"
|
|
#include "devices/nm-device.h"
|
|
#include "nm-exported-object.h"
|
|
#include "nm-manager.h"
|
|
#include "nm-utils.h"
|
|
|
|
/*****************************************************************************/
|
|
|
|
struct _NMCheckpointManager {
|
|
NMManager *_manager;
|
|
GHashTable *checkpoints;
|
|
guint rollback_timeout_id;
|
|
};
|
|
|
|
#define GET_MANAGER(self) \
|
|
({ \
|
|
typeof (self) _self = (self); \
|
|
\
|
|
_nm_unused NMCheckpointManager *_self2 = _self; \
|
|
\
|
|
nm_assert (_self); \
|
|
nm_assert (NM_IS_MANAGER (_self->_manager)); \
|
|
_self->_manager; \
|
|
})
|
|
|
|
/*****************************************************************************/
|
|
|
|
#define _NMLOG_DOMAIN LOGD_CORE
|
|
#define _NMLOG(level, ...) __NMLOG_DEFAULT (level, _NMLOG_DOMAIN, "checkpoint", __VA_ARGS__)
|
|
|
|
/*****************************************************************************/
|
|
|
|
static void update_rollback_timeout (NMCheckpointManager *self);
|
|
|
|
static void
|
|
checkpoint_destroy (gpointer checkpoint)
|
|
{
|
|
nm_exported_object_unexport (NM_EXPORTED_OBJECT (checkpoint));
|
|
g_object_unref (G_OBJECT (checkpoint));
|
|
}
|
|
|
|
static gboolean
|
|
rollback_timeout_cb (NMCheckpointManager *self)
|
|
{
|
|
NMCheckpoint *checkpoint;
|
|
GHashTableIter iter;
|
|
GVariant *result;
|
|
gint64 ts, now;
|
|
|
|
now = nm_utils_get_monotonic_timestamp_ms ();
|
|
|
|
g_hash_table_iter_init (&iter, self->checkpoints);
|
|
while (g_hash_table_iter_next (&iter, NULL, (gpointer *) &checkpoint)) {
|
|
ts = nm_checkpoint_get_rollback_ts (checkpoint);
|
|
if (ts && ts <= now) {
|
|
result = nm_checkpoint_rollback (checkpoint);
|
|
if (result)
|
|
g_variant_unref (result);
|
|
g_hash_table_iter_remove (&iter);
|
|
}
|
|
}
|
|
|
|
self->rollback_timeout_id = 0;
|
|
update_rollback_timeout (self);
|
|
|
|
return G_SOURCE_REMOVE;
|
|
}
|
|
|
|
static void
|
|
update_rollback_timeout (NMCheckpointManager *self)
|
|
{
|
|
NMCheckpoint *checkpoint;
|
|
GHashTableIter iter;
|
|
gint64 ts, delta, next = G_MAXINT64;
|
|
|
|
g_hash_table_iter_init (&iter, self->checkpoints);
|
|
while (g_hash_table_iter_next (&iter, NULL, (gpointer *) &checkpoint)) {
|
|
ts = nm_checkpoint_get_rollback_ts (checkpoint);
|
|
if (ts && ts < next)
|
|
next = ts;
|
|
}
|
|
|
|
nm_clear_g_source (&self->rollback_timeout_id);
|
|
|
|
if (next != G_MAXINT64) {
|
|
delta = MAX (next - nm_utils_get_monotonic_timestamp_ms (), 0);
|
|
self->rollback_timeout_id = g_timeout_add (delta,
|
|
(GSourceFunc) rollback_timeout_cb,
|
|
self);
|
|
_LOGT ("update timeout: next check in %" G_GINT64_FORMAT " ms", delta);
|
|
}
|
|
}
|
|
|
|
static NMCheckpoint *
|
|
find_checkpoint_for_device (NMCheckpointManager *self, NMDevice *device)
|
|
{
|
|
GHashTableIter iter;
|
|
NMCheckpoint *checkpoint;
|
|
|
|
g_hash_table_iter_init (&iter, self->checkpoints);
|
|
while (g_hash_table_iter_next (&iter, NULL, (gpointer *) &checkpoint)) {
|
|
if (nm_checkpoint_includes_device (checkpoint, device))
|
|
return checkpoint;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
NMCheckpoint *
|
|
nm_checkpoint_manager_create (NMCheckpointManager *self,
|
|
const char *const *device_paths,
|
|
guint32 rollback_timeout,
|
|
NMCheckpointCreateFlags flags,
|
|
GError **error)
|
|
{
|
|
NMManager *manager;
|
|
NMCheckpoint *checkpoint;
|
|
const char * const *path;
|
|
gs_unref_ptrarray GPtrArray *devices = NULL;
|
|
NMDevice *device;
|
|
const char *checkpoint_path;
|
|
gs_free const char **device_paths_free = NULL;
|
|
guint i;
|
|
|
|
g_return_val_if_fail (self, FALSE);
|
|
g_return_val_if_fail (!error || !*error, FALSE);
|
|
manager = GET_MANAGER (self);
|
|
|
|
if (!device_paths || !device_paths[0]) {
|
|
device_paths_free = nm_manager_get_device_paths (manager);
|
|
device_paths = (const char *const *) device_paths_free;
|
|
} else if (NM_FLAGS_HAS (flags, NM_CHECKPOINT_CREATE_FLAG_DISCONNECT_NEW_DEVICES)) {
|
|
g_set_error_literal (error, NM_MANAGER_ERROR, NM_MANAGER_ERROR_INVALID_ARGUMENTS,
|
|
"the DISCONNECT_NEW_DEVICES flag can only be used with an empty device list");
|
|
return NULL;
|
|
}
|
|
|
|
devices = g_ptr_array_new ();
|
|
for (path = device_paths; *path; path++) {
|
|
device = nm_manager_get_device_by_path (manager, *path);
|
|
if (!device) {
|
|
g_set_error (error, NM_MANAGER_ERROR, NM_MANAGER_ERROR_UNKNOWN_DEVICE,
|
|
"device %s does not exist", *path);
|
|
return NULL;
|
|
}
|
|
g_ptr_array_add (devices, device);
|
|
}
|
|
|
|
if (!NM_FLAGS_HAS (flags, NM_CHECKPOINT_CREATE_FLAG_DESTROY_ALL)) {
|
|
for (i = 0; i < devices->len; i++) {
|
|
device = devices->pdata[i];
|
|
if (find_checkpoint_for_device (self, device)) {
|
|
g_set_error (error, NM_MANAGER_ERROR, NM_MANAGER_ERROR_INVALID_ARGUMENTS,
|
|
"a checkpoint for device '%s' already exists",
|
|
nm_device_get_iface (device));
|
|
return NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
checkpoint = nm_checkpoint_new (manager, devices, rollback_timeout, flags, error);
|
|
if (!checkpoint)
|
|
return NULL;
|
|
|
|
if (NM_FLAGS_HAS (flags, NM_CHECKPOINT_CREATE_FLAG_DESTROY_ALL))
|
|
g_hash_table_remove_all (self->checkpoints);
|
|
|
|
nm_exported_object_export (NM_EXPORTED_OBJECT (checkpoint));
|
|
checkpoint_path = nm_exported_object_get_path (NM_EXPORTED_OBJECT (checkpoint));
|
|
|
|
if (!nm_g_hash_table_insert (self->checkpoints,
|
|
(gpointer) checkpoint_path,
|
|
checkpoint))
|
|
g_return_val_if_reached (NULL);
|
|
|
|
update_rollback_timeout (self);
|
|
|
|
return checkpoint;
|
|
}
|
|
|
|
gboolean
|
|
nm_checkpoint_manager_destroy_all (NMCheckpointManager *self,
|
|
GError **error)
|
|
{
|
|
g_return_val_if_fail (self, FALSE);
|
|
|
|
g_hash_table_remove_all (self->checkpoints);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean
|
|
nm_checkpoint_manager_destroy (NMCheckpointManager *self,
|
|
const char *checkpoint_path,
|
|
GError **error)
|
|
{
|
|
gboolean ret;
|
|
|
|
g_return_val_if_fail (self, FALSE);
|
|
g_return_val_if_fail (checkpoint_path && checkpoint_path[0] == '/', FALSE);
|
|
g_return_val_if_fail (!error || !*error, FALSE);
|
|
|
|
if (!nm_streq (checkpoint_path, "/")) {
|
|
ret = g_hash_table_remove (self->checkpoints, checkpoint_path);
|
|
if (!ret) {
|
|
g_set_error (error,
|
|
NM_MANAGER_ERROR,
|
|
NM_MANAGER_ERROR_INVALID_ARGUMENTS,
|
|
"checkpoint %s does not exist", checkpoint_path);
|
|
}
|
|
return ret;
|
|
} else
|
|
return nm_checkpoint_manager_destroy_all (self, error);
|
|
}
|
|
|
|
gboolean
|
|
nm_checkpoint_manager_rollback (NMCheckpointManager *self,
|
|
const char *checkpoint_path,
|
|
GVariant **results,
|
|
GError **error)
|
|
{
|
|
NMCheckpoint *cp;
|
|
|
|
g_return_val_if_fail (self, FALSE);
|
|
g_return_val_if_fail (checkpoint_path && checkpoint_path[0] == '/', FALSE);
|
|
g_return_val_if_fail (results, FALSE);
|
|
g_return_val_if_fail (!error || !*error, FALSE);
|
|
|
|
cp = g_hash_table_lookup (self->checkpoints, checkpoint_path);
|
|
if (!cp) {
|
|
g_set_error (error, NM_MANAGER_ERROR, NM_MANAGER_ERROR_FAILED,
|
|
"checkpoint %s does not exist", checkpoint_path);
|
|
return FALSE;
|
|
}
|
|
|
|
*results = nm_checkpoint_rollback (cp);
|
|
g_hash_table_remove (self->checkpoints, checkpoint_path);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
|
|
NMCheckpointManager *
|
|
nm_checkpoint_manager_new (NMManager *manager)
|
|
{
|
|
NMCheckpointManager *self;
|
|
|
|
g_return_val_if_fail (NM_IS_MANAGER (manager), FALSE);
|
|
|
|
self = g_slice_new0 (NMCheckpointManager);
|
|
|
|
/* the NMCheckpointManager instance is actually owned by NMManager.
|
|
* Thus, we cannot take a reference to it, and we also don't bother
|
|
* taking a weak-reference. Instead let GET_MANAGER() assert that
|
|
* self->_manager is alive -- which we always expect as the lifetime
|
|
* of NMManager shall surpass the lifetime of the NMCheckpointManager
|
|
* instance. */
|
|
self->_manager = manager;
|
|
self->checkpoints = g_hash_table_new_full (g_str_hash, g_str_equal,
|
|
NULL, checkpoint_destroy);
|
|
|
|
return self;
|
|
}
|
|
|
|
void
|
|
nm_checkpoint_manager_unref (NMCheckpointManager *self)
|
|
{
|
|
if (!self)
|
|
return;
|
|
|
|
nm_clear_g_source (&self->rollback_timeout_id);
|
|
g_hash_table_destroy (self->checkpoints);
|
|
|
|
g_slice_free (NMCheckpointManager, self);
|
|
}
|