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" 30dc623955SHyman 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), 200dc623955SHyman 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 246dc623955SHyman Huang(黄勇) bool migrate_dirty_limit(void) 247dc623955SHyman Huang(黄勇) { 248dc623955SHyman Huang(黄勇) MigrationState *s = migrate_get_current(); 249dc623955SHyman Huang(黄勇) 250dc623955SHyman Huang(黄勇) return s->capabilities[MIGRATION_CAPABILITY_DIRTY_LIMIT]; 251dc623955SHyman Huang(黄勇) } 252dc623955SHyman 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 444*82137e6cSWei Wang static bool migrate_incoming_started(void) 445*82137e6cSWei Wang { 446*82137e6cSWei Wang return !!migration_incoming_get_current()->transport_data; 447*82137e6cSWei Wang } 448*82137e6cSWei Wang 44977608706SJuan Quintela /** 45077608706SJuan Quintela * @migration_caps_check - check capability compatibility 45177608706SJuan Quintela * 45277608706SJuan Quintela * @old_caps: old capability list 45377608706SJuan Quintela * @new_caps: new capability list 45477608706SJuan Quintela * @errp: set *errp if the check failed, with reason 45577608706SJuan Quintela * 45677608706SJuan Quintela * Returns true if check passed, otherwise false. 45777608706SJuan Quintela */ 45877608706SJuan Quintela bool migrate_caps_check(bool *old_caps, bool *new_caps, Error **errp) 45977608706SJuan Quintela { 46077608706SJuan Quintela MigrationIncomingState *mis = migration_incoming_get_current(); 46177608706SJuan Quintela 46274c38cf7SPeter Xu ERRP_GUARD(); 46377608706SJuan Quintela #ifndef CONFIG_LIVE_BLOCK_MIGRATION 46477608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_BLOCK]) { 46577608706SJuan Quintela error_setg(errp, "QEMU compiled without old-style (blk/-b, inc/-i) " 46677608706SJuan Quintela "block migration"); 46777608706SJuan Quintela error_append_hint(errp, "Use drive_mirror+NBD instead.\n"); 46877608706SJuan Quintela return false; 46977608706SJuan Quintela } 47077608706SJuan Quintela #endif 47177608706SJuan Quintela 47277608706SJuan Quintela #ifndef CONFIG_REPLICATION 47377608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_X_COLO]) { 47477608706SJuan Quintela error_setg(errp, "QEMU compiled without replication module" 47577608706SJuan Quintela " can't enable COLO"); 47677608706SJuan Quintela error_append_hint(errp, "Please enable replication before COLO.\n"); 47777608706SJuan Quintela return false; 47877608706SJuan Quintela } 47977608706SJuan Quintela #endif 48077608706SJuan Quintela 48177608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_POSTCOPY_RAM]) { 48277608706SJuan Quintela /* This check is reasonably expensive, so only when it's being 48377608706SJuan Quintela * set the first time, also it's only the destination that needs 48477608706SJuan Quintela * special support. 48577608706SJuan Quintela */ 48677608706SJuan Quintela if (!old_caps[MIGRATION_CAPABILITY_POSTCOPY_RAM] && 48777608706SJuan Quintela runstate_check(RUN_STATE_INMIGRATE) && 48874c38cf7SPeter Xu !postcopy_ram_supported_by_host(mis, errp)) { 48974c38cf7SPeter Xu error_prepend(errp, "Postcopy is not supported: "); 49077608706SJuan Quintela return false; 49177608706SJuan Quintela } 49277608706SJuan Quintela 49377608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_X_IGNORE_SHARED]) { 49477608706SJuan Quintela error_setg(errp, "Postcopy is not compatible with ignore-shared"); 49577608706SJuan Quintela return false; 49677608706SJuan Quintela } 497b405dfffSLeonardo Bras 498b405dfffSLeonardo Bras if (new_caps[MIGRATION_CAPABILITY_MULTIFD]) { 499b405dfffSLeonardo Bras error_setg(errp, "Postcopy is not yet compatible with multifd"); 500b405dfffSLeonardo Bras return false; 501b405dfffSLeonardo Bras } 50277608706SJuan Quintela } 50377608706SJuan Quintela 50477608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_BACKGROUND_SNAPSHOT]) { 50577608706SJuan Quintela WriteTrackingSupport wt_support; 50677608706SJuan Quintela int idx; 50777608706SJuan Quintela /* 50877608706SJuan Quintela * Check if 'background-snapshot' capability is supported by 50977608706SJuan Quintela * host kernel and compatible with guest memory configuration. 51077608706SJuan Quintela */ 51177608706SJuan Quintela wt_support = migrate_query_write_tracking(); 51277608706SJuan Quintela if (wt_support < WT_SUPPORT_AVAILABLE) { 51377608706SJuan Quintela error_setg(errp, "Background-snapshot is not supported by host kernel"); 51477608706SJuan Quintela return false; 51577608706SJuan Quintela } 51677608706SJuan Quintela if (wt_support < WT_SUPPORT_COMPATIBLE) { 51777608706SJuan Quintela error_setg(errp, "Background-snapshot is not compatible " 51877608706SJuan Quintela "with guest memory configuration"); 51977608706SJuan Quintela return false; 52077608706SJuan Quintela } 52177608706SJuan Quintela 52277608706SJuan Quintela /* 52377608706SJuan Quintela * Check if there are any migration capabilities 52477608706SJuan Quintela * incompatible with 'background-snapshot'. 52577608706SJuan Quintela */ 52677608706SJuan Quintela for (idx = 0; idx < check_caps_background_snapshot.size; idx++) { 52777608706SJuan Quintela int incomp_cap = check_caps_background_snapshot.caps[idx]; 52877608706SJuan Quintela if (new_caps[incomp_cap]) { 52977608706SJuan Quintela error_setg(errp, 53077608706SJuan Quintela "Background-snapshot is not compatible with %s", 53177608706SJuan Quintela MigrationCapability_str(incomp_cap)); 53277608706SJuan Quintela return false; 53377608706SJuan Quintela } 53477608706SJuan Quintela } 53577608706SJuan Quintela } 53677608706SJuan Quintela 53777608706SJuan Quintela #ifdef CONFIG_LINUX 53877608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_ZERO_COPY_SEND] && 53977608706SJuan Quintela (!new_caps[MIGRATION_CAPABILITY_MULTIFD] || 54077608706SJuan Quintela new_caps[MIGRATION_CAPABILITY_COMPRESS] || 54177608706SJuan Quintela new_caps[MIGRATION_CAPABILITY_XBZRLE] || 54277608706SJuan Quintela migrate_multifd_compression() || 54310d4703bSJuan Quintela migrate_tls())) { 54477608706SJuan Quintela error_setg(errp, 54577608706SJuan Quintela "Zero copy only available for non-compressed non-TLS multifd migration"); 54677608706SJuan Quintela return false; 54777608706SJuan Quintela } 54877608706SJuan Quintela #else 54977608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_ZERO_COPY_SEND]) { 55077608706SJuan Quintela error_setg(errp, 55177608706SJuan Quintela "Zero copy currently only available on Linux"); 55277608706SJuan Quintela return false; 55377608706SJuan Quintela } 55477608706SJuan Quintela #endif 55577608706SJuan Quintela 55677608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_POSTCOPY_PREEMPT]) { 55777608706SJuan Quintela if (!new_caps[MIGRATION_CAPABILITY_POSTCOPY_RAM]) { 55877608706SJuan Quintela error_setg(errp, "Postcopy preempt requires postcopy-ram"); 55977608706SJuan Quintela return false; 56077608706SJuan Quintela } 56177608706SJuan Quintela 56277608706SJuan Quintela /* 56377608706SJuan Quintela * Preempt mode requires urgent pages to be sent in separate 56477608706SJuan Quintela * channel, OTOH compression logic will disorder all pages into 56577608706SJuan Quintela * different compression channels, which is not compatible with the 56677608706SJuan Quintela * preempt assumptions on channel assignments. 56777608706SJuan Quintela */ 56877608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_COMPRESS]) { 56977608706SJuan Quintela error_setg(errp, "Postcopy preempt not compatible with compress"); 57077608706SJuan Quintela return false; 57177608706SJuan Quintela } 572*82137e6cSWei Wang 573*82137e6cSWei Wang if (migrate_incoming_started()) { 574*82137e6cSWei Wang error_setg(errp, 575*82137e6cSWei Wang "Postcopy preempt must be set before incoming starts"); 576*82137e6cSWei Wang return false; 577*82137e6cSWei Wang } 57877608706SJuan Quintela } 57977608706SJuan Quintela 58077608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_MULTIFD]) { 58177608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_COMPRESS]) { 58277608706SJuan Quintela error_setg(errp, "Multifd is not compatible with compress"); 58377608706SJuan Quintela return false; 58477608706SJuan Quintela } 585*82137e6cSWei Wang if (migrate_incoming_started()) { 586*82137e6cSWei Wang error_setg(errp, "Multifd must be set before incoming starts"); 587*82137e6cSWei Wang return false; 588*82137e6cSWei Wang } 58977608706SJuan Quintela } 59077608706SJuan Quintela 5916574232fSAvihai Horon if (new_caps[MIGRATION_CAPABILITY_SWITCHOVER_ACK]) { 5926574232fSAvihai Horon if (!new_caps[MIGRATION_CAPABILITY_RETURN_PATH]) { 5936574232fSAvihai Horon error_setg(errp, "Capability 'switchover-ack' requires capability " 5946574232fSAvihai Horon "'return-path'"); 5956574232fSAvihai Horon return false; 5966574232fSAvihai Horon } 5976574232fSAvihai Horon } 598dc623955SHyman Huang(黄勇) if (new_caps[MIGRATION_CAPABILITY_DIRTY_LIMIT]) { 599dc623955SHyman Huang(黄勇) if (new_caps[MIGRATION_CAPABILITY_AUTO_CONVERGE]) { 600dc623955SHyman Huang(黄勇) error_setg(errp, "dirty-limit conflicts with auto-converge" 601dc623955SHyman Huang(黄勇) " either of then available currently"); 602dc623955SHyman Huang(黄勇) return false; 603dc623955SHyman Huang(黄勇) } 604dc623955SHyman Huang(黄勇) 605dc623955SHyman Huang(黄勇) if (!kvm_enabled() || !kvm_dirty_ring_enabled()) { 606dc623955SHyman Huang(黄勇) error_setg(errp, "dirty-limit requires KVM with accelerator" 607dc623955SHyman Huang(黄勇) " property 'dirty-ring-size' set"); 608dc623955SHyman Huang(黄勇) return false; 609dc623955SHyman Huang(黄勇) } 610dc623955SHyman Huang(黄勇) } 6116574232fSAvihai Horon 61277608706SJuan Quintela return true; 61377608706SJuan Quintela } 6144d0c6b69SJuan Quintela 615f80196b7SJuan Quintela bool migrate_cap_set(int cap, bool value, Error **errp) 616f80196b7SJuan Quintela { 617f80196b7SJuan Quintela MigrationState *s = migrate_get_current(); 618f80196b7SJuan Quintela bool new_caps[MIGRATION_CAPABILITY__MAX]; 619f80196b7SJuan Quintela 620f80196b7SJuan Quintela if (migration_is_running(s->state)) { 621f80196b7SJuan Quintela error_setg(errp, QERR_MIGRATION_ACTIVE); 622f80196b7SJuan Quintela return false; 623f80196b7SJuan Quintela } 624f80196b7SJuan Quintela 625f80196b7SJuan Quintela memcpy(new_caps, s->capabilities, sizeof(new_caps)); 626f80196b7SJuan Quintela new_caps[cap] = value; 627f80196b7SJuan Quintela 628f80196b7SJuan Quintela if (!migrate_caps_check(s->capabilities, new_caps, errp)) { 629f80196b7SJuan Quintela return false; 630f80196b7SJuan Quintela } 631f80196b7SJuan Quintela s->capabilities[cap] = value; 632f80196b7SJuan Quintela return true; 633f80196b7SJuan Quintela } 634f80196b7SJuan Quintela 6354d0c6b69SJuan Quintela MigrationCapabilityStatusList *qmp_query_migrate_capabilities(Error **errp) 6364d0c6b69SJuan Quintela { 6374d0c6b69SJuan Quintela MigrationCapabilityStatusList *head = NULL, **tail = &head; 6384d0c6b69SJuan Quintela MigrationCapabilityStatus *caps; 6394d0c6b69SJuan Quintela MigrationState *s = migrate_get_current(); 6404d0c6b69SJuan Quintela int i; 6414d0c6b69SJuan Quintela 6424d0c6b69SJuan Quintela for (i = 0; i < MIGRATION_CAPABILITY__MAX; i++) { 6434d0c6b69SJuan Quintela #ifndef CONFIG_LIVE_BLOCK_MIGRATION 6444d0c6b69SJuan Quintela if (i == MIGRATION_CAPABILITY_BLOCK) { 6454d0c6b69SJuan Quintela continue; 6464d0c6b69SJuan Quintela } 6474d0c6b69SJuan Quintela #endif 6484d0c6b69SJuan Quintela caps = g_malloc0(sizeof(*caps)); 6494d0c6b69SJuan Quintela caps->capability = i; 6504d0c6b69SJuan Quintela caps->state = s->capabilities[i]; 6514d0c6b69SJuan Quintela QAPI_LIST_APPEND(tail, caps); 6524d0c6b69SJuan Quintela } 6534d0c6b69SJuan Quintela 6544d0c6b69SJuan Quintela return head; 6554d0c6b69SJuan Quintela } 65645c1de13SJuan Quintela 65745c1de13SJuan Quintela void qmp_migrate_set_capabilities(MigrationCapabilityStatusList *params, 65845c1de13SJuan Quintela Error **errp) 65945c1de13SJuan Quintela { 66045c1de13SJuan Quintela MigrationState *s = migrate_get_current(); 66145c1de13SJuan Quintela MigrationCapabilityStatusList *cap; 66245c1de13SJuan Quintela bool new_caps[MIGRATION_CAPABILITY__MAX]; 66345c1de13SJuan Quintela 664d70178a8SVladimir Sementsov-Ogievskiy if (migration_is_running(s->state) || migration_in_colo_state()) { 66545c1de13SJuan Quintela error_setg(errp, QERR_MIGRATION_ACTIVE); 66645c1de13SJuan Quintela return; 66745c1de13SJuan Quintela } 66845c1de13SJuan Quintela 66945c1de13SJuan Quintela memcpy(new_caps, s->capabilities, sizeof(new_caps)); 67045c1de13SJuan Quintela for (cap = params; cap; cap = cap->next) { 67145c1de13SJuan Quintela new_caps[cap->value->capability] = cap->value->state; 67245c1de13SJuan Quintela } 67345c1de13SJuan Quintela 67445c1de13SJuan Quintela if (!migrate_caps_check(s->capabilities, new_caps, errp)) { 67545c1de13SJuan Quintela return; 67645c1de13SJuan Quintela } 67745c1de13SJuan Quintela 67845c1de13SJuan Quintela for (cap = params; cap; cap = cap->next) { 67945c1de13SJuan Quintela s->capabilities[cap->value->capability] = cap->value->state; 68045c1de13SJuan Quintela } 68145c1de13SJuan Quintela } 6821dfc4b9eSJuan Quintela 6831dfc4b9eSJuan Quintela /* parameters */ 6841dfc4b9eSJuan Quintela 685b804b35bSJuan Quintela const BitmapMigrationNodeAliasList *migrate_block_bitmap_mapping(void) 686b804b35bSJuan Quintela { 687b804b35bSJuan Quintela MigrationState *s = migrate_get_current(); 688b804b35bSJuan Quintela 689b804b35bSJuan Quintela return s->parameters.block_bitmap_mapping; 690b804b35bSJuan Quintela } 691b804b35bSJuan Quintela 6923cba22c9SJuan Quintela bool migrate_has_block_bitmap_mapping(void) 6933cba22c9SJuan Quintela { 6943cba22c9SJuan Quintela MigrationState *s = migrate_get_current(); 6953cba22c9SJuan Quintela 6963cba22c9SJuan Quintela return s->parameters.has_block_bitmap_mapping; 6973cba22c9SJuan Quintela } 6983cba22c9SJuan Quintela 6996f8be708SJuan Quintela bool migrate_block_incremental(void) 7006f8be708SJuan Quintela { 7018f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7026f8be708SJuan Quintela 7036f8be708SJuan Quintela return s->parameters.block_incremental; 7046f8be708SJuan Quintela } 7056f8be708SJuan Quintela 706f94a858fSJuan Quintela uint32_t migrate_checkpoint_delay(void) 707f94a858fSJuan Quintela { 7088f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 709f94a858fSJuan Quintela 710f94a858fSJuan Quintela return s->parameters.x_checkpoint_delay; 711f94a858fSJuan Quintela } 712f94a858fSJuan Quintela 7131dfc4b9eSJuan Quintela int migrate_compress_level(void) 7141dfc4b9eSJuan Quintela { 7158f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7161dfc4b9eSJuan Quintela 7171dfc4b9eSJuan Quintela return s->parameters.compress_level; 7181dfc4b9eSJuan Quintela } 7191dfc4b9eSJuan Quintela 7201dfc4b9eSJuan Quintela int migrate_compress_threads(void) 7211dfc4b9eSJuan Quintela { 7228f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7231dfc4b9eSJuan Quintela 7241dfc4b9eSJuan Quintela return s->parameters.compress_threads; 7251dfc4b9eSJuan Quintela } 7261dfc4b9eSJuan Quintela 7271dfc4b9eSJuan Quintela int migrate_compress_wait_thread(void) 7281dfc4b9eSJuan Quintela { 7298f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7301dfc4b9eSJuan Quintela 7311dfc4b9eSJuan Quintela return s->parameters.compress_wait_thread; 7321dfc4b9eSJuan Quintela } 7331dfc4b9eSJuan Quintela 7349605c2acSJuan Quintela uint8_t migrate_cpu_throttle_increment(void) 7359605c2acSJuan Quintela { 7368f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7379605c2acSJuan Quintela 7389605c2acSJuan Quintela return s->parameters.cpu_throttle_increment; 7399605c2acSJuan Quintela } 7409605c2acSJuan Quintela 7412a8ec380SJuan Quintela uint8_t migrate_cpu_throttle_initial(void) 7422a8ec380SJuan Quintela { 7438f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7442a8ec380SJuan Quintela 7452a8ec380SJuan Quintela return s->parameters.cpu_throttle_initial; 7462a8ec380SJuan Quintela } 7472a8ec380SJuan Quintela 748873f674cSJuan Quintela bool migrate_cpu_throttle_tailslow(void) 749873f674cSJuan Quintela { 7508f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 751873f674cSJuan Quintela 752873f674cSJuan Quintela return s->parameters.cpu_throttle_tailslow; 753873f674cSJuan Quintela } 754873f674cSJuan Quintela 7551dfc4b9eSJuan Quintela int migrate_decompress_threads(void) 7561dfc4b9eSJuan Quintela { 7578f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7581dfc4b9eSJuan Quintela 7591dfc4b9eSJuan Quintela return s->parameters.decompress_threads; 7601dfc4b9eSJuan Quintela } 7611dfc4b9eSJuan Quintela 762f5da8ba4SJuan Quintela uint64_t migrate_downtime_limit(void) 763f5da8ba4SJuan Quintela { 764f5da8ba4SJuan Quintela MigrationState *s = migrate_get_current(); 765f5da8ba4SJuan Quintela 766f5da8ba4SJuan Quintela return s->parameters.downtime_limit; 767f5da8ba4SJuan Quintela } 768f5da8ba4SJuan Quintela 76924155bd0SJuan Quintela uint8_t migrate_max_cpu_throttle(void) 77024155bd0SJuan Quintela { 7718f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 77224155bd0SJuan Quintela 77324155bd0SJuan Quintela return s->parameters.max_cpu_throttle; 77424155bd0SJuan Quintela } 77524155bd0SJuan Quintela 7769c894df3SJuan Quintela uint64_t migrate_max_bandwidth(void) 7779c894df3SJuan Quintela { 7788f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7799c894df3SJuan Quintela 7809c894df3SJuan Quintela return s->parameters.max_bandwidth; 7819c894df3SJuan Quintela } 7829c894df3SJuan Quintela 78352033349SJuan Quintela uint64_t migrate_max_postcopy_bandwidth(void) 7841dfc4b9eSJuan Quintela { 7858f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7861dfc4b9eSJuan Quintela 7871dfc4b9eSJuan Quintela return s->parameters.max_postcopy_bandwidth; 7881dfc4b9eSJuan Quintela } 7891dfc4b9eSJuan Quintela 7901dfc4b9eSJuan Quintela int migrate_multifd_channels(void) 7911dfc4b9eSJuan Quintela { 7928f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 7931dfc4b9eSJuan Quintela 7941dfc4b9eSJuan Quintela return s->parameters.multifd_channels; 7951dfc4b9eSJuan Quintela } 7961dfc4b9eSJuan Quintela 7971dfc4b9eSJuan Quintela MultiFDCompression migrate_multifd_compression(void) 7981dfc4b9eSJuan Quintela { 7998f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 8001dfc4b9eSJuan Quintela 8011dfc4b9eSJuan Quintela assert(s->parameters.multifd_compression < MULTIFD_COMPRESSION__MAX); 8021dfc4b9eSJuan Quintela return s->parameters.multifd_compression; 8031dfc4b9eSJuan Quintela } 8041dfc4b9eSJuan Quintela 8051dfc4b9eSJuan Quintela int migrate_multifd_zlib_level(void) 8061dfc4b9eSJuan Quintela { 8078f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 8081dfc4b9eSJuan Quintela 8091dfc4b9eSJuan Quintela return s->parameters.multifd_zlib_level; 8101dfc4b9eSJuan Quintela } 8111dfc4b9eSJuan Quintela 8121dfc4b9eSJuan Quintela int migrate_multifd_zstd_level(void) 8131dfc4b9eSJuan Quintela { 8148f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 8151dfc4b9eSJuan Quintela 8161dfc4b9eSJuan Quintela return s->parameters.multifd_zstd_level; 8171dfc4b9eSJuan Quintela } 8181dfc4b9eSJuan Quintela 8196499efdbSJuan Quintela uint8_t migrate_throttle_trigger_threshold(void) 8206499efdbSJuan Quintela { 8218f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 8226499efdbSJuan Quintela 8236499efdbSJuan Quintela return s->parameters.throttle_trigger_threshold; 8246499efdbSJuan Quintela } 8256499efdbSJuan Quintela 8262eb0308bSJuan Quintela const char *migrate_tls_authz(void) 8272eb0308bSJuan Quintela { 8282eb0308bSJuan Quintela MigrationState *s = migrate_get_current(); 8292eb0308bSJuan Quintela 8302eb0308bSJuan Quintela return s->parameters.tls_authz; 8312eb0308bSJuan Quintela } 8322eb0308bSJuan Quintela 833d5c3e195SJuan Quintela const char *migrate_tls_creds(void) 834d5c3e195SJuan Quintela { 835d5c3e195SJuan Quintela MigrationState *s = migrate_get_current(); 836d5c3e195SJuan Quintela 837d5c3e195SJuan Quintela return s->parameters.tls_creds; 838d5c3e195SJuan Quintela } 839d5c3e195SJuan Quintela 8401f2f366cSJuan Quintela const char *migrate_tls_hostname(void) 8411f2f366cSJuan Quintela { 8421f2f366cSJuan Quintela MigrationState *s = migrate_get_current(); 8431f2f366cSJuan Quintela 8441f2f366cSJuan Quintela return s->parameters.tls_hostname; 8451f2f366cSJuan Quintela } 8461f2f366cSJuan Quintela 8471dfc4b9eSJuan Quintela uint64_t migrate_xbzrle_cache_size(void) 8481dfc4b9eSJuan Quintela { 8498f9c5327SJuan Quintela MigrationState *s = migrate_get_current(); 8501dfc4b9eSJuan Quintela 8511dfc4b9eSJuan Quintela return s->parameters.xbzrle_cache_size; 8521dfc4b9eSJuan Quintela } 8532682c4eeSJuan Quintela 85487c22901SJuan Quintela /* parameter setters */ 85587c22901SJuan Quintela 85687c22901SJuan Quintela void migrate_set_block_incremental(bool value) 85787c22901SJuan Quintela { 85887c22901SJuan Quintela MigrationState *s = migrate_get_current(); 85987c22901SJuan Quintela 86087c22901SJuan Quintela s->parameters.block_incremental = value; 86187c22901SJuan Quintela } 86287c22901SJuan Quintela 8632682c4eeSJuan Quintela /* parameters helpers */ 8642682c4eeSJuan Quintela 865b1a87956SJuan Quintela void block_cleanup_parameters(void) 866b7b73122SJuan Quintela { 867b1a87956SJuan Quintela MigrationState *s = migrate_get_current(); 868b1a87956SJuan Quintela 869b7b73122SJuan Quintela if (s->must_remove_block_options) { 870b7b73122SJuan Quintela /* setting to false can never fail */ 871b7b73122SJuan Quintela migrate_cap_set(MIGRATION_CAPABILITY_BLOCK, false, &error_abort); 872b7b73122SJuan Quintela migrate_set_block_incremental(false); 873b7b73122SJuan Quintela s->must_remove_block_options = false; 874b7b73122SJuan Quintela } 875b7b73122SJuan Quintela } 876b7b73122SJuan Quintela 8772682c4eeSJuan Quintela AnnounceParameters *migrate_announce_params(void) 8782682c4eeSJuan Quintela { 8792682c4eeSJuan Quintela static AnnounceParameters ap; 8802682c4eeSJuan Quintela 8812682c4eeSJuan Quintela MigrationState *s = migrate_get_current(); 8822682c4eeSJuan Quintela 8832682c4eeSJuan Quintela ap.initial = s->parameters.announce_initial; 8842682c4eeSJuan Quintela ap.max = s->parameters.announce_max; 8852682c4eeSJuan Quintela ap.rounds = s->parameters.announce_rounds; 8862682c4eeSJuan Quintela ap.step = s->parameters.announce_step; 8872682c4eeSJuan Quintela 8882682c4eeSJuan Quintela return ≈ 8892682c4eeSJuan Quintela } 8909c894df3SJuan Quintela 8919c894df3SJuan Quintela MigrationParameters *qmp_query_migrate_parameters(Error **errp) 8929c894df3SJuan Quintela { 8939c894df3SJuan Quintela MigrationParameters *params; 8949c894df3SJuan Quintela MigrationState *s = migrate_get_current(); 8959c894df3SJuan Quintela 8969c894df3SJuan Quintela /* TODO use QAPI_CLONE() instead of duplicating it inline */ 8979c894df3SJuan Quintela params = g_malloc0(sizeof(*params)); 8989c894df3SJuan Quintela params->has_compress_level = true; 8999c894df3SJuan Quintela params->compress_level = s->parameters.compress_level; 9009c894df3SJuan Quintela params->has_compress_threads = true; 9019c894df3SJuan Quintela params->compress_threads = s->parameters.compress_threads; 9029c894df3SJuan Quintela params->has_compress_wait_thread = true; 9039c894df3SJuan Quintela params->compress_wait_thread = s->parameters.compress_wait_thread; 9049c894df3SJuan Quintela params->has_decompress_threads = true; 9059c894df3SJuan Quintela params->decompress_threads = s->parameters.decompress_threads; 9069c894df3SJuan Quintela params->has_throttle_trigger_threshold = true; 9079c894df3SJuan Quintela params->throttle_trigger_threshold = s->parameters.throttle_trigger_threshold; 9089c894df3SJuan Quintela params->has_cpu_throttle_initial = true; 9099c894df3SJuan Quintela params->cpu_throttle_initial = s->parameters.cpu_throttle_initial; 9109c894df3SJuan Quintela params->has_cpu_throttle_increment = true; 9119c894df3SJuan Quintela params->cpu_throttle_increment = s->parameters.cpu_throttle_increment; 9129c894df3SJuan Quintela params->has_cpu_throttle_tailslow = true; 9139c894df3SJuan Quintela params->cpu_throttle_tailslow = s->parameters.cpu_throttle_tailslow; 9149c894df3SJuan Quintela params->tls_creds = g_strdup(s->parameters.tls_creds); 9159c894df3SJuan Quintela params->tls_hostname = g_strdup(s->parameters.tls_hostname); 9169c894df3SJuan Quintela params->tls_authz = g_strdup(s->parameters.tls_authz ? 9179c894df3SJuan Quintela s->parameters.tls_authz : ""); 9189c894df3SJuan Quintela params->has_max_bandwidth = true; 9199c894df3SJuan Quintela params->max_bandwidth = s->parameters.max_bandwidth; 9209c894df3SJuan Quintela params->has_downtime_limit = true; 9219c894df3SJuan Quintela params->downtime_limit = s->parameters.downtime_limit; 9229c894df3SJuan Quintela params->has_x_checkpoint_delay = true; 9239c894df3SJuan Quintela params->x_checkpoint_delay = s->parameters.x_checkpoint_delay; 9249c894df3SJuan Quintela params->has_block_incremental = true; 9259c894df3SJuan Quintela params->block_incremental = s->parameters.block_incremental; 9269c894df3SJuan Quintela params->has_multifd_channels = true; 9279c894df3SJuan Quintela params->multifd_channels = s->parameters.multifd_channels; 9289c894df3SJuan Quintela params->has_multifd_compression = true; 9299c894df3SJuan Quintela params->multifd_compression = s->parameters.multifd_compression; 9309c894df3SJuan Quintela params->has_multifd_zlib_level = true; 9319c894df3SJuan Quintela params->multifd_zlib_level = s->parameters.multifd_zlib_level; 9329c894df3SJuan Quintela params->has_multifd_zstd_level = true; 9339c894df3SJuan Quintela params->multifd_zstd_level = s->parameters.multifd_zstd_level; 9349c894df3SJuan Quintela params->has_xbzrle_cache_size = true; 9359c894df3SJuan Quintela params->xbzrle_cache_size = s->parameters.xbzrle_cache_size; 9369c894df3SJuan Quintela params->has_max_postcopy_bandwidth = true; 9379c894df3SJuan Quintela params->max_postcopy_bandwidth = s->parameters.max_postcopy_bandwidth; 9389c894df3SJuan Quintela params->has_max_cpu_throttle = true; 9399c894df3SJuan Quintela params->max_cpu_throttle = s->parameters.max_cpu_throttle; 9409c894df3SJuan Quintela params->has_announce_initial = true; 9419c894df3SJuan Quintela params->announce_initial = s->parameters.announce_initial; 9429c894df3SJuan Quintela params->has_announce_max = true; 9439c894df3SJuan Quintela params->announce_max = s->parameters.announce_max; 9449c894df3SJuan Quintela params->has_announce_rounds = true; 9459c894df3SJuan Quintela params->announce_rounds = s->parameters.announce_rounds; 9469c894df3SJuan Quintela params->has_announce_step = true; 9479c894df3SJuan Quintela params->announce_step = s->parameters.announce_step; 9489c894df3SJuan Quintela 9499c894df3SJuan Quintela if (s->parameters.has_block_bitmap_mapping) { 9509c894df3SJuan Quintela params->has_block_bitmap_mapping = true; 9519c894df3SJuan Quintela params->block_bitmap_mapping = 9529c894df3SJuan Quintela QAPI_CLONE(BitmapMigrationNodeAliasList, 9539c894df3SJuan Quintela s->parameters.block_bitmap_mapping); 9549c894df3SJuan Quintela } 9559c894df3SJuan Quintela 9564d807857SHyman Huang(黄勇) params->has_x_vcpu_dirty_limit_period = true; 9574d807857SHyman Huang(黄勇) params->x_vcpu_dirty_limit_period = s->parameters.x_vcpu_dirty_limit_period; 95809f9ec99SHyman Huang(黄勇) params->has_vcpu_dirty_limit = true; 95909f9ec99SHyman Huang(黄勇) params->vcpu_dirty_limit = s->parameters.vcpu_dirty_limit; 9604d807857SHyman Huang(黄勇) 9619c894df3SJuan Quintela return params; 9629c894df3SJuan Quintela } 96309d6c965SJuan Quintela 96461a174e2SJuan Quintela void migrate_params_init(MigrationParameters *params) 96561a174e2SJuan Quintela { 96661a174e2SJuan Quintela params->tls_hostname = g_strdup(""); 96761a174e2SJuan Quintela params->tls_creds = g_strdup(""); 96861a174e2SJuan Quintela 96961a174e2SJuan Quintela /* Set has_* up only for parameter checks */ 97061a174e2SJuan Quintela params->has_compress_level = true; 97161a174e2SJuan Quintela params->has_compress_threads = true; 97261a174e2SJuan Quintela params->has_compress_wait_thread = true; 97361a174e2SJuan Quintela params->has_decompress_threads = true; 97461a174e2SJuan Quintela params->has_throttle_trigger_threshold = true; 97561a174e2SJuan Quintela params->has_cpu_throttle_initial = true; 97661a174e2SJuan Quintela params->has_cpu_throttle_increment = true; 97761a174e2SJuan Quintela params->has_cpu_throttle_tailslow = true; 97861a174e2SJuan Quintela params->has_max_bandwidth = true; 97961a174e2SJuan Quintela params->has_downtime_limit = true; 98061a174e2SJuan Quintela params->has_x_checkpoint_delay = true; 98161a174e2SJuan Quintela params->has_block_incremental = true; 98261a174e2SJuan Quintela params->has_multifd_channels = true; 98361a174e2SJuan Quintela params->has_multifd_compression = true; 98461a174e2SJuan Quintela params->has_multifd_zlib_level = true; 98561a174e2SJuan Quintela params->has_multifd_zstd_level = true; 98661a174e2SJuan Quintela params->has_xbzrle_cache_size = true; 98761a174e2SJuan Quintela params->has_max_postcopy_bandwidth = true; 98861a174e2SJuan Quintela params->has_max_cpu_throttle = true; 98961a174e2SJuan Quintela params->has_announce_initial = true; 99061a174e2SJuan Quintela params->has_announce_max = true; 99161a174e2SJuan Quintela params->has_announce_rounds = true; 99261a174e2SJuan Quintela params->has_announce_step = true; 9934d807857SHyman Huang(黄勇) params->has_x_vcpu_dirty_limit_period = true; 99409f9ec99SHyman Huang(黄勇) params->has_vcpu_dirty_limit = true; 99561a174e2SJuan Quintela } 99661a174e2SJuan Quintela 99709d6c965SJuan Quintela /* 99809d6c965SJuan Quintela * Check whether the parameters are valid. Error will be put into errp 99909d6c965SJuan Quintela * (if provided). Return true if valid, otherwise false. 100009d6c965SJuan Quintela */ 100109d6c965SJuan Quintela bool migrate_params_check(MigrationParameters *params, Error **errp) 100209d6c965SJuan Quintela { 100309d6c965SJuan Quintela if (params->has_compress_level && 100409d6c965SJuan Quintela (params->compress_level > 9)) { 100509d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, "compress_level", 100609d6c965SJuan Quintela "a value between 0 and 9"); 100709d6c965SJuan Quintela return false; 100809d6c965SJuan Quintela } 100909d6c965SJuan Quintela 101009d6c965SJuan Quintela if (params->has_compress_threads && (params->compress_threads < 1)) { 101109d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 101209d6c965SJuan Quintela "compress_threads", 101309d6c965SJuan Quintela "a value between 1 and 255"); 101409d6c965SJuan Quintela return false; 101509d6c965SJuan Quintela } 101609d6c965SJuan Quintela 101709d6c965SJuan Quintela if (params->has_decompress_threads && (params->decompress_threads < 1)) { 101809d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 101909d6c965SJuan Quintela "decompress_threads", 102009d6c965SJuan Quintela "a value between 1 and 255"); 102109d6c965SJuan Quintela return false; 102209d6c965SJuan Quintela } 102309d6c965SJuan Quintela 102409d6c965SJuan Quintela if (params->has_throttle_trigger_threshold && 102509d6c965SJuan Quintela (params->throttle_trigger_threshold < 1 || 102609d6c965SJuan Quintela params->throttle_trigger_threshold > 100)) { 102709d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 102809d6c965SJuan Quintela "throttle_trigger_threshold", 102909d6c965SJuan Quintela "an integer in the range of 1 to 100"); 103009d6c965SJuan Quintela return false; 103109d6c965SJuan Quintela } 103209d6c965SJuan Quintela 103309d6c965SJuan Quintela if (params->has_cpu_throttle_initial && 103409d6c965SJuan Quintela (params->cpu_throttle_initial < 1 || 103509d6c965SJuan Quintela params->cpu_throttle_initial > 99)) { 103609d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 103709d6c965SJuan Quintela "cpu_throttle_initial", 103809d6c965SJuan Quintela "an integer in the range of 1 to 99"); 103909d6c965SJuan Quintela return false; 104009d6c965SJuan Quintela } 104109d6c965SJuan Quintela 104209d6c965SJuan Quintela if (params->has_cpu_throttle_increment && 104309d6c965SJuan Quintela (params->cpu_throttle_increment < 1 || 104409d6c965SJuan Quintela params->cpu_throttle_increment > 99)) { 104509d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 104609d6c965SJuan Quintela "cpu_throttle_increment", 104709d6c965SJuan Quintela "an integer in the range of 1 to 99"); 104809d6c965SJuan Quintela return false; 104909d6c965SJuan Quintela } 105009d6c965SJuan Quintela 105109d6c965SJuan Quintela if (params->has_max_bandwidth && (params->max_bandwidth > SIZE_MAX)) { 105209d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 105309d6c965SJuan Quintela "max_bandwidth", 105409d6c965SJuan Quintela "an integer in the range of 0 to "stringify(SIZE_MAX) 105509d6c965SJuan Quintela " bytes/second"); 105609d6c965SJuan Quintela return false; 105709d6c965SJuan Quintela } 105809d6c965SJuan Quintela 105909d6c965SJuan Quintela if (params->has_downtime_limit && 106009d6c965SJuan Quintela (params->downtime_limit > MAX_MIGRATE_DOWNTIME)) { 106109d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 106209d6c965SJuan Quintela "downtime_limit", 106309d6c965SJuan Quintela "an integer in the range of 0 to " 106409d6c965SJuan Quintela stringify(MAX_MIGRATE_DOWNTIME)" ms"); 106509d6c965SJuan Quintela return false; 106609d6c965SJuan Quintela } 106709d6c965SJuan Quintela 106809d6c965SJuan Quintela /* x_checkpoint_delay is now always positive */ 106909d6c965SJuan Quintela 107009d6c965SJuan Quintela if (params->has_multifd_channels && (params->multifd_channels < 1)) { 107109d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 107209d6c965SJuan Quintela "multifd_channels", 107309d6c965SJuan Quintela "a value between 1 and 255"); 107409d6c965SJuan Quintela return false; 107509d6c965SJuan Quintela } 107609d6c965SJuan Quintela 107709d6c965SJuan Quintela if (params->has_multifd_zlib_level && 107809d6c965SJuan Quintela (params->multifd_zlib_level > 9)) { 107909d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, "multifd_zlib_level", 108009d6c965SJuan Quintela "a value between 0 and 9"); 108109d6c965SJuan Quintela return false; 108209d6c965SJuan Quintela } 108309d6c965SJuan Quintela 108409d6c965SJuan Quintela if (params->has_multifd_zstd_level && 108509d6c965SJuan Quintela (params->multifd_zstd_level > 20)) { 108609d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, "multifd_zstd_level", 108709d6c965SJuan Quintela "a value between 0 and 20"); 108809d6c965SJuan Quintela return false; 108909d6c965SJuan Quintela } 109009d6c965SJuan Quintela 109109d6c965SJuan Quintela if (params->has_xbzrle_cache_size && 109209d6c965SJuan Quintela (params->xbzrle_cache_size < qemu_target_page_size() || 109309d6c965SJuan Quintela !is_power_of_2(params->xbzrle_cache_size))) { 109409d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 109509d6c965SJuan Quintela "xbzrle_cache_size", 109609d6c965SJuan Quintela "a power of two no less than the target page size"); 109709d6c965SJuan Quintela return false; 109809d6c965SJuan Quintela } 109909d6c965SJuan Quintela 110009d6c965SJuan Quintela if (params->has_max_cpu_throttle && 110109d6c965SJuan Quintela (params->max_cpu_throttle < params->cpu_throttle_initial || 110209d6c965SJuan Quintela params->max_cpu_throttle > 99)) { 110309d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 110409d6c965SJuan Quintela "max_cpu_throttle", 110509d6c965SJuan Quintela "an integer in the range of cpu_throttle_initial to 99"); 110609d6c965SJuan Quintela return false; 110709d6c965SJuan Quintela } 110809d6c965SJuan Quintela 110909d6c965SJuan Quintela if (params->has_announce_initial && 111009d6c965SJuan Quintela params->announce_initial > 100000) { 111109d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 111209d6c965SJuan Quintela "announce_initial", 111309d6c965SJuan Quintela "a value between 0 and 100000"); 111409d6c965SJuan Quintela return false; 111509d6c965SJuan Quintela } 111609d6c965SJuan Quintela if (params->has_announce_max && 111709d6c965SJuan Quintela params->announce_max > 100000) { 111809d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 111909d6c965SJuan Quintela "announce_max", 112009d6c965SJuan Quintela "a value between 0 and 100000"); 112109d6c965SJuan Quintela return false; 112209d6c965SJuan Quintela } 112309d6c965SJuan Quintela if (params->has_announce_rounds && 112409d6c965SJuan Quintela params->announce_rounds > 1000) { 112509d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 112609d6c965SJuan Quintela "announce_rounds", 112709d6c965SJuan Quintela "a value between 0 and 1000"); 112809d6c965SJuan Quintela return false; 112909d6c965SJuan Quintela } 113009d6c965SJuan Quintela if (params->has_announce_step && 113109d6c965SJuan Quintela (params->announce_step < 1 || 113209d6c965SJuan Quintela params->announce_step > 10000)) { 113309d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 113409d6c965SJuan Quintela "announce_step", 113509d6c965SJuan Quintela "a value between 0 and 10000"); 113609d6c965SJuan Quintela return false; 113709d6c965SJuan Quintela } 113809d6c965SJuan Quintela 113909d6c965SJuan Quintela if (params->has_block_bitmap_mapping && 114009d6c965SJuan Quintela !check_dirty_bitmap_mig_alias_map(params->block_bitmap_mapping, errp)) { 114109d6c965SJuan Quintela error_prepend(errp, "Invalid mapping given for block-bitmap-mapping: "); 114209d6c965SJuan Quintela return false; 114309d6c965SJuan Quintela } 114409d6c965SJuan Quintela 114509d6c965SJuan Quintela #ifdef CONFIG_LINUX 114609d6c965SJuan Quintela if (migrate_zero_copy_send() && 114709d6c965SJuan Quintela ((params->has_multifd_compression && params->multifd_compression) || 114809d6c965SJuan Quintela (params->tls_creds && *params->tls_creds))) { 114909d6c965SJuan Quintela error_setg(errp, 115009d6c965SJuan Quintela "Zero copy only available for non-compressed non-TLS multifd migration"); 115109d6c965SJuan Quintela return false; 115209d6c965SJuan Quintela } 115309d6c965SJuan Quintela #endif 115409d6c965SJuan Quintela 11554d807857SHyman Huang(黄勇) if (params->has_x_vcpu_dirty_limit_period && 11564d807857SHyman Huang(黄勇) (params->x_vcpu_dirty_limit_period < 1 || 11574d807857SHyman Huang(黄勇) params->x_vcpu_dirty_limit_period > 1000)) { 11584d807857SHyman Huang(黄勇) error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 11594d807857SHyman Huang(黄勇) "x-vcpu-dirty-limit-period", 11604d807857SHyman Huang(黄勇) "a value between 1 and 1000"); 11614d807857SHyman Huang(黄勇) return false; 11624d807857SHyman Huang(黄勇) } 11634d807857SHyman Huang(黄勇) 116409f9ec99SHyman Huang(黄勇) if (params->has_vcpu_dirty_limit && 116509f9ec99SHyman Huang(黄勇) (params->vcpu_dirty_limit < 1)) { 116609f9ec99SHyman Huang(黄勇) error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 116709f9ec99SHyman Huang(黄勇) "vcpu_dirty_limit", 116809f9ec99SHyman Huang(黄勇) "is invalid, it must greater then 1 MB/s"); 116909f9ec99SHyman Huang(黄勇) return false; 117009f9ec99SHyman Huang(黄勇) } 117109f9ec99SHyman Huang(黄勇) 117209d6c965SJuan Quintela return true; 117309d6c965SJuan Quintela } 117409d6c965SJuan Quintela 117509d6c965SJuan Quintela static void migrate_params_test_apply(MigrateSetParameters *params, 117609d6c965SJuan Quintela MigrationParameters *dest) 117709d6c965SJuan Quintela { 117809d6c965SJuan Quintela *dest = migrate_get_current()->parameters; 117909d6c965SJuan Quintela 118009d6c965SJuan Quintela /* TODO use QAPI_CLONE() instead of duplicating it inline */ 118109d6c965SJuan Quintela 118209d6c965SJuan Quintela if (params->has_compress_level) { 118309d6c965SJuan Quintela dest->compress_level = params->compress_level; 118409d6c965SJuan Quintela } 118509d6c965SJuan Quintela 118609d6c965SJuan Quintela if (params->has_compress_threads) { 118709d6c965SJuan Quintela dest->compress_threads = params->compress_threads; 118809d6c965SJuan Quintela } 118909d6c965SJuan Quintela 119009d6c965SJuan Quintela if (params->has_compress_wait_thread) { 119109d6c965SJuan Quintela dest->compress_wait_thread = params->compress_wait_thread; 119209d6c965SJuan Quintela } 119309d6c965SJuan Quintela 119409d6c965SJuan Quintela if (params->has_decompress_threads) { 119509d6c965SJuan Quintela dest->decompress_threads = params->decompress_threads; 119609d6c965SJuan Quintela } 119709d6c965SJuan Quintela 119809d6c965SJuan Quintela if (params->has_throttle_trigger_threshold) { 119909d6c965SJuan Quintela dest->throttle_trigger_threshold = params->throttle_trigger_threshold; 120009d6c965SJuan Quintela } 120109d6c965SJuan Quintela 120209d6c965SJuan Quintela if (params->has_cpu_throttle_initial) { 120309d6c965SJuan Quintela dest->cpu_throttle_initial = params->cpu_throttle_initial; 120409d6c965SJuan Quintela } 120509d6c965SJuan Quintela 120609d6c965SJuan Quintela if (params->has_cpu_throttle_increment) { 120709d6c965SJuan Quintela dest->cpu_throttle_increment = params->cpu_throttle_increment; 120809d6c965SJuan Quintela } 120909d6c965SJuan Quintela 121009d6c965SJuan Quintela if (params->has_cpu_throttle_tailslow) { 121109d6c965SJuan Quintela dest->cpu_throttle_tailslow = params->cpu_throttle_tailslow; 121209d6c965SJuan Quintela } 121309d6c965SJuan Quintela 121409d6c965SJuan Quintela if (params->tls_creds) { 121509d6c965SJuan Quintela assert(params->tls_creds->type == QTYPE_QSTRING); 121609d6c965SJuan Quintela dest->tls_creds = params->tls_creds->u.s; 121709d6c965SJuan Quintela } 121809d6c965SJuan Quintela 121909d6c965SJuan Quintela if (params->tls_hostname) { 122009d6c965SJuan Quintela assert(params->tls_hostname->type == QTYPE_QSTRING); 122109d6c965SJuan Quintela dest->tls_hostname = params->tls_hostname->u.s; 122209d6c965SJuan Quintela } 122309d6c965SJuan Quintela 122409d6c965SJuan Quintela if (params->has_max_bandwidth) { 122509d6c965SJuan Quintela dest->max_bandwidth = params->max_bandwidth; 122609d6c965SJuan Quintela } 122709d6c965SJuan Quintela 122809d6c965SJuan Quintela if (params->has_downtime_limit) { 122909d6c965SJuan Quintela dest->downtime_limit = params->downtime_limit; 123009d6c965SJuan Quintela } 123109d6c965SJuan Quintela 123209d6c965SJuan Quintela if (params->has_x_checkpoint_delay) { 123309d6c965SJuan Quintela dest->x_checkpoint_delay = params->x_checkpoint_delay; 123409d6c965SJuan Quintela } 123509d6c965SJuan Quintela 123609d6c965SJuan Quintela if (params->has_block_incremental) { 123709d6c965SJuan Quintela dest->block_incremental = params->block_incremental; 123809d6c965SJuan Quintela } 123909d6c965SJuan Quintela if (params->has_multifd_channels) { 124009d6c965SJuan Quintela dest->multifd_channels = params->multifd_channels; 124109d6c965SJuan Quintela } 124209d6c965SJuan Quintela if (params->has_multifd_compression) { 124309d6c965SJuan Quintela dest->multifd_compression = params->multifd_compression; 124409d6c965SJuan Quintela } 124509d6c965SJuan Quintela if (params->has_xbzrle_cache_size) { 124609d6c965SJuan Quintela dest->xbzrle_cache_size = params->xbzrle_cache_size; 124709d6c965SJuan Quintela } 124809d6c965SJuan Quintela if (params->has_max_postcopy_bandwidth) { 124909d6c965SJuan Quintela dest->max_postcopy_bandwidth = params->max_postcopy_bandwidth; 125009d6c965SJuan Quintela } 125109d6c965SJuan Quintela if (params->has_max_cpu_throttle) { 125209d6c965SJuan Quintela dest->max_cpu_throttle = params->max_cpu_throttle; 125309d6c965SJuan Quintela } 125409d6c965SJuan Quintela if (params->has_announce_initial) { 125509d6c965SJuan Quintela dest->announce_initial = params->announce_initial; 125609d6c965SJuan Quintela } 125709d6c965SJuan Quintela if (params->has_announce_max) { 125809d6c965SJuan Quintela dest->announce_max = params->announce_max; 125909d6c965SJuan Quintela } 126009d6c965SJuan Quintela if (params->has_announce_rounds) { 126109d6c965SJuan Quintela dest->announce_rounds = params->announce_rounds; 126209d6c965SJuan Quintela } 126309d6c965SJuan Quintela if (params->has_announce_step) { 126409d6c965SJuan Quintela dest->announce_step = params->announce_step; 126509d6c965SJuan Quintela } 126609d6c965SJuan Quintela 126709d6c965SJuan Quintela if (params->has_block_bitmap_mapping) { 126809d6c965SJuan Quintela dest->has_block_bitmap_mapping = true; 126909d6c965SJuan Quintela dest->block_bitmap_mapping = params->block_bitmap_mapping; 127009d6c965SJuan Quintela } 12714d807857SHyman Huang(黄勇) 12724d807857SHyman Huang(黄勇) if (params->has_x_vcpu_dirty_limit_period) { 12734d807857SHyman Huang(黄勇) dest->x_vcpu_dirty_limit_period = 12744d807857SHyman Huang(黄勇) params->x_vcpu_dirty_limit_period; 12754d807857SHyman Huang(黄勇) } 127609f9ec99SHyman Huang(黄勇) if (params->has_vcpu_dirty_limit) { 127709f9ec99SHyman Huang(黄勇) dest->vcpu_dirty_limit = params->vcpu_dirty_limit; 127809f9ec99SHyman Huang(黄勇) } 127909d6c965SJuan Quintela } 128009d6c965SJuan Quintela 128109d6c965SJuan Quintela static void migrate_params_apply(MigrateSetParameters *params, Error **errp) 128209d6c965SJuan Quintela { 128309d6c965SJuan Quintela MigrationState *s = migrate_get_current(); 128409d6c965SJuan Quintela 128509d6c965SJuan Quintela /* TODO use QAPI_CLONE() instead of duplicating it inline */ 128609d6c965SJuan Quintela 128709d6c965SJuan Quintela if (params->has_compress_level) { 128809d6c965SJuan Quintela s->parameters.compress_level = params->compress_level; 128909d6c965SJuan Quintela } 129009d6c965SJuan Quintela 129109d6c965SJuan Quintela if (params->has_compress_threads) { 129209d6c965SJuan Quintela s->parameters.compress_threads = params->compress_threads; 129309d6c965SJuan Quintela } 129409d6c965SJuan Quintela 129509d6c965SJuan Quintela if (params->has_compress_wait_thread) { 129609d6c965SJuan Quintela s->parameters.compress_wait_thread = params->compress_wait_thread; 129709d6c965SJuan Quintela } 129809d6c965SJuan Quintela 129909d6c965SJuan Quintela if (params->has_decompress_threads) { 130009d6c965SJuan Quintela s->parameters.decompress_threads = params->decompress_threads; 130109d6c965SJuan Quintela } 130209d6c965SJuan Quintela 130309d6c965SJuan Quintela if (params->has_throttle_trigger_threshold) { 130409d6c965SJuan Quintela s->parameters.throttle_trigger_threshold = params->throttle_trigger_threshold; 130509d6c965SJuan Quintela } 130609d6c965SJuan Quintela 130709d6c965SJuan Quintela if (params->has_cpu_throttle_initial) { 130809d6c965SJuan Quintela s->parameters.cpu_throttle_initial = params->cpu_throttle_initial; 130909d6c965SJuan Quintela } 131009d6c965SJuan Quintela 131109d6c965SJuan Quintela if (params->has_cpu_throttle_increment) { 131209d6c965SJuan Quintela s->parameters.cpu_throttle_increment = params->cpu_throttle_increment; 131309d6c965SJuan Quintela } 131409d6c965SJuan Quintela 131509d6c965SJuan Quintela if (params->has_cpu_throttle_tailslow) { 131609d6c965SJuan Quintela s->parameters.cpu_throttle_tailslow = params->cpu_throttle_tailslow; 131709d6c965SJuan Quintela } 131809d6c965SJuan Quintela 131909d6c965SJuan Quintela if (params->tls_creds) { 132009d6c965SJuan Quintela g_free(s->parameters.tls_creds); 132109d6c965SJuan Quintela assert(params->tls_creds->type == QTYPE_QSTRING); 132209d6c965SJuan Quintela s->parameters.tls_creds = g_strdup(params->tls_creds->u.s); 132309d6c965SJuan Quintela } 132409d6c965SJuan Quintela 132509d6c965SJuan Quintela if (params->tls_hostname) { 132609d6c965SJuan Quintela g_free(s->parameters.tls_hostname); 132709d6c965SJuan Quintela assert(params->tls_hostname->type == QTYPE_QSTRING); 132809d6c965SJuan Quintela s->parameters.tls_hostname = g_strdup(params->tls_hostname->u.s); 132909d6c965SJuan Quintela } 133009d6c965SJuan Quintela 133109d6c965SJuan Quintela if (params->tls_authz) { 133209d6c965SJuan Quintela g_free(s->parameters.tls_authz); 133309d6c965SJuan Quintela assert(params->tls_authz->type == QTYPE_QSTRING); 133409d6c965SJuan Quintela s->parameters.tls_authz = g_strdup(params->tls_authz->u.s); 133509d6c965SJuan Quintela } 133609d6c965SJuan Quintela 133709d6c965SJuan Quintela if (params->has_max_bandwidth) { 133809d6c965SJuan Quintela s->parameters.max_bandwidth = params->max_bandwidth; 133909d6c965SJuan Quintela if (s->to_dst_file && !migration_in_postcopy()) { 1340e1fde0e0SJuan Quintela migration_rate_set(s->parameters.max_bandwidth); 134109d6c965SJuan Quintela } 134209d6c965SJuan Quintela } 134309d6c965SJuan Quintela 134409d6c965SJuan Quintela if (params->has_downtime_limit) { 134509d6c965SJuan Quintela s->parameters.downtime_limit = params->downtime_limit; 134609d6c965SJuan Quintela } 134709d6c965SJuan Quintela 134809d6c965SJuan Quintela if (params->has_x_checkpoint_delay) { 134909d6c965SJuan Quintela s->parameters.x_checkpoint_delay = params->x_checkpoint_delay; 13504332ffcdSVladimir Sementsov-Ogievskiy colo_checkpoint_delay_set(); 135109d6c965SJuan Quintela } 135209d6c965SJuan Quintela 135309d6c965SJuan Quintela if (params->has_block_incremental) { 135409d6c965SJuan Quintela s->parameters.block_incremental = params->block_incremental; 135509d6c965SJuan Quintela } 135609d6c965SJuan Quintela if (params->has_multifd_channels) { 135709d6c965SJuan Quintela s->parameters.multifd_channels = params->multifd_channels; 135809d6c965SJuan Quintela } 135909d6c965SJuan Quintela if (params->has_multifd_compression) { 136009d6c965SJuan Quintela s->parameters.multifd_compression = params->multifd_compression; 136109d6c965SJuan Quintela } 136209d6c965SJuan Quintela if (params->has_xbzrle_cache_size) { 136309d6c965SJuan Quintela s->parameters.xbzrle_cache_size = params->xbzrle_cache_size; 136409d6c965SJuan Quintela xbzrle_cache_resize(params->xbzrle_cache_size, errp); 136509d6c965SJuan Quintela } 136609d6c965SJuan Quintela if (params->has_max_postcopy_bandwidth) { 136709d6c965SJuan Quintela s->parameters.max_postcopy_bandwidth = params->max_postcopy_bandwidth; 136809d6c965SJuan Quintela if (s->to_dst_file && migration_in_postcopy()) { 1369e1fde0e0SJuan Quintela migration_rate_set(s->parameters.max_postcopy_bandwidth); 137009d6c965SJuan Quintela } 137109d6c965SJuan Quintela } 137209d6c965SJuan Quintela if (params->has_max_cpu_throttle) { 137309d6c965SJuan Quintela s->parameters.max_cpu_throttle = params->max_cpu_throttle; 137409d6c965SJuan Quintela } 137509d6c965SJuan Quintela if (params->has_announce_initial) { 137609d6c965SJuan Quintela s->parameters.announce_initial = params->announce_initial; 137709d6c965SJuan Quintela } 137809d6c965SJuan Quintela if (params->has_announce_max) { 137909d6c965SJuan Quintela s->parameters.announce_max = params->announce_max; 138009d6c965SJuan Quintela } 138109d6c965SJuan Quintela if (params->has_announce_rounds) { 138209d6c965SJuan Quintela s->parameters.announce_rounds = params->announce_rounds; 138309d6c965SJuan Quintela } 138409d6c965SJuan Quintela if (params->has_announce_step) { 138509d6c965SJuan Quintela s->parameters.announce_step = params->announce_step; 138609d6c965SJuan Quintela } 138709d6c965SJuan Quintela 138809d6c965SJuan Quintela if (params->has_block_bitmap_mapping) { 138909d6c965SJuan Quintela qapi_free_BitmapMigrationNodeAliasList( 139009d6c965SJuan Quintela s->parameters.block_bitmap_mapping); 139109d6c965SJuan Quintela 139209d6c965SJuan Quintela s->parameters.has_block_bitmap_mapping = true; 139309d6c965SJuan Quintela s->parameters.block_bitmap_mapping = 139409d6c965SJuan Quintela QAPI_CLONE(BitmapMigrationNodeAliasList, 139509d6c965SJuan Quintela params->block_bitmap_mapping); 139609d6c965SJuan Quintela } 13974d807857SHyman Huang(黄勇) 13984d807857SHyman Huang(黄勇) if (params->has_x_vcpu_dirty_limit_period) { 13994d807857SHyman Huang(黄勇) s->parameters.x_vcpu_dirty_limit_period = 14004d807857SHyman Huang(黄勇) params->x_vcpu_dirty_limit_period; 14014d807857SHyman Huang(黄勇) } 140209f9ec99SHyman Huang(黄勇) if (params->has_vcpu_dirty_limit) { 140309f9ec99SHyman Huang(黄勇) s->parameters.vcpu_dirty_limit = params->vcpu_dirty_limit; 140409f9ec99SHyman Huang(黄勇) } 140509d6c965SJuan Quintela } 140609d6c965SJuan Quintela 140709d6c965SJuan Quintela void qmp_migrate_set_parameters(MigrateSetParameters *params, Error **errp) 140809d6c965SJuan Quintela { 140909d6c965SJuan Quintela MigrationParameters tmp; 141009d6c965SJuan Quintela 141109d6c965SJuan Quintela /* TODO Rewrite "" to null instead */ 141209d6c965SJuan Quintela if (params->tls_creds 141309d6c965SJuan Quintela && params->tls_creds->type == QTYPE_QNULL) { 141409d6c965SJuan Quintela qobject_unref(params->tls_creds->u.n); 141509d6c965SJuan Quintela params->tls_creds->type = QTYPE_QSTRING; 141609d6c965SJuan Quintela params->tls_creds->u.s = strdup(""); 141709d6c965SJuan Quintela } 141809d6c965SJuan Quintela /* TODO Rewrite "" to null instead */ 141909d6c965SJuan Quintela if (params->tls_hostname 142009d6c965SJuan Quintela && params->tls_hostname->type == QTYPE_QNULL) { 142109d6c965SJuan Quintela qobject_unref(params->tls_hostname->u.n); 142209d6c965SJuan Quintela params->tls_hostname->type = QTYPE_QSTRING; 142309d6c965SJuan Quintela params->tls_hostname->u.s = strdup(""); 142409d6c965SJuan Quintela } 142509d6c965SJuan Quintela 142609d6c965SJuan Quintela migrate_params_test_apply(params, &tmp); 142709d6c965SJuan Quintela 142809d6c965SJuan Quintela if (!migrate_params_check(&tmp, errp)) { 142909d6c965SJuan Quintela /* Invalid parameter */ 143009d6c965SJuan Quintela return; 143109d6c965SJuan Quintela } 143209d6c965SJuan Quintela 143309d6c965SJuan Quintela migrate_params_apply(params, errp); 143409d6c965SJuan Quintela } 1435