xref: /openbmc/qemu/migration/options.c (revision 09d6c9658474e8573c5ada58dca8b20fe47dd99e)
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 &ap;
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