Add VIR_DOMAIN_VCPU_ASYNC_UNPLUG for virDomainSetVcpusFlags().
With this flag, success indicates that QEMU accepted the unplug
request, while final completion is reported by the vcpu-removed
event. Rejected requests continue to be reported by the
device-removal-failed event.
Wire the flag through the QEMU driver, document its semantics, and
add virsh support for the async path in the setvcpus subcommand.
Signed-off-by: Akash Kulhalli <akash.kulhalli@oracle.com>
Reviewed-by: Peter Krempa <pkrempa@redhat.com>
::
- setvcpus domain count [--maximum] [[--config] [--live] | [--current]] [--guest] [--hotpluggable]
+ setvcpus domain count [--maximum] [[--config] [--live] | [--current]] [--guest] [--hotpluggable] [--async]
Change the number of virtual CPUs active in a guest domain. By default,
this command works on active guest domains. To change the settings for an
therefore whether the XML configuration is adjusted to make the change
persistent.
+If *--async* is specified, live vCPU unplug requests are fired without waiting
+for the guest to comply. Final completion is reported by the ``vcpu-removed``
+domain event, while rejected unplug requests continue to be reported by
+``device-removal-failed``. This flag cannot be combined with *--guest*.
+
If *--guest* is specified, then the count of cpus is modified in the guest
instead of the hypervisor. This flag is usable only for live domains
and may require guest agent to be configured in the guest.
VIR_DOMAIN_VCPU_MAXIMUM = (1 << 2), /* Max rather than current count (Since: 0.8.5) */
VIR_DOMAIN_VCPU_GUEST = (1 << 3), /* Modify state of the cpu in the guest (Since: 1.1.0) */
VIR_DOMAIN_VCPU_HOTPLUGGABLE = (1 << 4), /* Make vcpus added hot(un)pluggable (Since: 2.4.0) */
+ VIR_DOMAIN_VCPU_ASYNC_UNPLUG = (1 << 5), /* Don't wait for the guest to comply with unplug request(s) (Since: 12.4.0) */
} virDomainVcpuFlags;
int virDomainSetVcpus (virDomainPtr domain,
* be used with live guests and is incompatible with VIR_DOMAIN_VCPU_MAXIMUM.
* The usage of this flag may require a guest agent configured.
*
+ * If @flags includes VIR_DOMAIN_VCPU_ASYNC_UNPLUG, live vCPU hot-unplug
+ * request(s) are fired without waiting for the guest to comply. Success in
+ * this mode only means that the unplug request(s) were accepted. Final
+ * completion is reported by VIR_DOMAIN_EVENT_ID_VCPU_REMOVED, carrying the
+ * XML ``<vcpu id='...'>`` value for each removed vCPU. Rejected unplug
+ * requests continue to be reported through the event
+ * VIR_DOMAIN_EVENT_ID_DEVICE_REMOVAL_FAILED. The success event may be
+ * delivered before this API call returns. This flag has no effect when this
+ * operation results in an increase in the live vCPU count.
+ *
* Not all hypervisors can support all flag combinations.
*
* Returns 0 in case of success, -1 in case of failure.
virDomainObj *vm = NULL;
virDomainDef *def;
virDomainDef *persistentDef;
+ bool async_unplug = !!(flags & VIR_DOMAIN_VCPU_ASYNC_UNPLUG);
bool hotpluggable = !!(flags & VIR_DOMAIN_VCPU_HOTPLUGGABLE);
bool useAgent = !!(flags & VIR_DOMAIN_VCPU_GUEST);
int ret = -1;
VIR_DOMAIN_AFFECT_CONFIG |
VIR_DOMAIN_VCPU_MAXIMUM |
VIR_DOMAIN_VCPU_GUEST |
- VIR_DOMAIN_VCPU_HOTPLUGGABLE, -1);
+ VIR_DOMAIN_VCPU_HOTPLUGGABLE |
+ VIR_DOMAIN_VCPU_ASYNC_UNPLUG, -1);
if (!(vm = qemuDomainObjFromDomain(dom)))
goto cleanup;
if (virDomainObjGetDefs(vm, flags, &def, &persistentDef) < 0)
goto endjob;
+ if (async_unplug && useAgent) {
+ virReportError(VIR_ERR_OPERATION_INVALID, "%s",
+ _("asynchronous mode is unsupported with VIR_DOMAIN_VCPU_GUEST"));
+ goto endjob;
+ }
+
if (useAgent)
ret = qemuDomainSetVcpusAgent(vm, nvcpus);
else if (flags & VIR_DOMAIN_VCPU_MAXIMUM)
ret = qemuDomainSetVcpusMax(driver, vm, def, persistentDef, nvcpus);
else
ret = qemuDomainSetVcpusInternal(driver, vm, def, persistentDef,
- nvcpus, hotpluggable, false);
+ nvcpus, hotpluggable, async_unplug);
endjob:
if (useAgent)
.type = VSH_OT_BOOL,
.help = N_("make added vcpus hot(un)pluggable")
},
+ {.name = "async",
+ .type = VSH_OT_BOOL,
+ .help = N_("return after firing vcpu unplug request(s)")
+ },
{.name = NULL}
};
bool current = vshCommandOptBool(cmd, "current");
bool guest = vshCommandOptBool(cmd, "guest");
bool hotpluggable = vshCommandOptBool(cmd, "hotpluggable");
+ bool async = vshCommandOptBool(cmd, "async");
unsigned int flags = VIR_DOMAIN_AFFECT_CURRENT;
VSH_EXCLUSIVE_OPTIONS_VAR(current, live);
VSH_EXCLUSIVE_OPTIONS_VAR(current, config);
VSH_EXCLUSIVE_OPTIONS_VAR(guest, config);
+ VSH_EXCLUSIVE_OPTIONS_VAR(async, guest);
VSH_REQUIRE_OPTION_VAR(maximum, config);
flags |= VIR_DOMAIN_VCPU_MAXIMUM;
if (hotpluggable)
flags |= VIR_DOMAIN_VCPU_HOTPLUGGABLE;
+ if (async)
+ flags |= VIR_DOMAIN_VCPU_ASYNC_UNPLUG;
if (!(dom = virshCommandOptDomain(ctl, cmd, NULL)))
return false;