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" 30*dc623955SHyman Huang(黄勇) #include "sysemu/kvm.h" 311f0776f1SJuan Quintela 3209d6c965SJuan Quintela /* Maximum migrate downtime set to 2000 seconds */ 3309d6c965SJuan Quintela #define MAX_MIGRATE_DOWNTIME_SECONDS 2000 3409d6c965SJuan Quintela #define MAX_MIGRATE_DOWNTIME (MAX_MIGRATE_DOWNTIME_SECONDS * 1000) 3509d6c965SJuan Quintela 36f9436522SJuan Quintela #define MAX_THROTTLE (128 << 20) /* Migration transfer speed throttling */ 37f9436522SJuan Quintela 38f9436522SJuan Quintela /* Time in milliseconds we are allowed to stop the source, 39f9436522SJuan Quintela * for sending the last part */ 40f9436522SJuan Quintela #define DEFAULT_MIGRATE_SET_DOWNTIME 300 41f9436522SJuan Quintela 42f9436522SJuan Quintela /* Default compression thread count */ 43f9436522SJuan Quintela #define DEFAULT_MIGRATE_COMPRESS_THREAD_COUNT 8 44f9436522SJuan Quintela /* Default decompression thread count, usually decompression is at 45f9436522SJuan Quintela * least 4 times as fast as compression.*/ 46f9436522SJuan Quintela #define DEFAULT_MIGRATE_DECOMPRESS_THREAD_COUNT 2 47f9436522SJuan Quintela /*0: means nocompress, 1: best speed, ... 9: best compress ratio */ 48f9436522SJuan Quintela #define DEFAULT_MIGRATE_COMPRESS_LEVEL 1 49f9436522SJuan Quintela /* Define default autoconverge cpu throttle migration parameters */ 50f9436522SJuan Quintela #define DEFAULT_MIGRATE_THROTTLE_TRIGGER_THRESHOLD 50 51f9436522SJuan Quintela #define DEFAULT_MIGRATE_CPU_THROTTLE_INITIAL 20 52f9436522SJuan Quintela #define DEFAULT_MIGRATE_CPU_THROTTLE_INCREMENT 10 53f9436522SJuan Quintela #define DEFAULT_MIGRATE_MAX_CPU_THROTTLE 99 54f9436522SJuan Quintela 55f9436522SJuan Quintela /* Migration XBZRLE default cache size */ 56f9436522SJuan Quintela #define DEFAULT_MIGRATE_XBZRLE_CACHE_SIZE (64 * 1024 * 1024) 57f9436522SJuan Quintela 58f9436522SJuan Quintela /* The delay time (in ms) between two COLO checkpoints */ 59f9436522SJuan Quintela #define DEFAULT_MIGRATE_X_CHECKPOINT_DELAY (200 * 100) 60f9436522SJuan Quintela #define DEFAULT_MIGRATE_MULTIFD_CHANNELS 2 61f9436522SJuan Quintela #define DEFAULT_MIGRATE_MULTIFD_COMPRESSION MULTIFD_COMPRESSION_NONE 62f9436522SJuan Quintela /* 0: means nocompress, 1: best speed, ... 9: best compress ratio */ 63f9436522SJuan Quintela #define DEFAULT_MIGRATE_MULTIFD_ZLIB_LEVEL 1 64f9436522SJuan Quintela /* 0: means nocompress, 1: best speed, ... 20: best compress ratio */ 65f9436522SJuan Quintela #define DEFAULT_MIGRATE_MULTIFD_ZSTD_LEVEL 1 66f9436522SJuan Quintela 67f9436522SJuan Quintela /* Background transfer rate for postcopy, 0 means unlimited, note 68f9436522SJuan Quintela * that page requests can still exceed this limit. 69f9436522SJuan Quintela */ 70f9436522SJuan Quintela #define DEFAULT_MIGRATE_MAX_POSTCOPY_BANDWIDTH 0 71f9436522SJuan Quintela 72f9436522SJuan Quintela /* 73f9436522SJuan Quintela * Parameters for self_announce_delay giving a stream of RARP/ARP 74f9436522SJuan Quintela * packets after migration. 75f9436522SJuan Quintela */ 76f9436522SJuan Quintela #define DEFAULT_MIGRATE_ANNOUNCE_INITIAL 50 77f9436522SJuan Quintela #define DEFAULT_MIGRATE_ANNOUNCE_MAX 550 78f9436522SJuan Quintela #define DEFAULT_MIGRATE_ANNOUNCE_ROUNDS 5 79f9436522SJuan Quintela #define DEFAULT_MIGRATE_ANNOUNCE_STEP 100 80f9436522SJuan Quintela 81f9436522SJuan Quintela #define DEFINE_PROP_MIG_CAP(name, x) \ 82f9436522SJuan Quintela DEFINE_PROP_BOOL(name, MigrationState, capabilities[x], false) 83f9436522SJuan Quintela 844d807857SHyman Huang(黄勇) #define DEFAULT_MIGRATE_VCPU_DIRTY_LIMIT_PERIOD 1000 /* milliseconds */ 8509f9ec99SHyman Huang(黄勇) #define DEFAULT_MIGRATE_VCPU_DIRTY_LIMIT 1 /* MB/s */ 864d807857SHyman Huang(黄勇) 87f9436522SJuan Quintela Property migration_properties[] = { 88f9436522SJuan Quintela DEFINE_PROP_BOOL("store-global-state", MigrationState, 89f9436522SJuan Quintela store_global_state, true), 90f9436522SJuan Quintela DEFINE_PROP_BOOL("send-configuration", MigrationState, 91f9436522SJuan Quintela send_configuration, true), 92f9436522SJuan Quintela DEFINE_PROP_BOOL("send-section-footer", MigrationState, 93f9436522SJuan Quintela send_section_footer, true), 94f9436522SJuan Quintela DEFINE_PROP_BOOL("decompress-error-check", MigrationState, 95f9436522SJuan Quintela decompress_error_check, true), 9677c259a4SJuan Quintela DEFINE_PROP_BOOL("multifd-flush-after-each-section", MigrationState, 97294e5a40SJuan Quintela multifd_flush_after_each_section, false), 98f9436522SJuan Quintela DEFINE_PROP_UINT8("x-clear-bitmap-shift", MigrationState, 99f9436522SJuan Quintela clear_bitmap_shift, CLEAR_BITMAP_SHIFT_DEFAULT), 100f9436522SJuan Quintela DEFINE_PROP_BOOL("x-preempt-pre-7-2", MigrationState, 101f9436522SJuan Quintela preempt_pre_7_2, false), 102f9436522SJuan Quintela 103f9436522SJuan Quintela /* Migration parameters */ 104f9436522SJuan Quintela DEFINE_PROP_UINT8("x-compress-level", MigrationState, 105f9436522SJuan Quintela parameters.compress_level, 106f9436522SJuan Quintela DEFAULT_MIGRATE_COMPRESS_LEVEL), 107f9436522SJuan Quintela DEFINE_PROP_UINT8("x-compress-threads", MigrationState, 108f9436522SJuan Quintela parameters.compress_threads, 109f9436522SJuan Quintela DEFAULT_MIGRATE_COMPRESS_THREAD_COUNT), 110f9436522SJuan Quintela DEFINE_PROP_BOOL("x-compress-wait-thread", MigrationState, 111f9436522SJuan Quintela parameters.compress_wait_thread, true), 112f9436522SJuan Quintela DEFINE_PROP_UINT8("x-decompress-threads", MigrationState, 113f9436522SJuan Quintela parameters.decompress_threads, 114f9436522SJuan Quintela DEFAULT_MIGRATE_DECOMPRESS_THREAD_COUNT), 115f9436522SJuan Quintela DEFINE_PROP_UINT8("x-throttle-trigger-threshold", MigrationState, 116f9436522SJuan Quintela parameters.throttle_trigger_threshold, 117f9436522SJuan Quintela DEFAULT_MIGRATE_THROTTLE_TRIGGER_THRESHOLD), 118f9436522SJuan Quintela DEFINE_PROP_UINT8("x-cpu-throttle-initial", MigrationState, 119f9436522SJuan Quintela parameters.cpu_throttle_initial, 120f9436522SJuan Quintela DEFAULT_MIGRATE_CPU_THROTTLE_INITIAL), 121f9436522SJuan Quintela DEFINE_PROP_UINT8("x-cpu-throttle-increment", MigrationState, 122f9436522SJuan Quintela parameters.cpu_throttle_increment, 123f9436522SJuan Quintela DEFAULT_MIGRATE_CPU_THROTTLE_INCREMENT), 124f9436522SJuan Quintela DEFINE_PROP_BOOL("x-cpu-throttle-tailslow", MigrationState, 125f9436522SJuan Quintela parameters.cpu_throttle_tailslow, false), 126f9436522SJuan Quintela DEFINE_PROP_SIZE("x-max-bandwidth", MigrationState, 127f9436522SJuan Quintela parameters.max_bandwidth, MAX_THROTTLE), 128f9436522SJuan Quintela DEFINE_PROP_UINT64("x-downtime-limit", MigrationState, 129f9436522SJuan Quintela parameters.downtime_limit, 130f9436522SJuan Quintela DEFAULT_MIGRATE_SET_DOWNTIME), 131f9436522SJuan Quintela DEFINE_PROP_UINT32("x-checkpoint-delay", MigrationState, 132f9436522SJuan Quintela parameters.x_checkpoint_delay, 133f9436522SJuan Quintela DEFAULT_MIGRATE_X_CHECKPOINT_DELAY), 134f9436522SJuan Quintela DEFINE_PROP_UINT8("multifd-channels", MigrationState, 135f9436522SJuan Quintela parameters.multifd_channels, 136f9436522SJuan Quintela DEFAULT_MIGRATE_MULTIFD_CHANNELS), 137f9436522SJuan Quintela DEFINE_PROP_MULTIFD_COMPRESSION("multifd-compression", MigrationState, 138f9436522SJuan Quintela parameters.multifd_compression, 139f9436522SJuan Quintela DEFAULT_MIGRATE_MULTIFD_COMPRESSION), 140f9436522SJuan Quintela DEFINE_PROP_UINT8("multifd-zlib-level", MigrationState, 141f9436522SJuan Quintela parameters.multifd_zlib_level, 142f9436522SJuan Quintela DEFAULT_MIGRATE_MULTIFD_ZLIB_LEVEL), 143f9436522SJuan Quintela DEFINE_PROP_UINT8("multifd-zstd-level", MigrationState, 144f9436522SJuan Quintela parameters.multifd_zstd_level, 145f9436522SJuan Quintela DEFAULT_MIGRATE_MULTIFD_ZSTD_LEVEL), 146f9436522SJuan Quintela DEFINE_PROP_SIZE("xbzrle-cache-size", MigrationState, 147f9436522SJuan Quintela parameters.xbzrle_cache_size, 148f9436522SJuan Quintela DEFAULT_MIGRATE_XBZRLE_CACHE_SIZE), 149f9436522SJuan Quintela DEFINE_PROP_SIZE("max-postcopy-bandwidth", MigrationState, 150f9436522SJuan Quintela parameters.max_postcopy_bandwidth, 151f9436522SJuan Quintela DEFAULT_MIGRATE_MAX_POSTCOPY_BANDWIDTH), 152f9436522SJuan Quintela DEFINE_PROP_UINT8("max-cpu-throttle", MigrationState, 153f9436522SJuan Quintela parameters.max_cpu_throttle, 154f9436522SJuan Quintela DEFAULT_MIGRATE_MAX_CPU_THROTTLE), 155f9436522SJuan Quintela DEFINE_PROP_SIZE("announce-initial", MigrationState, 156f9436522SJuan Quintela parameters.announce_initial, 157f9436522SJuan Quintela DEFAULT_MIGRATE_ANNOUNCE_INITIAL), 158f9436522SJuan Quintela DEFINE_PROP_SIZE("announce-max", MigrationState, 159f9436522SJuan Quintela parameters.announce_max, 160f9436522SJuan Quintela DEFAULT_MIGRATE_ANNOUNCE_MAX), 161f9436522SJuan Quintela DEFINE_PROP_SIZE("announce-rounds", MigrationState, 162f9436522SJuan Quintela parameters.announce_rounds, 163f9436522SJuan Quintela DEFAULT_MIGRATE_ANNOUNCE_ROUNDS), 164f9436522SJuan Quintela DEFINE_PROP_SIZE("announce-step", MigrationState, 165f9436522SJuan Quintela parameters.announce_step, 166f9436522SJuan Quintela DEFAULT_MIGRATE_ANNOUNCE_STEP), 167f9436522SJuan Quintela DEFINE_PROP_STRING("tls-creds", MigrationState, parameters.tls_creds), 168f9436522SJuan Quintela DEFINE_PROP_STRING("tls-hostname", MigrationState, parameters.tls_hostname), 169f9436522SJuan Quintela DEFINE_PROP_STRING("tls-authz", MigrationState, parameters.tls_authz), 1704d807857SHyman Huang(黄勇) DEFINE_PROP_UINT64("x-vcpu-dirty-limit-period", MigrationState, 1714d807857SHyman Huang(黄勇) parameters.x_vcpu_dirty_limit_period, 1724d807857SHyman Huang(黄勇) DEFAULT_MIGRATE_VCPU_DIRTY_LIMIT_PERIOD), 17309f9ec99SHyman Huang(黄勇) DEFINE_PROP_UINT64("vcpu-dirty-limit", MigrationState, 17409f9ec99SHyman Huang(黄勇) parameters.vcpu_dirty_limit, 17509f9ec99SHyman Huang(黄勇) DEFAULT_MIGRATE_VCPU_DIRTY_LIMIT), 176f9436522SJuan Quintela 177f9436522SJuan Quintela /* Migration capabilities */ 178f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-xbzrle", MIGRATION_CAPABILITY_XBZRLE), 179f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-rdma-pin-all", MIGRATION_CAPABILITY_RDMA_PIN_ALL), 180f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-auto-converge", MIGRATION_CAPABILITY_AUTO_CONVERGE), 181f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-zero-blocks", MIGRATION_CAPABILITY_ZERO_BLOCKS), 182f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-compress", MIGRATION_CAPABILITY_COMPRESS), 183f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-events", MIGRATION_CAPABILITY_EVENTS), 184f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-postcopy-ram", MIGRATION_CAPABILITY_POSTCOPY_RAM), 185f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-postcopy-preempt", 186f9436522SJuan Quintela MIGRATION_CAPABILITY_POSTCOPY_PREEMPT), 187f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-colo", MIGRATION_CAPABILITY_X_COLO), 188f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-release-ram", MIGRATION_CAPABILITY_RELEASE_RAM), 189f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-block", MIGRATION_CAPABILITY_BLOCK), 190f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-return-path", MIGRATION_CAPABILITY_RETURN_PATH), 191f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-multifd", MIGRATION_CAPABILITY_MULTIFD), 192f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-background-snapshot", 193f9436522SJuan Quintela MIGRATION_CAPABILITY_BACKGROUND_SNAPSHOT), 194f9436522SJuan Quintela #ifdef CONFIG_LINUX 195f9436522SJuan Quintela DEFINE_PROP_MIG_CAP("x-zero-copy-send", 196f9436522SJuan Quintela MIGRATION_CAPABILITY_ZERO_COPY_SEND), 197f9436522SJuan Quintela #endif 1986574232fSAvihai Horon DEFINE_PROP_MIG_CAP("x-switchover-ack", 1996574232fSAvihai Horon MIGRATION_CAPABILITY_SWITCHOVER_ACK), 200*dc623955SHyman Huang(黄勇) DEFINE_PROP_MIG_CAP("x-dirty-limit", MIGRATION_CAPABILITY_DIRTY_LIMIT), 201f9436522SJuan Quintela DEFINE_PROP_END_OF_LIST(), 202f9436522SJuan Quintela }; 203f9436522SJuan Quintela 2041f0776f1SJuan Quintela bool migrate_auto_converge(void) 2051f0776f1SJuan Quintela { 2068f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 2071f0776f1SJuan Quintela 2081f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_AUTO_CONVERGE]; 2091f0776f1SJuan Quintela } 2101f0776f1SJuan Quintela 2111f0776f1SJuan Quintela bool migrate_background_snapshot(void) 2121f0776f1SJuan Quintela { 2138f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 2141f0776f1SJuan Quintela 2151f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_BACKGROUND_SNAPSHOT]; 2161f0776f1SJuan Quintela } 2171f0776f1SJuan Quintela 2189d4b1e5fSJuan Quintela bool migrate_block(void) 2199d4b1e5fSJuan Quintela { 2208f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 2219d4b1e5fSJuan Quintela 2229d4b1e5fSJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_BLOCK]; 2239d4b1e5fSJuan Quintela } 2249d4b1e5fSJuan Quintela 2255e804644SJuan Quintela bool migrate_colo(void) 2265e804644SJuan Quintela { 2275e804644SJuan Quintela MigrationState *s = migrate_get_current(); 2288f9c5327SJuan Quintela 2295e804644SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_X_COLO]; 2305e804644SJuan Quintela } 2315e804644SJuan Quintela 232a7a94d14SJuan Quintela bool migrate_compress(void) 233a7a94d14SJuan Quintela { 2348f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 235a7a94d14SJuan Quintela 236a7a94d14SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_COMPRESS]; 237a7a94d14SJuan Quintela } 238a7a94d14SJuan Quintela 2391f0776f1SJuan Quintela bool migrate_dirty_bitmaps(void) 2401f0776f1SJuan Quintela { 2418f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 2421f0776f1SJuan Quintela 2431f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_DIRTY_BITMAPS]; 2441f0776f1SJuan Quintela } 2451f0776f1SJuan Quintela 246*dc623955SHyman Huang(黄勇) bool migrate_dirty_limit(void) 247*dc623955SHyman Huang(黄勇) { 248*dc623955SHyman Huang(黄勇) MigrationState *s = migrate_get_current(); 249*dc623955SHyman Huang(黄勇) 250*dc623955SHyman Huang(黄勇) return s->capabilities[MIGRATION_CAPABILITY_DIRTY_LIMIT]; 251*dc623955SHyman Huang(黄勇) } 252*dc623955SHyman Huang(黄勇) 253b890902cSJuan Quintela bool migrate_events(void) 254b890902cSJuan Quintela { 2558f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 256b890902cSJuan Quintela 257b890902cSJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_EVENTS]; 258b890902cSJuan Quintela } 259b890902cSJuan Quintela 2601f0776f1SJuan Quintela bool migrate_ignore_shared(void) 2611f0776f1SJuan Quintela { 2628f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 2631f0776f1SJuan Quintela 2641f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_X_IGNORE_SHARED]; 2651f0776f1SJuan Quintela } 2661f0776f1SJuan Quintela 2671f0776f1SJuan Quintela bool migrate_late_block_activate(void) 2681f0776f1SJuan Quintela { 2698f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 2701f0776f1SJuan Quintela 2711f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_LATE_BLOCK_ACTIVATE]; 2721f0776f1SJuan Quintela } 2731f0776f1SJuan Quintela 27451b07548SJuan Quintela bool migrate_multifd(void) 27551b07548SJuan Quintela { 2768f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 27751b07548SJuan Quintela 27851b07548SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_MULTIFD]; 27951b07548SJuan Quintela } 28051b07548SJuan Quintela 2811f0776f1SJuan Quintela bool migrate_pause_before_switchover(void) 2821f0776f1SJuan Quintela { 2838f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 2841f0776f1SJuan Quintela 2851f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_PAUSE_BEFORE_SWITCHOVER]; 2861f0776f1SJuan Quintela } 2871f0776f1SJuan Quintela 2881f0776f1SJuan Quintela bool migrate_postcopy_blocktime(void) 2891f0776f1SJuan Quintela { 2908f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 2911f0776f1SJuan Quintela 2921f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_POSTCOPY_BLOCKTIME]; 2931f0776f1SJuan Quintela } 2941f0776f1SJuan Quintela 2951f0776f1SJuan Quintela bool migrate_postcopy_preempt(void) 2961f0776f1SJuan Quintela { 2978f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 2981f0776f1SJuan Quintela 2991f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_POSTCOPY_PREEMPT]; 3001f0776f1SJuan Quintela } 3011f0776f1SJuan Quintela 3021f0776f1SJuan Quintela bool migrate_postcopy_ram(void) 3031f0776f1SJuan Quintela { 3048f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 3051f0776f1SJuan Quintela 3061f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_POSTCOPY_RAM]; 3071f0776f1SJuan Quintela } 3081f0776f1SJuan Quintela 30917cba690SJuan Quintela bool migrate_rdma_pin_all(void) 31017cba690SJuan Quintela { 31117cba690SJuan Quintela MigrationState *s = migrate_get_current(); 31217cba690SJuan Quintela 31317cba690SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_RDMA_PIN_ALL]; 31417cba690SJuan Quintela } 31517cba690SJuan Quintela 3161f0776f1SJuan Quintela bool migrate_release_ram(void) 3171f0776f1SJuan Quintela { 3188f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 3191f0776f1SJuan Quintela 3201f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_RELEASE_RAM]; 3211f0776f1SJuan Quintela } 3221f0776f1SJuan Quintela 32338ad1110SJuan Quintela bool migrate_return_path(void) 32438ad1110SJuan Quintela { 3258f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 32638ad1110SJuan Quintela 32738ad1110SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_RETURN_PATH]; 32838ad1110SJuan Quintela } 32938ad1110SJuan Quintela 3306574232fSAvihai Horon bool migrate_switchover_ack(void) 3316574232fSAvihai Horon { 3326574232fSAvihai Horon MigrationState *s = migrate_get_current(); 3336574232fSAvihai Horon 3346574232fSAvihai Horon return s->capabilities[MIGRATION_CAPABILITY_SWITCHOVER_ACK]; 3356574232fSAvihai Horon } 3366574232fSAvihai Horon 3371f0776f1SJuan Quintela bool migrate_validate_uuid(void) 3381f0776f1SJuan Quintela { 3398f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 3401f0776f1SJuan Quintela 3411f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_VALIDATE_UUID]; 3421f0776f1SJuan Quintela } 3431f0776f1SJuan Quintela 34487dca0c9SJuan Quintela bool migrate_xbzrle(void) 34587dca0c9SJuan Quintela { 3468f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 34787dca0c9SJuan Quintela 34887dca0c9SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_XBZRLE]; 34987dca0c9SJuan Quintela } 35087dca0c9SJuan Quintela 3511f0776f1SJuan Quintela bool migrate_zero_blocks(void) 3521f0776f1SJuan Quintela { 3538f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 3541f0776f1SJuan Quintela 3551f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_ZERO_BLOCKS]; 3561f0776f1SJuan Quintela } 357b4bc342cSJuan Quintela 358b4bc342cSJuan Quintela bool migrate_zero_copy_send(void) 359b4bc342cSJuan Quintela { 3608f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 361b4bc342cSJuan Quintela 362b4bc342cSJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_ZERO_COPY_SEND]; 363b4bc342cSJuan Quintela } 364f774fde5SJuan Quintela 365f774fde5SJuan Quintela /* pseudo capabilities */ 366f774fde5SJuan Quintela 36777c259a4SJuan Quintela bool migrate_multifd_flush_after_each_section(void) 36877c259a4SJuan Quintela { 36977c259a4SJuan Quintela MigrationState *s = migrate_get_current(); 37077c259a4SJuan Quintela 371294e5a40SJuan Quintela return s->multifd_flush_after_each_section; 37277c259a4SJuan Quintela } 37377c259a4SJuan Quintela 374f774fde5SJuan Quintela bool migrate_postcopy(void) 375f774fde5SJuan Quintela { 376f774fde5SJuan Quintela return migrate_postcopy_ram() || migrate_dirty_bitmaps(); 377f774fde5SJuan Quintela } 378f774fde5SJuan Quintela 37910d4703bSJuan Quintela bool migrate_tls(void) 38010d4703bSJuan Quintela { 3818f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 38210d4703bSJuan Quintela 38310d4703bSJuan Quintela return s->parameters.tls_creds && *s->parameters.tls_creds; 38410d4703bSJuan Quintela } 38510d4703bSJuan Quintela 38677608706SJuan Quintela typedef enum WriteTrackingSupport { 38777608706SJuan Quintela WT_SUPPORT_UNKNOWN = 0, 38877608706SJuan Quintela WT_SUPPORT_ABSENT, 38977608706SJuan Quintela WT_SUPPORT_AVAILABLE, 39077608706SJuan Quintela WT_SUPPORT_COMPATIBLE 39177608706SJuan Quintela } WriteTrackingSupport; 39277608706SJuan Quintela 39377608706SJuan Quintela static 39477608706SJuan Quintela WriteTrackingSupport migrate_query_write_tracking(void) 39577608706SJuan Quintela { 39677608706SJuan Quintela /* Check if kernel supports required UFFD features */ 39777608706SJuan Quintela if (!ram_write_tracking_available()) { 39877608706SJuan Quintela return WT_SUPPORT_ABSENT; 39977608706SJuan Quintela } 40077608706SJuan Quintela /* 40177608706SJuan Quintela * Check if current memory configuration is 40277608706SJuan Quintela * compatible with required UFFD features. 40377608706SJuan Quintela */ 40477608706SJuan Quintela if (!ram_write_tracking_compatible()) { 40577608706SJuan Quintela return WT_SUPPORT_AVAILABLE; 40677608706SJuan Quintela } 40777608706SJuan Quintela 40877608706SJuan Quintela return WT_SUPPORT_COMPATIBLE; 40977608706SJuan Quintela } 41077608706SJuan Quintela 41177608706SJuan Quintela /* Migration capabilities set */ 41277608706SJuan Quintela struct MigrateCapsSet { 41377608706SJuan Quintela int size; /* Capability set size */ 41477608706SJuan Quintela MigrationCapability caps[]; /* Variadic array of capabilities */ 41577608706SJuan Quintela }; 41677608706SJuan Quintela typedef struct MigrateCapsSet MigrateCapsSet; 41777608706SJuan Quintela 41877608706SJuan Quintela /* Define and initialize MigrateCapsSet */ 41977608706SJuan Quintela #define INITIALIZE_MIGRATE_CAPS_SET(_name, ...) \ 42077608706SJuan Quintela MigrateCapsSet _name = { \ 42177608706SJuan Quintela .size = sizeof((int []) { __VA_ARGS__ }) / sizeof(int), \ 42277608706SJuan Quintela .caps = { __VA_ARGS__ } \ 42377608706SJuan Quintela } 42477608706SJuan Quintela 42577608706SJuan Quintela /* Background-snapshot compatibility check list */ 42677608706SJuan Quintela static const 42777608706SJuan Quintela INITIALIZE_MIGRATE_CAPS_SET(check_caps_background_snapshot, 42877608706SJuan Quintela MIGRATION_CAPABILITY_POSTCOPY_RAM, 42977608706SJuan Quintela MIGRATION_CAPABILITY_DIRTY_BITMAPS, 43077608706SJuan Quintela MIGRATION_CAPABILITY_POSTCOPY_BLOCKTIME, 43177608706SJuan Quintela MIGRATION_CAPABILITY_LATE_BLOCK_ACTIVATE, 43277608706SJuan Quintela MIGRATION_CAPABILITY_RETURN_PATH, 43377608706SJuan Quintela MIGRATION_CAPABILITY_MULTIFD, 43477608706SJuan Quintela MIGRATION_CAPABILITY_PAUSE_BEFORE_SWITCHOVER, 43577608706SJuan Quintela MIGRATION_CAPABILITY_AUTO_CONVERGE, 43677608706SJuan Quintela MIGRATION_CAPABILITY_RELEASE_RAM, 43777608706SJuan Quintela MIGRATION_CAPABILITY_RDMA_PIN_ALL, 43877608706SJuan Quintela MIGRATION_CAPABILITY_COMPRESS, 43977608706SJuan Quintela MIGRATION_CAPABILITY_XBZRLE, 44077608706SJuan Quintela MIGRATION_CAPABILITY_X_COLO, 44177608706SJuan Quintela MIGRATION_CAPABILITY_VALIDATE_UUID, 44277608706SJuan Quintela MIGRATION_CAPABILITY_ZERO_COPY_SEND); 44377608706SJuan Quintela 44477608706SJuan Quintela /** 44577608706SJuan Quintela * @migration_caps_check - check capability compatibility 44677608706SJuan Quintela * 44777608706SJuan Quintela * @old_caps: old capability list 44877608706SJuan Quintela * @new_caps: new capability list 44977608706SJuan Quintela * @errp: set *errp if the check failed, with reason 45077608706SJuan Quintela * 45177608706SJuan Quintela * Returns true if check passed, otherwise false. 45277608706SJuan Quintela */ 45377608706SJuan Quintela bool migrate_caps_check(bool *old_caps, bool *new_caps, Error **errp) 45477608706SJuan Quintela { 45577608706SJuan Quintela MigrationIncomingState *mis = migration_incoming_get_current(); 45677608706SJuan Quintela 45774c38cf7SPeter Xu ERRP_GUARD(); 45877608706SJuan Quintela #ifndef CONFIG_LIVE_BLOCK_MIGRATION 45977608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_BLOCK]) { 46077608706SJuan Quintela error_setg(errp, "QEMU compiled without old-style (blk/-b, inc/-i) " 46177608706SJuan Quintela "block migration"); 46277608706SJuan Quintela error_append_hint(errp, "Use drive_mirror+NBD instead.\n"); 46377608706SJuan Quintela return false; 46477608706SJuan Quintela } 46577608706SJuan Quintela #endif 46677608706SJuan Quintela 46777608706SJuan Quintela #ifndef CONFIG_REPLICATION 46877608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_X_COLO]) { 46977608706SJuan Quintela error_setg(errp, "QEMU compiled without replication module" 47077608706SJuan Quintela " can't enable COLO"); 47177608706SJuan Quintela error_append_hint(errp, "Please enable replication before COLO.\n"); 47277608706SJuan Quintela return false; 47377608706SJuan Quintela } 47477608706SJuan Quintela #endif 47577608706SJuan Quintela 47677608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_POSTCOPY_RAM]) { 47777608706SJuan Quintela /* This check is reasonably expensive, so only when it's being 47877608706SJuan Quintela * set the first time, also it's only the destination that needs 47977608706SJuan Quintela * special support. 48077608706SJuan Quintela */ 48177608706SJuan Quintela if (!old_caps[MIGRATION_CAPABILITY_POSTCOPY_RAM] && 48277608706SJuan Quintela runstate_check(RUN_STATE_INMIGRATE) && 48374c38cf7SPeter Xu !postcopy_ram_supported_by_host(mis, errp)) { 48474c38cf7SPeter Xu error_prepend(errp, "Postcopy is not supported: "); 48577608706SJuan Quintela return false; 48677608706SJuan Quintela } 48777608706SJuan Quintela 48877608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_X_IGNORE_SHARED]) { 48977608706SJuan Quintela error_setg(errp, "Postcopy is not compatible with ignore-shared"); 49077608706SJuan Quintela return false; 49177608706SJuan Quintela } 492b405dfffSLeonardo Bras 493b405dfffSLeonardo Bras if (new_caps[MIGRATION_CAPABILITY_MULTIFD]) { 494b405dfffSLeonardo Bras error_setg(errp, "Postcopy is not yet compatible with multifd"); 495b405dfffSLeonardo Bras return false; 496b405dfffSLeonardo Bras } 49777608706SJuan Quintela } 49877608706SJuan Quintela 49977608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_BACKGROUND_SNAPSHOT]) { 50077608706SJuan Quintela WriteTrackingSupport wt_support; 50177608706SJuan Quintela int idx; 50277608706SJuan Quintela /* 50377608706SJuan Quintela * Check if 'background-snapshot' capability is supported by 50477608706SJuan Quintela * host kernel and compatible with guest memory configuration. 50577608706SJuan Quintela */ 50677608706SJuan Quintela wt_support = migrate_query_write_tracking(); 50777608706SJuan Quintela if (wt_support < WT_SUPPORT_AVAILABLE) { 50877608706SJuan Quintela error_setg(errp, "Background-snapshot is not supported by host kernel"); 50977608706SJuan Quintela return false; 51077608706SJuan Quintela } 51177608706SJuan Quintela if (wt_support < WT_SUPPORT_COMPATIBLE) { 51277608706SJuan Quintela error_setg(errp, "Background-snapshot is not compatible " 51377608706SJuan Quintela "with guest memory configuration"); 51477608706SJuan Quintela return false; 51577608706SJuan Quintela } 51677608706SJuan Quintela 51777608706SJuan Quintela /* 51877608706SJuan Quintela * Check if there are any migration capabilities 51977608706SJuan Quintela * incompatible with 'background-snapshot'. 52077608706SJuan Quintela */ 52177608706SJuan Quintela for (idx = 0; idx < check_caps_background_snapshot.size; idx++) { 52277608706SJuan Quintela int incomp_cap = check_caps_background_snapshot.caps[idx]; 52377608706SJuan Quintela if (new_caps[incomp_cap]) { 52477608706SJuan Quintela error_setg(errp, 52577608706SJuan Quintela "Background-snapshot is not compatible with %s", 52677608706SJuan Quintela MigrationCapability_str(incomp_cap)); 52777608706SJuan Quintela return false; 52877608706SJuan Quintela } 52977608706SJuan Quintela } 53077608706SJuan Quintela } 53177608706SJuan Quintela 53277608706SJuan Quintela #ifdef CONFIG_LINUX 53377608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_ZERO_COPY_SEND] && 53477608706SJuan Quintela (!new_caps[MIGRATION_CAPABILITY_MULTIFD] || 53577608706SJuan Quintela new_caps[MIGRATION_CAPABILITY_COMPRESS] || 53677608706SJuan Quintela new_caps[MIGRATION_CAPABILITY_XBZRLE] || 53777608706SJuan Quintela migrate_multifd_compression() || 53810d4703bSJuan Quintela migrate_tls())) { 53977608706SJuan Quintela error_setg(errp, 54077608706SJuan Quintela "Zero copy only available for non-compressed non-TLS multifd migration"); 54177608706SJuan Quintela return false; 54277608706SJuan Quintela } 54377608706SJuan Quintela #else 54477608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_ZERO_COPY_SEND]) { 54577608706SJuan Quintela error_setg(errp, 54677608706SJuan Quintela "Zero copy currently only available on Linux"); 54777608706SJuan Quintela return false; 54877608706SJuan Quintela } 54977608706SJuan Quintela #endif 55077608706SJuan Quintela 55177608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_POSTCOPY_PREEMPT]) { 55277608706SJuan Quintela if (!new_caps[MIGRATION_CAPABILITY_POSTCOPY_RAM]) { 55377608706SJuan Quintela error_setg(errp, "Postcopy preempt requires postcopy-ram"); 55477608706SJuan Quintela return false; 55577608706SJuan Quintela } 55677608706SJuan Quintela 55777608706SJuan Quintela /* 55877608706SJuan Quintela * Preempt mode requires urgent pages to be sent in separate 55977608706SJuan Quintela * channel, OTOH compression logic will disorder all pages into 56077608706SJuan Quintela * different compression channels, which is not compatible with the 56177608706SJuan Quintela * preempt assumptions on channel assignments. 56277608706SJuan Quintela */ 56377608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_COMPRESS]) { 56477608706SJuan Quintela error_setg(errp, "Postcopy preempt not compatible with compress"); 56577608706SJuan Quintela return false; 56677608706SJuan Quintela } 56777608706SJuan Quintela } 56877608706SJuan Quintela 56977608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_MULTIFD]) { 57077608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_COMPRESS]) { 57177608706SJuan Quintela error_setg(errp, "Multifd is not compatible with compress"); 57277608706SJuan Quintela return false; 57377608706SJuan Quintela } 57477608706SJuan Quintela } 57577608706SJuan Quintela 5766574232fSAvihai Horon if (new_caps[MIGRATION_CAPABILITY_SWITCHOVER_ACK]) { 5776574232fSAvihai Horon if (!new_caps[MIGRATION_CAPABILITY_RETURN_PATH]) { 5786574232fSAvihai Horon error_setg(errp, "Capability 'switchover-ack' requires capability " 5796574232fSAvihai Horon "'return-path'"); 5806574232fSAvihai Horon return false; 5816574232fSAvihai Horon } 5826574232fSAvihai Horon } 583*dc623955SHyman Huang(黄勇) if (new_caps[MIGRATION_CAPABILITY_DIRTY_LIMIT]) { 584*dc623955SHyman Huang(黄勇) if (new_caps[MIGRATION_CAPABILITY_AUTO_CONVERGE]) { 585*dc623955SHyman Huang(黄勇) error_setg(errp, "dirty-limit conflicts with auto-converge" 586*dc623955SHyman Huang(黄勇) " either of then available currently"); 587*dc623955SHyman Huang(黄勇) return false; 588*dc623955SHyman Huang(黄勇) } 589*dc623955SHyman Huang(黄勇) 590*dc623955SHyman Huang(黄勇) if (!kvm_enabled() || !kvm_dirty_ring_enabled()) { 591*dc623955SHyman Huang(黄勇) error_setg(errp, "dirty-limit requires KVM with accelerator" 592*dc623955SHyman Huang(黄勇) " property 'dirty-ring-size' set"); 593*dc623955SHyman Huang(黄勇) return false; 594*dc623955SHyman Huang(黄勇) } 595*dc623955SHyman Huang(黄勇) } 5966574232fSAvihai Horon 59777608706SJuan Quintela return true; 59877608706SJuan Quintela } 5994d0c6b69SJuan Quintela 600f80196b7SJuan Quintela bool migrate_cap_set(int cap, bool value, Error **errp) 601f80196b7SJuan Quintela { 602f80196b7SJuan Quintela MigrationState *s = migrate_get_current(); 603f80196b7SJuan Quintela bool new_caps[MIGRATION_CAPABILITY__MAX]; 604f80196b7SJuan Quintela 605f80196b7SJuan Quintela if (migration_is_running(s->state)) { 606f80196b7SJuan Quintela error_setg(errp, QERR_MIGRATION_ACTIVE); 607f80196b7SJuan Quintela return false; 608f80196b7SJuan Quintela } 609f80196b7SJuan Quintela 610f80196b7SJuan Quintela memcpy(new_caps, s->capabilities, sizeof(new_caps)); 611f80196b7SJuan Quintela new_caps[cap] = value; 612f80196b7SJuan Quintela 613f80196b7SJuan Quintela if (!migrate_caps_check(s->capabilities, new_caps, errp)) { 614f80196b7SJuan Quintela return false; 615f80196b7SJuan Quintela } 616f80196b7SJuan Quintela s->capabilities[cap] = value; 617f80196b7SJuan Quintela return true; 618f80196b7SJuan Quintela } 619f80196b7SJuan Quintela 6204d0c6b69SJuan Quintela MigrationCapabilityStatusList *qmp_query_migrate_capabilities(Error **errp) 6214d0c6b69SJuan Quintela { 6224d0c6b69SJuan Quintela MigrationCapabilityStatusList *head = NULL, **tail = &head; 6234d0c6b69SJuan Quintela MigrationCapabilityStatus *caps; 6244d0c6b69SJuan Quintela MigrationState *s = migrate_get_current(); 6254d0c6b69SJuan Quintela int i; 6264d0c6b69SJuan Quintela 6274d0c6b69SJuan Quintela for (i = 0; i < MIGRATION_CAPABILITY__MAX; i++) { 6284d0c6b69SJuan Quintela #ifndef CONFIG_LIVE_BLOCK_MIGRATION 6294d0c6b69SJuan Quintela if (i == MIGRATION_CAPABILITY_BLOCK) { 6304d0c6b69SJuan Quintela continue; 6314d0c6b69SJuan Quintela } 6324d0c6b69SJuan Quintela #endif 6334d0c6b69SJuan Quintela caps = g_malloc0(sizeof(*caps)); 6344d0c6b69SJuan Quintela caps->capability = i; 6354d0c6b69SJuan Quintela caps->state = s->capabilities[i]; 6364d0c6b69SJuan Quintela QAPI_LIST_APPEND(tail, caps); 6374d0c6b69SJuan Quintela } 6384d0c6b69SJuan Quintela 6394d0c6b69SJuan Quintela return head; 6404d0c6b69SJuan Quintela } 64145c1de13SJuan Quintela 64245c1de13SJuan Quintela void qmp_migrate_set_capabilities(MigrationCapabilityStatusList *params, 64345c1de13SJuan Quintela Error **errp) 64445c1de13SJuan Quintela { 64545c1de13SJuan Quintela MigrationState *s = migrate_get_current(); 64645c1de13SJuan Quintela MigrationCapabilityStatusList *cap; 64745c1de13SJuan Quintela bool new_caps[MIGRATION_CAPABILITY__MAX]; 64845c1de13SJuan Quintela 649d70178a8SVladimir Sementsov-Ogievskiy if (migration_is_running(s->state) || migration_in_colo_state()) { 65045c1de13SJuan Quintela error_setg(errp, QERR_MIGRATION_ACTIVE); 65145c1de13SJuan Quintela return; 65245c1de13SJuan Quintela } 65345c1de13SJuan Quintela 65445c1de13SJuan Quintela memcpy(new_caps, s->capabilities, sizeof(new_caps)); 65545c1de13SJuan Quintela for (cap = params; cap; cap = cap->next) { 65645c1de13SJuan Quintela new_caps[cap->value->capability] = cap->value->state; 65745c1de13SJuan Quintela } 65845c1de13SJuan Quintela 65945c1de13SJuan Quintela if (!migrate_caps_check(s->capabilities, new_caps, errp)) { 66045c1de13SJuan Quintela return; 66145c1de13SJuan Quintela } 66245c1de13SJuan Quintela 66345c1de13SJuan Quintela for (cap = params; cap; cap = cap->next) { 66445c1de13SJuan Quintela s->capabilities[cap->value->capability] = cap->value->state; 66545c1de13SJuan Quintela } 66645c1de13SJuan Quintela } 6671dfc4b9eSJuan Quintela 6681dfc4b9eSJuan Quintela /* parameters */ 6691dfc4b9eSJuan Quintela 670b804b35bSJuan Quintela const BitmapMigrationNodeAliasList *migrate_block_bitmap_mapping(void) 671b804b35bSJuan Quintela { 672b804b35bSJuan Quintela MigrationState *s = migrate_get_current(); 673b804b35bSJuan Quintela 674b804b35bSJuan Quintela return s->parameters.block_bitmap_mapping; 675b804b35bSJuan Quintela } 676b804b35bSJuan Quintela 6773cba22c9SJuan Quintela bool migrate_has_block_bitmap_mapping(void) 6783cba22c9SJuan Quintela { 6793cba22c9SJuan Quintela MigrationState *s = migrate_get_current(); 6803cba22c9SJuan Quintela 6813cba22c9SJuan Quintela return s->parameters.has_block_bitmap_mapping; 6823cba22c9SJuan Quintela } 6833cba22c9SJuan Quintela 6846f8be708SJuan Quintela bool migrate_block_incremental(void) 6856f8be708SJuan Quintela { 6868f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 6876f8be708SJuan Quintela 6886f8be708SJuan Quintela return s->parameters.block_incremental; 6896f8be708SJuan Quintela } 6906f8be708SJuan Quintela 691f94a858fSJuan Quintela uint32_t migrate_checkpoint_delay(void) 692f94a858fSJuan Quintela { 6938f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 694f94a858fSJuan Quintela 695f94a858fSJuan Quintela return s->parameters.x_checkpoint_delay; 696f94a858fSJuan Quintela } 697f94a858fSJuan Quintela 6981dfc4b9eSJuan Quintela int migrate_compress_level(void) 6991dfc4b9eSJuan Quintela { 7008f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7011dfc4b9eSJuan Quintela 7021dfc4b9eSJuan Quintela return s->parameters.compress_level; 7031dfc4b9eSJuan Quintela } 7041dfc4b9eSJuan Quintela 7051dfc4b9eSJuan Quintela int migrate_compress_threads(void) 7061dfc4b9eSJuan Quintela { 7078f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7081dfc4b9eSJuan Quintela 7091dfc4b9eSJuan Quintela return s->parameters.compress_threads; 7101dfc4b9eSJuan Quintela } 7111dfc4b9eSJuan Quintela 7121dfc4b9eSJuan Quintela int migrate_compress_wait_thread(void) 7131dfc4b9eSJuan Quintela { 7148f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7151dfc4b9eSJuan Quintela 7161dfc4b9eSJuan Quintela return s->parameters.compress_wait_thread; 7171dfc4b9eSJuan Quintela } 7181dfc4b9eSJuan Quintela 7199605c2acSJuan Quintela uint8_t migrate_cpu_throttle_increment(void) 7209605c2acSJuan Quintela { 7218f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7229605c2acSJuan Quintela 7239605c2acSJuan Quintela return s->parameters.cpu_throttle_increment; 7249605c2acSJuan Quintela } 7259605c2acSJuan Quintela 7262a8ec380SJuan Quintela uint8_t migrate_cpu_throttle_initial(void) 7272a8ec380SJuan Quintela { 7288f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7292a8ec380SJuan Quintela 7302a8ec380SJuan Quintela return s->parameters.cpu_throttle_initial; 7312a8ec380SJuan Quintela } 7322a8ec380SJuan Quintela 733873f674cSJuan Quintela bool migrate_cpu_throttle_tailslow(void) 734873f674cSJuan Quintela { 7358f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 736873f674cSJuan Quintela 737873f674cSJuan Quintela return s->parameters.cpu_throttle_tailslow; 738873f674cSJuan Quintela } 739873f674cSJuan Quintela 7401dfc4b9eSJuan Quintela int migrate_decompress_threads(void) 7411dfc4b9eSJuan Quintela { 7428f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7431dfc4b9eSJuan Quintela 7441dfc4b9eSJuan Quintela return s->parameters.decompress_threads; 7451dfc4b9eSJuan Quintela } 7461dfc4b9eSJuan Quintela 747f5da8ba4SJuan Quintela uint64_t migrate_downtime_limit(void) 748f5da8ba4SJuan Quintela { 749f5da8ba4SJuan Quintela MigrationState *s = migrate_get_current(); 750f5da8ba4SJuan Quintela 751f5da8ba4SJuan Quintela return s->parameters.downtime_limit; 752f5da8ba4SJuan Quintela } 753f5da8ba4SJuan Quintela 75424155bd0SJuan Quintela uint8_t migrate_max_cpu_throttle(void) 75524155bd0SJuan Quintela { 7568f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 75724155bd0SJuan Quintela 75824155bd0SJuan Quintela return s->parameters.max_cpu_throttle; 75924155bd0SJuan Quintela } 76024155bd0SJuan Quintela 7619c894df3SJuan Quintela uint64_t migrate_max_bandwidth(void) 7629c894df3SJuan Quintela { 7638f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7649c894df3SJuan Quintela 7659c894df3SJuan Quintela return s->parameters.max_bandwidth; 7669c894df3SJuan Quintela } 7679c894df3SJuan Quintela 76852033349SJuan Quintela uint64_t migrate_max_postcopy_bandwidth(void) 7691dfc4b9eSJuan Quintela { 7708f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7711dfc4b9eSJuan Quintela 7721dfc4b9eSJuan Quintela return s->parameters.max_postcopy_bandwidth; 7731dfc4b9eSJuan Quintela } 7741dfc4b9eSJuan Quintela 7751dfc4b9eSJuan Quintela int migrate_multifd_channels(void) 7761dfc4b9eSJuan Quintela { 7778f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7781dfc4b9eSJuan Quintela 7791dfc4b9eSJuan Quintela return s->parameters.multifd_channels; 7801dfc4b9eSJuan Quintela } 7811dfc4b9eSJuan Quintela 7821dfc4b9eSJuan Quintela MultiFDCompression migrate_multifd_compression(void) 7831dfc4b9eSJuan Quintela { 7848f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7851dfc4b9eSJuan Quintela 7861dfc4b9eSJuan Quintela assert(s->parameters.multifd_compression < MULTIFD_COMPRESSION__MAX); 7871dfc4b9eSJuan Quintela return s->parameters.multifd_compression; 7881dfc4b9eSJuan Quintela } 7891dfc4b9eSJuan Quintela 7901dfc4b9eSJuan Quintela int migrate_multifd_zlib_level(void) 7911dfc4b9eSJuan Quintela { 7928f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7931dfc4b9eSJuan Quintela 7941dfc4b9eSJuan Quintela return s->parameters.multifd_zlib_level; 7951dfc4b9eSJuan Quintela } 7961dfc4b9eSJuan Quintela 7971dfc4b9eSJuan Quintela int migrate_multifd_zstd_level(void) 7981dfc4b9eSJuan Quintela { 7998f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 8001dfc4b9eSJuan Quintela 8011dfc4b9eSJuan Quintela return s->parameters.multifd_zstd_level; 8021dfc4b9eSJuan Quintela } 8031dfc4b9eSJuan Quintela 8046499efdbSJuan Quintela uint8_t migrate_throttle_trigger_threshold(void) 8056499efdbSJuan Quintela { 8068f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 8076499efdbSJuan Quintela 8086499efdbSJuan Quintela return s->parameters.throttle_trigger_threshold; 8096499efdbSJuan Quintela } 8106499efdbSJuan Quintela 8112eb0308bSJuan Quintela const char *migrate_tls_authz(void) 8122eb0308bSJuan Quintela { 8132eb0308bSJuan Quintela MigrationState *s = migrate_get_current(); 8142eb0308bSJuan Quintela 8152eb0308bSJuan Quintela return s->parameters.tls_authz; 8162eb0308bSJuan Quintela } 8172eb0308bSJuan Quintela 818d5c3e195SJuan Quintela const char *migrate_tls_creds(void) 819d5c3e195SJuan Quintela { 820d5c3e195SJuan Quintela MigrationState *s = migrate_get_current(); 821d5c3e195SJuan Quintela 822d5c3e195SJuan Quintela return s->parameters.tls_creds; 823d5c3e195SJuan Quintela } 824d5c3e195SJuan Quintela 8251f2f366cSJuan Quintela const char *migrate_tls_hostname(void) 8261f2f366cSJuan Quintela { 8271f2f366cSJuan Quintela MigrationState *s = migrate_get_current(); 8281f2f366cSJuan Quintela 8291f2f366cSJuan Quintela return s->parameters.tls_hostname; 8301f2f366cSJuan Quintela } 8311f2f366cSJuan Quintela 8321dfc4b9eSJuan Quintela uint64_t migrate_xbzrle_cache_size(void) 8331dfc4b9eSJuan Quintela { 8348f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 8351dfc4b9eSJuan Quintela 8361dfc4b9eSJuan Quintela return s->parameters.xbzrle_cache_size; 8371dfc4b9eSJuan Quintela } 8382682c4eeSJuan Quintela 83987c22901SJuan Quintela /* parameter setters */ 84087c22901SJuan Quintela 84187c22901SJuan Quintela void migrate_set_block_incremental(bool value) 84287c22901SJuan Quintela { 84387c22901SJuan Quintela MigrationState *s = migrate_get_current(); 84487c22901SJuan Quintela 84587c22901SJuan Quintela s->parameters.block_incremental = value; 84687c22901SJuan Quintela } 84787c22901SJuan Quintela 8482682c4eeSJuan Quintela /* parameters helpers */ 8492682c4eeSJuan Quintela 850b1a87956SJuan Quintela void block_cleanup_parameters(void) 851b7b73122SJuan Quintela { 852b1a87956SJuan Quintela MigrationState *s = migrate_get_current(); 853b1a87956SJuan Quintela 854b7b73122SJuan Quintela if (s->must_remove_block_options) { 855b7b73122SJuan Quintela /* setting to false can never fail */ 856b7b73122SJuan Quintela migrate_cap_set(MIGRATION_CAPABILITY_BLOCK, false, &error_abort); 857b7b73122SJuan Quintela migrate_set_block_incremental(false); 858b7b73122SJuan Quintela s->must_remove_block_options = false; 859b7b73122SJuan Quintela } 860b7b73122SJuan Quintela } 861b7b73122SJuan Quintela 8622682c4eeSJuan Quintela AnnounceParameters *migrate_announce_params(void) 8632682c4eeSJuan Quintela { 8642682c4eeSJuan Quintela static AnnounceParameters ap; 8652682c4eeSJuan Quintela 8662682c4eeSJuan Quintela MigrationState *s = migrate_get_current(); 8672682c4eeSJuan Quintela 8682682c4eeSJuan Quintela ap.initial = s->parameters.announce_initial; 8692682c4eeSJuan Quintela ap.max = s->parameters.announce_max; 8702682c4eeSJuan Quintela ap.rounds = s->parameters.announce_rounds; 8712682c4eeSJuan Quintela ap.step = s->parameters.announce_step; 8722682c4eeSJuan Quintela 8732682c4eeSJuan Quintela return ≈ 8742682c4eeSJuan Quintela } 8759c894df3SJuan Quintela 8769c894df3SJuan Quintela MigrationParameters *qmp_query_migrate_parameters(Error **errp) 8779c894df3SJuan Quintela { 8789c894df3SJuan Quintela MigrationParameters *params; 8799c894df3SJuan Quintela MigrationState *s = migrate_get_current(); 8809c894df3SJuan Quintela 8819c894df3SJuan Quintela /* TODO use QAPI_CLONE() instead of duplicating it inline */ 8829c894df3SJuan Quintela params = g_malloc0(sizeof(*params)); 8839c894df3SJuan Quintela params->has_compress_level = true; 8849c894df3SJuan Quintela params->compress_level = s->parameters.compress_level; 8859c894df3SJuan Quintela params->has_compress_threads = true; 8869c894df3SJuan Quintela params->compress_threads = s->parameters.compress_threads; 8879c894df3SJuan Quintela params->has_compress_wait_thread = true; 8889c894df3SJuan Quintela params->compress_wait_thread = s->parameters.compress_wait_thread; 8899c894df3SJuan Quintela params->has_decompress_threads = true; 8909c894df3SJuan Quintela params->decompress_threads = s->parameters.decompress_threads; 8919c894df3SJuan Quintela params->has_throttle_trigger_threshold = true; 8929c894df3SJuan Quintela params->throttle_trigger_threshold = s->parameters.throttle_trigger_threshold; 8939c894df3SJuan Quintela params->has_cpu_throttle_initial = true; 8949c894df3SJuan Quintela params->cpu_throttle_initial = s->parameters.cpu_throttle_initial; 8959c894df3SJuan Quintela params->has_cpu_throttle_increment = true; 8969c894df3SJuan Quintela params->cpu_throttle_increment = s->parameters.cpu_throttle_increment; 8979c894df3SJuan Quintela params->has_cpu_throttle_tailslow = true; 8989c894df3SJuan Quintela params->cpu_throttle_tailslow = s->parameters.cpu_throttle_tailslow; 8999c894df3SJuan Quintela params->tls_creds = g_strdup(s->parameters.tls_creds); 9009c894df3SJuan Quintela params->tls_hostname = g_strdup(s->parameters.tls_hostname); 9019c894df3SJuan Quintela params->tls_authz = g_strdup(s->parameters.tls_authz ? 9029c894df3SJuan Quintela s->parameters.tls_authz : ""); 9039c894df3SJuan Quintela params->has_max_bandwidth = true; 9049c894df3SJuan Quintela params->max_bandwidth = s->parameters.max_bandwidth; 9059c894df3SJuan Quintela params->has_downtime_limit = true; 9069c894df3SJuan Quintela params->downtime_limit = s->parameters.downtime_limit; 9079c894df3SJuan Quintela params->has_x_checkpoint_delay = true; 9089c894df3SJuan Quintela params->x_checkpoint_delay = s->parameters.x_checkpoint_delay; 9099c894df3SJuan Quintela params->has_block_incremental = true; 9109c894df3SJuan Quintela params->block_incremental = s->parameters.block_incremental; 9119c894df3SJuan Quintela params->has_multifd_channels = true; 9129c894df3SJuan Quintela params->multifd_channels = s->parameters.multifd_channels; 9139c894df3SJuan Quintela params->has_multifd_compression = true; 9149c894df3SJuan Quintela params->multifd_compression = s->parameters.multifd_compression; 9159c894df3SJuan Quintela params->has_multifd_zlib_level = true; 9169c894df3SJuan Quintela params->multifd_zlib_level = s->parameters.multifd_zlib_level; 9179c894df3SJuan Quintela params->has_multifd_zstd_level = true; 9189c894df3SJuan Quintela params->multifd_zstd_level = s->parameters.multifd_zstd_level; 9199c894df3SJuan Quintela params->has_xbzrle_cache_size = true; 9209c894df3SJuan Quintela params->xbzrle_cache_size = s->parameters.xbzrle_cache_size; 9219c894df3SJuan Quintela params->has_max_postcopy_bandwidth = true; 9229c894df3SJuan Quintela params->max_postcopy_bandwidth = s->parameters.max_postcopy_bandwidth; 9239c894df3SJuan Quintela params->has_max_cpu_throttle = true; 9249c894df3SJuan Quintela params->max_cpu_throttle = s->parameters.max_cpu_throttle; 9259c894df3SJuan Quintela params->has_announce_initial = true; 9269c894df3SJuan Quintela params->announce_initial = s->parameters.announce_initial; 9279c894df3SJuan Quintela params->has_announce_max = true; 9289c894df3SJuan Quintela params->announce_max = s->parameters.announce_max; 9299c894df3SJuan Quintela params->has_announce_rounds = true; 9309c894df3SJuan Quintela params->announce_rounds = s->parameters.announce_rounds; 9319c894df3SJuan Quintela params->has_announce_step = true; 9329c894df3SJuan Quintela params->announce_step = s->parameters.announce_step; 9339c894df3SJuan Quintela 9349c894df3SJuan Quintela if (s->parameters.has_block_bitmap_mapping) { 9359c894df3SJuan Quintela params->has_block_bitmap_mapping = true; 9369c894df3SJuan Quintela params->block_bitmap_mapping = 9379c894df3SJuan Quintela QAPI_CLONE(BitmapMigrationNodeAliasList, 9389c894df3SJuan Quintela s->parameters.block_bitmap_mapping); 9399c894df3SJuan Quintela } 9409c894df3SJuan Quintela 9414d807857SHyman Huang(黄勇) params->has_x_vcpu_dirty_limit_period = true; 9424d807857SHyman Huang(黄勇) params->x_vcpu_dirty_limit_period = s->parameters.x_vcpu_dirty_limit_period; 94309f9ec99SHyman Huang(黄勇) params->has_vcpu_dirty_limit = true; 94409f9ec99SHyman Huang(黄勇) params->vcpu_dirty_limit = s->parameters.vcpu_dirty_limit; 9454d807857SHyman Huang(黄勇) 9469c894df3SJuan Quintela return params; 9479c894df3SJuan Quintela } 94809d6c965SJuan Quintela 94961a174e2SJuan Quintela void migrate_params_init(MigrationParameters *params) 95061a174e2SJuan Quintela { 95161a174e2SJuan Quintela params->tls_hostname = g_strdup(""); 95261a174e2SJuan Quintela params->tls_creds = g_strdup(""); 95361a174e2SJuan Quintela 95461a174e2SJuan Quintela /* Set has_* up only for parameter checks */ 95561a174e2SJuan Quintela params->has_compress_level = true; 95661a174e2SJuan Quintela params->has_compress_threads = true; 95761a174e2SJuan Quintela params->has_compress_wait_thread = true; 95861a174e2SJuan Quintela params->has_decompress_threads = true; 95961a174e2SJuan Quintela params->has_throttle_trigger_threshold = true; 96061a174e2SJuan Quintela params->has_cpu_throttle_initial = true; 96161a174e2SJuan Quintela params->has_cpu_throttle_increment = true; 96261a174e2SJuan Quintela params->has_cpu_throttle_tailslow = true; 96361a174e2SJuan Quintela params->has_max_bandwidth = true; 96461a174e2SJuan Quintela params->has_downtime_limit = true; 96561a174e2SJuan Quintela params->has_x_checkpoint_delay = true; 96661a174e2SJuan Quintela params->has_block_incremental = true; 96761a174e2SJuan Quintela params->has_multifd_channels = true; 96861a174e2SJuan Quintela params->has_multifd_compression = true; 96961a174e2SJuan Quintela params->has_multifd_zlib_level = true; 97061a174e2SJuan Quintela params->has_multifd_zstd_level = true; 97161a174e2SJuan Quintela params->has_xbzrle_cache_size = true; 97261a174e2SJuan Quintela params->has_max_postcopy_bandwidth = true; 97361a174e2SJuan Quintela params->has_max_cpu_throttle = true; 97461a174e2SJuan Quintela params->has_announce_initial = true; 97561a174e2SJuan Quintela params->has_announce_max = true; 97661a174e2SJuan Quintela params->has_announce_rounds = true; 97761a174e2SJuan Quintela params->has_announce_step = true; 9784d807857SHyman Huang(黄勇) params->has_x_vcpu_dirty_limit_period = true; 97909f9ec99SHyman Huang(黄勇) params->has_vcpu_dirty_limit = true; 98061a174e2SJuan Quintela } 98161a174e2SJuan Quintela 98209d6c965SJuan Quintela /* 98309d6c965SJuan Quintela * Check whether the parameters are valid. Error will be put into errp 98409d6c965SJuan Quintela * (if provided). Return true if valid, otherwise false. 98509d6c965SJuan Quintela */ 98609d6c965SJuan Quintela bool migrate_params_check(MigrationParameters *params, Error **errp) 98709d6c965SJuan Quintela { 98809d6c965SJuan Quintela if (params->has_compress_level && 98909d6c965SJuan Quintela (params->compress_level > 9)) { 99009d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, "compress_level", 99109d6c965SJuan Quintela "a value between 0 and 9"); 99209d6c965SJuan Quintela return false; 99309d6c965SJuan Quintela } 99409d6c965SJuan Quintela 99509d6c965SJuan Quintela if (params->has_compress_threads && (params->compress_threads < 1)) { 99609d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 99709d6c965SJuan Quintela "compress_threads", 99809d6c965SJuan Quintela "a value between 1 and 255"); 99909d6c965SJuan Quintela return false; 100009d6c965SJuan Quintela } 100109d6c965SJuan Quintela 100209d6c965SJuan Quintela if (params->has_decompress_threads && (params->decompress_threads < 1)) { 100309d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 100409d6c965SJuan Quintela "decompress_threads", 100509d6c965SJuan Quintela "a value between 1 and 255"); 100609d6c965SJuan Quintela return false; 100709d6c965SJuan Quintela } 100809d6c965SJuan Quintela 100909d6c965SJuan Quintela if (params->has_throttle_trigger_threshold && 101009d6c965SJuan Quintela (params->throttle_trigger_threshold < 1 || 101109d6c965SJuan Quintela params->throttle_trigger_threshold > 100)) { 101209d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 101309d6c965SJuan Quintela "throttle_trigger_threshold", 101409d6c965SJuan Quintela "an integer in the range of 1 to 100"); 101509d6c965SJuan Quintela return false; 101609d6c965SJuan Quintela } 101709d6c965SJuan Quintela 101809d6c965SJuan Quintela if (params->has_cpu_throttle_initial && 101909d6c965SJuan Quintela (params->cpu_throttle_initial < 1 || 102009d6c965SJuan Quintela params->cpu_throttle_initial > 99)) { 102109d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 102209d6c965SJuan Quintela "cpu_throttle_initial", 102309d6c965SJuan Quintela "an integer in the range of 1 to 99"); 102409d6c965SJuan Quintela return false; 102509d6c965SJuan Quintela } 102609d6c965SJuan Quintela 102709d6c965SJuan Quintela if (params->has_cpu_throttle_increment && 102809d6c965SJuan Quintela (params->cpu_throttle_increment < 1 || 102909d6c965SJuan Quintela params->cpu_throttle_increment > 99)) { 103009d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 103109d6c965SJuan Quintela "cpu_throttle_increment", 103209d6c965SJuan Quintela "an integer in the range of 1 to 99"); 103309d6c965SJuan Quintela return false; 103409d6c965SJuan Quintela } 103509d6c965SJuan Quintela 103609d6c965SJuan Quintela if (params->has_max_bandwidth && (params->max_bandwidth > SIZE_MAX)) { 103709d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 103809d6c965SJuan Quintela "max_bandwidth", 103909d6c965SJuan Quintela "an integer in the range of 0 to "stringify(SIZE_MAX) 104009d6c965SJuan Quintela " bytes/second"); 104109d6c965SJuan Quintela return false; 104209d6c965SJuan Quintela } 104309d6c965SJuan Quintela 104409d6c965SJuan Quintela if (params->has_downtime_limit && 104509d6c965SJuan Quintela (params->downtime_limit > MAX_MIGRATE_DOWNTIME)) { 104609d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 104709d6c965SJuan Quintela "downtime_limit", 104809d6c965SJuan Quintela "an integer in the range of 0 to " 104909d6c965SJuan Quintela stringify(MAX_MIGRATE_DOWNTIME)" ms"); 105009d6c965SJuan Quintela return false; 105109d6c965SJuan Quintela } 105209d6c965SJuan Quintela 105309d6c965SJuan Quintela /* x_checkpoint_delay is now always positive */ 105409d6c965SJuan Quintela 105509d6c965SJuan Quintela if (params->has_multifd_channels && (params->multifd_channels < 1)) { 105609d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 105709d6c965SJuan Quintela "multifd_channels", 105809d6c965SJuan Quintela "a value between 1 and 255"); 105909d6c965SJuan Quintela return false; 106009d6c965SJuan Quintela } 106109d6c965SJuan Quintela 106209d6c965SJuan Quintela if (params->has_multifd_zlib_level && 106309d6c965SJuan Quintela (params->multifd_zlib_level > 9)) { 106409d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, "multifd_zlib_level", 106509d6c965SJuan Quintela "a value between 0 and 9"); 106609d6c965SJuan Quintela return false; 106709d6c965SJuan Quintela } 106809d6c965SJuan Quintela 106909d6c965SJuan Quintela if (params->has_multifd_zstd_level && 107009d6c965SJuan Quintela (params->multifd_zstd_level > 20)) { 107109d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, "multifd_zstd_level", 107209d6c965SJuan Quintela "a value between 0 and 20"); 107309d6c965SJuan Quintela return false; 107409d6c965SJuan Quintela } 107509d6c965SJuan Quintela 107609d6c965SJuan Quintela if (params->has_xbzrle_cache_size && 107709d6c965SJuan Quintela (params->xbzrle_cache_size < qemu_target_page_size() || 107809d6c965SJuan Quintela !is_power_of_2(params->xbzrle_cache_size))) { 107909d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 108009d6c965SJuan Quintela "xbzrle_cache_size", 108109d6c965SJuan Quintela "a power of two no less than the target page size"); 108209d6c965SJuan Quintela return false; 108309d6c965SJuan Quintela } 108409d6c965SJuan Quintela 108509d6c965SJuan Quintela if (params->has_max_cpu_throttle && 108609d6c965SJuan Quintela (params->max_cpu_throttle < params->cpu_throttle_initial || 108709d6c965SJuan Quintela params->max_cpu_throttle > 99)) { 108809d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 108909d6c965SJuan Quintela "max_cpu_throttle", 109009d6c965SJuan Quintela "an integer in the range of cpu_throttle_initial to 99"); 109109d6c965SJuan Quintela return false; 109209d6c965SJuan Quintela } 109309d6c965SJuan Quintela 109409d6c965SJuan Quintela if (params->has_announce_initial && 109509d6c965SJuan Quintela params->announce_initial > 100000) { 109609d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 109709d6c965SJuan Quintela "announce_initial", 109809d6c965SJuan Quintela "a value between 0 and 100000"); 109909d6c965SJuan Quintela return false; 110009d6c965SJuan Quintela } 110109d6c965SJuan Quintela if (params->has_announce_max && 110209d6c965SJuan Quintela params->announce_max > 100000) { 110309d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 110409d6c965SJuan Quintela "announce_max", 110509d6c965SJuan Quintela "a value between 0 and 100000"); 110609d6c965SJuan Quintela return false; 110709d6c965SJuan Quintela } 110809d6c965SJuan Quintela if (params->has_announce_rounds && 110909d6c965SJuan Quintela params->announce_rounds > 1000) { 111009d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 111109d6c965SJuan Quintela "announce_rounds", 111209d6c965SJuan Quintela "a value between 0 and 1000"); 111309d6c965SJuan Quintela return false; 111409d6c965SJuan Quintela } 111509d6c965SJuan Quintela if (params->has_announce_step && 111609d6c965SJuan Quintela (params->announce_step < 1 || 111709d6c965SJuan Quintela params->announce_step > 10000)) { 111809d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 111909d6c965SJuan Quintela "announce_step", 112009d6c965SJuan Quintela "a value between 0 and 10000"); 112109d6c965SJuan Quintela return false; 112209d6c965SJuan Quintela } 112309d6c965SJuan Quintela 112409d6c965SJuan Quintela if (params->has_block_bitmap_mapping && 112509d6c965SJuan Quintela !check_dirty_bitmap_mig_alias_map(params->block_bitmap_mapping, errp)) { 112609d6c965SJuan Quintela error_prepend(errp, "Invalid mapping given for block-bitmap-mapping: "); 112709d6c965SJuan Quintela return false; 112809d6c965SJuan Quintela } 112909d6c965SJuan Quintela 113009d6c965SJuan Quintela #ifdef CONFIG_LINUX 113109d6c965SJuan Quintela if (migrate_zero_copy_send() && 113209d6c965SJuan Quintela ((params->has_multifd_compression && params->multifd_compression) || 113309d6c965SJuan Quintela (params->tls_creds && *params->tls_creds))) { 113409d6c965SJuan Quintela error_setg(errp, 113509d6c965SJuan Quintela "Zero copy only available for non-compressed non-TLS multifd migration"); 113609d6c965SJuan Quintela return false; 113709d6c965SJuan Quintela } 113809d6c965SJuan Quintela #endif 113909d6c965SJuan Quintela 11404d807857SHyman Huang(黄勇) if (params->has_x_vcpu_dirty_limit_period && 11414d807857SHyman Huang(黄勇) (params->x_vcpu_dirty_limit_period < 1 || 11424d807857SHyman Huang(黄勇) params->x_vcpu_dirty_limit_period > 1000)) { 11434d807857SHyman Huang(黄勇) error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 11444d807857SHyman Huang(黄勇) "x-vcpu-dirty-limit-period", 11454d807857SHyman Huang(黄勇) "a value between 1 and 1000"); 11464d807857SHyman Huang(黄勇) return false; 11474d807857SHyman Huang(黄勇) } 11484d807857SHyman Huang(黄勇) 114909f9ec99SHyman Huang(黄勇) if (params->has_vcpu_dirty_limit && 115009f9ec99SHyman Huang(黄勇) (params->vcpu_dirty_limit < 1)) { 115109f9ec99SHyman Huang(黄勇) error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 115209f9ec99SHyman Huang(黄勇) "vcpu_dirty_limit", 115309f9ec99SHyman Huang(黄勇) "is invalid, it must greater then 1 MB/s"); 115409f9ec99SHyman Huang(黄勇) return false; 115509f9ec99SHyman Huang(黄勇) } 115609f9ec99SHyman Huang(黄勇) 115709d6c965SJuan Quintela return true; 115809d6c965SJuan Quintela } 115909d6c965SJuan Quintela 116009d6c965SJuan Quintela static void migrate_params_test_apply(MigrateSetParameters *params, 116109d6c965SJuan Quintela MigrationParameters *dest) 116209d6c965SJuan Quintela { 116309d6c965SJuan Quintela *dest = migrate_get_current()->parameters; 116409d6c965SJuan Quintela 116509d6c965SJuan Quintela /* TODO use QAPI_CLONE() instead of duplicating it inline */ 116609d6c965SJuan Quintela 116709d6c965SJuan Quintela if (params->has_compress_level) { 116809d6c965SJuan Quintela dest->compress_level = params->compress_level; 116909d6c965SJuan Quintela } 117009d6c965SJuan Quintela 117109d6c965SJuan Quintela if (params->has_compress_threads) { 117209d6c965SJuan Quintela dest->compress_threads = params->compress_threads; 117309d6c965SJuan Quintela } 117409d6c965SJuan Quintela 117509d6c965SJuan Quintela if (params->has_compress_wait_thread) { 117609d6c965SJuan Quintela dest->compress_wait_thread = params->compress_wait_thread; 117709d6c965SJuan Quintela } 117809d6c965SJuan Quintela 117909d6c965SJuan Quintela if (params->has_decompress_threads) { 118009d6c965SJuan Quintela dest->decompress_threads = params->decompress_threads; 118109d6c965SJuan Quintela } 118209d6c965SJuan Quintela 118309d6c965SJuan Quintela if (params->has_throttle_trigger_threshold) { 118409d6c965SJuan Quintela dest->throttle_trigger_threshold = params->throttle_trigger_threshold; 118509d6c965SJuan Quintela } 118609d6c965SJuan Quintela 118709d6c965SJuan Quintela if (params->has_cpu_throttle_initial) { 118809d6c965SJuan Quintela dest->cpu_throttle_initial = params->cpu_throttle_initial; 118909d6c965SJuan Quintela } 119009d6c965SJuan Quintela 119109d6c965SJuan Quintela if (params->has_cpu_throttle_increment) { 119209d6c965SJuan Quintela dest->cpu_throttle_increment = params->cpu_throttle_increment; 119309d6c965SJuan Quintela } 119409d6c965SJuan Quintela 119509d6c965SJuan Quintela if (params->has_cpu_throttle_tailslow) { 119609d6c965SJuan Quintela dest->cpu_throttle_tailslow = params->cpu_throttle_tailslow; 119709d6c965SJuan Quintela } 119809d6c965SJuan Quintela 119909d6c965SJuan Quintela if (params->tls_creds) { 120009d6c965SJuan Quintela assert(params->tls_creds->type == QTYPE_QSTRING); 120109d6c965SJuan Quintela dest->tls_creds = params->tls_creds->u.s; 120209d6c965SJuan Quintela } 120309d6c965SJuan Quintela 120409d6c965SJuan Quintela if (params->tls_hostname) { 120509d6c965SJuan Quintela assert(params->tls_hostname->type == QTYPE_QSTRING); 120609d6c965SJuan Quintela dest->tls_hostname = params->tls_hostname->u.s; 120709d6c965SJuan Quintela } 120809d6c965SJuan Quintela 120909d6c965SJuan Quintela if (params->has_max_bandwidth) { 121009d6c965SJuan Quintela dest->max_bandwidth = params->max_bandwidth; 121109d6c965SJuan Quintela } 121209d6c965SJuan Quintela 121309d6c965SJuan Quintela if (params->has_downtime_limit) { 121409d6c965SJuan Quintela dest->downtime_limit = params->downtime_limit; 121509d6c965SJuan Quintela } 121609d6c965SJuan Quintela 121709d6c965SJuan Quintela if (params->has_x_checkpoint_delay) { 121809d6c965SJuan Quintela dest->x_checkpoint_delay = params->x_checkpoint_delay; 121909d6c965SJuan Quintela } 122009d6c965SJuan Quintela 122109d6c965SJuan Quintela if (params->has_block_incremental) { 122209d6c965SJuan Quintela dest->block_incremental = params->block_incremental; 122309d6c965SJuan Quintela } 122409d6c965SJuan Quintela if (params->has_multifd_channels) { 122509d6c965SJuan Quintela dest->multifd_channels = params->multifd_channels; 122609d6c965SJuan Quintela } 122709d6c965SJuan Quintela if (params->has_multifd_compression) { 122809d6c965SJuan Quintela dest->multifd_compression = params->multifd_compression; 122909d6c965SJuan Quintela } 123009d6c965SJuan Quintela if (params->has_xbzrle_cache_size) { 123109d6c965SJuan Quintela dest->xbzrle_cache_size = params->xbzrle_cache_size; 123209d6c965SJuan Quintela } 123309d6c965SJuan Quintela if (params->has_max_postcopy_bandwidth) { 123409d6c965SJuan Quintela dest->max_postcopy_bandwidth = params->max_postcopy_bandwidth; 123509d6c965SJuan Quintela } 123609d6c965SJuan Quintela if (params->has_max_cpu_throttle) { 123709d6c965SJuan Quintela dest->max_cpu_throttle = params->max_cpu_throttle; 123809d6c965SJuan Quintela } 123909d6c965SJuan Quintela if (params->has_announce_initial) { 124009d6c965SJuan Quintela dest->announce_initial = params->announce_initial; 124109d6c965SJuan Quintela } 124209d6c965SJuan Quintela if (params->has_announce_max) { 124309d6c965SJuan Quintela dest->announce_max = params->announce_max; 124409d6c965SJuan Quintela } 124509d6c965SJuan Quintela if (params->has_announce_rounds) { 124609d6c965SJuan Quintela dest->announce_rounds = params->announce_rounds; 124709d6c965SJuan Quintela } 124809d6c965SJuan Quintela if (params->has_announce_step) { 124909d6c965SJuan Quintela dest->announce_step = params->announce_step; 125009d6c965SJuan Quintela } 125109d6c965SJuan Quintela 125209d6c965SJuan Quintela if (params->has_block_bitmap_mapping) { 125309d6c965SJuan Quintela dest->has_block_bitmap_mapping = true; 125409d6c965SJuan Quintela dest->block_bitmap_mapping = params->block_bitmap_mapping; 125509d6c965SJuan Quintela } 12564d807857SHyman Huang(黄勇) 12574d807857SHyman Huang(黄勇) if (params->has_x_vcpu_dirty_limit_period) { 12584d807857SHyman Huang(黄勇) dest->x_vcpu_dirty_limit_period = 12594d807857SHyman Huang(黄勇) params->x_vcpu_dirty_limit_period; 12604d807857SHyman Huang(黄勇) } 126109f9ec99SHyman Huang(黄勇) if (params->has_vcpu_dirty_limit) { 126209f9ec99SHyman Huang(黄勇) dest->vcpu_dirty_limit = params->vcpu_dirty_limit; 126309f9ec99SHyman Huang(黄勇) } 126409d6c965SJuan Quintela } 126509d6c965SJuan Quintela 126609d6c965SJuan Quintela static void migrate_params_apply(MigrateSetParameters *params, Error **errp) 126709d6c965SJuan Quintela { 126809d6c965SJuan Quintela MigrationState *s = migrate_get_current(); 126909d6c965SJuan Quintela 127009d6c965SJuan Quintela /* TODO use QAPI_CLONE() instead of duplicating it inline */ 127109d6c965SJuan Quintela 127209d6c965SJuan Quintela if (params->has_compress_level) { 127309d6c965SJuan Quintela s->parameters.compress_level = params->compress_level; 127409d6c965SJuan Quintela } 127509d6c965SJuan Quintela 127609d6c965SJuan Quintela if (params->has_compress_threads) { 127709d6c965SJuan Quintela s->parameters.compress_threads = params->compress_threads; 127809d6c965SJuan Quintela } 127909d6c965SJuan Quintela 128009d6c965SJuan Quintela if (params->has_compress_wait_thread) { 128109d6c965SJuan Quintela s->parameters.compress_wait_thread = params->compress_wait_thread; 128209d6c965SJuan Quintela } 128309d6c965SJuan Quintela 128409d6c965SJuan Quintela if (params->has_decompress_threads) { 128509d6c965SJuan Quintela s->parameters.decompress_threads = params->decompress_threads; 128609d6c965SJuan Quintela } 128709d6c965SJuan Quintela 128809d6c965SJuan Quintela if (params->has_throttle_trigger_threshold) { 128909d6c965SJuan Quintela s->parameters.throttle_trigger_threshold = params->throttle_trigger_threshold; 129009d6c965SJuan Quintela } 129109d6c965SJuan Quintela 129209d6c965SJuan Quintela if (params->has_cpu_throttle_initial) { 129309d6c965SJuan Quintela s->parameters.cpu_throttle_initial = params->cpu_throttle_initial; 129409d6c965SJuan Quintela } 129509d6c965SJuan Quintela 129609d6c965SJuan Quintela if (params->has_cpu_throttle_increment) { 129709d6c965SJuan Quintela s->parameters.cpu_throttle_increment = params->cpu_throttle_increment; 129809d6c965SJuan Quintela } 129909d6c965SJuan Quintela 130009d6c965SJuan Quintela if (params->has_cpu_throttle_tailslow) { 130109d6c965SJuan Quintela s->parameters.cpu_throttle_tailslow = params->cpu_throttle_tailslow; 130209d6c965SJuan Quintela } 130309d6c965SJuan Quintela 130409d6c965SJuan Quintela if (params->tls_creds) { 130509d6c965SJuan Quintela g_free(s->parameters.tls_creds); 130609d6c965SJuan Quintela assert(params->tls_creds->type == QTYPE_QSTRING); 130709d6c965SJuan Quintela s->parameters.tls_creds = g_strdup(params->tls_creds->u.s); 130809d6c965SJuan Quintela } 130909d6c965SJuan Quintela 131009d6c965SJuan Quintela if (params->tls_hostname) { 131109d6c965SJuan Quintela g_free(s->parameters.tls_hostname); 131209d6c965SJuan Quintela assert(params->tls_hostname->type == QTYPE_QSTRING); 131309d6c965SJuan Quintela s->parameters.tls_hostname = g_strdup(params->tls_hostname->u.s); 131409d6c965SJuan Quintela } 131509d6c965SJuan Quintela 131609d6c965SJuan Quintela if (params->tls_authz) { 131709d6c965SJuan Quintela g_free(s->parameters.tls_authz); 131809d6c965SJuan Quintela assert(params->tls_authz->type == QTYPE_QSTRING); 131909d6c965SJuan Quintela s->parameters.tls_authz = g_strdup(params->tls_authz->u.s); 132009d6c965SJuan Quintela } 132109d6c965SJuan Quintela 132209d6c965SJuan Quintela if (params->has_max_bandwidth) { 132309d6c965SJuan Quintela s->parameters.max_bandwidth = params->max_bandwidth; 132409d6c965SJuan Quintela if (s->to_dst_file && !migration_in_postcopy()) { 1325e1fde0e0SJuan Quintela migration_rate_set(s->parameters.max_bandwidth); 132609d6c965SJuan Quintela } 132709d6c965SJuan Quintela } 132809d6c965SJuan Quintela 132909d6c965SJuan Quintela if (params->has_downtime_limit) { 133009d6c965SJuan Quintela s->parameters.downtime_limit = params->downtime_limit; 133109d6c965SJuan Quintela } 133209d6c965SJuan Quintela 133309d6c965SJuan Quintela if (params->has_x_checkpoint_delay) { 133409d6c965SJuan Quintela s->parameters.x_checkpoint_delay = params->x_checkpoint_delay; 13354332ffcdSVladimir Sementsov-Ogievskiy colo_checkpoint_delay_set(); 133609d6c965SJuan Quintela } 133709d6c965SJuan Quintela 133809d6c965SJuan Quintela if (params->has_block_incremental) { 133909d6c965SJuan Quintela s->parameters.block_incremental = params->block_incremental; 134009d6c965SJuan Quintela } 134109d6c965SJuan Quintela if (params->has_multifd_channels) { 134209d6c965SJuan Quintela s->parameters.multifd_channels = params->multifd_channels; 134309d6c965SJuan Quintela } 134409d6c965SJuan Quintela if (params->has_multifd_compression) { 134509d6c965SJuan Quintela s->parameters.multifd_compression = params->multifd_compression; 134609d6c965SJuan Quintela } 134709d6c965SJuan Quintela if (params->has_xbzrle_cache_size) { 134809d6c965SJuan Quintela s->parameters.xbzrle_cache_size = params->xbzrle_cache_size; 134909d6c965SJuan Quintela xbzrle_cache_resize(params->xbzrle_cache_size, errp); 135009d6c965SJuan Quintela } 135109d6c965SJuan Quintela if (params->has_max_postcopy_bandwidth) { 135209d6c965SJuan Quintela s->parameters.max_postcopy_bandwidth = params->max_postcopy_bandwidth; 135309d6c965SJuan Quintela if (s->to_dst_file && migration_in_postcopy()) { 1354e1fde0e0SJuan Quintela migration_rate_set(s->parameters.max_postcopy_bandwidth); 135509d6c965SJuan Quintela } 135609d6c965SJuan Quintela } 135709d6c965SJuan Quintela if (params->has_max_cpu_throttle) { 135809d6c965SJuan Quintela s->parameters.max_cpu_throttle = params->max_cpu_throttle; 135909d6c965SJuan Quintela } 136009d6c965SJuan Quintela if (params->has_announce_initial) { 136109d6c965SJuan Quintela s->parameters.announce_initial = params->announce_initial; 136209d6c965SJuan Quintela } 136309d6c965SJuan Quintela if (params->has_announce_max) { 136409d6c965SJuan Quintela s->parameters.announce_max = params->announce_max; 136509d6c965SJuan Quintela } 136609d6c965SJuan Quintela if (params->has_announce_rounds) { 136709d6c965SJuan Quintela s->parameters.announce_rounds = params->announce_rounds; 136809d6c965SJuan Quintela } 136909d6c965SJuan Quintela if (params->has_announce_step) { 137009d6c965SJuan Quintela s->parameters.announce_step = params->announce_step; 137109d6c965SJuan Quintela } 137209d6c965SJuan Quintela 137309d6c965SJuan Quintela if (params->has_block_bitmap_mapping) { 137409d6c965SJuan Quintela qapi_free_BitmapMigrationNodeAliasList( 137509d6c965SJuan Quintela s->parameters.block_bitmap_mapping); 137609d6c965SJuan Quintela 137709d6c965SJuan Quintela s->parameters.has_block_bitmap_mapping = true; 137809d6c965SJuan Quintela s->parameters.block_bitmap_mapping = 137909d6c965SJuan Quintela QAPI_CLONE(BitmapMigrationNodeAliasList, 138009d6c965SJuan Quintela params->block_bitmap_mapping); 138109d6c965SJuan Quintela } 13824d807857SHyman Huang(黄勇) 13834d807857SHyman Huang(黄勇) if (params->has_x_vcpu_dirty_limit_period) { 13844d807857SHyman Huang(黄勇) s->parameters.x_vcpu_dirty_limit_period = 13854d807857SHyman Huang(黄勇) params->x_vcpu_dirty_limit_period; 13864d807857SHyman Huang(黄勇) } 138709f9ec99SHyman Huang(黄勇) if (params->has_vcpu_dirty_limit) { 138809f9ec99SHyman Huang(黄勇) s->parameters.vcpu_dirty_limit = params->vcpu_dirty_limit; 138909f9ec99SHyman Huang(黄勇) } 139009d6c965SJuan Quintela } 139109d6c965SJuan Quintela 139209d6c965SJuan Quintela void qmp_migrate_set_parameters(MigrateSetParameters *params, Error **errp) 139309d6c965SJuan Quintela { 139409d6c965SJuan Quintela MigrationParameters tmp; 139509d6c965SJuan Quintela 139609d6c965SJuan Quintela /* TODO Rewrite "" to null instead */ 139709d6c965SJuan Quintela if (params->tls_creds 139809d6c965SJuan Quintela && params->tls_creds->type == QTYPE_QNULL) { 139909d6c965SJuan Quintela qobject_unref(params->tls_creds->u.n); 140009d6c965SJuan Quintela params->tls_creds->type = QTYPE_QSTRING; 140109d6c965SJuan Quintela params->tls_creds->u.s = strdup(""); 140209d6c965SJuan Quintela } 140309d6c965SJuan Quintela /* TODO Rewrite "" to null instead */ 140409d6c965SJuan Quintela if (params->tls_hostname 140509d6c965SJuan Quintela && params->tls_hostname->type == QTYPE_QNULL) { 140609d6c965SJuan Quintela qobject_unref(params->tls_hostname->u.n); 140709d6c965SJuan Quintela params->tls_hostname->type = QTYPE_QSTRING; 140809d6c965SJuan Quintela params->tls_hostname->u.s = strdup(""); 140909d6c965SJuan Quintela } 141009d6c965SJuan Quintela 141109d6c965SJuan Quintela migrate_params_test_apply(params, &tmp); 141209d6c965SJuan Quintela 141309d6c965SJuan Quintela if (!migrate_params_check(&tmp, errp)) { 141409d6c965SJuan Quintela /* Invalid parameter */ 141509d6c965SJuan Quintela return; 141609d6c965SJuan Quintela } 141709d6c965SJuan Quintela 141809d6c965SJuan Quintela migrate_params_apply(params, errp); 141909d6c965SJuan Quintela } 1420