2020-10-29 20:05:43 +03:00
|
|
|
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
|
|
From: Stefan Reiter <s.reiter@proxmox.com>
|
|
|
|
Date: Thu, 22 Oct 2020 17:34:18 +0200
|
|
|
|
Subject: [PATCH] PVE: Migrate dirty bitmap state via savevm
|
|
|
|
|
|
|
|
QEMU provides 'savevm' registrations as a mechanism for arbitrary state
|
|
|
|
to be migrated along with a VM. Use this to send a serialized version of
|
|
|
|
dirty bitmap state data from proxmox-backup-qemu, and restore it on the
|
|
|
|
target node.
|
|
|
|
|
|
|
|
Also add a flag to query-proxmox-support so qemu-server can determine if
|
|
|
|
safe migration is possible and makes sense.
|
|
|
|
|
|
|
|
Signed-off-by: Stefan Reiter <s.reiter@proxmox.com>
|
2022-01-13 12:34:33 +03:00
|
|
|
Signed-off-by: Thomas Lamprecht <t.lamprecht@proxmox.com>
|
2020-10-29 20:05:43 +03:00
|
|
|
---
|
2020-11-10 19:16:15 +03:00
|
|
|
include/migration/misc.h | 3 ++
|
2021-02-11 19:11:11 +03:00
|
|
|
migration/meson.build | 2 +
|
2021-05-27 13:43:32 +03:00
|
|
|
migration/migration.c | 1 +
|
2020-11-10 19:16:15 +03:00
|
|
|
migration/pbs-state.c | 106 +++++++++++++++++++++++++++++++++++++++
|
|
|
|
pve-backup.c | 1 +
|
|
|
|
qapi/block-core.json | 6 +++
|
2021-02-11 19:11:11 +03:00
|
|
|
6 files changed, 119 insertions(+)
|
2020-10-29 20:05:43 +03:00
|
|
|
create mode 100644 migration/pbs-state.c
|
|
|
|
|
|
|
|
diff --git a/include/migration/misc.h b/include/migration/misc.h
|
2021-10-11 14:55:34 +03:00
|
|
|
index 465906710d..4f0aeceb6f 100644
|
2020-10-29 20:05:43 +03:00
|
|
|
--- a/include/migration/misc.h
|
|
|
|
+++ b/include/migration/misc.h
|
2021-10-11 14:55:34 +03:00
|
|
|
@@ -75,4 +75,7 @@ bool migration_in_bg_snapshot(void);
|
2020-10-29 20:05:43 +03:00
|
|
|
/* migration/block-dirty-bitmap.c */
|
|
|
|
void dirty_bitmap_mig_init(void);
|
|
|
|
|
|
|
|
+/* migration/pbs-state.c */
|
|
|
|
+void pbs_state_mig_init(void);
|
|
|
|
+
|
|
|
|
#endif
|
2021-02-11 19:11:11 +03:00
|
|
|
diff --git a/migration/meson.build b/migration/meson.build
|
2021-10-11 14:55:34 +03:00
|
|
|
index ea9aedeefc..c27dc9bd97 100644
|
2021-02-11 19:11:11 +03:00
|
|
|
--- a/migration/meson.build
|
|
|
|
+++ b/migration/meson.build
|
|
|
|
@@ -7,8 +7,10 @@ migration_files = files(
|
|
|
|
'qemu-file-channel.c',
|
|
|
|
'qemu-file.c',
|
2021-05-27 13:43:32 +03:00
|
|
|
'yank_functions.c',
|
2021-02-11 19:11:11 +03:00
|
|
|
+ 'pbs-state.c',
|
|
|
|
)
|
|
|
|
softmmu_ss.add(migration_files)
|
|
|
|
+softmmu_ss.add(libproxmox_backup_qemu)
|
2020-10-29 20:05:43 +03:00
|
|
|
|
2021-02-11 19:11:11 +03:00
|
|
|
softmmu_ss.add(files(
|
|
|
|
'block-dirty-bitmap.c',
|
2021-05-27 13:43:32 +03:00
|
|
|
diff --git a/migration/migration.c b/migration/migration.c
|
2022-02-11 12:24:33 +03:00
|
|
|
index abaf6f9e3d..d925fd7488 100644
|
2021-05-27 13:43:32 +03:00
|
|
|
--- a/migration/migration.c
|
|
|
|
+++ b/migration/migration.c
|
2022-02-11 12:24:33 +03:00
|
|
|
@@ -213,6 +213,7 @@ void migration_object_init(void)
|
2021-05-27 13:43:32 +03:00
|
|
|
blk_mig_init();
|
|
|
|
ram_mig_init();
|
|
|
|
dirty_bitmap_mig_init();
|
|
|
|
+ pbs_state_mig_init();
|
|
|
|
}
|
|
|
|
|
2022-02-11 12:24:33 +03:00
|
|
|
void migration_cancel(const Error *error)
|
2020-10-29 20:05:43 +03:00
|
|
|
diff --git a/migration/pbs-state.c b/migration/pbs-state.c
|
|
|
|
new file mode 100644
|
2020-11-10 19:16:15 +03:00
|
|
|
index 0000000000..29f2b3860d
|
2020-10-29 20:05:43 +03:00
|
|
|
--- /dev/null
|
|
|
|
+++ b/migration/pbs-state.c
|
2020-11-10 19:16:15 +03:00
|
|
|
@@ -0,0 +1,106 @@
|
2020-10-29 20:05:43 +03:00
|
|
|
+/*
|
|
|
|
+ * PBS (dirty-bitmap) state migration
|
|
|
|
+ */
|
|
|
|
+
|
|
|
|
+#include "qemu/osdep.h"
|
|
|
|
+#include "migration/misc.h"
|
|
|
|
+#include "qemu-file.h"
|
|
|
|
+#include "migration/vmstate.h"
|
|
|
|
+#include "migration/register.h"
|
|
|
|
+#include "proxmox-backup-qemu.h"
|
|
|
|
+
|
fix dirty-bitmap state migration freeze
The idea in general is to migrate all the state, which is small for
us, in a single step once. But, QEMU only calls save state if we
return active true.
Hardcoding is-active to return true, like done initially, makes the
migration freeze, as QEMU thinks this is never done, and only stops
calling us and finishes after a few seconds.
So, add a state with an "active" boolean, set to true when
initializing a migration, and set it to false when the state was
saved.
Signed-off-by: Thomas Lamprecht <t.lamprecht@proxmox.com>
2020-11-05 20:43:01 +03:00
|
|
|
+typedef struct PBSState {
|
|
|
|
+ bool active;
|
|
|
|
+} PBSState;
|
|
|
|
+
|
2020-11-10 19:16:15 +03:00
|
|
|
+/* state is accessed via this static variable directly, 'opaque' is NULL */
|
fix dirty-bitmap state migration freeze
The idea in general is to migrate all the state, which is small for
us, in a single step once. But, QEMU only calls save state if we
return active true.
Hardcoding is-active to return true, like done initially, makes the
migration freeze, as QEMU thinks this is never done, and only stops
calling us and finishes after a few seconds.
So, add a state with an "active" boolean, set to true when
initializing a migration, and set it to false when the state was
saved.
Signed-off-by: Thomas Lamprecht <t.lamprecht@proxmox.com>
2020-11-05 20:43:01 +03:00
|
|
|
+static PBSState pbs_state;
|
|
|
|
+
|
2020-10-29 20:05:43 +03:00
|
|
|
+static void pbs_state_save_pending(QEMUFile *f, void *opaque,
|
|
|
|
+ uint64_t max_size,
|
|
|
|
+ uint64_t *res_precopy_only,
|
|
|
|
+ uint64_t *res_compatible,
|
|
|
|
+ uint64_t *res_postcopy_only)
|
|
|
|
+{
|
|
|
|
+ /* we send everything in save_setup, so nothing is ever pending */
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/* receive PBS state via f and deserialize, called on target */
|
|
|
|
+static int pbs_state_load(QEMUFile *f, void *opaque, int version_id)
|
|
|
|
+{
|
|
|
|
+ /* safe cast, we cannot migrate to target with less bits than source */
|
|
|
|
+ size_t buf_size = (size_t)qemu_get_be64(f);
|
|
|
|
+
|
|
|
|
+ uint8_t *buf = (uint8_t *)malloc(buf_size);
|
|
|
|
+ size_t read = qemu_get_buffer(f, buf, buf_size);
|
|
|
|
+
|
|
|
|
+ if (read < buf_size) {
|
|
|
|
+ fprintf(stderr, "error receiving PBS state: not enough data\n");
|
|
|
|
+ return -EIO;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ proxmox_import_state(buf, buf_size);
|
|
|
|
+
|
|
|
|
+ free(buf);
|
|
|
|
+ return 0;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/* serialize PBS state and send to target via f, called on source */
|
|
|
|
+static int pbs_state_save_setup(QEMUFile *f, void *opaque)
|
|
|
|
+{
|
|
|
|
+ size_t buf_size;
|
|
|
|
+ uint8_t *buf = proxmox_export_state(&buf_size);
|
|
|
|
+
|
|
|
|
+ /* LV encoding */
|
|
|
|
+ qemu_put_be64(f, buf_size);
|
|
|
|
+ qemu_put_buffer(f, buf, buf_size);
|
|
|
|
+
|
|
|
|
+ proxmox_free_state_buf(buf);
|
fix dirty-bitmap state migration freeze
The idea in general is to migrate all the state, which is small for
us, in a single step once. But, QEMU only calls save state if we
return active true.
Hardcoding is-active to return true, like done initially, makes the
migration freeze, as QEMU thinks this is never done, and only stops
calling us and finishes after a few seconds.
So, add a state with an "active" boolean, set to true when
initializing a migration, and set it to false when the state was
saved.
Signed-off-by: Thomas Lamprecht <t.lamprecht@proxmox.com>
2020-11-05 20:43:01 +03:00
|
|
|
+ pbs_state.active = false;
|
2020-10-29 20:05:43 +03:00
|
|
|
+ return 0;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static bool pbs_state_is_active(void *opaque)
|
|
|
|
+{
|
2020-11-10 19:16:15 +03:00
|
|
|
+ /* we need to return active exactly once, else .save_setup is never called,
|
|
|
|
+ * but if we'd just return true the migration doesn't make progress since
|
|
|
|
+ * it'd be waiting for us */
|
fix dirty-bitmap state migration freeze
The idea in general is to migrate all the state, which is small for
us, in a single step once. But, QEMU only calls save state if we
return active true.
Hardcoding is-active to return true, like done initially, makes the
migration freeze, as QEMU thinks this is never done, and only stops
calling us and finishes after a few seconds.
So, add a state with an "active" boolean, set to true when
initializing a migration, and set it to false when the state was
saved.
Signed-off-by: Thomas Lamprecht <t.lamprecht@proxmox.com>
2020-11-05 20:43:01 +03:00
|
|
|
+ return pbs_state.active;
|
2020-10-29 20:05:43 +03:00
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static bool pbs_state_is_active_iterate(void *opaque)
|
|
|
|
+{
|
|
|
|
+ /* we don't iterate, everything is sent in save_setup */
|
fix dirty-bitmap state migration freeze
The idea in general is to migrate all the state, which is small for
us, in a single step once. But, QEMU only calls save state if we
return active true.
Hardcoding is-active to return true, like done initially, makes the
migration freeze, as QEMU thinks this is never done, and only stops
calling us and finishes after a few seconds.
So, add a state with an "active" boolean, set to true when
initializing a migration, and set it to false when the state was
saved.
Signed-off-by: Thomas Lamprecht <t.lamprecht@proxmox.com>
2020-11-05 20:43:01 +03:00
|
|
|
+ return pbs_state_is_active(opaque);
|
2020-10-29 20:05:43 +03:00
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static bool pbs_state_has_postcopy(void *opaque)
|
|
|
|
+{
|
|
|
|
+ /* PBS state can't change during a migration (since that's blocking any
|
|
|
|
+ * potential backups), so we can copy everything before the VM is stopped */
|
|
|
|
+ return false;
|
|
|
|
+}
|
|
|
|
+
|
2020-11-10 19:16:15 +03:00
|
|
|
+static void pbs_state_save_cleanup(void *opaque)
|
|
|
|
+{
|
|
|
|
+ /* reset active after migration succeeds or fails */
|
|
|
|
+ pbs_state.active = false;
|
|
|
|
+}
|
|
|
|
+
|
2020-10-29 20:05:43 +03:00
|
|
|
+static SaveVMHandlers savevm_pbs_state_handlers = {
|
|
|
|
+ .save_setup = pbs_state_save_setup,
|
|
|
|
+ .has_postcopy = pbs_state_has_postcopy,
|
|
|
|
+ .save_live_pending = pbs_state_save_pending,
|
|
|
|
+ .is_active_iterate = pbs_state_is_active_iterate,
|
|
|
|
+ .load_state = pbs_state_load,
|
|
|
|
+ .is_active = pbs_state_is_active,
|
2020-11-10 19:16:15 +03:00
|
|
|
+ .save_cleanup = pbs_state_save_cleanup,
|
2020-10-29 20:05:43 +03:00
|
|
|
+};
|
|
|
|
+
|
|
|
|
+void pbs_state_mig_init(void)
|
|
|
|
+{
|
fix dirty-bitmap state migration freeze
The idea in general is to migrate all the state, which is small for
us, in a single step once. But, QEMU only calls save state if we
return active true.
Hardcoding is-active to return true, like done initially, makes the
migration freeze, as QEMU thinks this is never done, and only stops
calling us and finishes after a few seconds.
So, add a state with an "active" boolean, set to true when
initializing a migration, and set it to false when the state was
saved.
Signed-off-by: Thomas Lamprecht <t.lamprecht@proxmox.com>
2020-11-05 20:43:01 +03:00
|
|
|
+ pbs_state.active = true;
|
2020-10-29 20:05:43 +03:00
|
|
|
+ register_savevm_live("pbs-state", 0, 1,
|
|
|
|
+ &savevm_pbs_state_handlers,
|
2020-11-10 19:16:15 +03:00
|
|
|
+ NULL);
|
2020-10-29 20:05:43 +03:00
|
|
|
+}
|
|
|
|
diff --git a/pve-backup.c b/pve-backup.c
|
2022-02-11 12:24:33 +03:00
|
|
|
index 6f05796fad..5fa3cc1352 100644
|
2020-10-29 20:05:43 +03:00
|
|
|
--- a/pve-backup.c
|
|
|
|
+++ b/pve-backup.c
|
2021-05-27 13:43:32 +03:00
|
|
|
@@ -1132,6 +1132,7 @@ ProxmoxSupportStatus *qmp_query_proxmox_support(Error **errp)
|
2020-11-24 18:41:20 +03:00
|
|
|
ret->pbs_library_version = g_strdup(proxmox_backup_qemu_version());
|
2020-10-29 20:05:43 +03:00
|
|
|
ret->pbs_dirty_bitmap = true;
|
2021-03-16 19:30:22 +03:00
|
|
|
ret->pbs_dirty_bitmap_savevm = true;
|
2020-10-29 20:05:43 +03:00
|
|
|
+ ret->pbs_dirty_bitmap_migration = true;
|
2021-03-16 19:30:22 +03:00
|
|
|
ret->query_bitmap_info = true;
|
2020-10-29 20:05:43 +03:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
diff --git a/qapi/block-core.json b/qapi/block-core.json
|
2022-04-25 11:07:01 +03:00
|
|
|
index 6a67adf923..c99ddf8628 100644
|
2020-10-29 20:05:43 +03:00
|
|
|
--- a/qapi/block-core.json
|
|
|
|
+++ b/qapi/block-core.json
|
2022-02-11 12:24:33 +03:00
|
|
|
@@ -883,6 +883,11 @@
|
2021-03-16 19:30:22 +03:00
|
|
|
# @pbs-dirty-bitmap-savevm: True if 'dirty-bitmaps' migration capability can
|
|
|
|
# safely be set for savevm-async.
|
2020-10-29 20:05:43 +03:00
|
|
|
#
|
|
|
|
+# @pbs-dirty-bitmap-migration: True if safe migration of dirty-bitmaps including
|
|
|
|
+# PBS state is supported. Enabling 'dirty-bitmaps'
|
|
|
|
+# migration cap if this is false/unset may lead
|
|
|
|
+# to crashes on migration!
|
|
|
|
+#
|
2020-11-24 18:41:20 +03:00
|
|
|
# @pbs-library-version: Running version of libproxmox-backup-qemu0 library.
|
|
|
|
#
|
2020-10-29 20:05:43 +03:00
|
|
|
##
|
2022-02-11 12:24:33 +03:00
|
|
|
@@ -890,6 +895,7 @@
|
2020-11-24 18:41:20 +03:00
|
|
|
'data': { 'pbs-dirty-bitmap': 'bool',
|
|
|
|
'query-bitmap-info': 'bool',
|
2021-03-16 19:30:22 +03:00
|
|
|
'pbs-dirty-bitmap-savevm': 'bool',
|
2020-11-24 18:41:20 +03:00
|
|
|
+ 'pbs-dirty-bitmap-migration': 'bool',
|
|
|
|
'pbs-library-version': 'str' } }
|
2020-10-29 20:05:43 +03:00
|
|
|
|
|
|
|
##
|