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" 15*09d6c965SJuan 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" 21*09d6c965SJuan Quintela #include "qapi/qmp/qnull.h" 2277608706SJuan Quintela #include "sysemu/runstate.h" 23*09d6c965SJuan Quintela #include "migration/colo.h" 242682c4eeSJuan Quintela #include "migration/misc.h" 251f0776f1SJuan Quintela #include "migration.h" 26*09d6c965SJuan Quintela #include "qemu-file.h" 2777608706SJuan Quintela #include "ram.h" 281f0776f1SJuan Quintela #include "options.h" 291f0776f1SJuan Quintela 30*09d6c965SJuan Quintela /* Maximum migrate downtime set to 2000 seconds */ 31*09d6c965SJuan Quintela #define MAX_MIGRATE_DOWNTIME_SECONDS 2000 32*09d6c965SJuan Quintela #define MAX_MIGRATE_DOWNTIME (MAX_MIGRATE_DOWNTIME_SECONDS * 1000) 33*09d6c965SJuan Quintela 341f0776f1SJuan Quintela bool migrate_auto_converge(void) 351f0776f1SJuan Quintela { 361f0776f1SJuan Quintela MigrationState *s; 371f0776f1SJuan Quintela 381f0776f1SJuan Quintela s = migrate_get_current(); 391f0776f1SJuan Quintela 401f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_AUTO_CONVERGE]; 411f0776f1SJuan Quintela } 421f0776f1SJuan Quintela 431f0776f1SJuan Quintela bool migrate_background_snapshot(void) 441f0776f1SJuan Quintela { 451f0776f1SJuan Quintela MigrationState *s; 461f0776f1SJuan Quintela 471f0776f1SJuan Quintela s = migrate_get_current(); 481f0776f1SJuan Quintela 491f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_BACKGROUND_SNAPSHOT]; 501f0776f1SJuan Quintela } 511f0776f1SJuan Quintela 529d4b1e5fSJuan Quintela bool migrate_block(void) 539d4b1e5fSJuan Quintela { 549d4b1e5fSJuan Quintela MigrationState *s; 559d4b1e5fSJuan Quintela 569d4b1e5fSJuan Quintela s = migrate_get_current(); 579d4b1e5fSJuan Quintela 589d4b1e5fSJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_BLOCK]; 599d4b1e5fSJuan Quintela } 609d4b1e5fSJuan Quintela 615e804644SJuan Quintela bool migrate_colo(void) 625e804644SJuan Quintela { 635e804644SJuan Quintela MigrationState *s = migrate_get_current(); 645e804644SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_X_COLO]; 655e804644SJuan Quintela } 665e804644SJuan Quintela 67a7a94d14SJuan Quintela bool migrate_compress(void) 68a7a94d14SJuan Quintela { 69a7a94d14SJuan Quintela MigrationState *s; 70a7a94d14SJuan Quintela 71a7a94d14SJuan Quintela s = migrate_get_current(); 72a7a94d14SJuan Quintela 73a7a94d14SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_COMPRESS]; 74a7a94d14SJuan Quintela } 75a7a94d14SJuan Quintela 761f0776f1SJuan Quintela bool migrate_dirty_bitmaps(void) 771f0776f1SJuan Quintela { 781f0776f1SJuan Quintela MigrationState *s; 791f0776f1SJuan Quintela 801f0776f1SJuan Quintela s = migrate_get_current(); 811f0776f1SJuan Quintela 821f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_DIRTY_BITMAPS]; 831f0776f1SJuan Quintela } 841f0776f1SJuan Quintela 85b890902cSJuan Quintela bool migrate_events(void) 86b890902cSJuan Quintela { 87b890902cSJuan Quintela MigrationState *s; 88b890902cSJuan Quintela 89b890902cSJuan Quintela s = migrate_get_current(); 90b890902cSJuan Quintela 91b890902cSJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_EVENTS]; 92b890902cSJuan Quintela } 93b890902cSJuan Quintela 941f0776f1SJuan Quintela bool migrate_ignore_shared(void) 951f0776f1SJuan Quintela { 961f0776f1SJuan Quintela MigrationState *s; 971f0776f1SJuan Quintela 981f0776f1SJuan Quintela s = migrate_get_current(); 991f0776f1SJuan Quintela 1001f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_X_IGNORE_SHARED]; 1011f0776f1SJuan Quintela } 1021f0776f1SJuan Quintela 1031f0776f1SJuan Quintela bool migrate_late_block_activate(void) 1041f0776f1SJuan Quintela { 1051f0776f1SJuan Quintela MigrationState *s; 1061f0776f1SJuan Quintela 1071f0776f1SJuan Quintela s = migrate_get_current(); 1081f0776f1SJuan Quintela 1091f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_LATE_BLOCK_ACTIVATE]; 1101f0776f1SJuan Quintela } 1111f0776f1SJuan Quintela 11251b07548SJuan Quintela bool migrate_multifd(void) 11351b07548SJuan Quintela { 11451b07548SJuan Quintela MigrationState *s; 11551b07548SJuan Quintela 11651b07548SJuan Quintela s = migrate_get_current(); 11751b07548SJuan Quintela 11851b07548SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_MULTIFD]; 11951b07548SJuan Quintela } 12051b07548SJuan Quintela 1211f0776f1SJuan Quintela bool migrate_pause_before_switchover(void) 1221f0776f1SJuan Quintela { 1231f0776f1SJuan Quintela MigrationState *s; 1241f0776f1SJuan Quintela 1251f0776f1SJuan Quintela s = migrate_get_current(); 1261f0776f1SJuan Quintela 1271f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_PAUSE_BEFORE_SWITCHOVER]; 1281f0776f1SJuan Quintela } 1291f0776f1SJuan Quintela 1301f0776f1SJuan Quintela bool migrate_postcopy_blocktime(void) 1311f0776f1SJuan Quintela { 1321f0776f1SJuan Quintela MigrationState *s; 1331f0776f1SJuan Quintela 1341f0776f1SJuan Quintela s = migrate_get_current(); 1351f0776f1SJuan Quintela 1361f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_POSTCOPY_BLOCKTIME]; 1371f0776f1SJuan Quintela } 1381f0776f1SJuan Quintela 1391f0776f1SJuan Quintela bool migrate_postcopy_preempt(void) 1401f0776f1SJuan Quintela { 1411f0776f1SJuan Quintela MigrationState *s; 1421f0776f1SJuan Quintela 1431f0776f1SJuan Quintela s = migrate_get_current(); 1441f0776f1SJuan Quintela 1451f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_POSTCOPY_PREEMPT]; 1461f0776f1SJuan Quintela } 1471f0776f1SJuan Quintela 1481f0776f1SJuan Quintela bool migrate_postcopy_ram(void) 1491f0776f1SJuan Quintela { 1501f0776f1SJuan Quintela MigrationState *s; 1511f0776f1SJuan Quintela 1521f0776f1SJuan Quintela s = migrate_get_current(); 1531f0776f1SJuan Quintela 1541f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_POSTCOPY_RAM]; 1551f0776f1SJuan Quintela } 1561f0776f1SJuan Quintela 15717cba690SJuan Quintela bool migrate_rdma_pin_all(void) 15817cba690SJuan Quintela { 15917cba690SJuan Quintela MigrationState *s = migrate_get_current(); 16017cba690SJuan Quintela 16117cba690SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_RDMA_PIN_ALL]; 16217cba690SJuan Quintela } 16317cba690SJuan Quintela 1641f0776f1SJuan Quintela bool migrate_release_ram(void) 1651f0776f1SJuan Quintela { 1661f0776f1SJuan Quintela MigrationState *s; 1671f0776f1SJuan Quintela 1681f0776f1SJuan Quintela s = migrate_get_current(); 1691f0776f1SJuan Quintela 1701f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_RELEASE_RAM]; 1711f0776f1SJuan Quintela } 1721f0776f1SJuan Quintela 17338ad1110SJuan Quintela bool migrate_return_path(void) 17438ad1110SJuan Quintela { 17538ad1110SJuan Quintela MigrationState *s; 17638ad1110SJuan Quintela 17738ad1110SJuan Quintela s = migrate_get_current(); 17838ad1110SJuan Quintela 17938ad1110SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_RETURN_PATH]; 18038ad1110SJuan Quintela } 18138ad1110SJuan Quintela 1821f0776f1SJuan Quintela bool migrate_validate_uuid(void) 1831f0776f1SJuan Quintela { 1841f0776f1SJuan Quintela MigrationState *s; 1851f0776f1SJuan Quintela 1861f0776f1SJuan Quintela s = migrate_get_current(); 1871f0776f1SJuan Quintela 1881f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_VALIDATE_UUID]; 1891f0776f1SJuan Quintela } 1901f0776f1SJuan Quintela 19187dca0c9SJuan Quintela bool migrate_xbzrle(void) 19287dca0c9SJuan Quintela { 19387dca0c9SJuan Quintela MigrationState *s; 19487dca0c9SJuan Quintela 19587dca0c9SJuan Quintela s = migrate_get_current(); 19687dca0c9SJuan Quintela 19787dca0c9SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_XBZRLE]; 19887dca0c9SJuan Quintela } 19987dca0c9SJuan Quintela 2001f0776f1SJuan Quintela bool migrate_zero_blocks(void) 2011f0776f1SJuan Quintela { 2021f0776f1SJuan Quintela MigrationState *s; 2031f0776f1SJuan Quintela 2041f0776f1SJuan Quintela s = migrate_get_current(); 2051f0776f1SJuan Quintela 2061f0776f1SJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_ZERO_BLOCKS]; 2071f0776f1SJuan Quintela } 208b4bc342cSJuan Quintela 209b4bc342cSJuan Quintela bool migrate_zero_copy_send(void) 210b4bc342cSJuan Quintela { 211b4bc342cSJuan Quintela MigrationState *s; 212b4bc342cSJuan Quintela 213b4bc342cSJuan Quintela s = migrate_get_current(); 214b4bc342cSJuan Quintela 215b4bc342cSJuan Quintela return s->capabilities[MIGRATION_CAPABILITY_ZERO_COPY_SEND]; 216b4bc342cSJuan Quintela } 217f774fde5SJuan Quintela 218f774fde5SJuan Quintela /* pseudo capabilities */ 219f774fde5SJuan Quintela 220f774fde5SJuan Quintela bool migrate_postcopy(void) 221f774fde5SJuan Quintela { 222f774fde5SJuan Quintela return migrate_postcopy_ram() || migrate_dirty_bitmaps(); 223f774fde5SJuan Quintela } 224f774fde5SJuan Quintela 22510d4703bSJuan Quintela bool migrate_tls(void) 22610d4703bSJuan Quintela { 22710d4703bSJuan Quintela MigrationState *s; 22810d4703bSJuan Quintela 22910d4703bSJuan Quintela s = migrate_get_current(); 23010d4703bSJuan Quintela 23110d4703bSJuan Quintela return s->parameters.tls_creds && *s->parameters.tls_creds; 23210d4703bSJuan Quintela } 23310d4703bSJuan Quintela 23477608706SJuan Quintela typedef enum WriteTrackingSupport { 23577608706SJuan Quintela WT_SUPPORT_UNKNOWN = 0, 23677608706SJuan Quintela WT_SUPPORT_ABSENT, 23777608706SJuan Quintela WT_SUPPORT_AVAILABLE, 23877608706SJuan Quintela WT_SUPPORT_COMPATIBLE 23977608706SJuan Quintela } WriteTrackingSupport; 24077608706SJuan Quintela 24177608706SJuan Quintela static 24277608706SJuan Quintela WriteTrackingSupport migrate_query_write_tracking(void) 24377608706SJuan Quintela { 24477608706SJuan Quintela /* Check if kernel supports required UFFD features */ 24577608706SJuan Quintela if (!ram_write_tracking_available()) { 24677608706SJuan Quintela return WT_SUPPORT_ABSENT; 24777608706SJuan Quintela } 24877608706SJuan Quintela /* 24977608706SJuan Quintela * Check if current memory configuration is 25077608706SJuan Quintela * compatible with required UFFD features. 25177608706SJuan Quintela */ 25277608706SJuan Quintela if (!ram_write_tracking_compatible()) { 25377608706SJuan Quintela return WT_SUPPORT_AVAILABLE; 25477608706SJuan Quintela } 25577608706SJuan Quintela 25677608706SJuan Quintela return WT_SUPPORT_COMPATIBLE; 25777608706SJuan Quintela } 25877608706SJuan Quintela 25977608706SJuan Quintela /* Migration capabilities set */ 26077608706SJuan Quintela struct MigrateCapsSet { 26177608706SJuan Quintela int size; /* Capability set size */ 26277608706SJuan Quintela MigrationCapability caps[]; /* Variadic array of capabilities */ 26377608706SJuan Quintela }; 26477608706SJuan Quintela typedef struct MigrateCapsSet MigrateCapsSet; 26577608706SJuan Quintela 26677608706SJuan Quintela /* Define and initialize MigrateCapsSet */ 26777608706SJuan Quintela #define INITIALIZE_MIGRATE_CAPS_SET(_name, ...) \ 26877608706SJuan Quintela MigrateCapsSet _name = { \ 26977608706SJuan Quintela .size = sizeof((int []) { __VA_ARGS__ }) / sizeof(int), \ 27077608706SJuan Quintela .caps = { __VA_ARGS__ } \ 27177608706SJuan Quintela } 27277608706SJuan Quintela 27377608706SJuan Quintela /* Background-snapshot compatibility check list */ 27477608706SJuan Quintela static const 27577608706SJuan Quintela INITIALIZE_MIGRATE_CAPS_SET(check_caps_background_snapshot, 27677608706SJuan Quintela MIGRATION_CAPABILITY_POSTCOPY_RAM, 27777608706SJuan Quintela MIGRATION_CAPABILITY_DIRTY_BITMAPS, 27877608706SJuan Quintela MIGRATION_CAPABILITY_POSTCOPY_BLOCKTIME, 27977608706SJuan Quintela MIGRATION_CAPABILITY_LATE_BLOCK_ACTIVATE, 28077608706SJuan Quintela MIGRATION_CAPABILITY_RETURN_PATH, 28177608706SJuan Quintela MIGRATION_CAPABILITY_MULTIFD, 28277608706SJuan Quintela MIGRATION_CAPABILITY_PAUSE_BEFORE_SWITCHOVER, 28377608706SJuan Quintela MIGRATION_CAPABILITY_AUTO_CONVERGE, 28477608706SJuan Quintela MIGRATION_CAPABILITY_RELEASE_RAM, 28577608706SJuan Quintela MIGRATION_CAPABILITY_RDMA_PIN_ALL, 28677608706SJuan Quintela MIGRATION_CAPABILITY_COMPRESS, 28777608706SJuan Quintela MIGRATION_CAPABILITY_XBZRLE, 28877608706SJuan Quintela MIGRATION_CAPABILITY_X_COLO, 28977608706SJuan Quintela MIGRATION_CAPABILITY_VALIDATE_UUID, 29077608706SJuan Quintela MIGRATION_CAPABILITY_ZERO_COPY_SEND); 29177608706SJuan Quintela 29277608706SJuan Quintela /** 29377608706SJuan Quintela * @migration_caps_check - check capability compatibility 29477608706SJuan Quintela * 29577608706SJuan Quintela * @old_caps: old capability list 29677608706SJuan Quintela * @new_caps: new capability list 29777608706SJuan Quintela * @errp: set *errp if the check failed, with reason 29877608706SJuan Quintela * 29977608706SJuan Quintela * Returns true if check passed, otherwise false. 30077608706SJuan Quintela */ 30177608706SJuan Quintela bool migrate_caps_check(bool *old_caps, bool *new_caps, Error **errp) 30277608706SJuan Quintela { 30377608706SJuan Quintela MigrationIncomingState *mis = migration_incoming_get_current(); 30477608706SJuan Quintela 30577608706SJuan Quintela #ifndef CONFIG_LIVE_BLOCK_MIGRATION 30677608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_BLOCK]) { 30777608706SJuan Quintela error_setg(errp, "QEMU compiled without old-style (blk/-b, inc/-i) " 30877608706SJuan Quintela "block migration"); 30977608706SJuan Quintela error_append_hint(errp, "Use drive_mirror+NBD instead.\n"); 31077608706SJuan Quintela return false; 31177608706SJuan Quintela } 31277608706SJuan Quintela #endif 31377608706SJuan Quintela 31477608706SJuan Quintela #ifndef CONFIG_REPLICATION 31577608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_X_COLO]) { 31677608706SJuan Quintela error_setg(errp, "QEMU compiled without replication module" 31777608706SJuan Quintela " can't enable COLO"); 31877608706SJuan Quintela error_append_hint(errp, "Please enable replication before COLO.\n"); 31977608706SJuan Quintela return false; 32077608706SJuan Quintela } 32177608706SJuan Quintela #endif 32277608706SJuan Quintela 32377608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_POSTCOPY_RAM]) { 32477608706SJuan Quintela /* This check is reasonably expensive, so only when it's being 32577608706SJuan Quintela * set the first time, also it's only the destination that needs 32677608706SJuan Quintela * special support. 32777608706SJuan Quintela */ 32877608706SJuan Quintela if (!old_caps[MIGRATION_CAPABILITY_POSTCOPY_RAM] && 32977608706SJuan Quintela runstate_check(RUN_STATE_INMIGRATE) && 33077608706SJuan Quintela !postcopy_ram_supported_by_host(mis)) { 33177608706SJuan Quintela /* postcopy_ram_supported_by_host will have emitted a more 33277608706SJuan Quintela * detailed message 33377608706SJuan Quintela */ 33477608706SJuan Quintela error_setg(errp, "Postcopy is not supported"); 33577608706SJuan Quintela return false; 33677608706SJuan Quintela } 33777608706SJuan Quintela 33877608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_X_IGNORE_SHARED]) { 33977608706SJuan Quintela error_setg(errp, "Postcopy is not compatible with ignore-shared"); 34077608706SJuan Quintela return false; 34177608706SJuan Quintela } 342b405dfffSLeonardo Bras 343b405dfffSLeonardo Bras if (new_caps[MIGRATION_CAPABILITY_MULTIFD]) { 344b405dfffSLeonardo Bras error_setg(errp, "Postcopy is not yet compatible with multifd"); 345b405dfffSLeonardo Bras return false; 346b405dfffSLeonardo Bras } 34777608706SJuan Quintela } 34877608706SJuan Quintela 34977608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_BACKGROUND_SNAPSHOT]) { 35077608706SJuan Quintela WriteTrackingSupport wt_support; 35177608706SJuan Quintela int idx; 35277608706SJuan Quintela /* 35377608706SJuan Quintela * Check if 'background-snapshot' capability is supported by 35477608706SJuan Quintela * host kernel and compatible with guest memory configuration. 35577608706SJuan Quintela */ 35677608706SJuan Quintela wt_support = migrate_query_write_tracking(); 35777608706SJuan Quintela if (wt_support < WT_SUPPORT_AVAILABLE) { 35877608706SJuan Quintela error_setg(errp, "Background-snapshot is not supported by host kernel"); 35977608706SJuan Quintela return false; 36077608706SJuan Quintela } 36177608706SJuan Quintela if (wt_support < WT_SUPPORT_COMPATIBLE) { 36277608706SJuan Quintela error_setg(errp, "Background-snapshot is not compatible " 36377608706SJuan Quintela "with guest memory configuration"); 36477608706SJuan Quintela return false; 36577608706SJuan Quintela } 36677608706SJuan Quintela 36777608706SJuan Quintela /* 36877608706SJuan Quintela * Check if there are any migration capabilities 36977608706SJuan Quintela * incompatible with 'background-snapshot'. 37077608706SJuan Quintela */ 37177608706SJuan Quintela for (idx = 0; idx < check_caps_background_snapshot.size; idx++) { 37277608706SJuan Quintela int incomp_cap = check_caps_background_snapshot.caps[idx]; 37377608706SJuan Quintela if (new_caps[incomp_cap]) { 37477608706SJuan Quintela error_setg(errp, 37577608706SJuan Quintela "Background-snapshot is not compatible with %s", 37677608706SJuan Quintela MigrationCapability_str(incomp_cap)); 37777608706SJuan Quintela return false; 37877608706SJuan Quintela } 37977608706SJuan Quintela } 38077608706SJuan Quintela } 38177608706SJuan Quintela 38277608706SJuan Quintela #ifdef CONFIG_LINUX 38377608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_ZERO_COPY_SEND] && 38477608706SJuan Quintela (!new_caps[MIGRATION_CAPABILITY_MULTIFD] || 38577608706SJuan Quintela new_caps[MIGRATION_CAPABILITY_COMPRESS] || 38677608706SJuan Quintela new_caps[MIGRATION_CAPABILITY_XBZRLE] || 38777608706SJuan Quintela migrate_multifd_compression() || 38810d4703bSJuan Quintela migrate_tls())) { 38977608706SJuan Quintela error_setg(errp, 39077608706SJuan Quintela "Zero copy only available for non-compressed non-TLS multifd migration"); 39177608706SJuan Quintela return false; 39277608706SJuan Quintela } 39377608706SJuan Quintela #else 39477608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_ZERO_COPY_SEND]) { 39577608706SJuan Quintela error_setg(errp, 39677608706SJuan Quintela "Zero copy currently only available on Linux"); 39777608706SJuan Quintela return false; 39877608706SJuan Quintela } 39977608706SJuan Quintela #endif 40077608706SJuan Quintela 40177608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_POSTCOPY_PREEMPT]) { 40277608706SJuan Quintela if (!new_caps[MIGRATION_CAPABILITY_POSTCOPY_RAM]) { 40377608706SJuan Quintela error_setg(errp, "Postcopy preempt requires postcopy-ram"); 40477608706SJuan Quintela return false; 40577608706SJuan Quintela } 40677608706SJuan Quintela 40777608706SJuan Quintela /* 40877608706SJuan Quintela * Preempt mode requires urgent pages to be sent in separate 40977608706SJuan Quintela * channel, OTOH compression logic will disorder all pages into 41077608706SJuan Quintela * different compression channels, which is not compatible with the 41177608706SJuan Quintela * preempt assumptions on channel assignments. 41277608706SJuan Quintela */ 41377608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_COMPRESS]) { 41477608706SJuan Quintela error_setg(errp, "Postcopy preempt not compatible with compress"); 41577608706SJuan Quintela return false; 41677608706SJuan Quintela } 41777608706SJuan Quintela } 41877608706SJuan Quintela 41977608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_MULTIFD]) { 42077608706SJuan Quintela if (new_caps[MIGRATION_CAPABILITY_COMPRESS]) { 42177608706SJuan Quintela error_setg(errp, "Multifd is not compatible with compress"); 42277608706SJuan Quintela return false; 42377608706SJuan Quintela } 42477608706SJuan Quintela } 42577608706SJuan Quintela 42677608706SJuan Quintela return true; 42777608706SJuan Quintela } 4284d0c6b69SJuan Quintela 429f80196b7SJuan Quintela bool migrate_cap_set(int cap, bool value, Error **errp) 430f80196b7SJuan Quintela { 431f80196b7SJuan Quintela MigrationState *s = migrate_get_current(); 432f80196b7SJuan Quintela bool new_caps[MIGRATION_CAPABILITY__MAX]; 433f80196b7SJuan Quintela 434f80196b7SJuan Quintela if (migration_is_running(s->state)) { 435f80196b7SJuan Quintela error_setg(errp, QERR_MIGRATION_ACTIVE); 436f80196b7SJuan Quintela return false; 437f80196b7SJuan Quintela } 438f80196b7SJuan Quintela 439f80196b7SJuan Quintela memcpy(new_caps, s->capabilities, sizeof(new_caps)); 440f80196b7SJuan Quintela new_caps[cap] = value; 441f80196b7SJuan Quintela 442f80196b7SJuan Quintela if (!migrate_caps_check(s->capabilities, new_caps, errp)) { 443f80196b7SJuan Quintela return false; 444f80196b7SJuan Quintela } 445f80196b7SJuan Quintela s->capabilities[cap] = value; 446f80196b7SJuan Quintela return true; 447f80196b7SJuan Quintela } 448f80196b7SJuan Quintela 4494d0c6b69SJuan Quintela MigrationCapabilityStatusList *qmp_query_migrate_capabilities(Error **errp) 4504d0c6b69SJuan Quintela { 4514d0c6b69SJuan Quintela MigrationCapabilityStatusList *head = NULL, **tail = &head; 4524d0c6b69SJuan Quintela MigrationCapabilityStatus *caps; 4534d0c6b69SJuan Quintela MigrationState *s = migrate_get_current(); 4544d0c6b69SJuan Quintela int i; 4554d0c6b69SJuan Quintela 4564d0c6b69SJuan Quintela for (i = 0; i < MIGRATION_CAPABILITY__MAX; i++) { 4574d0c6b69SJuan Quintela #ifndef CONFIG_LIVE_BLOCK_MIGRATION 4584d0c6b69SJuan Quintela if (i == MIGRATION_CAPABILITY_BLOCK) { 4594d0c6b69SJuan Quintela continue; 4604d0c6b69SJuan Quintela } 4614d0c6b69SJuan Quintela #endif 4624d0c6b69SJuan Quintela caps = g_malloc0(sizeof(*caps)); 4634d0c6b69SJuan Quintela caps->capability = i; 4644d0c6b69SJuan Quintela caps->state = s->capabilities[i]; 4654d0c6b69SJuan Quintela QAPI_LIST_APPEND(tail, caps); 4664d0c6b69SJuan Quintela } 4674d0c6b69SJuan Quintela 4684d0c6b69SJuan Quintela return head; 4694d0c6b69SJuan Quintela } 47045c1de13SJuan Quintela 47145c1de13SJuan Quintela void qmp_migrate_set_capabilities(MigrationCapabilityStatusList *params, 47245c1de13SJuan Quintela Error **errp) 47345c1de13SJuan Quintela { 47445c1de13SJuan Quintela MigrationState *s = migrate_get_current(); 47545c1de13SJuan Quintela MigrationCapabilityStatusList *cap; 47645c1de13SJuan Quintela bool new_caps[MIGRATION_CAPABILITY__MAX]; 47745c1de13SJuan Quintela 47845c1de13SJuan Quintela if (migration_is_running(s->state)) { 47945c1de13SJuan Quintela error_setg(errp, QERR_MIGRATION_ACTIVE); 48045c1de13SJuan Quintela return; 48145c1de13SJuan Quintela } 48245c1de13SJuan Quintela 48345c1de13SJuan Quintela memcpy(new_caps, s->capabilities, sizeof(new_caps)); 48445c1de13SJuan Quintela for (cap = params; cap; cap = cap->next) { 48545c1de13SJuan Quintela new_caps[cap->value->capability] = cap->value->state; 48645c1de13SJuan Quintela } 48745c1de13SJuan Quintela 48845c1de13SJuan Quintela if (!migrate_caps_check(s->capabilities, new_caps, errp)) { 48945c1de13SJuan Quintela return; 49045c1de13SJuan Quintela } 49145c1de13SJuan Quintela 49245c1de13SJuan Quintela for (cap = params; cap; cap = cap->next) { 49345c1de13SJuan Quintela s->capabilities[cap->value->capability] = cap->value->state; 49445c1de13SJuan Quintela } 49545c1de13SJuan Quintela } 4961dfc4b9eSJuan Quintela 4971dfc4b9eSJuan Quintela /* parameters */ 4981dfc4b9eSJuan Quintela 4996f8be708SJuan Quintela bool migrate_block_incremental(void) 5006f8be708SJuan Quintela { 5016f8be708SJuan Quintela MigrationState *s; 5026f8be708SJuan Quintela 5036f8be708SJuan Quintela s = migrate_get_current(); 5046f8be708SJuan Quintela 5056f8be708SJuan Quintela return s->parameters.block_incremental; 5066f8be708SJuan Quintela } 5076f8be708SJuan Quintela 508f94a858fSJuan Quintela uint32_t migrate_checkpoint_delay(void) 509f94a858fSJuan Quintela { 510f94a858fSJuan Quintela MigrationState *s; 511f94a858fSJuan Quintela 512f94a858fSJuan Quintela s = migrate_get_current(); 513f94a858fSJuan Quintela 514f94a858fSJuan Quintela return s->parameters.x_checkpoint_delay; 515f94a858fSJuan Quintela } 516f94a858fSJuan Quintela 5171dfc4b9eSJuan Quintela int migrate_compress_level(void) 5181dfc4b9eSJuan Quintela { 5191dfc4b9eSJuan Quintela MigrationState *s; 5201dfc4b9eSJuan Quintela 5211dfc4b9eSJuan Quintela s = migrate_get_current(); 5221dfc4b9eSJuan Quintela 5231dfc4b9eSJuan Quintela return s->parameters.compress_level; 5241dfc4b9eSJuan Quintela } 5251dfc4b9eSJuan Quintela 5261dfc4b9eSJuan Quintela int migrate_compress_threads(void) 5271dfc4b9eSJuan Quintela { 5281dfc4b9eSJuan Quintela MigrationState *s; 5291dfc4b9eSJuan Quintela 5301dfc4b9eSJuan Quintela s = migrate_get_current(); 5311dfc4b9eSJuan Quintela 5321dfc4b9eSJuan Quintela return s->parameters.compress_threads; 5331dfc4b9eSJuan Quintela } 5341dfc4b9eSJuan Quintela 5351dfc4b9eSJuan Quintela int migrate_compress_wait_thread(void) 5361dfc4b9eSJuan Quintela { 5371dfc4b9eSJuan Quintela MigrationState *s; 5381dfc4b9eSJuan Quintela 5391dfc4b9eSJuan Quintela s = migrate_get_current(); 5401dfc4b9eSJuan Quintela 5411dfc4b9eSJuan Quintela return s->parameters.compress_wait_thread; 5421dfc4b9eSJuan Quintela } 5431dfc4b9eSJuan Quintela 5449605c2acSJuan Quintela uint8_t migrate_cpu_throttle_increment(void) 5459605c2acSJuan Quintela { 5469605c2acSJuan Quintela MigrationState *s; 5479605c2acSJuan Quintela 5489605c2acSJuan Quintela s = migrate_get_current(); 5499605c2acSJuan Quintela 5509605c2acSJuan Quintela return s->parameters.cpu_throttle_increment; 5519605c2acSJuan Quintela } 5529605c2acSJuan Quintela 5532a8ec380SJuan Quintela uint8_t migrate_cpu_throttle_initial(void) 5542a8ec380SJuan Quintela { 5552a8ec380SJuan Quintela MigrationState *s; 5562a8ec380SJuan Quintela 5572a8ec380SJuan Quintela s = migrate_get_current(); 5582a8ec380SJuan Quintela 5592a8ec380SJuan Quintela return s->parameters.cpu_throttle_initial; 5602a8ec380SJuan Quintela } 5612a8ec380SJuan Quintela 562873f674cSJuan Quintela bool migrate_cpu_throttle_tailslow(void) 563873f674cSJuan Quintela { 564873f674cSJuan Quintela MigrationState *s; 565873f674cSJuan Quintela 566873f674cSJuan Quintela s = migrate_get_current(); 567873f674cSJuan Quintela 568873f674cSJuan Quintela return s->parameters.cpu_throttle_tailslow; 569873f674cSJuan Quintela } 570873f674cSJuan Quintela 5711dfc4b9eSJuan Quintela int migrate_decompress_threads(void) 5721dfc4b9eSJuan Quintela { 5731dfc4b9eSJuan Quintela MigrationState *s; 5741dfc4b9eSJuan Quintela 5751dfc4b9eSJuan Quintela s = migrate_get_current(); 5761dfc4b9eSJuan Quintela 5771dfc4b9eSJuan Quintela return s->parameters.decompress_threads; 5781dfc4b9eSJuan Quintela } 5791dfc4b9eSJuan Quintela 58024155bd0SJuan Quintela uint8_t migrate_max_cpu_throttle(void) 58124155bd0SJuan Quintela { 58224155bd0SJuan Quintela MigrationState *s; 58324155bd0SJuan Quintela 58424155bd0SJuan Quintela s = migrate_get_current(); 58524155bd0SJuan Quintela 58624155bd0SJuan Quintela return s->parameters.max_cpu_throttle; 58724155bd0SJuan Quintela } 58824155bd0SJuan Quintela 5899c894df3SJuan Quintela uint64_t migrate_max_bandwidth(void) 5909c894df3SJuan Quintela { 5919c894df3SJuan Quintela MigrationState *s; 5929c894df3SJuan Quintela 5939c894df3SJuan Quintela s = migrate_get_current(); 5949c894df3SJuan Quintela 5959c894df3SJuan Quintela return s->parameters.max_bandwidth; 5969c894df3SJuan Quintela } 5979c894df3SJuan Quintela 5981dfc4b9eSJuan Quintela int64_t migrate_max_postcopy_bandwidth(void) 5991dfc4b9eSJuan Quintela { 6001dfc4b9eSJuan Quintela MigrationState *s; 6011dfc4b9eSJuan Quintela 6021dfc4b9eSJuan Quintela s = migrate_get_current(); 6031dfc4b9eSJuan Quintela 6041dfc4b9eSJuan Quintela return s->parameters.max_postcopy_bandwidth; 6051dfc4b9eSJuan Quintela } 6061dfc4b9eSJuan Quintela 6071dfc4b9eSJuan Quintela int migrate_multifd_channels(void) 6081dfc4b9eSJuan Quintela { 6091dfc4b9eSJuan Quintela MigrationState *s; 6101dfc4b9eSJuan Quintela 6111dfc4b9eSJuan Quintela s = migrate_get_current(); 6121dfc4b9eSJuan Quintela 6131dfc4b9eSJuan Quintela return s->parameters.multifd_channels; 6141dfc4b9eSJuan Quintela } 6151dfc4b9eSJuan Quintela 6161dfc4b9eSJuan Quintela MultiFDCompression migrate_multifd_compression(void) 6171dfc4b9eSJuan Quintela { 6181dfc4b9eSJuan Quintela MigrationState *s; 6191dfc4b9eSJuan Quintela 6201dfc4b9eSJuan Quintela s = migrate_get_current(); 6211dfc4b9eSJuan Quintela 6221dfc4b9eSJuan Quintela assert(s->parameters.multifd_compression < MULTIFD_COMPRESSION__MAX); 6231dfc4b9eSJuan Quintela return s->parameters.multifd_compression; 6241dfc4b9eSJuan Quintela } 6251dfc4b9eSJuan Quintela 6261dfc4b9eSJuan Quintela int migrate_multifd_zlib_level(void) 6271dfc4b9eSJuan Quintela { 6281dfc4b9eSJuan Quintela MigrationState *s; 6291dfc4b9eSJuan Quintela 6301dfc4b9eSJuan Quintela s = migrate_get_current(); 6311dfc4b9eSJuan Quintela 6321dfc4b9eSJuan Quintela return s->parameters.multifd_zlib_level; 6331dfc4b9eSJuan Quintela } 6341dfc4b9eSJuan Quintela 6351dfc4b9eSJuan Quintela int migrate_multifd_zstd_level(void) 6361dfc4b9eSJuan Quintela { 6371dfc4b9eSJuan Quintela MigrationState *s; 6381dfc4b9eSJuan Quintela 6391dfc4b9eSJuan Quintela s = migrate_get_current(); 6401dfc4b9eSJuan Quintela 6411dfc4b9eSJuan Quintela return s->parameters.multifd_zstd_level; 6421dfc4b9eSJuan Quintela } 6431dfc4b9eSJuan Quintela 6446499efdbSJuan Quintela uint8_t migrate_throttle_trigger_threshold(void) 6456499efdbSJuan Quintela { 6466499efdbSJuan Quintela MigrationState *s; 6476499efdbSJuan Quintela 6486499efdbSJuan Quintela s = migrate_get_current(); 6496499efdbSJuan Quintela 6506499efdbSJuan Quintela return s->parameters.throttle_trigger_threshold; 6516499efdbSJuan Quintela } 6526499efdbSJuan Quintela 6531dfc4b9eSJuan Quintela uint64_t migrate_xbzrle_cache_size(void) 6541dfc4b9eSJuan Quintela { 6551dfc4b9eSJuan Quintela MigrationState *s; 6561dfc4b9eSJuan Quintela 6571dfc4b9eSJuan Quintela s = migrate_get_current(); 6581dfc4b9eSJuan Quintela 6591dfc4b9eSJuan Quintela return s->parameters.xbzrle_cache_size; 6601dfc4b9eSJuan Quintela } 6612682c4eeSJuan Quintela 6622682c4eeSJuan Quintela /* parameters helpers */ 6632682c4eeSJuan Quintela 6642682c4eeSJuan Quintela AnnounceParameters *migrate_announce_params(void) 6652682c4eeSJuan Quintela { 6662682c4eeSJuan Quintela static AnnounceParameters ap; 6672682c4eeSJuan Quintela 6682682c4eeSJuan Quintela MigrationState *s = migrate_get_current(); 6692682c4eeSJuan Quintela 6702682c4eeSJuan Quintela ap.initial = s->parameters.announce_initial; 6712682c4eeSJuan Quintela ap.max = s->parameters.announce_max; 6722682c4eeSJuan Quintela ap.rounds = s->parameters.announce_rounds; 6732682c4eeSJuan Quintela ap.step = s->parameters.announce_step; 6742682c4eeSJuan Quintela 6752682c4eeSJuan Quintela return ≈ 6762682c4eeSJuan Quintela } 6779c894df3SJuan Quintela 6789c894df3SJuan Quintela MigrationParameters *qmp_query_migrate_parameters(Error **errp) 6799c894df3SJuan Quintela { 6809c894df3SJuan Quintela MigrationParameters *params; 6819c894df3SJuan Quintela MigrationState *s = migrate_get_current(); 6829c894df3SJuan Quintela 6839c894df3SJuan Quintela /* TODO use QAPI_CLONE() instead of duplicating it inline */ 6849c894df3SJuan Quintela params = g_malloc0(sizeof(*params)); 6859c894df3SJuan Quintela params->has_compress_level = true; 6869c894df3SJuan Quintela params->compress_level = s->parameters.compress_level; 6879c894df3SJuan Quintela params->has_compress_threads = true; 6889c894df3SJuan Quintela params->compress_threads = s->parameters.compress_threads; 6899c894df3SJuan Quintela params->has_compress_wait_thread = true; 6909c894df3SJuan Quintela params->compress_wait_thread = s->parameters.compress_wait_thread; 6919c894df3SJuan Quintela params->has_decompress_threads = true; 6929c894df3SJuan Quintela params->decompress_threads = s->parameters.decompress_threads; 6939c894df3SJuan Quintela params->has_throttle_trigger_threshold = true; 6949c894df3SJuan Quintela params->throttle_trigger_threshold = s->parameters.throttle_trigger_threshold; 6959c894df3SJuan Quintela params->has_cpu_throttle_initial = true; 6969c894df3SJuan Quintela params->cpu_throttle_initial = s->parameters.cpu_throttle_initial; 6979c894df3SJuan Quintela params->has_cpu_throttle_increment = true; 6989c894df3SJuan Quintela params->cpu_throttle_increment = s->parameters.cpu_throttle_increment; 6999c894df3SJuan Quintela params->has_cpu_throttle_tailslow = true; 7009c894df3SJuan Quintela params->cpu_throttle_tailslow = s->parameters.cpu_throttle_tailslow; 7019c894df3SJuan Quintela params->tls_creds = g_strdup(s->parameters.tls_creds); 7029c894df3SJuan Quintela params->tls_hostname = g_strdup(s->parameters.tls_hostname); 7039c894df3SJuan Quintela params->tls_authz = g_strdup(s->parameters.tls_authz ? 7049c894df3SJuan Quintela s->parameters.tls_authz : ""); 7059c894df3SJuan Quintela params->has_max_bandwidth = true; 7069c894df3SJuan Quintela params->max_bandwidth = s->parameters.max_bandwidth; 7079c894df3SJuan Quintela params->has_downtime_limit = true; 7089c894df3SJuan Quintela params->downtime_limit = s->parameters.downtime_limit; 7099c894df3SJuan Quintela params->has_x_checkpoint_delay = true; 7109c894df3SJuan Quintela params->x_checkpoint_delay = s->parameters.x_checkpoint_delay; 7119c894df3SJuan Quintela params->has_block_incremental = true; 7129c894df3SJuan Quintela params->block_incremental = s->parameters.block_incremental; 7139c894df3SJuan Quintela params->has_multifd_channels = true; 7149c894df3SJuan Quintela params->multifd_channels = s->parameters.multifd_channels; 7159c894df3SJuan Quintela params->has_multifd_compression = true; 7169c894df3SJuan Quintela params->multifd_compression = s->parameters.multifd_compression; 7179c894df3SJuan Quintela params->has_multifd_zlib_level = true; 7189c894df3SJuan Quintela params->multifd_zlib_level = s->parameters.multifd_zlib_level; 7199c894df3SJuan Quintela params->has_multifd_zstd_level = true; 7209c894df3SJuan Quintela params->multifd_zstd_level = s->parameters.multifd_zstd_level; 7219c894df3SJuan Quintela params->has_xbzrle_cache_size = true; 7229c894df3SJuan Quintela params->xbzrle_cache_size = s->parameters.xbzrle_cache_size; 7239c894df3SJuan Quintela params->has_max_postcopy_bandwidth = true; 7249c894df3SJuan Quintela params->max_postcopy_bandwidth = s->parameters.max_postcopy_bandwidth; 7259c894df3SJuan Quintela params->has_max_cpu_throttle = true; 7269c894df3SJuan Quintela params->max_cpu_throttle = s->parameters.max_cpu_throttle; 7279c894df3SJuan Quintela params->has_announce_initial = true; 7289c894df3SJuan Quintela params->announce_initial = s->parameters.announce_initial; 7299c894df3SJuan Quintela params->has_announce_max = true; 7309c894df3SJuan Quintela params->announce_max = s->parameters.announce_max; 7319c894df3SJuan Quintela params->has_announce_rounds = true; 7329c894df3SJuan Quintela params->announce_rounds = s->parameters.announce_rounds; 7339c894df3SJuan Quintela params->has_announce_step = true; 7349c894df3SJuan Quintela params->announce_step = s->parameters.announce_step; 7359c894df3SJuan Quintela 7369c894df3SJuan Quintela if (s->parameters.has_block_bitmap_mapping) { 7379c894df3SJuan Quintela params->has_block_bitmap_mapping = true; 7389c894df3SJuan Quintela params->block_bitmap_mapping = 7399c894df3SJuan Quintela QAPI_CLONE(BitmapMigrationNodeAliasList, 7409c894df3SJuan Quintela s->parameters.block_bitmap_mapping); 7419c894df3SJuan Quintela } 7429c894df3SJuan Quintela 7439c894df3SJuan Quintela return params; 7449c894df3SJuan Quintela } 745*09d6c965SJuan Quintela 746*09d6c965SJuan Quintela /* 747*09d6c965SJuan Quintela * Check whether the parameters are valid. Error will be put into errp 748*09d6c965SJuan Quintela * (if provided). Return true if valid, otherwise false. 749*09d6c965SJuan Quintela */ 750*09d6c965SJuan Quintela bool migrate_params_check(MigrationParameters *params, Error **errp) 751*09d6c965SJuan Quintela { 752*09d6c965SJuan Quintela if (params->has_compress_level && 753*09d6c965SJuan Quintela (params->compress_level > 9)) { 754*09d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, "compress_level", 755*09d6c965SJuan Quintela "a value between 0 and 9"); 756*09d6c965SJuan Quintela return false; 757*09d6c965SJuan Quintela } 758*09d6c965SJuan Quintela 759*09d6c965SJuan Quintela if (params->has_compress_threads && (params->compress_threads < 1)) { 760*09d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 761*09d6c965SJuan Quintela "compress_threads", 762*09d6c965SJuan Quintela "a value between 1 and 255"); 763*09d6c965SJuan Quintela return false; 764*09d6c965SJuan Quintela } 765*09d6c965SJuan Quintela 766*09d6c965SJuan Quintela if (params->has_decompress_threads && (params->decompress_threads < 1)) { 767*09d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 768*09d6c965SJuan Quintela "decompress_threads", 769*09d6c965SJuan Quintela "a value between 1 and 255"); 770*09d6c965SJuan Quintela return false; 771*09d6c965SJuan Quintela } 772*09d6c965SJuan Quintela 773*09d6c965SJuan Quintela if (params->has_throttle_trigger_threshold && 774*09d6c965SJuan Quintela (params->throttle_trigger_threshold < 1 || 775*09d6c965SJuan Quintela params->throttle_trigger_threshold > 100)) { 776*09d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 777*09d6c965SJuan Quintela "throttle_trigger_threshold", 778*09d6c965SJuan Quintela "an integer in the range of 1 to 100"); 779*09d6c965SJuan Quintela return false; 780*09d6c965SJuan Quintela } 781*09d6c965SJuan Quintela 782*09d6c965SJuan Quintela if (params->has_cpu_throttle_initial && 783*09d6c965SJuan Quintela (params->cpu_throttle_initial < 1 || 784*09d6c965SJuan Quintela params->cpu_throttle_initial > 99)) { 785*09d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 786*09d6c965SJuan Quintela "cpu_throttle_initial", 787*09d6c965SJuan Quintela "an integer in the range of 1 to 99"); 788*09d6c965SJuan Quintela return false; 789*09d6c965SJuan Quintela } 790*09d6c965SJuan Quintela 791*09d6c965SJuan Quintela if (params->has_cpu_throttle_increment && 792*09d6c965SJuan Quintela (params->cpu_throttle_increment < 1 || 793*09d6c965SJuan Quintela params->cpu_throttle_increment > 99)) { 794*09d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 795*09d6c965SJuan Quintela "cpu_throttle_increment", 796*09d6c965SJuan Quintela "an integer in the range of 1 to 99"); 797*09d6c965SJuan Quintela return false; 798*09d6c965SJuan Quintela } 799*09d6c965SJuan Quintela 800*09d6c965SJuan Quintela if (params->has_max_bandwidth && (params->max_bandwidth > SIZE_MAX)) { 801*09d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 802*09d6c965SJuan Quintela "max_bandwidth", 803*09d6c965SJuan Quintela "an integer in the range of 0 to "stringify(SIZE_MAX) 804*09d6c965SJuan Quintela " bytes/second"); 805*09d6c965SJuan Quintela return false; 806*09d6c965SJuan Quintela } 807*09d6c965SJuan Quintela 808*09d6c965SJuan Quintela if (params->has_downtime_limit && 809*09d6c965SJuan Quintela (params->downtime_limit > MAX_MIGRATE_DOWNTIME)) { 810*09d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 811*09d6c965SJuan Quintela "downtime_limit", 812*09d6c965SJuan Quintela "an integer in the range of 0 to " 813*09d6c965SJuan Quintela stringify(MAX_MIGRATE_DOWNTIME)" ms"); 814*09d6c965SJuan Quintela return false; 815*09d6c965SJuan Quintela } 816*09d6c965SJuan Quintela 817*09d6c965SJuan Quintela /* x_checkpoint_delay is now always positive */ 818*09d6c965SJuan Quintela 819*09d6c965SJuan Quintela if (params->has_multifd_channels && (params->multifd_channels < 1)) { 820*09d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 821*09d6c965SJuan Quintela "multifd_channels", 822*09d6c965SJuan Quintela "a value between 1 and 255"); 823*09d6c965SJuan Quintela return false; 824*09d6c965SJuan Quintela } 825*09d6c965SJuan Quintela 826*09d6c965SJuan Quintela if (params->has_multifd_zlib_level && 827*09d6c965SJuan Quintela (params->multifd_zlib_level > 9)) { 828*09d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, "multifd_zlib_level", 829*09d6c965SJuan Quintela "a value between 0 and 9"); 830*09d6c965SJuan Quintela return false; 831*09d6c965SJuan Quintela } 832*09d6c965SJuan Quintela 833*09d6c965SJuan Quintela if (params->has_multifd_zstd_level && 834*09d6c965SJuan Quintela (params->multifd_zstd_level > 20)) { 835*09d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, "multifd_zstd_level", 836*09d6c965SJuan Quintela "a value between 0 and 20"); 837*09d6c965SJuan Quintela return false; 838*09d6c965SJuan Quintela } 839*09d6c965SJuan Quintela 840*09d6c965SJuan Quintela if (params->has_xbzrle_cache_size && 841*09d6c965SJuan Quintela (params->xbzrle_cache_size < qemu_target_page_size() || 842*09d6c965SJuan Quintela !is_power_of_2(params->xbzrle_cache_size))) { 843*09d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 844*09d6c965SJuan Quintela "xbzrle_cache_size", 845*09d6c965SJuan Quintela "a power of two no less than the target page size"); 846*09d6c965SJuan Quintela return false; 847*09d6c965SJuan Quintela } 848*09d6c965SJuan Quintela 849*09d6c965SJuan Quintela if (params->has_max_cpu_throttle && 850*09d6c965SJuan Quintela (params->max_cpu_throttle < params->cpu_throttle_initial || 851*09d6c965SJuan Quintela params->max_cpu_throttle > 99)) { 852*09d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 853*09d6c965SJuan Quintela "max_cpu_throttle", 854*09d6c965SJuan Quintela "an integer in the range of cpu_throttle_initial to 99"); 855*09d6c965SJuan Quintela return false; 856*09d6c965SJuan Quintela } 857*09d6c965SJuan Quintela 858*09d6c965SJuan Quintela if (params->has_announce_initial && 859*09d6c965SJuan Quintela params->announce_initial > 100000) { 860*09d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 861*09d6c965SJuan Quintela "announce_initial", 862*09d6c965SJuan Quintela "a value between 0 and 100000"); 863*09d6c965SJuan Quintela return false; 864*09d6c965SJuan Quintela } 865*09d6c965SJuan Quintela if (params->has_announce_max && 866*09d6c965SJuan Quintela params->announce_max > 100000) { 867*09d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 868*09d6c965SJuan Quintela "announce_max", 869*09d6c965SJuan Quintela "a value between 0 and 100000"); 870*09d6c965SJuan Quintela return false; 871*09d6c965SJuan Quintela } 872*09d6c965SJuan Quintela if (params->has_announce_rounds && 873*09d6c965SJuan Quintela params->announce_rounds > 1000) { 874*09d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 875*09d6c965SJuan Quintela "announce_rounds", 876*09d6c965SJuan Quintela "a value between 0 and 1000"); 877*09d6c965SJuan Quintela return false; 878*09d6c965SJuan Quintela } 879*09d6c965SJuan Quintela if (params->has_announce_step && 880*09d6c965SJuan Quintela (params->announce_step < 1 || 881*09d6c965SJuan Quintela params->announce_step > 10000)) { 882*09d6c965SJuan Quintela error_setg(errp, QERR_INVALID_PARAMETER_VALUE, 883*09d6c965SJuan Quintela "announce_step", 884*09d6c965SJuan Quintela "a value between 0 and 10000"); 885*09d6c965SJuan Quintela return false; 886*09d6c965SJuan Quintela } 887*09d6c965SJuan Quintela 888*09d6c965SJuan Quintela if (params->has_block_bitmap_mapping && 889*09d6c965SJuan Quintela !check_dirty_bitmap_mig_alias_map(params->block_bitmap_mapping, errp)) { 890*09d6c965SJuan Quintela error_prepend(errp, "Invalid mapping given for block-bitmap-mapping: "); 891*09d6c965SJuan Quintela return false; 892*09d6c965SJuan Quintela } 893*09d6c965SJuan Quintela 894*09d6c965SJuan Quintela #ifdef CONFIG_LINUX 895*09d6c965SJuan Quintela if (migrate_zero_copy_send() && 896*09d6c965SJuan Quintela ((params->has_multifd_compression && params->multifd_compression) || 897*09d6c965SJuan Quintela (params->tls_creds && *params->tls_creds))) { 898*09d6c965SJuan Quintela error_setg(errp, 899*09d6c965SJuan Quintela "Zero copy only available for non-compressed non-TLS multifd migration"); 900*09d6c965SJuan Quintela return false; 901*09d6c965SJuan Quintela } 902*09d6c965SJuan Quintela #endif 903*09d6c965SJuan Quintela 904*09d6c965SJuan Quintela return true; 905*09d6c965SJuan Quintela } 906*09d6c965SJuan Quintela 907*09d6c965SJuan Quintela static void migrate_params_test_apply(MigrateSetParameters *params, 908*09d6c965SJuan Quintela MigrationParameters *dest) 909*09d6c965SJuan Quintela { 910*09d6c965SJuan Quintela *dest = migrate_get_current()->parameters; 911*09d6c965SJuan Quintela 912*09d6c965SJuan Quintela /* TODO use QAPI_CLONE() instead of duplicating it inline */ 913*09d6c965SJuan Quintela 914*09d6c965SJuan Quintela if (params->has_compress_level) { 915*09d6c965SJuan Quintela dest->compress_level = params->compress_level; 916*09d6c965SJuan Quintela } 917*09d6c965SJuan Quintela 918*09d6c965SJuan Quintela if (params->has_compress_threads) { 919*09d6c965SJuan Quintela dest->compress_threads = params->compress_threads; 920*09d6c965SJuan Quintela } 921*09d6c965SJuan Quintela 922*09d6c965SJuan Quintela if (params->has_compress_wait_thread) { 923*09d6c965SJuan Quintela dest->compress_wait_thread = params->compress_wait_thread; 924*09d6c965SJuan Quintela } 925*09d6c965SJuan Quintela 926*09d6c965SJuan Quintela if (params->has_decompress_threads) { 927*09d6c965SJuan Quintela dest->decompress_threads = params->decompress_threads; 928*09d6c965SJuan Quintela } 929*09d6c965SJuan Quintela 930*09d6c965SJuan Quintela if (params->has_throttle_trigger_threshold) { 931*09d6c965SJuan Quintela dest->throttle_trigger_threshold = params->throttle_trigger_threshold; 932*09d6c965SJuan Quintela } 933*09d6c965SJuan Quintela 934*09d6c965SJuan Quintela if (params->has_cpu_throttle_initial) { 935*09d6c965SJuan Quintela dest->cpu_throttle_initial = params->cpu_throttle_initial; 936*09d6c965SJuan Quintela } 937*09d6c965SJuan Quintela 938*09d6c965SJuan Quintela if (params->has_cpu_throttle_increment) { 939*09d6c965SJuan Quintela dest->cpu_throttle_increment = params->cpu_throttle_increment; 940*09d6c965SJuan Quintela } 941*09d6c965SJuan Quintela 942*09d6c965SJuan Quintela if (params->has_cpu_throttle_tailslow) { 943*09d6c965SJuan Quintela dest->cpu_throttle_tailslow = params->cpu_throttle_tailslow; 944*09d6c965SJuan Quintela } 945*09d6c965SJuan Quintela 946*09d6c965SJuan Quintela if (params->tls_creds) { 947*09d6c965SJuan Quintela assert(params->tls_creds->type == QTYPE_QSTRING); 948*09d6c965SJuan Quintela dest->tls_creds = params->tls_creds->u.s; 949*09d6c965SJuan Quintela } 950*09d6c965SJuan Quintela 951*09d6c965SJuan Quintela if (params->tls_hostname) { 952*09d6c965SJuan Quintela assert(params->tls_hostname->type == QTYPE_QSTRING); 953*09d6c965SJuan Quintela dest->tls_hostname = params->tls_hostname->u.s; 954*09d6c965SJuan Quintela } 955*09d6c965SJuan Quintela 956*09d6c965SJuan Quintela if (params->has_max_bandwidth) { 957*09d6c965SJuan Quintela dest->max_bandwidth = params->max_bandwidth; 958*09d6c965SJuan Quintela } 959*09d6c965SJuan Quintela 960*09d6c965SJuan Quintela if (params->has_downtime_limit) { 961*09d6c965SJuan Quintela dest->downtime_limit = params->downtime_limit; 962*09d6c965SJuan Quintela } 963*09d6c965SJuan Quintela 964*09d6c965SJuan Quintela if (params->has_x_checkpoint_delay) { 965*09d6c965SJuan Quintela dest->x_checkpoint_delay = params->x_checkpoint_delay; 966*09d6c965SJuan Quintela } 967*09d6c965SJuan Quintela 968*09d6c965SJuan Quintela if (params->has_block_incremental) { 969*09d6c965SJuan Quintela dest->block_incremental = params->block_incremental; 970*09d6c965SJuan Quintela } 971*09d6c965SJuan Quintela if (params->has_multifd_channels) { 972*09d6c965SJuan Quintela dest->multifd_channels = params->multifd_channels; 973*09d6c965SJuan Quintela } 974*09d6c965SJuan Quintela if (params->has_multifd_compression) { 975*09d6c965SJuan Quintela dest->multifd_compression = params->multifd_compression; 976*09d6c965SJuan Quintela } 977*09d6c965SJuan Quintela if (params->has_xbzrle_cache_size) { 978*09d6c965SJuan Quintela dest->xbzrle_cache_size = params->xbzrle_cache_size; 979*09d6c965SJuan Quintela } 980*09d6c965SJuan Quintela if (params->has_max_postcopy_bandwidth) { 981*09d6c965SJuan Quintela dest->max_postcopy_bandwidth = params->max_postcopy_bandwidth; 982*09d6c965SJuan Quintela } 983*09d6c965SJuan Quintela if (params->has_max_cpu_throttle) { 984*09d6c965SJuan Quintela dest->max_cpu_throttle = params->max_cpu_throttle; 985*09d6c965SJuan Quintela } 986*09d6c965SJuan Quintela if (params->has_announce_initial) { 987*09d6c965SJuan Quintela dest->announce_initial = params->announce_initial; 988*09d6c965SJuan Quintela } 989*09d6c965SJuan Quintela if (params->has_announce_max) { 990*09d6c965SJuan Quintela dest->announce_max = params->announce_max; 991*09d6c965SJuan Quintela } 992*09d6c965SJuan Quintela if (params->has_announce_rounds) { 993*09d6c965SJuan Quintela dest->announce_rounds = params->announce_rounds; 994*09d6c965SJuan Quintela } 995*09d6c965SJuan Quintela if (params->has_announce_step) { 996*09d6c965SJuan Quintela dest->announce_step = params->announce_step; 997*09d6c965SJuan Quintela } 998*09d6c965SJuan Quintela 999*09d6c965SJuan Quintela if (params->has_block_bitmap_mapping) { 1000*09d6c965SJuan Quintela dest->has_block_bitmap_mapping = true; 1001*09d6c965SJuan Quintela dest->block_bitmap_mapping = params->block_bitmap_mapping; 1002*09d6c965SJuan Quintela } 1003*09d6c965SJuan Quintela } 1004*09d6c965SJuan Quintela 1005*09d6c965SJuan Quintela static void migrate_params_apply(MigrateSetParameters *params, Error **errp) 1006*09d6c965SJuan Quintela { 1007*09d6c965SJuan Quintela MigrationState *s = migrate_get_current(); 1008*09d6c965SJuan Quintela 1009*09d6c965SJuan Quintela /* TODO use QAPI_CLONE() instead of duplicating it inline */ 1010*09d6c965SJuan Quintela 1011*09d6c965SJuan Quintela if (params->has_compress_level) { 1012*09d6c965SJuan Quintela s->parameters.compress_level = params->compress_level; 1013*09d6c965SJuan Quintela } 1014*09d6c965SJuan Quintela 1015*09d6c965SJuan Quintela if (params->has_compress_threads) { 1016*09d6c965SJuan Quintela s->parameters.compress_threads = params->compress_threads; 1017*09d6c965SJuan Quintela } 1018*09d6c965SJuan Quintela 1019*09d6c965SJuan Quintela if (params->has_compress_wait_thread) { 1020*09d6c965SJuan Quintela s->parameters.compress_wait_thread = params->compress_wait_thread; 1021*09d6c965SJuan Quintela } 1022*09d6c965SJuan Quintela 1023*09d6c965SJuan Quintela if (params->has_decompress_threads) { 1024*09d6c965SJuan Quintela s->parameters.decompress_threads = params->decompress_threads; 1025*09d6c965SJuan Quintela } 1026*09d6c965SJuan Quintela 1027*09d6c965SJuan Quintela if (params->has_throttle_trigger_threshold) { 1028*09d6c965SJuan Quintela s->parameters.throttle_trigger_threshold = params->throttle_trigger_threshold; 1029*09d6c965SJuan Quintela } 1030*09d6c965SJuan Quintela 1031*09d6c965SJuan Quintela if (params->has_cpu_throttle_initial) { 1032*09d6c965SJuan Quintela s->parameters.cpu_throttle_initial = params->cpu_throttle_initial; 1033*09d6c965SJuan Quintela } 1034*09d6c965SJuan Quintela 1035*09d6c965SJuan Quintela if (params->has_cpu_throttle_increment) { 1036*09d6c965SJuan Quintela s->parameters.cpu_throttle_increment = params->cpu_throttle_increment; 1037*09d6c965SJuan Quintela } 1038*09d6c965SJuan Quintela 1039*09d6c965SJuan Quintela if (params->has_cpu_throttle_tailslow) { 1040*09d6c965SJuan Quintela s->parameters.cpu_throttle_tailslow = params->cpu_throttle_tailslow; 1041*09d6c965SJuan Quintela } 1042*09d6c965SJuan Quintela 1043*09d6c965SJuan Quintela if (params->tls_creds) { 1044*09d6c965SJuan Quintela g_free(s->parameters.tls_creds); 1045*09d6c965SJuan Quintela assert(params->tls_creds->type == QTYPE_QSTRING); 1046*09d6c965SJuan Quintela s->parameters.tls_creds = g_strdup(params->tls_creds->u.s); 1047*09d6c965SJuan Quintela } 1048*09d6c965SJuan Quintela 1049*09d6c965SJuan Quintela if (params->tls_hostname) { 1050*09d6c965SJuan Quintela g_free(s->parameters.tls_hostname); 1051*09d6c965SJuan Quintela assert(params->tls_hostname->type == QTYPE_QSTRING); 1052*09d6c965SJuan Quintela s->parameters.tls_hostname = g_strdup(params->tls_hostname->u.s); 1053*09d6c965SJuan Quintela } 1054*09d6c965SJuan Quintela 1055*09d6c965SJuan Quintela if (params->tls_authz) { 1056*09d6c965SJuan Quintela g_free(s->parameters.tls_authz); 1057*09d6c965SJuan Quintela assert(params->tls_authz->type == QTYPE_QSTRING); 1058*09d6c965SJuan Quintela s->parameters.tls_authz = g_strdup(params->tls_authz->u.s); 1059*09d6c965SJuan Quintela } 1060*09d6c965SJuan Quintela 1061*09d6c965SJuan Quintela if (params->has_max_bandwidth) { 1062*09d6c965SJuan Quintela s->parameters.max_bandwidth = params->max_bandwidth; 1063*09d6c965SJuan Quintela if (s->to_dst_file && !migration_in_postcopy()) { 1064*09d6c965SJuan Quintela qemu_file_set_rate_limit(s->to_dst_file, 1065*09d6c965SJuan Quintela s->parameters.max_bandwidth / XFER_LIMIT_RATIO); 1066*09d6c965SJuan Quintela } 1067*09d6c965SJuan Quintela } 1068*09d6c965SJuan Quintela 1069*09d6c965SJuan Quintela if (params->has_downtime_limit) { 1070*09d6c965SJuan Quintela s->parameters.downtime_limit = params->downtime_limit; 1071*09d6c965SJuan Quintela } 1072*09d6c965SJuan Quintela 1073*09d6c965SJuan Quintela if (params->has_x_checkpoint_delay) { 1074*09d6c965SJuan Quintela s->parameters.x_checkpoint_delay = params->x_checkpoint_delay; 1075*09d6c965SJuan Quintela if (migration_in_colo_state()) { 1076*09d6c965SJuan Quintela colo_checkpoint_notify(s); 1077*09d6c965SJuan Quintela } 1078*09d6c965SJuan Quintela } 1079*09d6c965SJuan Quintela 1080*09d6c965SJuan Quintela if (params->has_block_incremental) { 1081*09d6c965SJuan Quintela s->parameters.block_incremental = params->block_incremental; 1082*09d6c965SJuan Quintela } 1083*09d6c965SJuan Quintela if (params->has_multifd_channels) { 1084*09d6c965SJuan Quintela s->parameters.multifd_channels = params->multifd_channels; 1085*09d6c965SJuan Quintela } 1086*09d6c965SJuan Quintela if (params->has_multifd_compression) { 1087*09d6c965SJuan Quintela s->parameters.multifd_compression = params->multifd_compression; 1088*09d6c965SJuan Quintela } 1089*09d6c965SJuan Quintela if (params->has_xbzrle_cache_size) { 1090*09d6c965SJuan Quintela s->parameters.xbzrle_cache_size = params->xbzrle_cache_size; 1091*09d6c965SJuan Quintela xbzrle_cache_resize(params->xbzrle_cache_size, errp); 1092*09d6c965SJuan Quintela } 1093*09d6c965SJuan Quintela if (params->has_max_postcopy_bandwidth) { 1094*09d6c965SJuan Quintela s->parameters.max_postcopy_bandwidth = params->max_postcopy_bandwidth; 1095*09d6c965SJuan Quintela if (s->to_dst_file && migration_in_postcopy()) { 1096*09d6c965SJuan Quintela qemu_file_set_rate_limit(s->to_dst_file, 1097*09d6c965SJuan Quintela s->parameters.max_postcopy_bandwidth / XFER_LIMIT_RATIO); 1098*09d6c965SJuan Quintela } 1099*09d6c965SJuan Quintela } 1100*09d6c965SJuan Quintela if (params->has_max_cpu_throttle) { 1101*09d6c965SJuan Quintela s->parameters.max_cpu_throttle = params->max_cpu_throttle; 1102*09d6c965SJuan Quintela } 1103*09d6c965SJuan Quintela if (params->has_announce_initial) { 1104*09d6c965SJuan Quintela s->parameters.announce_initial = params->announce_initial; 1105*09d6c965SJuan Quintela } 1106*09d6c965SJuan Quintela if (params->has_announce_max) { 1107*09d6c965SJuan Quintela s->parameters.announce_max = params->announce_max; 1108*09d6c965SJuan Quintela } 1109*09d6c965SJuan Quintela if (params->has_announce_rounds) { 1110*09d6c965SJuan Quintela s->parameters.announce_rounds = params->announce_rounds; 1111*09d6c965SJuan Quintela } 1112*09d6c965SJuan Quintela if (params->has_announce_step) { 1113*09d6c965SJuan Quintela s->parameters.announce_step = params->announce_step; 1114*09d6c965SJuan Quintela } 1115*09d6c965SJuan Quintela 1116*09d6c965SJuan Quintela if (params->has_block_bitmap_mapping) { 1117*09d6c965SJuan Quintela qapi_free_BitmapMigrationNodeAliasList( 1118*09d6c965SJuan Quintela s->parameters.block_bitmap_mapping); 1119*09d6c965SJuan Quintela 1120*09d6c965SJuan Quintela s->parameters.has_block_bitmap_mapping = true; 1121*09d6c965SJuan Quintela s->parameters.block_bitmap_mapping = 1122*09d6c965SJuan Quintela QAPI_CLONE(BitmapMigrationNodeAliasList, 1123*09d6c965SJuan Quintela params->block_bitmap_mapping); 1124*09d6c965SJuan Quintela } 1125*09d6c965SJuan Quintela } 1126*09d6c965SJuan Quintela 1127*09d6c965SJuan Quintela void qmp_migrate_set_parameters(MigrateSetParameters *params, Error **errp) 1128*09d6c965SJuan Quintela { 1129*09d6c965SJuan Quintela MigrationParameters tmp; 1130*09d6c965SJuan Quintela 1131*09d6c965SJuan Quintela /* TODO Rewrite "" to null instead */ 1132*09d6c965SJuan Quintela if (params->tls_creds 1133*09d6c965SJuan Quintela && params->tls_creds->type == QTYPE_QNULL) { 1134*09d6c965SJuan Quintela qobject_unref(params->tls_creds->u.n); 1135*09d6c965SJuan Quintela params->tls_creds->type = QTYPE_QSTRING; 1136*09d6c965SJuan Quintela params->tls_creds->u.s = strdup(""); 1137*09d6c965SJuan Quintela } 1138*09d6c965SJuan Quintela /* TODO Rewrite "" to null instead */ 1139*09d6c965SJuan Quintela if (params->tls_hostname 1140*09d6c965SJuan Quintela && params->tls_hostname->type == QTYPE_QNULL) { 1141*09d6c965SJuan Quintela qobject_unref(params->tls_hostname->u.n); 1142*09d6c965SJuan Quintela params->tls_hostname->type = QTYPE_QSTRING; 1143*09d6c965SJuan Quintela params->tls_hostname->u.s = strdup(""); 1144*09d6c965SJuan Quintela } 1145*09d6c965SJuan Quintela 1146*09d6c965SJuan Quintela migrate_params_test_apply(params, &tmp); 1147*09d6c965SJuan Quintela 1148*09d6c965SJuan Quintela if (!migrate_params_check(&tmp, errp)) { 1149*09d6c965SJuan Quintela /* Invalid parameter */ 1150*09d6c965SJuan Quintela return; 1151*09d6c965SJuan Quintela } 1152*09d6c965SJuan Quintela 1153*09d6c965SJuan Quintela migrate_params_apply(params, errp); 1154*09d6c965SJuan Quintela } 1155