11f0776f1SJuan Quintela /* 21f0776f1SJuan Quintela * QEMU migration capabilities 31f0776f1SJuan Quintela * 41f0776f1SJuan Quintela * Copyright (c) 2012-2023 Red Hat Inc 51f0776f1SJuan Quintela * 61f0776f1SJuan Quintela * Authors: 71f0776f1SJuan Quintela * Orit Wasserman <owasserm@redhat.com> 81f0776f1SJuan Quintela * Juan Quintela <quintela@redhat.com> 91f0776f1SJuan Quintela * 101f0776f1SJuan Quintela * This work is licensed under the terms of the GNU GPL, version 2 or later. 111f0776f1SJuan Quintela * See the COPYING file in the top-level directory. 121f0776f1SJuan Quintela */ 131f0776f1SJuan Quintela 141f0776f1SJuan Quintela #include "qemu/osdep.h" 1509d6c965SJuan Quintela #include "exec/target_page.h" 169c894df3SJuan Quintela #include "qapi/clone-visitor.h" 1777608706SJuan Quintela #include "qapi/error.h" 184d0c6b69SJuan Quintela #include "qapi/qapi-commands-migration.h" 199c894df3SJuan Quintela #include "qapi/qapi-visit-migration.h" 20f80196b7SJuan Quintela #include "qapi/qmp/qerror.h" 2109d6c965SJuan Quintela #include "qapi/qmp/qnull.h" 2277608706SJuan Quintela #include "sysemu/runstate.h" 2309d6c965SJuan Quintela #include "migration/colo.h" 242682c4eeSJuan Quintela #include "migration/misc.h" 251f0776f1SJuan Quintela #include "migration.h" 26e1fde0e0SJuan Quintela #include "migration-stats.h" 2709d6c965SJuan Quintela #include "qemu-file.h" 2877608706SJuan Quintela #include "ram.h" 291f0776f1SJuan Quintela #include "options.h" 301f0776f1SJuan Quintela 3109d6c965SJuan Quintela /* Maximum migrate downtime set to 2000 seconds */ 3209d6c965SJuan Quintela #define MAX_MIGRATE_DOWNTIME_SECONDS 2000 3309d6c965SJuan Quintela #define MAX_MIGRATE_DOWNTIME (MAX_MIGRATE_DOWNTIME_SECONDS * 1000) 3409d6c965SJuan Quintela 35f9436522SJuan Quintela #define MAX_THROTTLE (128 << 20) /* Migration transfer speed throttling */ 36f9436522SJuan Quintela 37f9436522SJuan Quintela /* Time in milliseconds we are allowed to stop the source, 38f9436522SJuan Quintela * for sending the last part */ 39f9436522SJuan Quintela #define DEFAULT_MIGRATE_SET_DOWNTIME 300 40f9436522SJuan Quintela 41f9436522SJuan Quintela /* Default compression thread count */ 42f9436522SJuan Quintela #define DEFAULT_MIGRATE_COMPRESS_THREAD_COUNT 8 43f9436522SJuan Quintela /* Default decompression thread count, usually decompression is at 44f9436522SJuan Quintela * least 4 times as fast as compression.*/ 45f9436522SJuan Quintela #define DEFAULT_MIGRATE_DECOMPRESS_THREAD_COUNT 2 46f9436522SJuan Quintela /*0: means nocompress, 1: best speed, ... 9: best compress ratio */ 47f9436522SJuan Quintela #define DEFAULT_MIGRATE_COMPRESS_LEVEL 1 48f9436522SJuan Quintela /* Define default autoconverge cpu throttle migration parameters */ 49f9436522SJuan Quintela #define DEFAULT_MIGRATE_THROTTLE_TRIGGER_THRESHOLD 50 50f9436522SJuan Quintela #define DEFAULT_MIGRATE_CPU_THROTTLE_INITIAL 20 51f9436522SJuan Quintela #define DEFAULT_MIGRATE_CPU_THROTTLE_INCREMENT 10 52f9436522SJuan Quintela #define DEFAULT_MIGRATE_MAX_CPU_THROTTLE 99 53f9436522SJuan Quintela 54f9436522SJuan Quintela /* Migration XBZRLE default cache size */ 55f9436522SJuan Quintela #define DEFAULT_MIGRATE_XBZRLE_CACHE_SIZE (64 * 1024 * 1024) 56f9436522SJuan Quintela 57f9436522SJuan Quintela /* The delay time (in ms) between two COLO checkpoints */ 58f9436522SJuan Quintela #define DEFAULT_MIGRATE_X_CHECKPOINT_DELAY (200 * 100) 59f9436522SJuan Quintela #define DEFAULT_MIGRATE_MULTIFD_CHANNELS 2 60f9436522SJuan Quintela #define DEFAULT_MIGRATE_MULTIFD_COMPRESSION MULTIFD_COMPRESSION_NONE 61f9436522SJuan Quintela /* 0: means nocompress, 1: best speed, ... 9: best compress ratio */ 62f9436522SJuan Quintela #define DEFAULT_MIGRATE_MULTIFD_ZLIB_LEVEL 1 63f9436522SJuan Quintela /* 0: means nocompress, 1: best speed, ... 20: best compress ratio */ 64f9436522SJuan Quintela #define DEFAULT_MIGRATE_MULTIFD_ZSTD_LEVEL 1 65f9436522SJuan Quintela 66f9436522SJuan Quintela /* Background transfer rate for postcopy, 0 means unlimited, note 67f9436522SJuan Quintela * that page requests can still exceed this limit. 68f9436522SJuan Quintela */ 69f9436522SJuan Quintela #define DEFAULT_MIGRATE_MAX_POSTCOPY_BANDWIDTH 0 70f9436522SJuan Quintela 71f9436522SJuan Quintela /* 72f9436522SJuan Quintela * Parameters for self_announce_delay giving a stream of RARP/ARP 73f9436522SJuan Quintela * packets after migration. 74f9436522SJuan Quintela */ 75f9436522SJuan Quintela #define DEFAULT_MIGRATE_ANNOUNCE_INITIAL 50 76f9436522SJuan Quintela #define DEFAULT_MIGRATE_ANNOUNCE_MAX 550 77f9436522SJuan Quintela #define DEFAULT_MIGRATE_ANNOUNCE_ROUNDS 5 78f9436522SJuan Quintela #define DEFAULT_MIGRATE_ANNOUNCE_STEP 100 79f9436522SJuan Quintela 80f9436522SJuan Quintela #define DEFINE_PROP_MIG_CAP(name, x) \ 81f9436522SJuan Quintela DEFINE_PROP_BOOL(name, MigrationState, capabilities[x], false) 82f9436522SJuan Quintela 83f9436522SJuan Quintela Property migration_properties[] = { 84f9436522SJuan Quintela DEFINE_PROP_BOOL("store-global-state", MigrationState, 85f9436522SJuan Quintela store_global_state, true), 86f9436522SJuan Quintela DEFINE_PROP_BOOL("send-configuration", MigrationState, 87f9436522SJuan Quintela send_configuration, true), 88f9436522SJuan Quintela DEFINE_PROP_BOOL("send-section-footer", MigrationState, 89f9436522SJuan Quintela send_section_footer, true), 90f9436522SJuan Quintela DEFINE_PROP_BOOL("decompress-error-check", MigrationState, 91f9436522SJuan Quintela decompress_error_check, true), 9277c259a4SJuan Quintela DEFINE_PROP_BOOL("multifd-flush-after-each-section", MigrationState, 93294e5a40SJuan Quintela multifd_flush_after_each_section, false), 94f9436522SJuan Quintela DEFINE_PROP_UINT8("x-clear-bitmap-shift", MigrationState, 95f9436522SJuan Quintela clear_bitmap_shift, CLEAR_BITMAP_SHIFT_DEFAULT), 96f9436522SJuan Quintela DEFINE_PROP_BOOL("x-preempt-pre-7-2", MigrationState, 97f9436522SJuan Quintela preempt_pre_7_2, false), 98f9436522SJuan Quintela 99f9436522SJuan Quintela /* Migration parameters */ 100f9436522SJuan Quintela DEFINE_PROP_UINT8("x-compress-level", MigrationState, 101f9436522SJuan Quintela parameters.compress_level, 102f9436522SJuan Quintela DEFAULT_MIGRATE_COMPRESS_LEVEL), 103f9436522SJuan Quintela DEFINE_PROP_UINT8("x-compress-threads", MigrationState, 104f9436522SJuan Quintela parameters.compress_threads, 105f9436522SJuan Quintela DEFAULT_MIGRATE_COMPRESS_THREAD_COUNT), 106f9436522SJuan Quintela DEFINE_PROP_BOOL("x-compress-wait-thread", MigrationState, 107f9436522SJuan Quintela parameters.compress_wait_thread, true), 108f9436522SJuan Quintela DEFINE_PROP_UINT8("x-decompress-threads", MigrationState, 109f9436522SJuan Quintela parameters.decompress_threads, 110f9436522SJuan Quintela DEFAULT_MIGRATE_DECOMPRESS_THREAD_COUNT), 111f9436522SJuan Quintela DEFINE_PROP_UINT8("x-throttle-trigger-threshold", MigrationState, 112f9436522SJuan Quintela parameters.throttle_trigger_threshold, 113f9436522SJuan Quintela DEFAULT_MIGRATE_THROTTLE_TRIGGER_THRESHOLD), 114f9436522SJuan Quintela DEFINE_PROP_UINT8("x-cpu-throttle-initial", MigrationState, 115f9436522SJuan Quintela parameters.cpu_throttle_initial, 116f9436522SJuan Quintela DEFAULT_MIGRATE_CPU_THROTTLE_INITIAL), 117f9436522SJuan Quintela DEFINE_PROP_UINT8("x-cpu-throttle-increment", MigrationState, 118f9436522SJuan Quintela parameters.cpu_throttle_increment, 119f9436522SJuan Quintela DEFAULT_MIGRATE_CPU_THROTTLE_INCREMENT), 120f9436522SJuan Quintela DEFINE_PROP_BOOL("x-cpu-throttle-tailslow", MigrationState, 121f9436522SJuan Quintela parameters.cpu_throttle_tailslow, false), 122f9436522SJuan Quintela DEFINE_PROP_SIZE("x-max-bandwidth", MigrationState, 123f9436522SJuan Quintela parameters.max_bandwidth, MAX_THROTTLE), 124f9436522SJuan Quintela DEFINE_PROP_UINT64("x-downtime-limit", MigrationState, 125f9436522SJuan Quintela parameters.downtime_limit, 126f9436522SJuan Quintela DEFAULT_MIGRATE_SET_DOWNTIME), 127f9436522SJuan Quintela DEFINE_PROP_UINT32("x-checkpoint-delay", MigrationState, 128f9436522SJuan Quintela parameters.x_checkpoint_delay, 129f9436522SJuan Quintela DEFAULT_MIGRATE_X_CHECKPOINT_DELAY), 130f9436522SJuan Quintela DEFINE_PROP_UINT8("multifd-channels", MigrationState, 131f9436522SJuan Quintela parameters.multifd_channels, 132f9436522SJuan Quintela DEFAULT_MIGRATE_MULTIFD_CHANNELS), 133f9436522SJuan Quintela DEFINE_PROP_MULTIFD_COMPRESSION("multifd-compression", MigrationState, 134f9436522SJuan Quintela parameters.multifd_compression, 135f9436522SJuan Quintela DEFAULT_MIGRATE_MULTIFD_COMPRESSION), 136f9436522SJuan Quintela DEFINE_PROP_UINT8("multifd-zlib-level", MigrationState, 137f9436522SJuan Quintela parameters.multifd_zlib_level, 138f9436522SJuan Quintela DEFAULT_MIGRATE_MULTIFD_ZLIB_LEVEL), 139f9436522SJuan Quintela DEFINE_PROP_UINT8("multifd-zstd-level", MigrationState, 140f9436522SJuan Quintela parameters.multifd_zstd_level, 141f9436522SJuan Quintela DEFAULT_MIGRATE_MULTIFD_ZSTD_LEVEL), 142f9436522SJuan Quintela DEFINE_PROP_SIZE("xbzrle-cache-size", MigrationState, 143f9436522SJuan Quintela parameters.xbzrle_cache_size, 144f9436522SJuan Quintela DEFAULT_MIGRATE_XBZRLE_CACHE_SIZE), 145f9436522SJuan Quintela DEFINE_PROP_SIZE("max-postcopy-bandwidth", MigrationState, 146f9436522SJuan Quintela parameters.max_postcopy_bandwidth, 147f9436522SJuan Quintela DEFAULT_MIGRATE_MAX_POSTCOPY_BANDWIDTH), 148f9436522SJuan Quintela DEFINE_PROP_UINT8("max-cpu-throttle", MigrationState, 149f9436522SJuan Quintela parameters.max_cpu_throttle, 150f9436522SJuan Quintela DEFAULT_MIGRATE_MAX_CPU_THROTTLE), 151f9436522SJuan Quintela DEFINE_PROP_SIZE("announce-initial", MigrationState, 152f9436522SJuan Quintela parameters.announce_initial, 153f9436522SJuan Quintela DEFAULT_MIGRATE_ANNOUNCE_INITIAL), 154f9436522SJuan Quintela DEFINE_PROP_SIZE("announce-max", MigrationState, 155f9436522SJuan Quintela parameters.announce_max, 156f9436522SJuan Quintela DEFAULT_MIGRATE_ANNOUNCE_MAX), 157f9436522SJuan Quintela DEFINE_PROP_SIZE("announce-rounds", MigrationState, 158f9436522SJuan Quintela parameters.announce_rounds, 159f9436522SJuan Quintela DEFAULT_MIGRATE_ANNOUNCE_ROUNDS), 160f9436522SJuan Quintela DEFINE_PROP_SIZE("announce-step", MigrationState, 161f9436522SJuan Quintela parameters.announce_step, 162f9436522SJuan Quintela DEFAULT_MIGRATE_ANNOUNCE_STEP), 163f9436522SJuan Quintela DEFINE_PROP_STRING("tls-creds", MigrationState, parameters.tls_creds), 164f9436522SJuan Quintela DEFINE_PROP_STRING("tls-hostname", MigrationState, parameters.tls_hostname), 165f9436522SJuan Quintela DEFINE_PROP_STRING("tls-authz", MigrationState, parameters.tls_authz), 166f9436522SJuan Quintela 167f9436522SJuan Quintela /* Migration capabilities */ 168f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-xbzrle", MIGRATION_CAPABILITY_XBZRLE), 169f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-rdma-pin-all", MIGRATION_CAPABILITY_RDMA_PIN_ALL), 170f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-auto-converge", MIGRATION_CAPABILITY_AUTO_CONVERGE), 171f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-zero-blocks", MIGRATION_CAPABILITY_ZERO_BLOCKS), 172f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-compress", MIGRATION_CAPABILITY_COMPRESS), 173f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-events", MIGRATION_CAPABILITY_EVENTS), 174f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-postcopy-ram", MIGRATION_CAPABILITY_POSTCOPY_RAM), 175f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-postcopy-preempt", 176f9436522SJuan Quintela MIGRATION_CAPABILITY_POSTCOPY_PREEMPT), 177f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-colo", MIGRATION_CAPABILITY_X_COLO), 178f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-release-ram", MIGRATION_CAPABILITY_RELEASE_RAM), 179f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-block", MIGRATION_CAPABILITY_BLOCK), 180f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-return-path", MIGRATION_CAPABILITY_RETURN_PATH), 181f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-multifd", MIGRATION_CAPABILITY_MULTIFD), 182f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-background-snapshot", 183f9436522SJuan Quintela MIGRATION_CAPABILITY_BACKGROUND_SNAPSHOT), 184f9436522SJuan Quintela #ifdef CONFIG_LINUX 185f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-zero-copy-send", 186f9436522SJuan Quintela MIGRATION_CAPABILITY_ZERO_COPY_SEND), 187f9436522SJuan Quintela #endif 188*6574232fSAvihai Horon DEFINE_PROP_MIG_CAP("x-switchover-ack", 189*6574232fSAvihai Horon MIGRATION_CAPABILITY_SWITCHOVER_ACK), 190f9436522SJuan Quintela 191f9436522SJuan Quintela DEFINE_PROP_END_OF_LIST(), 192f9436522SJuan Quintela }; 193f9436522SJuan Quintela 1941f0776f1SJuan Quintela bool migrate_auto_converge(void) 1951f0776f1SJuan Quintela { 1968f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 1971f0776f1SJuan Quintela 1981f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_AUTO_CONVERGE]; 1991f0776f1SJuan Quintela } 2001f0776f1SJuan Quintela 2011f0776f1SJuan Quintela bool migrate_background_snapshot(void) 2021f0776f1SJuan Quintela { 2038f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 2041f0776f1SJuan Quintela 2051f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_BACKGROUND_SNAPSHOT]; 2061f0776f1SJuan Quintela } 2071f0776f1SJuan Quintela 2089d4b1e5fSJuan Quintela bool migrate_block(void) 2099d4b1e5fSJuan Quintela { 2108f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 2119d4b1e5fSJuan Quintela 2129d4b1e5fSJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_BLOCK]; 2139d4b1e5fSJuan Quintela } 2149d4b1e5fSJuan Quintela 2155e804644SJuan Quintela bool migrate_colo(void) 2165e804644SJuan Quintela { 2175e804644SJuan Quintela MigrationState *s = migrate_get_current(); 2188f9c5327SJuan Quintela 2195e804644SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_X_COLO]; 2205e804644SJuan Quintela } 2215e804644SJuan Quintela 222a7a94d14SJuan Quintela bool migrate_compress(void) 223a7a94d14SJuan Quintela { 2248f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 225a7a94d14SJuan Quintela 226a7a94d14SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_COMPRESS]; 227a7a94d14SJuan Quintela } 228a7a94d14SJuan Quintela 2291f0776f1SJuan Quintela bool migrate_dirty_bitmaps(void) 2301f0776f1SJuan Quintela { 2318f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 2321f0776f1SJuan Quintela 2331f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_DIRTY_BITMAPS]; 2341f0776f1SJuan Quintela } 2351f0776f1SJuan Quintela 236b890902cSJuan Quintela bool migrate_events(void) 237b890902cSJuan Quintela { 2388f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 239b890902cSJuan Quintela 240b890902cSJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_EVENTS]; 241b890902cSJuan Quintela } 242b890902cSJuan Quintela 2431f0776f1SJuan Quintela bool migrate_ignore_shared(void) 2441f0776f1SJuan Quintela { 2458f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 2461f0776f1SJuan Quintela 2471f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_X_IGNORE_SHARED]; 2481f0776f1SJuan Quintela } 2491f0776f1SJuan Quintela 2501f0776f1SJuan Quintela bool migrate_late_block_activate(void) 2511f0776f1SJuan Quintela { 2528f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 2531f0776f1SJuan Quintela 2541f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_LATE_BLOCK_ACTIVATE]; 2551f0776f1SJuan Quintela } 2561f0776f1SJuan Quintela 25751b07548SJuan Quintela bool migrate_multifd(void) 25851b07548SJuan Quintela { 2598f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 26051b07548SJuan Quintela 26151b07548SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_MULTIFD]; 26251b07548SJuan Quintela } 26351b07548SJuan Quintela 2641f0776f1SJuan Quintela bool migrate_pause_before_switchover(void) 2651f0776f1SJuan Quintela { 2668f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 2671f0776f1SJuan Quintela 2681f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_PAUSE_BEFORE_SWITCHOVER]; 2691f0776f1SJuan Quintela } 2701f0776f1SJuan Quintela 2711f0776f1SJuan Quintela bool migrate_postcopy_blocktime(void) 2721f0776f1SJuan Quintela { 2738f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 2741f0776f1SJuan Quintela 2751f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_POSTCOPY_BLOCKTIME]; 2761f0776f1SJuan Quintela } 2771f0776f1SJuan Quintela 2781f0776f1SJuan Quintela bool migrate_postcopy_preempt(void) 2791f0776f1SJuan Quintela { 2808f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 2811f0776f1SJuan Quintela 2821f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_POSTCOPY_PREEMPT]; 2831f0776f1SJuan Quintela } 2841f0776f1SJuan Quintela 2851f0776f1SJuan Quintela bool migrate_postcopy_ram(void) 2861f0776f1SJuan Quintela { 2878f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 2881f0776f1SJuan Quintela 2891f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_POSTCOPY_RAM]; 2901f0776f1SJuan Quintela } 2911f0776f1SJuan Quintela 29217cba690SJuan Quintela bool migrate_rdma_pin_all(void) 29317cba690SJuan Quintela { 29417cba690SJuan Quintela MigrationState *s = migrate_get_current(); 29517cba690SJuan Quintela 29617cba690SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_RDMA_PIN_ALL]; 29717cba690SJuan Quintela } 29817cba690SJuan Quintela 2991f0776f1SJuan Quintela bool migrate_release_ram(void) 3001f0776f1SJuan Quintela { 3018f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 3021f0776f1SJuan Quintela 3031f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_RELEASE_RAM]; 3041f0776f1SJuan Quintela } 3051f0776f1SJuan Quintela 30638ad1110SJuan Quintela bool migrate_return_path(void) 30738ad1110SJuan Quintela { 3088f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 30938ad1110SJuan Quintela 31038ad1110SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_RETURN_PATH]; 31138ad1110SJuan Quintela } 31238ad1110SJuan Quintela 313*6574232fSAvihai Horon bool migrate_switchover_ack(void) 314*6574232fSAvihai Horon { 315*6574232fSAvihai Horon MigrationState *s = migrate_get_current(); 316*6574232fSAvihai Horon 317*6574232fSAvihai Horon return s->capabilities[MIGRATION_CAPABILITY_SWITCHOVER_ACK]; 318*6574232fSAvihai Horon } 319*6574232fSAvihai Horon 3201f0776f1SJuan Quintela bool migrate_validate_uuid(void) 3211f0776f1SJuan Quintela { 3228f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 3231f0776f1SJuan Quintela 3241f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_VALIDATE_UUID]; 3251f0776f1SJuan Quintela } 3261f0776f1SJuan Quintela 32787dca0c9SJuan Quintela bool migrate_xbzrle(void) 32887dca0c9SJuan Quintela { 3298f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 33087dca0c9SJuan Quintela 33187dca0c9SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_XBZRLE]; 33287dca0c9SJuan Quintela } 33387dca0c9SJuan Quintela 3341f0776f1SJuan Quintela bool migrate_zero_blocks(void) 3351f0776f1SJuan Quintela { 3368f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 3371f0776f1SJuan Quintela 3381f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_ZERO_BLOCKS]; 3391f0776f1SJuan Quintela } 340b4bc342cSJuan Quintela 341b4bc342cSJuan Quintela bool migrate_zero_copy_send(void) 342b4bc342cSJuan Quintela { 3438f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 344b4bc342cSJuan Quintela 345b4bc342cSJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_ZERO_COPY_SEND]; 346b4bc342cSJuan Quintela } 347f774fde5SJuan Quintela 348f774fde5SJuan Quintela /* pseudo capabilities */ 349f774fde5SJuan Quintela 35077c259a4SJuan Quintela bool migrate_multifd_flush_after_each_section(void) 35177c259a4SJuan Quintela { 35277c259a4SJuan Quintela MigrationState *s = migrate_get_current(); 35377c259a4SJuan Quintela 354294e5a40SJuan Quintela return s->multifd_flush_after_each_section; 35577c259a4SJuan Quintela } 35677c259a4SJuan Quintela 357f774fde5SJuan Quintela bool migrate_postcopy(void) 358f774fde5SJuan Quintela { 359f774fde5SJuan Quintela return migrate_postcopy_ram() || migrate_dirty_bitmaps(); 360f774fde5SJuan Quintela } 361f774fde5SJuan Quintela 36210d4703bSJuan Quintela bool migrate_tls(void) 36310d4703bSJuan Quintela { 3648f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 36510d4703bSJuan Quintela 36610d4703bSJuan Quintela return s->parameters.tls_creds && *s->parameters.tls_creds; 36710d4703bSJuan Quintela } 36810d4703bSJuan Quintela 36977608706SJuan Quintela typedef enum WriteTrackingSupport { 37077608706SJuan Quintela WT_SUPPORT_UNKNOWN = 0, 37177608706SJuan Quintela WT_SUPPORT_ABSENT, 37277608706SJuan Quintela WT_SUPPORT_AVAILABLE, 37377608706SJuan Quintela WT_SUPPORT_COMPATIBLE 37477608706SJuan Quintela } WriteTrackingSupport; 37577608706SJuan Quintela 37677608706SJuan Quintela static 37777608706SJuan Quintela WriteTrackingSupport migrate_query_write_tracking(void) 37877608706SJuan Quintela { 37977608706SJuan Quintela /* Check if kernel supports required UFFD features */ 38077608706SJuan Quintela if (!ram_write_tracking_available()) { 38177608706SJuan Quintela return WT_SUPPORT_ABSENT; 38277608706SJuan Quintela } 38377608706SJuan Quintela /* 38477608706SJuan Quintela * Check if current memory configuration is 38577608706SJuan Quintela * compatible with required UFFD features. 38677608706SJuan Quintela */ 38777608706SJuan Quintela if (!ram_write_tracking_compatible()) { 38877608706SJuan Quintela return WT_SUPPORT_AVAILABLE; 38977608706SJuan Quintela } 39077608706SJuan Quintela 39177608706SJuan Quintela return WT_SUPPORT_COMPATIBLE; 39277608706SJuan Quintela } 39377608706SJuan Quintela 39477608706SJuan Quintela /* Migration capabilities set */ 39577608706SJuan Quintela struct MigrateCapsSet { 39677608706SJuan Quintela int size; /* Capability set size */ 39777608706SJuan Quintela MigrationCapability caps[]; /* Variadic array of capabilities */ 39877608706SJuan Quintela }; 39977608706SJuan Quintela typedef struct MigrateCapsSet MigrateCapsSet; 40077608706SJuan Quintela 40177608706SJuan Quintela /* Define and initialize MigrateCapsSet */ 40277608706SJuan Quintela #define INITIALIZE_MIGRATE_CAPS_SET(_name, ...) \ 40377608706SJuan Quintela MigrateCapsSet _name = { \ 40477608706SJuan Quintela .size = sizeof((int []) { __VA_ARGS__ }) / sizeof(int), \ 40577608706SJuan Quintela .caps = { __VA_ARGS__ } \ 40677608706SJuan Quintela } 40777608706SJuan Quintela 40877608706SJuan Quintela /* Background-snapshot compatibility check list */ 40977608706SJuan Quintela static const 41077608706SJuan Quintela INITIALIZE_MIGRATE_CAPS_SET(check_caps_background_snapshot, 41177608706SJuan Quintela MIGRATION_CAPABILITY_POSTCOPY_RAM, 41277608706SJuan Quintela MIGRATION_CAPABILITY_DIRTY_BITMAPS, 41377608706SJuan Quintela MIGRATION_CAPABILITY_POSTCOPY_BLOCKTIME, 41477608706SJuan Quintela MIGRATION_CAPABILITY_LATE_BLOCK_ACTIVATE, 41577608706SJuan Quintela MIGRATION_CAPABILITY_RETURN_PATH, 41677608706SJuan Quintela MIGRATION_CAPABILITY_MULTIFD, 41777608706SJuan Quintela MIGRATION_CAPABILITY_PAUSE_BEFORE_SWITCHOVER, 41877608706SJuan Quintela MIGRATION_CAPABILITY_AUTO_CONVERGE, 41977608706SJuan Quintela MIGRATION_CAPABILITY_RELEASE_RAM, 42077608706SJuan Quintela MIGRATION_CAPABILITY_RDMA_PIN_ALL, 42177608706SJuan Quintela MIGRATION_CAPABILITY_COMPRESS, 42277608706SJuan Quintela MIGRATION_CAPABILITY_XBZRLE, 42377608706SJuan Quintela MIGRATION_CAPABILITY_X_COLO, 42477608706SJuan Quintela MIGRATION_CAPABILITY_VALIDATE_UUID, 42577608706SJuan Quintela MIGRATION_CAPABILITY_ZERO_COPY_SEND); 42677608706SJuan Quintela 42777608706SJuan Quintela /** 42877608706SJuan Quintela * @migration_caps_check - check capability compatibility 42977608706SJuan Quintela * 43077608706SJuan Quintela * @old_caps: old capability list 43177608706SJuan Quintela * @new_caps: new capability list 43277608706SJuan Quintela * @errp: set *errp if the check failed, with reason 43377608706SJuan Quintela * 43477608706SJuan Quintela * Returns true if check passed, otherwise false. 43577608706SJuan Quintela */ 43677608706SJuan Quintela bool migrate_caps_check(bool *old_caps, bool *new_caps, Error **errp) 43777608706SJuan Quintela { 43877608706SJuan Quintela MigrationIncomingState *mis = migration_incoming_get_current(); 43977608706SJuan Quintela 44074c38cf7SPeter Xu ERRP_GUARD(); 44177608706SJuan Quintela #ifndef CONFIG_LIVE_BLOCK_MIGRATION 44277608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_BLOCK]) { 44377608706SJuan Quintela error_setg(errp, "QEMU compiled without old-style (blk/-b, inc/-i) " 44477608706SJuan Quintela "block migration"); 44577608706SJuan Quintela error_append_hint(errp, "Use drive_mirror+NBD instead.\n"); 44677608706SJuan Quintela return false; 44777608706SJuan Quintela } 44877608706SJuan Quintela #endif 44977608706SJuan Quintela 45077608706SJuan Quintela #ifndef CONFIG_REPLICATION 45177608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_X_COLO]) { 45277608706SJuan Quintela error_setg(errp, "QEMU compiled without replication module" 45377608706SJuan Quintela " can't enable COLO"); 45477608706SJuan Quintela error_append_hint(errp, "Please enable replication before COLO.\n"); 45577608706SJuan Quintela return false; 45677608706SJuan Quintela } 45777608706SJuan Quintela #endif 45877608706SJuan Quintela 45977608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_POSTCOPY_RAM]) { 46077608706SJuan Quintela /* This check is reasonably expensive, so only when it's being 46177608706SJuan Quintela * set the first time, also it's only the destination that needs 46277608706SJuan Quintela * special support. 46377608706SJuan Quintela */ 46477608706SJuan Quintela if (!old_caps[MIGRATION_CAPABILITY_POSTCOPY_RAM] && 46577608706SJuan Quintela runstate_check(RUN_STATE_INMIGRATE) && 46674c38cf7SPeter Xu !postcopy_ram_supported_by_host(mis, errp)) { 46774c38cf7SPeter Xu error_prepend(errp, "Postcopy is not supported: "); 46877608706SJuan Quintela return false; 46977608706SJuan Quintela } 47077608706SJuan Quintela 47177608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_X_IGNORE_SHARED]) { 47277608706SJuan Quintela error_setg(errp, "Postcopy is not compatible with ignore-shared"); 47377608706SJuan Quintela return false; 47477608706SJuan Quintela } 475b405dfffSLeonardo Bras 476b405dfffSLeonardo Bras if (new_caps[MIGRATION_CAPABILITY_MULTIFD]) { 477b405dfffSLeonardo Bras error_setg(errp, "Postcopy is not yet compatible with multifd"); 478b405dfffSLeonardo Bras return false; 479b405dfffSLeonardo Bras } 48077608706SJuan Quintela } 48177608706SJuan Quintela 48277608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_BACKGROUND_SNAPSHOT]) { 48377608706SJuan Quintela WriteTrackingSupport wt_support; 48477608706SJuan Quintela int idx; 48577608706SJuan Quintela /* 48677608706SJuan Quintela * Check if 'background-snapshot' capability is supported by 48777608706SJuan Quintela * host kernel and compatible with guest memory configuration. 48877608706SJuan Quintela */ 48977608706SJuan Quintela wt_support = migrate_query_write_tracking(); 49077608706SJuan Quintela if (wt_support < WT_SUPPORT_AVAILABLE) { 49177608706SJuan Quintela error_setg(errp, "Background-snapshot is not supported by host kernel"); 49277608706SJuan Quintela return false; 49377608706SJuan Quintela } 49477608706SJuan Quintela if (wt_support < WT_SUPPORT_COMPATIBLE) { 49577608706SJuan Quintela error_setg(errp, "Background-snapshot is not compatible " 49677608706SJuan Quintela "with guest memory configuration"); 49777608706SJuan Quintela return false; 49877608706SJuan Quintela } 49977608706SJuan Quintela 50077608706SJuan Quintela /* 50177608706SJuan Quintela * Check if there are any migration capabilities 50277608706SJuan Quintela * incompatible with 'background-snapshot'. 50377608706SJuan Quintela */ 50477608706SJuan Quintela for (idx = 0; idx < check_caps_background_snapshot.size; idx++) { 50577608706SJuan Quintela int incomp_cap = check_caps_background_snapshot.caps[idx]; 50677608706SJuan Quintela if (new_caps[incomp_cap]) { 50777608706SJuan Quintela error_setg(errp, 50877608706SJuan Quintela "Background-snapshot is not compatible with %s", 50977608706SJuan Quintela MigrationCapability_str(incomp_cap)); 51077608706SJuan Quintela return false; 51177608706SJuan Quintela } 51277608706SJuan Quintela } 51377608706SJuan Quintela } 51477608706SJuan Quintela 51577608706SJuan Quintela #ifdef CONFIG_LINUX 51677608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_ZERO_COPY_SEND] && 51777608706SJuan Quintela (!new_caps[MIGRATION_CAPABILITY_MULTIFD] || 51877608706SJuan Quintela new_caps[MIGRATION_CAPABILITY_COMPRESS] || 51977608706SJuan Quintela new_caps[MIGRATION_CAPABILITY_XBZRLE] || 52077608706SJuan Quintela migrate_multifd_compression() || 52110d4703bSJuan Quintela migrate_tls())) { 52277608706SJuan Quintela error_setg(errp, 52377608706SJuan Quintela "Zero copy only available for non-compressed non-TLS multifd migration"); 52477608706SJuan Quintela return false; 52577608706SJuan Quintela } 52677608706SJuan Quintela #else 52777608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_ZERO_COPY_SEND]) { 52877608706SJuan Quintela error_setg(errp, 52977608706SJuan Quintela "Zero copy currently only available on Linux"); 53077608706SJuan Quintela return false; 53177608706SJuan Quintela } 53277608706SJuan Quintela #endif 53377608706SJuan Quintela 53477608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_POSTCOPY_PREEMPT]) { 53577608706SJuan Quintela if (!new_caps[MIGRATION_CAPABILITY_POSTCOPY_RAM]) { 53677608706SJuan Quintela error_setg(errp, "Postcopy preempt requires postcopy-ram"); 53777608706SJuan Quintela return false; 53877608706SJuan Quintela } 53977608706SJuan Quintela 54077608706SJuan Quintela /* 54177608706SJuan Quintela * Preempt mode requires urgent pages to be sent in separate 54277608706SJuan Quintela * channel, OTOH compression logic will disorder all pages into 54377608706SJuan Quintela * different compression channels, which is not compatible with the 54477608706SJuan Quintela * preempt assumptions on channel assignments. 54577608706SJuan Quintela */ 54677608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_COMPRESS]) { 54777608706SJuan Quintela error_setg(errp, "Postcopy preempt not compatible with compress"); 54877608706SJuan Quintela return false; 54977608706SJuan Quintela } 55077608706SJuan Quintela } 55177608706SJuan Quintela 55277608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_MULTIFD]) { 55377608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_COMPRESS]) { 55477608706SJuan Quintela error_setg(errp, "Multifd is not compatible with compress"); 55577608706SJuan Quintela return false; 55677608706SJuan Quintela } 55777608706SJuan Quintela } 55877608706SJuan Quintela 559*6574232fSAvihai Horon if (new_caps[MIGRATION_CAPABILITY_SWITCHOVER_ACK]) { 560*6574232fSAvihai Horon if (!new_caps[MIGRATION_CAPABILITY_RETURN_PATH]) { 561*6574232fSAvihai Horon error_setg(errp, "Capability 'switchover-ack' requires capability " 562*6574232fSAvihai Horon "'return-path'"); 563*6574232fSAvihai Horon return false; 564*6574232fSAvihai Horon } 565*6574232fSAvihai Horon 566*6574232fSAvihai Horon /* Disable this capability until it's implemented */ 567*6574232fSAvihai Horon error_setg(errp, "'switchover-ack' is not implemented yet"); 568*6574232fSAvihai Horon return false; 569*6574232fSAvihai Horon } 570*6574232fSAvihai Horon 57177608706SJuan Quintela return true; 57277608706SJuan Quintela } 5734d0c6b69SJuan Quintela 574f80196b7SJuan Quintela bool migrate_cap_set(int cap, bool value, Error **errp) 575f80196b7SJuan Quintela { 576f80196b7SJuan Quintela MigrationState *s = migrate_get_current(); 577f80196b7SJuan Quintela bool new_caps[MIGRATION_CAPABILITY__MAX]; 578f80196b7SJuan Quintela 579f80196b7SJuan Quintela if (migration_is_running(s->state)) { 580f80196b7SJuan Quintela error_setg(errp, QERR_MIGRATION_ACTIVE); 581f80196b7SJuan Quintela return false; 582f80196b7SJuan Quintela } 583f80196b7SJuan Quintela 584f80196b7SJuan Quintela memcpy(new_caps, s->capabilities, sizeof(new_caps)); 585f80196b7SJuan Quintela new_caps[cap] = value; 586f80196b7SJuan Quintela 587f80196b7SJuan Quintela if (!migrate_caps_check(s->capabilities, new_caps, errp)) { 588f80196b7SJuan Quintela return false; 589f80196b7SJuan Quintela } 590f80196b7SJuan Quintela s->capabilities[cap] = value; 591f80196b7SJuan Quintela return true; 592f80196b7SJuan Quintela } 593f80196b7SJuan Quintela 5944d0c6b69SJuan Quintela MigrationCapabilityStatusList *qmp_query_migrate_capabilities(Error **errp) 5954d0c6b69SJuan Quintela { 5964d0c6b69SJuan Quintela MigrationCapabilityStatusList *head = NULL, **tail = &head; 5974d0c6b69SJuan Quintela MigrationCapabilityStatus *caps; 5984d0c6b69SJuan Quintela MigrationState *s = migrate_get_current(); 5994d0c6b69SJuan Quintela int i; 6004d0c6b69SJuan Quintela 6014d0c6b69SJuan Quintela for (i = 0; i < MIGRATION_CAPABILITY__MAX; i++) { 6024d0c6b69SJuan Quintela #ifndef CONFIG_LIVE_BLOCK_MIGRATION 6034d0c6b69SJuan Quintela if (i == MIGRATION_CAPABILITY_BLOCK) { 6044d0c6b69SJuan Quintela continue; 6054d0c6b69SJuan Quintela } 6064d0c6b69SJuan Quintela #endif 6074d0c6b69SJuan Quintela caps = g_malloc0(sizeof(*caps)); 6084d0c6b69SJuan Quintela caps->capability = i; 6094d0c6b69SJuan Quintela caps->state = s->capabilities[i]; 6104d0c6b69SJuan Quintela QAPI_LIST_APPEND(tail, caps); 6114d0c6b69SJuan Quintela } 6124d0c6b69SJuan Quintela 6134d0c6b69SJuan Quintela return head; 6144d0c6b69SJuan Quintela } 61545c1de13SJuan Quintela 61645c1de13SJuan Quintela void qmp_migrate_set_capabilities(MigrationCapabilityStatusList *params, 61745c1de13SJuan Quintela Error **errp) 61845c1de13SJuan Quintela { 61945c1de13SJuan Quintela MigrationState *s = migrate_get_current(); 62045c1de13SJuan Quintela MigrationCapabilityStatusList *cap; 62145c1de13SJuan Quintela bool new_caps[MIGRATION_CAPABILITY__MAX]; 62245c1de13SJuan Quintela 623d70178a8SVladimir Sementsov-Ogievskiy if (migration_is_running(s->state) || migration_in_colo_state()) { 62445c1de13SJuan Quintela error_setg(errp, QERR_MIGRATION_ACTIVE); 62545c1de13SJuan Quintela return; 62645c1de13SJuan Quintela } 62745c1de13SJuan Quintela 62845c1de13SJuan Quintela memcpy(new_caps, s->capabilities, sizeof(new_caps)); 62945c1de13SJuan Quintela for (cap = params; cap; cap = cap->next) { 63045c1de13SJuan Quintela new_caps[cap->value->capability] = cap->value->state; 63145c1de13SJuan Quintela } 63245c1de13SJuan Quintela 63345c1de13SJuan Quintela if (!migrate_caps_check(s->capabilities, new_caps, errp)) { 63445c1de13SJuan Quintela return; 63545c1de13SJuan Quintela } 63645c1de13SJuan Quintela 63745c1de13SJuan Quintela for (cap = params; cap; cap = cap->next) { 63845c1de13SJuan Quintela s->capabilities[cap->value->capability] = cap->value->state; 63945c1de13SJuan Quintela } 64045c1de13SJuan Quintela } 6411dfc4b9eSJuan Quintela 6421dfc4b9eSJuan Quintela /* parameters */ 6431dfc4b9eSJuan Quintela 644b804b35bSJuan Quintela const BitmapMigrationNodeAliasList *migrate_block_bitmap_mapping(void) 645b804b35bSJuan Quintela { 646b804b35bSJuan Quintela MigrationState *s = migrate_get_current(); 647b804b35bSJuan Quintela 648b804b35bSJuan Quintela return s->parameters.block_bitmap_mapping; 649b804b35bSJuan Quintela } 650b804b35bSJuan Quintela 6513cba22c9SJuan Quintela bool migrate_has_block_bitmap_mapping(void) 6523cba22c9SJuan Quintela { 6533cba22c9SJuan Quintela MigrationState *s = migrate_get_current(); 6543cba22c9SJuan Quintela 6553cba22c9SJuan Quintela return s->parameters.has_block_bitmap_mapping; 6563cba22c9SJuan Quintela } 6573cba22c9SJuan Quintela 6586f8be708SJuan Quintela bool migrate_block_incremental(void) 6596f8be708SJuan Quintela { 6608f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 6616f8be708SJuan Quintela 6626f8be708SJuan Quintela return s->parameters.block_incremental; 6636f8be708SJuan Quintela } 6646f8be708SJuan Quintela 665f94a858fSJuan Quintela uint32_t migrate_checkpoint_delay(void) 666f94a858fSJuan Quintela { 6678f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 668f94a858fSJuan Quintela 669f94a858fSJuan Quintela return s->parameters.x_checkpoint_delay; 670f94a858fSJuan Quintela } 671f94a858fSJuan Quintela 6721dfc4b9eSJuan Quintela int migrate_compress_level(void) 6731dfc4b9eSJuan Quintela { 6748f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 6751dfc4b9eSJuan Quintela 6761dfc4b9eSJuan Quintela return s->parameters.compress_level; 6771dfc4b9eSJuan Quintela } 6781dfc4b9eSJuan Quintela 6791dfc4b9eSJuan Quintela int migrate_compress_threads(void) 6801dfc4b9eSJuan Quintela { 6818f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 6821dfc4b9eSJuan Quintela 6831dfc4b9eSJuan Quintela return s->parameters.compress_threads; 6841dfc4b9eSJuan Quintela } 6851dfc4b9eSJuan Quintela 6861dfc4b9eSJuan Quintela int migrate_compress_wait_thread(void) 6871dfc4b9eSJuan Quintela { 6888f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 6891dfc4b9eSJuan Quintela 6901dfc4b9eSJuan Quintela return s->parameters.compress_wait_thread; 6911dfc4b9eSJuan Quintela } 6921dfc4b9eSJuan Quintela 6939605c2acSJuan Quintela uint8_t migrate_cpu_throttle_increment(void) 6949605c2acSJuan Quintela { 6958f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 6969605c2acSJuan Quintela 6979605c2acSJuan Quintela return s->parameters.cpu_throttle_increment; 6989605c2acSJuan Quintela } 6999605c2acSJuan Quintela 7002a8ec380SJuan Quintela uint8_t migrate_cpu_throttle_initial(void) 7012a8ec380SJuan Quintela { 7028f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7032a8ec380SJuan Quintela 7042a8ec380SJuan Quintela return s->parameters.cpu_throttle_initial; 7052a8ec380SJuan Quintela } 7062a8ec380SJuan Quintela 707873f674cSJuan Quintela bool migrate_cpu_throttle_tailslow(void) 708873f674cSJuan Quintela { 7098f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 710873f674cSJuan Quintela 711873f674cSJuan Quintela return s->parameters.cpu_throttle_tailslow; 712873f674cSJuan Quintela } 713873f674cSJuan Quintela 7141dfc4b9eSJuan Quintela int migrate_decompress_threads(void) 7151dfc4b9eSJuan Quintela { 7168f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7171dfc4b9eSJuan Quintela 7181dfc4b9eSJuan Quintela return s->parameters.decompress_threads; 7191dfc4b9eSJuan Quintela } 7201dfc4b9eSJuan Quintela 721f5da8ba4SJuan Quintela uint64_t migrate_downtime_limit(void) 722f5da8ba4SJuan Quintela { 723f5da8ba4SJuan Quintela MigrationState *s = migrate_get_current(); 724f5da8ba4SJuan Quintela 725f5da8ba4SJuan Quintela return s->parameters.downtime_limit; 726f5da8ba4SJuan Quintela } 727f5da8ba4SJuan Quintela 72824155bd0SJuan Quintela uint8_t migrate_max_cpu_throttle(void) 72924155bd0SJuan Quintela { 7308f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 73124155bd0SJuan Quintela 73224155bd0SJuan Quintela return s->parameters.max_cpu_throttle; 73324155bd0SJuan Quintela } 73424155bd0SJuan Quintela 7359c894df3SJuan Quintela uint64_t migrate_max_bandwidth(void) 7369c894df3SJuan Quintela { 7378f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7389c894df3SJuan Quintela 7399c894df3SJuan Quintela return s->parameters.max_bandwidth; 7409c894df3SJuan Quintela } 7419c894df3SJuan Quintela 74252033349SJuan Quintela uint64_t migrate_max_postcopy_bandwidth(void) 7431dfc4b9eSJuan Quintela { 7448f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7451dfc4b9eSJuan Quintela 7461dfc4b9eSJuan Quintela return s->parameters.max_postcopy_bandwidth; 7471dfc4b9eSJuan Quintela } 7481dfc4b9eSJuan Quintela 7491dfc4b9eSJuan Quintela int migrate_multifd_channels(void) 7501dfc4b9eSJuan Quintela { 7518f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7521dfc4b9eSJuan Quintela 7531dfc4b9eSJuan Quintela return s->parameters.multifd_channels; 7541dfc4b9eSJuan Quintela } 7551dfc4b9eSJuan Quintela 7561dfc4b9eSJuan Quintela MultiFDCompression migrate_multifd_compression(void) 7571dfc4b9eSJuan Quintela { 7588f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7591dfc4b9eSJuan Quintela 7601dfc4b9eSJuan Quintela assert(s->parameters.multifd_compression < MULTIFD_COMPRESSION__MAX); 7611dfc4b9eSJuan Quintela return s->parameters.multifd_compression; 7621dfc4b9eSJuan Quintela } 7631dfc4b9eSJuan Quintela 7641dfc4b9eSJuan Quintela int migrate_multifd_zlib_level(void) 7651dfc4b9eSJuan Quintela { 7668f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7671dfc4b9eSJuan Quintela 7681dfc4b9eSJuan Quintela return s->parameters.multifd_zlib_level; 7691dfc4b9eSJuan Quintela } 7701dfc4b9eSJuan Quintela 7711dfc4b9eSJuan Quintela int migrate_multifd_zstd_level(void) 7721dfc4b9eSJuan Quintela { 7738f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7741dfc4b9eSJuan Quintela 7751dfc4b9eSJuan Quintela return s->parameters.multifd_zstd_level; 7761dfc4b9eSJuan Quintela } 7771dfc4b9eSJuan Quintela 7786499efdbSJuan Quintela uint8_t migrate_throttle_trigger_threshold(void) 7796499efdbSJuan Quintela { 7808f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7816499efdbSJuan Quintela 7826499efdbSJuan Quintela return s->parameters.throttle_trigger_threshold; 7836499efdbSJuan Quintela } 7846499efdbSJuan Quintela 7852eb0308bSJuan Quintela const char *migrate_tls_authz(void) 7862eb0308bSJuan Quintela { 7872eb0308bSJuan Quintela MigrationState *s = migrate_get_current(); 7882eb0308bSJuan Quintela 7892eb0308bSJuan Quintela return s->parameters.tls_authz; 7902eb0308bSJuan Quintela } 7912eb0308bSJuan Quintela 792d5c3e195SJuan Quintela const char *migrate_tls_creds(void) 793d5c3e195SJuan Quintela { 794d5c3e195SJuan Quintela MigrationState *s = migrate_get_current(); 795d5c3e195SJuan Quintela 796d5c3e195SJuan Quintela return s->parameters.tls_creds; 797d5c3e195SJuan Quintela } 798d5c3e195SJuan Quintela 7991f2f366cSJuan Quintela const char *migrate_tls_hostname(void) 8001f2f366cSJuan Quintela { 8011f2f366cSJuan Quintela MigrationState *s = migrate_get_current(); 8021f2f366cSJuan Quintela 8031f2f366cSJuan Quintela return s->parameters.tls_hostname; 8041f2f366cSJuan Quintela } 8051f2f366cSJuan Quintela 8061dfc4b9eSJuan Quintela uint64_t migrate_xbzrle_cache_size(void) 8071dfc4b9eSJuan Quintela { 8088f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 8091dfc4b9eSJuan Quintela 8101dfc4b9eSJuan Quintela return s->parameters.xbzrle_cache_size; 8111dfc4b9eSJuan Quintela } 8122682c4eeSJuan Quintela 81387c22901SJuan Quintela /* parameter setters */ 81487c22901SJuan Quintela 81587c22901SJuan Quintela void migrate_set_block_incremental(bool value) 81687c22901SJuan Quintela { 81787c22901SJuan Quintela MigrationState *s = migrate_get_current(); 81887c22901SJuan Quintela 81987c22901SJuan Quintela s->parameters.block_incremental = value; 82087c22901SJuan Quintela } 82187c22901SJuan Quintela 8222682c4eeSJuan Quintela /* parameters helpers */ 8232682c4eeSJuan Quintela 824b1a87956SJuan Quintela void block_cleanup_parameters(void) 825b7b73122SJuan Quintela { 826b1a87956SJuan Quintela MigrationState *s = migrate_get_current(); 827b1a87956SJuan Quintela 828b7b73122SJuan Quintela if (s->must_remove_block_options) { 829b7b73122SJuan Quintela /* setting to false can never fail */ 830b7b73122SJuan Quintela migrate_cap_set(MIGRATION_CAPABILITY_BLOCK, false, &error_abort); 831b7b73122SJuan Quintela migrate_set_block_incremental(false); 832b7b73122SJuan Quintela s->must_remove_block_options = false; 833b7b73122SJuan Quintela } 834b7b73122SJuan Quintela } 835b7b73122SJuan Quintela 8362682c4eeSJuan Quintela AnnounceParameters *migrate_announce_params(void) 8372682c4eeSJuan Quintela { 8382682c4eeSJuan Quintela static AnnounceParameters ap; 8392682c4eeSJuan Quintela 8402682c4eeSJuan Quintela MigrationState *s = migrate_get_current(); 8412682c4eeSJuan Quintela 8422682c4eeSJuan Quintela ap.initial = s->parameters.announce_initial; 8432682c4eeSJuan Quintela ap.max = s->parameters.announce_max; 8442682c4eeSJuan Quintela ap.rounds = s->parameters.announce_rounds; 8452682c4eeSJuan Quintela ap.step = s->parameters.announce_step; 8462682c4eeSJuan Quintela 8472682c4eeSJuan Quintela return ≈ 8482682c4eeSJuan Quintela } 8499c894df3SJuan Quintela 8509c894df3SJuan Quintela MigrationParameters *qmp_query_migrate_parameters(Error **errp) 8519c894df3SJuan Quintela { 8529c894df3SJuan Quintela MigrationParameters *params; 8539c894df3SJuan Quintela MigrationState *s = migrate_get_current(); 8549c894df3SJuan Quintela 8559c894df3SJuan Quintela /* TODO use QAPI_CLONE() instead of duplicating it inline */ 8569c894df3SJuan Quintela params = g_malloc0(sizeof(*params)); 8579c894df3SJuan Quintela params->has_compress_level = true; 8589c894df3SJuan Quintela params->compress_level = s->parameters.compress_level; 8599c894df3SJuan Quintela params->has_compress_threads = true; 8609c894df3SJuan Quintela params->compress_threads = s->parameters.compress_threads; 8619c894df3SJuan Quintela params->has_compress_wait_thread = true; 8629c894df3SJuan Quintela params->compress_wait_thread = s->parameters.compress_wait_thread; 8639c894df3SJuan Quintela params->has_decompress_threads = true; 8649c894df3SJuan Quintela params->decompress_threads = s->parameters.decompress_threads; 8659c894df3SJuan Quintela params->has_throttle_trigger_threshold = true; 8669c894df3SJuan Quintela params->throttle_trigger_threshold = s->parameters.throttle_trigger_threshold; 8679c894df3SJuan Quintela params->has_cpu_throttle_initial = true; 8689c894df3SJuan Quintela params->cpu_throttle_initial = s->parameters.cpu_throttle_initial; 8699c894df3SJuan Quintela params->has_cpu_throttle_increment = true; 8709c894df3SJuan Quintela params->cpu_throttle_increment = s->parameters.cpu_throttle_increment; 8719c894df3SJuan Quintela params->has_cpu_throttle_tailslow = true; 8729c894df3SJuan Quintela params->cpu_throttle_tailslow = s->parameters.cpu_throttle_tailslow; 8739c894df3SJuan Quintela params->tls_creds = g_strdup(s->parameters.tls_creds); 8749c894df3SJuan Quintela params->tls_hostname = g_strdup(s->parameters.tls_hostname); 8759c894df3SJuan Quintela params->tls_authz = g_strdup(s->parameters.tls_authz ? 8769c894df3SJuan Quintela s->parameters.tls_authz : ""); 8779c894df3SJuan Quintela params->has_max_bandwidth = true; 8789c894df3SJuan Quintela params->max_bandwidth = s->parameters.max_bandwidth; 8799c894df3SJuan Quintela params->has_downtime_limit = true; 8809c894df3SJuan Quintela params->downtime_limit = s->parameters.downtime_limit; 8819c894df3SJuan Quintela params->has_x_checkpoint_delay = true; 8829c894df3SJuan Quintela params->x_checkpoint_delay = s->parameters.x_checkpoint_delay; 8839c894df3SJuan Quintela params->has_block_incremental = true; 8849c894df3SJuan Quintela params->block_incremental = s->parameters.block_incremental; 8859c894df3SJuan Quintela params->has_multifd_channels = true; 8869c894df3SJuan Quintela params->multifd_channels = s->parameters.multifd_channels; 8879c894df3SJuan Quintela params->has_multifd_compression = true; 8889c894df3SJuan Quintela params->multifd_compression = s->parameters.multifd_compression; 8899c894df3SJuan Quintela params->has_multifd_zlib_level = true; 8909c894df3SJuan Quintela params->multifd_zlib_level = s->parameters.multifd_zlib_level; 8919c894df3SJuan Quintela params->has_multifd_zstd_level = true; 8929c894df3SJuan Quintela params->multifd_zstd_level = s->parameters.multifd_zstd_level; 8939c894df3SJuan Quintela params->has_xbzrle_cache_size = true; 8949c894df3SJuan Quintela params->xbzrle_cache_size = s->parameters.xbzrle_cache_size; 8959c894df3SJuan Quintela params->has_max_postcopy_bandwidth = true; 8969c894df3SJuan Quintela params->max_postcopy_bandwidth = s->parameters.max_postcopy_bandwidth; 8979c894df3SJuan Quintela params->has_max_cpu_throttle = true; 8989c894df3SJuan Quintela params->max_cpu_throttle = s->parameters.max_cpu_throttle; 8999c894df3SJuan Quintela params->has_announce_initial = true; 9009c894df3SJuan Quintela params->announce_initial = s->parameters.announce_initial; 9019c894df3SJuan Quintela params->has_announce_max = true; 9029c894df3SJuan Quintela params->announce_max = s->parameters.announce_max; 9039c894df3SJuan Quintela params->has_announce_rounds = true; 9049c894df3SJuan Quintela params->announce_rounds = s->parameters.announce_rounds; 9059c894df3SJuan Quintela params->has_announce_step = true; 9069c894df3SJuan Quintela params->announce_step = s->parameters.announce_step; 9079c894df3SJuan Quintela 9089c894df3SJuan Quintela if (s->parameters.has_block_bitmap_mapping) { 9099c894df3SJuan Quintela params->has_block_bitmap_mapping = true; 9109c894df3SJuan Quintela params->block_bitmap_mapping = 9119c894df3SJuan Quintela QAPI_CLONE(BitmapMigrationNodeAliasList, 9129c894df3SJuan Quintela s->parameters.block_bitmap_mapping); 9139c894df3SJuan Quintela } 9149c894df3SJuan Quintela 9159c894df3SJuan Quintela return params; 9169c894df3SJuan Quintela } 91709d6c965SJuan Quintela 91861a174e2SJuan Quintela void migrate_params_init(MigrationParameters *params) 91961a174e2SJuan Quintela { 92061a174e2SJuan Quintela params->tls_hostname = g_strdup(""); 92161a174e2SJuan Quintela params->tls_creds = g_strdup(""); 92261a174e2SJuan Quintela 92361a174e2SJuan Quintela /* Set has_* up only for parameter checks */ 92461a174e2SJuan Quintela params->has_compress_level = true; 92561a174e2SJuan Quintela params->has_compress_threads = true; 92661a174e2SJuan Quintela params->has_compress_wait_thread = true; 92761a174e2SJuan Quintela params->has_decompress_threads = true; 92861a174e2SJuan Quintela params->has_throttle_trigger_threshold = true; 92961a174e2SJuan Quintela params->has_cpu_throttle_initial = true; 93061a174e2SJuan Quintela params->has_cpu_throttle_increment = true; 93161a174e2SJuan Quintela params->has_cpu_throttle_tailslow = true; 93261a174e2SJuan Quintela params->has_max_bandwidth = true; 93361a174e2SJuan Quintela params->has_downtime_limit = true; 93461a174e2SJuan Quintela params->has_x_checkpoint_delay = true; 93561a174e2SJuan Quintela params->has_block_incremental = true; 93661a174e2SJuan Quintela params->has_multifd_channels = true; 93761a174e2SJuan Quintela params->has_multifd_compression = true; 93861a174e2SJuan Quintela params->has_multifd_zlib_level = true; 93961a174e2SJuan Quintela params->has_multifd_zstd_level = true; 94061a174e2SJuan Quintela params->has_xbzrle_cache_size = true; 94161a174e2SJuan Quintela params->has_max_postcopy_bandwidth = true; 94261a174e2SJuan Quintela params->has_max_cpu_throttle = true; 94361a174e2SJuan Quintela params->has_announce_initial = true; 94461a174e2SJuan Quintela params->has_announce_max = true; 94561a174e2SJuan Quintela params->has_announce_rounds = true; 94661a174e2SJuan Quintela params->has_announce_step = true; 94761a174e2SJuan Quintela } 94861a174e2SJuan Quintela 94909d6c965SJuan Quintela /* 95009d6c965SJuan Quintela * Check whether the parameters are valid. Error will be put into errp 95109d6c965SJuan Quintela * (if provided). Return true if valid, otherwise false. 95209d6c965SJuan Quintela */ 95309d6c965SJuan Quintela bool migrate_params_check(MigrationParameters *params, Error **errp) 95409d6c965SJuan Quintela { 95509d6c965SJuan Quintela if (params->has_compress_level && 95609d6c965SJuan Quintela (params->compress_level > 9)) { 95709d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, "compress_level", 95809d6c965SJuan Quintela "a value between 0 and 9"); 95909d6c965SJuan Quintela return false; 96009d6c965SJuan Quintela } 96109d6c965SJuan Quintela 96209d6c965SJuan Quintela if (params->has_compress_threads && (params->compress_threads < 1)) { 96309d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 96409d6c965SJuan Quintela "compress_threads", 96509d6c965SJuan Quintela "a value between 1 and 255"); 96609d6c965SJuan Quintela return false; 96709d6c965SJuan Quintela } 96809d6c965SJuan Quintela 96909d6c965SJuan Quintela if (params->has_decompress_threads && (params->decompress_threads < 1)) { 97009d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 97109d6c965SJuan Quintela "decompress_threads", 97209d6c965SJuan Quintela "a value between 1 and 255"); 97309d6c965SJuan Quintela return false; 97409d6c965SJuan Quintela } 97509d6c965SJuan Quintela 97609d6c965SJuan Quintela if (params->has_throttle_trigger_threshold && 97709d6c965SJuan Quintela (params->throttle_trigger_threshold < 1 || 97809d6c965SJuan Quintela params->throttle_trigger_threshold > 100)) { 97909d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 98009d6c965SJuan Quintela "throttle_trigger_threshold", 98109d6c965SJuan Quintela "an integer in the range of 1 to 100"); 98209d6c965SJuan Quintela return false; 98309d6c965SJuan Quintela } 98409d6c965SJuan Quintela 98509d6c965SJuan Quintela if (params->has_cpu_throttle_initial && 98609d6c965SJuan Quintela (params->cpu_throttle_initial < 1 || 98709d6c965SJuan Quintela params->cpu_throttle_initial > 99)) { 98809d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 98909d6c965SJuan Quintela "cpu_throttle_initial", 99009d6c965SJuan Quintela "an integer in the range of 1 to 99"); 99109d6c965SJuan Quintela return false; 99209d6c965SJuan Quintela } 99309d6c965SJuan Quintela 99409d6c965SJuan Quintela if (params->has_cpu_throttle_increment && 99509d6c965SJuan Quintela (params->cpu_throttle_increment < 1 || 99609d6c965SJuan Quintela params->cpu_throttle_increment > 99)) { 99709d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 99809d6c965SJuan Quintela "cpu_throttle_increment", 99909d6c965SJuan Quintela "an integer in the range of 1 to 99"); 100009d6c965SJuan Quintela return false; 100109d6c965SJuan Quintela } 100209d6c965SJuan Quintela 100309d6c965SJuan Quintela if (params->has_max_bandwidth && (params->max_bandwidth > SIZE_MAX)) { 100409d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 100509d6c965SJuan Quintela "max_bandwidth", 100609d6c965SJuan Quintela "an integer in the range of 0 to "stringify(SIZE_MAX) 100709d6c965SJuan Quintela " bytes/second"); 100809d6c965SJuan Quintela return false; 100909d6c965SJuan Quintela } 101009d6c965SJuan Quintela 101109d6c965SJuan Quintela if (params->has_downtime_limit && 101209d6c965SJuan Quintela (params->downtime_limit > MAX_MIGRATE_DOWNTIME)) { 101309d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 101409d6c965SJuan Quintela "downtime_limit", 101509d6c965SJuan Quintela "an integer in the range of 0 to " 101609d6c965SJuan Quintela stringify(MAX_MIGRATE_DOWNTIME)" ms"); 101709d6c965SJuan Quintela return false; 101809d6c965SJuan Quintela } 101909d6c965SJuan Quintela 102009d6c965SJuan Quintela /* x_checkpoint_delay is now always positive */ 102109d6c965SJuan Quintela 102209d6c965SJuan Quintela if (params->has_multifd_channels && (params->multifd_channels < 1)) { 102309d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 102409d6c965SJuan Quintela "multifd_channels", 102509d6c965SJuan Quintela "a value between 1 and 255"); 102609d6c965SJuan Quintela return false; 102709d6c965SJuan Quintela } 102809d6c965SJuan Quintela 102909d6c965SJuan Quintela if (params->has_multifd_zlib_level && 103009d6c965SJuan Quintela (params->multifd_zlib_level > 9)) { 103109d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, "multifd_zlib_level", 103209d6c965SJuan Quintela "a value between 0 and 9"); 103309d6c965SJuan Quintela return false; 103409d6c965SJuan Quintela } 103509d6c965SJuan Quintela 103609d6c965SJuan Quintela if (params->has_multifd_zstd_level && 103709d6c965SJuan Quintela (params->multifd_zstd_level > 20)) { 103809d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, "multifd_zstd_level", 103909d6c965SJuan Quintela "a value between 0 and 20"); 104009d6c965SJuan Quintela return false; 104109d6c965SJuan Quintela } 104209d6c965SJuan Quintela 104309d6c965SJuan Quintela if (params->has_xbzrle_cache_size && 104409d6c965SJuan Quintela (params->xbzrle_cache_size < qemu_target_page_size() || 104509d6c965SJuan Quintela !is_power_of_2(params->xbzrle_cache_size))) { 104609d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 104709d6c965SJuan Quintela "xbzrle_cache_size", 104809d6c965SJuan Quintela "a power of two no less than the target page size"); 104909d6c965SJuan Quintela return false; 105009d6c965SJuan Quintela } 105109d6c965SJuan Quintela 105209d6c965SJuan Quintela if (params->has_max_cpu_throttle && 105309d6c965SJuan Quintela (params->max_cpu_throttle < params->cpu_throttle_initial || 105409d6c965SJuan Quintela params->max_cpu_throttle > 99)) { 105509d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 105609d6c965SJuan Quintela "max_cpu_throttle", 105709d6c965SJuan Quintela "an integer in the range of cpu_throttle_initial to 99"); 105809d6c965SJuan Quintela return false; 105909d6c965SJuan Quintela } 106009d6c965SJuan Quintela 106109d6c965SJuan Quintela if (params->has_announce_initial && 106209d6c965SJuan Quintela params->announce_initial > 100000) { 106309d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 106409d6c965SJuan Quintela "announce_initial", 106509d6c965SJuan Quintela "a value between 0 and 100000"); 106609d6c965SJuan Quintela return false; 106709d6c965SJuan Quintela } 106809d6c965SJuan Quintela if (params->has_announce_max && 106909d6c965SJuan Quintela params->announce_max > 100000) { 107009d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 107109d6c965SJuan Quintela "announce_max", 107209d6c965SJuan Quintela "a value between 0 and 100000"); 107309d6c965SJuan Quintela return false; 107409d6c965SJuan Quintela } 107509d6c965SJuan Quintela if (params->has_announce_rounds && 107609d6c965SJuan Quintela params->announce_rounds > 1000) { 107709d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 107809d6c965SJuan Quintela "announce_rounds", 107909d6c965SJuan Quintela "a value between 0 and 1000"); 108009d6c965SJuan Quintela return false; 108109d6c965SJuan Quintela } 108209d6c965SJuan Quintela if (params->has_announce_step && 108309d6c965SJuan Quintela (params->announce_step < 1 || 108409d6c965SJuan Quintela params->announce_step > 10000)) { 108509d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 108609d6c965SJuan Quintela "announce_step", 108709d6c965SJuan Quintela "a value between 0 and 10000"); 108809d6c965SJuan Quintela return false; 108909d6c965SJuan Quintela } 109009d6c965SJuan Quintela 109109d6c965SJuan Quintela if (params->has_block_bitmap_mapping && 109209d6c965SJuan Quintela !check_dirty_bitmap_mig_alias_map(params->block_bitmap_mapping, errp)) { 109309d6c965SJuan Quintela error_prepend(errp, "Invalid mapping given for block-bitmap-mapping: "); 109409d6c965SJuan Quintela return false; 109509d6c965SJuan Quintela } 109609d6c965SJuan Quintela 109709d6c965SJuan Quintela #ifdef CONFIG_LINUX 109809d6c965SJuan Quintela if (migrate_zero_copy_send() && 109909d6c965SJuan Quintela ((params->has_multifd_compression && params->multifd_compression) || 110009d6c965SJuan Quintela (params->tls_creds && *params->tls_creds))) { 110109d6c965SJuan Quintela error_setg(errp, 110209d6c965SJuan Quintela "Zero copy only available for non-compressed non-TLS multifd migration"); 110309d6c965SJuan Quintela return false; 110409d6c965SJuan Quintela } 110509d6c965SJuan Quintela #endif 110609d6c965SJuan Quintela 110709d6c965SJuan Quintela return true; 110809d6c965SJuan Quintela } 110909d6c965SJuan Quintela 111009d6c965SJuan Quintela static void migrate_params_test_apply(MigrateSetParameters *params, 111109d6c965SJuan Quintela MigrationParameters *dest) 111209d6c965SJuan Quintela { 111309d6c965SJuan Quintela *dest = migrate_get_current()->parameters; 111409d6c965SJuan Quintela 111509d6c965SJuan Quintela /* TODO use QAPI_CLONE() instead of duplicating it inline */ 111609d6c965SJuan Quintela 111709d6c965SJuan Quintela if (params->has_compress_level) { 111809d6c965SJuan Quintela dest->compress_level = params->compress_level; 111909d6c965SJuan Quintela } 112009d6c965SJuan Quintela 112109d6c965SJuan Quintela if (params->has_compress_threads) { 112209d6c965SJuan Quintela dest->compress_threads = params->compress_threads; 112309d6c965SJuan Quintela } 112409d6c965SJuan Quintela 112509d6c965SJuan Quintela if (params->has_compress_wait_thread) { 112609d6c965SJuan Quintela dest->compress_wait_thread = params->compress_wait_thread; 112709d6c965SJuan Quintela } 112809d6c965SJuan Quintela 112909d6c965SJuan Quintela if (params->has_decompress_threads) { 113009d6c965SJuan Quintela dest->decompress_threads = params->decompress_threads; 113109d6c965SJuan Quintela } 113209d6c965SJuan Quintela 113309d6c965SJuan Quintela if (params->has_throttle_trigger_threshold) { 113409d6c965SJuan Quintela dest->throttle_trigger_threshold = params->throttle_trigger_threshold; 113509d6c965SJuan Quintela } 113609d6c965SJuan Quintela 113709d6c965SJuan Quintela if (params->has_cpu_throttle_initial) { 113809d6c965SJuan Quintela dest->cpu_throttle_initial = params->cpu_throttle_initial; 113909d6c965SJuan Quintela } 114009d6c965SJuan Quintela 114109d6c965SJuan Quintela if (params->has_cpu_throttle_increment) { 114209d6c965SJuan Quintela dest->cpu_throttle_increment = params->cpu_throttle_increment; 114309d6c965SJuan Quintela } 114409d6c965SJuan Quintela 114509d6c965SJuan Quintela if (params->has_cpu_throttle_tailslow) { 114609d6c965SJuan Quintela dest->cpu_throttle_tailslow = params->cpu_throttle_tailslow; 114709d6c965SJuan Quintela } 114809d6c965SJuan Quintela 114909d6c965SJuan Quintela if (params->tls_creds) { 115009d6c965SJuan Quintela assert(params->tls_creds->type == QTYPE_QSTRING); 115109d6c965SJuan Quintela dest->tls_creds = params->tls_creds->u.s; 115209d6c965SJuan Quintela } 115309d6c965SJuan Quintela 115409d6c965SJuan Quintela if (params->tls_hostname) { 115509d6c965SJuan Quintela assert(params->tls_hostname->type == QTYPE_QSTRING); 115609d6c965SJuan Quintela dest->tls_hostname = params->tls_hostname->u.s; 115709d6c965SJuan Quintela } 115809d6c965SJuan Quintela 115909d6c965SJuan Quintela if (params->has_max_bandwidth) { 116009d6c965SJuan Quintela dest->max_bandwidth = params->max_bandwidth; 116109d6c965SJuan Quintela } 116209d6c965SJuan Quintela 116309d6c965SJuan Quintela if (params->has_downtime_limit) { 116409d6c965SJuan Quintela dest->downtime_limit = params->downtime_limit; 116509d6c965SJuan Quintela } 116609d6c965SJuan Quintela 116709d6c965SJuan Quintela if (params->has_x_checkpoint_delay) { 116809d6c965SJuan Quintela dest->x_checkpoint_delay = params->x_checkpoint_delay; 116909d6c965SJuan Quintela } 117009d6c965SJuan Quintela 117109d6c965SJuan Quintela if (params->has_block_incremental) { 117209d6c965SJuan Quintela dest->block_incremental = params->block_incremental; 117309d6c965SJuan Quintela } 117409d6c965SJuan Quintela if (params->has_multifd_channels) { 117509d6c965SJuan Quintela dest->multifd_channels = params->multifd_channels; 117609d6c965SJuan Quintela } 117709d6c965SJuan Quintela if (params->has_multifd_compression) { 117809d6c965SJuan Quintela dest->multifd_compression = params->multifd_compression; 117909d6c965SJuan Quintela } 118009d6c965SJuan Quintela if (params->has_xbzrle_cache_size) { 118109d6c965SJuan Quintela dest->xbzrle_cache_size = params->xbzrle_cache_size; 118209d6c965SJuan Quintela } 118309d6c965SJuan Quintela if (params->has_max_postcopy_bandwidth) { 118409d6c965SJuan Quintela dest->max_postcopy_bandwidth = params->max_postcopy_bandwidth; 118509d6c965SJuan Quintela } 118609d6c965SJuan Quintela if (params->has_max_cpu_throttle) { 118709d6c965SJuan Quintela dest->max_cpu_throttle = params->max_cpu_throttle; 118809d6c965SJuan Quintela } 118909d6c965SJuan Quintela if (params->has_announce_initial) { 119009d6c965SJuan Quintela dest->announce_initial = params->announce_initial; 119109d6c965SJuan Quintela } 119209d6c965SJuan Quintela if (params->has_announce_max) { 119309d6c965SJuan Quintela dest->announce_max = params->announce_max; 119409d6c965SJuan Quintela } 119509d6c965SJuan Quintela if (params->has_announce_rounds) { 119609d6c965SJuan Quintela dest->announce_rounds = params->announce_rounds; 119709d6c965SJuan Quintela } 119809d6c965SJuan Quintela if (params->has_announce_step) { 119909d6c965SJuan Quintela dest->announce_step = params->announce_step; 120009d6c965SJuan Quintela } 120109d6c965SJuan Quintela 120209d6c965SJuan Quintela if (params->has_block_bitmap_mapping) { 120309d6c965SJuan Quintela dest->has_block_bitmap_mapping = true; 120409d6c965SJuan Quintela dest->block_bitmap_mapping = params->block_bitmap_mapping; 120509d6c965SJuan Quintela } 120609d6c965SJuan Quintela } 120709d6c965SJuan Quintela 120809d6c965SJuan Quintela static void migrate_params_apply(MigrateSetParameters *params, Error **errp) 120909d6c965SJuan Quintela { 121009d6c965SJuan Quintela MigrationState *s = migrate_get_current(); 121109d6c965SJuan Quintela 121209d6c965SJuan Quintela /* TODO use QAPI_CLONE() instead of duplicating it inline */ 121309d6c965SJuan Quintela 121409d6c965SJuan Quintela if (params->has_compress_level) { 121509d6c965SJuan Quintela s->parameters.compress_level = params->compress_level; 121609d6c965SJuan Quintela } 121709d6c965SJuan Quintela 121809d6c965SJuan Quintela if (params->has_compress_threads) { 121909d6c965SJuan Quintela s->parameters.compress_threads = params->compress_threads; 122009d6c965SJuan Quintela } 122109d6c965SJuan Quintela 122209d6c965SJuan Quintela if (params->has_compress_wait_thread) { 122309d6c965SJuan Quintela s->parameters.compress_wait_thread = params->compress_wait_thread; 122409d6c965SJuan Quintela } 122509d6c965SJuan Quintela 122609d6c965SJuan Quintela if (params->has_decompress_threads) { 122709d6c965SJuan Quintela s->parameters.decompress_threads = params->decompress_threads; 122809d6c965SJuan Quintela } 122909d6c965SJuan Quintela 123009d6c965SJuan Quintela if (params->has_throttle_trigger_threshold) { 123109d6c965SJuan Quintela s->parameters.throttle_trigger_threshold = params->throttle_trigger_threshold; 123209d6c965SJuan Quintela } 123309d6c965SJuan Quintela 123409d6c965SJuan Quintela if (params->has_cpu_throttle_initial) { 123509d6c965SJuan Quintela s->parameters.cpu_throttle_initial = params->cpu_throttle_initial; 123609d6c965SJuan Quintela } 123709d6c965SJuan Quintela 123809d6c965SJuan Quintela if (params->has_cpu_throttle_increment) { 123909d6c965SJuan Quintela s->parameters.cpu_throttle_increment = params->cpu_throttle_increment; 124009d6c965SJuan Quintela } 124109d6c965SJuan Quintela 124209d6c965SJuan Quintela if (params->has_cpu_throttle_tailslow) { 124309d6c965SJuan Quintela s->parameters.cpu_throttle_tailslow = params->cpu_throttle_tailslow; 124409d6c965SJuan Quintela } 124509d6c965SJuan Quintela 124609d6c965SJuan Quintela if (params->tls_creds) { 124709d6c965SJuan Quintela g_free(s->parameters.tls_creds); 124809d6c965SJuan Quintela assert(params->tls_creds->type == QTYPE_QSTRING); 124909d6c965SJuan Quintela s->parameters.tls_creds = g_strdup(params->tls_creds->u.s); 125009d6c965SJuan Quintela } 125109d6c965SJuan Quintela 125209d6c965SJuan Quintela if (params->tls_hostname) { 125309d6c965SJuan Quintela g_free(s->parameters.tls_hostname); 125409d6c965SJuan Quintela assert(params->tls_hostname->type == QTYPE_QSTRING); 125509d6c965SJuan Quintela s->parameters.tls_hostname = g_strdup(params->tls_hostname->u.s); 125609d6c965SJuan Quintela } 125709d6c965SJuan Quintela 125809d6c965SJuan Quintela if (params->tls_authz) { 125909d6c965SJuan Quintela g_free(s->parameters.tls_authz); 126009d6c965SJuan Quintela assert(params->tls_authz->type == QTYPE_QSTRING); 126109d6c965SJuan Quintela s->parameters.tls_authz = g_strdup(params->tls_authz->u.s); 126209d6c965SJuan Quintela } 126309d6c965SJuan Quintela 126409d6c965SJuan Quintela if (params->has_max_bandwidth) { 126509d6c965SJuan Quintela s->parameters.max_bandwidth = params->max_bandwidth; 126609d6c965SJuan Quintela if (s->to_dst_file && !migration_in_postcopy()) { 1267e1fde0e0SJuan Quintela migration_rate_set(s->parameters.max_bandwidth); 126809d6c965SJuan Quintela } 126909d6c965SJuan Quintela } 127009d6c965SJuan Quintela 127109d6c965SJuan Quintela if (params->has_downtime_limit) { 127209d6c965SJuan Quintela s->parameters.downtime_limit = params->downtime_limit; 127309d6c965SJuan Quintela } 127409d6c965SJuan Quintela 127509d6c965SJuan Quintela if (params->has_x_checkpoint_delay) { 127609d6c965SJuan Quintela s->parameters.x_checkpoint_delay = params->x_checkpoint_delay; 12774332ffcdSVladimir Sementsov-Ogievskiy colo_checkpoint_delay_set(); 127809d6c965SJuan Quintela } 127909d6c965SJuan Quintela 128009d6c965SJuan Quintela if (params->has_block_incremental) { 128109d6c965SJuan Quintela s->parameters.block_incremental = params->block_incremental; 128209d6c965SJuan Quintela } 128309d6c965SJuan Quintela if (params->has_multifd_channels) { 128409d6c965SJuan Quintela s->parameters.multifd_channels = params->multifd_channels; 128509d6c965SJuan Quintela } 128609d6c965SJuan Quintela if (params->has_multifd_compression) { 128709d6c965SJuan Quintela s->parameters.multifd_compression = params->multifd_compression; 128809d6c965SJuan Quintela } 128909d6c965SJuan Quintela if (params->has_xbzrle_cache_size) { 129009d6c965SJuan Quintela s->parameters.xbzrle_cache_size = params->xbzrle_cache_size; 129109d6c965SJuan Quintela xbzrle_cache_resize(params->xbzrle_cache_size, errp); 129209d6c965SJuan Quintela } 129309d6c965SJuan Quintela if (params->has_max_postcopy_bandwidth) { 129409d6c965SJuan Quintela s->parameters.max_postcopy_bandwidth = params->max_postcopy_bandwidth; 129509d6c965SJuan Quintela if (s->to_dst_file && migration_in_postcopy()) { 1296e1fde0e0SJuan Quintela migration_rate_set(s->parameters.max_postcopy_bandwidth); 129709d6c965SJuan Quintela } 129809d6c965SJuan Quintela } 129909d6c965SJuan Quintela if (params->has_max_cpu_throttle) { 130009d6c965SJuan Quintela s->parameters.max_cpu_throttle = params->max_cpu_throttle; 130109d6c965SJuan Quintela } 130209d6c965SJuan Quintela if (params->has_announce_initial) { 130309d6c965SJuan Quintela s->parameters.announce_initial = params->announce_initial; 130409d6c965SJuan Quintela } 130509d6c965SJuan Quintela if (params->has_announce_max) { 130609d6c965SJuan Quintela s->parameters.announce_max = params->announce_max; 130709d6c965SJuan Quintela } 130809d6c965SJuan Quintela if (params->has_announce_rounds) { 130909d6c965SJuan Quintela s->parameters.announce_rounds = params->announce_rounds; 131009d6c965SJuan Quintela } 131109d6c965SJuan Quintela if (params->has_announce_step) { 131209d6c965SJuan Quintela s->parameters.announce_step = params->announce_step; 131309d6c965SJuan Quintela } 131409d6c965SJuan Quintela 131509d6c965SJuan Quintela if (params->has_block_bitmap_mapping) { 131609d6c965SJuan Quintela qapi_free_BitmapMigrationNodeAliasList( 131709d6c965SJuan Quintela s->parameters.block_bitmap_mapping); 131809d6c965SJuan Quintela 131909d6c965SJuan Quintela s->parameters.has_block_bitmap_mapping = true; 132009d6c965SJuan Quintela s->parameters.block_bitmap_mapping = 132109d6c965SJuan Quintela QAPI_CLONE(BitmapMigrationNodeAliasList, 132209d6c965SJuan Quintela params->block_bitmap_mapping); 132309d6c965SJuan Quintela } 132409d6c965SJuan Quintela } 132509d6c965SJuan Quintela 132609d6c965SJuan Quintela void qmp_migrate_set_parameters(MigrateSetParameters *params, Error **errp) 132709d6c965SJuan Quintela { 132809d6c965SJuan Quintela MigrationParameters tmp; 132909d6c965SJuan Quintela 133009d6c965SJuan Quintela /* TODO Rewrite "" to null instead */ 133109d6c965SJuan Quintela if (params->tls_creds 133209d6c965SJuan Quintela && params->tls_creds->type == QTYPE_QNULL) { 133309d6c965SJuan Quintela qobject_unref(params->tls_creds->u.n); 133409d6c965SJuan Quintela params->tls_creds->type = QTYPE_QSTRING; 133509d6c965SJuan Quintela params->tls_creds->u.s = strdup(""); 133609d6c965SJuan Quintela } 133709d6c965SJuan Quintela /* TODO Rewrite "" to null instead */ 133809d6c965SJuan Quintela if (params->tls_hostname 133909d6c965SJuan Quintela && params->tls_hostname->type == QTYPE_QNULL) { 134009d6c965SJuan Quintela qobject_unref(params->tls_hostname->u.n); 134109d6c965SJuan Quintela params->tls_hostname->type = QTYPE_QSTRING; 134209d6c965SJuan Quintela params->tls_hostname->u.s = strdup(""); 134309d6c965SJuan Quintela } 134409d6c965SJuan Quintela 134509d6c965SJuan Quintela migrate_params_test_apply(params, &tmp); 134609d6c965SJuan Quintela 134709d6c965SJuan Quintela if (!migrate_params_check(&tmp, errp)) { 134809d6c965SJuan Quintela /* Invalid parameter */ 134909d6c965SJuan Quintela return; 135009d6c965SJuan Quintela } 135109d6c965SJuan Quintela 135209d6c965SJuan Quintela migrate_params_apply(params, errp); 135309d6c965SJuan Quintela } 1354