mirror of
https://gitlab.freedesktop.org/NetworkManager/NetworkManager.git
synced 2026-01-07 21:50:18 +01:00
When a global checkpoint is created (one with empty device list) we save the status of all devices to restore it later. After the checkpoint new interfaces and connections may appear and they can significantly influence the overall networking status, but we don't consider them at the moment. Introduce a new flag DELETE_NEW_CONNECTIONS to delete any connection added after the checkpoint and similarly a DISCONNECT_NEW_DEVICES to ensure that the connection active on newly appeared devices doesn't disrupt network connectivity. https://bugzilla.redhat.com/show_bug.cgi?id=1378393
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 "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);
|
|
}
|