mirror of https://github.com/xemu-project/xemu.git
migration/ram: Discard RAM when growing RAM blocks after ram_postcopy_incoming_init()
In case we grow our RAM after ram_postcopy_incoming_init() (e.g., when synchronizing the RAM block state with the migration source), the resized part would not get discarded. Let's perform that when being notified about a resize while postcopy has been advised, but is not listening yet. With precopy, the process is as following: 1. VM created - RAM blocks are created 2. Incomming migration started - Postcopy is advised - All pages in RAM blocks are discarded 3. Precopy starts - RAM blocks are resized to match the size on the migration source. - RAM pages from precopy stream are loaded - Uffd handler is registered, postcopy starts listening 4. Guest started, postcopy running - Pagefaults get resolved, pages get placed Reviewed-by: Peter Xu <peterx@redhat.com> Signed-off-by: David Hildenbrand <david@redhat.com> Message-Id: <20210429112708.12291-7-david@redhat.com> Signed-off-by: Dr. David Alan Gilbert <dgilbert@redhat.com>
This commit is contained in:
parent
dcdc460767
commit
cc61c703b6
migration
|
@ -4099,6 +4099,7 @@ static SaveVMHandlers savevm_ram_handlers = {
|
||||||
static void ram_mig_ram_block_resized(RAMBlockNotifier *n, void *host,
|
static void ram_mig_ram_block_resized(RAMBlockNotifier *n, void *host,
|
||||||
size_t old_size, size_t new_size)
|
size_t old_size, size_t new_size)
|
||||||
{
|
{
|
||||||
|
PostcopyState ps = postcopy_state_get();
|
||||||
ram_addr_t offset;
|
ram_addr_t offset;
|
||||||
RAMBlock *rb = qemu_ram_block_from_host(host, false, &offset);
|
RAMBlock *rb = qemu_ram_block_from_host(host, false, &offset);
|
||||||
Error *err = NULL;
|
Error *err = NULL;
|
||||||
|
@ -4119,6 +4120,35 @@ static void ram_mig_ram_block_resized(RAMBlockNotifier *n, void *host,
|
||||||
error_free(err);
|
error_free(err);
|
||||||
migration_cancel();
|
migration_cancel();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
switch (ps) {
|
||||||
|
case POSTCOPY_INCOMING_ADVISE:
|
||||||
|
/*
|
||||||
|
* Update what ram_postcopy_incoming_init()->init_range() does at the
|
||||||
|
* time postcopy was advised. Syncing RAM blocks with the source will
|
||||||
|
* result in RAM resizes.
|
||||||
|
*/
|
||||||
|
if (old_size < new_size) {
|
||||||
|
if (ram_discard_range(rb->idstr, old_size, new_size - old_size)) {
|
||||||
|
error_report("RAM block '%s' discard of resized RAM failed",
|
||||||
|
rb->idstr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case POSTCOPY_INCOMING_NONE:
|
||||||
|
case POSTCOPY_INCOMING_RUNNING:
|
||||||
|
case POSTCOPY_INCOMING_END:
|
||||||
|
/*
|
||||||
|
* Once our guest is running, postcopy does no longer care about
|
||||||
|
* resizes. When growing, the new memory was not available on the
|
||||||
|
* source, no handler needed.
|
||||||
|
*/
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
error_report("RAM block '%s' resized during postcopy state: %d",
|
||||||
|
rb->idstr, ps);
|
||||||
|
exit(-1);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static RAMBlockNotifier ram_mig_ram_notifier = {
|
static RAMBlockNotifier ram_mig_ram_notifier = {
|
||||||
|
|
Loading…
Reference in New Issue