2009-11-03 13:59:18 -05:00
|
|
|
/*
|
|
|
|
* qemu_monitor_json.c: interaction with QEMU monitor console
|
|
|
|
*
|
2016-02-10 07:33:47 -05:00
|
|
|
* Copyright (C) 2006-2016 Red Hat, Inc.
|
2009-11-03 13:59:18 -05:00
|
|
|
* Copyright (C) 2006 Daniel P. Berrange
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
2012-09-20 16:30:55 -06:00
|
|
|
* License along with this library. If not, see
|
2012-07-21 18:06:23 +08:00
|
|
|
* <http://www.gnu.org/licenses/>.
|
2009-11-03 13:59:18 -05:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include <poll.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <sys/time.h>
|
|
|
|
|
|
|
|
#include "qemu_monitor_json.h"
|
2016-06-29 12:37:39 -04:00
|
|
|
#include "qemu_alias.h"
|
2012-02-13 12:19:24 +01:00
|
|
|
#include "qemu_capabilities.h"
|
2012-12-12 18:06:53 +00:00
|
|
|
#include "viralloc.h"
|
2012-12-12 17:59:27 +00:00
|
|
|
#include "virlog.h"
|
2009-11-03 13:59:18 -05:00
|
|
|
#include "driver.h"
|
|
|
|
#include "datatypes.h"
|
2012-12-13 18:21:53 +00:00
|
|
|
#include "virerror.h"
|
2012-12-12 17:53:50 +00:00
|
|
|
#include "virjson.h"
|
2014-02-27 13:41:11 +00:00
|
|
|
#include "virprobe.h"
|
2013-04-12 16:55:45 -04:00
|
|
|
#include "virstring.h"
|
2013-07-22 13:07:23 +02:00
|
|
|
#include "cpu/cpu_x86.h"
|
2019-04-01 12:14:26 +02:00
|
|
|
#include "virenum.h"
|
2020-01-14 17:38:59 +00:00
|
|
|
#include "virsocket.h"
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2012-04-02 18:24:29 +01:00
|
|
|
#ifdef WITH_DTRACE_PROBES
|
|
|
|
# include "libvirt_qemu_probes.h"
|
|
|
|
#endif
|
|
|
|
|
2009-11-03 13:59:18 -05:00
|
|
|
#define VIR_FROM_THIS VIR_FROM_QEMU
|
|
|
|
|
2014-02-28 12:16:17 +00:00
|
|
|
VIR_LOG_INIT("qemu.qemu_monitor_json");
|
|
|
|
|
2013-07-22 13:07:23 +02:00
|
|
|
#define QOM_CPU_PATH "/machine/unattached/device[0]"
|
2009-11-03 13:59:18 -05:00
|
|
|
|
|
|
|
#define LINE_ENDING "\r\n"
|
|
|
|
|
2018-12-04 17:58:38 +01:00
|
|
|
VIR_ENUM_IMPL(qemuMonitorJob,
|
|
|
|
QEMU_MONITOR_JOB_TYPE_LAST,
|
|
|
|
"",
|
|
|
|
"commit",
|
|
|
|
"stream",
|
|
|
|
"mirror",
|
|
|
|
"backup",
|
|
|
|
"create");
|
|
|
|
|
|
|
|
VIR_ENUM_IMPL(qemuMonitorJobStatus,
|
|
|
|
QEMU_MONITOR_JOB_STATUS_LAST,
|
|
|
|
"",
|
|
|
|
"created",
|
|
|
|
"running",
|
|
|
|
"paused",
|
|
|
|
"ready",
|
|
|
|
"standby",
|
|
|
|
"waiting",
|
|
|
|
"pending",
|
|
|
|
"aborting",
|
|
|
|
"concluded",
|
|
|
|
"undefined",
|
|
|
|
"null");
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static void qemuMonitorJSONHandleShutdown(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleReset(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandlePowerdown(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleStop(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleResume(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleRTCChange(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleWatchdog(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleIOError(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleVNCConnect(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleVNCInitialize(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleVNCDisconnect(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleSPICEConnect(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleSPICEInitialize(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleSPICEDisconnect(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleTrayChange(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandlePMWakeup(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandlePMSuspend(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleBlockJobCompleted(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleBlockJobCanceled(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleBlockJobReady(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleJobStatusChange(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleBalloonChange(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandlePMSuspendDisk(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleGuestCrashloaded(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleGuestPanic(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleDeviceDeleted(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleNicRxFilterChanged(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleSerialChange(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleSpiceMigrated(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleMigrationStatus(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleMigrationPass(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleAcpiOstInfo(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleBlockThreshold(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleDumpCompleted(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandlePRManagerStatusChanged(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleRdmaGidStatusChanged(qemuMonitor *mon, virJSONValue *data);
|
|
|
|
static void qemuMonitorJSONHandleMemoryFailure(qemuMonitor *mon, virJSONValue *data);
|
2009-11-26 13:06:24 +00:00
|
|
|
|
2012-04-11 08:58:09 -06:00
|
|
|
typedef struct {
|
2009-11-26 13:06:24 +00:00
|
|
|
const char *type;
|
2021-03-11 08:16:13 +01:00
|
|
|
void (*handler)(qemuMonitor *mon, virJSONValue *data);
|
2012-04-11 08:58:09 -06:00
|
|
|
} qemuEventHandler;
|
|
|
|
|
|
|
|
static qemuEventHandler eventHandlers[] = {
|
2016-04-01 16:41:08 +02:00
|
|
|
{ "ACPI_DEVICE_OST", qemuMonitorJSONHandleAcpiOstInfo, },
|
2012-07-12 23:45:57 +08:00
|
|
|
{ "BALLOON_CHANGE", qemuMonitorJSONHandleBalloonChange, },
|
2010-02-16 12:07:49 +00:00
|
|
|
{ "BLOCK_IO_ERROR", qemuMonitorJSONHandleIOError, },
|
2012-04-11 16:17:36 -06:00
|
|
|
{ "BLOCK_JOB_CANCELLED", qemuMonitorJSONHandleBlockJobCanceled, },
|
|
|
|
{ "BLOCK_JOB_COMPLETED", qemuMonitorJSONHandleBlockJobCompleted, },
|
2012-10-12 14:06:10 -06:00
|
|
|
{ "BLOCK_JOB_READY", qemuMonitorJSONHandleBlockJobReady, },
|
2017-02-22 16:52:22 +01:00
|
|
|
{ "BLOCK_WRITE_THRESHOLD", qemuMonitorJSONHandleBlockThreshold, },
|
2013-07-11 17:07:26 +02:00
|
|
|
{ "DEVICE_DELETED", qemuMonitorJSONHandleDeviceDeleted, },
|
2012-04-11 08:58:09 -06:00
|
|
|
{ "DEVICE_TRAY_MOVED", qemuMonitorJSONHandleTrayChange, },
|
2017-11-20 08:56:24 -05:00
|
|
|
{ "DUMP_COMPLETED", qemuMonitorJSONHandleDumpCompleted, },
|
qemu: support Panic Crashloaded event handling
Pvpanic device supports bit 1 as crashloaded event, it means that
guest actually panicked and run kexec to handle error by guest side.
Handle crashloaded as a lifecyle event in libvirt.
Test case:
Guest side:
before testing, we need make sure kdump is enabled,
1, build new pvpanic driver (with commit from upstream
e0b9a42735f2672ca2764cfbea6e55a81098d5ba
191941692a3d1b6a9614502b279be062926b70f5)
2, insmod new kmod
3, enable crash_kexec_post_notifiers,
# echo 1 > /sys/module/kernel/parameters/crash_kexec_post_notifiers
4, trigger kernel panic
# echo 1 > /proc/sys/kernel/sysrq
# echo c > /proc/sysrq-trigger
Host side:
1, build new qemu with pvpanic patches (with commit from upstream
600d7b47e8f5085919fd1d1157f25950ea8dbc11
7dc58deea79a343ac3adc5cadb97215086054c86)
2, build libvirt with this patch
3, handle lifecycle event and trigger guest side panic
# virsh event stretch --event lifecycle
event 'lifecycle' for domain stretch: Crashed Crashloaded
events received: 1
Reviewed-by: Daniel P. Berrangé <berrange@redhat.com>
Signed-off-by: zhenwei pi <pizhenwei@bytedance.com>
2020-02-04 15:41:00 +08:00
|
|
|
{ "GUEST_CRASHLOADED", qemuMonitorJSONHandleGuestCrashloaded, },
|
2013-06-07 18:23:34 +08:00
|
|
|
{ "GUEST_PANICKED", qemuMonitorJSONHandleGuestPanic, },
|
2018-12-04 17:57:23 +01:00
|
|
|
{ "JOB_STATUS_CHANGE", qemuMonitorJSONHandleJobStatusChange, },
|
2020-10-14 18:37:51 +08:00
|
|
|
{ "MEMORY_FAILURE", qemuMonitorJSONHandleMemoryFailure, },
|
2015-05-28 13:35:52 +02:00
|
|
|
{ "MIGRATION", qemuMonitorJSONHandleMigrationStatus, },
|
2015-12-08 15:23:35 +01:00
|
|
|
{ "MIGRATION_PASS", qemuMonitorJSONHandleMigrationPass, },
|
2014-09-17 13:07:50 -04:00
|
|
|
{ "NIC_RX_FILTER_CHANGED", qemuMonitorJSONHandleNicRxFilterChanged, },
|
2012-04-11 08:58:09 -06:00
|
|
|
{ "POWERDOWN", qemuMonitorJSONHandlePowerdown, },
|
2018-06-27 12:17:59 +02:00
|
|
|
{ "PR_MANAGER_STATUS_CHANGED", qemuMonitorJSONHandlePRManagerStatusChanged, },
|
2018-12-24 12:15:12 +02:00
|
|
|
{ "RDMA_GID_STATUS_CHANGED", qemuMonitorJSONHandleRdmaGidStatusChanged, },
|
2012-04-11 08:58:09 -06:00
|
|
|
{ "RESET", qemuMonitorJSONHandleReset, },
|
2013-01-07 16:25:01 -05:00
|
|
|
{ "RESUME", qemuMonitorJSONHandleResume, },
|
2012-04-11 08:58:09 -06:00
|
|
|
{ "RTC_CHANGE", qemuMonitorJSONHandleRTCChange, },
|
|
|
|
{ "SHUTDOWN", qemuMonitorJSONHandleShutdown, },
|
2012-03-14 01:41:35 -04:00
|
|
|
{ "SPICE_CONNECTED", qemuMonitorJSONHandleSPICEConnect, },
|
|
|
|
{ "SPICE_DISCONNECTED", qemuMonitorJSONHandleSPICEDisconnect, },
|
2012-04-11 08:58:09 -06:00
|
|
|
{ "SPICE_INITIALIZED", qemuMonitorJSONHandleSPICEInitialize, },
|
2015-05-28 13:35:06 +02:00
|
|
|
{ "SPICE_MIGRATE_COMPLETED", qemuMonitorJSONHandleSpiceMigrated, },
|
2012-04-11 08:58:09 -06:00
|
|
|
{ "STOP", qemuMonitorJSONHandleStop, },
|
2012-03-23 22:50:36 +08:00
|
|
|
{ "SUSPEND", qemuMonitorJSONHandlePMSuspend, },
|
2012-10-12 21:13:39 +02:00
|
|
|
{ "SUSPEND_DISK", qemuMonitorJSONHandlePMSuspendDisk, },
|
2012-04-11 08:58:09 -06:00
|
|
|
{ "VNC_CONNECTED", qemuMonitorJSONHandleVNCConnect, },
|
|
|
|
{ "VNC_DISCONNECTED", qemuMonitorJSONHandleVNCDisconnect, },
|
|
|
|
{ "VNC_INITIALIZED", qemuMonitorJSONHandleVNCInitialize, },
|
2014-11-13 14:09:39 +01:00
|
|
|
{ "VSERPORT_CHANGE", qemuMonitorJSONHandleSerialChange, },
|
2012-04-11 08:58:09 -06:00
|
|
|
{ "WAKEUP", qemuMonitorJSONHandlePMWakeup, },
|
|
|
|
{ "WATCHDOG", qemuMonitorJSONHandleWatchdog, },
|
|
|
|
/* We use bsearch, so keep this list sorted. */
|
2009-11-26 13:06:24 +00:00
|
|
|
};
|
|
|
|
|
2012-04-11 08:58:09 -06:00
|
|
|
static int
|
|
|
|
qemuMonitorEventCompare(const void *key, const void *elt)
|
|
|
|
{
|
|
|
|
const char *type = key;
|
|
|
|
const qemuEventHandler *handler = elt;
|
|
|
|
return strcmp(type, handler->type);
|
|
|
|
}
|
2009-11-26 13:06:24 +00:00
|
|
|
|
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONIOProcessEvent(qemuMonitor *mon,
|
|
|
|
virJSONValue *obj)
|
2009-11-26 13:06:24 +00:00
|
|
|
{
|
2010-01-22 13:22:53 +00:00
|
|
|
const char *type;
|
2012-04-11 08:58:09 -06:00
|
|
|
qemuEventHandler *handler;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *data;
|
2014-01-29 17:14:44 -07:00
|
|
|
char *details = NULL;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *timestamp;
|
2014-01-29 17:14:44 -07:00
|
|
|
long long seconds = -1;
|
|
|
|
unsigned int micros = 0;
|
|
|
|
|
2009-11-26 13:06:24 +00:00
|
|
|
VIR_DEBUG("mon=%p obj=%p", mon, obj);
|
|
|
|
|
|
|
|
type = virJSONValueObjectGetString(obj, "event");
|
|
|
|
if (!type) {
|
2011-05-09 17:24:09 +08:00
|
|
|
VIR_WARN("missing event type in message");
|
2009-11-26 13:06:24 +00:00
|
|
|
errno = EINVAL;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2014-01-29 17:14:44 -07:00
|
|
|
/* Not all events have data; and event reporting is best-effort only */
|
|
|
|
if ((data = virJSONValueObjectGet(obj, "data")))
|
|
|
|
details = virJSONValueToString(data, false);
|
|
|
|
if ((timestamp = virJSONValueObjectGet(obj, "timestamp"))) {
|
qemu: simplify json parsing
Rather than grabbing an arbitrary JSON value and then checking
if it has the right type, we might as well request the correct
type to begin with.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONIOProcessEvent)
(qemuMonitorJSONCommandWithFd, qemuMonitorJSONHandleGraphics)
(qemuMonitorJSONGetStatus, qemuMonitorJSONExtractCPUInfo)
(qemuMonitorJSONGetVirtType, qemuMonitorJSONGetBalloonInfo)
(qemuMonitorJSONGetMemoryStats)
(qemuMonitorJSONDevGetBlockExtent)
(qemuMonitorJSONGetOneBlockStatsInfo)
(qemuMonitorJSONGetAllBlockStatsInfo)
(qemuMonitorJSONBlockStatsUpdateCapacityOne)
(qemuMonitorJSONBlockStatsUpdateCapacity)
(qemuMonitorJSONGetBlockExtent)
(qemuMonitorJSONGetMigrationStatusReply)
(qemuMonitorJSONGetDumpGuestMemoryCapability)
(qemuMonitorJSONAddFd, qemuMonitorJSONQueryRxFilterParse)
(qemuMonitorJSONExtractChardevInfo)
(qemuMonitorJSONDiskNameLookupOne)
(qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONGetAllBlockJobInfo)
(qemuMonitorJSONBlockIoThrottleInfo, qemuMonitorJSONGetVersion)
(qemuMonitorJSONGetMachines, qemuMonitorJSONGetCPUDefinitions)
(qemuMonitorJSONGetCommands, qemuMonitorJSONGetEvents)
(qemuMonitorJSONGetKVMState, qemuMonitorJSONGetObjectTypes)
(qemuMonitorJSONGetObjectListPaths)
(qemuMonitorJSONGetObjectProps, qemuMonitorJSONGetTargetArch)
(qemuMonitorJSONGetMigrationCapabilities)
(qemuMonitorJSONGetStringArray, qemuMonitorJSONAttachCharDev)
(qemuMonitorJSONGetCPUx86Data, qemuMonitorJSONGetIOThreads)
(qemuMonitorJSONGetMemoryDeviceInfo): Use shorter idioms.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-06-20 13:40:21 -06:00
|
|
|
ignore_value(virJSONValueObjectGetNumberLong(timestamp, "seconds",
|
|
|
|
&seconds));
|
|
|
|
ignore_value(virJSONValueObjectGetNumberUint(timestamp, "microseconds",
|
|
|
|
µs));
|
2014-01-29 17:14:44 -07:00
|
|
|
}
|
|
|
|
qemuMonitorEmitEvent(mon, type, seconds, micros, details);
|
|
|
|
VIR_FREE(details);
|
|
|
|
|
2019-10-15 13:55:26 +02:00
|
|
|
handler = bsearch(type, eventHandlers, G_N_ELEMENTS(eventHandlers),
|
2012-04-11 08:58:09 -06:00
|
|
|
sizeof(eventHandlers[0]), qemuMonitorEventCompare);
|
|
|
|
if (handler) {
|
|
|
|
VIR_DEBUG("handle %s handler=%p data=%p", type,
|
|
|
|
handler->handler, data);
|
|
|
|
(handler->handler)(mon, data);
|
2009-11-26 13:06:24 +00:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-04-27 16:50:43 +02:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONIOProcessLine(qemuMonitor *mon,
|
2009-11-03 13:59:18 -05:00
|
|
|
const char *line,
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorMessage *msg)
|
2009-11-03 13:59:18 -05:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *obj = NULL;
|
2009-11-03 13:59:18 -05:00
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
VIR_DEBUG("Line [%s]", line);
|
|
|
|
|
2011-05-29 20:51:08 +08:00
|
|
|
if (!(obj = virJSONValueFromString(line)))
|
2009-11-03 13:59:18 -05:00
|
|
|
goto cleanup;
|
|
|
|
|
2018-03-29 20:30:05 +02:00
|
|
|
if (virJSONValueGetType(obj) != VIR_JSON_TYPE_OBJECT) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Parsed JSON reply '%s' isn't an object"), line);
|
2011-05-29 20:51:08 +08:00
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectHasKey(obj, "QMP") == 1) {
|
|
|
|
ret = 0;
|
2011-05-29 20:51:08 +08:00
|
|
|
} else if (virJSONValueObjectHasKey(obj, "event") == 1) {
|
2011-10-24 15:31:37 +01:00
|
|
|
PROBE(QEMU_MONITOR_RECV_EVENT,
|
|
|
|
"mon=%p event=%s", mon, line);
|
2009-11-26 13:06:24 +00:00
|
|
|
ret = qemuMonitorJSONIOProcessEvent(mon, obj);
|
2011-05-29 20:51:08 +08:00
|
|
|
} else if (virJSONValueObjectHasKey(obj, "error") == 1 ||
|
|
|
|
virJSONValueObjectHasKey(obj, "return") == 1) {
|
2011-10-24 15:31:37 +01:00
|
|
|
PROBE(QEMU_MONITOR_RECV_REPLY,
|
|
|
|
"mon=%p reply=%s", mon, line);
|
2011-05-29 20:51:08 +08:00
|
|
|
if (msg) {
|
2021-03-24 10:32:58 +01:00
|
|
|
msg->rxObject = g_steal_pointer(&obj);
|
2011-05-29 20:51:08 +08:00
|
|
|
msg->finished = 1;
|
|
|
|
ret = 0;
|
|
|
|
} else {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Unexpected JSON reply '%s'"), line);
|
2009-11-03 13:59:18 -05:00
|
|
|
}
|
|
|
|
} else {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Unknown JSON reply '%s'"), line);
|
2009-11-03 13:59:18 -05:00
|
|
|
}
|
|
|
|
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2011-06-30 15:03:31 +01:00
|
|
|
virJSONValueFree(obj);
|
2009-11-03 13:59:18 -05:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONIOProcess(qemuMonitor *mon,
|
2009-11-03 13:59:18 -05:00
|
|
|
const char *data,
|
|
|
|
size_t len,
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorMessage *msg)
|
2009-11-03 13:59:18 -05:00
|
|
|
{
|
|
|
|
int used = 0;
|
|
|
|
/*VIR_DEBUG("Data %d bytes [%s]", len, data);*/
|
|
|
|
|
|
|
|
while (used < len) {
|
|
|
|
char *nl = strstr(data + used, LINE_ENDING);
|
|
|
|
|
|
|
|
if (nl) {
|
|
|
|
int got = nl - (data + used);
|
2013-05-20 11:23:13 +02:00
|
|
|
char *line;
|
2019-10-24 23:34:40 +02:00
|
|
|
line = g_strndup(data + used, got);
|
2009-11-03 13:59:18 -05:00
|
|
|
used += got + strlen(LINE_ENDING);
|
|
|
|
line[got] = '\0'; /* kill \n */
|
|
|
|
if (qemuMonitorJSONIOProcessLine(mon, line, msg) < 0) {
|
|
|
|
VIR_FREE(line);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
VIR_FREE(line);
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-20 13:09:07 +01:00
|
|
|
#if DEBUG_IO
|
2009-11-03 13:59:18 -05:00
|
|
|
VIR_DEBUG("Total used %d bytes out of %zd available in buffer", used, len);
|
2017-12-20 13:09:07 +01:00
|
|
|
#endif
|
|
|
|
|
2009-11-03 13:59:18 -05:00
|
|
|
return used;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONCommandWithFd(qemuMonitor *mon,
|
|
|
|
virJSONValue *cmd,
|
2009-11-03 13:59:18 -05:00
|
|
|
int scm_fd,
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue **reply)
|
2009-11-03 13:59:18 -05:00
|
|
|
{
|
|
|
|
int ret = -1;
|
|
|
|
qemuMonitorMessage msg;
|
2019-10-15 14:47:50 +02:00
|
|
|
g_auto(virBuffer) cmdbuf = VIR_BUFFER_INITIALIZER;
|
2011-05-29 20:51:08 +08:00
|
|
|
char *id = NULL;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
|
|
|
*reply = NULL;
|
|
|
|
|
2012-03-29 10:52:04 +01:00
|
|
|
memset(&msg, 0, sizeof(msg));
|
2009-11-03 13:59:18 -05:00
|
|
|
|
qemu: simplify json parsing
Rather than grabbing an arbitrary JSON value and then checking
if it has the right type, we might as well request the correct
type to begin with.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONIOProcessEvent)
(qemuMonitorJSONCommandWithFd, qemuMonitorJSONHandleGraphics)
(qemuMonitorJSONGetStatus, qemuMonitorJSONExtractCPUInfo)
(qemuMonitorJSONGetVirtType, qemuMonitorJSONGetBalloonInfo)
(qemuMonitorJSONGetMemoryStats)
(qemuMonitorJSONDevGetBlockExtent)
(qemuMonitorJSONGetOneBlockStatsInfo)
(qemuMonitorJSONGetAllBlockStatsInfo)
(qemuMonitorJSONBlockStatsUpdateCapacityOne)
(qemuMonitorJSONBlockStatsUpdateCapacity)
(qemuMonitorJSONGetBlockExtent)
(qemuMonitorJSONGetMigrationStatusReply)
(qemuMonitorJSONGetDumpGuestMemoryCapability)
(qemuMonitorJSONAddFd, qemuMonitorJSONQueryRxFilterParse)
(qemuMonitorJSONExtractChardevInfo)
(qemuMonitorJSONDiskNameLookupOne)
(qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONGetAllBlockJobInfo)
(qemuMonitorJSONBlockIoThrottleInfo, qemuMonitorJSONGetVersion)
(qemuMonitorJSONGetMachines, qemuMonitorJSONGetCPUDefinitions)
(qemuMonitorJSONGetCommands, qemuMonitorJSONGetEvents)
(qemuMonitorJSONGetKVMState, qemuMonitorJSONGetObjectTypes)
(qemuMonitorJSONGetObjectListPaths)
(qemuMonitorJSONGetObjectProps, qemuMonitorJSONGetTargetArch)
(qemuMonitorJSONGetMigrationCapabilities)
(qemuMonitorJSONGetStringArray, qemuMonitorJSONAttachCharDev)
(qemuMonitorJSONGetCPUx86Data, qemuMonitorJSONGetIOThreads)
(qemuMonitorJSONGetMemoryDeviceInfo): Use shorter idioms.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-06-20 13:40:21 -06:00
|
|
|
if (virJSONValueObjectHasKey(cmd, "execute") == 1) {
|
2011-05-29 20:51:08 +08:00
|
|
|
if (!(id = qemuMonitorNextCommandID(mon)))
|
|
|
|
goto cleanup;
|
|
|
|
if (virJSONValueObjectAppendString(cmd, "id", id) < 0) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Unable to append command 'id' string"));
|
2011-05-29 20:51:08 +08:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-26 18:33:01 +01:00
|
|
|
if (virJSONValueToBuffer(cmd, &cmdbuf, false) < 0)
|
2009-11-03 13:59:18 -05:00
|
|
|
goto cleanup;
|
2019-03-26 18:33:01 +01:00
|
|
|
virBufferAddLit(&cmdbuf, "\r\n");
|
|
|
|
|
|
|
|
msg.txLength = virBufferUse(&cmdbuf);
|
|
|
|
msg.txBuffer = virBufferContentAndReset(&cmdbuf);
|
2009-11-03 13:59:18 -05:00
|
|
|
msg.txFD = scm_fd;
|
|
|
|
|
|
|
|
ret = qemuMonitorSend(mon, &msg);
|
|
|
|
|
|
|
|
if (ret == 0) {
|
2011-05-29 20:51:08 +08:00
|
|
|
if (!msg.rxObject) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Missing monitor reply object"));
|
2011-05-29 20:51:08 +08:00
|
|
|
ret = -1;
|
|
|
|
} else {
|
|
|
|
*reply = msg.rxObject;
|
2009-11-03 13:59:18 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2011-05-29 20:51:08 +08:00
|
|
|
VIR_FREE(id);
|
2009-11-03 13:59:18 -05:00
|
|
|
VIR_FREE(msg.txBuffer);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONCommand(qemuMonitor *mon,
|
|
|
|
virJSONValue *cmd,
|
|
|
|
virJSONValue **reply)
|
2014-03-18 09:15:21 +01:00
|
|
|
{
|
2009-11-03 13:59:18 -05:00
|
|
|
return qemuMonitorJSONCommandWithFd(mon, cmd, -1, reply);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Ignoring OOM in this method, since we're already reporting
|
|
|
|
* a more important error
|
|
|
|
*
|
|
|
|
* XXX see qerror.h for different klasses & fill out useful params
|
|
|
|
*/
|
2010-02-12 16:45:11 +00:00
|
|
|
static const char *
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONStringifyError(virJSONValue *error)
|
2009-11-03 13:59:18 -05:00
|
|
|
{
|
2010-01-22 13:22:53 +00:00
|
|
|
const char *klass = virJSONValueObjectGetString(error, "class");
|
2010-02-12 16:45:11 +00:00
|
|
|
const char *detail = NULL;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2010-02-12 16:45:11 +00:00
|
|
|
/* The QMP 'desc' field is usually sufficient for our generic
|
|
|
|
* error reporting needs.
|
|
|
|
*/
|
|
|
|
if (klass)
|
|
|
|
detail = virJSONValueObjectGetString(error, "desc");
|
|
|
|
|
|
|
|
|
|
|
|
if (!detail)
|
|
|
|
detail = "unknown QEMU command error";
|
|
|
|
|
|
|
|
return detail;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char *
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONCommandName(virJSONValue *cmd)
|
2010-02-12 16:45:11 +00:00
|
|
|
{
|
|
|
|
const char *name = virJSONValueObjectGetString(cmd, "execute");
|
|
|
|
if (name)
|
|
|
|
return name;
|
|
|
|
else
|
|
|
|
return "<unknown>";
|
2009-11-03 13:59:18 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONCheckErrorFull(virJSONValue *cmd,
|
|
|
|
virJSONValue *reply,
|
2020-03-18 11:08:58 +01:00
|
|
|
bool report)
|
2009-11-03 13:59:18 -05:00
|
|
|
{
|
|
|
|
if (virJSONValueObjectHasKey(reply, "error")) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *error = virJSONValueObjectGet(reply, "error");
|
2020-03-18 10:34:32 +01:00
|
|
|
g_autofree char *cmdstr = virJSONValueToString(cmd, false);
|
|
|
|
g_autofree char *replystr = virJSONValueToString(reply, false);
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2010-02-12 16:45:11 +00:00
|
|
|
/* Log the full JSON formatted command & error */
|
|
|
|
VIR_DEBUG("unable to execute QEMU command %s: %s",
|
2013-02-22 09:41:38 -07:00
|
|
|
NULLSTR(cmdstr), NULLSTR(replystr));
|
2010-02-12 16:45:11 +00:00
|
|
|
|
2020-03-18 11:08:58 +01:00
|
|
|
if (!report)
|
|
|
|
return -1;
|
|
|
|
|
2010-02-12 16:45:11 +00:00
|
|
|
/* Only send the user the command name + friendly error */
|
|
|
|
if (!error)
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("unable to execute QEMU command '%s'"),
|
|
|
|
qemuMonitorJSONCommandName(cmd));
|
2010-02-12 16:45:11 +00:00
|
|
|
else
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("unable to execute QEMU command '%s': %s"),
|
|
|
|
qemuMonitorJSONCommandName(cmd),
|
|
|
|
qemuMonitorJSONStringifyError(error));
|
2010-02-12 16:45:11 +00:00
|
|
|
|
2009-11-03 13:59:18 -05:00
|
|
|
return -1;
|
|
|
|
} else if (!virJSONValueObjectHasKey(reply, "return")) {
|
2020-03-18 10:34:32 +01:00
|
|
|
g_autofree char *cmdstr = virJSONValueToString(cmd, false);
|
|
|
|
g_autofree char *replystr = virJSONValueToString(reply, false);
|
2009-11-03 13:59:18 -05:00
|
|
|
|
|
|
|
VIR_DEBUG("Neither 'return' nor 'error' is set in the JSON reply %s: %s",
|
2013-02-22 09:41:38 -07:00
|
|
|
NULLSTR(cmdstr), NULLSTR(replystr));
|
2020-03-18 11:08:58 +01:00
|
|
|
|
|
|
|
if (!report)
|
|
|
|
return -1;
|
|
|
|
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("unable to execute QEMU command '%s'"),
|
|
|
|
qemuMonitorJSONCommandName(cmd));
|
2009-11-03 13:59:18 -05:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-03-18 11:08:58 +01:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONCheckError(virJSONValue *cmd,
|
|
|
|
virJSONValue *reply)
|
2020-03-18 11:08:58 +01:00
|
|
|
{
|
|
|
|
return qemuMonitorJSONCheckErrorFull(cmd, reply, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONCheckReply(virJSONValue *cmd,
|
|
|
|
virJSONValue *reply,
|
2018-03-28 12:45:21 +02:00
|
|
|
virJSONType type)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *data;
|
2018-03-28 12:45:21 +02:00
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
data = virJSONValueObjectGet(reply, "return");
|
|
|
|
if (virJSONValueGetType(data) != type) {
|
2020-10-21 17:23:40 +08:00
|
|
|
g_autofree char *cmdstr = virJSONValueToString(cmd, false);
|
|
|
|
g_autofree char *retstr = virJSONValueToString(data, false);
|
2018-03-28 12:45:21 +02:00
|
|
|
|
|
|
|
VIR_DEBUG("Unexpected return type %d (expecting %d) for command %s: %s",
|
|
|
|
virJSONValueGetType(data), type, cmdstr, retstr);
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("unexpected type returned by QEMU command '%s'"),
|
|
|
|
qemuMonitorJSONCommandName(cmd));
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-04-09 11:26:43 +02:00
|
|
|
static bool
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONErrorIsClass(virJSONValue *error,
|
2015-04-09 11:26:43 +02:00
|
|
|
const char *klass)
|
|
|
|
{
|
|
|
|
return STREQ_NULLABLE(virJSONValueObjectGetString(error, "class"), klass);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static bool
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHasError(virJSONValue *reply,
|
2009-11-03 13:59:18 -05:00
|
|
|
const char *klass)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *error;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2015-04-09 11:26:43 +02:00
|
|
|
if (!(error = virJSONValueObjectGet(reply, "error")))
|
|
|
|
return false;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2015-04-09 11:26:43 +02:00
|
|
|
return qemuMonitorJSONErrorIsClass(error, klass);
|
2009-11-03 13:59:18 -05:00
|
|
|
}
|
|
|
|
|
2015-04-09 11:26:43 +02:00
|
|
|
|
2018-07-03 09:24:40 +02:00
|
|
|
/**
|
|
|
|
* qemuMonitorJSONTransactionAdd:
|
|
|
|
* @actions: array of actions for the 'transaction' command
|
|
|
|
* @cmdname: command to add to @actions
|
|
|
|
* @...: arguments for @cmdname (see virJSONValueObjectAddVArgs for formatting)
|
|
|
|
*
|
|
|
|
* Add a new command with arguments to the existing ones. The resulting array
|
|
|
|
* is intended to be used as argument for the 'transaction' command.
|
|
|
|
*
|
|
|
|
* Returns 0 on success and -1 on error.
|
|
|
|
*/
|
2019-09-26 16:37:44 +02:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONTransactionAdd(virJSONValue *actions,
|
2018-07-03 09:24:40 +02:00
|
|
|
const char *cmdname,
|
|
|
|
...)
|
|
|
|
{
|
2021-02-12 11:51:59 +01:00
|
|
|
g_autoptr(virJSONValue) entry = NULL;
|
|
|
|
g_autoptr(virJSONValue) data = NULL;
|
2018-07-03 09:24:40 +02:00
|
|
|
va_list args;
|
|
|
|
|
|
|
|
va_start(args, cmdname);
|
|
|
|
|
2021-02-12 11:51:59 +01:00
|
|
|
if (virJSONValueObjectCreateVArgs(&data, args) < 0) {
|
|
|
|
va_end(args);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
va_end(args);
|
2018-07-03 09:24:40 +02:00
|
|
|
|
|
|
|
if (virJSONValueObjectCreate(&entry,
|
|
|
|
"s:type", cmdname,
|
|
|
|
"A:data", &data, NULL) < 0)
|
2021-02-12 11:51:59 +01:00
|
|
|
return -1;
|
2018-07-03 09:24:40 +02:00
|
|
|
|
2021-02-11 17:57:45 +01:00
|
|
|
if (virJSONValueArrayAppend(actions, &entry) < 0)
|
2021-02-12 11:51:59 +01:00
|
|
|
return -1;
|
2018-07-03 09:24:40 +02:00
|
|
|
|
2021-02-12 11:51:59 +01:00
|
|
|
return 0;
|
2018-07-03 09:24:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-02-20 17:56:46 +01:00
|
|
|
/**
|
|
|
|
* qemuMonitorJSONMakeCommandInternal:
|
|
|
|
* @cmdname: QMP command name
|
|
|
|
* @arguments: a JSON object containing command arguments or NULL
|
|
|
|
*
|
2018-07-03 14:21:56 +02:00
|
|
|
* Create a JSON object used on the QMP monitor to call a command.
|
2018-02-20 17:56:46 +01:00
|
|
|
*
|
2020-11-30 15:34:56 +01:00
|
|
|
* Note that @arguments is consumed and cleared.
|
2018-02-20 17:56:46 +01:00
|
|
|
*/
|
2021-03-11 08:16:13 +01:00
|
|
|
static virJSONValue *
|
2018-02-20 17:56:46 +01:00
|
|
|
qemuMonitorJSONMakeCommandInternal(const char *cmdname,
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue **arguments)
|
2018-02-20 17:56:46 +01:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *ret = NULL;
|
2018-02-20 17:56:46 +01:00
|
|
|
|
2018-07-03 14:21:56 +02:00
|
|
|
ignore_value(virJSONValueObjectCreate(&ret,
|
|
|
|
"s:execute", cmdname,
|
2020-11-30 15:34:56 +01:00
|
|
|
"A:arguments", arguments, NULL));
|
2018-02-20 17:56:46 +01:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static virJSONValue *G_GNUC_NULL_TERMINATED
|
2018-07-03 14:21:56 +02:00
|
|
|
qemuMonitorJSONMakeCommand(const char *cmdname,
|
|
|
|
...)
|
2009-11-03 13:59:18 -05:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *obj = NULL;
|
2020-11-30 15:34:56 +01:00
|
|
|
g_autoptr(virJSONValue) jargs = NULL;
|
2009-11-03 13:59:18 -05:00
|
|
|
va_list args;
|
|
|
|
|
|
|
|
va_start(args, cmdname);
|
|
|
|
|
2014-09-22 16:35:02 +02:00
|
|
|
if (virJSONValueObjectCreateVArgs(&jargs, args) < 0)
|
2018-02-20 17:56:46 +01:00
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2020-11-30 15:34:56 +01:00
|
|
|
obj = qemuMonitorJSONMakeCommandInternal(cmdname, &jargs);
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2018-02-20 17:56:46 +01:00
|
|
|
cleanup:
|
2009-11-03 13:59:18 -05:00
|
|
|
va_end(args);
|
|
|
|
|
|
|
|
return obj;
|
|
|
|
}
|
|
|
|
|
2010-04-15 14:52:03 +01:00
|
|
|
|
2019-06-17 14:18:51 +02:00
|
|
|
static void
|
|
|
|
qemuMonitorJSONParseKeywordsFree(int nkeywords,
|
|
|
|
char **keywords,
|
|
|
|
char **values)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
for (i = 0; i < nkeywords; i++) {
|
2021-02-03 14:36:01 -05:00
|
|
|
g_free(keywords[i]);
|
|
|
|
g_free(values[i]);
|
2019-06-17 14:18:51 +02:00
|
|
|
}
|
2021-02-03 14:36:01 -05:00
|
|
|
g_free(keywords);
|
|
|
|
g_free(values);
|
2019-06-17 14:18:51 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Takes a string containing a set of key=value,key=value,key...
|
|
|
|
* parameters and splits them up, returning two arrays with
|
2020-05-13 20:54:24 +02:00
|
|
|
* the individual keys and values.
|
|
|
|
* The "=value" part is optional and if a key with no value is found,
|
|
|
|
* NULL will be placed into corresponding place in retvalues.
|
2019-06-17 14:18:51 +02:00
|
|
|
*/
|
|
|
|
static int
|
|
|
|
qemuMonitorJSONParseKeywords(const char *str,
|
|
|
|
char ***retkeywords,
|
|
|
|
char ***retvalues,
|
2020-05-13 20:54:24 +02:00
|
|
|
int *retnkeywords)
|
2019-06-17 14:18:51 +02:00
|
|
|
{
|
|
|
|
int keywordCount = 0;
|
|
|
|
int keywordAlloc = 0;
|
|
|
|
char **keywords = NULL;
|
|
|
|
char **values = NULL;
|
|
|
|
const char *start = str;
|
|
|
|
const char *end;
|
|
|
|
|
|
|
|
*retkeywords = NULL;
|
|
|
|
*retvalues = NULL;
|
|
|
|
*retnkeywords = 0;
|
|
|
|
end = start + strlen(str);
|
|
|
|
|
|
|
|
while (start) {
|
|
|
|
const char *separator;
|
|
|
|
const char *endmark;
|
|
|
|
char *keyword;
|
|
|
|
char *value = NULL;
|
|
|
|
|
|
|
|
endmark = start;
|
|
|
|
do {
|
|
|
|
/* QEMU accepts ',,' as an escape for a literal comma;
|
|
|
|
* skip past those here while searching for the end of the
|
|
|
|
* value, then strip them down below */
|
|
|
|
endmark = strchr(endmark, ',');
|
|
|
|
} while (endmark && endmark[1] == ',' && (endmark += 2));
|
|
|
|
if (!endmark)
|
|
|
|
endmark = end;
|
|
|
|
if (!(separator = strchr(start, '=')))
|
|
|
|
separator = end;
|
|
|
|
|
2020-05-13 20:54:24 +02:00
|
|
|
if (separator >= endmark)
|
2019-06-17 14:18:51 +02:00
|
|
|
separator = endmark;
|
|
|
|
|
2019-10-24 19:41:34 +02:00
|
|
|
keyword = g_strndup(start, separator - start);
|
2019-06-17 14:18:51 +02:00
|
|
|
|
|
|
|
if (separator < endmark) {
|
|
|
|
separator++;
|
2019-10-24 19:41:34 +02:00
|
|
|
value = g_strndup(separator, endmark - separator);
|
2019-06-17 14:18:51 +02:00
|
|
|
if (strchr(value, ',')) {
|
|
|
|
char *p = strchr(value, ',') + 1;
|
|
|
|
char *q = p + 1;
|
|
|
|
while (*q) {
|
|
|
|
if (*q == ',')
|
|
|
|
q++;
|
|
|
|
*p++ = *q++;
|
|
|
|
}
|
|
|
|
*p = '\0';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (keywordAlloc == keywordCount) {
|
2021-03-20 00:37:05 +01:00
|
|
|
VIR_REALLOC_N(keywords, keywordAlloc + 10);
|
|
|
|
VIR_REALLOC_N(values, keywordAlloc + 10);
|
2019-06-17 14:18:51 +02:00
|
|
|
keywordAlloc += 10;
|
|
|
|
}
|
|
|
|
|
|
|
|
keywords[keywordCount] = keyword;
|
|
|
|
values[keywordCount] = value;
|
|
|
|
keywordCount++;
|
|
|
|
|
|
|
|
start = endmark < end ? endmark + 1 : NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
*retkeywords = keywords;
|
|
|
|
*retvalues = values;
|
|
|
|
*retnkeywords = keywordCount;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static virJSONValue *
|
2010-04-15 14:52:03 +01:00
|
|
|
qemuMonitorJSONKeywordStringToJSON(const char *str, const char *firstkeyword)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *ret = virJSONValueNewObject();
|
2010-04-15 14:52:03 +01:00
|
|
|
char **keywords = NULL;
|
|
|
|
char **values = NULL;
|
|
|
|
int nkeywords = 0;
|
Convert 'int i' to 'size_t i' in src/qemu files
Convert the type of loop iterators named 'i', 'j', k',
'ii', 'jj', 'kk', to be 'size_t' instead of 'int' or
'unsigned int', also santizing 'ii', 'jj', 'kk' to use
the normal 'i', 'j', 'k' naming
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2013-07-08 15:09:33 +01:00
|
|
|
size_t i;
|
2010-04-15 14:52:03 +01:00
|
|
|
|
2020-05-13 20:54:24 +02:00
|
|
|
if (qemuMonitorJSONParseKeywords(str, &keywords, &values, &nkeywords) < 0)
|
2010-04-15 14:52:03 +01:00
|
|
|
goto error;
|
|
|
|
|
2013-05-21 15:21:20 +08:00
|
|
|
for (i = 0; i < nkeywords; i++) {
|
2010-04-15 14:52:03 +01:00
|
|
|
if (values[i] == NULL) {
|
|
|
|
if (i != 0) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("unexpected empty keyword in %s"), str);
|
2010-04-15 14:52:03 +01:00
|
|
|
goto error;
|
|
|
|
} else {
|
|
|
|
/* This 3rd arg isn't a typo - the way the parser works is
|
|
|
|
* that the value ended up in the keyword field */
|
|
|
|
if (virJSONValueObjectAppendString(ret, firstkeyword, keywords[i]) < 0)
|
2013-07-04 12:14:12 +02:00
|
|
|
goto error;
|
2010-04-15 14:52:03 +01:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (virJSONValueObjectAppendString(ret, keywords[i], values[i]) < 0)
|
2013-07-04 12:14:12 +02:00
|
|
|
goto error;
|
2010-04-15 14:52:03 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-17 14:18:51 +02:00
|
|
|
qemuMonitorJSONParseKeywordsFree(nkeywords, keywords, values);
|
2010-04-15 14:52:03 +01:00
|
|
|
return ret;
|
|
|
|
|
2014-03-25 07:49:44 +01:00
|
|
|
error:
|
2019-06-17 14:18:51 +02:00
|
|
|
qemuMonitorJSONParseKeywordsFree(nkeywords, keywords, values);
|
2010-04-15 14:52:03 +01:00
|
|
|
virJSONValueFree(ret);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static void qemuMonitorJSONHandleShutdown(qemuMonitor *mon, virJSONValue *data)
|
2009-11-26 13:06:24 +00:00
|
|
|
{
|
2017-04-12 12:00:37 +02:00
|
|
|
bool guest = false;
|
|
|
|
virTristateBool guest_initiated = VIR_TRISTATE_BOOL_ABSENT;
|
|
|
|
|
2017-05-30 10:28:18 +02:00
|
|
|
if (data && virJSONValueObjectGetBoolean(data, "guest", &guest) == 0)
|
2017-04-12 12:00:37 +02:00
|
|
|
guest_initiated = guest ? VIR_TRISTATE_BOOL_YES : VIR_TRISTATE_BOOL_NO;
|
|
|
|
|
|
|
|
qemuMonitorEmitShutdown(mon, guest_initiated);
|
2009-11-26 13:06:24 +00:00
|
|
|
}
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static void qemuMonitorJSONHandleReset(qemuMonitor *mon, virJSONValue *data G_GNUC_UNUSED)
|
2009-11-26 13:06:24 +00:00
|
|
|
{
|
|
|
|
qemuMonitorEmitReset(mon);
|
|
|
|
}
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static void qemuMonitorJSONHandlePowerdown(qemuMonitor *mon, virJSONValue *data G_GNUC_UNUSED)
|
2009-11-26 13:06:24 +00:00
|
|
|
{
|
|
|
|
qemuMonitorEmitPowerdown(mon);
|
|
|
|
}
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static void qemuMonitorJSONHandleStop(qemuMonitor *mon, virJSONValue *data G_GNUC_UNUSED)
|
2009-11-26 13:06:24 +00:00
|
|
|
{
|
|
|
|
qemuMonitorEmitStop(mon);
|
|
|
|
}
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static void qemuMonitorJSONHandleResume(qemuMonitor *mon, virJSONValue *data G_GNUC_UNUSED)
|
2013-01-07 16:25:01 -05:00
|
|
|
{
|
|
|
|
qemuMonitorEmitResume(mon);
|
|
|
|
}
|
|
|
|
|
2017-03-20 14:35:33 +01:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static qemuMonitorEventPanicInfo *
|
|
|
|
qemuMonitorJSONGuestPanicExtractInfoHyperv(virJSONValue *data)
|
2013-06-07 18:23:34 +08:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorEventPanicInfo *ret;
|
2017-03-20 14:35:33 +01:00
|
|
|
|
2020-10-05 12:26:34 +02:00
|
|
|
ret = g_new0(qemuMonitorEventPanicInfo, 1);
|
2017-03-20 14:35:33 +01:00
|
|
|
|
|
|
|
ret->type = QEMU_MONITOR_EVENT_PANIC_INFO_TYPE_HYPERV;
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberUlong(data, "arg1", &ret->data.hyperv.arg1) < 0 ||
|
|
|
|
virJSONValueObjectGetNumberUlong(data, "arg2", &ret->data.hyperv.arg2) < 0 ||
|
|
|
|
virJSONValueObjectGetNumberUlong(data, "arg3", &ret->data.hyperv.arg3) < 0 ||
|
|
|
|
virJSONValueObjectGetNumberUlong(data, "arg4", &ret->data.hyperv.arg4) < 0 ||
|
|
|
|
virJSONValueObjectGetNumberUlong(data, "arg5", &ret->data.hyperv.arg5) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("malformed hyperv panic data"));
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
error:
|
|
|
|
qemuMonitorEventPanicInfoFree(ret);
|
|
|
|
return NULL;
|
2013-06-07 18:23:34 +08:00
|
|
|
}
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static qemuMonitorEventPanicInfo *
|
|
|
|
qemuMonitorJSONGuestPanicExtractInfoS390(virJSONValue *data)
|
qemu: log the crash information for S390
Since QEMU 2.12 commit id '4ada99ade' guest crash information for
S390 is available in the QEMU monitor, e.g.:
{
"timestamp": {
"seconds": 1518004739,
"microseconds": 552563
},
"event": "GUEST_PANICKED",
"data": {
"action": "pause",
"info": {
"core": 0,
"psw-addr": 1102832,
"reason": "disabled-wait",
"psw-mask": 562956395872256,
"type": "s390"
}
}
}
Let's log this information into the domain log file, e.g.:
2018-02-08 13:11:26.075+0000: panic s390: core='0' psw-mask='0x0002000180000000' psw-addr='0x000000000010f146' reason='disabled-wait'
Reviewed-by: Boris Fiuczynski <fiuczy@linux.vnet.ibm.com>
Signed-off-by: Bjoern Walk <bwalk@linux.vnet.ibm.com>
Reviewed-by: John Ferlan <jferlan@redhat.com>
2018-02-27 10:32:56 +01:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorEventPanicInfo *ret;
|
qemu: log the crash information for S390
Since QEMU 2.12 commit id '4ada99ade' guest crash information for
S390 is available in the QEMU monitor, e.g.:
{
"timestamp": {
"seconds": 1518004739,
"microseconds": 552563
},
"event": "GUEST_PANICKED",
"data": {
"action": "pause",
"info": {
"core": 0,
"psw-addr": 1102832,
"reason": "disabled-wait",
"psw-mask": 562956395872256,
"type": "s390"
}
}
}
Let's log this information into the domain log file, e.g.:
2018-02-08 13:11:26.075+0000: panic s390: core='0' psw-mask='0x0002000180000000' psw-addr='0x000000000010f146' reason='disabled-wait'
Reviewed-by: Boris Fiuczynski <fiuczy@linux.vnet.ibm.com>
Signed-off-by: Bjoern Walk <bwalk@linux.vnet.ibm.com>
Reviewed-by: John Ferlan <jferlan@redhat.com>
2018-02-27 10:32:56 +01:00
|
|
|
int core;
|
|
|
|
unsigned long long psw_mask, psw_addr;
|
|
|
|
const char *reason = NULL;
|
|
|
|
|
2020-10-05 12:26:34 +02:00
|
|
|
ret = g_new0(qemuMonitorEventPanicInfo, 1);
|
qemu: log the crash information for S390
Since QEMU 2.12 commit id '4ada99ade' guest crash information for
S390 is available in the QEMU monitor, e.g.:
{
"timestamp": {
"seconds": 1518004739,
"microseconds": 552563
},
"event": "GUEST_PANICKED",
"data": {
"action": "pause",
"info": {
"core": 0,
"psw-addr": 1102832,
"reason": "disabled-wait",
"psw-mask": 562956395872256,
"type": "s390"
}
}
}
Let's log this information into the domain log file, e.g.:
2018-02-08 13:11:26.075+0000: panic s390: core='0' psw-mask='0x0002000180000000' psw-addr='0x000000000010f146' reason='disabled-wait'
Reviewed-by: Boris Fiuczynski <fiuczy@linux.vnet.ibm.com>
Signed-off-by: Bjoern Walk <bwalk@linux.vnet.ibm.com>
Reviewed-by: John Ferlan <jferlan@redhat.com>
2018-02-27 10:32:56 +01:00
|
|
|
|
|
|
|
ret->type = QEMU_MONITOR_EVENT_PANIC_INFO_TYPE_S390;
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberInt(data, "core", &core) < 0 ||
|
|
|
|
virJSONValueObjectGetNumberUlong(data, "psw-mask", &psw_mask) < 0 ||
|
|
|
|
virJSONValueObjectGetNumberUlong(data, "psw-addr", &psw_addr) < 0 ||
|
|
|
|
!(reason = virJSONValueObjectGetString(data, "reason"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("malformed s390 panic data"));
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret->data.s390.core = core;
|
|
|
|
ret->data.s390.psw_mask = psw_mask;
|
|
|
|
ret->data.s390.psw_addr = psw_addr;
|
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
ret->data.s390.reason = g_strdup(reason);
|
qemu: log the crash information for S390
Since QEMU 2.12 commit id '4ada99ade' guest crash information for
S390 is available in the QEMU monitor, e.g.:
{
"timestamp": {
"seconds": 1518004739,
"microseconds": 552563
},
"event": "GUEST_PANICKED",
"data": {
"action": "pause",
"info": {
"core": 0,
"psw-addr": 1102832,
"reason": "disabled-wait",
"psw-mask": 562956395872256,
"type": "s390"
}
}
}
Let's log this information into the domain log file, e.g.:
2018-02-08 13:11:26.075+0000: panic s390: core='0' psw-mask='0x0002000180000000' psw-addr='0x000000000010f146' reason='disabled-wait'
Reviewed-by: Boris Fiuczynski <fiuczy@linux.vnet.ibm.com>
Signed-off-by: Bjoern Walk <bwalk@linux.vnet.ibm.com>
Reviewed-by: John Ferlan <jferlan@redhat.com>
2018-02-27 10:32:56 +01:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
error:
|
|
|
|
qemuMonitorEventPanicInfoFree(ret);
|
|
|
|
return NULL;
|
|
|
|
}
|
2017-03-20 14:35:33 +01:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static qemuMonitorEventPanicInfo *
|
|
|
|
qemuMonitorJSONGuestPanicExtractInfo(virJSONValue *data)
|
2017-03-20 14:35:33 +01:00
|
|
|
{
|
|
|
|
const char *type = virJSONValueObjectGetString(data, "type");
|
|
|
|
|
|
|
|
if (STREQ_NULLABLE(type, "hyper-v"))
|
|
|
|
return qemuMonitorJSONGuestPanicExtractInfoHyperv(data);
|
qemu: log the crash information for S390
Since QEMU 2.12 commit id '4ada99ade' guest crash information for
S390 is available in the QEMU monitor, e.g.:
{
"timestamp": {
"seconds": 1518004739,
"microseconds": 552563
},
"event": "GUEST_PANICKED",
"data": {
"action": "pause",
"info": {
"core": 0,
"psw-addr": 1102832,
"reason": "disabled-wait",
"psw-mask": 562956395872256,
"type": "s390"
}
}
}
Let's log this information into the domain log file, e.g.:
2018-02-08 13:11:26.075+0000: panic s390: core='0' psw-mask='0x0002000180000000' psw-addr='0x000000000010f146' reason='disabled-wait'
Reviewed-by: Boris Fiuczynski <fiuczy@linux.vnet.ibm.com>
Signed-off-by: Bjoern Walk <bwalk@linux.vnet.ibm.com>
Reviewed-by: John Ferlan <jferlan@redhat.com>
2018-02-27 10:32:56 +01:00
|
|
|
else if (STREQ_NULLABLE(type, "s390"))
|
|
|
|
return qemuMonitorJSONGuestPanicExtractInfoS390(data);
|
2017-03-20 14:35:33 +01:00
|
|
|
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("unknown panic info type '%s'"), NULLSTR(type));
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHandleGuestPanic(qemuMonitor *mon,
|
|
|
|
virJSONValue *data)
|
2017-03-20 14:35:33 +01:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *infojson = virJSONValueObjectGetObject(data, "info");
|
|
|
|
qemuMonitorEventPanicInfo *info = NULL;
|
2017-03-20 14:35:33 +01:00
|
|
|
|
|
|
|
if (infojson)
|
|
|
|
info = qemuMonitorJSONGuestPanicExtractInfo(infojson);
|
|
|
|
|
|
|
|
qemuMonitorEmitGuestPanic(mon, info);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static void qemuMonitorJSONHandleRTCChange(qemuMonitor *mon, virJSONValue *data)
|
2010-03-18 18:28:15 +00:00
|
|
|
{
|
|
|
|
long long offset = 0;
|
|
|
|
if (virJSONValueObjectGetNumberLong(data, "offset", &offset) < 0) {
|
2011-05-09 17:24:09 +08:00
|
|
|
VIR_WARN("missing offset in RTC change event");
|
2010-03-18 18:28:15 +00:00
|
|
|
offset = 0;
|
|
|
|
}
|
|
|
|
qemuMonitorEmitRTCChange(mon, offset);
|
|
|
|
}
|
|
|
|
|
2019-01-20 11:04:56 -05:00
|
|
|
VIR_ENUM_DECL(qemuMonitorWatchdogAction);
|
2019-03-16 14:20:32 -04:00
|
|
|
VIR_ENUM_IMPL(qemuMonitorWatchdogAction,
|
|
|
|
VIR_DOMAIN_EVENT_WATCHDOG_LAST,
|
2019-01-20 11:30:15 -05:00
|
|
|
"none", "pause", "reset", "poweroff", "shutdown", "debug", "inject-nmi",
|
|
|
|
);
|
Add support for an explicit watchdog event
This introduces a new event type
VIR_DOMAIN_EVENT_ID_WATCHDOG
This event includes the action that is about to be taken
as a result of the watchdog triggering
typedef enum {
VIR_DOMAIN_EVENT_WATCHDOG_NONE = 0,
VIR_DOMAIN_EVENT_WATCHDOG_PAUSE,
VIR_DOMAIN_EVENT_WATCHDOG_RESET,
VIR_DOMAIN_EVENT_WATCHDOG_POWEROFF,
VIR_DOMAIN_EVENT_WATCHDOG_SHUTDOWN,
VIR_DOMAIN_EVENT_WATCHDOG_DEBUG,
} virDomainEventWatchdogAction;
Thus there is a new callback definition for this event type
typedef void (*virConnectDomainEventWatchdogCallback)(virConnectPtr conn,
virDomainPtr dom,
int action,
void *opaque);
* daemon/remote.c: Dispatch watchdog events to client
* examples/domain-events/events-c/event-test.c: Watch for
watchdog events
* include/libvirt/libvirt.h.in: Define new watchdg event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle watchdog events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for watchdogs and emit a libvirt watchdog event
* src/remote/remote_driver.c: Receive and dispatch watchdog
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
watchdog events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for WATCHDOG event
from QEMU monitor
2010-03-18 19:07:48 +00:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static void qemuMonitorJSONHandleWatchdog(qemuMonitor *mon, virJSONValue *data)
|
Add support for an explicit watchdog event
This introduces a new event type
VIR_DOMAIN_EVENT_ID_WATCHDOG
This event includes the action that is about to be taken
as a result of the watchdog triggering
typedef enum {
VIR_DOMAIN_EVENT_WATCHDOG_NONE = 0,
VIR_DOMAIN_EVENT_WATCHDOG_PAUSE,
VIR_DOMAIN_EVENT_WATCHDOG_RESET,
VIR_DOMAIN_EVENT_WATCHDOG_POWEROFF,
VIR_DOMAIN_EVENT_WATCHDOG_SHUTDOWN,
VIR_DOMAIN_EVENT_WATCHDOG_DEBUG,
} virDomainEventWatchdogAction;
Thus there is a new callback definition for this event type
typedef void (*virConnectDomainEventWatchdogCallback)(virConnectPtr conn,
virDomainPtr dom,
int action,
void *opaque);
* daemon/remote.c: Dispatch watchdog events to client
* examples/domain-events/events-c/event-test.c: Watch for
watchdog events
* include/libvirt/libvirt.h.in: Define new watchdg event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle watchdog events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for watchdogs and emit a libvirt watchdog event
* src/remote/remote_driver.c: Receive and dispatch watchdog
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
watchdog events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for WATCHDOG event
from QEMU monitor
2010-03-18 19:07:48 +00:00
|
|
|
{
|
|
|
|
const char *action;
|
|
|
|
int actionID;
|
2014-11-13 15:25:30 +01:00
|
|
|
if (!(action = virJSONValueObjectGetString(data, "action")))
|
2011-05-09 17:24:09 +08:00
|
|
|
VIR_WARN("missing action in watchdog event");
|
Add support for an explicit watchdog event
This introduces a new event type
VIR_DOMAIN_EVENT_ID_WATCHDOG
This event includes the action that is about to be taken
as a result of the watchdog triggering
typedef enum {
VIR_DOMAIN_EVENT_WATCHDOG_NONE = 0,
VIR_DOMAIN_EVENT_WATCHDOG_PAUSE,
VIR_DOMAIN_EVENT_WATCHDOG_RESET,
VIR_DOMAIN_EVENT_WATCHDOG_POWEROFF,
VIR_DOMAIN_EVENT_WATCHDOG_SHUTDOWN,
VIR_DOMAIN_EVENT_WATCHDOG_DEBUG,
} virDomainEventWatchdogAction;
Thus there is a new callback definition for this event type
typedef void (*virConnectDomainEventWatchdogCallback)(virConnectPtr conn,
virDomainPtr dom,
int action,
void *opaque);
* daemon/remote.c: Dispatch watchdog events to client
* examples/domain-events/events-c/event-test.c: Watch for
watchdog events
* include/libvirt/libvirt.h.in: Define new watchdg event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle watchdog events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for watchdogs and emit a libvirt watchdog event
* src/remote/remote_driver.c: Receive and dispatch watchdog
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
watchdog events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for WATCHDOG event
from QEMU monitor
2010-03-18 19:07:48 +00:00
|
|
|
if (action) {
|
|
|
|
if ((actionID = qemuMonitorWatchdogActionTypeFromString(action)) < 0) {
|
|
|
|
VIR_WARN("unknown action %s in watchdog event", action);
|
|
|
|
actionID = VIR_DOMAIN_EVENT_WATCHDOG_NONE;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
actionID = VIR_DOMAIN_EVENT_WATCHDOG_NONE;
|
|
|
|
}
|
2010-04-08 16:01:00 +01:00
|
|
|
qemuMonitorEmitWatchdog(mon, actionID);
|
Add support for an explicit watchdog event
This introduces a new event type
VIR_DOMAIN_EVENT_ID_WATCHDOG
This event includes the action that is about to be taken
as a result of the watchdog triggering
typedef enum {
VIR_DOMAIN_EVENT_WATCHDOG_NONE = 0,
VIR_DOMAIN_EVENT_WATCHDOG_PAUSE,
VIR_DOMAIN_EVENT_WATCHDOG_RESET,
VIR_DOMAIN_EVENT_WATCHDOG_POWEROFF,
VIR_DOMAIN_EVENT_WATCHDOG_SHUTDOWN,
VIR_DOMAIN_EVENT_WATCHDOG_DEBUG,
} virDomainEventWatchdogAction;
Thus there is a new callback definition for this event type
typedef void (*virConnectDomainEventWatchdogCallback)(virConnectPtr conn,
virDomainPtr dom,
int action,
void *opaque);
* daemon/remote.c: Dispatch watchdog events to client
* examples/domain-events/events-c/event-test.c: Watch for
watchdog events
* include/libvirt/libvirt.h.in: Define new watchdg event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle watchdog events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for watchdogs and emit a libvirt watchdog event
* src/remote/remote_driver.c: Receive and dispatch watchdog
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
watchdog events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for WATCHDOG event
from QEMU monitor
2010-03-18 19:07:48 +00:00
|
|
|
}
|
|
|
|
|
2019-01-20 11:04:56 -05:00
|
|
|
VIR_ENUM_DECL(qemuMonitorIOErrorAction);
|
2019-03-16 14:20:32 -04:00
|
|
|
VIR_ENUM_IMPL(qemuMonitorIOErrorAction,
|
|
|
|
VIR_DOMAIN_EVENT_IO_ERROR_LAST,
|
2019-01-20 11:30:15 -05:00
|
|
|
"ignore", "stop", "report",
|
|
|
|
);
|
Add support for an explicit IO error event
This introduces a new event type
VIR_DOMAIN_EVENT_ID_IO_ERROR
This event includes the action that is about to be taken
as a result of the watchdog triggering
typedef enum {
VIR_DOMAIN_EVENT_IO_ERROR_NONE = 0,
VIR_DOMAIN_EVENT_IO_ERROR_PAUSE,
VIR_DOMAIN_EVENT_IO_ERROR_REPORT,
} virDomainEventIOErrorAction;
In addition it has the source path of the disk that had the
error and its unique device alias. It does not include the
target device name (/dev/sda), since this would preclude
triggering IO errors from other file backed devices (eg
serial ports connected to a file)
Thus there is a new callback definition for this event type
typedef void (*virConnectDomainEventIOErrorCallback)(virConnectPtr conn,
virDomainPtr dom,
const char *srcPath,
const char *devAlias,
int action,
void *opaque);
This is currently wired up to the QEMU block IO error events
* daemon/remote.c: Dispatch IO error events to client
* examples/domain-events/events-c/event-test.c: Watch for
IO error events
* include/libvirt/libvirt.h.in: Define new IO error event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle IO error events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for block IO errors and emit a libvirt IO error event
* src/remote/remote_driver.c: Receive and dispatch IO error
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
IO error events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for BLOCK_IO_ERROR event
from QEMU monitor
2010-03-18 19:37:44 +00:00
|
|
|
|
|
|
|
|
qemu: support nospace reason in io error event
Aeons ago (commit 34dcbbb4, v0.8.2), we added a new libvirt event
(VIR_DOMAIN_EVENT_ID_IO_ERROR_REASON) in order to tell the user WHY
the guest halted. This is because at least VDSM wants to react
differently to ENOSPC events (resize the lvm partition to be larger,
and resume the guest as if nothing had happened) from all other events
(I/O is hosed, throw up our hands and flag things as broken). At the
time this was done, downstream RHEL qemu added a vendor extension
'__com.redhat_reason', which would be exactly one of these strings:
"enospc", "eperm", "eio", and "eother". In our stupidity, we exposed
those exact strings to clients, rather than an enum, and we also
return "" if we did not have access to a reason (which was the case
for upstream qemu).
Fast forward to now: upstream qemu commit c7c2ff0c (will be qemu 2.2)
FINALLY adds a 'nospace' boolean, after discussion with multiple
projects determined that VDSM really doesn't care about distinction
between any other error types. So this patch converts 'nospace' into
the string "enospc" for compatibility with RHEL clients that were
already used to the downstream extension, while leaving the reason
blank for all other cases (no change from the status quo).
See also https://bugzilla.redhat.com/show_bug.cgi?id=1119784
* src/qemu/qemu_monitor_json.c (qewmuMonitorJSONHandleIOError):
Parse reason field from modern qemu.
* include/libvirt/libvirt.h.in
(virConnectDomainEventIOErrorReasonCallback): Document it.
Signed-off-by: Eric Blake <eblake@redhat.com>
2014-10-03 08:46:25 -06:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHandleIOError(qemuMonitor *mon, virJSONValue *data)
|
Add support for an explicit IO error event
This introduces a new event type
VIR_DOMAIN_EVENT_ID_IO_ERROR
This event includes the action that is about to be taken
as a result of the watchdog triggering
typedef enum {
VIR_DOMAIN_EVENT_IO_ERROR_NONE = 0,
VIR_DOMAIN_EVENT_IO_ERROR_PAUSE,
VIR_DOMAIN_EVENT_IO_ERROR_REPORT,
} virDomainEventIOErrorAction;
In addition it has the source path of the disk that had the
error and its unique device alias. It does not include the
target device name (/dev/sda), since this would preclude
triggering IO errors from other file backed devices (eg
serial ports connected to a file)
Thus there is a new callback definition for this event type
typedef void (*virConnectDomainEventIOErrorCallback)(virConnectPtr conn,
virDomainPtr dom,
const char *srcPath,
const char *devAlias,
int action,
void *opaque);
This is currently wired up to the QEMU block IO error events
* daemon/remote.c: Dispatch IO error events to client
* examples/domain-events/events-c/event-test.c: Watch for
IO error events
* include/libvirt/libvirt.h.in: Define new IO error event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle IO error events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for block IO errors and emit a libvirt IO error event
* src/remote/remote_driver.c: Receive and dispatch IO error
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
IO error events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for BLOCK_IO_ERROR event
from QEMU monitor
2010-03-18 19:37:44 +00:00
|
|
|
{
|
|
|
|
const char *device;
|
2018-08-13 17:02:38 +02:00
|
|
|
const char *nodename;
|
Add support for an explicit IO error event
This introduces a new event type
VIR_DOMAIN_EVENT_ID_IO_ERROR
This event includes the action that is about to be taken
as a result of the watchdog triggering
typedef enum {
VIR_DOMAIN_EVENT_IO_ERROR_NONE = 0,
VIR_DOMAIN_EVENT_IO_ERROR_PAUSE,
VIR_DOMAIN_EVENT_IO_ERROR_REPORT,
} virDomainEventIOErrorAction;
In addition it has the source path of the disk that had the
error and its unique device alias. It does not include the
target device name (/dev/sda), since this would preclude
triggering IO errors from other file backed devices (eg
serial ports connected to a file)
Thus there is a new callback definition for this event type
typedef void (*virConnectDomainEventIOErrorCallback)(virConnectPtr conn,
virDomainPtr dom,
const char *srcPath,
const char *devAlias,
int action,
void *opaque);
This is currently wired up to the QEMU block IO error events
* daemon/remote.c: Dispatch IO error events to client
* examples/domain-events/events-c/event-test.c: Watch for
IO error events
* include/libvirt/libvirt.h.in: Define new IO error event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle IO error events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for block IO errors and emit a libvirt IO error event
* src/remote/remote_driver.c: Receive and dispatch IO error
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
IO error events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for BLOCK_IO_ERROR event
from QEMU monitor
2010-03-18 19:37:44 +00:00
|
|
|
const char *action;
|
qemu: support nospace reason in io error event
Aeons ago (commit 34dcbbb4, v0.8.2), we added a new libvirt event
(VIR_DOMAIN_EVENT_ID_IO_ERROR_REASON) in order to tell the user WHY
the guest halted. This is because at least VDSM wants to react
differently to ENOSPC events (resize the lvm partition to be larger,
and resume the guest as if nothing had happened) from all other events
(I/O is hosed, throw up our hands and flag things as broken). At the
time this was done, downstream RHEL qemu added a vendor extension
'__com.redhat_reason', which would be exactly one of these strings:
"enospc", "eperm", "eio", and "eother". In our stupidity, we exposed
those exact strings to clients, rather than an enum, and we also
return "" if we did not have access to a reason (which was the case
for upstream qemu).
Fast forward to now: upstream qemu commit c7c2ff0c (will be qemu 2.2)
FINALLY adds a 'nospace' boolean, after discussion with multiple
projects determined that VDSM really doesn't care about distinction
between any other error types. So this patch converts 'nospace' into
the string "enospc" for compatibility with RHEL clients that were
already used to the downstream extension, while leaving the reason
blank for all other cases (no change from the status quo).
See also https://bugzilla.redhat.com/show_bug.cgi?id=1119784
* src/qemu/qemu_monitor_json.c (qewmuMonitorJSONHandleIOError):
Parse reason field from modern qemu.
* include/libvirt/libvirt.h.in
(virConnectDomainEventIOErrorReasonCallback): Document it.
Signed-off-by: Eric Blake <eblake@redhat.com>
2014-10-03 08:46:25 -06:00
|
|
|
const char *reason = "";
|
|
|
|
bool nospc = false;
|
Add support for an explicit IO error event
This introduces a new event type
VIR_DOMAIN_EVENT_ID_IO_ERROR
This event includes the action that is about to be taken
as a result of the watchdog triggering
typedef enum {
VIR_DOMAIN_EVENT_IO_ERROR_NONE = 0,
VIR_DOMAIN_EVENT_IO_ERROR_PAUSE,
VIR_DOMAIN_EVENT_IO_ERROR_REPORT,
} virDomainEventIOErrorAction;
In addition it has the source path of the disk that had the
error and its unique device alias. It does not include the
target device name (/dev/sda), since this would preclude
triggering IO errors from other file backed devices (eg
serial ports connected to a file)
Thus there is a new callback definition for this event type
typedef void (*virConnectDomainEventIOErrorCallback)(virConnectPtr conn,
virDomainPtr dom,
const char *srcPath,
const char *devAlias,
int action,
void *opaque);
This is currently wired up to the QEMU block IO error events
* daemon/remote.c: Dispatch IO error events to client
* examples/domain-events/events-c/event-test.c: Watch for
IO error events
* include/libvirt/libvirt.h.in: Define new IO error event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle IO error events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for block IO errors and emit a libvirt IO error event
* src/remote/remote_driver.c: Receive and dispatch IO error
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
IO error events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for BLOCK_IO_ERROR event
from QEMU monitor
2010-03-18 19:37:44 +00:00
|
|
|
int actionID;
|
|
|
|
|
|
|
|
/* Throughout here we try our best to carry on upon errors,
|
2018-12-04 19:08:14 +02:00
|
|
|
since it's important to get as much info as possible out
|
Add support for an explicit IO error event
This introduces a new event type
VIR_DOMAIN_EVENT_ID_IO_ERROR
This event includes the action that is about to be taken
as a result of the watchdog triggering
typedef enum {
VIR_DOMAIN_EVENT_IO_ERROR_NONE = 0,
VIR_DOMAIN_EVENT_IO_ERROR_PAUSE,
VIR_DOMAIN_EVENT_IO_ERROR_REPORT,
} virDomainEventIOErrorAction;
In addition it has the source path of the disk that had the
error and its unique device alias. It does not include the
target device name (/dev/sda), since this would preclude
triggering IO errors from other file backed devices (eg
serial ports connected to a file)
Thus there is a new callback definition for this event type
typedef void (*virConnectDomainEventIOErrorCallback)(virConnectPtr conn,
virDomainPtr dom,
const char *srcPath,
const char *devAlias,
int action,
void *opaque);
This is currently wired up to the QEMU block IO error events
* daemon/remote.c: Dispatch IO error events to client
* examples/domain-events/events-c/event-test.c: Watch for
IO error events
* include/libvirt/libvirt.h.in: Define new IO error event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle IO error events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for block IO errors and emit a libvirt IO error event
* src/remote/remote_driver.c: Receive and dispatch IO error
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
IO error events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for BLOCK_IO_ERROR event
from QEMU monitor
2010-03-18 19:37:44 +00:00
|
|
|
to the application */
|
|
|
|
|
|
|
|
if ((action = virJSONValueObjectGetString(data, "action")) == NULL) {
|
2011-05-09 17:24:09 +08:00
|
|
|
VIR_WARN("Missing action in disk io error event");
|
Add support for an explicit IO error event
This introduces a new event type
VIR_DOMAIN_EVENT_ID_IO_ERROR
This event includes the action that is about to be taken
as a result of the watchdog triggering
typedef enum {
VIR_DOMAIN_EVENT_IO_ERROR_NONE = 0,
VIR_DOMAIN_EVENT_IO_ERROR_PAUSE,
VIR_DOMAIN_EVENT_IO_ERROR_REPORT,
} virDomainEventIOErrorAction;
In addition it has the source path of the disk that had the
error and its unique device alias. It does not include the
target device name (/dev/sda), since this would preclude
triggering IO errors from other file backed devices (eg
serial ports connected to a file)
Thus there is a new callback definition for this event type
typedef void (*virConnectDomainEventIOErrorCallback)(virConnectPtr conn,
virDomainPtr dom,
const char *srcPath,
const char *devAlias,
int action,
void *opaque);
This is currently wired up to the QEMU block IO error events
* daemon/remote.c: Dispatch IO error events to client
* examples/domain-events/events-c/event-test.c: Watch for
IO error events
* include/libvirt/libvirt.h.in: Define new IO error event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle IO error events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for block IO errors and emit a libvirt IO error event
* src/remote/remote_driver.c: Receive and dispatch IO error
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
IO error events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for BLOCK_IO_ERROR event
from QEMU monitor
2010-03-18 19:37:44 +00:00
|
|
|
action = "ignore";
|
|
|
|
}
|
|
|
|
|
2014-11-13 15:25:30 +01:00
|
|
|
if ((device = virJSONValueObjectGetString(data, "device")) == NULL)
|
2011-05-09 17:24:09 +08:00
|
|
|
VIR_WARN("missing device in disk io error event");
|
Add support for an explicit IO error event
This introduces a new event type
VIR_DOMAIN_EVENT_ID_IO_ERROR
This event includes the action that is about to be taken
as a result of the watchdog triggering
typedef enum {
VIR_DOMAIN_EVENT_IO_ERROR_NONE = 0,
VIR_DOMAIN_EVENT_IO_ERROR_PAUSE,
VIR_DOMAIN_EVENT_IO_ERROR_REPORT,
} virDomainEventIOErrorAction;
In addition it has the source path of the disk that had the
error and its unique device alias. It does not include the
target device name (/dev/sda), since this would preclude
triggering IO errors from other file backed devices (eg
serial ports connected to a file)
Thus there is a new callback definition for this event type
typedef void (*virConnectDomainEventIOErrorCallback)(virConnectPtr conn,
virDomainPtr dom,
const char *srcPath,
const char *devAlias,
int action,
void *opaque);
This is currently wired up to the QEMU block IO error events
* daemon/remote.c: Dispatch IO error events to client
* examples/domain-events/events-c/event-test.c: Watch for
IO error events
* include/libvirt/libvirt.h.in: Define new IO error event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle IO error events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for block IO errors and emit a libvirt IO error event
* src/remote/remote_driver.c: Receive and dispatch IO error
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
IO error events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for BLOCK_IO_ERROR event
from QEMU monitor
2010-03-18 19:37:44 +00:00
|
|
|
|
2018-08-13 17:02:38 +02:00
|
|
|
nodename = virJSONValueObjectGetString(data, "node-name");
|
|
|
|
|
qemu: support nospace reason in io error event
Aeons ago (commit 34dcbbb4, v0.8.2), we added a new libvirt event
(VIR_DOMAIN_EVENT_ID_IO_ERROR_REASON) in order to tell the user WHY
the guest halted. This is because at least VDSM wants to react
differently to ENOSPC events (resize the lvm partition to be larger,
and resume the guest as if nothing had happened) from all other events
(I/O is hosed, throw up our hands and flag things as broken). At the
time this was done, downstream RHEL qemu added a vendor extension
'__com.redhat_reason', which would be exactly one of these strings:
"enospc", "eperm", "eio", and "eother". In our stupidity, we exposed
those exact strings to clients, rather than an enum, and we also
return "" if we did not have access to a reason (which was the case
for upstream qemu).
Fast forward to now: upstream qemu commit c7c2ff0c (will be qemu 2.2)
FINALLY adds a 'nospace' boolean, after discussion with multiple
projects determined that VDSM really doesn't care about distinction
between any other error types. So this patch converts 'nospace' into
the string "enospc" for compatibility with RHEL clients that were
already used to the downstream extension, while leaving the reason
blank for all other cases (no change from the status quo).
See also https://bugzilla.redhat.com/show_bug.cgi?id=1119784
* src/qemu/qemu_monitor_json.c (qewmuMonitorJSONHandleIOError):
Parse reason field from modern qemu.
* include/libvirt/libvirt.h.in
(virConnectDomainEventIOErrorReasonCallback): Document it.
Signed-off-by: Eric Blake <eblake@redhat.com>
2014-10-03 08:46:25 -06:00
|
|
|
if (virJSONValueObjectGetBoolean(data, "nospace", &nospc) == 0 && nospc)
|
|
|
|
reason = "enospc";
|
Add support for another explicit IO error event
This introduces a new event type
VIR_DOMAIN_EVENT_ID_IO_ERROR_REASON
This event is the same as the previous VIR_DOMAIN_ID_IO_ERROR
event, but also includes a string describing the cause of
the event.
Thus there is a new callback definition for this event type
typedef void (*virConnectDomainEventIOErrorReasonCallback)(virConnectPtr conn,
virDomainPtr dom,
const char *srcPath,
const char *devAlias,
int action,
const char *reason,
void *opaque);
This is currently wired up to the QEMU block IO error events
* daemon/remote.c: Dispatch IO error events to client
* examples/domain-events/events-c/event-test.c: Watch for
IO error events
* include/libvirt/libvirt.h.in: Define new IO error event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle IO error events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for block IO errors and emit a libvirt IO error event
* src/remote/remote_driver.c: Receive and dispatch IO error
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
IO error events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for BLOCK_IO_ERROR event
from QEMU monitor
2010-03-18 19:37:44 +00:00
|
|
|
|
Add support for an explicit IO error event
This introduces a new event type
VIR_DOMAIN_EVENT_ID_IO_ERROR
This event includes the action that is about to be taken
as a result of the watchdog triggering
typedef enum {
VIR_DOMAIN_EVENT_IO_ERROR_NONE = 0,
VIR_DOMAIN_EVENT_IO_ERROR_PAUSE,
VIR_DOMAIN_EVENT_IO_ERROR_REPORT,
} virDomainEventIOErrorAction;
In addition it has the source path of the disk that had the
error and its unique device alias. It does not include the
target device name (/dev/sda), since this would preclude
triggering IO errors from other file backed devices (eg
serial ports connected to a file)
Thus there is a new callback definition for this event type
typedef void (*virConnectDomainEventIOErrorCallback)(virConnectPtr conn,
virDomainPtr dom,
const char *srcPath,
const char *devAlias,
int action,
void *opaque);
This is currently wired up to the QEMU block IO error events
* daemon/remote.c: Dispatch IO error events to client
* examples/domain-events/events-c/event-test.c: Watch for
IO error events
* include/libvirt/libvirt.h.in: Define new IO error event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle IO error events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for block IO errors and emit a libvirt IO error event
* src/remote/remote_driver.c: Receive and dispatch IO error
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
IO error events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for BLOCK_IO_ERROR event
from QEMU monitor
2010-03-18 19:37:44 +00:00
|
|
|
if ((actionID = qemuMonitorIOErrorActionTypeFromString(action)) < 0) {
|
|
|
|
VIR_WARN("unknown disk io error action '%s'", action);
|
|
|
|
actionID = VIR_DOMAIN_EVENT_IO_ERROR_NONE;
|
|
|
|
}
|
|
|
|
|
2018-08-13 17:02:38 +02:00
|
|
|
qemuMonitorEmitIOError(mon, device, nodename, actionID, reason);
|
Add support for an explicit IO error event
This introduces a new event type
VIR_DOMAIN_EVENT_ID_IO_ERROR
This event includes the action that is about to be taken
as a result of the watchdog triggering
typedef enum {
VIR_DOMAIN_EVENT_IO_ERROR_NONE = 0,
VIR_DOMAIN_EVENT_IO_ERROR_PAUSE,
VIR_DOMAIN_EVENT_IO_ERROR_REPORT,
} virDomainEventIOErrorAction;
In addition it has the source path of the disk that had the
error and its unique device alias. It does not include the
target device name (/dev/sda), since this would preclude
triggering IO errors from other file backed devices (eg
serial ports connected to a file)
Thus there is a new callback definition for this event type
typedef void (*virConnectDomainEventIOErrorCallback)(virConnectPtr conn,
virDomainPtr dom,
const char *srcPath,
const char *devAlias,
int action,
void *opaque);
This is currently wired up to the QEMU block IO error events
* daemon/remote.c: Dispatch IO error events to client
* examples/domain-events/events-c/event-test.c: Watch for
IO error events
* include/libvirt/libvirt.h.in: Define new IO error event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle IO error events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for block IO errors and emit a libvirt IO error event
* src/remote/remote_driver.c: Receive and dispatch IO error
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
IO error events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for BLOCK_IO_ERROR event
from QEMU monitor
2010-03-18 19:37:44 +00:00
|
|
|
}
|
|
|
|
|
2009-11-26 13:06:24 +00:00
|
|
|
|
2019-01-20 11:04:56 -05:00
|
|
|
VIR_ENUM_DECL(qemuMonitorGraphicsAddressFamily);
|
2012-01-20 14:00:58 -07:00
|
|
|
VIR_ENUM_IMPL(qemuMonitorGraphicsAddressFamily,
|
|
|
|
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_LAST,
|
2019-01-20 11:30:15 -05:00
|
|
|
"ipv4", "ipv6", "unix",
|
|
|
|
);
|
Add domain events for graphics network clients
This introduces a new event type
VIR_DOMAIN_EVENT_ID_GRAPHICS
The same event can be emitted in 3 scenarios
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_CONNECT = 0,
VIR_DOMAIN_EVENT_GRAPHICS_INITIALIZE,
VIR_DOMAIN_EVENT_GRAPHICS_DISCONNECT,
} virDomainEventGraphicsPhase;
Connect/disconnect are triggered at socket accept/close.
The initialize phase is immediately after the protocol
setup and authentication has completed. ie when the
client is authorized and about to start interacting with
the graphical desktop
This event comes with *a lot* of potential information
- IP address, port & address family of client
- IP address, port & address family of server
- Authentication scheme (arbitrary string)
- Authenticated subject identity. A subject may have
multiple identities with some authentication schemes.
For example, vencrypt+sasl results in a x509dname
and saslUsername identities.
This results in a very complicated callback :-(
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV4,
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV6,
} virDomainEventGraphicsAddressType;
struct _virDomainEventGraphicsAddress {
int family;
const char *node;
const char *service;
};
typedef struct _virDomainEventGraphicsAddress virDomainEventGraphicsAddress;
typedef virDomainEventGraphicsAddress *virDomainEventGraphicsAddressPtr;
struct _virDomainEventGraphicsSubject {
int nidentity;
struct {
const char *type;
const char *name;
} *identities;
};
typedef struct _virDomainEventGraphicsSubject virDomainEventGraphicsSubject;
typedef virDomainEventGraphicsSubject *virDomainEventGraphicsSubjectPtr;
typedef void (*virConnectDomainEventGraphicsCallback)(virConnectPtr conn,
virDomainPtr dom,
int phase,
virDomainEventGraphicsAddressPtr local,
virDomainEventGraphicsAddressPtr remote,
const char *authScheme,
virDomainEventGraphicsSubjectPtr subject,
void *opaque);
The wire protocol is similarly complex
struct remote_domain_event_graphics_address {
int family;
remote_nonnull_string node;
remote_nonnull_string service;
};
const REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX = 20;
struct remote_domain_event_graphics_identity {
remote_nonnull_string type;
remote_nonnull_string name;
};
struct remote_domain_event_graphics_msg {
remote_nonnull_domain dom;
int phase;
remote_domain_event_graphics_address local;
remote_domain_event_graphics_address remote;
remote_nonnull_string authScheme;
remote_domain_event_graphics_identity subject<REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX>;
};
This is currently implemented in QEMU for the VNC graphics
protocol, but designed to be usable with SPICE graphics in
the future too.
* daemon/remote.c: Dispatch graphics events to client
* examples/domain-events/events-c/event-test.c: Watch for
graphics events
* include/libvirt/libvirt.h.in: Define new graphics event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle graphics events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for VNC events and emit a libvirt graphics event
* src/remote/remote_driver.c: Receive and dispatch graphics
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
graphics events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for VNC_CONNECTED,
VNC_INITIALIZED & VNC_DISCONNETED events from QEMU monitor
2010-03-19 13:27:45 +00:00
|
|
|
|
2015-06-29 16:10:51 +02:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHandleGraphicsVNC(qemuMonitor *mon,
|
|
|
|
virJSONValue *data,
|
2015-06-29 16:10:51 +02:00
|
|
|
int phase)
|
Add domain events for graphics network clients
This introduces a new event type
VIR_DOMAIN_EVENT_ID_GRAPHICS
The same event can be emitted in 3 scenarios
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_CONNECT = 0,
VIR_DOMAIN_EVENT_GRAPHICS_INITIALIZE,
VIR_DOMAIN_EVENT_GRAPHICS_DISCONNECT,
} virDomainEventGraphicsPhase;
Connect/disconnect are triggered at socket accept/close.
The initialize phase is immediately after the protocol
setup and authentication has completed. ie when the
client is authorized and about to start interacting with
the graphical desktop
This event comes with *a lot* of potential information
- IP address, port & address family of client
- IP address, port & address family of server
- Authentication scheme (arbitrary string)
- Authenticated subject identity. A subject may have
multiple identities with some authentication schemes.
For example, vencrypt+sasl results in a x509dname
and saslUsername identities.
This results in a very complicated callback :-(
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV4,
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV6,
} virDomainEventGraphicsAddressType;
struct _virDomainEventGraphicsAddress {
int family;
const char *node;
const char *service;
};
typedef struct _virDomainEventGraphicsAddress virDomainEventGraphicsAddress;
typedef virDomainEventGraphicsAddress *virDomainEventGraphicsAddressPtr;
struct _virDomainEventGraphicsSubject {
int nidentity;
struct {
const char *type;
const char *name;
} *identities;
};
typedef struct _virDomainEventGraphicsSubject virDomainEventGraphicsSubject;
typedef virDomainEventGraphicsSubject *virDomainEventGraphicsSubjectPtr;
typedef void (*virConnectDomainEventGraphicsCallback)(virConnectPtr conn,
virDomainPtr dom,
int phase,
virDomainEventGraphicsAddressPtr local,
virDomainEventGraphicsAddressPtr remote,
const char *authScheme,
virDomainEventGraphicsSubjectPtr subject,
void *opaque);
The wire protocol is similarly complex
struct remote_domain_event_graphics_address {
int family;
remote_nonnull_string node;
remote_nonnull_string service;
};
const REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX = 20;
struct remote_domain_event_graphics_identity {
remote_nonnull_string type;
remote_nonnull_string name;
};
struct remote_domain_event_graphics_msg {
remote_nonnull_domain dom;
int phase;
remote_domain_event_graphics_address local;
remote_domain_event_graphics_address remote;
remote_nonnull_string authScheme;
remote_domain_event_graphics_identity subject<REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX>;
};
This is currently implemented in QEMU for the VNC graphics
protocol, but designed to be usable with SPICE graphics in
the future too.
* daemon/remote.c: Dispatch graphics events to client
* examples/domain-events/events-c/event-test.c: Watch for
graphics events
* include/libvirt/libvirt.h.in: Define new graphics event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle graphics events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for VNC events and emit a libvirt graphics event
* src/remote/remote_driver.c: Receive and dispatch graphics
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
graphics events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for VNC_CONNECTED,
VNC_INITIALIZED & VNC_DISCONNETED events from QEMU monitor
2010-03-19 13:27:45 +00:00
|
|
|
{
|
|
|
|
const char *localNode, *localService, *localFamily;
|
|
|
|
const char *remoteNode, *remoteService, *remoteFamily;
|
|
|
|
const char *authScheme, *saslUsername, *x509dname;
|
|
|
|
int localFamilyID, remoteFamilyID;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *client;
|
|
|
|
virJSONValue *server;
|
Add domain events for graphics network clients
This introduces a new event type
VIR_DOMAIN_EVENT_ID_GRAPHICS
The same event can be emitted in 3 scenarios
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_CONNECT = 0,
VIR_DOMAIN_EVENT_GRAPHICS_INITIALIZE,
VIR_DOMAIN_EVENT_GRAPHICS_DISCONNECT,
} virDomainEventGraphicsPhase;
Connect/disconnect are triggered at socket accept/close.
The initialize phase is immediately after the protocol
setup and authentication has completed. ie when the
client is authorized and about to start interacting with
the graphical desktop
This event comes with *a lot* of potential information
- IP address, port & address family of client
- IP address, port & address family of server
- Authentication scheme (arbitrary string)
- Authenticated subject identity. A subject may have
multiple identities with some authentication schemes.
For example, vencrypt+sasl results in a x509dname
and saslUsername identities.
This results in a very complicated callback :-(
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV4,
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV6,
} virDomainEventGraphicsAddressType;
struct _virDomainEventGraphicsAddress {
int family;
const char *node;
const char *service;
};
typedef struct _virDomainEventGraphicsAddress virDomainEventGraphicsAddress;
typedef virDomainEventGraphicsAddress *virDomainEventGraphicsAddressPtr;
struct _virDomainEventGraphicsSubject {
int nidentity;
struct {
const char *type;
const char *name;
} *identities;
};
typedef struct _virDomainEventGraphicsSubject virDomainEventGraphicsSubject;
typedef virDomainEventGraphicsSubject *virDomainEventGraphicsSubjectPtr;
typedef void (*virConnectDomainEventGraphicsCallback)(virConnectPtr conn,
virDomainPtr dom,
int phase,
virDomainEventGraphicsAddressPtr local,
virDomainEventGraphicsAddressPtr remote,
const char *authScheme,
virDomainEventGraphicsSubjectPtr subject,
void *opaque);
The wire protocol is similarly complex
struct remote_domain_event_graphics_address {
int family;
remote_nonnull_string node;
remote_nonnull_string service;
};
const REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX = 20;
struct remote_domain_event_graphics_identity {
remote_nonnull_string type;
remote_nonnull_string name;
};
struct remote_domain_event_graphics_msg {
remote_nonnull_domain dom;
int phase;
remote_domain_event_graphics_address local;
remote_domain_event_graphics_address remote;
remote_nonnull_string authScheme;
remote_domain_event_graphics_identity subject<REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX>;
};
This is currently implemented in QEMU for the VNC graphics
protocol, but designed to be usable with SPICE graphics in
the future too.
* daemon/remote.c: Dispatch graphics events to client
* examples/domain-events/events-c/event-test.c: Watch for
graphics events
* include/libvirt/libvirt.h.in: Define new graphics event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle graphics events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for VNC events and emit a libvirt graphics event
* src/remote/remote_driver.c: Receive and dispatch graphics
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
graphics events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for VNC_CONNECTED,
VNC_INITIALIZED & VNC_DISCONNETED events from QEMU monitor
2010-03-19 13:27:45 +00:00
|
|
|
|
qemu: simplify json parsing
Rather than grabbing an arbitrary JSON value and then checking
if it has the right type, we might as well request the correct
type to begin with.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONIOProcessEvent)
(qemuMonitorJSONCommandWithFd, qemuMonitorJSONHandleGraphics)
(qemuMonitorJSONGetStatus, qemuMonitorJSONExtractCPUInfo)
(qemuMonitorJSONGetVirtType, qemuMonitorJSONGetBalloonInfo)
(qemuMonitorJSONGetMemoryStats)
(qemuMonitorJSONDevGetBlockExtent)
(qemuMonitorJSONGetOneBlockStatsInfo)
(qemuMonitorJSONGetAllBlockStatsInfo)
(qemuMonitorJSONBlockStatsUpdateCapacityOne)
(qemuMonitorJSONBlockStatsUpdateCapacity)
(qemuMonitorJSONGetBlockExtent)
(qemuMonitorJSONGetMigrationStatusReply)
(qemuMonitorJSONGetDumpGuestMemoryCapability)
(qemuMonitorJSONAddFd, qemuMonitorJSONQueryRxFilterParse)
(qemuMonitorJSONExtractChardevInfo)
(qemuMonitorJSONDiskNameLookupOne)
(qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONGetAllBlockJobInfo)
(qemuMonitorJSONBlockIoThrottleInfo, qemuMonitorJSONGetVersion)
(qemuMonitorJSONGetMachines, qemuMonitorJSONGetCPUDefinitions)
(qemuMonitorJSONGetCommands, qemuMonitorJSONGetEvents)
(qemuMonitorJSONGetKVMState, qemuMonitorJSONGetObjectTypes)
(qemuMonitorJSONGetObjectListPaths)
(qemuMonitorJSONGetObjectProps, qemuMonitorJSONGetTargetArch)
(qemuMonitorJSONGetMigrationCapabilities)
(qemuMonitorJSONGetStringArray, qemuMonitorJSONAttachCharDev)
(qemuMonitorJSONGetCPUx86Data, qemuMonitorJSONGetIOThreads)
(qemuMonitorJSONGetMemoryDeviceInfo): Use shorter idioms.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-06-20 13:40:21 -06:00
|
|
|
if (!(client = virJSONValueObjectGetObject(data, "client"))) {
|
2011-05-09 17:24:09 +08:00
|
|
|
VIR_WARN("missing client info in VNC event");
|
Add domain events for graphics network clients
This introduces a new event type
VIR_DOMAIN_EVENT_ID_GRAPHICS
The same event can be emitted in 3 scenarios
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_CONNECT = 0,
VIR_DOMAIN_EVENT_GRAPHICS_INITIALIZE,
VIR_DOMAIN_EVENT_GRAPHICS_DISCONNECT,
} virDomainEventGraphicsPhase;
Connect/disconnect are triggered at socket accept/close.
The initialize phase is immediately after the protocol
setup and authentication has completed. ie when the
client is authorized and about to start interacting with
the graphical desktop
This event comes with *a lot* of potential information
- IP address, port & address family of client
- IP address, port & address family of server
- Authentication scheme (arbitrary string)
- Authenticated subject identity. A subject may have
multiple identities with some authentication schemes.
For example, vencrypt+sasl results in a x509dname
and saslUsername identities.
This results in a very complicated callback :-(
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV4,
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV6,
} virDomainEventGraphicsAddressType;
struct _virDomainEventGraphicsAddress {
int family;
const char *node;
const char *service;
};
typedef struct _virDomainEventGraphicsAddress virDomainEventGraphicsAddress;
typedef virDomainEventGraphicsAddress *virDomainEventGraphicsAddressPtr;
struct _virDomainEventGraphicsSubject {
int nidentity;
struct {
const char *type;
const char *name;
} *identities;
};
typedef struct _virDomainEventGraphicsSubject virDomainEventGraphicsSubject;
typedef virDomainEventGraphicsSubject *virDomainEventGraphicsSubjectPtr;
typedef void (*virConnectDomainEventGraphicsCallback)(virConnectPtr conn,
virDomainPtr dom,
int phase,
virDomainEventGraphicsAddressPtr local,
virDomainEventGraphicsAddressPtr remote,
const char *authScheme,
virDomainEventGraphicsSubjectPtr subject,
void *opaque);
The wire protocol is similarly complex
struct remote_domain_event_graphics_address {
int family;
remote_nonnull_string node;
remote_nonnull_string service;
};
const REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX = 20;
struct remote_domain_event_graphics_identity {
remote_nonnull_string type;
remote_nonnull_string name;
};
struct remote_domain_event_graphics_msg {
remote_nonnull_domain dom;
int phase;
remote_domain_event_graphics_address local;
remote_domain_event_graphics_address remote;
remote_nonnull_string authScheme;
remote_domain_event_graphics_identity subject<REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX>;
};
This is currently implemented in QEMU for the VNC graphics
protocol, but designed to be usable with SPICE graphics in
the future too.
* daemon/remote.c: Dispatch graphics events to client
* examples/domain-events/events-c/event-test.c: Watch for
graphics events
* include/libvirt/libvirt.h.in: Define new graphics event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle graphics events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for VNC events and emit a libvirt graphics event
* src/remote/remote_driver.c: Receive and dispatch graphics
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
graphics events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for VNC_CONNECTED,
VNC_INITIALIZED & VNC_DISCONNETED events from QEMU monitor
2010-03-19 13:27:45 +00:00
|
|
|
return;
|
|
|
|
}
|
qemu: simplify json parsing
Rather than grabbing an arbitrary JSON value and then checking
if it has the right type, we might as well request the correct
type to begin with.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONIOProcessEvent)
(qemuMonitorJSONCommandWithFd, qemuMonitorJSONHandleGraphics)
(qemuMonitorJSONGetStatus, qemuMonitorJSONExtractCPUInfo)
(qemuMonitorJSONGetVirtType, qemuMonitorJSONGetBalloonInfo)
(qemuMonitorJSONGetMemoryStats)
(qemuMonitorJSONDevGetBlockExtent)
(qemuMonitorJSONGetOneBlockStatsInfo)
(qemuMonitorJSONGetAllBlockStatsInfo)
(qemuMonitorJSONBlockStatsUpdateCapacityOne)
(qemuMonitorJSONBlockStatsUpdateCapacity)
(qemuMonitorJSONGetBlockExtent)
(qemuMonitorJSONGetMigrationStatusReply)
(qemuMonitorJSONGetDumpGuestMemoryCapability)
(qemuMonitorJSONAddFd, qemuMonitorJSONQueryRxFilterParse)
(qemuMonitorJSONExtractChardevInfo)
(qemuMonitorJSONDiskNameLookupOne)
(qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONGetAllBlockJobInfo)
(qemuMonitorJSONBlockIoThrottleInfo, qemuMonitorJSONGetVersion)
(qemuMonitorJSONGetMachines, qemuMonitorJSONGetCPUDefinitions)
(qemuMonitorJSONGetCommands, qemuMonitorJSONGetEvents)
(qemuMonitorJSONGetKVMState, qemuMonitorJSONGetObjectTypes)
(qemuMonitorJSONGetObjectListPaths)
(qemuMonitorJSONGetObjectProps, qemuMonitorJSONGetTargetArch)
(qemuMonitorJSONGetMigrationCapabilities)
(qemuMonitorJSONGetStringArray, qemuMonitorJSONAttachCharDev)
(qemuMonitorJSONGetCPUx86Data, qemuMonitorJSONGetIOThreads)
(qemuMonitorJSONGetMemoryDeviceInfo): Use shorter idioms.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-06-20 13:40:21 -06:00
|
|
|
if (!(server = virJSONValueObjectGetObject(data, "server"))) {
|
2011-05-09 17:24:09 +08:00
|
|
|
VIR_WARN("missing server info in VNC event");
|
Add domain events for graphics network clients
This introduces a new event type
VIR_DOMAIN_EVENT_ID_GRAPHICS
The same event can be emitted in 3 scenarios
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_CONNECT = 0,
VIR_DOMAIN_EVENT_GRAPHICS_INITIALIZE,
VIR_DOMAIN_EVENT_GRAPHICS_DISCONNECT,
} virDomainEventGraphicsPhase;
Connect/disconnect are triggered at socket accept/close.
The initialize phase is immediately after the protocol
setup and authentication has completed. ie when the
client is authorized and about to start interacting with
the graphical desktop
This event comes with *a lot* of potential information
- IP address, port & address family of client
- IP address, port & address family of server
- Authentication scheme (arbitrary string)
- Authenticated subject identity. A subject may have
multiple identities with some authentication schemes.
For example, vencrypt+sasl results in a x509dname
and saslUsername identities.
This results in a very complicated callback :-(
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV4,
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV6,
} virDomainEventGraphicsAddressType;
struct _virDomainEventGraphicsAddress {
int family;
const char *node;
const char *service;
};
typedef struct _virDomainEventGraphicsAddress virDomainEventGraphicsAddress;
typedef virDomainEventGraphicsAddress *virDomainEventGraphicsAddressPtr;
struct _virDomainEventGraphicsSubject {
int nidentity;
struct {
const char *type;
const char *name;
} *identities;
};
typedef struct _virDomainEventGraphicsSubject virDomainEventGraphicsSubject;
typedef virDomainEventGraphicsSubject *virDomainEventGraphicsSubjectPtr;
typedef void (*virConnectDomainEventGraphicsCallback)(virConnectPtr conn,
virDomainPtr dom,
int phase,
virDomainEventGraphicsAddressPtr local,
virDomainEventGraphicsAddressPtr remote,
const char *authScheme,
virDomainEventGraphicsSubjectPtr subject,
void *opaque);
The wire protocol is similarly complex
struct remote_domain_event_graphics_address {
int family;
remote_nonnull_string node;
remote_nonnull_string service;
};
const REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX = 20;
struct remote_domain_event_graphics_identity {
remote_nonnull_string type;
remote_nonnull_string name;
};
struct remote_domain_event_graphics_msg {
remote_nonnull_domain dom;
int phase;
remote_domain_event_graphics_address local;
remote_domain_event_graphics_address remote;
remote_nonnull_string authScheme;
remote_domain_event_graphics_identity subject<REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX>;
};
This is currently implemented in QEMU for the VNC graphics
protocol, but designed to be usable with SPICE graphics in
the future too.
* daemon/remote.c: Dispatch graphics events to client
* examples/domain-events/events-c/event-test.c: Watch for
graphics events
* include/libvirt/libvirt.h.in: Define new graphics event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle graphics events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for VNC events and emit a libvirt graphics event
* src/remote/remote_driver.c: Receive and dispatch graphics
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
graphics events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for VNC_CONNECTED,
VNC_INITIALIZED & VNC_DISCONNETED events from QEMU monitor
2010-03-19 13:27:45 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-06-29 17:03:14 +02:00
|
|
|
if (!(authScheme = virJSONValueObjectGetString(server, "auth"))) {
|
2012-09-04 16:52:47 +02:00
|
|
|
/* not all events are required to contain auth scheme */
|
2015-06-29 17:03:14 +02:00
|
|
|
VIR_DEBUG("missing auth scheme in VNC event");
|
2012-09-04 16:52:47 +02:00
|
|
|
authScheme = "";
|
2012-03-14 01:41:35 -04:00
|
|
|
}
|
Add domain events for graphics network clients
This introduces a new event type
VIR_DOMAIN_EVENT_ID_GRAPHICS
The same event can be emitted in 3 scenarios
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_CONNECT = 0,
VIR_DOMAIN_EVENT_GRAPHICS_INITIALIZE,
VIR_DOMAIN_EVENT_GRAPHICS_DISCONNECT,
} virDomainEventGraphicsPhase;
Connect/disconnect are triggered at socket accept/close.
The initialize phase is immediately after the protocol
setup and authentication has completed. ie when the
client is authorized and about to start interacting with
the graphical desktop
This event comes with *a lot* of potential information
- IP address, port & address family of client
- IP address, port & address family of server
- Authentication scheme (arbitrary string)
- Authenticated subject identity. A subject may have
multiple identities with some authentication schemes.
For example, vencrypt+sasl results in a x509dname
and saslUsername identities.
This results in a very complicated callback :-(
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV4,
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV6,
} virDomainEventGraphicsAddressType;
struct _virDomainEventGraphicsAddress {
int family;
const char *node;
const char *service;
};
typedef struct _virDomainEventGraphicsAddress virDomainEventGraphicsAddress;
typedef virDomainEventGraphicsAddress *virDomainEventGraphicsAddressPtr;
struct _virDomainEventGraphicsSubject {
int nidentity;
struct {
const char *type;
const char *name;
} *identities;
};
typedef struct _virDomainEventGraphicsSubject virDomainEventGraphicsSubject;
typedef virDomainEventGraphicsSubject *virDomainEventGraphicsSubjectPtr;
typedef void (*virConnectDomainEventGraphicsCallback)(virConnectPtr conn,
virDomainPtr dom,
int phase,
virDomainEventGraphicsAddressPtr local,
virDomainEventGraphicsAddressPtr remote,
const char *authScheme,
virDomainEventGraphicsSubjectPtr subject,
void *opaque);
The wire protocol is similarly complex
struct remote_domain_event_graphics_address {
int family;
remote_nonnull_string node;
remote_nonnull_string service;
};
const REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX = 20;
struct remote_domain_event_graphics_identity {
remote_nonnull_string type;
remote_nonnull_string name;
};
struct remote_domain_event_graphics_msg {
remote_nonnull_domain dom;
int phase;
remote_domain_event_graphics_address local;
remote_domain_event_graphics_address remote;
remote_nonnull_string authScheme;
remote_domain_event_graphics_identity subject<REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX>;
};
This is currently implemented in QEMU for the VNC graphics
protocol, but designed to be usable with SPICE graphics in
the future too.
* daemon/remote.c: Dispatch graphics events to client
* examples/domain-events/events-c/event-test.c: Watch for
graphics events
* include/libvirt/libvirt.h.in: Define new graphics event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle graphics events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for VNC events and emit a libvirt graphics event
* src/remote/remote_driver.c: Receive and dispatch graphics
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
graphics events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for VNC_CONNECTED,
VNC_INITIALIZED & VNC_DISCONNETED events from QEMU monitor
2010-03-19 13:27:45 +00:00
|
|
|
|
2015-06-29 17:03:14 +02:00
|
|
|
if (!(localFamily = virJSONValueObjectGetString(server, "family"))) {
|
|
|
|
VIR_WARN("missing local address family in VNC event");
|
2012-03-14 01:41:35 -04:00
|
|
|
return;
|
|
|
|
}
|
2015-06-29 17:03:14 +02:00
|
|
|
if (!(localNode = virJSONValueObjectGetString(server, "host"))) {
|
|
|
|
VIR_WARN("missing local hostname in VNC event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!(localService = virJSONValueObjectGetString(server, "service"))) {
|
|
|
|
VIR_WARN("missing local service in VNC event");
|
2012-03-14 01:41:35 -04:00
|
|
|
return;
|
|
|
|
}
|
Add domain events for graphics network clients
This introduces a new event type
VIR_DOMAIN_EVENT_ID_GRAPHICS
The same event can be emitted in 3 scenarios
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_CONNECT = 0,
VIR_DOMAIN_EVENT_GRAPHICS_INITIALIZE,
VIR_DOMAIN_EVENT_GRAPHICS_DISCONNECT,
} virDomainEventGraphicsPhase;
Connect/disconnect are triggered at socket accept/close.
The initialize phase is immediately after the protocol
setup and authentication has completed. ie when the
client is authorized and about to start interacting with
the graphical desktop
This event comes with *a lot* of potential information
- IP address, port & address family of client
- IP address, port & address family of server
- Authentication scheme (arbitrary string)
- Authenticated subject identity. A subject may have
multiple identities with some authentication schemes.
For example, vencrypt+sasl results in a x509dname
and saslUsername identities.
This results in a very complicated callback :-(
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV4,
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV6,
} virDomainEventGraphicsAddressType;
struct _virDomainEventGraphicsAddress {
int family;
const char *node;
const char *service;
};
typedef struct _virDomainEventGraphicsAddress virDomainEventGraphicsAddress;
typedef virDomainEventGraphicsAddress *virDomainEventGraphicsAddressPtr;
struct _virDomainEventGraphicsSubject {
int nidentity;
struct {
const char *type;
const char *name;
} *identities;
};
typedef struct _virDomainEventGraphicsSubject virDomainEventGraphicsSubject;
typedef virDomainEventGraphicsSubject *virDomainEventGraphicsSubjectPtr;
typedef void (*virConnectDomainEventGraphicsCallback)(virConnectPtr conn,
virDomainPtr dom,
int phase,
virDomainEventGraphicsAddressPtr local,
virDomainEventGraphicsAddressPtr remote,
const char *authScheme,
virDomainEventGraphicsSubjectPtr subject,
void *opaque);
The wire protocol is similarly complex
struct remote_domain_event_graphics_address {
int family;
remote_nonnull_string node;
remote_nonnull_string service;
};
const REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX = 20;
struct remote_domain_event_graphics_identity {
remote_nonnull_string type;
remote_nonnull_string name;
};
struct remote_domain_event_graphics_msg {
remote_nonnull_domain dom;
int phase;
remote_domain_event_graphics_address local;
remote_domain_event_graphics_address remote;
remote_nonnull_string authScheme;
remote_domain_event_graphics_identity subject<REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX>;
};
This is currently implemented in QEMU for the VNC graphics
protocol, but designed to be usable with SPICE graphics in
the future too.
* daemon/remote.c: Dispatch graphics events to client
* examples/domain-events/events-c/event-test.c: Watch for
graphics events
* include/libvirt/libvirt.h.in: Define new graphics event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle graphics events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for VNC events and emit a libvirt graphics event
* src/remote/remote_driver.c: Receive and dispatch graphics
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
graphics events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for VNC_CONNECTED,
VNC_INITIALIZED & VNC_DISCONNETED events from QEMU monitor
2010-03-19 13:27:45 +00:00
|
|
|
|
2015-06-29 17:03:14 +02:00
|
|
|
if (!(remoteFamily = virJSONValueObjectGetString(client, "family"))) {
|
|
|
|
VIR_WARN("missing remote address family in VNC event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!(remoteNode = virJSONValueObjectGetString(client, "host"))) {
|
|
|
|
VIR_WARN("missing remote hostname in VNC event");
|
2012-03-14 01:41:35 -04:00
|
|
|
return;
|
|
|
|
}
|
2015-06-29 17:03:14 +02:00
|
|
|
if (!(remoteService = virJSONValueObjectGetString(client, "service"))) {
|
|
|
|
VIR_WARN("missing remote service in VNC event");
|
2012-03-14 01:41:35 -04:00
|
|
|
return;
|
|
|
|
}
|
Add domain events for graphics network clients
This introduces a new event type
VIR_DOMAIN_EVENT_ID_GRAPHICS
The same event can be emitted in 3 scenarios
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_CONNECT = 0,
VIR_DOMAIN_EVENT_GRAPHICS_INITIALIZE,
VIR_DOMAIN_EVENT_GRAPHICS_DISCONNECT,
} virDomainEventGraphicsPhase;
Connect/disconnect are triggered at socket accept/close.
The initialize phase is immediately after the protocol
setup and authentication has completed. ie when the
client is authorized and about to start interacting with
the graphical desktop
This event comes with *a lot* of potential information
- IP address, port & address family of client
- IP address, port & address family of server
- Authentication scheme (arbitrary string)
- Authenticated subject identity. A subject may have
multiple identities with some authentication schemes.
For example, vencrypt+sasl results in a x509dname
and saslUsername identities.
This results in a very complicated callback :-(
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV4,
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV6,
} virDomainEventGraphicsAddressType;
struct _virDomainEventGraphicsAddress {
int family;
const char *node;
const char *service;
};
typedef struct _virDomainEventGraphicsAddress virDomainEventGraphicsAddress;
typedef virDomainEventGraphicsAddress *virDomainEventGraphicsAddressPtr;
struct _virDomainEventGraphicsSubject {
int nidentity;
struct {
const char *type;
const char *name;
} *identities;
};
typedef struct _virDomainEventGraphicsSubject virDomainEventGraphicsSubject;
typedef virDomainEventGraphicsSubject *virDomainEventGraphicsSubjectPtr;
typedef void (*virConnectDomainEventGraphicsCallback)(virConnectPtr conn,
virDomainPtr dom,
int phase,
virDomainEventGraphicsAddressPtr local,
virDomainEventGraphicsAddressPtr remote,
const char *authScheme,
virDomainEventGraphicsSubjectPtr subject,
void *opaque);
The wire protocol is similarly complex
struct remote_domain_event_graphics_address {
int family;
remote_nonnull_string node;
remote_nonnull_string service;
};
const REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX = 20;
struct remote_domain_event_graphics_identity {
remote_nonnull_string type;
remote_nonnull_string name;
};
struct remote_domain_event_graphics_msg {
remote_nonnull_domain dom;
int phase;
remote_domain_event_graphics_address local;
remote_domain_event_graphics_address remote;
remote_nonnull_string authScheme;
remote_domain_event_graphics_identity subject<REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX>;
};
This is currently implemented in QEMU for the VNC graphics
protocol, but designed to be usable with SPICE graphics in
the future too.
* daemon/remote.c: Dispatch graphics events to client
* examples/domain-events/events-c/event-test.c: Watch for
graphics events
* include/libvirt/libvirt.h.in: Define new graphics event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle graphics events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for VNC events and emit a libvirt graphics event
* src/remote/remote_driver.c: Receive and dispatch graphics
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
graphics events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for VNC_CONNECTED,
VNC_INITIALIZED & VNC_DISCONNETED events from QEMU monitor
2010-03-19 13:27:45 +00:00
|
|
|
|
|
|
|
saslUsername = virJSONValueObjectGetString(client, "sasl_username");
|
|
|
|
x509dname = virJSONValueObjectGetString(client, "x509_dname");
|
|
|
|
|
|
|
|
if ((localFamilyID = qemuMonitorGraphicsAddressFamilyTypeFromString(localFamily)) < 0) {
|
|
|
|
VIR_WARN("unknown address family '%s'", localFamily);
|
|
|
|
localFamilyID = VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV4;
|
|
|
|
}
|
|
|
|
if ((remoteFamilyID = qemuMonitorGraphicsAddressFamilyTypeFromString(remoteFamily)) < 0) {
|
|
|
|
VIR_WARN("unknown address family '%s'", remoteFamily);
|
|
|
|
remoteFamilyID = VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV4;
|
|
|
|
}
|
|
|
|
|
|
|
|
qemuMonitorEmitGraphics(mon, phase,
|
|
|
|
localFamilyID, localNode, localService,
|
|
|
|
remoteFamilyID, remoteNode, remoteService,
|
|
|
|
authScheme, x509dname, saslUsername);
|
|
|
|
}
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static void qemuMonitorJSONHandleVNCConnect(qemuMonitor *mon, virJSONValue *data)
|
Add domain events for graphics network clients
This introduces a new event type
VIR_DOMAIN_EVENT_ID_GRAPHICS
The same event can be emitted in 3 scenarios
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_CONNECT = 0,
VIR_DOMAIN_EVENT_GRAPHICS_INITIALIZE,
VIR_DOMAIN_EVENT_GRAPHICS_DISCONNECT,
} virDomainEventGraphicsPhase;
Connect/disconnect are triggered at socket accept/close.
The initialize phase is immediately after the protocol
setup and authentication has completed. ie when the
client is authorized and about to start interacting with
the graphical desktop
This event comes with *a lot* of potential information
- IP address, port & address family of client
- IP address, port & address family of server
- Authentication scheme (arbitrary string)
- Authenticated subject identity. A subject may have
multiple identities with some authentication schemes.
For example, vencrypt+sasl results in a x509dname
and saslUsername identities.
This results in a very complicated callback :-(
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV4,
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV6,
} virDomainEventGraphicsAddressType;
struct _virDomainEventGraphicsAddress {
int family;
const char *node;
const char *service;
};
typedef struct _virDomainEventGraphicsAddress virDomainEventGraphicsAddress;
typedef virDomainEventGraphicsAddress *virDomainEventGraphicsAddressPtr;
struct _virDomainEventGraphicsSubject {
int nidentity;
struct {
const char *type;
const char *name;
} *identities;
};
typedef struct _virDomainEventGraphicsSubject virDomainEventGraphicsSubject;
typedef virDomainEventGraphicsSubject *virDomainEventGraphicsSubjectPtr;
typedef void (*virConnectDomainEventGraphicsCallback)(virConnectPtr conn,
virDomainPtr dom,
int phase,
virDomainEventGraphicsAddressPtr local,
virDomainEventGraphicsAddressPtr remote,
const char *authScheme,
virDomainEventGraphicsSubjectPtr subject,
void *opaque);
The wire protocol is similarly complex
struct remote_domain_event_graphics_address {
int family;
remote_nonnull_string node;
remote_nonnull_string service;
};
const REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX = 20;
struct remote_domain_event_graphics_identity {
remote_nonnull_string type;
remote_nonnull_string name;
};
struct remote_domain_event_graphics_msg {
remote_nonnull_domain dom;
int phase;
remote_domain_event_graphics_address local;
remote_domain_event_graphics_address remote;
remote_nonnull_string authScheme;
remote_domain_event_graphics_identity subject<REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX>;
};
This is currently implemented in QEMU for the VNC graphics
protocol, but designed to be usable with SPICE graphics in
the future too.
* daemon/remote.c: Dispatch graphics events to client
* examples/domain-events/events-c/event-test.c: Watch for
graphics events
* include/libvirt/libvirt.h.in: Define new graphics event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle graphics events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for VNC events and emit a libvirt graphics event
* src/remote/remote_driver.c: Receive and dispatch graphics
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
graphics events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for VNC_CONNECTED,
VNC_INITIALIZED & VNC_DISCONNETED events from QEMU monitor
2010-03-19 13:27:45 +00:00
|
|
|
{
|
2015-06-29 16:10:51 +02:00
|
|
|
qemuMonitorJSONHandleGraphicsVNC(mon, data, VIR_DOMAIN_EVENT_GRAPHICS_CONNECT);
|
Add domain events for graphics network clients
This introduces a new event type
VIR_DOMAIN_EVENT_ID_GRAPHICS
The same event can be emitted in 3 scenarios
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_CONNECT = 0,
VIR_DOMAIN_EVENT_GRAPHICS_INITIALIZE,
VIR_DOMAIN_EVENT_GRAPHICS_DISCONNECT,
} virDomainEventGraphicsPhase;
Connect/disconnect are triggered at socket accept/close.
The initialize phase is immediately after the protocol
setup and authentication has completed. ie when the
client is authorized and about to start interacting with
the graphical desktop
This event comes with *a lot* of potential information
- IP address, port & address family of client
- IP address, port & address family of server
- Authentication scheme (arbitrary string)
- Authenticated subject identity. A subject may have
multiple identities with some authentication schemes.
For example, vencrypt+sasl results in a x509dname
and saslUsername identities.
This results in a very complicated callback :-(
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV4,
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV6,
} virDomainEventGraphicsAddressType;
struct _virDomainEventGraphicsAddress {
int family;
const char *node;
const char *service;
};
typedef struct _virDomainEventGraphicsAddress virDomainEventGraphicsAddress;
typedef virDomainEventGraphicsAddress *virDomainEventGraphicsAddressPtr;
struct _virDomainEventGraphicsSubject {
int nidentity;
struct {
const char *type;
const char *name;
} *identities;
};
typedef struct _virDomainEventGraphicsSubject virDomainEventGraphicsSubject;
typedef virDomainEventGraphicsSubject *virDomainEventGraphicsSubjectPtr;
typedef void (*virConnectDomainEventGraphicsCallback)(virConnectPtr conn,
virDomainPtr dom,
int phase,
virDomainEventGraphicsAddressPtr local,
virDomainEventGraphicsAddressPtr remote,
const char *authScheme,
virDomainEventGraphicsSubjectPtr subject,
void *opaque);
The wire protocol is similarly complex
struct remote_domain_event_graphics_address {
int family;
remote_nonnull_string node;
remote_nonnull_string service;
};
const REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX = 20;
struct remote_domain_event_graphics_identity {
remote_nonnull_string type;
remote_nonnull_string name;
};
struct remote_domain_event_graphics_msg {
remote_nonnull_domain dom;
int phase;
remote_domain_event_graphics_address local;
remote_domain_event_graphics_address remote;
remote_nonnull_string authScheme;
remote_domain_event_graphics_identity subject<REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX>;
};
This is currently implemented in QEMU for the VNC graphics
protocol, but designed to be usable with SPICE graphics in
the future too.
* daemon/remote.c: Dispatch graphics events to client
* examples/domain-events/events-c/event-test.c: Watch for
graphics events
* include/libvirt/libvirt.h.in: Define new graphics event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle graphics events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for VNC events and emit a libvirt graphics event
* src/remote/remote_driver.c: Receive and dispatch graphics
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
graphics events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for VNC_CONNECTED,
VNC_INITIALIZED & VNC_DISCONNETED events from QEMU monitor
2010-03-19 13:27:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static void qemuMonitorJSONHandleVNCInitialize(qemuMonitor *mon, virJSONValue *data)
|
Add domain events for graphics network clients
This introduces a new event type
VIR_DOMAIN_EVENT_ID_GRAPHICS
The same event can be emitted in 3 scenarios
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_CONNECT = 0,
VIR_DOMAIN_EVENT_GRAPHICS_INITIALIZE,
VIR_DOMAIN_EVENT_GRAPHICS_DISCONNECT,
} virDomainEventGraphicsPhase;
Connect/disconnect are triggered at socket accept/close.
The initialize phase is immediately after the protocol
setup and authentication has completed. ie when the
client is authorized and about to start interacting with
the graphical desktop
This event comes with *a lot* of potential information
- IP address, port & address family of client
- IP address, port & address family of server
- Authentication scheme (arbitrary string)
- Authenticated subject identity. A subject may have
multiple identities with some authentication schemes.
For example, vencrypt+sasl results in a x509dname
and saslUsername identities.
This results in a very complicated callback :-(
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV4,
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV6,
} virDomainEventGraphicsAddressType;
struct _virDomainEventGraphicsAddress {
int family;
const char *node;
const char *service;
};
typedef struct _virDomainEventGraphicsAddress virDomainEventGraphicsAddress;
typedef virDomainEventGraphicsAddress *virDomainEventGraphicsAddressPtr;
struct _virDomainEventGraphicsSubject {
int nidentity;
struct {
const char *type;
const char *name;
} *identities;
};
typedef struct _virDomainEventGraphicsSubject virDomainEventGraphicsSubject;
typedef virDomainEventGraphicsSubject *virDomainEventGraphicsSubjectPtr;
typedef void (*virConnectDomainEventGraphicsCallback)(virConnectPtr conn,
virDomainPtr dom,
int phase,
virDomainEventGraphicsAddressPtr local,
virDomainEventGraphicsAddressPtr remote,
const char *authScheme,
virDomainEventGraphicsSubjectPtr subject,
void *opaque);
The wire protocol is similarly complex
struct remote_domain_event_graphics_address {
int family;
remote_nonnull_string node;
remote_nonnull_string service;
};
const REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX = 20;
struct remote_domain_event_graphics_identity {
remote_nonnull_string type;
remote_nonnull_string name;
};
struct remote_domain_event_graphics_msg {
remote_nonnull_domain dom;
int phase;
remote_domain_event_graphics_address local;
remote_domain_event_graphics_address remote;
remote_nonnull_string authScheme;
remote_domain_event_graphics_identity subject<REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX>;
};
This is currently implemented in QEMU for the VNC graphics
protocol, but designed to be usable with SPICE graphics in
the future too.
* daemon/remote.c: Dispatch graphics events to client
* examples/domain-events/events-c/event-test.c: Watch for
graphics events
* include/libvirt/libvirt.h.in: Define new graphics event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle graphics events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for VNC events and emit a libvirt graphics event
* src/remote/remote_driver.c: Receive and dispatch graphics
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
graphics events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for VNC_CONNECTED,
VNC_INITIALIZED & VNC_DISCONNETED events from QEMU monitor
2010-03-19 13:27:45 +00:00
|
|
|
{
|
2015-06-29 16:10:51 +02:00
|
|
|
qemuMonitorJSONHandleGraphicsVNC(mon, data, VIR_DOMAIN_EVENT_GRAPHICS_INITIALIZE);
|
Add domain events for graphics network clients
This introduces a new event type
VIR_DOMAIN_EVENT_ID_GRAPHICS
The same event can be emitted in 3 scenarios
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_CONNECT = 0,
VIR_DOMAIN_EVENT_GRAPHICS_INITIALIZE,
VIR_DOMAIN_EVENT_GRAPHICS_DISCONNECT,
} virDomainEventGraphicsPhase;
Connect/disconnect are triggered at socket accept/close.
The initialize phase is immediately after the protocol
setup and authentication has completed. ie when the
client is authorized and about to start interacting with
the graphical desktop
This event comes with *a lot* of potential information
- IP address, port & address family of client
- IP address, port & address family of server
- Authentication scheme (arbitrary string)
- Authenticated subject identity. A subject may have
multiple identities with some authentication schemes.
For example, vencrypt+sasl results in a x509dname
and saslUsername identities.
This results in a very complicated callback :-(
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV4,
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV6,
} virDomainEventGraphicsAddressType;
struct _virDomainEventGraphicsAddress {
int family;
const char *node;
const char *service;
};
typedef struct _virDomainEventGraphicsAddress virDomainEventGraphicsAddress;
typedef virDomainEventGraphicsAddress *virDomainEventGraphicsAddressPtr;
struct _virDomainEventGraphicsSubject {
int nidentity;
struct {
const char *type;
const char *name;
} *identities;
};
typedef struct _virDomainEventGraphicsSubject virDomainEventGraphicsSubject;
typedef virDomainEventGraphicsSubject *virDomainEventGraphicsSubjectPtr;
typedef void (*virConnectDomainEventGraphicsCallback)(virConnectPtr conn,
virDomainPtr dom,
int phase,
virDomainEventGraphicsAddressPtr local,
virDomainEventGraphicsAddressPtr remote,
const char *authScheme,
virDomainEventGraphicsSubjectPtr subject,
void *opaque);
The wire protocol is similarly complex
struct remote_domain_event_graphics_address {
int family;
remote_nonnull_string node;
remote_nonnull_string service;
};
const REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX = 20;
struct remote_domain_event_graphics_identity {
remote_nonnull_string type;
remote_nonnull_string name;
};
struct remote_domain_event_graphics_msg {
remote_nonnull_domain dom;
int phase;
remote_domain_event_graphics_address local;
remote_domain_event_graphics_address remote;
remote_nonnull_string authScheme;
remote_domain_event_graphics_identity subject<REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX>;
};
This is currently implemented in QEMU for the VNC graphics
protocol, but designed to be usable with SPICE graphics in
the future too.
* daemon/remote.c: Dispatch graphics events to client
* examples/domain-events/events-c/event-test.c: Watch for
graphics events
* include/libvirt/libvirt.h.in: Define new graphics event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle graphics events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for VNC events and emit a libvirt graphics event
* src/remote/remote_driver.c: Receive and dispatch graphics
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
graphics events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for VNC_CONNECTED,
VNC_INITIALIZED & VNC_DISCONNETED events from QEMU monitor
2010-03-19 13:27:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static void qemuMonitorJSONHandleVNCDisconnect(qemuMonitor *mon, virJSONValue *data)
|
Add domain events for graphics network clients
This introduces a new event type
VIR_DOMAIN_EVENT_ID_GRAPHICS
The same event can be emitted in 3 scenarios
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_CONNECT = 0,
VIR_DOMAIN_EVENT_GRAPHICS_INITIALIZE,
VIR_DOMAIN_EVENT_GRAPHICS_DISCONNECT,
} virDomainEventGraphicsPhase;
Connect/disconnect are triggered at socket accept/close.
The initialize phase is immediately after the protocol
setup and authentication has completed. ie when the
client is authorized and about to start interacting with
the graphical desktop
This event comes with *a lot* of potential information
- IP address, port & address family of client
- IP address, port & address family of server
- Authentication scheme (arbitrary string)
- Authenticated subject identity. A subject may have
multiple identities with some authentication schemes.
For example, vencrypt+sasl results in a x509dname
and saslUsername identities.
This results in a very complicated callback :-(
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV4,
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV6,
} virDomainEventGraphicsAddressType;
struct _virDomainEventGraphicsAddress {
int family;
const char *node;
const char *service;
};
typedef struct _virDomainEventGraphicsAddress virDomainEventGraphicsAddress;
typedef virDomainEventGraphicsAddress *virDomainEventGraphicsAddressPtr;
struct _virDomainEventGraphicsSubject {
int nidentity;
struct {
const char *type;
const char *name;
} *identities;
};
typedef struct _virDomainEventGraphicsSubject virDomainEventGraphicsSubject;
typedef virDomainEventGraphicsSubject *virDomainEventGraphicsSubjectPtr;
typedef void (*virConnectDomainEventGraphicsCallback)(virConnectPtr conn,
virDomainPtr dom,
int phase,
virDomainEventGraphicsAddressPtr local,
virDomainEventGraphicsAddressPtr remote,
const char *authScheme,
virDomainEventGraphicsSubjectPtr subject,
void *opaque);
The wire protocol is similarly complex
struct remote_domain_event_graphics_address {
int family;
remote_nonnull_string node;
remote_nonnull_string service;
};
const REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX = 20;
struct remote_domain_event_graphics_identity {
remote_nonnull_string type;
remote_nonnull_string name;
};
struct remote_domain_event_graphics_msg {
remote_nonnull_domain dom;
int phase;
remote_domain_event_graphics_address local;
remote_domain_event_graphics_address remote;
remote_nonnull_string authScheme;
remote_domain_event_graphics_identity subject<REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX>;
};
This is currently implemented in QEMU for the VNC graphics
protocol, but designed to be usable with SPICE graphics in
the future too.
* daemon/remote.c: Dispatch graphics events to client
* examples/domain-events/events-c/event-test.c: Watch for
graphics events
* include/libvirt/libvirt.h.in: Define new graphics event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle graphics events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for VNC events and emit a libvirt graphics event
* src/remote/remote_driver.c: Receive and dispatch graphics
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
graphics events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for VNC_CONNECTED,
VNC_INITIALIZED & VNC_DISCONNETED events from QEMU monitor
2010-03-19 13:27:45 +00:00
|
|
|
{
|
2015-06-29 16:10:51 +02:00
|
|
|
qemuMonitorJSONHandleGraphicsVNC(mon, data, VIR_DOMAIN_EVENT_GRAPHICS_DISCONNECT);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHandleGraphicsSPICE(qemuMonitor *mon,
|
|
|
|
virJSONValue *data,
|
2015-06-29 16:10:51 +02:00
|
|
|
int phase)
|
|
|
|
{
|
|
|
|
const char *lhost, *lport, *lfamily;
|
|
|
|
const char *rhost, *rport, *rfamily;
|
|
|
|
const char *auth = "";
|
|
|
|
int lfamilyID, rfamilyID;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *client;
|
|
|
|
virJSONValue *server;
|
2015-06-29 16:10:51 +02:00
|
|
|
|
|
|
|
if (!(client = virJSONValueObjectGetObject(data, "client")) ||
|
|
|
|
!(server = virJSONValueObjectGetObject(data, "server"))) {
|
|
|
|
VIR_WARN("missing server or client info in SPICE event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (phase == VIR_DOMAIN_EVENT_GRAPHICS_INITIALIZE &&
|
|
|
|
!(auth = virJSONValueObjectGetString(server, "auth"))) {
|
|
|
|
VIR_DEBUG("missing auth scheme in SPICE event");
|
|
|
|
auth = "";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(lfamily = virJSONValueObjectGetString(server, "family"))) {
|
|
|
|
VIR_WARN("missing local address family in SPICE event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!(lhost = virJSONValueObjectGetString(server, "host"))) {
|
|
|
|
VIR_WARN("missing local hostname in SPICE event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!(lport = virJSONValueObjectGetString(server, "port"))) {
|
|
|
|
VIR_WARN("missing local port in SPICE event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(rfamily = virJSONValueObjectGetString(client, "family"))) {
|
|
|
|
VIR_WARN("missing remote address family in SPICE event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!(rhost = virJSONValueObjectGetString(client, "host"))) {
|
|
|
|
VIR_WARN("missing remote hostname in SPICE event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!(rport = virJSONValueObjectGetString(client, "port"))) {
|
|
|
|
VIR_WARN("missing remote service in SPICE event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((lfamilyID = qemuMonitorGraphicsAddressFamilyTypeFromString(lfamily)) < 0) {
|
|
|
|
VIR_WARN("unknown address family '%s'", lfamily);
|
|
|
|
lfamilyID = VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV4;
|
|
|
|
}
|
|
|
|
if ((rfamilyID = qemuMonitorGraphicsAddressFamilyTypeFromString(rfamily)) < 0) {
|
|
|
|
VIR_WARN("unknown address family '%s'", rfamily);
|
|
|
|
rfamilyID = VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV4;
|
|
|
|
}
|
|
|
|
|
|
|
|
qemuMonitorEmitGraphics(mon, phase, lfamilyID, lhost, lport, rfamilyID,
|
|
|
|
rhost, rport, auth, NULL, NULL);
|
2012-03-14 01:41:35 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static void qemuMonitorJSONHandleSPICEConnect(qemuMonitor *mon, virJSONValue *data)
|
2012-03-14 01:41:35 -04:00
|
|
|
{
|
2015-06-29 16:10:51 +02:00
|
|
|
qemuMonitorJSONHandleGraphicsSPICE(mon, data, VIR_DOMAIN_EVENT_GRAPHICS_CONNECT);
|
2012-03-14 01:41:35 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static void qemuMonitorJSONHandleSPICEInitialize(qemuMonitor *mon, virJSONValue *data)
|
2012-03-14 01:41:35 -04:00
|
|
|
{
|
2015-06-29 16:10:51 +02:00
|
|
|
qemuMonitorJSONHandleGraphicsSPICE(mon, data, VIR_DOMAIN_EVENT_GRAPHICS_INITIALIZE);
|
2012-03-14 01:41:35 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static void qemuMonitorJSONHandleSPICEDisconnect(qemuMonitor *mon, virJSONValue *data)
|
2012-03-14 01:41:35 -04:00
|
|
|
{
|
2015-06-29 16:10:51 +02:00
|
|
|
qemuMonitorJSONHandleGraphicsSPICE(mon, data, VIR_DOMAIN_EVENT_GRAPHICS_DISCONNECT);
|
Add domain events for graphics network clients
This introduces a new event type
VIR_DOMAIN_EVENT_ID_GRAPHICS
The same event can be emitted in 3 scenarios
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_CONNECT = 0,
VIR_DOMAIN_EVENT_GRAPHICS_INITIALIZE,
VIR_DOMAIN_EVENT_GRAPHICS_DISCONNECT,
} virDomainEventGraphicsPhase;
Connect/disconnect are triggered at socket accept/close.
The initialize phase is immediately after the protocol
setup and authentication has completed. ie when the
client is authorized and about to start interacting with
the graphical desktop
This event comes with *a lot* of potential information
- IP address, port & address family of client
- IP address, port & address family of server
- Authentication scheme (arbitrary string)
- Authenticated subject identity. A subject may have
multiple identities with some authentication schemes.
For example, vencrypt+sasl results in a x509dname
and saslUsername identities.
This results in a very complicated callback :-(
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV4,
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV6,
} virDomainEventGraphicsAddressType;
struct _virDomainEventGraphicsAddress {
int family;
const char *node;
const char *service;
};
typedef struct _virDomainEventGraphicsAddress virDomainEventGraphicsAddress;
typedef virDomainEventGraphicsAddress *virDomainEventGraphicsAddressPtr;
struct _virDomainEventGraphicsSubject {
int nidentity;
struct {
const char *type;
const char *name;
} *identities;
};
typedef struct _virDomainEventGraphicsSubject virDomainEventGraphicsSubject;
typedef virDomainEventGraphicsSubject *virDomainEventGraphicsSubjectPtr;
typedef void (*virConnectDomainEventGraphicsCallback)(virConnectPtr conn,
virDomainPtr dom,
int phase,
virDomainEventGraphicsAddressPtr local,
virDomainEventGraphicsAddressPtr remote,
const char *authScheme,
virDomainEventGraphicsSubjectPtr subject,
void *opaque);
The wire protocol is similarly complex
struct remote_domain_event_graphics_address {
int family;
remote_nonnull_string node;
remote_nonnull_string service;
};
const REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX = 20;
struct remote_domain_event_graphics_identity {
remote_nonnull_string type;
remote_nonnull_string name;
};
struct remote_domain_event_graphics_msg {
remote_nonnull_domain dom;
int phase;
remote_domain_event_graphics_address local;
remote_domain_event_graphics_address remote;
remote_nonnull_string authScheme;
remote_domain_event_graphics_identity subject<REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX>;
};
This is currently implemented in QEMU for the VNC graphics
protocol, but designed to be usable with SPICE graphics in
the future too.
* daemon/remote.c: Dispatch graphics events to client
* examples/domain-events/events-c/event-test.c: Watch for
graphics events
* include/libvirt/libvirt.h.in: Define new graphics event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle graphics events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for VNC events and emit a libvirt graphics event
* src/remote/remote_driver.c: Receive and dispatch graphics
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
graphics events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for VNC_CONNECTED,
VNC_INITIALIZED & VNC_DISCONNETED events from QEMU monitor
2010-03-19 13:27:45 +00:00
|
|
|
}
|
|
|
|
|
2012-04-11 16:17:36 -06:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHandleBlockJobImpl(qemuMonitor *mon,
|
|
|
|
virJSONValue *data,
|
2012-04-11 16:17:36 -06:00
|
|
|
int event)
|
2011-07-22 13:57:42 +08:00
|
|
|
{
|
|
|
|
const char *device;
|
|
|
|
const char *type_str;
|
2017-10-27 15:37:22 +03:00
|
|
|
const char *error = NULL;
|
2011-07-22 13:57:42 +08:00
|
|
|
int type = VIR_DOMAIN_BLOCK_JOB_TYPE_UNKNOWN;
|
|
|
|
unsigned long long offset, len;
|
|
|
|
|
|
|
|
if ((device = virJSONValueObjectGetString(data, "device")) == NULL) {
|
|
|
|
VIR_WARN("missing device in block job event");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberUlong(data, "offset", &offset) < 0) {
|
|
|
|
VIR_WARN("missing offset in block job event");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberUlong(data, "len", &len) < 0) {
|
|
|
|
VIR_WARN("missing len in block job event");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((type_str = virJSONValueObjectGetString(data, "type")) == NULL) {
|
|
|
|
VIR_WARN("missing type in block job event");
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (STREQ(type_str, "stream"))
|
|
|
|
type = VIR_DOMAIN_BLOCK_JOB_TYPE_PULL;
|
2012-10-03 15:13:21 -06:00
|
|
|
else if (STREQ(type_str, "commit"))
|
|
|
|
type = VIR_DOMAIN_BLOCK_JOB_TYPE_COMMIT;
|
2012-10-12 14:06:10 -06:00
|
|
|
else if (STREQ(type_str, "mirror"))
|
|
|
|
type = VIR_DOMAIN_BLOCK_JOB_TYPE_COPY;
|
2019-10-18 15:10:33 +02:00
|
|
|
else if (STREQ(type_str, "backup"))
|
|
|
|
type = VIR_DOMAIN_BLOCK_JOB_TYPE_BACKUP;
|
2011-07-22 13:57:42 +08:00
|
|
|
|
2012-04-11 16:17:36 -06:00
|
|
|
switch ((virConnectDomainEventBlockJobStatus) event) {
|
|
|
|
case VIR_DOMAIN_BLOCK_JOB_COMPLETED:
|
2017-10-27 15:37:22 +03:00
|
|
|
error = virJSONValueObjectGetString(data, "error");
|
2012-04-11 16:17:36 -06:00
|
|
|
/* Make sure the whole device has been processed */
|
2018-01-19 22:03:05 +08:00
|
|
|
if (offset != len || error)
|
2012-04-11 16:17:36 -06:00
|
|
|
event = VIR_DOMAIN_BLOCK_JOB_FAILED;
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_BLOCK_JOB_CANCELED:
|
2012-10-12 14:06:10 -06:00
|
|
|
case VIR_DOMAIN_BLOCK_JOB_READY:
|
2012-04-11 16:17:36 -06:00
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_BLOCK_JOB_FAILED:
|
|
|
|
case VIR_DOMAIN_BLOCK_JOB_LAST:
|
2012-10-12 14:06:10 -06:00
|
|
|
VIR_DEBUG("should not get here");
|
|
|
|
break;
|
2012-04-11 16:17:36 -06:00
|
|
|
}
|
2011-07-22 13:57:42 +08:00
|
|
|
|
2014-03-25 07:49:44 +01:00
|
|
|
out:
|
2017-10-27 15:37:22 +03:00
|
|
|
qemuMonitorEmitBlockJob(mon, device, type, event, error);
|
2011-07-22 13:57:42 +08:00
|
|
|
}
|
|
|
|
|
2018-12-04 17:57:23 +01:00
|
|
|
|
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHandleJobStatusChange(qemuMonitor *mon,
|
|
|
|
virJSONValue *data)
|
2018-12-04 17:57:23 +01:00
|
|
|
{
|
|
|
|
const char *jobname = virJSONValueObjectGetString(data, "id");
|
|
|
|
const char *statusstr = virJSONValueObjectGetString(data, "status");
|
|
|
|
int status;
|
|
|
|
|
|
|
|
if (!jobname) {
|
|
|
|
VIR_WARN("missing job name in JOB_STATUS_CHANGE event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((status = qemuMonitorJobStatusTypeFromString(statusstr)) < 0) {
|
|
|
|
VIR_WARN("unknown job status '%s' for job '%s' in JOB_STATUS_CHANGE event",
|
|
|
|
statusstr, jobname);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
qemuMonitorEmitJobStatusChange(mon, jobname, status);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-03-23 21:44:50 +08:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHandleTrayChange(qemuMonitor *mon,
|
|
|
|
virJSONValue *data)
|
2012-03-23 21:44:50 +08:00
|
|
|
{
|
2018-08-13 16:02:38 +02:00
|
|
|
const char *devAlias = virJSONValueObjectGetString(data, "device");
|
|
|
|
const char *devid = virJSONValueObjectGetString(data, "id");
|
2012-03-23 21:44:50 +08:00
|
|
|
bool trayOpened;
|
|
|
|
int reason;
|
|
|
|
|
2018-08-13 16:02:38 +02:00
|
|
|
/* drive alias is always reported but empty for -blockdev */
|
|
|
|
if (*devAlias == '\0')
|
|
|
|
devAlias = NULL;
|
|
|
|
|
|
|
|
if (!devAlias && !devid) {
|
2012-03-23 21:44:50 +08:00
|
|
|
VIR_WARN("missing device in tray change event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetBoolean(data, "tray-open", &trayOpened) < 0) {
|
|
|
|
VIR_WARN("missing tray-open in tray change event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (trayOpened)
|
|
|
|
reason = VIR_DOMAIN_EVENT_TRAY_CHANGE_OPEN;
|
|
|
|
else
|
|
|
|
reason = VIR_DOMAIN_EVENT_TRAY_CHANGE_CLOSE;
|
|
|
|
|
2018-08-13 16:02:38 +02:00
|
|
|
qemuMonitorEmitTrayChange(mon, devAlias, devid, reason);
|
2012-03-23 21:44:50 +08:00
|
|
|
}
|
Add domain events for graphics network clients
This introduces a new event type
VIR_DOMAIN_EVENT_ID_GRAPHICS
The same event can be emitted in 3 scenarios
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_CONNECT = 0,
VIR_DOMAIN_EVENT_GRAPHICS_INITIALIZE,
VIR_DOMAIN_EVENT_GRAPHICS_DISCONNECT,
} virDomainEventGraphicsPhase;
Connect/disconnect are triggered at socket accept/close.
The initialize phase is immediately after the protocol
setup and authentication has completed. ie when the
client is authorized and about to start interacting with
the graphical desktop
This event comes with *a lot* of potential information
- IP address, port & address family of client
- IP address, port & address family of server
- Authentication scheme (arbitrary string)
- Authenticated subject identity. A subject may have
multiple identities with some authentication schemes.
For example, vencrypt+sasl results in a x509dname
and saslUsername identities.
This results in a very complicated callback :-(
typedef enum {
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV4,
VIR_DOMAIN_EVENT_GRAPHICS_ADDRESS_IPV6,
} virDomainEventGraphicsAddressType;
struct _virDomainEventGraphicsAddress {
int family;
const char *node;
const char *service;
};
typedef struct _virDomainEventGraphicsAddress virDomainEventGraphicsAddress;
typedef virDomainEventGraphicsAddress *virDomainEventGraphicsAddressPtr;
struct _virDomainEventGraphicsSubject {
int nidentity;
struct {
const char *type;
const char *name;
} *identities;
};
typedef struct _virDomainEventGraphicsSubject virDomainEventGraphicsSubject;
typedef virDomainEventGraphicsSubject *virDomainEventGraphicsSubjectPtr;
typedef void (*virConnectDomainEventGraphicsCallback)(virConnectPtr conn,
virDomainPtr dom,
int phase,
virDomainEventGraphicsAddressPtr local,
virDomainEventGraphicsAddressPtr remote,
const char *authScheme,
virDomainEventGraphicsSubjectPtr subject,
void *opaque);
The wire protocol is similarly complex
struct remote_domain_event_graphics_address {
int family;
remote_nonnull_string node;
remote_nonnull_string service;
};
const REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX = 20;
struct remote_domain_event_graphics_identity {
remote_nonnull_string type;
remote_nonnull_string name;
};
struct remote_domain_event_graphics_msg {
remote_nonnull_domain dom;
int phase;
remote_domain_event_graphics_address local;
remote_domain_event_graphics_address remote;
remote_nonnull_string authScheme;
remote_domain_event_graphics_identity subject<REMOTE_DOMAIN_EVENT_GRAPHICS_IDENTITY_MAX>;
};
This is currently implemented in QEMU for the VNC graphics
protocol, but designed to be usable with SPICE graphics in
the future too.
* daemon/remote.c: Dispatch graphics events to client
* examples/domain-events/events-c/event-test.c: Watch for
graphics events
* include/libvirt/libvirt.h.in: Define new graphics event ID
and callback signature
* src/conf/domain_event.c, src/conf/domain_event.h,
src/libvirt_private.syms: Extend API to handle graphics events
* src/qemu/qemu_driver.c: Connect to the QEMU monitor event
for VNC events and emit a libvirt graphics event
* src/remote/remote_driver.c: Receive and dispatch graphics
events to application
* src/remote/remote_protocol.x: Wire protocol definition for
graphics events
* src/qemu/qemu_monitor.c, src/qemu/qemu_monitor.h,
src/qemu/qemu_monitor_json.c: Watch for VNC_CONNECTED,
VNC_INITIALIZED & VNC_DISCONNETED events from QEMU monitor
2010-03-19 13:27:45 +00:00
|
|
|
|
2012-03-23 22:43:14 +08:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHandlePMWakeup(qemuMonitor *mon,
|
|
|
|
virJSONValue *data G_GNUC_UNUSED)
|
2012-03-23 22:43:14 +08:00
|
|
|
{
|
|
|
|
qemuMonitorEmitPMWakeup(mon);
|
|
|
|
}
|
|
|
|
|
2012-03-23 22:50:36 +08:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHandlePMSuspend(qemuMonitor *mon,
|
|
|
|
virJSONValue *data G_GNUC_UNUSED)
|
2012-03-23 22:50:36 +08:00
|
|
|
{
|
|
|
|
qemuMonitorEmitPMSuspend(mon);
|
|
|
|
}
|
|
|
|
|
2012-04-11 16:17:36 -06:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHandleBlockJobCompleted(qemuMonitor *mon,
|
|
|
|
virJSONValue *data)
|
2012-04-11 16:17:36 -06:00
|
|
|
{
|
|
|
|
qemuMonitorJSONHandleBlockJobImpl(mon, data,
|
|
|
|
VIR_DOMAIN_BLOCK_JOB_COMPLETED);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHandleBlockJobCanceled(qemuMonitor *mon,
|
|
|
|
virJSONValue *data)
|
2012-04-11 16:17:36 -06:00
|
|
|
{
|
|
|
|
qemuMonitorJSONHandleBlockJobImpl(mon, data,
|
|
|
|
VIR_DOMAIN_BLOCK_JOB_CANCELED);
|
|
|
|
}
|
|
|
|
|
2012-10-12 14:06:10 -06:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHandleBlockJobReady(qemuMonitor *mon,
|
|
|
|
virJSONValue *data)
|
2012-10-12 14:06:10 -06:00
|
|
|
{
|
|
|
|
qemuMonitorJSONHandleBlockJobImpl(mon, data,
|
|
|
|
VIR_DOMAIN_BLOCK_JOB_READY);
|
|
|
|
}
|
|
|
|
|
2012-07-12 23:45:57 +08:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHandleBalloonChange(qemuMonitor *mon,
|
|
|
|
virJSONValue *data)
|
2012-07-12 23:45:57 +08:00
|
|
|
{
|
|
|
|
unsigned long long actual = 0;
|
|
|
|
if (virJSONValueObjectGetNumberUlong(data, "actual", &actual) < 0) {
|
|
|
|
VIR_WARN("missing actual in balloon change event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
actual = VIR_DIV_UP(actual, 1024);
|
|
|
|
qemuMonitorEmitBalloonChange(mon, actual);
|
|
|
|
}
|
|
|
|
|
2012-10-12 21:13:39 +02:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHandlePMSuspendDisk(qemuMonitor *mon,
|
|
|
|
virJSONValue *data G_GNUC_UNUSED)
|
2012-10-12 21:13:39 +02:00
|
|
|
{
|
|
|
|
qemuMonitorEmitPMSuspendDisk(mon);
|
|
|
|
}
|
|
|
|
|
2013-07-11 17:07:26 +02:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHandleDeviceDeleted(qemuMonitor *mon, virJSONValue *data)
|
2013-07-11 17:07:26 +02:00
|
|
|
{
|
|
|
|
const char *device;
|
|
|
|
|
|
|
|
if (!(device = virJSONValueObjectGetString(data, "device"))) {
|
2016-09-02 14:52:15 +02:00
|
|
|
VIR_DEBUG("missing device in device deleted event");
|
2013-07-11 17:07:26 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
qemuMonitorEmitDeviceDeleted(mon, device);
|
|
|
|
}
|
|
|
|
|
2014-09-17 13:07:50 -04:00
|
|
|
|
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHandleNicRxFilterChanged(qemuMonitor *mon, virJSONValue *data)
|
2014-09-17 13:07:50 -04:00
|
|
|
{
|
|
|
|
const char *name;
|
|
|
|
|
|
|
|
if (!(name = virJSONValueObjectGetString(data, "name"))) {
|
|
|
|
VIR_WARN("missing device in NIC_RX_FILTER_CHANGED event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
qemuMonitorEmitNicRxFilterChanged(mon, name);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-11-13 14:09:39 +01:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHandleSerialChange(qemuMonitor *mon,
|
|
|
|
virJSONValue *data)
|
2014-11-13 14:09:39 +01:00
|
|
|
{
|
|
|
|
const char *name;
|
|
|
|
bool connected;
|
|
|
|
|
|
|
|
if (!(name = virJSONValueObjectGetString(data, "id"))) {
|
|
|
|
VIR_WARN("missing device alias in VSERPORT_CHANGE event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetBoolean(data, "open", &connected) < 0) {
|
|
|
|
VIR_WARN("missing port state for '%s' in VSERPORT_CHANGE event", name);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
qemuMonitorEmitSerialChange(mon, name, connected);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-05-28 13:35:06 +02:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHandleSpiceMigrated(qemuMonitor *mon,
|
|
|
|
virJSONValue *data G_GNUC_UNUSED)
|
2015-05-28 13:35:06 +02:00
|
|
|
{
|
|
|
|
qemuMonitorEmitSpiceMigrated(mon);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-14 18:37:51 +08:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHandleMemoryFailure(qemuMonitor *mon,
|
|
|
|
virJSONValue *data)
|
2020-10-14 18:37:51 +08:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *flagsjson = virJSONValueObjectGetObject(data, "flags");
|
2020-10-14 18:37:51 +08:00
|
|
|
const char *str;
|
|
|
|
int recipient;
|
|
|
|
int action;
|
|
|
|
bool ar = false;
|
|
|
|
bool recursive = false;
|
|
|
|
qemuMonitorEventMemoryFailure mf = {0};
|
|
|
|
|
|
|
|
if (!(str = virJSONValueObjectGetString(data, "recipient"))) {
|
|
|
|
VIR_WARN("missing recipient in memory failure event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
recipient = qemuMonitorMemoryFailureRecipientTypeFromString(str);
|
|
|
|
if (recipient < 0) {
|
|
|
|
VIR_WARN("unknown recipient '%s' in memory_failure event", str);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(str = virJSONValueObjectGetString(data, "action"))) {
|
|
|
|
VIR_WARN("missing action in memory failure event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
action = qemuMonitorMemoryFailureActionTypeFromString(str);
|
|
|
|
if (action < 0) {
|
|
|
|
VIR_WARN("unknown action '%s' in memory_failure event", str);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (flagsjson) {
|
|
|
|
virJSONValueObjectGetBoolean(flagsjson, "action-required", &ar);
|
|
|
|
virJSONValueObjectGetBoolean(flagsjson, "recursive", &recursive);
|
|
|
|
}
|
|
|
|
|
|
|
|
mf.recipient = recipient;
|
|
|
|
mf.action = action;
|
|
|
|
mf.action_required = ar;
|
|
|
|
mf.recursive = recursive;
|
|
|
|
qemuMonitorEmitMemoryFailure(mon, &mf);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-05-28 13:35:52 +02:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHandleMigrationStatus(qemuMonitor *mon,
|
|
|
|
virJSONValue *data)
|
2015-05-28 13:35:52 +02:00
|
|
|
{
|
|
|
|
const char *str;
|
|
|
|
int status;
|
|
|
|
|
|
|
|
if (!(str = virJSONValueObjectGetString(data, "status"))) {
|
|
|
|
VIR_WARN("missing status in migration event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((status = qemuMonitorMigrationStatusTypeFromString(str)) == -1) {
|
|
|
|
VIR_WARN("unknown status '%s' in migration event", str);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
qemuMonitorEmitMigrationStatus(mon, status);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-12-08 15:23:35 +01:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHandleMigrationPass(qemuMonitor *mon,
|
|
|
|
virJSONValue *data)
|
2015-12-08 15:23:35 +01:00
|
|
|
{
|
|
|
|
int pass;
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberInt(data, "pass", &pass) < 0) {
|
|
|
|
VIR_WARN("missing dirty-sync-count in migration-pass event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
qemuMonitorEmitMigrationPass(mon, pass);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-04-01 16:41:08 +02:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHandleAcpiOstInfo(qemuMonitor *mon, virJSONValue *data)
|
2016-04-01 16:41:08 +02:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *info;
|
2016-04-01 16:41:08 +02:00
|
|
|
const char *alias;
|
|
|
|
const char *slotType;
|
|
|
|
const char *slot;
|
|
|
|
unsigned int source;
|
|
|
|
unsigned int status;
|
|
|
|
|
|
|
|
if (!(info = virJSONValueObjectGetObject(data, "info")))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
/* optional */
|
|
|
|
alias = virJSONValueObjectGetString(info, "device");
|
|
|
|
|
|
|
|
if (!(slotType = virJSONValueObjectGetString(info, "slot-type")))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (!(slot = virJSONValueObjectGetString(info, "slot")))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberUint(info, "source", &source) < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberUint(info, "status", &status) < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
qemuMonitorEmitAcpiOstInfo(mon, alias, slotType, slot, source, status);
|
|
|
|
return;
|
|
|
|
|
|
|
|
error:
|
|
|
|
VIR_WARN("malformed ACPI_DEVICE_OST event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-02-22 16:52:22 +01:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHandleBlockThreshold(qemuMonitor *mon, virJSONValue *data)
|
2017-02-22 16:52:22 +01:00
|
|
|
{
|
|
|
|
const char *nodename;
|
|
|
|
unsigned long long threshold;
|
|
|
|
unsigned long long excess;
|
|
|
|
|
|
|
|
if (!(nodename = virJSONValueObjectGetString(data, "node-name")))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberUlong(data, "write-threshold", &threshold) < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberUlong(data, "amount-exceeded", &excess) < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
qemuMonitorEmitBlockThreshold(mon, nodename, threshold, excess);
|
|
|
|
return;
|
|
|
|
|
|
|
|
error:
|
|
|
|
VIR_WARN("malformed 'BLOCK_WRITE_THRESHOLD' event");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-11-20 08:56:24 -05:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONExtractDumpStats(virJSONValue *result,
|
|
|
|
qemuMonitorDumpStats *ret)
|
2017-11-20 08:56:24 -05:00
|
|
|
{
|
|
|
|
const char *statusstr;
|
|
|
|
|
|
|
|
if (!(statusstr = virJSONValueObjectGetString(result, "status"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("incomplete result, failed to get status"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret->status = qemuMonitorDumpStatusTypeFromString(statusstr);
|
|
|
|
if (ret->status < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("incomplete result, unknown status string '%s'"),
|
|
|
|
statusstr);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberUlong(result, "completed", &ret->completed) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("incomplete result, failed to get completed"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberUlong(result, "total", &ret->total) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("incomplete result, failed to get total"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHandleDumpCompleted(qemuMonitor *mon,
|
|
|
|
virJSONValue *data)
|
2017-11-20 08:56:24 -05:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *result;
|
2017-11-20 08:56:24 -05:00
|
|
|
int status;
|
|
|
|
qemuMonitorDumpStats stats = { 0 };
|
|
|
|
const char *error = NULL;
|
|
|
|
|
|
|
|
if (!(result = virJSONValueObjectGetObject(data, "result"))) {
|
|
|
|
VIR_WARN("missing result in dump completed event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
status = qemuMonitorJSONExtractDumpStats(result, &stats);
|
|
|
|
|
|
|
|
error = virJSONValueObjectGetString(data, "error");
|
|
|
|
|
|
|
|
qemuMonitorEmitDumpCompleted(mon, status, &stats, error);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static void qemuMonitorJSONHandlePRManagerStatusChanged(qemuMonitor *mon,
|
|
|
|
virJSONValue *data)
|
2018-06-27 12:17:59 +02:00
|
|
|
{
|
|
|
|
const char *name;
|
|
|
|
bool connected;
|
|
|
|
|
|
|
|
if (!(name = virJSONValueObjectGetString(data, "id"))) {
|
|
|
|
VIR_WARN("missing pr-manager alias in PR_MANAGER_STATUS_CHANGED event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetBoolean(data, "connected", &connected) < 0) {
|
|
|
|
VIR_WARN("missing connected state for %s "
|
|
|
|
"in PR_MANAGER_STATUS_CHANGED event", name);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
qemuMonitorEmitPRManagerStatusChanged(mon, name, connected);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static void qemuMonitorJSONHandleRdmaGidStatusChanged(qemuMonitor *mon,
|
|
|
|
virJSONValue *data)
|
2018-12-24 12:15:12 +02:00
|
|
|
{
|
|
|
|
const char *netdev;
|
|
|
|
bool gid_status;
|
|
|
|
unsigned long long subnet_prefix, interface_id;
|
|
|
|
|
|
|
|
if (!(netdev = virJSONValueObjectGetString(data, "netdev"))) {
|
|
|
|
VIR_WARN("missing netdev in GID_STATUS_CHANGED event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetBoolean(data, "gid-status", &gid_status)) {
|
|
|
|
VIR_WARN("missing gid-status in GID_STATUS_CHANGED event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberUlong(data, "subnet-prefix",
|
|
|
|
&subnet_prefix)) {
|
|
|
|
VIR_WARN("missing subnet-prefix in GID_STATUS_CHANGED event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberUlong(data, "interface-id",
|
|
|
|
&interface_id)) {
|
|
|
|
VIR_WARN("missing interface-id in GID_STATUS_CHANGED event");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
qemuMonitorEmitRdmaGidStatusChanged(mon, netdev, gid_status, subnet_prefix,
|
|
|
|
interface_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
qemu: support Panic Crashloaded event handling
Pvpanic device supports bit 1 as crashloaded event, it means that
guest actually panicked and run kexec to handle error by guest side.
Handle crashloaded as a lifecyle event in libvirt.
Test case:
Guest side:
before testing, we need make sure kdump is enabled,
1, build new pvpanic driver (with commit from upstream
e0b9a42735f2672ca2764cfbea6e55a81098d5ba
191941692a3d1b6a9614502b279be062926b70f5)
2, insmod new kmod
3, enable crash_kexec_post_notifiers,
# echo 1 > /sys/module/kernel/parameters/crash_kexec_post_notifiers
4, trigger kernel panic
# echo 1 > /proc/sys/kernel/sysrq
# echo c > /proc/sysrq-trigger
Host side:
1, build new qemu with pvpanic patches (with commit from upstream
600d7b47e8f5085919fd1d1157f25950ea8dbc11
7dc58deea79a343ac3adc5cadb97215086054c86)
2, build libvirt with this patch
3, handle lifecycle event and trigger guest side panic
# virsh event stretch --event lifecycle
event 'lifecycle' for domain stretch: Crashed Crashloaded
events received: 1
Reviewed-by: Daniel P. Berrangé <berrange@redhat.com>
Signed-off-by: zhenwei pi <pizhenwei@bytedance.com>
2020-02-04 15:41:00 +08:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHandleGuestCrashloaded(qemuMonitor *mon,
|
|
|
|
virJSONValue *data)
|
qemu: support Panic Crashloaded event handling
Pvpanic device supports bit 1 as crashloaded event, it means that
guest actually panicked and run kexec to handle error by guest side.
Handle crashloaded as a lifecyle event in libvirt.
Test case:
Guest side:
before testing, we need make sure kdump is enabled,
1, build new pvpanic driver (with commit from upstream
e0b9a42735f2672ca2764cfbea6e55a81098d5ba
191941692a3d1b6a9614502b279be062926b70f5)
2, insmod new kmod
3, enable crash_kexec_post_notifiers,
# echo 1 > /sys/module/kernel/parameters/crash_kexec_post_notifiers
4, trigger kernel panic
# echo 1 > /proc/sys/kernel/sysrq
# echo c > /proc/sysrq-trigger
Host side:
1, build new qemu with pvpanic patches (with commit from upstream
600d7b47e8f5085919fd1d1157f25950ea8dbc11
7dc58deea79a343ac3adc5cadb97215086054c86)
2, build libvirt with this patch
3, handle lifecycle event and trigger guest side panic
# virsh event stretch --event lifecycle
event 'lifecycle' for domain stretch: Crashed Crashloaded
events received: 1
Reviewed-by: Daniel P. Berrangé <berrange@redhat.com>
Signed-off-by: zhenwei pi <pizhenwei@bytedance.com>
2020-02-04 15:41:00 +08:00
|
|
|
{
|
|
|
|
VIR_DEBUG("qemuMonitorJSONHandleGuestCrashloaded event, mon %p, data %p", mon, data);
|
|
|
|
|
|
|
|
qemuMonitorEmitGuestCrashloaded(mon);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-03-09 21:24:04 +01:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONHumanCommand(qemuMonitor *mon,
|
2019-09-19 17:13:31 +02:00
|
|
|
const char *cmd_str,
|
|
|
|
char **reply_str)
|
2011-03-09 21:24:04 +01:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = NULL;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *obj;
|
2011-03-09 21:24:04 +01:00
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
cmd = qemuMonitorJSONMakeCommand("human-monitor-command",
|
|
|
|
"s:command-line", cmd_str,
|
|
|
|
NULL);
|
|
|
|
|
2019-09-19 17:13:31 +02:00
|
|
|
if (!cmd || qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2011-03-09 21:24:04 +01:00
|
|
|
goto cleanup;
|
|
|
|
|
2012-09-06 16:28:53 +01:00
|
|
|
if (qemuMonitorJSONHasError(reply, "CommandNotFound")) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
|
|
|
|
_("Human monitor command is not available to run %s"),
|
|
|
|
cmd_str);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2011-03-09 21:24:04 +01:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply))
|
|
|
|
goto cleanup;
|
|
|
|
|
2016-05-03 10:55:41 +02:00
|
|
|
obj = virJSONValueObjectGet(reply, "return");
|
2011-03-09 21:24:04 +01:00
|
|
|
|
|
|
|
if (reply_str) {
|
|
|
|
const char *data;
|
|
|
|
|
2013-05-20 11:23:13 +02:00
|
|
|
data = virJSONValueGetString(obj);
|
2019-10-20 13:49:46 +02:00
|
|
|
*reply_str = g_strdup(NULLSTR_EMPTY(data));
|
2011-03-09 21:24:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2011-03-09 21:24:04 +01:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-02-12 13:45:20 +00:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONSetCapabilities(qemuMonitor *mon)
|
2010-02-12 13:45:20 +00:00
|
|
|
{
|
2021-03-19 23:28:38 +01:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("qmp_capabilities",
|
|
|
|
NULL)))
|
2010-02-12 13:45:20 +00:00
|
|
|
return -1;
|
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2021-03-19 23:28:38 +01:00
|
|
|
return -1;
|
2010-02-12 13:45:20 +00:00
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
2021-03-19 23:28:38 +01:00
|
|
|
return -1;
|
2010-02-12 13:45:20 +00:00
|
|
|
|
2021-03-19 23:28:38 +01:00
|
|
|
return 0;
|
2010-02-12 13:45:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-11-03 13:59:18 -05:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONStartCPUs(qemuMonitor *mon)
|
2009-11-03 13:59:18 -05:00
|
|
|
{
|
|
|
|
int ret;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = qemuMonitorJSONMakeCommand("cont", NULL);
|
|
|
|
virJSONValue *reply = NULL;
|
Convert 'int i' to 'size_t i' in src/qemu files
Convert the type of loop iterators named 'i', 'j', k',
'ii', 'jj', 'kk', to be 'size_t' instead of 'int' or
'unsigned int', also santizing 'ii', 'jj', 'kk' to use
the normal 'i', 'j', 'k' naming
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2013-07-08 15:09:33 +01:00
|
|
|
size_t i = 0;
|
|
|
|
int timeout = 3;
|
2009-11-03 13:59:18 -05:00
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2011-01-13 12:52:23 -07:00
|
|
|
do {
|
|
|
|
ret = qemuMonitorJSONCommand(mon, cmd, &reply);
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2011-01-13 12:52:23 -07:00
|
|
|
if (ret != 0)
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* If no error, we're done */
|
|
|
|
if ((ret = qemuMonitorJSONCheckError(cmd, reply)) == 0)
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* If error class is not MigrationExpected, we're done.
|
|
|
|
* Otherwise try 'cont' cmd again */
|
|
|
|
if (!qemuMonitorJSONHasError(reply, "MigrationExpected"))
|
|
|
|
break;
|
|
|
|
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
reply = NULL;
|
2019-10-02 18:01:11 +01:00
|
|
|
g_usleep(250000);
|
2011-01-13 12:52:23 -07:00
|
|
|
} while (++i <= timeout);
|
2009-11-03 13:59:18 -05:00
|
|
|
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONStopCPUs(qemuMonitor *mon)
|
2009-11-03 13:59:18 -05:00
|
|
|
{
|
2016-05-03 11:29:11 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = qemuMonitorJSONMakeCommand("stop", NULL);
|
|
|
|
virJSONValue *reply = NULL;
|
2009-11-03 13:59:18 -05:00
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2009-11-03 13:59:18 -05:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-05-05 13:50:25 +02:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetStatus(qemuMonitor *mon,
|
2011-09-27 11:42:04 +02:00
|
|
|
bool *running,
|
|
|
|
virDomainPausedReason *reason)
|
2011-05-05 13:50:25 +02:00
|
|
|
{
|
2016-05-03 11:29:11 +02:00
|
|
|
int ret = -1;
|
2011-09-27 11:42:04 +02:00
|
|
|
const char *status;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *data;
|
2011-05-05 13:50:25 +02:00
|
|
|
|
2011-09-27 11:42:04 +02:00
|
|
|
if (reason)
|
|
|
|
*reason = VIR_DOMAIN_PAUSED_UNKNOWN;
|
|
|
|
|
2011-05-05 13:50:25 +02:00
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-status", NULL)))
|
|
|
|
return -1;
|
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2011-05-05 13:50:25 +02:00
|
|
|
goto cleanup;
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_OBJECT) < 0)
|
2016-05-03 11:29:11 +02:00
|
|
|
goto cleanup;
|
2011-05-05 13:50:25 +02:00
|
|
|
|
2016-11-21 00:10:06 +01:00
|
|
|
data = virJSONValueObjectGetObject(reply, "return");
|
2011-05-05 13:50:25 +02:00
|
|
|
|
|
|
|
if (virJSONValueObjectGetBoolean(data, "running", running) < 0) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-status reply was missing running state"));
|
2011-05-05 13:50:25 +02:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2011-09-27 11:42:04 +02:00
|
|
|
if ((status = virJSONValueObjectGetString(data, "status"))) {
|
|
|
|
if (!*running && reason)
|
|
|
|
*reason = qemuMonitorVMStatusToPausedReason(status);
|
|
|
|
} else if (!*running) {
|
|
|
|
VIR_DEBUG("query-status reply was missing status details");
|
|
|
|
}
|
|
|
|
|
2011-05-05 13:50:25 +02:00
|
|
|
ret = 0;
|
|
|
|
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2011-05-05 13:50:25 +02:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONSystemPowerdown(qemuMonitor *mon)
|
2009-11-03 13:59:18 -05:00
|
|
|
{
|
2016-05-03 11:29:11 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = qemuMonitorJSONMakeCommand("system_powerdown", NULL);
|
|
|
|
virJSONValue *reply = NULL;
|
2009-11-03 13:59:18 -05:00
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2009-11-03 13:59:18 -05:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONSetLink(qemuMonitor *mon,
|
2011-09-06 16:18:57 +08:00
|
|
|
const char *name,
|
2014-05-31 21:22:30 -03:00
|
|
|
virDomainNetInterfaceLinkState state)
|
2011-09-06 16:18:57 +08:00
|
|
|
{
|
2016-05-03 11:29:11 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *cmd = qemuMonitorJSONMakeCommand("set_link",
|
2011-09-06 16:18:57 +08:00
|
|
|
"s:name", name,
|
|
|
|
"b:up", state != VIR_DOMAIN_NET_INTERFACE_LINK_STATE_DOWN,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2011-09-06 16:18:57 +08:00
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2011-09-06 16:18:57 +08:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONSystemReset(qemuMonitor *mon)
|
2011-06-15 17:49:58 +01:00
|
|
|
{
|
2016-05-03 11:29:11 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = qemuMonitorJSONMakeCommand("system_reset", NULL);
|
|
|
|
virJSONValue *reply = NULL;
|
2011-06-15 17:49:58 +01:00
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2011-06-15 17:49:58 +01:00
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2011-06-15 17:49:58 +01:00
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2011-06-15 17:49:58 +01:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-04-04 16:45:05 +02:00
|
|
|
/**
|
|
|
|
* qemuMonitorJSONExtractCPUS390Info:
|
|
|
|
* @jsoncpu: pointer to a single JSON cpu entry
|
|
|
|
* @cpu: pointer to a single cpu entry
|
|
|
|
*
|
|
|
|
* Derive the legacy cpu info 'halted' information
|
|
|
|
* from the more accurate s390 cpu state. @cpu is
|
|
|
|
* modified only on success.
|
|
|
|
*
|
|
|
|
* Note: the 'uninitialized' s390 cpu state can't be
|
|
|
|
* mapped to halted yes/no.
|
|
|
|
*
|
|
|
|
* A s390 cpu entry could look like this
|
|
|
|
* { "arch": "s390",
|
|
|
|
* "cpu-index": 0,
|
|
|
|
* "qom-path": "/machine/unattached/device[0]",
|
|
|
|
* "thread_id": 3081,
|
|
|
|
* "cpu-state": "operating" }
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONExtractCPUS390Info(virJSONValue *jsoncpu,
|
2018-04-04 16:45:05 +02:00
|
|
|
struct qemuMonitorQueryCpusEntry *cpu)
|
|
|
|
{
|
|
|
|
const char *cpu_state = virJSONValueObjectGetString(jsoncpu, "cpu-state");
|
|
|
|
|
|
|
|
if (STREQ_NULLABLE(cpu_state, "operating") ||
|
|
|
|
STREQ_NULLABLE(cpu_state, "load"))
|
|
|
|
cpu->halted = false;
|
|
|
|
else if (STREQ_NULLABLE(cpu_state, "stopped") ||
|
|
|
|
STREQ_NULLABLE(cpu_state, "check-stop"))
|
|
|
|
cpu->halted = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* qemuMonitorJSONExtractCPUInfo:
|
|
|
|
* @data: JSON response data
|
|
|
|
* @entries: filled with detected cpu entries on success
|
|
|
|
* @nentries: number of entries returned
|
|
|
|
* @fast: true if this is a response from query-cpus-fast
|
2016-07-28 10:33:10 +02:00
|
|
|
*
|
2018-04-04 16:45:05 +02:00
|
|
|
* The JSON response @data will have the following format
|
|
|
|
* in case @fast == false
|
2016-07-28 10:33:10 +02:00
|
|
|
* [{ "arch": "x86",
|
|
|
|
* "current": true,
|
|
|
|
* "CPU": 0,
|
|
|
|
* "qom_path": "/machine/unattached/device[0]",
|
|
|
|
* "pc": -2130415978,
|
|
|
|
* "halted": true,
|
2018-04-04 16:45:05 +02:00
|
|
|
* "thread_id": 2631237,
|
|
|
|
* ...},
|
|
|
|
* {...}
|
|
|
|
* ]
|
|
|
|
* and for @fast == true
|
|
|
|
* [{ "arch": "x86",
|
|
|
|
* "cpu-index": 0,
|
|
|
|
* "props": {
|
|
|
|
* "core-id": 0,
|
|
|
|
* "thread-id": 0,
|
|
|
|
* "socket-id": 0
|
|
|
|
* },
|
|
|
|
* "qom-path": "/machine/unattached/device[0]",
|
|
|
|
* "thread-id": 2631237,
|
|
|
|
* ...},
|
|
|
|
* {...}
|
|
|
|
* ]
|
|
|
|
* or for s390
|
|
|
|
* [{ "arch": "s390",
|
|
|
|
* "cpu-index": 0,
|
|
|
|
* "props": {
|
|
|
|
* "core-id": 0
|
|
|
|
* },
|
|
|
|
* "qom-path": "/machine/unattached/device[0]",
|
|
|
|
* "thread-id": 1237,
|
|
|
|
* "cpu-state": "operating",
|
|
|
|
* ...},
|
2016-07-28 10:33:10 +02:00
|
|
|
* {...}
|
|
|
|
* ]
|
2018-07-09 12:32:39 +02:00
|
|
|
*
|
|
|
|
* Note that since QEMU 2.13.0 the "arch" output member of the
|
|
|
|
* "query-cpus-fast" command is replaced by "target".
|
2010-01-22 14:52:05 +00:00
|
|
|
*/
|
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONExtractCPUInfo(virJSONValue *data,
|
2016-08-01 13:44:25 +02:00
|
|
|
struct qemuMonitorQueryCpusEntry **entries,
|
2018-04-04 16:45:03 +02:00
|
|
|
size_t *nentries,
|
|
|
|
bool fast)
|
2010-01-22 14:52:05 +00:00
|
|
|
{
|
2018-07-09 12:53:51 +02:00
|
|
|
const char *arch = NULL;
|
2016-08-01 13:44:25 +02:00
|
|
|
struct qemuMonitorQueryCpusEntry *cpus = NULL;
|
2010-01-22 14:52:05 +00:00
|
|
|
int ret = -1;
|
Convert 'int i' to 'size_t i' in src/qemu files
Convert the type of loop iterators named 'i', 'j', k',
'ii', 'jj', 'kk', to be 'size_t' instead of 'int' or
'unsigned int', also santizing 'ii', 'jj', 'kk' to use
the normal 'i', 'j', 'k' naming
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2013-07-08 15:09:33 +01:00
|
|
|
size_t i;
|
2018-04-19 17:29:02 -04:00
|
|
|
size_t ncpus;
|
2010-01-22 14:52:05 +00:00
|
|
|
|
2018-04-19 17:29:02 -04:00
|
|
|
if ((ncpus = virJSONValueArraySize(data)) == 0)
|
2016-08-01 13:44:25 +02:00
|
|
|
return -2;
|
2010-01-22 14:52:05 +00:00
|
|
|
|
2020-10-05 12:26:34 +02:00
|
|
|
cpus = g_new0(struct qemuMonitorQueryCpusEntry, ncpus);
|
2010-01-22 14:52:05 +00:00
|
|
|
|
2013-05-21 15:21:20 +08:00
|
|
|
for (i = 0; i < ncpus; i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *entry = virJSONValueArrayGet(data, i);
|
2016-11-21 14:57:54 +01:00
|
|
|
int cpuid = -1;
|
2016-08-01 13:44:25 +02:00
|
|
|
int thread = 0;
|
2016-10-13 13:42:44 +02:00
|
|
|
bool halted = false;
|
2016-07-28 10:33:10 +02:00
|
|
|
const char *qom_path;
|
2010-01-22 14:52:05 +00:00
|
|
|
if (!entry) {
|
2016-08-01 13:44:25 +02:00
|
|
|
ret = -2;
|
2010-01-22 14:52:05 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2016-08-01 13:44:25 +02:00
|
|
|
/* Some older qemu versions don't report the thread_id so treat this as
|
2018-04-04 16:45:03 +02:00
|
|
|
* non-fatal, simply returning no data.
|
|
|
|
* The return data of query-cpus-fast has different field names
|
|
|
|
*/
|
|
|
|
if (fast) {
|
2018-07-09 12:32:39 +02:00
|
|
|
if (!(arch = virJSONValueObjectGetString(entry, "target")))
|
|
|
|
arch = virJSONValueObjectGetString(entry, "arch");
|
2018-04-04 16:45:03 +02:00
|
|
|
ignore_value(virJSONValueObjectGetNumberInt(entry, "cpu-index", &cpuid));
|
|
|
|
ignore_value(virJSONValueObjectGetNumberInt(entry, "thread-id", &thread));
|
|
|
|
qom_path = virJSONValueObjectGetString(entry, "qom-path");
|
|
|
|
} else {
|
2018-07-09 12:53:51 +02:00
|
|
|
arch = virJSONValueObjectGetString(entry, "arch");
|
2018-04-04 16:45:03 +02:00
|
|
|
ignore_value(virJSONValueObjectGetNumberInt(entry, "CPU", &cpuid));
|
|
|
|
ignore_value(virJSONValueObjectGetNumberInt(entry, "thread_id", &thread));
|
|
|
|
ignore_value(virJSONValueObjectGetBoolean(entry, "halted", &halted));
|
|
|
|
qom_path = virJSONValueObjectGetString(entry, "qom_path");
|
|
|
|
}
|
2010-01-22 14:52:05 +00:00
|
|
|
|
2016-11-21 14:57:54 +01:00
|
|
|
cpus[i].qemu_id = cpuid;
|
2016-08-01 13:44:25 +02:00
|
|
|
cpus[i].tid = thread;
|
2016-10-13 13:42:44 +02:00
|
|
|
cpus[i].halted = halted;
|
2019-10-20 13:49:46 +02:00
|
|
|
cpus[i].qom_path = g_strdup(qom_path);
|
2018-04-04 16:45:05 +02:00
|
|
|
|
|
|
|
/* process optional architecture-specific data */
|
2019-03-01 11:29:51 +01:00
|
|
|
if (STREQ_NULLABLE(arch, "s390") || STREQ_NULLABLE(arch, "s390x"))
|
2018-07-09 12:53:51 +02:00
|
|
|
qemuMonitorJSONExtractCPUS390Info(entry, cpus + i);
|
2010-01-22 14:52:05 +00:00
|
|
|
}
|
|
|
|
|
2019-10-16 13:43:18 +02:00
|
|
|
*entries = g_steal_pointer(&cpus);
|
2016-08-01 13:44:25 +02:00
|
|
|
*nentries = ncpus;
|
|
|
|
ret = 0;
|
2010-01-22 14:52:05 +00:00
|
|
|
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2016-08-01 13:44:25 +02:00
|
|
|
qemuMonitorQueryCpusFree(cpus, ncpus);
|
2010-01-22 14:52:05 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-08-01 13:44:25 +02:00
|
|
|
/**
|
|
|
|
* qemuMonitorJSONQueryCPUs:
|
|
|
|
*
|
|
|
|
* @mon: monitor object
|
|
|
|
* @entries: filled with detected entries on success
|
|
|
|
* @nentries: number of entries returned
|
2018-04-04 16:45:03 +02:00
|
|
|
* @force: force exit on error
|
|
|
|
* @fast: use query-cpus-fast
|
2016-08-01 13:44:25 +02:00
|
|
|
*
|
|
|
|
* Queries qemu for cpu-related information. Failure to execute the command or
|
|
|
|
* extract results does not produce an error as libvirt can continue without
|
2018-04-04 16:45:03 +02:00
|
|
|
* this information, unless the caller has specified @force == true.
|
2016-08-01 13:44:25 +02:00
|
|
|
*
|
2017-04-22 22:06:20 +03:00
|
|
|
* Returns 0 on success, -1 on a fatal error (oom ...) and -2 if the
|
2016-08-01 13:44:25 +02:00
|
|
|
* query failed gracefully.
|
|
|
|
*/
|
2016-07-08 15:36:27 +02:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONQueryCPUs(qemuMonitor *mon,
|
2016-08-01 13:44:25 +02:00
|
|
|
struct qemuMonitorQueryCpusEntry **entries,
|
2016-12-04 18:53:03 +01:00
|
|
|
size_t *nentries,
|
2018-04-04 16:45:03 +02:00
|
|
|
bool force,
|
|
|
|
bool fast)
|
2009-11-03 13:59:18 -05:00
|
|
|
{
|
2016-05-03 11:29:11 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *data;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2018-04-04 16:45:03 +02:00
|
|
|
if (fast)
|
|
|
|
cmd = qemuMonitorJSONMakeCommand("query-cpus-fast", NULL);
|
|
|
|
else
|
|
|
|
cmd = qemuMonitorJSONMakeCommand("query-cpus", NULL);
|
|
|
|
|
2009-11-03 13:59:18 -05:00
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-12-04 18:53:03 +01:00
|
|
|
if (force && qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2016-08-01 13:44:25 +02:00
|
|
|
if (!(data = virJSONValueObjectGetArray(reply, "return"))) {
|
|
|
|
ret = -2;
|
2016-05-03 11:29:11 +02:00
|
|
|
goto cleanup;
|
2016-08-01 13:44:25 +02:00
|
|
|
}
|
|
|
|
|
2018-04-04 16:45:03 +02:00
|
|
|
ret = qemuMonitorJSONExtractCPUInfo(data, entries, nentries, fast);
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
cleanup:
|
2009-11-03 13:59:18 -05:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-12-10 15:31:23 +01:00
|
|
|
/**
|
|
|
|
* Loads correct video memory size values from QEMU and update the video
|
|
|
|
* definition.
|
|
|
|
*
|
|
|
|
* Return 0 on success, -1 on failure and set proper error message.
|
|
|
|
*/
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONUpdateVideoMemorySize(qemuMonitor *mon,
|
|
|
|
virDomainVideoDef *video,
|
2014-12-10 15:31:23 +01:00
|
|
|
char *path)
|
|
|
|
{
|
|
|
|
qemuMonitorJSONObjectProperty prop = {
|
|
|
|
QEMU_MONITOR_OBJECT_PROPERTY_ULONG,
|
|
|
|
{0}
|
|
|
|
};
|
|
|
|
|
|
|
|
switch (video->type) {
|
|
|
|
case VIR_DOMAIN_VIDEO_TYPE_VGA:
|
|
|
|
if (qemuMonitorJSONGetObjectProperty(mon, path, "vgamem_mb", &prop) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
2015-02-25 14:12:33 +01:00
|
|
|
_("QOM Object '%s' has no property 'vgamem_mb'"),
|
2014-12-10 15:31:23 +01:00
|
|
|
path);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
video->vram = prop.val.ul * 1024;
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_VIDEO_TYPE_QXL:
|
|
|
|
if (qemuMonitorJSONGetObjectProperty(mon, path, "vram_size", &prop) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
2015-02-25 14:12:33 +01:00
|
|
|
_("QOM Object '%s' has no property 'vram_size'"),
|
2014-12-10 15:31:23 +01:00
|
|
|
path);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
video->vram = prop.val.ul / 1024;
|
2016-02-23 17:04:19 +01:00
|
|
|
|
2014-12-10 15:31:23 +01:00
|
|
|
if (qemuMonitorJSONGetObjectProperty(mon, path, "ram_size", &prop) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
2015-02-25 14:12:33 +01:00
|
|
|
_("QOM Object '%s' has no property 'ram_size'"),
|
2014-12-10 15:31:23 +01:00
|
|
|
path);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
video->ram = prop.val.ul / 1024;
|
|
|
|
if (qemuMonitorJSONGetObjectProperty(mon, path, "vgamem_mb", &prop) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
2015-02-25 14:12:33 +01:00
|
|
|
_("QOM Object '%s' has no property 'vgamem_mb'"),
|
2014-12-10 15:31:23 +01:00
|
|
|
path);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
video->vgamem = prop.val.ul * 1024;
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_VIDEO_TYPE_VMVGA:
|
|
|
|
if (qemuMonitorJSONGetObjectProperty(mon, path, "vgamem_mb", &prop) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
2015-02-25 14:12:33 +01:00
|
|
|
_("QOM Object '%s' has no property 'vgamem_mb'"),
|
2014-12-10 15:31:23 +01:00
|
|
|
path);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
video->vram = prop.val.ul * 1024;
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_VIDEO_TYPE_CIRRUS:
|
|
|
|
case VIR_DOMAIN_VIDEO_TYPE_XEN:
|
|
|
|
case VIR_DOMAIN_VIDEO_TYPE_VBOX:
|
|
|
|
case VIR_DOMAIN_VIDEO_TYPE_LAST:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-02-23 17:04:19 +01:00
|
|
|
/**
|
|
|
|
* Loads correct video vram64 size value from QEMU and update the video
|
|
|
|
* definition.
|
|
|
|
*
|
|
|
|
* Return 0 on success, -1 on failure and set proper error message.
|
|
|
|
*/
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONUpdateVideoVram64Size(qemuMonitor *mon,
|
|
|
|
virDomainVideoDef *video,
|
2016-02-23 17:04:19 +01:00
|
|
|
char *path)
|
|
|
|
{
|
|
|
|
qemuMonitorJSONObjectProperty prop = {
|
|
|
|
QEMU_MONITOR_OBJECT_PROPERTY_ULONG,
|
|
|
|
{0}
|
|
|
|
};
|
|
|
|
|
|
|
|
switch (video->type) {
|
|
|
|
case VIR_DOMAIN_VIDEO_TYPE_QXL:
|
|
|
|
if (video->vram64 != 0) {
|
|
|
|
if (qemuMonitorJSONGetObjectProperty(mon, path,
|
|
|
|
"vram64_size_mb", &prop) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("QOM Object '%s' has no property 'vram64_size_mb'"),
|
|
|
|
path);
|
|
|
|
return -1;
|
|
|
|
}
|
2016-04-13 10:13:16 +02:00
|
|
|
video->vram64 = prop.val.ul * 1024;
|
2016-02-23 17:04:19 +01:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_VIDEO_TYPE_VGA:
|
|
|
|
case VIR_DOMAIN_VIDEO_TYPE_VMVGA:
|
|
|
|
case VIR_DOMAIN_VIDEO_TYPE_CIRRUS:
|
|
|
|
case VIR_DOMAIN_VIDEO_TYPE_XEN:
|
|
|
|
case VIR_DOMAIN_VIDEO_TYPE_VBOX:
|
|
|
|
case VIR_DOMAIN_VIDEO_TYPE_LAST:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-05-27 13:42:32 +02:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetBalloonInfo(qemuMonitor *mon,
|
2015-05-27 13:42:32 +02:00
|
|
|
unsigned long long *currmem)
|
2009-11-03 13:59:18 -05:00
|
|
|
{
|
2016-05-03 11:29:11 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *data;
|
2016-05-03 11:29:11 +02:00
|
|
|
unsigned long long mem;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = qemuMonitorJSONMakeCommand("query-balloon",
|
2009-11-03 13:59:18 -05:00
|
|
|
NULL);
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *reply = NULL;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
|
|
|
*currmem = 0;
|
|
|
|
|
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
/* See if balloon soft-failed */
|
|
|
|
if (qemuMonitorJSONHasError(reply, "DeviceNotActive") ||
|
|
|
|
qemuMonitorJSONHasError(reply, "KVMMissingCap")) {
|
|
|
|
ret = 0;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
/* See if any other fatal error occurred */
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_OBJECT) < 0)
|
2016-05-03 11:29:11 +02:00
|
|
|
goto cleanup;
|
2010-01-22 13:22:53 +00:00
|
|
|
|
2016-11-21 00:10:06 +01:00
|
|
|
data = virJSONValueObjectGetObject(reply, "return");
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
if (virJSONValueObjectGetNumberUlong(data, "actual", &mem) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("info balloon reply was missing balloon data"));
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
}
|
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
*currmem = (mem/1024);
|
|
|
|
ret = 1;
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2009-11-03 13:59:18 -05:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-07-11 19:18:48 -04:00
|
|
|
/* Process the balloon driver statistics. The request and data returned
|
|
|
|
* will be as follows (although the 'child[#]' entry will differ based on
|
|
|
|
* where it's run).
|
|
|
|
*
|
|
|
|
* { "execute": "qom-get","arguments": \
|
|
|
|
* { "path": "/machine/i440fx/pci.0/child[7]","property": "guest-stats"} }
|
|
|
|
*
|
|
|
|
* {"return": {"stats": \
|
|
|
|
* {"stat-swap-out": 0,
|
|
|
|
* "stat-free-memory": 686350336,
|
|
|
|
* "stat-minor-faults": 697283,
|
|
|
|
* "stat-major-faults": 951,
|
|
|
|
* "stat-total-memory": 1019924480,
|
|
|
|
* "stat-swap-in": 0},
|
|
|
|
* "last-update": 1371221540}}
|
|
|
|
*
|
|
|
|
* A value in "stats" can be -1 indicating it's never been collected/stored.
|
|
|
|
* The 'last-update' value could be used in the future in order to determine
|
|
|
|
* rates and/or whether data has been collected since a previous cycle.
|
|
|
|
* It's currently unused.
|
|
|
|
*/
|
2017-11-03 13:09:47 +01:00
|
|
|
#define GET_BALLOON_STATS(OBJECT, FIELD, TAG, DIVISOR) \
|
|
|
|
if (virJSONValueObjectHasKey(OBJECT, FIELD) && \
|
|
|
|
(got < nr_stats)) { \
|
|
|
|
if (virJSONValueObjectGetNumberUlong(OBJECT, FIELD, &mem) < 0) { \
|
|
|
|
VIR_DEBUG("Failed to get '%s' value", FIELD); \
|
|
|
|
} else { \
|
|
|
|
/* Not being collected? No point in providing bad data */ \
|
|
|
|
if (mem != -1UL) { \
|
|
|
|
stats[got].tag = TAG; \
|
|
|
|
stats[got].val = mem / DIVISOR; \
|
|
|
|
got++; \
|
|
|
|
} \
|
|
|
|
} \
|
2013-07-11 19:18:48 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONGetMemoryStats(qemuMonitor *mon,
|
2013-07-11 19:18:48 -04:00
|
|
|
char *balloonpath,
|
2010-04-12 12:31:15 +01:00
|
|
|
virDomainMemoryStatPtr stats,
|
|
|
|
unsigned int nr_stats)
|
|
|
|
{
|
2016-05-03 11:29:11 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = NULL;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *data;
|
|
|
|
virJSONValue *statsdata;
|
2013-07-11 19:18:48 -04:00
|
|
|
unsigned long long mem;
|
|
|
|
int got = 0;
|
2010-04-12 12:31:15 +01:00
|
|
|
|
2013-07-11 19:18:48 -04:00
|
|
|
ret = qemuMonitorJSONGetBalloonInfo(mon, &mem);
|
|
|
|
if (ret == 1 && (got < nr_stats)) {
|
|
|
|
stats[got].tag = VIR_DOMAIN_MEMORY_STAT_ACTUAL_BALLOON;
|
|
|
|
stats[got].val = mem;
|
|
|
|
got++;
|
|
|
|
}
|
2010-04-12 12:31:15 +01:00
|
|
|
|
2013-07-11 19:18:48 -04:00
|
|
|
if (!balloonpath)
|
|
|
|
goto cleanup;
|
2010-04-12 12:31:15 +01:00
|
|
|
|
2013-07-11 19:18:48 -04:00
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("qom-get",
|
|
|
|
"s:path", balloonpath,
|
|
|
|
"s:property", "guest-stats",
|
|
|
|
NULL)))
|
|
|
|
goto cleanup;
|
2010-04-12 12:31:15 +01:00
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2014-05-14 09:35:18 +02:00
|
|
|
goto cleanup;
|
2010-04-12 12:31:15 +01:00
|
|
|
|
qemu: simplify json parsing
Rather than grabbing an arbitrary JSON value and then checking
if it has the right type, we might as well request the correct
type to begin with.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONIOProcessEvent)
(qemuMonitorJSONCommandWithFd, qemuMonitorJSONHandleGraphics)
(qemuMonitorJSONGetStatus, qemuMonitorJSONExtractCPUInfo)
(qemuMonitorJSONGetVirtType, qemuMonitorJSONGetBalloonInfo)
(qemuMonitorJSONGetMemoryStats)
(qemuMonitorJSONDevGetBlockExtent)
(qemuMonitorJSONGetOneBlockStatsInfo)
(qemuMonitorJSONGetAllBlockStatsInfo)
(qemuMonitorJSONBlockStatsUpdateCapacityOne)
(qemuMonitorJSONBlockStatsUpdateCapacity)
(qemuMonitorJSONGetBlockExtent)
(qemuMonitorJSONGetMigrationStatusReply)
(qemuMonitorJSONGetDumpGuestMemoryCapability)
(qemuMonitorJSONAddFd, qemuMonitorJSONQueryRxFilterParse)
(qemuMonitorJSONExtractChardevInfo)
(qemuMonitorJSONDiskNameLookupOne)
(qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONGetAllBlockJobInfo)
(qemuMonitorJSONBlockIoThrottleInfo, qemuMonitorJSONGetVersion)
(qemuMonitorJSONGetMachines, qemuMonitorJSONGetCPUDefinitions)
(qemuMonitorJSONGetCommands, qemuMonitorJSONGetEvents)
(qemuMonitorJSONGetKVMState, qemuMonitorJSONGetObjectTypes)
(qemuMonitorJSONGetObjectListPaths)
(qemuMonitorJSONGetObjectProps, qemuMonitorJSONGetTargetArch)
(qemuMonitorJSONGetMigrationCapabilities)
(qemuMonitorJSONGetStringArray, qemuMonitorJSONAttachCharDev)
(qemuMonitorJSONGetCPUx86Data, qemuMonitorJSONGetIOThreads)
(qemuMonitorJSONGetMemoryDeviceInfo): Use shorter idioms.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-06-20 13:40:21 -06:00
|
|
|
if ((data = virJSONValueObjectGetObject(reply, "error"))) {
|
2014-05-14 09:35:18 +02:00
|
|
|
const char *klass = virJSONValueObjectGetString(data, "class");
|
|
|
|
const char *desc = virJSONValueObjectGetString(data, "desc");
|
2010-04-12 12:31:15 +01:00
|
|
|
|
2014-05-14 09:35:18 +02:00
|
|
|
if (STREQ_NULLABLE(klass, "GenericError") &&
|
|
|
|
STREQ_NULLABLE(desc, "guest hasn't updated any stats yet")) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID, "%s",
|
|
|
|
_("the guest hasn't updated any stats yet"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_OBJECT) < 0)
|
2013-07-11 19:18:48 -04:00
|
|
|
goto cleanup;
|
2010-04-12 12:31:15 +01:00
|
|
|
|
2016-11-21 00:10:06 +01:00
|
|
|
data = virJSONValueObjectGetObject(reply, "return");
|
qemu: Parse current balloon value returned by query_balloon
Qemu once supported following memory stats which will returned by
"query_balloon":
stat_put(dict, "actual", actual);
stat_put(dict, "mem_swapped_in", dev->stats[VIRTIO_BALLOON_S_SWAP_IN]);
stat_put(dict, "mem_swapped_out", dev->stats[VIRTIO_BALLOON_S_SWAP_OUT]);
stat_put(dict, "major_page_faults", dev->stats[VIRTIO_BALLOON_S_MAJFLT]);
stat_put(dict, "minor_page_faults", dev->stats[VIRTIO_BALLOON_S_MINFLT]);
stat_put(dict, "free_mem", dev->stats[VIRTIO_BALLOON_S_MEMFREE]);
stat_put(dict, "total_mem", dev->stats[VIRTIO_BALLOON_S_MEMTOT]);
But it later disabled all the stats except "actual" by commit
07b0403dfc2b2ac179ae5b48105096cc2d03375a.
libvirt doesn't parse "actual", so user will always see a empty result
with "virsh dommemstat $domain". Even qemu haven't disabled the stats,
we should support parsing "actual".
2011-06-14 11:21:35 +08:00
|
|
|
|
2013-07-11 19:18:48 -04:00
|
|
|
if (!(statsdata = virJSONValueObjectGet(data, "stats"))) {
|
|
|
|
VIR_DEBUG("data does not include 'stats'");
|
|
|
|
goto cleanup;
|
2010-04-12 12:31:15 +01:00
|
|
|
}
|
|
|
|
|
2016-06-01 20:07:07 +03:00
|
|
|
GET_BALLOON_STATS(statsdata, "stat-swap-in",
|
2013-07-11 19:18:48 -04:00
|
|
|
VIR_DOMAIN_MEMORY_STAT_SWAP_IN, 1024);
|
2016-06-01 20:07:07 +03:00
|
|
|
GET_BALLOON_STATS(statsdata, "stat-swap-out",
|
2013-07-11 19:18:48 -04:00
|
|
|
VIR_DOMAIN_MEMORY_STAT_SWAP_OUT, 1024);
|
2016-06-01 20:07:07 +03:00
|
|
|
GET_BALLOON_STATS(statsdata, "stat-major-faults",
|
2013-07-11 19:18:48 -04:00
|
|
|
VIR_DOMAIN_MEMORY_STAT_MAJOR_FAULT, 1);
|
2016-06-01 20:07:07 +03:00
|
|
|
GET_BALLOON_STATS(statsdata, "stat-minor-faults",
|
2013-07-11 19:18:48 -04:00
|
|
|
VIR_DOMAIN_MEMORY_STAT_MINOR_FAULT, 1);
|
2016-06-01 20:07:07 +03:00
|
|
|
GET_BALLOON_STATS(statsdata, "stat-free-memory",
|
2013-07-11 19:18:48 -04:00
|
|
|
VIR_DOMAIN_MEMORY_STAT_UNUSED, 1024);
|
2016-06-01 20:07:07 +03:00
|
|
|
GET_BALLOON_STATS(statsdata, "stat-total-memory",
|
2013-07-11 19:18:48 -04:00
|
|
|
VIR_DOMAIN_MEMORY_STAT_AVAILABLE, 1024);
|
2016-06-01 20:07:07 +03:00
|
|
|
GET_BALLOON_STATS(statsdata, "stat-available-memory",
|
2016-06-01 20:07:06 +03:00
|
|
|
VIR_DOMAIN_MEMORY_STAT_USABLE, 1024);
|
2016-06-01 20:07:07 +03:00
|
|
|
GET_BALLOON_STATS(data, "last-update",
|
|
|
|
VIR_DOMAIN_MEMORY_STAT_LAST_UPDATE, 1);
|
2018-07-02 13:19:42 +02:00
|
|
|
GET_BALLOON_STATS(statsdata, "stat-disk-caches",
|
|
|
|
VIR_DOMAIN_MEMORY_STAT_DISK_CACHES, 1024);
|
2019-04-28 17:18:05 +08:00
|
|
|
GET_BALLOON_STATS(statsdata, "stat-htlb-pgalloc",
|
|
|
|
VIR_DOMAIN_MEMORY_STAT_HUGETLB_PGALLOC, 1);
|
|
|
|
GET_BALLOON_STATS(statsdata, "stat-htlb-pgfail",
|
|
|
|
VIR_DOMAIN_MEMORY_STAT_HUGETLB_PGFAIL, 1);
|
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
ret = got;
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2010-04-12 12:31:15 +01:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2013-07-11 19:18:48 -04:00
|
|
|
#undef GET_BALLOON_STATS
|
2010-04-12 12:31:15 +01:00
|
|
|
|
|
|
|
|
2013-06-27 11:00:31 -04:00
|
|
|
/*
|
|
|
|
* Using the provided balloonpath, determine if we need to set the
|
|
|
|
* collection interval property to enable statistics gathering.
|
|
|
|
*/
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONSetMemoryStatsPeriod(qemuMonitor *mon,
|
2013-06-27 11:00:31 -04:00
|
|
|
char *balloonpath,
|
|
|
|
int period)
|
|
|
|
{
|
|
|
|
qemuMonitorJSONObjectProperty prop;
|
|
|
|
|
|
|
|
/* Set to the value in memballoon (could enable or disable) */
|
|
|
|
memset(&prop, 0, sizeof(qemuMonitorJSONObjectProperty));
|
|
|
|
prop.type = QEMU_MONITOR_OBJECT_PROPERTY_INT;
|
|
|
|
prop.val.iv = period;
|
|
|
|
if (qemuMonitorJSONSetObjectProperty(mon, balloonpath,
|
|
|
|
"guest-stats-polling-interval",
|
|
|
|
&prop) < 0) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-02-25 10:55:11 +01:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONSetDBusVMStateIdList(qemuMonitor *mon,
|
2020-02-25 10:55:11 +01:00
|
|
|
const char *vmstatepath,
|
2021-02-05 10:48:51 +01:00
|
|
|
const char *idstr)
|
2020-02-25 10:55:11 +01:00
|
|
|
{
|
|
|
|
qemuMonitorJSONObjectProperty prop = {
|
|
|
|
.type = QEMU_MONITOR_OBJECT_PROPERTY_STRING,
|
2021-02-05 10:48:51 +01:00
|
|
|
.val.str = (char *) idstr,
|
2020-02-25 10:55:11 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
return qemuMonitorJSONSetObjectProperty(mon, vmstatepath, "id-list", &prop);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-09-27 09:39:21 -04:00
|
|
|
/* qemuMonitorJSONQueryBlock:
|
|
|
|
* @mon: Monitor pointer
|
|
|
|
*
|
|
|
|
* This helper will attempt to make a "query-block" call and check for
|
|
|
|
* errors before returning with the reply.
|
|
|
|
*
|
|
|
|
* Returns: NULL on error, reply on success
|
|
|
|
*/
|
2021-03-11 08:16:13 +01:00
|
|
|
static virJSONValue *
|
|
|
|
qemuMonitorJSONQueryBlock(qemuMonitor *mon)
|
2011-09-13 15:49:50 +02:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *devices = NULL;
|
2011-09-13 15:49:50 +02:00
|
|
|
|
2016-09-27 09:39:21 -04:00
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-block", NULL)))
|
|
|
|
return NULL;
|
2011-09-13 15:49:50 +02:00
|
|
|
|
2016-09-27 09:39:21 -04:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0 ||
|
2018-03-28 12:45:21 +02:00
|
|
|
qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_ARRAY) < 0)
|
2016-05-03 11:29:11 +02:00
|
|
|
goto cleanup;
|
2011-09-13 15:49:50 +02:00
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
devices = virJSONValueObjectStealArray(reply, "return");
|
2011-09-13 15:49:50 +02:00
|
|
|
|
2016-09-27 09:39:21 -04:00
|
|
|
cleanup:
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return devices;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static virJSONValue *
|
|
|
|
qemuMonitorJSONGetBlockDev(virJSONValue *devices,
|
2016-10-03 14:58:59 -04:00
|
|
|
size_t idx)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *dev = virJSONValueArrayGet(devices, idx);
|
2016-10-03 14:58:59 -04:00
|
|
|
|
2018-03-29 20:30:05 +02:00
|
|
|
if (!dev || virJSONValueGetType(dev) != VIR_JSON_TYPE_OBJECT) {
|
2016-10-03 14:58:59 -04:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-block device entry was not in expected format"));
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
return dev;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-10-03 15:04:32 -04:00
|
|
|
static const char *
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetBlockDevDevice(virJSONValue *dev)
|
2016-10-03 15:04:32 -04:00
|
|
|
{
|
|
|
|
const char *thisdev;
|
|
|
|
|
|
|
|
if (!(thisdev = virJSONValueObjectGetString(dev, "device"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-block device entry was not in expected format"));
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return thisdev;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-08-06 14:38:55 +02:00
|
|
|
static int
|
2020-10-22 19:04:18 +02:00
|
|
|
qemuMonitorJSONBlockInfoAdd(GHashTable *table,
|
2018-08-06 14:38:55 +02:00
|
|
|
struct qemuDomainDiskInfo *info,
|
|
|
|
const char *entryname)
|
|
|
|
{
|
|
|
|
struct qemuDomainDiskInfo *tmp = NULL;
|
|
|
|
int ret = -1;
|
|
|
|
|
2020-10-05 12:26:34 +02:00
|
|
|
tmp = g_new0(struct qemuDomainDiskInfo, 1);
|
2018-08-06 14:38:55 +02:00
|
|
|
|
|
|
|
*tmp = *info;
|
|
|
|
tmp->nodename = NULL;
|
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
if (info->nodename)
|
|
|
|
tmp->nodename = g_strdup(info->nodename);
|
2018-08-06 14:38:55 +02:00
|
|
|
|
|
|
|
if (virHashAddEntry(table, entryname, tmp) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
tmp = NULL;
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
if (tmp)
|
|
|
|
VIR_FREE(tmp->nodename);
|
|
|
|
VIR_FREE(tmp);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONGetBlockInfo(qemuMonitor *mon,
|
2020-10-22 19:04:18 +02:00
|
|
|
GHashTable *table)
|
2016-09-27 09:39:21 -04:00
|
|
|
{
|
|
|
|
int ret = -1;
|
|
|
|
size_t i;
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *devices;
|
2016-09-27 09:39:21 -04:00
|
|
|
|
|
|
|
if (!(devices = qemuMonitorJSONQueryBlock(mon)))
|
|
|
|
return -1;
|
|
|
|
|
2011-09-13 15:49:50 +02:00
|
|
|
for (i = 0; i < virJSONValueArraySize(devices); i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *dev;
|
|
|
|
virJSONValue *image;
|
2018-08-06 14:38:55 +02:00
|
|
|
struct qemuDomainDiskInfo info = { false };
|
2011-09-13 15:49:50 +02:00
|
|
|
const char *thisdev;
|
2012-01-19 17:58:58 +01:00
|
|
|
const char *status;
|
2018-08-06 14:38:55 +02:00
|
|
|
const char *qdev;
|
2011-09-13 15:49:50 +02:00
|
|
|
|
2016-10-03 14:58:59 -04:00
|
|
|
if (!(dev = qemuMonitorJSONGetBlockDev(devices, i)))
|
2011-09-13 15:49:50 +02:00
|
|
|
goto cleanup;
|
|
|
|
|
2016-10-03 15:04:32 -04:00
|
|
|
if (!(thisdev = qemuMonitorJSONGetBlockDevDevice(dev)))
|
2011-09-13 15:49:50 +02:00
|
|
|
goto cleanup;
|
|
|
|
|
2016-06-29 13:34:00 -04:00
|
|
|
thisdev = qemuAliasDiskDriveSkipPrefix(thisdev);
|
2018-08-06 14:38:55 +02:00
|
|
|
qdev = virJSONValueObjectGetString(dev, "qdev");
|
2011-09-13 15:49:50 +02:00
|
|
|
|
2018-08-06 14:38:55 +02:00
|
|
|
if (*thisdev == '\0')
|
|
|
|
thisdev = NULL;
|
2012-01-18 22:01:30 +01:00
|
|
|
|
2018-08-06 14:38:55 +02:00
|
|
|
if (!qdev && !thisdev) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-block device entry was not in expected format"));
|
2012-01-18 22:01:30 +01:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2011-09-13 15:49:50 +02:00
|
|
|
|
2018-08-06 14:38:55 +02:00
|
|
|
if (virJSONValueObjectGetBoolean(dev, "removable", &info.removable) < 0) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("cannot read %s value"),
|
|
|
|
"removable");
|
2011-09-13 15:49:50 +02:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2016-05-19 14:57:41 +02:00
|
|
|
/* 'tray_open' is present only if the device has a tray */
|
2018-08-06 14:38:55 +02:00
|
|
|
if (virJSONValueObjectGetBoolean(dev, "tray_open", &info.tray_open) == 0)
|
|
|
|
info.tray = true;
|
2016-05-19 14:57:41 +02:00
|
|
|
|
|
|
|
/* presence of 'inserted' notifies that a medium is in the device */
|
2017-02-23 19:36:52 +01:00
|
|
|
if ((image = virJSONValueObjectGetObject(dev, "inserted"))) {
|
2018-08-06 14:38:55 +02:00
|
|
|
info.nodename = (char *) virJSONValueObjectGetString(image, "node-name");
|
2017-02-23 19:36:52 +01:00
|
|
|
} else {
|
2018-08-06 14:38:55 +02:00
|
|
|
info.empty = true;
|
2017-02-23 19:36:52 +01:00
|
|
|
}
|
2012-01-19 17:58:58 +01:00
|
|
|
|
|
|
|
/* Missing io-status indicates no error */
|
|
|
|
if ((status = virJSONValueObjectGetString(dev, "io-status"))) {
|
2018-08-06 14:38:55 +02:00
|
|
|
info.io_status = qemuMonitorBlockIOStatusToError(status);
|
|
|
|
if (info.io_status < 0)
|
2012-01-19 17:58:58 +01:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2018-08-06 14:38:55 +02:00
|
|
|
|
|
|
|
if (thisdev &&
|
|
|
|
qemuMonitorJSONBlockInfoAdd(table, &info, thisdev) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (qdev && STRNEQ_NULLABLE(thisdev, qdev) &&
|
|
|
|
qemuMonitorJSONBlockInfoAdd(table, &info, qdev) < 0)
|
|
|
|
goto cleanup;
|
2011-09-13 15:49:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2016-09-27 09:39:21 -04:00
|
|
|
virJSONValueFree(devices);
|
2011-09-13 15:49:50 +02:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static qemuBlockStats *
|
|
|
|
qemuMonitorJSONBlockStatsCollectData(virJSONValue *dev,
|
2017-07-31 14:35:45 +02:00
|
|
|
int *nstats)
|
2014-12-11 15:28:41 -07:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
g_autofree qemuBlockStats *bstats = NULL;
|
|
|
|
virJSONValue *parent;
|
|
|
|
virJSONValue *parentstats;
|
|
|
|
virJSONValue *stats;
|
2014-12-11 15:28:41 -07:00
|
|
|
|
qemu: simplify json parsing
Rather than grabbing an arbitrary JSON value and then checking
if it has the right type, we might as well request the correct
type to begin with.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONIOProcessEvent)
(qemuMonitorJSONCommandWithFd, qemuMonitorJSONHandleGraphics)
(qemuMonitorJSONGetStatus, qemuMonitorJSONExtractCPUInfo)
(qemuMonitorJSONGetVirtType, qemuMonitorJSONGetBalloonInfo)
(qemuMonitorJSONGetMemoryStats)
(qemuMonitorJSONDevGetBlockExtent)
(qemuMonitorJSONGetOneBlockStatsInfo)
(qemuMonitorJSONGetAllBlockStatsInfo)
(qemuMonitorJSONBlockStatsUpdateCapacityOne)
(qemuMonitorJSONBlockStatsUpdateCapacity)
(qemuMonitorJSONGetBlockExtent)
(qemuMonitorJSONGetMigrationStatusReply)
(qemuMonitorJSONGetDumpGuestMemoryCapability)
(qemuMonitorJSONAddFd, qemuMonitorJSONQueryRxFilterParse)
(qemuMonitorJSONExtractChardevInfo)
(qemuMonitorJSONDiskNameLookupOne)
(qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONGetAllBlockJobInfo)
(qemuMonitorJSONBlockIoThrottleInfo, qemuMonitorJSONGetVersion)
(qemuMonitorJSONGetMachines, qemuMonitorJSONGetCPUDefinitions)
(qemuMonitorJSONGetCommands, qemuMonitorJSONGetEvents)
(qemuMonitorJSONGetKVMState, qemuMonitorJSONGetObjectTypes)
(qemuMonitorJSONGetObjectListPaths)
(qemuMonitorJSONGetObjectProps, qemuMonitorJSONGetTargetArch)
(qemuMonitorJSONGetMigrationCapabilities)
(qemuMonitorJSONGetStringArray, qemuMonitorJSONAttachCharDev)
(qemuMonitorJSONGetCPUx86Data, qemuMonitorJSONGetIOThreads)
(qemuMonitorJSONGetMemoryDeviceInfo): Use shorter idioms.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-06-20 13:40:21 -06:00
|
|
|
if ((stats = virJSONValueObjectGetObject(dev, "stats")) == NULL) {
|
2014-12-11 15:28:41 -07:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("blockstats stats entry was not "
|
|
|
|
"in expected format"));
|
2019-09-16 16:19:33 +02:00
|
|
|
return NULL;
|
2014-12-11 15:28:41 -07:00
|
|
|
}
|
|
|
|
|
2020-10-05 12:26:34 +02:00
|
|
|
bstats = g_new0(qemuBlockStats, 1);
|
2017-07-31 14:35:45 +02:00
|
|
|
|
2017-11-03 13:09:47 +01:00
|
|
|
#define QEMU_MONITOR_BLOCK_STAT_GET(NAME, VAR, MANDATORY) \
|
|
|
|
if (MANDATORY || virJSONValueObjectHasKey(stats, NAME)) { \
|
2018-08-01 09:15:22 +02:00
|
|
|
(*nstats)++; \
|
2019-08-15 11:32:28 +02:00
|
|
|
if (virJSONValueObjectGetNumberUlong(stats, NAME, &VAR) < 0) { \
|
2017-11-03 13:09:47 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, \
|
|
|
|
_("cannot read %s statistic"), NAME); \
|
2019-09-16 16:19:33 +02:00
|
|
|
return NULL; \
|
2017-11-03 13:09:47 +01:00
|
|
|
} \
|
2015-03-10 14:32:46 +01:00
|
|
|
}
|
2015-06-23 15:11:17 +02:00
|
|
|
QEMU_MONITOR_BLOCK_STAT_GET("rd_bytes", bstats->rd_bytes, true);
|
|
|
|
QEMU_MONITOR_BLOCK_STAT_GET("wr_bytes", bstats->wr_bytes, true);
|
|
|
|
QEMU_MONITOR_BLOCK_STAT_GET("rd_operations", bstats->rd_req, true);
|
|
|
|
QEMU_MONITOR_BLOCK_STAT_GET("wr_operations", bstats->wr_req, true);
|
|
|
|
QEMU_MONITOR_BLOCK_STAT_GET("rd_total_time_ns", bstats->rd_total_times, false);
|
|
|
|
QEMU_MONITOR_BLOCK_STAT_GET("wr_total_time_ns", bstats->wr_total_times, false);
|
|
|
|
QEMU_MONITOR_BLOCK_STAT_GET("flush_operations", bstats->flush_req, false);
|
|
|
|
QEMU_MONITOR_BLOCK_STAT_GET("flush_total_time_ns", bstats->flush_total_times, false);
|
2015-03-10 14:32:46 +01:00
|
|
|
#undef QEMU_MONITOR_BLOCK_STAT_GET
|
2014-12-11 15:28:41 -07:00
|
|
|
|
2015-06-23 16:58:07 +02:00
|
|
|
if ((parent = virJSONValueObjectGetObject(dev, "parent")) &&
|
|
|
|
(parentstats = virJSONValueObjectGetObject(parent, "stats"))) {
|
|
|
|
if (virJSONValueObjectGetNumberUlong(parentstats, "wr_highest_offset",
|
|
|
|
&bstats->wr_highest_offset) == 0)
|
|
|
|
bstats->wr_highest_offset_valid = true;
|
|
|
|
}
|
2014-12-11 15:28:41 -07:00
|
|
|
|
2019-10-16 13:35:54 +02:00
|
|
|
return g_steal_pointer(&bstats);
|
2017-07-31 14:35:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-08-03 16:22:19 +02:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONAddOneBlockStatsInfo(qemuBlockStats *bstats,
|
2018-08-03 16:22:19 +02:00
|
|
|
const char *name,
|
2020-10-22 19:04:18 +02:00
|
|
|
GHashTable *stats)
|
2018-08-03 16:22:19 +02:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuBlockStats *copy = NULL;
|
2018-08-03 16:22:19 +02:00
|
|
|
|
2020-10-05 12:26:34 +02:00
|
|
|
copy = g_new0(qemuBlockStats, 1);
|
2018-08-03 16:22:19 +02:00
|
|
|
|
|
|
|
if (bstats)
|
|
|
|
*copy = *bstats;
|
|
|
|
|
|
|
|
if (virHashAddEntry(stats, name, copy) < 0) {
|
|
|
|
VIR_FREE(copy);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-07-31 14:35:45 +02:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetOneBlockStatsInfo(virJSONValue *dev,
|
2017-07-31 14:35:45 +02:00
|
|
|
const char *dev_name,
|
|
|
|
int depth,
|
2020-10-22 19:04:18 +02:00
|
|
|
GHashTable *hash,
|
2017-07-31 14:35:45 +02:00
|
|
|
bool backingChain)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
g_autofree qemuBlockStats *bstats = NULL;
|
2017-07-31 14:35:45 +02:00
|
|
|
int nstats = 0;
|
2018-08-03 16:22:19 +02:00
|
|
|
const char *qdevname = NULL;
|
|
|
|
const char *nodename = NULL;
|
2019-10-15 15:16:31 +02:00
|
|
|
g_autofree char *devicename = NULL;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *backing;
|
2017-07-31 14:35:45 +02:00
|
|
|
|
2018-08-03 16:22:19 +02:00
|
|
|
if (dev_name &&
|
|
|
|
!(devicename = qemuDomainStorageAlias(dev_name, depth)))
|
2019-09-16 16:19:33 +02:00
|
|
|
return -1;
|
2018-08-03 16:22:19 +02:00
|
|
|
|
|
|
|
qdevname = virJSONValueObjectGetString(dev, "qdev");
|
|
|
|
nodename = virJSONValueObjectGetString(dev, "node-name");
|
|
|
|
|
|
|
|
if (!devicename && !qdevname && !nodename) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("blockstats device entry was not in expected format"));
|
2019-09-16 16:19:33 +02:00
|
|
|
return -1;
|
2018-08-03 16:22:19 +02:00
|
|
|
}
|
2017-07-31 14:35:45 +02:00
|
|
|
|
|
|
|
if (!(bstats = qemuMonitorJSONBlockStatsCollectData(dev, &nstats)))
|
2019-09-16 16:19:33 +02:00
|
|
|
return -1;
|
2017-07-31 14:35:45 +02:00
|
|
|
|
2018-08-03 16:22:19 +02:00
|
|
|
if (devicename &&
|
|
|
|
qemuMonitorJSONAddOneBlockStatsInfo(bstats, devicename, hash) < 0)
|
2019-09-16 16:19:33 +02:00
|
|
|
return -1;
|
2018-08-03 16:22:19 +02:00
|
|
|
|
|
|
|
if (qdevname && STRNEQ_NULLABLE(qdevname, devicename) &&
|
|
|
|
qemuMonitorJSONAddOneBlockStatsInfo(bstats, qdevname, hash) < 0)
|
2019-09-16 16:19:33 +02:00
|
|
|
return -1;
|
2018-08-03 16:22:19 +02:00
|
|
|
|
|
|
|
if (nodename &&
|
|
|
|
qemuMonitorJSONAddOneBlockStatsInfo(bstats, nodename, hash) < 0)
|
2019-09-16 16:19:33 +02:00
|
|
|
return -1;
|
2014-12-12 09:53:33 -07:00
|
|
|
|
|
|
|
if (backingChain &&
|
qemu: simplify json parsing
Rather than grabbing an arbitrary JSON value and then checking
if it has the right type, we might as well request the correct
type to begin with.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONIOProcessEvent)
(qemuMonitorJSONCommandWithFd, qemuMonitorJSONHandleGraphics)
(qemuMonitorJSONGetStatus, qemuMonitorJSONExtractCPUInfo)
(qemuMonitorJSONGetVirtType, qemuMonitorJSONGetBalloonInfo)
(qemuMonitorJSONGetMemoryStats)
(qemuMonitorJSONDevGetBlockExtent)
(qemuMonitorJSONGetOneBlockStatsInfo)
(qemuMonitorJSONGetAllBlockStatsInfo)
(qemuMonitorJSONBlockStatsUpdateCapacityOne)
(qemuMonitorJSONBlockStatsUpdateCapacity)
(qemuMonitorJSONGetBlockExtent)
(qemuMonitorJSONGetMigrationStatusReply)
(qemuMonitorJSONGetDumpGuestMemoryCapability)
(qemuMonitorJSONAddFd, qemuMonitorJSONQueryRxFilterParse)
(qemuMonitorJSONExtractChardevInfo)
(qemuMonitorJSONDiskNameLookupOne)
(qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONGetAllBlockJobInfo)
(qemuMonitorJSONBlockIoThrottleInfo, qemuMonitorJSONGetVersion)
(qemuMonitorJSONGetMachines, qemuMonitorJSONGetCPUDefinitions)
(qemuMonitorJSONGetCommands, qemuMonitorJSONGetEvents)
(qemuMonitorJSONGetKVMState, qemuMonitorJSONGetObjectTypes)
(qemuMonitorJSONGetObjectListPaths)
(qemuMonitorJSONGetObjectProps, qemuMonitorJSONGetTargetArch)
(qemuMonitorJSONGetMigrationCapabilities)
(qemuMonitorJSONGetStringArray, qemuMonitorJSONAttachCharDev)
(qemuMonitorJSONGetCPUx86Data, qemuMonitorJSONGetIOThreads)
(qemuMonitorJSONGetMemoryDeviceInfo): Use shorter idioms.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-06-20 13:40:21 -06:00
|
|
|
(backing = virJSONValueObjectGetObject(dev, "backing")) &&
|
2014-12-12 09:53:33 -07:00
|
|
|
qemuMonitorJSONGetOneBlockStatsInfo(backing, dev_name, depth + 1,
|
|
|
|
hash, true) < 0)
|
2019-09-16 16:19:33 +02:00
|
|
|
return -1;
|
2014-12-12 09:53:33 -07:00
|
|
|
|
2019-09-16 16:19:33 +02:00
|
|
|
return nstats;
|
2014-12-11 15:28:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *
|
|
|
|
qemuMonitorJSONQueryBlockstats(qemuMonitor *mon)
|
2014-09-15 10:48:09 +02:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *ret = NULL;
|
2014-09-15 10:48:09 +02:00
|
|
|
|
2018-07-30 17:08:37 +02:00
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-blockstats", NULL)))
|
2017-07-24 12:51:10 +02:00
|
|
|
return NULL;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2017-07-24 12:51:10 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2014-09-25 10:12:15 +02:00
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_ARRAY) < 0)
|
2010-07-12 14:07:02 +01:00
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
ret = virJSONValueObjectStealArray(reply, "return");
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2017-07-24 12:51:10 +02:00
|
|
|
cleanup:
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetAllBlockStatsInfo(qemuMonitor *mon,
|
2020-10-22 19:04:18 +02:00
|
|
|
GHashTable *hash,
|
2017-07-24 12:51:10 +02:00
|
|
|
bool backingChain)
|
|
|
|
{
|
|
|
|
int nstats = 0;
|
|
|
|
int rc;
|
|
|
|
size_t i;
|
2019-10-15 14:47:50 +02:00
|
|
|
g_autoptr(virJSONValue) devices = NULL;
|
2017-07-24 12:51:10 +02:00
|
|
|
|
2018-07-30 17:08:37 +02:00
|
|
|
if (!(devices = qemuMonitorJSONQueryBlockstats(mon)))
|
2017-07-24 12:51:10 +02:00
|
|
|
return -1;
|
|
|
|
|
2014-09-25 10:12:15 +02:00
|
|
|
for (i = 0; i < virJSONValueArraySize(devices); i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *dev = virJSONValueArrayGet(devices, i);
|
2014-09-30 11:41:43 +02:00
|
|
|
const char *dev_name;
|
2014-09-25 10:12:15 +02:00
|
|
|
|
2018-03-29 20:30:05 +02:00
|
|
|
if (!dev || virJSONValueGetType(dev) != VIR_JSON_TYPE_OBJECT) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2014-09-15 10:48:09 +02:00
|
|
|
_("blockstats device entry was not "
|
|
|
|
"in expected format"));
|
2019-09-16 16:19:33 +02:00
|
|
|
return -1;
|
2009-11-03 13:59:18 -05:00
|
|
|
}
|
|
|
|
|
2014-09-30 11:41:43 +02:00
|
|
|
if (!(dev_name = virJSONValueObjectGetString(dev, "device"))) {
|
2014-09-25 10:12:15 +02:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("blockstats device entry was not "
|
|
|
|
"in expected format"));
|
2019-09-16 16:19:33 +02:00
|
|
|
return -1;
|
2014-09-15 10:48:09 +02:00
|
|
|
}
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2018-08-03 16:22:19 +02:00
|
|
|
if (*dev_name == '\0')
|
|
|
|
dev_name = NULL;
|
|
|
|
|
2015-03-10 14:40:58 +01:00
|
|
|
rc = qemuMonitorJSONGetOneBlockStatsInfo(dev, dev_name, 0, hash,
|
|
|
|
backingChain);
|
|
|
|
|
|
|
|
if (rc < 0)
|
2019-09-16 16:19:33 +02:00
|
|
|
return -1;
|
2014-09-15 17:42:52 +02:00
|
|
|
|
2015-03-10 14:40:58 +01:00
|
|
|
if (rc > nstats)
|
|
|
|
nstats = rc;
|
2009-11-03 13:59:18 -05:00
|
|
|
}
|
2014-09-15 10:48:09 +02:00
|
|
|
|
2019-09-16 16:19:33 +02:00
|
|
|
return nstats;
|
2009-11-03 13:59:18 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-12-11 15:28:41 -07:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBlockStatsUpdateCapacityData(virJSONValue *image,
|
2018-06-20 10:43:07 +02:00
|
|
|
const char *name,
|
2020-10-22 19:04:18 +02:00
|
|
|
GHashTable *stats,
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuBlockStats **entry)
|
2014-12-11 15:28:41 -07:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuBlockStats *bstats;
|
2014-12-11 15:28:41 -07:00
|
|
|
|
2018-06-20 10:43:07 +02:00
|
|
|
if (!(bstats = virHashLookup(stats, name))) {
|
2020-10-05 12:26:34 +02:00
|
|
|
bstats = g_new0(qemuBlockStats, 1);
|
2014-12-11 15:28:41 -07:00
|
|
|
|
2018-06-20 10:43:07 +02:00
|
|
|
if (virHashAddEntry(stats, name, bstats) < 0) {
|
2014-12-11 15:28:41 -07:00
|
|
|
VIR_FREE(bstats);
|
2018-06-20 10:43:07 +02:00
|
|
|
return -1;
|
2014-12-11 15:28:41 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-29 15:28:31 +02:00
|
|
|
if (entry)
|
|
|
|
*entry = bstats;
|
|
|
|
|
2018-06-20 10:43:07 +02:00
|
|
|
/* failures can be ignored after this point */
|
2014-12-11 15:28:41 -07:00
|
|
|
if (virJSONValueObjectGetNumberUlong(image, "virtual-size",
|
|
|
|
&bstats->capacity) < 0)
|
2018-06-20 10:43:07 +02:00
|
|
|
return 0;
|
2014-12-11 15:28:41 -07:00
|
|
|
|
|
|
|
/* if actual-size is missing, image is not thin provisioned */
|
|
|
|
if (virJSONValueObjectGetNumberUlong(image, "actual-size",
|
|
|
|
&bstats->physical) < 0)
|
|
|
|
bstats->physical = bstats->capacity;
|
2014-12-12 09:53:33 -07:00
|
|
|
|
2018-06-20 10:43:07 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBlockStatsUpdateCapacityOne(virJSONValue *image,
|
2018-06-20 10:43:07 +02:00
|
|
|
const char *dev_name,
|
|
|
|
int depth,
|
2020-10-22 19:04:18 +02:00
|
|
|
GHashTable *stats,
|
2018-06-20 10:43:07 +02:00
|
|
|
bool backingChain)
|
|
|
|
{
|
|
|
|
int ret = -1;
|
|
|
|
char *entry_name = qemuDomainStorageAlias(dev_name, depth);
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *backing;
|
2018-06-20 10:43:07 +02:00
|
|
|
|
2018-06-29 15:28:31 +02:00
|
|
|
if (qemuMonitorJSONBlockStatsUpdateCapacityData(image, entry_name,
|
|
|
|
stats, NULL) < 0)
|
2018-06-20 10:43:07 +02:00
|
|
|
goto cleanup;
|
|
|
|
|
2014-12-12 09:53:33 -07:00
|
|
|
if (backingChain &&
|
2018-07-26 16:37:27 +02:00
|
|
|
(backing = virJSONValueObjectGetObject(image, "backing-image")) &&
|
|
|
|
qemuMonitorJSONBlockStatsUpdateCapacityOne(backing,
|
|
|
|
dev_name,
|
|
|
|
depth + 1,
|
|
|
|
stats,
|
|
|
|
true) < 0)
|
|
|
|
goto cleanup;
|
2014-12-12 09:53:33 -07:00
|
|
|
|
2018-07-26 16:37:27 +02:00
|
|
|
ret = 0;
|
2014-12-11 15:28:41 -07:00
|
|
|
cleanup:
|
2014-12-12 09:53:33 -07:00
|
|
|
VIR_FREE(entry_name);
|
2014-12-11 15:28:41 -07:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBlockStatsUpdateCapacity(qemuMonitor *mon,
|
2020-10-22 19:04:18 +02:00
|
|
|
GHashTable *stats,
|
2014-12-11 15:28:41 -07:00
|
|
|
bool backingChain)
|
2014-09-25 12:03:26 +02:00
|
|
|
{
|
|
|
|
int ret = -1;
|
|
|
|
size_t i;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *devices;
|
2014-09-25 12:03:26 +02:00
|
|
|
|
2016-09-27 09:39:21 -04:00
|
|
|
if (!(devices = qemuMonitorJSONQueryBlock(mon)))
|
2014-09-25 12:03:26 +02:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
for (i = 0; i < virJSONValueArraySize(devices); i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *dev;
|
|
|
|
virJSONValue *inserted;
|
|
|
|
virJSONValue *image;
|
2014-10-01 14:39:23 +02:00
|
|
|
const char *dev_name;
|
2014-09-25 12:03:26 +02:00
|
|
|
|
2016-10-03 14:58:59 -04:00
|
|
|
if (!(dev = qemuMonitorJSONGetBlockDev(devices, i)))
|
2014-09-25 12:03:26 +02:00
|
|
|
goto cleanup;
|
|
|
|
|
2016-10-03 15:04:32 -04:00
|
|
|
if (!(dev_name = qemuMonitorJSONGetBlockDevDevice(dev)))
|
2014-09-25 12:03:26 +02:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
/* drive may be empty */
|
qemu: simplify json parsing
Rather than grabbing an arbitrary JSON value and then checking
if it has the right type, we might as well request the correct
type to begin with.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONIOProcessEvent)
(qemuMonitorJSONCommandWithFd, qemuMonitorJSONHandleGraphics)
(qemuMonitorJSONGetStatus, qemuMonitorJSONExtractCPUInfo)
(qemuMonitorJSONGetVirtType, qemuMonitorJSONGetBalloonInfo)
(qemuMonitorJSONGetMemoryStats)
(qemuMonitorJSONDevGetBlockExtent)
(qemuMonitorJSONGetOneBlockStatsInfo)
(qemuMonitorJSONGetAllBlockStatsInfo)
(qemuMonitorJSONBlockStatsUpdateCapacityOne)
(qemuMonitorJSONBlockStatsUpdateCapacity)
(qemuMonitorJSONGetBlockExtent)
(qemuMonitorJSONGetMigrationStatusReply)
(qemuMonitorJSONGetDumpGuestMemoryCapability)
(qemuMonitorJSONAddFd, qemuMonitorJSONQueryRxFilterParse)
(qemuMonitorJSONExtractChardevInfo)
(qemuMonitorJSONDiskNameLookupOne)
(qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONGetAllBlockJobInfo)
(qemuMonitorJSONBlockIoThrottleInfo, qemuMonitorJSONGetVersion)
(qemuMonitorJSONGetMachines, qemuMonitorJSONGetCPUDefinitions)
(qemuMonitorJSONGetCommands, qemuMonitorJSONGetEvents)
(qemuMonitorJSONGetKVMState, qemuMonitorJSONGetObjectTypes)
(qemuMonitorJSONGetObjectListPaths)
(qemuMonitorJSONGetObjectProps, qemuMonitorJSONGetTargetArch)
(qemuMonitorJSONGetMigrationCapabilities)
(qemuMonitorJSONGetStringArray, qemuMonitorJSONAttachCharDev)
(qemuMonitorJSONGetCPUx86Data, qemuMonitorJSONGetIOThreads)
(qemuMonitorJSONGetMemoryDeviceInfo): Use shorter idioms.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-06-20 13:40:21 -06:00
|
|
|
if (!(inserted = virJSONValueObjectGetObject(dev, "inserted")) ||
|
|
|
|
!(image = virJSONValueObjectGetObject(inserted, "image")))
|
2014-09-25 12:03:26 +02:00
|
|
|
continue;
|
|
|
|
|
2014-12-12 09:53:33 -07:00
|
|
|
if (qemuMonitorJSONBlockStatsUpdateCapacityOne(image, dev_name, 0,
|
|
|
|
stats,
|
2014-12-11 15:28:41 -07:00
|
|
|
backingChain) < 0)
|
|
|
|
goto cleanup;
|
2014-09-25 12:03:26 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
2016-09-27 09:39:21 -04:00
|
|
|
virJSONValueFree(devices);
|
2014-09-25 12:03:26 +02:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-06-28 12:35:16 +02:00
|
|
|
static int
|
2019-10-14 14:45:33 +02:00
|
|
|
qemuMonitorJSONBlockStatsUpdateCapacityBlockdevWorker(size_t pos G_GNUC_UNUSED,
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *val,
|
2018-06-28 12:35:16 +02:00
|
|
|
void *opaque)
|
|
|
|
{
|
2020-10-22 19:04:18 +02:00
|
|
|
GHashTable *stats = opaque;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *image;
|
2018-06-28 12:35:16 +02:00
|
|
|
const char *nodename;
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuBlockStats *entry;
|
2018-06-28 12:35:16 +02:00
|
|
|
|
|
|
|
if (!(nodename = virJSONValueObjectGetString(val, "node-name")) ||
|
|
|
|
!(image = virJSONValueObjectGetObject(val, "image"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-named-block-nodes entry was not in expected format"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2018-06-29 15:28:31 +02:00
|
|
|
if (qemuMonitorJSONBlockStatsUpdateCapacityData(image, nodename, stats, &entry) < 0)
|
2018-06-28 12:35:16 +02:00
|
|
|
return -1;
|
|
|
|
|
2018-06-29 15:28:31 +02:00
|
|
|
if (entry)
|
|
|
|
ignore_value(virJSONValueObjectGetNumberUlong(val, "write_threshold",
|
|
|
|
&entry->write_threshold));
|
|
|
|
|
2018-06-28 12:35:16 +02:00
|
|
|
return 1; /* we don't want to steal the value from the JSON array */
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBlockStatsUpdateCapacityBlockdev(qemuMonitor *mon,
|
2020-10-22 19:04:18 +02:00
|
|
|
GHashTable *stats)
|
2018-06-28 12:35:16 +02:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *nodes;
|
2018-06-28 12:35:16 +02:00
|
|
|
int ret = -1;
|
|
|
|
|
2020-01-21 16:42:49 +01:00
|
|
|
if (!(nodes = qemuMonitorJSONQueryNamedBlockNodes(mon, false)))
|
2018-06-28 12:35:16 +02:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (virJSONValueArrayForeachSteal(nodes,
|
|
|
|
qemuMonitorJSONBlockStatsUpdateCapacityBlockdevWorker,
|
|
|
|
stats) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virJSONValueFree(nodes);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-11-28 14:27:00 +01:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBlockNamedNodeDataBitmapFree(qemuBlockNamedNodeDataBitmap *bitmap)
|
2019-11-28 14:27:00 +01:00
|
|
|
{
|
|
|
|
if (!bitmap)
|
|
|
|
return;
|
|
|
|
|
|
|
|
g_free(bitmap->name);
|
|
|
|
g_free(bitmap);
|
|
|
|
}
|
|
|
|
|
|
|
|
G_DEFINE_AUTOPTR_CLEANUP_FUNC(qemuBlockNamedNodeDataBitmap,
|
|
|
|
qemuMonitorJSONBlockNamedNodeDataBitmapFree);
|
|
|
|
|
|
|
|
|
2019-10-09 14:13:10 +02:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBlockNamedNodeDataFree(qemuBlockNamedNodeData *data)
|
2019-10-09 14:13:10 +02:00
|
|
|
{
|
2019-11-28 14:27:00 +01:00
|
|
|
size_t i;
|
|
|
|
|
|
|
|
if (!data)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (i = 0; i < data->nbitmaps; i++)
|
|
|
|
qemuMonitorJSONBlockNamedNodeDataBitmapFree(data->bitmaps[i]);
|
|
|
|
g_free(data->bitmaps);
|
2019-10-09 14:13:10 +02:00
|
|
|
g_free(data);
|
|
|
|
}
|
|
|
|
G_DEFINE_AUTOPTR_CLEANUP_FUNC(qemuBlockNamedNodeData, qemuMonitorJSONBlockNamedNodeDataFree);
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static qemuBlockNamedNodeDataBitmap *
|
|
|
|
qemuMonitorJSONBlockGetNamedNodeDataBitmapOne(virJSONValue *val)
|
2019-11-28 14:27:00 +01:00
|
|
|
{
|
|
|
|
g_autoptr(qemuBlockNamedNodeDataBitmap) bitmap = NULL;
|
|
|
|
const char *name;
|
|
|
|
|
|
|
|
bitmap = g_new0(qemuBlockNamedNodeDataBitmap, 1);
|
|
|
|
|
|
|
|
if (!(name = virJSONValueObjectGetString(val, "name")))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
bitmap->name = g_strdup(name);
|
|
|
|
|
|
|
|
ignore_value(virJSONValueObjectGetBoolean(val, "recording", &bitmap->recording));
|
|
|
|
ignore_value(virJSONValueObjectGetBoolean(val, "persistent", &bitmap->persistent));
|
|
|
|
ignore_value(virJSONValueObjectGetBoolean(val, "busy", &bitmap->busy));
|
|
|
|
ignore_value(virJSONValueObjectGetBoolean(val, "inconsistent", &bitmap->inconsistent));
|
|
|
|
ignore_value(virJSONValueObjectGetNumberUlong(val, "granularity", &bitmap->granularity));
|
|
|
|
ignore_value(virJSONValueObjectGetNumberUlong(val, "count", &bitmap->dirtybytes));
|
|
|
|
|
|
|
|
return g_steal_pointer(&bitmap);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBlockGetNamedNodeDataBitmaps(virJSONValue *bitmaps,
|
|
|
|
qemuBlockNamedNodeData *data)
|
2019-11-28 14:27:00 +01:00
|
|
|
{
|
|
|
|
size_t nbitmaps = virJSONValueArraySize(bitmaps);
|
|
|
|
size_t i;
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
data->bitmaps = g_new0(qemuBlockNamedNodeDataBitmap *, nbitmaps);
|
2019-11-28 14:27:00 +01:00
|
|
|
|
|
|
|
for (i = 0; i < nbitmaps; i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *bitmap = virJSONValueArrayGet(bitmaps, i);
|
|
|
|
qemuBlockNamedNodeDataBitmap *tmp;
|
2019-11-28 14:27:00 +01:00
|
|
|
|
|
|
|
if (!bitmap)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (!(tmp = qemuMonitorJSONBlockGetNamedNodeDataBitmapOne(bitmap)))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
data->bitmaps[data->nbitmaps++] = tmp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-10-09 14:13:10 +02:00
|
|
|
static int
|
|
|
|
qemuMonitorJSONBlockGetNamedNodeDataWorker(size_t pos G_GNUC_UNUSED,
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *val,
|
2019-10-09 14:13:10 +02:00
|
|
|
void *opaque)
|
|
|
|
{
|
2020-10-22 19:04:18 +02:00
|
|
|
GHashTable *nodes = opaque;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *img;
|
|
|
|
virJSONValue *bitmaps;
|
|
|
|
virJSONValue *format_specific;
|
2019-10-09 14:13:10 +02:00
|
|
|
const char *nodename;
|
|
|
|
g_autoptr(qemuBlockNamedNodeData) ent = NULL;
|
|
|
|
|
|
|
|
ent = g_new0(qemuBlockNamedNodeData, 1);
|
|
|
|
|
|
|
|
if (!(nodename = virJSONValueObjectGetString(val, "node-name")) ||
|
|
|
|
!(img = virJSONValueObjectGetObject(val, "image")))
|
|
|
|
goto broken;
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberUlong(img, "virtual-size", &ent->capacity) < 0)
|
|
|
|
goto broken;
|
|
|
|
|
|
|
|
/* if actual-size is missing, image is not thin provisioned */
|
|
|
|
if (virJSONValueObjectGetNumberUlong(img, "actual-size", &ent->physical) < 0)
|
|
|
|
ent->physical = ent->capacity;
|
|
|
|
|
2020-08-26 16:41:17 +02:00
|
|
|
/* try looking up the cluster size */
|
|
|
|
ignore_value(virJSONValueObjectGetNumberUlong(img, "cluster-size", &ent->clusterSize));
|
|
|
|
|
2019-11-28 14:27:00 +01:00
|
|
|
if ((bitmaps = virJSONValueObjectGetArray(val, "dirty-bitmaps")))
|
|
|
|
qemuMonitorJSONBlockGetNamedNodeDataBitmaps(bitmaps, ent);
|
|
|
|
|
2021-01-29 14:35:57 +01:00
|
|
|
/* query qcow2 format specific props */
|
|
|
|
if ((format_specific = virJSONValueObjectGetObject(img, "format-specific")) &&
|
|
|
|
STREQ_NULLABLE(virJSONValueObjectGetString(format_specific, "type"), "qcow2")) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *qcow2props = virJSONValueObjectGetObject(format_specific, "data");
|
2021-01-29 14:35:57 +01:00
|
|
|
|
|
|
|
if (qcow2props &&
|
|
|
|
STREQ_NULLABLE(virJSONValueObjectGetString(qcow2props, "compat"), "0.10"))
|
|
|
|
ent->qcow2v2 = true;
|
|
|
|
}
|
|
|
|
|
2019-10-09 14:13:10 +02:00
|
|
|
if (virHashAddEntry(nodes, nodename, ent) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
ent = NULL;
|
|
|
|
|
|
|
|
return 1; /* we don't want to steal the value from the JSON array */
|
|
|
|
|
|
|
|
broken:
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-named-block-nodes entry was not in expected format"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-22 19:04:18 +02:00
|
|
|
GHashTable *
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBlockGetNamedNodeDataJSON(virJSONValue *nodes)
|
2019-10-09 14:13:10 +02:00
|
|
|
{
|
2020-10-22 19:04:18 +02:00
|
|
|
g_autoptr(GHashTable) ret = NULL;
|
2019-10-09 14:13:10 +02:00
|
|
|
|
2020-01-28 13:40:24 +01:00
|
|
|
ret = virHashNew((virHashDataFree) qemuMonitorJSONBlockNamedNodeDataFree);
|
2019-10-09 14:13:10 +02:00
|
|
|
|
|
|
|
if (virJSONValueArrayForeachSteal(nodes,
|
|
|
|
qemuMonitorJSONBlockGetNamedNodeDataWorker,
|
|
|
|
ret) < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
return g_steal_pointer(&ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-22 19:04:18 +02:00
|
|
|
GHashTable *
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBlockGetNamedNodeData(qemuMonitor *mon,
|
2020-01-21 16:51:40 +01:00
|
|
|
bool supports_flat)
|
2019-12-09 07:46:53 +01:00
|
|
|
{
|
|
|
|
g_autoptr(virJSONValue) nodes = NULL;
|
|
|
|
|
2020-01-21 16:51:40 +01:00
|
|
|
if (!(nodes = qemuMonitorJSONQueryNamedBlockNodes(mon, supports_flat)))
|
2019-12-09 07:46:53 +01:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
return qemuMonitorJSONBlockGetNamedNodeDataJSON(nodes);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONBlockResize(qemuMonitor *mon,
|
2011-11-29 15:34:53 +08:00
|
|
|
const char *device,
|
2018-08-07 09:31:04 +02:00
|
|
|
const char *nodename,
|
2011-11-29 15:34:53 +08:00
|
|
|
unsigned long long size)
|
|
|
|
{
|
2016-05-03 11:29:11 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2011-11-29 15:34:53 +08:00
|
|
|
|
|
|
|
cmd = qemuMonitorJSONMakeCommand("block_resize",
|
2018-08-07 09:31:04 +02:00
|
|
|
"S:device", device,
|
|
|
|
"S:node-name", nodename,
|
2012-03-03 07:43:22 -07:00
|
|
|
"U:size", size,
|
2011-11-29 15:34:53 +08:00
|
|
|
NULL);
|
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2011-11-29 15:34:53 +08:00
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = 0;
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2011-11-29 15:34:53 +08:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2010-05-14 09:10:01 -04:00
|
|
|
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONSetPassword(qemuMonitor *mon,
|
2011-01-10 12:12:32 +01:00
|
|
|
const char *protocol,
|
|
|
|
const char *password,
|
|
|
|
const char *action_if_connected)
|
|
|
|
{
|
2016-05-03 11:29:11 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = qemuMonitorJSONMakeCommand("set_password",
|
2011-01-10 12:12:32 +01:00
|
|
|
"s:protocol", protocol,
|
|
|
|
"s:password", password,
|
|
|
|
"s:connected", action_if_connected,
|
|
|
|
NULL);
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *reply = NULL;
|
2011-01-10 12:12:32 +01:00
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2011-01-10 12:12:32 +01:00
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = 0;
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2011-01-10 12:12:32 +01:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONExpirePassword(qemuMonitor *mon,
|
2011-01-10 12:12:32 +01:00
|
|
|
const char *protocol,
|
|
|
|
const char *expire_time)
|
|
|
|
{
|
2016-05-03 11:29:11 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = qemuMonitorJSONMakeCommand("expire_password",
|
2011-01-10 12:12:32 +01:00
|
|
|
"s:protocol", protocol,
|
|
|
|
"s:time", expire_time,
|
|
|
|
NULL);
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *reply = NULL;
|
2011-01-10 12:12:32 +01:00
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2011-01-18 18:37:45 +00:00
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = 0;
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2011-01-10 12:12:32 +01:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2015-05-27 14:03:17 +02:00
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONSetBalloon(qemuMonitor *mon,
|
2015-05-27 14:03:17 +02:00
|
|
|
unsigned long long newmem)
|
2009-11-03 13:59:18 -05:00
|
|
|
{
|
2016-05-03 11:29:11 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = qemuMonitorJSONMakeCommand("balloon",
|
2015-05-27 14:03:17 +02:00
|
|
|
"U:value", newmem * 1024,
|
2009-11-03 13:59:18 -05:00
|
|
|
NULL);
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *reply = NULL;
|
2009-11-03 13:59:18 -05:00
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
/* See if balloon soft-failed */
|
|
|
|
if (qemuMonitorJSONHasError(reply, "DeviceNotActive") ||
|
|
|
|
qemuMonitorJSONHasError(reply, "KVMMissingCap")) {
|
|
|
|
ret = 0;
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
}
|
|
|
|
|
2016-05-03 11:29:11 +02:00
|
|
|
/* See if any other fatal error occurred */
|
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
/* Real success */
|
|
|
|
ret = 1;
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2009-11-03 13:59:18 -05:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONSetCPU(qemuMonitor *mon,
|
2013-05-27 15:35:35 +02:00
|
|
|
int cpu, bool online)
|
2010-02-08 16:37:17 +00:00
|
|
|
{
|
2013-05-27 16:08:30 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = NULL;
|
|
|
|
virJSONValue *reply = NULL;
|
2013-05-27 16:08:30 +02:00
|
|
|
|
|
|
|
if (online) {
|
|
|
|
cmd = qemuMonitorJSONMakeCommand("cpu-add",
|
|
|
|
"i:id", cpu,
|
|
|
|
NULL);
|
|
|
|
} else {
|
2019-06-20 13:12:04 +02:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("vCPU unplug is not supported by this QEMU"));
|
|
|
|
goto cleanup;
|
2013-05-27 16:08:30 +02:00
|
|
|
}
|
|
|
|
if (!cmd)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if ((ret = qemuMonitorJSONCommand(mon, cmd, &reply)) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2019-06-20 13:12:04 +02:00
|
|
|
ret = qemuMonitorJSONCheckError(cmd, reply);
|
2013-05-27 16:08:30 +02:00
|
|
|
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2013-05-27 16:08:30 +02:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
2010-02-08 16:37:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-06-29 16:18:53 +02:00
|
|
|
/**
|
|
|
|
* Run QMP command to eject a media from ejectable device.
|
|
|
|
*
|
|
|
|
* Returns:
|
2016-03-17 16:38:28 +01:00
|
|
|
* -1 on error
|
2015-06-29 16:18:53 +02:00
|
|
|
* 0 on success
|
|
|
|
*/
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONEjectMedia(qemuMonitor *mon,
|
2011-09-16 14:05:58 +02:00
|
|
|
const char *dev_name,
|
2010-11-08 12:52:48 -05:00
|
|
|
bool force)
|
2009-11-03 13:59:18 -05:00
|
|
|
{
|
2016-05-04 10:28:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = qemuMonitorJSONMakeCommand("eject",
|
2011-09-16 14:05:58 +02:00
|
|
|
"s:device", dev_name,
|
2010-11-08 12:52:48 -05:00
|
|
|
"b:force", force ? 1 : 0,
|
2009-11-03 13:59:18 -05:00
|
|
|
NULL);
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *reply = NULL;
|
2009-11-03 13:59:18 -05:00
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2009-11-03 13:59:18 -05:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONChangeMedia(qemuMonitor *mon,
|
2011-09-16 14:05:58 +02:00
|
|
|
const char *dev_name,
|
2009-11-03 13:59:18 -05:00
|
|
|
const char *newmedia,
|
|
|
|
const char *format)
|
|
|
|
{
|
2016-05-04 10:28:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2014-05-13 17:28:45 +02:00
|
|
|
|
|
|
|
cmd = qemuMonitorJSONMakeCommand("change",
|
|
|
|
"s:device", dev_name,
|
|
|
|
"s:target", newmedia,
|
|
|
|
"S:arg", format,
|
|
|
|
NULL);
|
|
|
|
|
2009-11-03 13:59:18 -05:00
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2009-11-03 13:59:18 -05:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static int qemuMonitorJSONSaveMemory(qemuMonitor *mon,
|
2009-11-03 13:59:18 -05:00
|
|
|
const char *cmdtype,
|
|
|
|
unsigned long long offset,
|
2019-09-27 13:29:53 +02:00
|
|
|
unsigned long long length,
|
2009-11-03 13:59:18 -05:00
|
|
|
const char *path)
|
|
|
|
{
|
2016-05-04 10:28:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = qemuMonitorJSONMakeCommand(cmdtype,
|
2009-11-03 13:59:18 -05:00
|
|
|
"U:val", offset,
|
2019-08-30 16:37:46 +02:00
|
|
|
"U:size", length,
|
2009-11-03 13:59:18 -05:00
|
|
|
"s:filename", path,
|
|
|
|
NULL);
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *reply = NULL;
|
2009-11-03 13:59:18 -05:00
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2009-11-03 13:59:18 -05:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONSaveVirtualMemory(qemuMonitor *mon,
|
2009-11-03 13:59:18 -05:00
|
|
|
unsigned long long offset,
|
2019-09-27 13:29:53 +02:00
|
|
|
unsigned long long length,
|
2009-11-03 13:59:18 -05:00
|
|
|
const char *path)
|
|
|
|
{
|
|
|
|
return qemuMonitorJSONSaveMemory(mon, "memsave", offset, length, path);
|
|
|
|
}
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONSavePhysicalMemory(qemuMonitor *mon,
|
2009-11-03 13:59:18 -05:00
|
|
|
unsigned long long offset,
|
2019-09-27 13:29:53 +02:00
|
|
|
unsigned long long length,
|
2009-11-03 13:59:18 -05:00
|
|
|
const char *path)
|
|
|
|
{
|
|
|
|
return qemuMonitorJSONSaveMemory(mon, "pmemsave", offset, length, path);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONSetMigrationSpeed(qemuMonitor *mon,
|
2009-11-03 13:59:18 -05:00
|
|
|
unsigned long bandwidth)
|
|
|
|
{
|
2016-05-04 10:28:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2009-11-03 13:59:18 -05:00
|
|
|
cmd = qemuMonitorJSONMakeCommand("migrate_set_speed",
|
2010-11-25 16:38:32 +08:00
|
|
|
"U:value", bandwidth * 1024ULL * 1024ULL,
|
2009-11-03 13:59:18 -05:00
|
|
|
NULL);
|
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2009-11-03 13:59:18 -05:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONSetMigrationDowntime(qemuMonitor *mon,
|
2010-03-17 16:53:14 +01:00
|
|
|
unsigned long long downtime)
|
|
|
|
{
|
2016-05-04 10:28:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2010-08-19 15:59:25 +02:00
|
|
|
|
2010-03-17 16:53:14 +01:00
|
|
|
cmd = qemuMonitorJSONMakeCommand("migrate_set_downtime",
|
2010-08-19 15:59:25 +02:00
|
|
|
"d:value", downtime / 1000.0,
|
2010-03-17 16:53:14 +01:00
|
|
|
NULL);
|
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2010-03-17 16:53:14 +01:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2010-03-17 16:53:14 +01:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2010-03-17 16:53:14 +01:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-02-18 21:54:58 +01:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetMigrationCacheSize(qemuMonitor *mon,
|
2013-02-18 21:54:58 +01:00
|
|
|
unsigned long long *cacheSize)
|
|
|
|
{
|
2016-05-04 10:28:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2013-02-18 21:54:58 +01:00
|
|
|
|
|
|
|
*cacheSize = 0;
|
|
|
|
|
|
|
|
cmd = qemuMonitorJSONMakeCommand("query-migrate-cache-size", NULL);
|
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2013-02-18 21:54:58 +01:00
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_NUMBER) < 0)
|
2013-02-18 21:54:58 +01:00
|
|
|
goto cleanup;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (virJSONValueObjectGetNumberUlong(reply, "return", cacheSize) < 0) {
|
2013-02-18 21:54:58 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2018-03-28 12:45:21 +02:00
|
|
|
_("invalid cache size in query-migrate-cache-size reply"));
|
2013-02-18 21:54:58 +01:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2013-02-18 21:54:58 +01:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONSetMigrationCacheSize(qemuMonitor *mon,
|
2013-02-18 21:54:58 +01:00
|
|
|
unsigned long long cacheSize)
|
|
|
|
{
|
2016-05-04 10:28:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2013-02-18 21:54:58 +01:00
|
|
|
|
|
|
|
cmd = qemuMonitorJSONMakeCommand("migrate-set-cache-size",
|
|
|
|
"U:value", cacheSize,
|
|
|
|
NULL);
|
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2013-02-18 21:54:58 +01:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2013-02-18 21:54:58 +01:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2013-02-18 21:54:58 +01:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-04-14 13:33:49 +03:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetMigrationParams(qemuMonitor *mon,
|
|
|
|
virJSONValue **params)
|
2016-04-14 13:33:49 +03:00
|
|
|
{
|
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = NULL;
|
|
|
|
virJSONValue *reply = NULL;
|
2016-04-14 13:33:49 +03:00
|
|
|
|
2018-03-15 18:06:01 +01:00
|
|
|
*params = NULL;
|
2016-06-20 16:54:24 +02:00
|
|
|
|
2016-04-14 13:33:49 +03:00
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-migrate-parameters", NULL)))
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2016-04-14 13:33:49 +03:00
|
|
|
goto cleanup;
|
|
|
|
|
2017-04-26 23:18:35 +02:00
|
|
|
if (qemuMonitorJSONHasError(reply, "CommandNotFound")) {
|
|
|
|
ret = 0;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_OBJECT) < 0)
|
2016-04-14 13:33:49 +03:00
|
|
|
goto cleanup;
|
|
|
|
|
2018-03-15 18:06:01 +01:00
|
|
|
*params = virJSONValueObjectStealObject(reply, "return");
|
2016-04-14 13:33:49 +03:00
|
|
|
ret = 0;
|
2018-03-15 18:06:01 +01:00
|
|
|
|
2016-04-14 13:33:49 +03:00
|
|
|
cleanup:
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONSetMigrationParams(qemuMonitor *mon,
|
|
|
|
virJSONValue **params)
|
2016-04-14 13:33:49 +03:00
|
|
|
{
|
2020-03-04 10:10:56 +01:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2016-04-14 13:33:49 +03:00
|
|
|
|
2020-11-30 15:34:56 +01:00
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommandInternal("migrate-set-parameters", params)))
|
2020-03-04 10:10:56 +01:00
|
|
|
return -1;
|
2016-04-14 13:33:49 +03:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2020-03-04 10:10:56 +01:00
|
|
|
return -1;
|
2016-04-14 13:33:49 +03:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
2020-03-04 10:10:56 +01:00
|
|
|
return -1;
|
2016-04-14 13:33:49 +03:00
|
|
|
|
2020-03-04 10:10:56 +01:00
|
|
|
return 0;
|
2016-04-14 13:33:49 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-11-03 13:59:18 -05:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetMigrationStatsReply(virJSONValue *reply,
|
|
|
|
qemuMonitorMigrationStats *stats,
|
2017-10-12 15:19:19 +02:00
|
|
|
char **error)
|
2009-11-03 13:59:18 -05:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *ret;
|
|
|
|
virJSONValue *ram;
|
|
|
|
virJSONValue *disk;
|
|
|
|
virJSONValue *comp;
|
2010-01-22 13:22:53 +00:00
|
|
|
const char *statusstr;
|
2013-02-08 09:58:03 +01:00
|
|
|
int rc;
|
2014-01-13 14:28:10 +08:00
|
|
|
double mbps;
|
2017-10-12 15:19:19 +02:00
|
|
|
const char *tmp;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-11-21 00:10:06 +01:00
|
|
|
ret = virJSONValueObjectGetObject(reply, "return");
|
2009-11-03 13:59:18 -05:00
|
|
|
|
|
|
|
if (!(statusstr = virJSONValueObjectGetString(ret, "status"))) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("info migration reply was missing return status"));
|
2009-11-03 13:59:18 -05:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2015-11-26 13:23:08 +01:00
|
|
|
stats->status = qemuMonitorMigrationStatusTypeFromString(statusstr);
|
|
|
|
if (stats->status < 0) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("unexpected migration status in %s"), statusstr);
|
2009-11-03 13:59:18 -05:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2014-09-09 10:17:46 +02:00
|
|
|
ignore_value(virJSONValueObjectGetNumberUlong(ret, "total-time",
|
2015-11-26 13:23:08 +01:00
|
|
|
&stats->total_time));
|
|
|
|
if (stats->status == QEMU_MONITOR_MIGRATION_STATUS_COMPLETED) {
|
2013-02-08 09:58:03 +01:00
|
|
|
rc = virJSONValueObjectGetNumberUlong(ret, "downtime",
|
2015-11-26 13:23:08 +01:00
|
|
|
&stats->downtime);
|
2013-02-08 09:58:03 +01:00
|
|
|
} else {
|
|
|
|
rc = virJSONValueObjectGetNumberUlong(ret, "expected-downtime",
|
2015-11-26 13:23:08 +01:00
|
|
|
&stats->downtime);
|
2013-02-08 09:58:03 +01:00
|
|
|
}
|
|
|
|
if (rc == 0)
|
2015-11-26 13:23:08 +01:00
|
|
|
stats->downtime_set = true;
|
2013-02-08 09:58:03 +01:00
|
|
|
|
2014-01-13 14:28:10 +08:00
|
|
|
if (virJSONValueObjectGetNumberUlong(ret, "setup-time",
|
2015-11-26 13:23:08 +01:00
|
|
|
&stats->setup_time) == 0)
|
|
|
|
stats->setup_time_set = true;
|
2014-01-13 14:28:10 +08:00
|
|
|
|
2016-06-21 13:40:33 +02:00
|
|
|
ignore_value(virJSONValueObjectGetNumberInt(ret, "cpu-throttle-percentage",
|
|
|
|
&stats->cpu_throttle_percentage));
|
|
|
|
|
2015-11-26 13:24:31 +01:00
|
|
|
switch ((qemuMonitorMigrationStatus) stats->status) {
|
|
|
|
case QEMU_MONITOR_MIGRATION_STATUS_INACTIVE:
|
|
|
|
case QEMU_MONITOR_MIGRATION_STATUS_SETUP:
|
|
|
|
case QEMU_MONITOR_MIGRATION_STATUS_CANCELLED:
|
2020-01-15 16:38:57 -05:00
|
|
|
case QEMU_MONITOR_MIGRATION_STATUS_WAIT_UNPLUG:
|
2015-11-26 13:24:31 +01:00
|
|
|
case QEMU_MONITOR_MIGRATION_STATUS_LAST:
|
|
|
|
break;
|
|
|
|
|
2017-10-12 15:19:19 +02:00
|
|
|
case QEMU_MONITOR_MIGRATION_STATUS_ERROR:
|
|
|
|
if (error) {
|
|
|
|
tmp = virJSONValueObjectGetString(ret, "error-desc");
|
2019-10-20 13:49:46 +02:00
|
|
|
if (tmp)
|
|
|
|
*error = g_strdup(tmp);
|
2017-10-12 15:19:19 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2015-11-26 13:24:31 +01:00
|
|
|
case QEMU_MONITOR_MIGRATION_STATUS_ACTIVE:
|
2015-11-26 15:37:23 +01:00
|
|
|
case QEMU_MONITOR_MIGRATION_STATUS_POSTCOPY:
|
2015-11-26 13:24:31 +01:00
|
|
|
case QEMU_MONITOR_MIGRATION_STATUS_COMPLETED:
|
|
|
|
case QEMU_MONITOR_MIGRATION_STATUS_CANCELLING:
|
2017-10-20 10:11:32 +02:00
|
|
|
case QEMU_MONITOR_MIGRATION_STATUS_PRE_SWITCHOVER:
|
|
|
|
case QEMU_MONITOR_MIGRATION_STATUS_DEVICE:
|
2015-11-26 13:24:31 +01:00
|
|
|
ram = virJSONValueObjectGetObject(ret, "ram");
|
2009-11-03 13:59:18 -05:00
|
|
|
if (!ram) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("migration was active, but no RAM info was set"));
|
2009-11-03 13:59:18 -05:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2011-08-17 08:30:02 +02:00
|
|
|
if (virJSONValueObjectGetNumberUlong(ram, "transferred",
|
2015-11-26 13:23:08 +01:00
|
|
|
&stats->ram_transferred) < 0) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("migration was active, but RAM 'transferred' "
|
|
|
|
"data was missing"));
|
2009-11-03 13:59:18 -05:00
|
|
|
return -1;
|
|
|
|
}
|
2013-02-08 09:58:03 +01:00
|
|
|
if (virJSONValueObjectGetNumberUlong(ram, "remaining",
|
2015-11-26 13:23:08 +01:00
|
|
|
&stats->ram_remaining) < 0) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("migration was active, but RAM 'remaining' "
|
|
|
|
"data was missing"));
|
2009-11-03 13:59:18 -05:00
|
|
|
return -1;
|
|
|
|
}
|
2013-02-08 09:58:03 +01:00
|
|
|
if (virJSONValueObjectGetNumberUlong(ram, "total",
|
2015-11-26 13:23:08 +01:00
|
|
|
&stats->ram_total) < 0) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("migration was active, but RAM 'total' "
|
|
|
|
"data was missing"));
|
2009-11-03 13:59:18 -05:00
|
|
|
return -1;
|
|
|
|
}
|
2011-08-17 08:30:02 +02:00
|
|
|
|
2014-01-13 14:28:10 +08:00
|
|
|
if (virJSONValueObjectGetNumberDouble(ram, "mbps", &mbps) == 0 &&
|
|
|
|
mbps > 0) {
|
|
|
|
/* mpbs from QEMU reports Mbits/s (M as in 10^6 not Mi as 2^20) */
|
2015-11-26 13:23:08 +01:00
|
|
|
stats->ram_bps = mbps * (1000 * 1000 / 8);
|
2014-01-13 14:28:10 +08:00
|
|
|
}
|
|
|
|
|
2013-02-08 09:58:03 +01:00
|
|
|
if (virJSONValueObjectGetNumberUlong(ram, "duplicate",
|
2015-11-26 13:23:08 +01:00
|
|
|
&stats->ram_duplicate) == 0)
|
|
|
|
stats->ram_duplicate_set = true;
|
2014-09-09 10:17:46 +02:00
|
|
|
ignore_value(virJSONValueObjectGetNumberUlong(ram, "normal",
|
2015-11-26 13:23:08 +01:00
|
|
|
&stats->ram_normal));
|
2014-09-09 10:17:46 +02:00
|
|
|
ignore_value(virJSONValueObjectGetNumberUlong(ram, "normal-bytes",
|
2015-11-26 13:23:08 +01:00
|
|
|
&stats->ram_normal_bytes));
|
2015-11-27 12:30:09 +01:00
|
|
|
ignore_value(virJSONValueObjectGetNumberUlong(ram, "dirty-pages-rate",
|
|
|
|
&stats->ram_dirty_rate));
|
2017-10-09 10:00:03 +08:00
|
|
|
ignore_value(virJSONValueObjectGetNumberUlong(ram, "page-size",
|
|
|
|
&stats->ram_page_size));
|
2015-11-27 12:30:09 +01:00
|
|
|
ignore_value(virJSONValueObjectGetNumberUlong(ram, "dirty-sync-count",
|
|
|
|
&stats->ram_iteration));
|
2018-11-15 15:25:46 +01:00
|
|
|
ignore_value(virJSONValueObjectGetNumberUlong(ram, "postcopy-requests",
|
|
|
|
&stats->ram_postcopy_reqs));
|
2013-02-08 09:58:03 +01:00
|
|
|
|
2015-11-26 13:24:31 +01:00
|
|
|
disk = virJSONValueObjectGetObject(ret, "disk");
|
2013-02-08 09:58:03 +01:00
|
|
|
if (disk) {
|
|
|
|
rc = virJSONValueObjectGetNumberUlong(disk, "transferred",
|
2015-11-26 13:23:08 +01:00
|
|
|
&stats->disk_transferred);
|
2013-02-08 09:58:03 +01:00
|
|
|
if (rc < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("disk migration was active, but "
|
|
|
|
"'transferred' data was missing"));
|
|
|
|
return -1;
|
|
|
|
}
|
2011-08-17 08:30:02 +02:00
|
|
|
|
2013-02-08 09:58:03 +01:00
|
|
|
rc = virJSONValueObjectGetNumberUlong(disk, "remaining",
|
2015-11-26 13:23:08 +01:00
|
|
|
&stats->disk_remaining);
|
2013-02-08 09:58:03 +01:00
|
|
|
if (rc < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("disk migration was active, but 'remaining' "
|
|
|
|
"data was missing"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = virJSONValueObjectGetNumberUlong(disk, "total",
|
2015-11-26 13:23:08 +01:00
|
|
|
&stats->disk_total);
|
2013-02-08 09:58:03 +01:00
|
|
|
if (rc < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("disk migration was active, but 'total' "
|
|
|
|
"data was missing"));
|
|
|
|
return -1;
|
|
|
|
}
|
2014-01-13 14:28:10 +08:00
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberDouble(disk, "mbps", &mbps) == 0 &&
|
|
|
|
mbps > 0) {
|
|
|
|
/* mpbs from QEMU reports Mbits/s (M as in 10^6 not Mi as 2^20) */
|
2015-11-26 13:23:08 +01:00
|
|
|
stats->disk_bps = mbps * (1000 * 1000 / 8);
|
2014-01-13 14:28:10 +08:00
|
|
|
}
|
2011-08-17 08:30:02 +02:00
|
|
|
}
|
2013-02-08 09:58:03 +01:00
|
|
|
|
2015-11-26 13:24:31 +01:00
|
|
|
comp = virJSONValueObjectGetObject(ret, "xbzrle-cache");
|
2013-02-08 09:58:03 +01:00
|
|
|
if (comp) {
|
2015-11-26 13:23:08 +01:00
|
|
|
stats->xbzrle_set = true;
|
2013-02-08 09:58:03 +01:00
|
|
|
rc = virJSONValueObjectGetNumberUlong(comp, "cache-size",
|
2015-11-26 13:23:08 +01:00
|
|
|
&stats->xbzrle_cache_size);
|
2013-02-08 09:58:03 +01:00
|
|
|
if (rc < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("XBZRLE is active, but 'cache-size' data "
|
|
|
|
"was missing"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = virJSONValueObjectGetNumberUlong(comp, "bytes",
|
2015-11-26 13:23:08 +01:00
|
|
|
&stats->xbzrle_bytes);
|
2013-02-08 09:58:03 +01:00
|
|
|
if (rc < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("XBZRLE is active, but 'bytes' data "
|
|
|
|
"was missing"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = virJSONValueObjectGetNumberUlong(comp, "pages",
|
2015-11-26 13:23:08 +01:00
|
|
|
&stats->xbzrle_pages);
|
2013-02-08 09:58:03 +01:00
|
|
|
if (rc < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("XBZRLE is active, but 'pages' data "
|
|
|
|
"was missing"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = virJSONValueObjectGetNumberUlong(comp, "cache-miss",
|
2015-11-26 13:23:08 +01:00
|
|
|
&stats->xbzrle_cache_miss);
|
2013-02-08 09:58:03 +01:00
|
|
|
if (rc < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("XBZRLE is active, but 'cache-miss' data "
|
|
|
|
"was missing"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = virJSONValueObjectGetNumberUlong(comp, "overflow",
|
2015-11-26 13:23:08 +01:00
|
|
|
&stats->xbzrle_overflow);
|
2013-02-08 09:58:03 +01:00
|
|
|
if (rc < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("XBZRLE is active, but 'overflow' data "
|
|
|
|
"was missing"));
|
|
|
|
return -1;
|
|
|
|
}
|
2011-08-17 08:30:02 +02:00
|
|
|
}
|
2015-11-26 13:24:31 +01:00
|
|
|
break;
|
2009-11-03 13:59:18 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONGetMigrationStats(qemuMonitor *mon,
|
|
|
|
qemuMonitorMigrationStats *stats,
|
2017-10-12 15:19:19 +02:00
|
|
|
char **error)
|
2009-11-03 13:59:18 -05:00
|
|
|
{
|
2016-05-04 10:28:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = qemuMonitorJSONMakeCommand("query-migrate",
|
2009-11-03 13:59:18 -05:00
|
|
|
NULL);
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *reply = NULL;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2015-11-26 13:23:08 +01:00
|
|
|
memset(stats, 0, sizeof(*stats));
|
2009-11-03 13:59:18 -05:00
|
|
|
|
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_OBJECT) < 0)
|
2016-05-04 10:28:43 +02:00
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2017-10-12 15:19:19 +02:00
|
|
|
if (qemuMonitorJSONGetMigrationStatsReply(reply, stats, error) < 0)
|
2016-05-04 10:28:43 +02:00
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2013-02-08 09:58:03 +01:00
|
|
|
if (ret < 0)
|
2015-11-26 13:23:08 +01:00
|
|
|
memset(stats, 0, sizeof(*stats));
|
2009-11-03 13:59:18 -05:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONMigrate(qemuMonitor *mon,
|
2011-03-04 11:51:48 -07:00
|
|
|
unsigned int flags,
|
|
|
|
const char *uri)
|
2009-11-03 13:59:18 -05:00
|
|
|
{
|
2016-05-04 10:28:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd =
|
2010-06-24 14:58:36 -04:00
|
|
|
qemuMonitorJSONMakeCommand("migrate",
|
2010-09-09 18:05:03 -03:00
|
|
|
"b:detach", flags & QEMU_MONITOR_MIGRATE_BACKGROUND ? 1 : 0,
|
|
|
|
"b:blk", flags & QEMU_MONITOR_MIGRATE_NON_SHARED_DISK ? 1 : 0,
|
|
|
|
"b:inc", flags & QEMU_MONITOR_MIGRATE_NON_SHARED_INC ? 1 : 0,
|
2010-06-24 14:58:36 -04:00
|
|
|
"s:uri", uri,
|
|
|
|
NULL);
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *reply = NULL;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2009-11-03 13:59:18 -05:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONMigrateCancel(qemuMonitor *mon)
|
2009-11-03 13:59:18 -05:00
|
|
|
{
|
2016-05-04 10:28:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = qemuMonitorJSONMakeCommand("migrate_cancel", NULL);
|
|
|
|
virJSONValue *reply = NULL;
|
2009-11-03 13:59:18 -05:00
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2009-11-03 13:59:18 -05:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2017-11-20 15:02:59 -05:00
|
|
|
|
|
|
|
/* qemuMonitorJSONQueryDump:
|
|
|
|
* @mon: Monitor pointer
|
|
|
|
* @stats: Monitor dump stats
|
|
|
|
*
|
|
|
|
* Attempt to make a "query-dump" call, check for errors, and get/return
|
|
|
|
* the current from the reply
|
|
|
|
*
|
|
|
|
* Returns: 0 on success, -1 on failure
|
|
|
|
*/
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONQueryDump(qemuMonitor *mon,
|
|
|
|
qemuMonitorDumpStats *stats)
|
2017-11-20 15:02:59 -05:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = qemuMonitorJSONMakeCommand("query-dump", NULL);
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *result = NULL;
|
2017-11-20 15:02:59 -05:00
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_OBJECT) < 0)
|
2017-11-20 15:02:59 -05:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
result = virJSONValueObjectGetObject(reply, "return");
|
|
|
|
|
|
|
|
ret = qemuMonitorJSONExtractDumpStats(result, stats);
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-23 11:51:13 +08:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetDumpGuestMemoryCapability(qemuMonitor *mon,
|
2014-03-23 11:51:13 +08:00
|
|
|
const char *capability)
|
|
|
|
{
|
2016-05-04 10:28:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *caps;
|
|
|
|
virJSONValue *formats;
|
2014-03-23 11:51:13 +08:00
|
|
|
size_t i;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-dump-guest-memory-capability",
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2014-03-23 11:51:13 +08:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONHasError(reply, "CommandNotFound")) {
|
|
|
|
ret = 0;
|
|
|
|
goto cleanup;
|
2014-03-23 11:51:13 +08:00
|
|
|
}
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_OBJECT) < 0)
|
2014-03-23 11:51:13 +08:00
|
|
|
goto cleanup;
|
|
|
|
|
2016-11-21 00:10:06 +01:00
|
|
|
caps = virJSONValueObjectGetObject(reply, "return");
|
2014-03-23 11:51:13 +08:00
|
|
|
|
qemu: simplify json parsing
Rather than grabbing an arbitrary JSON value and then checking
if it has the right type, we might as well request the correct
type to begin with.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONIOProcessEvent)
(qemuMonitorJSONCommandWithFd, qemuMonitorJSONHandleGraphics)
(qemuMonitorJSONGetStatus, qemuMonitorJSONExtractCPUInfo)
(qemuMonitorJSONGetVirtType, qemuMonitorJSONGetBalloonInfo)
(qemuMonitorJSONGetMemoryStats)
(qemuMonitorJSONDevGetBlockExtent)
(qemuMonitorJSONGetOneBlockStatsInfo)
(qemuMonitorJSONGetAllBlockStatsInfo)
(qemuMonitorJSONBlockStatsUpdateCapacityOne)
(qemuMonitorJSONBlockStatsUpdateCapacity)
(qemuMonitorJSONGetBlockExtent)
(qemuMonitorJSONGetMigrationStatusReply)
(qemuMonitorJSONGetDumpGuestMemoryCapability)
(qemuMonitorJSONAddFd, qemuMonitorJSONQueryRxFilterParse)
(qemuMonitorJSONExtractChardevInfo)
(qemuMonitorJSONDiskNameLookupOne)
(qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONGetAllBlockJobInfo)
(qemuMonitorJSONBlockIoThrottleInfo, qemuMonitorJSONGetVersion)
(qemuMonitorJSONGetMachines, qemuMonitorJSONGetCPUDefinitions)
(qemuMonitorJSONGetCommands, qemuMonitorJSONGetEvents)
(qemuMonitorJSONGetKVMState, qemuMonitorJSONGetObjectTypes)
(qemuMonitorJSONGetObjectListPaths)
(qemuMonitorJSONGetObjectProps, qemuMonitorJSONGetTargetArch)
(qemuMonitorJSONGetMigrationCapabilities)
(qemuMonitorJSONGetStringArray, qemuMonitorJSONAttachCharDev)
(qemuMonitorJSONGetCPUx86Data, qemuMonitorJSONGetIOThreads)
(qemuMonitorJSONGetMemoryDeviceInfo): Use shorter idioms.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-06-20 13:40:21 -06:00
|
|
|
if (!(formats = virJSONValueObjectGetArray(caps, "formats"))) {
|
2014-03-23 11:51:13 +08:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("missing supported dump formats"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < virJSONValueArraySize(formats); i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *dumpformat = virJSONValueArrayGet(formats, i);
|
2014-03-23 11:51:13 +08:00
|
|
|
|
2018-03-29 20:30:05 +02:00
|
|
|
if (!dumpformat || virJSONValueGetType(dumpformat) != VIR_JSON_TYPE_STRING) {
|
2014-03-23 11:51:13 +08:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("missing entry in supported dump formats"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (STREQ(virJSONValueGetString(dumpformat), capability)) {
|
|
|
|
ret = 1;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2014-03-23 11:51:13 +08:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2012-09-17 13:05:29 -06:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONDump(qemuMonitor *mon,
|
2014-03-23 11:51:14 +08:00
|
|
|
const char *protocol,
|
2017-11-20 15:05:23 -05:00
|
|
|
const char *dumpformat,
|
|
|
|
bool detach)
|
2012-06-12 11:04:51 +08:00
|
|
|
{
|
2016-05-04 10:28:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = NULL;
|
|
|
|
virJSONValue *reply = NULL;
|
2012-06-12 11:04:51 +08:00
|
|
|
|
2017-11-17 14:25:01 -05:00
|
|
|
cmd = qemuMonitorJSONMakeCommand("dump-guest-memory",
|
|
|
|
"b:paging", false,
|
|
|
|
"s:protocol", protocol,
|
|
|
|
"S:format", dumpformat,
|
2017-11-20 15:05:23 -05:00
|
|
|
"B:detach", detach,
|
2017-11-17 14:25:01 -05:00
|
|
|
NULL);
|
2012-06-12 11:04:51 +08:00
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2012-06-12 11:04:51 +08:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2012-06-12 11:04:51 +08:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2012-06-12 11:04:51 +08:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONGraphicsRelocate(qemuMonitor *mon,
|
2011-02-17 13:39:36 +00:00
|
|
|
int type,
|
|
|
|
const char *hostname,
|
|
|
|
int port,
|
|
|
|
int tlsPort,
|
|
|
|
const char *tlsSubject)
|
|
|
|
{
|
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = qemuMonitorJSONMakeCommand("client_migrate_info",
|
2011-02-17 13:39:36 +00:00
|
|
|
"s:protocol",
|
|
|
|
(type == VIR_DOMAIN_GRAPHICS_TYPE_SPICE ? "spice" : "vnc"),
|
|
|
|
"s:hostname", hostname,
|
|
|
|
"i:port", port,
|
|
|
|
"i:tls-port", tlsPort,
|
2014-05-13 17:28:45 +02:00
|
|
|
"S:cert-subject", tlsSubject,
|
2011-02-17 13:39:36 +00:00
|
|
|
NULL);
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *reply = NULL;
|
2011-02-17 13:39:36 +00:00
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2011-02-17 13:39:36 +00:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2011-02-17 13:39:36 +00:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2011-02-17 13:39:36 +00:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-14 12:08:28 -05:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAddfdInfoParse(virJSONValue *msg,
|
|
|
|
qemuMonitorAddFdInfo *fdinfo)
|
2020-10-14 12:08:28 -05:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *returnObj;
|
2020-10-14 12:08:28 -05:00
|
|
|
|
|
|
|
if (!(returnObj = virJSONValueObjectGetObject(msg, "return"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Missing or invalid return data in add-fd response"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberInt(returnObj, "fd", &fdinfo->fd) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Missing or invalid fd in add-fd response"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberInt(returnObj, "fdset-id", &fdinfo->fdset) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Missing or invalid fdset-id in add-fd response"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* if fdset is negative, qemu will create a new fdset and add the fd to that */
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONAddFileHandleToSet(qemuMonitor *mon,
|
2020-10-14 12:08:28 -05:00
|
|
|
int fd,
|
|
|
|
int fdset,
|
|
|
|
const char *opaque,
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorAddFdInfo *fdinfo)
|
2020-10-14 12:08:28 -05:00
|
|
|
{
|
2020-10-23 16:40:08 -05:00
|
|
|
g_autoptr(virJSONValue) args = NULL;
|
2020-10-14 12:08:28 -05:00
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
|
|
|
|
if (virJSONValueObjectCreate(&args, "S:opaque", opaque, NULL) < 0)
|
|
|
|
return -1;
|
|
|
|
|
2020-10-23 16:40:09 -05:00
|
|
|
if (fdset >= 0 &&
|
|
|
|
virJSONValueObjectAdd(args, "j:fdset-id", fdset, NULL) < 0) {
|
|
|
|
return -1;
|
|
|
|
}
|
2020-10-14 12:08:28 -05:00
|
|
|
|
2020-11-30 15:34:56 +01:00
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommandInternal("add-fd", &args)))
|
2020-10-14 12:08:28 -05:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommandWithFd(mon, cmd, fd, &reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuAddfdInfoParse(reply, fdinfo) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONQueryFdsetsParse(virJSONValue *msg,
|
|
|
|
qemuMonitorFdsets **fdsets)
|
2020-10-14 12:08:28 -05:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *returnArray;
|
|
|
|
virJSONValue *entry;
|
2020-10-14 12:08:28 -05:00
|
|
|
size_t i;
|
|
|
|
g_autoptr(qemuMonitorFdsets) sets = g_new0(qemuMonitorFdsets, 1);
|
|
|
|
int ninfo;
|
|
|
|
|
|
|
|
returnArray = virJSONValueObjectGetArray(msg, "return");
|
|
|
|
|
|
|
|
ninfo = virJSONValueArraySize(returnArray);
|
|
|
|
if (ninfo > 0)
|
|
|
|
sets->fdsets = g_new0(qemuMonitorFdsetInfo, ninfo);
|
|
|
|
sets->nfdsets = ninfo;
|
|
|
|
|
|
|
|
for (i = 0; i < ninfo; i++) {
|
|
|
|
size_t j;
|
|
|
|
const char *tmp;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *fdarray;
|
|
|
|
qemuMonitorFdsetInfo *fdsetinfo = &sets->fdsets[i];
|
2020-10-14 12:08:28 -05:00
|
|
|
|
|
|
|
if (!(entry = virJSONValueArrayGet(returnArray, i))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-fdsets return data missing fdset array element"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberInt(entry, "fdset-id", &fdsetinfo->id) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-fdsets reply was missing 'fdset-id'"));
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
fdarray = virJSONValueObjectGetArray(entry, "fds");
|
|
|
|
fdsetinfo->nfds = virJSONValueArraySize(fdarray);
|
|
|
|
if (fdsetinfo->nfds > 0)
|
|
|
|
fdsetinfo->fds = g_new0(qemuMonitorFdsetFdInfo, fdsetinfo->nfds);
|
|
|
|
|
|
|
|
for (j = 0; j < fdsetinfo->nfds; j++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorFdsetFdInfo *fdinfo = &fdsetinfo->fds[j];
|
|
|
|
virJSONValue *fdentry;
|
2020-10-14 12:08:28 -05:00
|
|
|
|
|
|
|
if (!(fdentry = virJSONValueArrayGet(fdarray, j))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-fdsets return data missing fd array element"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberInt(fdentry, "fd", &fdinfo->fd) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-fdsets return data missing 'fd'"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* opaque is optional and may be missing */
|
|
|
|
tmp = virJSONValueObjectGetString(fdentry, "opaque");
|
|
|
|
if (tmp)
|
|
|
|
fdinfo->opaque = g_strdup(tmp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*fdsets = g_steal_pointer(&sets);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONQueryFdsets(qemuMonitor *mon,
|
|
|
|
qemuMonitorFdsets **fdsets)
|
2020-10-14 12:08:28 -05:00
|
|
|
{
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
|
|
|
g_autoptr(virJSONValue) cmd = qemuMonitorJSONMakeCommand("query-fdsets",
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONQueryFdsetsParse(reply, fdsets) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONRemoveFdset(qemuMonitor *mon,
|
2020-10-14 12:08:28 -05:00
|
|
|
int fdset)
|
|
|
|
{
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
|
|
|
g_autoptr(virJSONValue) cmd = qemuMonitorJSONMakeCommand("remove-fd",
|
|
|
|
"i:fdset-id", fdset,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONSendFileHandle(qemuMonitor *mon,
|
2009-11-03 13:59:18 -05:00
|
|
|
const char *fdname,
|
|
|
|
int fd)
|
|
|
|
{
|
2016-05-04 10:28:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = qemuMonitorJSONMakeCommand("getfd",
|
2009-11-03 13:59:18 -05:00
|
|
|
"s:fdname", fdname,
|
|
|
|
NULL);
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *reply = NULL;
|
2009-11-03 13:59:18 -05:00
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommandWithFd(mon, cmd, fd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2009-11-03 13:59:18 -05:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONCloseFileHandle(qemuMonitor *mon,
|
2009-11-03 13:59:18 -05:00
|
|
|
const char *fdname)
|
|
|
|
{
|
2016-05-04 10:28:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = qemuMonitorJSONMakeCommand("closefd",
|
2009-11-03 13:59:18 -05:00
|
|
|
"s:fdname", fdname,
|
|
|
|
NULL);
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *reply = NULL;
|
2009-11-03 13:59:18 -05:00
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2009-11-03 13:59:18 -05:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2009-11-03 13:59:18 -05:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-05-15 11:16:32 +02:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONAddNetdev(qemuMonitor *mon,
|
|
|
|
virJSONValue **props)
|
2010-04-15 14:52:03 +01:00
|
|
|
{
|
2020-05-15 11:16:32 +02:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2010-04-15 14:52:03 +01:00
|
|
|
|
2020-11-30 15:34:56 +01:00
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommandInternal("netdev_add", props)))
|
2020-05-15 11:16:32 +02:00
|
|
|
return -1;
|
2010-04-15 14:52:03 +01:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2020-05-15 11:16:32 +02:00
|
|
|
return -1;
|
2010-04-15 14:52:03 +01:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
2020-05-15 11:16:32 +02:00
|
|
|
return -1;
|
2010-04-15 14:52:03 +01:00
|
|
|
|
2020-05-15 11:16:32 +02:00
|
|
|
return 0;
|
2010-04-15 14:52:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-05-15 11:16:32 +02:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONRemoveNetdev(qemuMonitor *mon,
|
2020-05-15 11:16:32 +02:00
|
|
|
const char *alias)
|
2010-04-15 14:52:03 +01:00
|
|
|
{
|
2020-05-15 11:16:32 +02:00
|
|
|
g_autoptr(virJSONValue) cmd = qemuMonitorJSONMakeCommand("netdev_del",
|
|
|
|
"s:id", alias,
|
|
|
|
NULL);
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
|
|
|
|
2010-04-15 14:52:03 +01:00
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2020-05-15 11:16:32 +02:00
|
|
|
return -1;
|
2010-04-15 14:52:03 +01:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
2020-05-15 11:16:32 +02:00
|
|
|
return -1;
|
2010-04-15 14:52:03 +01:00
|
|
|
|
2020-05-15 11:16:32 +02:00
|
|
|
return 0;
|
2010-04-15 14:52:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
qemu: qemuMonitorQueryRxFilter - retrieve guest netdev rx-filter
This function can be called at any time to get the current status of a
guest's network device rx-filter. In particular it is useful to call
after libvirt recieves a NIC_RX_FILTER_CHANGED event - this event only
tells you that something has changed in the rx-filter, the details are
retrieved with the query-rx-filter monitor command (only available in
the json monitor). The command sent to the qemu monitor looks like this:
{"execute":"query-rx-filter", "arguments": {"name":"net2"} }'
and the results will look something like this:
{
"return": [
{
"promiscuous": false,
"name": "net2",
"main-mac": "52:54:00:98:2d:e3",
"unicast": "normal",
"vlan": "normal",
"vlan-table": [
42,
0
],
"unicast-table": [
],
"multicast": "normal",
"multicast-overflow": false,
"unicast-overflow": false,
"multicast-table": [
"33:33:ff:98:2d:e3",
"01:80:c2:00:00:21",
"01:00:5e:00:00:fb",
"33:33:ff:98:2d:e2",
"01:00:5e:00:00:01",
"33:33:00:00:00:01"
],
"broadcast-allowed": false
}
],
"id": "libvirt-14"
}
This is all parsed from JSON into a virNetDevRxFilter object for
easier consumption. (unicast-table is usually empty, but is also an
array of mac addresses similar to multicast-table).
(NB: LIBNL_CFLAGS was added to tests/Makefile.am because virnetdev.h
now includes util/virnetlink.h, which includes netlink/msg.h when
appropriate. Without LIBNL_CFLAGS, gcc can't find that file (if
libnl/netlink isn't available, LIBNL_CFLAGS will be empty and
virnetlink.h won't try to include netlink/msg.h anyway).)
2014-09-22 12:19:41 -04:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONQueryRxFilterParse(virJSONValue *msg,
|
|
|
|
virNetDevRxFilter **filter)
|
qemu: qemuMonitorQueryRxFilter - retrieve guest netdev rx-filter
This function can be called at any time to get the current status of a
guest's network device rx-filter. In particular it is useful to call
after libvirt recieves a NIC_RX_FILTER_CHANGED event - this event only
tells you that something has changed in the rx-filter, the details are
retrieved with the query-rx-filter monitor command (only available in
the json monitor). The command sent to the qemu monitor looks like this:
{"execute":"query-rx-filter", "arguments": {"name":"net2"} }'
and the results will look something like this:
{
"return": [
{
"promiscuous": false,
"name": "net2",
"main-mac": "52:54:00:98:2d:e3",
"unicast": "normal",
"vlan": "normal",
"vlan-table": [
42,
0
],
"unicast-table": [
],
"multicast": "normal",
"multicast-overflow": false,
"unicast-overflow": false,
"multicast-table": [
"33:33:ff:98:2d:e3",
"01:80:c2:00:00:21",
"01:00:5e:00:00:fb",
"33:33:ff:98:2d:e2",
"01:00:5e:00:00:01",
"33:33:00:00:00:01"
],
"broadcast-allowed": false
}
],
"id": "libvirt-14"
}
This is all parsed from JSON into a virNetDevRxFilter object for
easier consumption. (unicast-table is usually empty, but is also an
array of mac addresses similar to multicast-table).
(NB: LIBNL_CFLAGS was added to tests/Makefile.am because virnetdev.h
now includes util/virnetlink.h, which includes netlink/msg.h when
appropriate. Without LIBNL_CFLAGS, gcc can't find that file (if
libnl/netlink isn't available, LIBNL_CFLAGS will be empty and
virnetlink.h won't try to include netlink/msg.h anyway).)
2014-09-22 12:19:41 -04:00
|
|
|
{
|
|
|
|
int ret = -1;
|
|
|
|
const char *tmp;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *returnArray;
|
|
|
|
virJSONValue *entry;
|
|
|
|
virJSONValue *table;
|
|
|
|
virJSONValue *element;
|
2018-04-19 17:29:02 -04:00
|
|
|
size_t nTable;
|
qemu: qemuMonitorQueryRxFilter - retrieve guest netdev rx-filter
This function can be called at any time to get the current status of a
guest's network device rx-filter. In particular it is useful to call
after libvirt recieves a NIC_RX_FILTER_CHANGED event - this event only
tells you that something has changed in the rx-filter, the details are
retrieved with the query-rx-filter monitor command (only available in
the json monitor). The command sent to the qemu monitor looks like this:
{"execute":"query-rx-filter", "arguments": {"name":"net2"} }'
and the results will look something like this:
{
"return": [
{
"promiscuous": false,
"name": "net2",
"main-mac": "52:54:00:98:2d:e3",
"unicast": "normal",
"vlan": "normal",
"vlan-table": [
42,
0
],
"unicast-table": [
],
"multicast": "normal",
"multicast-overflow": false,
"unicast-overflow": false,
"multicast-table": [
"33:33:ff:98:2d:e3",
"01:80:c2:00:00:21",
"01:00:5e:00:00:fb",
"33:33:ff:98:2d:e2",
"01:00:5e:00:00:01",
"33:33:00:00:00:01"
],
"broadcast-allowed": false
}
],
"id": "libvirt-14"
}
This is all parsed from JSON into a virNetDevRxFilter object for
easier consumption. (unicast-table is usually empty, but is also an
array of mac addresses similar to multicast-table).
(NB: LIBNL_CFLAGS was added to tests/Makefile.am because virnetdev.h
now includes util/virnetlink.h, which includes netlink/msg.h when
appropriate. Without LIBNL_CFLAGS, gcc can't find that file (if
libnl/netlink isn't available, LIBNL_CFLAGS will be empty and
virnetlink.h won't try to include netlink/msg.h anyway).)
2014-09-22 12:19:41 -04:00
|
|
|
size_t i;
|
2021-03-11 08:16:13 +01:00
|
|
|
virNetDevRxFilter *fil = virNetDevRxFilterNew();
|
qemu: qemuMonitorQueryRxFilter - retrieve guest netdev rx-filter
This function can be called at any time to get the current status of a
guest's network device rx-filter. In particular it is useful to call
after libvirt recieves a NIC_RX_FILTER_CHANGED event - this event only
tells you that something has changed in the rx-filter, the details are
retrieved with the query-rx-filter monitor command (only available in
the json monitor). The command sent to the qemu monitor looks like this:
{"execute":"query-rx-filter", "arguments": {"name":"net2"} }'
and the results will look something like this:
{
"return": [
{
"promiscuous": false,
"name": "net2",
"main-mac": "52:54:00:98:2d:e3",
"unicast": "normal",
"vlan": "normal",
"vlan-table": [
42,
0
],
"unicast-table": [
],
"multicast": "normal",
"multicast-overflow": false,
"unicast-overflow": false,
"multicast-table": [
"33:33:ff:98:2d:e3",
"01:80:c2:00:00:21",
"01:00:5e:00:00:fb",
"33:33:ff:98:2d:e2",
"01:00:5e:00:00:01",
"33:33:00:00:00:01"
],
"broadcast-allowed": false
}
],
"id": "libvirt-14"
}
This is all parsed from JSON into a virNetDevRxFilter object for
easier consumption. (unicast-table is usually empty, but is also an
array of mac addresses similar to multicast-table).
(NB: LIBNL_CFLAGS was added to tests/Makefile.am because virnetdev.h
now includes util/virnetlink.h, which includes netlink/msg.h when
appropriate. Without LIBNL_CFLAGS, gcc can't find that file (if
libnl/netlink isn't available, LIBNL_CFLAGS will be empty and
virnetlink.h won't try to include netlink/msg.h anyway).)
2014-09-22 12:19:41 -04:00
|
|
|
|
|
|
|
if (!fil)
|
|
|
|
goto cleanup;
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
returnArray = virJSONValueObjectGetArray(msg, "return");
|
|
|
|
|
qemu: qemuMonitorQueryRxFilter - retrieve guest netdev rx-filter
This function can be called at any time to get the current status of a
guest's network device rx-filter. In particular it is useful to call
after libvirt recieves a NIC_RX_FILTER_CHANGED event - this event only
tells you that something has changed in the rx-filter, the details are
retrieved with the query-rx-filter monitor command (only available in
the json monitor). The command sent to the qemu monitor looks like this:
{"execute":"query-rx-filter", "arguments": {"name":"net2"} }'
and the results will look something like this:
{
"return": [
{
"promiscuous": false,
"name": "net2",
"main-mac": "52:54:00:98:2d:e3",
"unicast": "normal",
"vlan": "normal",
"vlan-table": [
42,
0
],
"unicast-table": [
],
"multicast": "normal",
"multicast-overflow": false,
"unicast-overflow": false,
"multicast-table": [
"33:33:ff:98:2d:e3",
"01:80:c2:00:00:21",
"01:00:5e:00:00:fb",
"33:33:ff:98:2d:e2",
"01:00:5e:00:00:01",
"33:33:00:00:00:01"
],
"broadcast-allowed": false
}
],
"id": "libvirt-14"
}
This is all parsed from JSON into a virNetDevRxFilter object for
easier consumption. (unicast-table is usually empty, but is also an
array of mac addresses similar to multicast-table).
(NB: LIBNL_CFLAGS was added to tests/Makefile.am because virnetdev.h
now includes util/virnetlink.h, which includes netlink/msg.h when
appropriate. Without LIBNL_CFLAGS, gcc can't find that file (if
libnl/netlink isn't available, LIBNL_CFLAGS will be empty and
virnetlink.h won't try to include netlink/msg.h anyway).)
2014-09-22 12:19:41 -04:00
|
|
|
if (!(entry = virJSONValueArrayGet(returnArray, 0))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query -rx-filter return data missing array element"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(tmp = virJSONValueObjectGetString(entry, "name"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Missing or invalid name "
|
|
|
|
"in query-rx-filter response"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2019-10-20 13:49:46 +02:00
|
|
|
fil->name = g_strdup(tmp);
|
qemu: qemuMonitorQueryRxFilter - retrieve guest netdev rx-filter
This function can be called at any time to get the current status of a
guest's network device rx-filter. In particular it is useful to call
after libvirt recieves a NIC_RX_FILTER_CHANGED event - this event only
tells you that something has changed in the rx-filter, the details are
retrieved with the query-rx-filter monitor command (only available in
the json monitor). The command sent to the qemu monitor looks like this:
{"execute":"query-rx-filter", "arguments": {"name":"net2"} }'
and the results will look something like this:
{
"return": [
{
"promiscuous": false,
"name": "net2",
"main-mac": "52:54:00:98:2d:e3",
"unicast": "normal",
"vlan": "normal",
"vlan-table": [
42,
0
],
"unicast-table": [
],
"multicast": "normal",
"multicast-overflow": false,
"unicast-overflow": false,
"multicast-table": [
"33:33:ff:98:2d:e3",
"01:80:c2:00:00:21",
"01:00:5e:00:00:fb",
"33:33:ff:98:2d:e2",
"01:00:5e:00:00:01",
"33:33:00:00:00:01"
],
"broadcast-allowed": false
}
],
"id": "libvirt-14"
}
This is all parsed from JSON into a virNetDevRxFilter object for
easier consumption. (unicast-table is usually empty, but is also an
array of mac addresses similar to multicast-table).
(NB: LIBNL_CFLAGS was added to tests/Makefile.am because virnetdev.h
now includes util/virnetlink.h, which includes netlink/msg.h when
appropriate. Without LIBNL_CFLAGS, gcc can't find that file (if
libnl/netlink isn't available, LIBNL_CFLAGS will be empty and
virnetlink.h won't try to include netlink/msg.h anyway).)
2014-09-22 12:19:41 -04:00
|
|
|
if ((!(tmp = virJSONValueObjectGetString(entry, "main-mac"))) ||
|
|
|
|
virMacAddrParse(tmp, &fil->mac) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Missing or invalid 'main-mac' "
|
|
|
|
"in query-rx-filter response"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
if (virJSONValueObjectGetBoolean(entry, "promiscuous",
|
|
|
|
&fil->promiscuous) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Missing or invalid 'promiscuous' "
|
|
|
|
"in query-rx-filter response"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
if (virJSONValueObjectGetBoolean(entry, "broadcast-allowed",
|
|
|
|
&fil->broadcastAllowed) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Missing or invalid 'broadcast-allowed' "
|
|
|
|
"in query-rx-filter response"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((!(tmp = virJSONValueObjectGetString(entry, "unicast"))) ||
|
|
|
|
((fil->unicast.mode
|
|
|
|
= virNetDevRxFilterModeTypeFromString(tmp)) < 0)) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Missing or invalid 'unicast' "
|
|
|
|
"in query-rx-filter response"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
if (virJSONValueObjectGetBoolean(entry, "unicast-overflow",
|
|
|
|
&fil->unicast.overflow) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Missing or invalid 'unicast-overflow' "
|
|
|
|
"in query-rx-filter response"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
if ((!(table = virJSONValueObjectGet(entry, "unicast-table"))) ||
|
2018-04-19 17:29:02 -04:00
|
|
|
(!virJSONValueIsArray(table))) {
|
qemu: qemuMonitorQueryRxFilter - retrieve guest netdev rx-filter
This function can be called at any time to get the current status of a
guest's network device rx-filter. In particular it is useful to call
after libvirt recieves a NIC_RX_FILTER_CHANGED event - this event only
tells you that something has changed in the rx-filter, the details are
retrieved with the query-rx-filter monitor command (only available in
the json monitor). The command sent to the qemu monitor looks like this:
{"execute":"query-rx-filter", "arguments": {"name":"net2"} }'
and the results will look something like this:
{
"return": [
{
"promiscuous": false,
"name": "net2",
"main-mac": "52:54:00:98:2d:e3",
"unicast": "normal",
"vlan": "normal",
"vlan-table": [
42,
0
],
"unicast-table": [
],
"multicast": "normal",
"multicast-overflow": false,
"unicast-overflow": false,
"multicast-table": [
"33:33:ff:98:2d:e3",
"01:80:c2:00:00:21",
"01:00:5e:00:00:fb",
"33:33:ff:98:2d:e2",
"01:00:5e:00:00:01",
"33:33:00:00:00:01"
],
"broadcast-allowed": false
}
],
"id": "libvirt-14"
}
This is all parsed from JSON into a virNetDevRxFilter object for
easier consumption. (unicast-table is usually empty, but is also an
array of mac addresses similar to multicast-table).
(NB: LIBNL_CFLAGS was added to tests/Makefile.am because virnetdev.h
now includes util/virnetlink.h, which includes netlink/msg.h when
appropriate. Without LIBNL_CFLAGS, gcc can't find that file (if
libnl/netlink isn't available, LIBNL_CFLAGS will be empty and
virnetlink.h won't try to include netlink/msg.h anyway).)
2014-09-22 12:19:41 -04:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Missing or invalid 'unicast-table' array "
|
|
|
|
"in query-rx-filter response"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2018-04-19 17:29:02 -04:00
|
|
|
nTable = virJSONValueArraySize(table);
|
2020-10-05 12:26:34 +02:00
|
|
|
fil->unicast.table = g_new0(virMacAddr, nTable);
|
qemu: qemuMonitorQueryRxFilter - retrieve guest netdev rx-filter
This function can be called at any time to get the current status of a
guest's network device rx-filter. In particular it is useful to call
after libvirt recieves a NIC_RX_FILTER_CHANGED event - this event only
tells you that something has changed in the rx-filter, the details are
retrieved with the query-rx-filter monitor command (only available in
the json monitor). The command sent to the qemu monitor looks like this:
{"execute":"query-rx-filter", "arguments": {"name":"net2"} }'
and the results will look something like this:
{
"return": [
{
"promiscuous": false,
"name": "net2",
"main-mac": "52:54:00:98:2d:e3",
"unicast": "normal",
"vlan": "normal",
"vlan-table": [
42,
0
],
"unicast-table": [
],
"multicast": "normal",
"multicast-overflow": false,
"unicast-overflow": false,
"multicast-table": [
"33:33:ff:98:2d:e3",
"01:80:c2:00:00:21",
"01:00:5e:00:00:fb",
"33:33:ff:98:2d:e2",
"01:00:5e:00:00:01",
"33:33:00:00:00:01"
],
"broadcast-allowed": false
}
],
"id": "libvirt-14"
}
This is all parsed from JSON into a virNetDevRxFilter object for
easier consumption. (unicast-table is usually empty, but is also an
array of mac addresses similar to multicast-table).
(NB: LIBNL_CFLAGS was added to tests/Makefile.am because virnetdev.h
now includes util/virnetlink.h, which includes netlink/msg.h when
appropriate. Without LIBNL_CFLAGS, gcc can't find that file (if
libnl/netlink isn't available, LIBNL_CFLAGS will be empty and
virnetlink.h won't try to include netlink/msg.h anyway).)
2014-09-22 12:19:41 -04:00
|
|
|
for (i = 0; i < nTable; i++) {
|
|
|
|
if (!(element = virJSONValueArrayGet(table, i)) ||
|
|
|
|
!(tmp = virJSONValueGetString(element))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Missing or invalid element %zu of 'unicast' "
|
|
|
|
"list in query-rx-filter response"), i);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
if (virMacAddrParse(tmp, &fil->unicast.table[i]) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("invalid mac address '%s' in 'unicast-table' "
|
|
|
|
"array in query-rx-filter response"), tmp);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fil->unicast.nTable = nTable;
|
|
|
|
|
|
|
|
if ((!(tmp = virJSONValueObjectGetString(entry, "multicast"))) ||
|
|
|
|
((fil->multicast.mode
|
|
|
|
= virNetDevRxFilterModeTypeFromString(tmp)) < 0)) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Missing or invalid 'multicast' "
|
|
|
|
"in query-rx-filter response"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
if (virJSONValueObjectGetBoolean(entry, "multicast-overflow",
|
|
|
|
&fil->multicast.overflow) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Missing or invalid 'multicast-overflow' "
|
|
|
|
"in query-rx-filter response"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
if ((!(table = virJSONValueObjectGet(entry, "multicast-table"))) ||
|
2018-04-19 17:29:02 -04:00
|
|
|
(!virJSONValueIsArray(table))) {
|
qemu: qemuMonitorQueryRxFilter - retrieve guest netdev rx-filter
This function can be called at any time to get the current status of a
guest's network device rx-filter. In particular it is useful to call
after libvirt recieves a NIC_RX_FILTER_CHANGED event - this event only
tells you that something has changed in the rx-filter, the details are
retrieved with the query-rx-filter monitor command (only available in
the json monitor). The command sent to the qemu monitor looks like this:
{"execute":"query-rx-filter", "arguments": {"name":"net2"} }'
and the results will look something like this:
{
"return": [
{
"promiscuous": false,
"name": "net2",
"main-mac": "52:54:00:98:2d:e3",
"unicast": "normal",
"vlan": "normal",
"vlan-table": [
42,
0
],
"unicast-table": [
],
"multicast": "normal",
"multicast-overflow": false,
"unicast-overflow": false,
"multicast-table": [
"33:33:ff:98:2d:e3",
"01:80:c2:00:00:21",
"01:00:5e:00:00:fb",
"33:33:ff:98:2d:e2",
"01:00:5e:00:00:01",
"33:33:00:00:00:01"
],
"broadcast-allowed": false
}
],
"id": "libvirt-14"
}
This is all parsed from JSON into a virNetDevRxFilter object for
easier consumption. (unicast-table is usually empty, but is also an
array of mac addresses similar to multicast-table).
(NB: LIBNL_CFLAGS was added to tests/Makefile.am because virnetdev.h
now includes util/virnetlink.h, which includes netlink/msg.h when
appropriate. Without LIBNL_CFLAGS, gcc can't find that file (if
libnl/netlink isn't available, LIBNL_CFLAGS will be empty and
virnetlink.h won't try to include netlink/msg.h anyway).)
2014-09-22 12:19:41 -04:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Missing or invalid 'multicast-table' array "
|
|
|
|
"in query-rx-filter response"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2018-04-19 17:29:02 -04:00
|
|
|
nTable = virJSONValueArraySize(table);
|
2020-10-05 12:26:34 +02:00
|
|
|
fil->multicast.table = g_new0(virMacAddr, nTable);
|
qemu: qemuMonitorQueryRxFilter - retrieve guest netdev rx-filter
This function can be called at any time to get the current status of a
guest's network device rx-filter. In particular it is useful to call
after libvirt recieves a NIC_RX_FILTER_CHANGED event - this event only
tells you that something has changed in the rx-filter, the details are
retrieved with the query-rx-filter monitor command (only available in
the json monitor). The command sent to the qemu monitor looks like this:
{"execute":"query-rx-filter", "arguments": {"name":"net2"} }'
and the results will look something like this:
{
"return": [
{
"promiscuous": false,
"name": "net2",
"main-mac": "52:54:00:98:2d:e3",
"unicast": "normal",
"vlan": "normal",
"vlan-table": [
42,
0
],
"unicast-table": [
],
"multicast": "normal",
"multicast-overflow": false,
"unicast-overflow": false,
"multicast-table": [
"33:33:ff:98:2d:e3",
"01:80:c2:00:00:21",
"01:00:5e:00:00:fb",
"33:33:ff:98:2d:e2",
"01:00:5e:00:00:01",
"33:33:00:00:00:01"
],
"broadcast-allowed": false
}
],
"id": "libvirt-14"
}
This is all parsed from JSON into a virNetDevRxFilter object for
easier consumption. (unicast-table is usually empty, but is also an
array of mac addresses similar to multicast-table).
(NB: LIBNL_CFLAGS was added to tests/Makefile.am because virnetdev.h
now includes util/virnetlink.h, which includes netlink/msg.h when
appropriate. Without LIBNL_CFLAGS, gcc can't find that file (if
libnl/netlink isn't available, LIBNL_CFLAGS will be empty and
virnetlink.h won't try to include netlink/msg.h anyway).)
2014-09-22 12:19:41 -04:00
|
|
|
for (i = 0; i < nTable; i++) {
|
|
|
|
if (!(element = virJSONValueArrayGet(table, i)) ||
|
|
|
|
!(tmp = virJSONValueGetString(element))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Missing or invalid element %zu of 'multicast' "
|
|
|
|
"list in query-rx-filter response"), i);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
if (virMacAddrParse(tmp, &fil->multicast.table[i]) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("invalid mac address '%s' in 'multicast-table' "
|
|
|
|
"array in query-rx-filter response"), tmp);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fil->multicast.nTable = nTable;
|
|
|
|
|
|
|
|
if ((!(tmp = virJSONValueObjectGetString(entry, "vlan"))) ||
|
|
|
|
((fil->vlan.mode
|
|
|
|
= virNetDevRxFilterModeTypeFromString(tmp)) < 0)) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Missing or invalid 'vlan' "
|
|
|
|
"in query-rx-filter response"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
if ((!(table = virJSONValueObjectGet(entry, "vlan-table"))) ||
|
2018-04-19 17:29:02 -04:00
|
|
|
(!virJSONValueIsArray(table))) {
|
qemu: qemuMonitorQueryRxFilter - retrieve guest netdev rx-filter
This function can be called at any time to get the current status of a
guest's network device rx-filter. In particular it is useful to call
after libvirt recieves a NIC_RX_FILTER_CHANGED event - this event only
tells you that something has changed in the rx-filter, the details are
retrieved with the query-rx-filter monitor command (only available in
the json monitor). The command sent to the qemu monitor looks like this:
{"execute":"query-rx-filter", "arguments": {"name":"net2"} }'
and the results will look something like this:
{
"return": [
{
"promiscuous": false,
"name": "net2",
"main-mac": "52:54:00:98:2d:e3",
"unicast": "normal",
"vlan": "normal",
"vlan-table": [
42,
0
],
"unicast-table": [
],
"multicast": "normal",
"multicast-overflow": false,
"unicast-overflow": false,
"multicast-table": [
"33:33:ff:98:2d:e3",
"01:80:c2:00:00:21",
"01:00:5e:00:00:fb",
"33:33:ff:98:2d:e2",
"01:00:5e:00:00:01",
"33:33:00:00:00:01"
],
"broadcast-allowed": false
}
],
"id": "libvirt-14"
}
This is all parsed from JSON into a virNetDevRxFilter object for
easier consumption. (unicast-table is usually empty, but is also an
array of mac addresses similar to multicast-table).
(NB: LIBNL_CFLAGS was added to tests/Makefile.am because virnetdev.h
now includes util/virnetlink.h, which includes netlink/msg.h when
appropriate. Without LIBNL_CFLAGS, gcc can't find that file (if
libnl/netlink isn't available, LIBNL_CFLAGS will be empty and
virnetlink.h won't try to include netlink/msg.h anyway).)
2014-09-22 12:19:41 -04:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Missing or invalid 'vlan-table' array "
|
|
|
|
"in query-rx-filter response"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2018-04-19 17:29:02 -04:00
|
|
|
nTable = virJSONValueArraySize(table);
|
2020-10-05 12:26:34 +02:00
|
|
|
fil->vlan.table = g_new0(unsigned int, nTable);
|
qemu: qemuMonitorQueryRxFilter - retrieve guest netdev rx-filter
This function can be called at any time to get the current status of a
guest's network device rx-filter. In particular it is useful to call
after libvirt recieves a NIC_RX_FILTER_CHANGED event - this event only
tells you that something has changed in the rx-filter, the details are
retrieved with the query-rx-filter monitor command (only available in
the json monitor). The command sent to the qemu monitor looks like this:
{"execute":"query-rx-filter", "arguments": {"name":"net2"} }'
and the results will look something like this:
{
"return": [
{
"promiscuous": false,
"name": "net2",
"main-mac": "52:54:00:98:2d:e3",
"unicast": "normal",
"vlan": "normal",
"vlan-table": [
42,
0
],
"unicast-table": [
],
"multicast": "normal",
"multicast-overflow": false,
"unicast-overflow": false,
"multicast-table": [
"33:33:ff:98:2d:e3",
"01:80:c2:00:00:21",
"01:00:5e:00:00:fb",
"33:33:ff:98:2d:e2",
"01:00:5e:00:00:01",
"33:33:00:00:00:01"
],
"broadcast-allowed": false
}
],
"id": "libvirt-14"
}
This is all parsed from JSON into a virNetDevRxFilter object for
easier consumption. (unicast-table is usually empty, but is also an
array of mac addresses similar to multicast-table).
(NB: LIBNL_CFLAGS was added to tests/Makefile.am because virnetdev.h
now includes util/virnetlink.h, which includes netlink/msg.h when
appropriate. Without LIBNL_CFLAGS, gcc can't find that file (if
libnl/netlink isn't available, LIBNL_CFLAGS will be empty and
virnetlink.h won't try to include netlink/msg.h anyway).)
2014-09-22 12:19:41 -04:00
|
|
|
for (i = 0; i < nTable; i++) {
|
|
|
|
if (!(element = virJSONValueArrayGet(table, i)) ||
|
|
|
|
virJSONValueGetNumberUint(element, &fil->vlan.table[i]) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Missing or invalid element %zu of 'vlan-table' "
|
|
|
|
"array in query-rx-filter response"), i);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fil->vlan.nTable = nTable;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
|
|
if (ret < 0) {
|
|
|
|
virNetDevRxFilterFree(fil);
|
|
|
|
fil = NULL;
|
|
|
|
}
|
|
|
|
*filter = fil;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONQueryRxFilter(qemuMonitor *mon, const char *alias,
|
|
|
|
virNetDevRxFilter **filter)
|
qemu: qemuMonitorQueryRxFilter - retrieve guest netdev rx-filter
This function can be called at any time to get the current status of a
guest's network device rx-filter. In particular it is useful to call
after libvirt recieves a NIC_RX_FILTER_CHANGED event - this event only
tells you that something has changed in the rx-filter, the details are
retrieved with the query-rx-filter monitor command (only available in
the json monitor). The command sent to the qemu monitor looks like this:
{"execute":"query-rx-filter", "arguments": {"name":"net2"} }'
and the results will look something like this:
{
"return": [
{
"promiscuous": false,
"name": "net2",
"main-mac": "52:54:00:98:2d:e3",
"unicast": "normal",
"vlan": "normal",
"vlan-table": [
42,
0
],
"unicast-table": [
],
"multicast": "normal",
"multicast-overflow": false,
"unicast-overflow": false,
"multicast-table": [
"33:33:ff:98:2d:e3",
"01:80:c2:00:00:21",
"01:00:5e:00:00:fb",
"33:33:ff:98:2d:e2",
"01:00:5e:00:00:01",
"33:33:00:00:00:01"
],
"broadcast-allowed": false
}
],
"id": "libvirt-14"
}
This is all parsed from JSON into a virNetDevRxFilter object for
easier consumption. (unicast-table is usually empty, but is also an
array of mac addresses similar to multicast-table).
(NB: LIBNL_CFLAGS was added to tests/Makefile.am because virnetdev.h
now includes util/virnetlink.h, which includes netlink/msg.h when
appropriate. Without LIBNL_CFLAGS, gcc can't find that file (if
libnl/netlink isn't available, LIBNL_CFLAGS will be empty and
virnetlink.h won't try to include netlink/msg.h anyway).)
2014-09-22 12:19:41 -04:00
|
|
|
{
|
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = qemuMonitorJSONMakeCommand("query-rx-filter",
|
qemu: qemuMonitorQueryRxFilter - retrieve guest netdev rx-filter
This function can be called at any time to get the current status of a
guest's network device rx-filter. In particular it is useful to call
after libvirt recieves a NIC_RX_FILTER_CHANGED event - this event only
tells you that something has changed in the rx-filter, the details are
retrieved with the query-rx-filter monitor command (only available in
the json monitor). The command sent to the qemu monitor looks like this:
{"execute":"query-rx-filter", "arguments": {"name":"net2"} }'
and the results will look something like this:
{
"return": [
{
"promiscuous": false,
"name": "net2",
"main-mac": "52:54:00:98:2d:e3",
"unicast": "normal",
"vlan": "normal",
"vlan-table": [
42,
0
],
"unicast-table": [
],
"multicast": "normal",
"multicast-overflow": false,
"unicast-overflow": false,
"multicast-table": [
"33:33:ff:98:2d:e3",
"01:80:c2:00:00:21",
"01:00:5e:00:00:fb",
"33:33:ff:98:2d:e2",
"01:00:5e:00:00:01",
"33:33:00:00:00:01"
],
"broadcast-allowed": false
}
],
"id": "libvirt-14"
}
This is all parsed from JSON into a virNetDevRxFilter object for
easier consumption. (unicast-table is usually empty, but is also an
array of mac addresses similar to multicast-table).
(NB: LIBNL_CFLAGS was added to tests/Makefile.am because virnetdev.h
now includes util/virnetlink.h, which includes netlink/msg.h when
appropriate. Without LIBNL_CFLAGS, gcc can't find that file (if
libnl/netlink isn't available, LIBNL_CFLAGS will be empty and
virnetlink.h won't try to include netlink/msg.h anyway).)
2014-09-22 12:19:41 -04:00
|
|
|
"s:name", alias,
|
|
|
|
NULL);
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *reply = NULL;
|
qemu: qemuMonitorQueryRxFilter - retrieve guest netdev rx-filter
This function can be called at any time to get the current status of a
guest's network device rx-filter. In particular it is useful to call
after libvirt recieves a NIC_RX_FILTER_CHANGED event - this event only
tells you that something has changed in the rx-filter, the details are
retrieved with the query-rx-filter monitor command (only available in
the json monitor). The command sent to the qemu monitor looks like this:
{"execute":"query-rx-filter", "arguments": {"name":"net2"} }'
and the results will look something like this:
{
"return": [
{
"promiscuous": false,
"name": "net2",
"main-mac": "52:54:00:98:2d:e3",
"unicast": "normal",
"vlan": "normal",
"vlan-table": [
42,
0
],
"unicast-table": [
],
"multicast": "normal",
"multicast-overflow": false,
"unicast-overflow": false,
"multicast-table": [
"33:33:ff:98:2d:e3",
"01:80:c2:00:00:21",
"01:00:5e:00:00:fb",
"33:33:ff:98:2d:e2",
"01:00:5e:00:00:01",
"33:33:00:00:00:01"
],
"broadcast-allowed": false
}
],
"id": "libvirt-14"
}
This is all parsed from JSON into a virNetDevRxFilter object for
easier consumption. (unicast-table is usually empty, but is also an
array of mac addresses similar to multicast-table).
(NB: LIBNL_CFLAGS was added to tests/Makefile.am because virnetdev.h
now includes util/virnetlink.h, which includes netlink/msg.h when
appropriate. Without LIBNL_CFLAGS, gcc can't find that file (if
libnl/netlink isn't available, LIBNL_CFLAGS will be empty and
virnetlink.h won't try to include netlink/msg.h anyway).)
2014-09-22 12:19:41 -04:00
|
|
|
|
|
|
|
if (!cmd)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_ARRAY) < 0)
|
2016-05-03 10:44:13 +02:00
|
|
|
goto cleanup;
|
|
|
|
|
qemu: qemuMonitorQueryRxFilter - retrieve guest netdev rx-filter
This function can be called at any time to get the current status of a
guest's network device rx-filter. In particular it is useful to call
after libvirt recieves a NIC_RX_FILTER_CHANGED event - this event only
tells you that something has changed in the rx-filter, the details are
retrieved with the query-rx-filter monitor command (only available in
the json monitor). The command sent to the qemu monitor looks like this:
{"execute":"query-rx-filter", "arguments": {"name":"net2"} }'
and the results will look something like this:
{
"return": [
{
"promiscuous": false,
"name": "net2",
"main-mac": "52:54:00:98:2d:e3",
"unicast": "normal",
"vlan": "normal",
"vlan-table": [
42,
0
],
"unicast-table": [
],
"multicast": "normal",
"multicast-overflow": false,
"unicast-overflow": false,
"multicast-table": [
"33:33:ff:98:2d:e3",
"01:80:c2:00:00:21",
"01:00:5e:00:00:fb",
"33:33:ff:98:2d:e2",
"01:00:5e:00:00:01",
"33:33:00:00:00:01"
],
"broadcast-allowed": false
}
],
"id": "libvirt-14"
}
This is all parsed from JSON into a virNetDevRxFilter object for
easier consumption. (unicast-table is usually empty, but is also an
array of mac addresses similar to multicast-table).
(NB: LIBNL_CFLAGS was added to tests/Makefile.am because virnetdev.h
now includes util/virnetlink.h, which includes netlink/msg.h when
appropriate. Without LIBNL_CFLAGS, gcc can't find that file (if
libnl/netlink isn't available, LIBNL_CFLAGS will be empty and
virnetlink.h won't try to include netlink/msg.h anyway).)
2014-09-22 12:19:41 -04:00
|
|
|
if (qemuMonitorJSONQueryRxFilterParse(reply, filter) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
|
|
if (ret < 0) {
|
|
|
|
virNetDevRxFilterFree(*filter);
|
|
|
|
*filter = NULL;
|
|
|
|
}
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-01-22 13:22:53 +00:00
|
|
|
/*
|
|
|
|
* Example return data
|
|
|
|
*
|
|
|
|
* {"return": [
|
|
|
|
* {"filename": "stdio", "label": "monitor"},
|
2014-11-13 19:29:14 +01:00
|
|
|
* {"filename": "pty:/dev/pts/6", "label": "serial0", "frontend-open": true},
|
2010-01-22 13:22:53 +00:00
|
|
|
* {"filename": "pty:/dev/pts/7", "label": "parallel0"}
|
|
|
|
* ]}
|
|
|
|
*
|
|
|
|
*/
|
2014-11-13 16:17:21 +01:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONExtractChardevInfo(virJSONValue *reply,
|
2020-10-22 19:04:18 +02:00
|
|
|
GHashTable *info)
|
2010-01-22 13:22:53 +00:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *data;
|
2010-01-22 13:22:53 +00:00
|
|
|
int ret = -1;
|
Convert 'int i' to 'size_t i' in src/qemu files
Convert the type of loop iterators named 'i', 'j', k',
'ii', 'jj', 'kk', to be 'size_t' instead of 'int' or
'unsigned int', also santizing 'ii', 'jj', 'kk' to use
the normal 'i', 'j', 'k' naming
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2013-07-08 15:09:33 +01:00
|
|
|
size_t i;
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorChardevInfo *entry = NULL;
|
2010-01-22 13:22:53 +00:00
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
data = virJSONValueObjectGetArray(reply, "return");
|
2010-01-22 13:22:53 +00:00
|
|
|
|
2013-05-21 15:21:20 +08:00
|
|
|
for (i = 0; i < virJSONValueArraySize(data); i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *chardev = virJSONValueArrayGet(data, i);
|
2010-01-22 13:22:53 +00:00
|
|
|
const char *type;
|
2014-11-13 19:29:14 +01:00
|
|
|
const char *alias;
|
|
|
|
bool connected;
|
|
|
|
|
|
|
|
if (!chardev) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("character device information was missing array element"));
|
2010-01-22 13:22:53 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2014-11-13 19:29:14 +01:00
|
|
|
if (!(alias = virJSONValueObjectGetString(chardev, "label"))) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2014-11-13 19:29:14 +01:00
|
|
|
_("character device information was missing label"));
|
2010-01-22 13:22:53 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2014-11-13 19:29:14 +01:00
|
|
|
if (!(type = virJSONValueObjectGetString(chardev, "filename"))) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("character device information was missing filename"));
|
2010-01-22 13:22:53 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2020-10-05 12:26:34 +02:00
|
|
|
entry = g_new0(qemuMonitorChardevInfo, 1);
|
2010-01-22 13:22:53 +00:00
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
if (STRPREFIX(type, "pty:"))
|
|
|
|
entry->ptyPath = g_strdup(type + strlen("pty:"));
|
2014-11-13 19:29:14 +01:00
|
|
|
|
|
|
|
if (virJSONValueObjectGetBoolean(chardev, "frontend-open", &connected) == 0) {
|
|
|
|
if (connected)
|
|
|
|
entry->state = VIR_DOMAIN_CHR_DEVICE_STATE_CONNECTED;
|
|
|
|
else
|
|
|
|
entry->state = VIR_DOMAIN_CHR_DEVICE_STATE_DISCONNECTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virHashAddEntry(info, alias, entry) < 0) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED,
|
|
|
|
_("failed to add chardev '%s' info"), alias);
|
|
|
|
goto cleanup;
|
2010-01-22 13:22:53 +00:00
|
|
|
}
|
2014-11-13 19:29:14 +01:00
|
|
|
|
|
|
|
entry = NULL;
|
2010-01-22 13:22:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2014-11-13 19:29:14 +01:00
|
|
|
if (entry) {
|
|
|
|
VIR_FREE(entry->ptyPath);
|
|
|
|
VIR_FREE(entry);
|
|
|
|
}
|
|
|
|
|
2010-01-22 13:22:53 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2014-11-13 16:17:21 +01:00
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetChardevInfo(qemuMonitor *mon,
|
2020-10-22 19:04:18 +02:00
|
|
|
GHashTable *info)
|
2010-01-22 13:22:53 +00:00
|
|
|
|
|
|
|
{
|
2016-05-04 10:28:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = qemuMonitorJSONMakeCommand("query-chardev",
|
2010-01-22 13:22:53 +00:00
|
|
|
NULL);
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *reply = NULL;
|
2010-01-22 13:22:53 +00:00
|
|
|
|
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2010-01-22 13:22:53 +00:00
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_ARRAY) < 0)
|
2016-05-04 10:28:43 +02:00
|
|
|
goto cleanup;
|
2010-01-22 13:22:53 +00:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = qemuMonitorJSONExtractChardevInfo(reply, info);
|
|
|
|
cleanup:
|
2010-01-22 13:22:53 +00:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONDelDevice(qemuMonitor *mon,
|
2010-04-14 15:36:42 +01:00
|
|
|
const char *devalias)
|
2010-03-02 09:40:51 +01:00
|
|
|
{
|
2016-05-04 10:28:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2010-03-02 09:40:51 +01:00
|
|
|
|
|
|
|
cmd = qemuMonitorJSONMakeCommand("device_del",
|
2010-04-15 12:17:29 +01:00
|
|
|
"s:id", devalias,
|
2010-03-02 09:40:51 +01:00
|
|
|
NULL);
|
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2010-03-02 09:40:51 +01:00
|
|
|
|
2019-03-14 08:46:37 +01:00
|
|
|
if (qemuMonitorJSONHasError(reply, "DeviceNotFound")) {
|
|
|
|
ret = -2;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2010-03-02 09:40:51 +01:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2010-03-02 09:40:51 +01:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-07-31 15:26:12 +02:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONAddDeviceArgs(qemuMonitor *mon,
|
|
|
|
virJSONValue *args)
|
2010-01-26 15:34:46 +00:00
|
|
|
{
|
2010-04-14 16:02:37 +01:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = NULL;
|
|
|
|
virJSONValue *reply = NULL;
|
2010-01-26 15:34:46 +00:00
|
|
|
|
2016-07-31 15:26:12 +02:00
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("device_add", NULL)))
|
2010-04-14 16:02:37 +01:00
|
|
|
goto cleanup;
|
|
|
|
|
2021-02-11 17:57:45 +01:00
|
|
|
if (virJSONValueObjectAppend(cmd, "arguments", &args) < 0)
|
2010-04-14 16:02:37 +01:00
|
|
|
goto cleanup;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2010-01-26 15:34:46 +00:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2010-01-26 15:34:46 +00:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2010-04-14 16:02:37 +01:00
|
|
|
virJSONValueFree(args);
|
2010-01-26 15:34:46 +00:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-07-31 15:26:12 +02:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONAddDevice(qemuMonitor *mon,
|
2016-07-31 15:26:12 +02:00
|
|
|
const char *devicestr)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *args;
|
2016-07-31 15:26:12 +02:00
|
|
|
|
|
|
|
if (!(args = qemuMonitorJSONKeywordStringToJSON(devicestr, "driver")))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return qemuMonitorJSONAddDeviceArgs(mon, args);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-05-17 16:43:58 +02:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONAddObject(qemuMonitor *mon,
|
|
|
|
virJSONValue **props)
|
2014-09-23 13:25:25 +02:00
|
|
|
{
|
2020-03-18 10:29:41 +01:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2014-09-23 13:25:25 +02:00
|
|
|
|
2020-11-30 15:34:56 +01:00
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommandInternal("object-add", props)))
|
2020-03-18 10:29:41 +01:00
|
|
|
return -1;
|
2014-09-23 13:25:25 +02:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2020-03-18 10:29:41 +01:00
|
|
|
return -1;
|
2014-09-23 13:25:25 +02:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
2020-03-18 10:29:41 +01:00
|
|
|
return -1;
|
2014-09-23 13:25:25 +02:00
|
|
|
|
2020-03-18 10:29:41 +01:00
|
|
|
return 0;
|
2014-09-23 13:25:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-03-18 10:29:41 +01:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONDelObject(qemuMonitor *mon,
|
2020-03-18 12:24:40 +01:00
|
|
|
const char *objalias,
|
|
|
|
bool report_error)
|
2014-09-23 13:25:25 +02:00
|
|
|
{
|
2020-03-18 10:29:41 +01:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2014-09-23 13:25:25 +02:00
|
|
|
|
2020-03-18 10:29:41 +01:00
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("object-del", "s:id", objalias, NULL)))
|
2014-09-23 13:25:25 +02:00
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2020-03-18 10:29:41 +01:00
|
|
|
return -1;
|
2014-09-23 13:25:25 +02:00
|
|
|
|
2020-03-18 12:24:40 +01:00
|
|
|
if (qemuMonitorJSONCheckErrorFull(cmd, reply, report_error) < 0)
|
2020-03-18 10:29:41 +01:00
|
|
|
return -1;
|
2014-09-23 13:25:25 +02:00
|
|
|
|
2020-03-18 10:29:41 +01:00
|
|
|
return 0;
|
2014-09-23 13:25:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
blockjob: add qemu capabilities related to block jobs
Upstream qemu 1.3 is adding two new monitor commands, 'drive-mirror'
and 'block-job-complete'[1], which can drive live block copy and
storage migration. [Additionally, RHEL 6.3 had backported an earlier
version of most of the same functionality, but under the names
'__com.redhat_drive-mirror' and '__com.redhat_drive-reopen' and with
slightly different JSON arguments, and has been using patches similar
to these upstream patches for several months now.]
The libvirt API virDomainBlockRebase as already committed for 0.9.12
is flexible enough to expose the basics of block copy, but some
additional features in the 'drive-mirror' qemu command, such as
setting error policy, setting granularity, or using a persistent
bitmap, may later require a new libvirt API virDomainBlockCopy. I
will wait to add that API until we know more about what qemu 1.3
will finally provide.
This patch caters only to the upstream qemu 1.3 interface, although
I have proven that the changes for RHEL 6.3 can be isolated to
just qemu_monitor_json.c, and the rest of this series will
gracefully handle either interface once the JSON differences are
papered over in a downstream patch.
For consistency with other block job commands, libvirt must handle
the bandwidth argument as MiB/sec from the user, even though qemu
exposes the speed argument as bytes/sec; then again, qemu rounds
up to cluster size internally, so using MiB hides the worst effects
of that rounding if you pass small numbers.
[1]https://lists.gnu.org/archive/html/qemu-devel/2012-10/msg04123.html
* src/qemu/qemu_capabilities.h (QEMU_CAPS_DRIVE_MIRROR)
(QEMU_CAPS_DRIVE_REOPEN): New bits.
* src/qemu/qemu_capabilities.c (qemuCaps): Name them.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONCheckCommands): Set
them.
(qemuMonitorJSONDriveMirror, qemuMonitorDrivePivot): New functions.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDriveMirror)
(qemuMonitorDrivePivot): Declare them.
* src/qemu/qemu_monitor.c (qemuMonitorDriveMirror)
(qemuMonitorDrivePivot): New passthroughs.
* src/qemu/qemu_monitor.h (qemuMonitorDriveMirror)
(qemuMonitorDrivePivot): Declare them.
2012-09-28 17:29:53 -06:00
|
|
|
/* speed is in bytes/sec */
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONDriveMirror(qemuMonitor *mon,
|
blockjob: add qemu capabilities related to block jobs
Upstream qemu 1.3 is adding two new monitor commands, 'drive-mirror'
and 'block-job-complete'[1], which can drive live block copy and
storage migration. [Additionally, RHEL 6.3 had backported an earlier
version of most of the same functionality, but under the names
'__com.redhat_drive-mirror' and '__com.redhat_drive-reopen' and with
slightly different JSON arguments, and has been using patches similar
to these upstream patches for several months now.]
The libvirt API virDomainBlockRebase as already committed for 0.9.12
is flexible enough to expose the basics of block copy, but some
additional features in the 'drive-mirror' qemu command, such as
setting error policy, setting granularity, or using a persistent
bitmap, may later require a new libvirt API virDomainBlockCopy. I
will wait to add that API until we know more about what qemu 1.3
will finally provide.
This patch caters only to the upstream qemu 1.3 interface, although
I have proven that the changes for RHEL 6.3 can be isolated to
just qemu_monitor_json.c, and the rest of this series will
gracefully handle either interface once the JSON differences are
papered over in a downstream patch.
For consistency with other block job commands, libvirt must handle
the bandwidth argument as MiB/sec from the user, even though qemu
exposes the speed argument as bytes/sec; then again, qemu rounds
up to cluster size internally, so using MiB hides the worst effects
of that rounding if you pass small numbers.
[1]https://lists.gnu.org/archive/html/qemu-devel/2012-10/msg04123.html
* src/qemu/qemu_capabilities.h (QEMU_CAPS_DRIVE_MIRROR)
(QEMU_CAPS_DRIVE_REOPEN): New bits.
* src/qemu/qemu_capabilities.c (qemuCaps): Name them.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONCheckCommands): Set
them.
(qemuMonitorJSONDriveMirror, qemuMonitorDrivePivot): New functions.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDriveMirror)
(qemuMonitorDrivePivot): Declare them.
* src/qemu/qemu_monitor.c (qemuMonitorDriveMirror)
(qemuMonitorDrivePivot): New passthroughs.
* src/qemu/qemu_monitor.h (qemuMonitorDriveMirror)
(qemuMonitorDrivePivot): Declare them.
2012-09-28 17:29:53 -06:00
|
|
|
const char *device, const char *file,
|
|
|
|
const char *format, unsigned long long speed,
|
blockcopy: add qemu implementation of new tunables
Upstream qemu 1.4 added some drive-mirror tunables not present
when it was first introduced in 1.3. Management apps may want
to set these in some cases (for example, without tuning
granularity down to sector size, a copy may end up occupying
more bytes than the original because an entire cluster is
copied even when only a sector within the cluster is dirty,
although tuning it down results in more CPU time to do the
copy). I haven't personally needed to use the parameters, but
since they exist, and since the new API supports virTypedParams,
we might as well expose them.
Since the tuning parameters aren't often used, and omitted from
the QMP command when unspecified, I think it is safe to rely on
qemu 1.3 to issue an error about them being unsupported, rather
than trying to create a new capability bit in libvirt.
Meanwhile, all versions of qemu from 1.4 to 2.1 have a bug where
a bad granularity (such as non-power-of-2) gives a poor message:
error: internal error: unable to execute QEMU command 'drive-mirror': Invalid parameter 'drive-virtio-disk0'
because of abuse of QERR_INVALID_PARAMETER (which is supposed to
name the parameter that was given a bad value, rather than the
value passed to some other parameter). I don't see that a
capability check will help, so we'll just live with it (and it
has since been improved in upstream qemu).
* src/qemu/qemu_monitor.h (qemuMonitorDriveMirror): Add
parameters.
* src/qemu/qemu_monitor.c (qemuMonitorDriveMirror): Likewise.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDriveMirror):
Likewise.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONDriveMirror):
Likewise.
* src/qemu/qemu_driver.c (qemuDomainBlockCopyCommon): Likewise.
(qemuDomainBlockRebase, qemuDomainBlockCopy): Adjust callers.
* src/qemu/qemu_migration.c (qemuMigrationDriveMirror): Likewise.
* tests/qemumonitorjsontest.c (qemuMonitorJSONDriveMirror): Likewise.
Signed-off-by: Eric Blake <eblake@redhat.com>
2014-09-08 14:53:12 -06:00
|
|
|
unsigned int granularity,
|
|
|
|
unsigned long long buf_size,
|
2019-05-17 18:13:53 +02:00
|
|
|
bool shallow,
|
|
|
|
bool reuse)
|
blockjob: add qemu capabilities related to block jobs
Upstream qemu 1.3 is adding two new monitor commands, 'drive-mirror'
and 'block-job-complete'[1], which can drive live block copy and
storage migration. [Additionally, RHEL 6.3 had backported an earlier
version of most of the same functionality, but under the names
'__com.redhat_drive-mirror' and '__com.redhat_drive-reopen' and with
slightly different JSON arguments, and has been using patches similar
to these upstream patches for several months now.]
The libvirt API virDomainBlockRebase as already committed for 0.9.12
is flexible enough to expose the basics of block copy, but some
additional features in the 'drive-mirror' qemu command, such as
setting error policy, setting granularity, or using a persistent
bitmap, may later require a new libvirt API virDomainBlockCopy. I
will wait to add that API until we know more about what qemu 1.3
will finally provide.
This patch caters only to the upstream qemu 1.3 interface, although
I have proven that the changes for RHEL 6.3 can be isolated to
just qemu_monitor_json.c, and the rest of this series will
gracefully handle either interface once the JSON differences are
papered over in a downstream patch.
For consistency with other block job commands, libvirt must handle
the bandwidth argument as MiB/sec from the user, even though qemu
exposes the speed argument as bytes/sec; then again, qemu rounds
up to cluster size internally, so using MiB hides the worst effects
of that rounding if you pass small numbers.
[1]https://lists.gnu.org/archive/html/qemu-devel/2012-10/msg04123.html
* src/qemu/qemu_capabilities.h (QEMU_CAPS_DRIVE_MIRROR)
(QEMU_CAPS_DRIVE_REOPEN): New bits.
* src/qemu/qemu_capabilities.c (qemuCaps): Name them.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONCheckCommands): Set
them.
(qemuMonitorJSONDriveMirror, qemuMonitorDrivePivot): New functions.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDriveMirror)
(qemuMonitorDrivePivot): Declare them.
* src/qemu/qemu_monitor.c (qemuMonitorDriveMirror)
(qemuMonitorDrivePivot): New passthroughs.
* src/qemu/qemu_monitor.h (qemuMonitorDriveMirror)
(qemuMonitorDrivePivot): Declare them.
2012-09-28 17:29:53 -06:00
|
|
|
{
|
2019-10-15 14:47:50 +02:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
blockjob: add qemu capabilities related to block jobs
Upstream qemu 1.3 is adding two new monitor commands, 'drive-mirror'
and 'block-job-complete'[1], which can drive live block copy and
storage migration. [Additionally, RHEL 6.3 had backported an earlier
version of most of the same functionality, but under the names
'__com.redhat_drive-mirror' and '__com.redhat_drive-reopen' and with
slightly different JSON arguments, and has been using patches similar
to these upstream patches for several months now.]
The libvirt API virDomainBlockRebase as already committed for 0.9.12
is flexible enough to expose the basics of block copy, but some
additional features in the 'drive-mirror' qemu command, such as
setting error policy, setting granularity, or using a persistent
bitmap, may later require a new libvirt API virDomainBlockCopy. I
will wait to add that API until we know more about what qemu 1.3
will finally provide.
This patch caters only to the upstream qemu 1.3 interface, although
I have proven that the changes for RHEL 6.3 can be isolated to
just qemu_monitor_json.c, and the rest of this series will
gracefully handle either interface once the JSON differences are
papered over in a downstream patch.
For consistency with other block job commands, libvirt must handle
the bandwidth argument as MiB/sec from the user, even though qemu
exposes the speed argument as bytes/sec; then again, qemu rounds
up to cluster size internally, so using MiB hides the worst effects
of that rounding if you pass small numbers.
[1]https://lists.gnu.org/archive/html/qemu-devel/2012-10/msg04123.html
* src/qemu/qemu_capabilities.h (QEMU_CAPS_DRIVE_MIRROR)
(QEMU_CAPS_DRIVE_REOPEN): New bits.
* src/qemu/qemu_capabilities.c (qemuCaps): Name them.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONCheckCommands): Set
them.
(qemuMonitorJSONDriveMirror, qemuMonitorDrivePivot): New functions.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDriveMirror)
(qemuMonitorDrivePivot): Declare them.
* src/qemu/qemu_monitor.c (qemuMonitorDriveMirror)
(qemuMonitorDrivePivot): New passthroughs.
* src/qemu/qemu_monitor.h (qemuMonitorDriveMirror)
(qemuMonitorDrivePivot): Declare them.
2012-09-28 17:29:53 -06:00
|
|
|
|
|
|
|
cmd = qemuMonitorJSONMakeCommand("drive-mirror",
|
|
|
|
"s:device", device,
|
|
|
|
"s:target", file,
|
2014-09-05 05:31:47 -06:00
|
|
|
"Y:speed", speed,
|
blockcopy: add qemu implementation of new tunables
Upstream qemu 1.4 added some drive-mirror tunables not present
when it was first introduced in 1.3. Management apps may want
to set these in some cases (for example, without tuning
granularity down to sector size, a copy may end up occupying
more bytes than the original because an entire cluster is
copied even when only a sector within the cluster is dirty,
although tuning it down results in more CPU time to do the
copy). I haven't personally needed to use the parameters, but
since they exist, and since the new API supports virTypedParams,
we might as well expose them.
Since the tuning parameters aren't often used, and omitted from
the QMP command when unspecified, I think it is safe to rely on
qemu 1.3 to issue an error about them being unsupported, rather
than trying to create a new capability bit in libvirt.
Meanwhile, all versions of qemu from 1.4 to 2.1 have a bug where
a bad granularity (such as non-power-of-2) gives a poor message:
error: internal error: unable to execute QEMU command 'drive-mirror': Invalid parameter 'drive-virtio-disk0'
because of abuse of QERR_INVALID_PARAMETER (which is supposed to
name the parameter that was given a bad value, rather than the
value passed to some other parameter). I don't see that a
capability check will help, so we'll just live with it (and it
has since been improved in upstream qemu).
* src/qemu/qemu_monitor.h (qemuMonitorDriveMirror): Add
parameters.
* src/qemu/qemu_monitor.c (qemuMonitorDriveMirror): Likewise.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDriveMirror):
Likewise.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONDriveMirror):
Likewise.
* src/qemu/qemu_driver.c (qemuDomainBlockCopyCommon): Likewise.
(qemuDomainBlockRebase, qemuDomainBlockCopy): Adjust callers.
* src/qemu/qemu_migration.c (qemuMigrationDriveMirror): Likewise.
* tests/qemumonitorjsontest.c (qemuMonitorJSONDriveMirror): Likewise.
Signed-off-by: Eric Blake <eblake@redhat.com>
2014-09-08 14:53:12 -06:00
|
|
|
"z:granularity", granularity,
|
|
|
|
"P:buf-size", buf_size,
|
blockjob: add qemu capabilities related to block jobs
Upstream qemu 1.3 is adding two new monitor commands, 'drive-mirror'
and 'block-job-complete'[1], which can drive live block copy and
storage migration. [Additionally, RHEL 6.3 had backported an earlier
version of most of the same functionality, but under the names
'__com.redhat_drive-mirror' and '__com.redhat_drive-reopen' and with
slightly different JSON arguments, and has been using patches similar
to these upstream patches for several months now.]
The libvirt API virDomainBlockRebase as already committed for 0.9.12
is flexible enough to expose the basics of block copy, but some
additional features in the 'drive-mirror' qemu command, such as
setting error policy, setting granularity, or using a persistent
bitmap, may later require a new libvirt API virDomainBlockCopy. I
will wait to add that API until we know more about what qemu 1.3
will finally provide.
This patch caters only to the upstream qemu 1.3 interface, although
I have proven that the changes for RHEL 6.3 can be isolated to
just qemu_monitor_json.c, and the rest of this series will
gracefully handle either interface once the JSON differences are
papered over in a downstream patch.
For consistency with other block job commands, libvirt must handle
the bandwidth argument as MiB/sec from the user, even though qemu
exposes the speed argument as bytes/sec; then again, qemu rounds
up to cluster size internally, so using MiB hides the worst effects
of that rounding if you pass small numbers.
[1]https://lists.gnu.org/archive/html/qemu-devel/2012-10/msg04123.html
* src/qemu/qemu_capabilities.h (QEMU_CAPS_DRIVE_MIRROR)
(QEMU_CAPS_DRIVE_REOPEN): New bits.
* src/qemu/qemu_capabilities.c (qemuCaps): Name them.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONCheckCommands): Set
them.
(qemuMonitorJSONDriveMirror, qemuMonitorDrivePivot): New functions.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDriveMirror)
(qemuMonitorDrivePivot): Declare them.
* src/qemu/qemu_monitor.c (qemuMonitorDriveMirror)
(qemuMonitorDrivePivot): New passthroughs.
* src/qemu/qemu_monitor.h (qemuMonitorDriveMirror)
(qemuMonitorDrivePivot): Declare them.
2012-09-28 17:29:53 -06:00
|
|
|
"s:sync", shallow ? "top" : "full",
|
2014-05-13 17:28:45 +02:00
|
|
|
"s:mode", reuse ? "existing" : "absolute-paths",
|
|
|
|
"S:format", format,
|
blockjob: add qemu capabilities related to block jobs
Upstream qemu 1.3 is adding two new monitor commands, 'drive-mirror'
and 'block-job-complete'[1], which can drive live block copy and
storage migration. [Additionally, RHEL 6.3 had backported an earlier
version of most of the same functionality, but under the names
'__com.redhat_drive-mirror' and '__com.redhat_drive-reopen' and with
slightly different JSON arguments, and has been using patches similar
to these upstream patches for several months now.]
The libvirt API virDomainBlockRebase as already committed for 0.9.12
is flexible enough to expose the basics of block copy, but some
additional features in the 'drive-mirror' qemu command, such as
setting error policy, setting granularity, or using a persistent
bitmap, may later require a new libvirt API virDomainBlockCopy. I
will wait to add that API until we know more about what qemu 1.3
will finally provide.
This patch caters only to the upstream qemu 1.3 interface, although
I have proven that the changes for RHEL 6.3 can be isolated to
just qemu_monitor_json.c, and the rest of this series will
gracefully handle either interface once the JSON differences are
papered over in a downstream patch.
For consistency with other block job commands, libvirt must handle
the bandwidth argument as MiB/sec from the user, even though qemu
exposes the speed argument as bytes/sec; then again, qemu rounds
up to cluster size internally, so using MiB hides the worst effects
of that rounding if you pass small numbers.
[1]https://lists.gnu.org/archive/html/qemu-devel/2012-10/msg04123.html
* src/qemu/qemu_capabilities.h (QEMU_CAPS_DRIVE_MIRROR)
(QEMU_CAPS_DRIVE_REOPEN): New bits.
* src/qemu/qemu_capabilities.c (qemuCaps): Name them.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONCheckCommands): Set
them.
(qemuMonitorJSONDriveMirror, qemuMonitorDrivePivot): New functions.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDriveMirror)
(qemuMonitorDrivePivot): Declare them.
* src/qemu/qemu_monitor.c (qemuMonitorDriveMirror)
(qemuMonitorDrivePivot): New passthroughs.
* src/qemu/qemu_monitor.h (qemuMonitorDriveMirror)
(qemuMonitorDrivePivot): Declare them.
2012-09-28 17:29:53 -06:00
|
|
|
NULL);
|
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2019-05-17 18:17:26 +02:00
|
|
|
return -1;
|
blockjob: add qemu capabilities related to block jobs
Upstream qemu 1.3 is adding two new monitor commands, 'drive-mirror'
and 'block-job-complete'[1], which can drive live block copy and
storage migration. [Additionally, RHEL 6.3 had backported an earlier
version of most of the same functionality, but under the names
'__com.redhat_drive-mirror' and '__com.redhat_drive-reopen' and with
slightly different JSON arguments, and has been using patches similar
to these upstream patches for several months now.]
The libvirt API virDomainBlockRebase as already committed for 0.9.12
is flexible enough to expose the basics of block copy, but some
additional features in the 'drive-mirror' qemu command, such as
setting error policy, setting granularity, or using a persistent
bitmap, may later require a new libvirt API virDomainBlockCopy. I
will wait to add that API until we know more about what qemu 1.3
will finally provide.
This patch caters only to the upstream qemu 1.3 interface, although
I have proven that the changes for RHEL 6.3 can be isolated to
just qemu_monitor_json.c, and the rest of this series will
gracefully handle either interface once the JSON differences are
papered over in a downstream patch.
For consistency with other block job commands, libvirt must handle
the bandwidth argument as MiB/sec from the user, even though qemu
exposes the speed argument as bytes/sec; then again, qemu rounds
up to cluster size internally, so using MiB hides the worst effects
of that rounding if you pass small numbers.
[1]https://lists.gnu.org/archive/html/qemu-devel/2012-10/msg04123.html
* src/qemu/qemu_capabilities.h (QEMU_CAPS_DRIVE_MIRROR)
(QEMU_CAPS_DRIVE_REOPEN): New bits.
* src/qemu/qemu_capabilities.c (qemuCaps): Name them.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONCheckCommands): Set
them.
(qemuMonitorJSONDriveMirror, qemuMonitorDrivePivot): New functions.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDriveMirror)
(qemuMonitorDrivePivot): Declare them.
* src/qemu/qemu_monitor.c (qemuMonitorDriveMirror)
(qemuMonitorDrivePivot): New passthroughs.
* src/qemu/qemu_monitor.h (qemuMonitorDriveMirror)
(qemuMonitorDrivePivot): Declare them.
2012-09-28 17:29:53 -06:00
|
|
|
|
2019-05-17 18:17:26 +02:00
|
|
|
return qemuMonitorJSONCheckError(cmd, reply);
|
blockjob: add qemu capabilities related to block jobs
Upstream qemu 1.3 is adding two new monitor commands, 'drive-mirror'
and 'block-job-complete'[1], which can drive live block copy and
storage migration. [Additionally, RHEL 6.3 had backported an earlier
version of most of the same functionality, but under the names
'__com.redhat_drive-mirror' and '__com.redhat_drive-reopen' and with
slightly different JSON arguments, and has been using patches similar
to these upstream patches for several months now.]
The libvirt API virDomainBlockRebase as already committed for 0.9.12
is flexible enough to expose the basics of block copy, but some
additional features in the 'drive-mirror' qemu command, such as
setting error policy, setting granularity, or using a persistent
bitmap, may later require a new libvirt API virDomainBlockCopy. I
will wait to add that API until we know more about what qemu 1.3
will finally provide.
This patch caters only to the upstream qemu 1.3 interface, although
I have proven that the changes for RHEL 6.3 can be isolated to
just qemu_monitor_json.c, and the rest of this series will
gracefully handle either interface once the JSON differences are
papered over in a downstream patch.
For consistency with other block job commands, libvirt must handle
the bandwidth argument as MiB/sec from the user, even though qemu
exposes the speed argument as bytes/sec; then again, qemu rounds
up to cluster size internally, so using MiB hides the worst effects
of that rounding if you pass small numbers.
[1]https://lists.gnu.org/archive/html/qemu-devel/2012-10/msg04123.html
* src/qemu/qemu_capabilities.h (QEMU_CAPS_DRIVE_MIRROR)
(QEMU_CAPS_DRIVE_REOPEN): New bits.
* src/qemu/qemu_capabilities.c (qemuCaps): Name them.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONCheckCommands): Set
them.
(qemuMonitorJSONDriveMirror, qemuMonitorDrivePivot): New functions.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDriveMirror)
(qemuMonitorDrivePivot): Declare them.
* src/qemu/qemu_monitor.c (qemuMonitorDriveMirror)
(qemuMonitorDrivePivot): New passthroughs.
* src/qemu/qemu_monitor.h (qemuMonitorDriveMirror)
(qemuMonitorDrivePivot): Declare them.
2012-09-28 17:29:53 -06:00
|
|
|
}
|
|
|
|
|
2016-03-01 14:55:34 +01:00
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBlockdevMirror(qemuMonitor *mon,
|
2016-03-01 14:55:34 +01:00
|
|
|
const char *jobname,
|
2018-08-16 18:47:20 +02:00
|
|
|
bool persistjob,
|
2016-03-01 14:55:34 +01:00
|
|
|
const char *device,
|
|
|
|
const char *target,
|
|
|
|
unsigned long long speed,
|
|
|
|
unsigned int granularity,
|
|
|
|
unsigned long long buf_size,
|
2019-05-17 18:13:53 +02:00
|
|
|
bool shallow)
|
2016-03-01 14:55:34 +01:00
|
|
|
{
|
2019-10-15 14:47:50 +02:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2018-08-16 18:47:20 +02:00
|
|
|
virTristateBool autofinalize = VIR_TRISTATE_BOOL_ABSENT;
|
|
|
|
virTristateBool autodismiss = VIR_TRISTATE_BOOL_ABSENT;
|
|
|
|
|
|
|
|
if (persistjob) {
|
|
|
|
autofinalize = VIR_TRISTATE_BOOL_YES;
|
|
|
|
autodismiss = VIR_TRISTATE_BOOL_NO;
|
|
|
|
}
|
2016-03-01 14:55:34 +01:00
|
|
|
|
|
|
|
cmd = qemuMonitorJSONMakeCommand("blockdev-mirror",
|
|
|
|
"S:job-id", jobname,
|
|
|
|
"s:device", device,
|
|
|
|
"s:target", target,
|
|
|
|
"Y:speed", speed,
|
|
|
|
"z:granularity", granularity,
|
|
|
|
"P:buf-size", buf_size,
|
|
|
|
"s:sync", shallow ? "top" : "full",
|
2018-08-16 18:47:20 +02:00
|
|
|
"T:auto-finalize", autofinalize,
|
|
|
|
"T:auto-dismiss", autodismiss,
|
2016-03-01 14:55:34 +01:00
|
|
|
NULL);
|
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2019-05-17 18:17:26 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
return -1;
|
2016-03-01 14:55:34 +01:00
|
|
|
|
2019-05-17 18:17:26 +02:00
|
|
|
return qemuMonitorJSONCheckError(cmd, reply);
|
2016-03-01 14:55:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-03-16 22:17:28 -06:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONTransaction(qemuMonitor *mon, virJSONValue **actions)
|
2012-03-16 22:17:28 -06:00
|
|
|
{
|
2012-04-05 13:15:03 -06:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2012-03-16 22:17:28 -06:00
|
|
|
|
|
|
|
cmd = qemuMonitorJSONMakeCommand("transaction",
|
2018-03-30 12:35:52 +02:00
|
|
|
"a:actions", actions,
|
2012-03-16 22:17:28 -06:00
|
|
|
NULL);
|
2012-04-05 13:15:03 -06:00
|
|
|
if (!cmd)
|
|
|
|
goto cleanup;
|
2012-03-16 22:17:28 -06:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2012-03-16 22:17:28 -06:00
|
|
|
goto cleanup;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2011-08-15 17:25:54 -06:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2011-08-15 17:25:54 -06:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2018-08-14 15:52:01 +02:00
|
|
|
|
blockjob: allow omitted arguments to QMP block-commit
We are about to turn on support for active block commit. Although
qemu 2.0 was the first version to mostly support it, that version
mis-handles 0-length files, and doesn't have anything available for
easy probing. But qemu 2.1 fixed bugs, and made life simpler by
letting the 'top' argument be optional. Unless someone begs for
active commit with qemu 2.0, for now we are just going to enable
it only by probing for qemu 2.1 behavior (anyone backporting active
commit can also backport the optional argument behavior). This
requires qemu.git commit 7676e2c597000eff3a7233b40cca768b358f9bc9.
Although all our actual uses of block-commit supply arguments for
both base and top, we can omit both arguments and use a bogus
device string to trigger an interesting behavior in qemu. All QMP
commands first do argument validation, failing with GenericError
if a mandatory argument is missing. Once that passes, the code
in the specific command gets to do further checking, and the qemu
developers made sure that if device is the only supplied argument,
then the block-commit code will look up the device first, with a
failure of DeviceNotFound, before attempting any further argument
validation (most other validations fail with GenericError). Thus,
the category of error class can reliably be used to decipher
whether the top argument was optional, which in turn implies a
working active commit. Since we expect our bogus device string to
trigger an error either way, the code is written to return a
distinct return value without spamming the logs.
* src/qemu/qemu_monitor.h (qemuMonitorSupportsActiveCommit): New
prototype.
* src/qemu/qemu_monitor.c (qemuMonitorSupportsActiveCommit):
Implement it.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONBlockCommit):
Allow NULL for top and base, for probing purposes.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONBlockCommit):
Likewise, implementing the probe.
* tests/qemumonitorjsontest.c (mymain): Enable...
(testQemuMonitorJSONqemuMonitorSupportsActiveCommit): ...a new test.
Signed-off-by: Eric Blake <eblake@redhat.com>
2014-06-16 21:42:49 -06:00
|
|
|
/* speed is in bytes/sec. Returns 0 on success, -1 with error message
|
2018-08-14 15:52:01 +02:00
|
|
|
* emitted on failure. */
|
2012-10-03 15:13:21 -06:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBlockCommit(qemuMonitor *mon,
|
2018-08-16 18:20:25 +02:00
|
|
|
const char *device,
|
|
|
|
const char *jobname,
|
|
|
|
bool persistjob,
|
|
|
|
const char *top,
|
|
|
|
const char *topNode,
|
|
|
|
const char *base,
|
|
|
|
const char *baseNode,
|
2014-05-13 17:41:33 +02:00
|
|
|
const char *backingName,
|
2012-10-03 15:13:21 -06:00
|
|
|
unsigned long long speed)
|
|
|
|
{
|
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2018-08-16 18:20:25 +02:00
|
|
|
virTristateBool autofinalize = VIR_TRISTATE_BOOL_ABSENT;
|
|
|
|
virTristateBool autodismiss = VIR_TRISTATE_BOOL_ABSENT;
|
|
|
|
|
|
|
|
if (persistjob) {
|
|
|
|
autofinalize = VIR_TRISTATE_BOOL_YES;
|
|
|
|
autodismiss = VIR_TRISTATE_BOOL_NO;
|
|
|
|
}
|
2012-10-03 15:13:21 -06:00
|
|
|
|
|
|
|
cmd = qemuMonitorJSONMakeCommand("block-commit",
|
|
|
|
"s:device", device,
|
2018-08-16 18:20:25 +02:00
|
|
|
"S:job-id", jobname,
|
2014-09-05 05:31:47 -06:00
|
|
|
"Y:speed", speed,
|
blockjob: allow omitted arguments to QMP block-commit
We are about to turn on support for active block commit. Although
qemu 2.0 was the first version to mostly support it, that version
mis-handles 0-length files, and doesn't have anything available for
easy probing. But qemu 2.1 fixed bugs, and made life simpler by
letting the 'top' argument be optional. Unless someone begs for
active commit with qemu 2.0, for now we are just going to enable
it only by probing for qemu 2.1 behavior (anyone backporting active
commit can also backport the optional argument behavior). This
requires qemu.git commit 7676e2c597000eff3a7233b40cca768b358f9bc9.
Although all our actual uses of block-commit supply arguments for
both base and top, we can omit both arguments and use a bogus
device string to trigger an interesting behavior in qemu. All QMP
commands first do argument validation, failing with GenericError
if a mandatory argument is missing. Once that passes, the code
in the specific command gets to do further checking, and the qemu
developers made sure that if device is the only supplied argument,
then the block-commit code will look up the device first, with a
failure of DeviceNotFound, before attempting any further argument
validation (most other validations fail with GenericError). Thus,
the category of error class can reliably be used to decipher
whether the top argument was optional, which in turn implies a
working active commit. Since we expect our bogus device string to
trigger an error either way, the code is written to return a
distinct return value without spamming the logs.
* src/qemu/qemu_monitor.h (qemuMonitorSupportsActiveCommit): New
prototype.
* src/qemu/qemu_monitor.c (qemuMonitorSupportsActiveCommit):
Implement it.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONBlockCommit):
Allow NULL for top and base, for probing purposes.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONBlockCommit):
Likewise, implementing the probe.
* tests/qemumonitorjsontest.c (mymain): Enable...
(testQemuMonitorJSONqemuMonitorSupportsActiveCommit): ...a new test.
Signed-off-by: Eric Blake <eblake@redhat.com>
2014-06-16 21:42:49 -06:00
|
|
|
"S:top", top,
|
2018-08-16 18:20:25 +02:00
|
|
|
"S:top-node", topNode,
|
blockjob: allow omitted arguments to QMP block-commit
We are about to turn on support for active block commit. Although
qemu 2.0 was the first version to mostly support it, that version
mis-handles 0-length files, and doesn't have anything available for
easy probing. But qemu 2.1 fixed bugs, and made life simpler by
letting the 'top' argument be optional. Unless someone begs for
active commit with qemu 2.0, for now we are just going to enable
it only by probing for qemu 2.1 behavior (anyone backporting active
commit can also backport the optional argument behavior). This
requires qemu.git commit 7676e2c597000eff3a7233b40cca768b358f9bc9.
Although all our actual uses of block-commit supply arguments for
both base and top, we can omit both arguments and use a bogus
device string to trigger an interesting behavior in qemu. All QMP
commands first do argument validation, failing with GenericError
if a mandatory argument is missing. Once that passes, the code
in the specific command gets to do further checking, and the qemu
developers made sure that if device is the only supplied argument,
then the block-commit code will look up the device first, with a
failure of DeviceNotFound, before attempting any further argument
validation (most other validations fail with GenericError). Thus,
the category of error class can reliably be used to decipher
whether the top argument was optional, which in turn implies a
working active commit. Since we expect our bogus device string to
trigger an error either way, the code is written to return a
distinct return value without spamming the logs.
* src/qemu/qemu_monitor.h (qemuMonitorSupportsActiveCommit): New
prototype.
* src/qemu/qemu_monitor.c (qemuMonitorSupportsActiveCommit):
Implement it.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONBlockCommit):
Allow NULL for top and base, for probing purposes.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONBlockCommit):
Likewise, implementing the probe.
* tests/qemumonitorjsontest.c (mymain): Enable...
(testQemuMonitorJSONqemuMonitorSupportsActiveCommit): ...a new test.
Signed-off-by: Eric Blake <eblake@redhat.com>
2014-06-16 21:42:49 -06:00
|
|
|
"S:base", base,
|
2018-08-16 18:20:25 +02:00
|
|
|
"S:base-node", baseNode,
|
2014-05-13 17:41:33 +02:00
|
|
|
"S:backing-file", backingName,
|
2018-08-16 18:20:25 +02:00
|
|
|
"T:auto-finalize", autofinalize,
|
|
|
|
"T:auto-dismiss", autodismiss,
|
2012-10-03 15:13:21 -06:00
|
|
|
NULL);
|
blockjob: add qemu capabilities related to block jobs
Upstream qemu 1.3 is adding two new monitor commands, 'drive-mirror'
and 'block-job-complete'[1], which can drive live block copy and
storage migration. [Additionally, RHEL 6.3 had backported an earlier
version of most of the same functionality, but under the names
'__com.redhat_drive-mirror' and '__com.redhat_drive-reopen' and with
slightly different JSON arguments, and has been using patches similar
to these upstream patches for several months now.]
The libvirt API virDomainBlockRebase as already committed for 0.9.12
is flexible enough to expose the basics of block copy, but some
additional features in the 'drive-mirror' qemu command, such as
setting error policy, setting granularity, or using a persistent
bitmap, may later require a new libvirt API virDomainBlockCopy. I
will wait to add that API until we know more about what qemu 1.3
will finally provide.
This patch caters only to the upstream qemu 1.3 interface, although
I have proven that the changes for RHEL 6.3 can be isolated to
just qemu_monitor_json.c, and the rest of this series will
gracefully handle either interface once the JSON differences are
papered over in a downstream patch.
For consistency with other block job commands, libvirt must handle
the bandwidth argument as MiB/sec from the user, even though qemu
exposes the speed argument as bytes/sec; then again, qemu rounds
up to cluster size internally, so using MiB hides the worst effects
of that rounding if you pass small numbers.
[1]https://lists.gnu.org/archive/html/qemu-devel/2012-10/msg04123.html
* src/qemu/qemu_capabilities.h (QEMU_CAPS_DRIVE_MIRROR)
(QEMU_CAPS_DRIVE_REOPEN): New bits.
* src/qemu/qemu_capabilities.c (qemuCaps): Name them.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONCheckCommands): Set
them.
(qemuMonitorJSONDriveMirror, qemuMonitorDrivePivot): New functions.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDriveMirror)
(qemuMonitorDrivePivot): Declare them.
* src/qemu/qemu_monitor.c (qemuMonitorDriveMirror)
(qemuMonitorDrivePivot): New passthroughs.
* src/qemu/qemu_monitor.h (qemuMonitorDriveMirror)
(qemuMonitorDrivePivot): Declare them.
2012-09-28 17:29:53 -06:00
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
blockjob: add qemu capabilities related to block jobs
Upstream qemu 1.3 is adding two new monitor commands, 'drive-mirror'
and 'block-job-complete'[1], which can drive live block copy and
storage migration. [Additionally, RHEL 6.3 had backported an earlier
version of most of the same functionality, but under the names
'__com.redhat_drive-mirror' and '__com.redhat_drive-reopen' and with
slightly different JSON arguments, and has been using patches similar
to these upstream patches for several months now.]
The libvirt API virDomainBlockRebase as already committed for 0.9.12
is flexible enough to expose the basics of block copy, but some
additional features in the 'drive-mirror' qemu command, such as
setting error policy, setting granularity, or using a persistent
bitmap, may later require a new libvirt API virDomainBlockCopy. I
will wait to add that API until we know more about what qemu 1.3
will finally provide.
This patch caters only to the upstream qemu 1.3 interface, although
I have proven that the changes for RHEL 6.3 can be isolated to
just qemu_monitor_json.c, and the rest of this series will
gracefully handle either interface once the JSON differences are
papered over in a downstream patch.
For consistency with other block job commands, libvirt must handle
the bandwidth argument as MiB/sec from the user, even though qemu
exposes the speed argument as bytes/sec; then again, qemu rounds
up to cluster size internally, so using MiB hides the worst effects
of that rounding if you pass small numbers.
[1]https://lists.gnu.org/archive/html/qemu-devel/2012-10/msg04123.html
* src/qemu/qemu_capabilities.h (QEMU_CAPS_DRIVE_MIRROR)
(QEMU_CAPS_DRIVE_REOPEN): New bits.
* src/qemu/qemu_capabilities.c (qemuCaps): Name them.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONCheckCommands): Set
them.
(qemuMonitorJSONDriveMirror, qemuMonitorDrivePivot): New functions.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDriveMirror)
(qemuMonitorDrivePivot): Declare them.
* src/qemu/qemu_monitor.c (qemuMonitorDriveMirror)
(qemuMonitorDrivePivot): New passthroughs.
* src/qemu/qemu_monitor.h (qemuMonitorDriveMirror)
(qemuMonitorDrivePivot): Declare them.
2012-09-28 17:29:53 -06:00
|
|
|
goto cleanup;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = 0;
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
blockjob: add qemu capabilities related to block jobs
Upstream qemu 1.3 is adding two new monitor commands, 'drive-mirror'
and 'block-job-complete'[1], which can drive live block copy and
storage migration. [Additionally, RHEL 6.3 had backported an earlier
version of most of the same functionality, but under the names
'__com.redhat_drive-mirror' and '__com.redhat_drive-reopen' and with
slightly different JSON arguments, and has been using patches similar
to these upstream patches for several months now.]
The libvirt API virDomainBlockRebase as already committed for 0.9.12
is flexible enough to expose the basics of block copy, but some
additional features in the 'drive-mirror' qemu command, such as
setting error policy, setting granularity, or using a persistent
bitmap, may later require a new libvirt API virDomainBlockCopy. I
will wait to add that API until we know more about what qemu 1.3
will finally provide.
This patch caters only to the upstream qemu 1.3 interface, although
I have proven that the changes for RHEL 6.3 can be isolated to
just qemu_monitor_json.c, and the rest of this series will
gracefully handle either interface once the JSON differences are
papered over in a downstream patch.
For consistency with other block job commands, libvirt must handle
the bandwidth argument as MiB/sec from the user, even though qemu
exposes the speed argument as bytes/sec; then again, qemu rounds
up to cluster size internally, so using MiB hides the worst effects
of that rounding if you pass small numbers.
[1]https://lists.gnu.org/archive/html/qemu-devel/2012-10/msg04123.html
* src/qemu/qemu_capabilities.h (QEMU_CAPS_DRIVE_MIRROR)
(QEMU_CAPS_DRIVE_REOPEN): New bits.
* src/qemu/qemu_capabilities.c (qemuCaps): Name them.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONCheckCommands): Set
them.
(qemuMonitorJSONDriveMirror, qemuMonitorDrivePivot): New functions.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDriveMirror)
(qemuMonitorDrivePivot): Declare them.
* src/qemu/qemu_monitor.c (qemuMonitorDriveMirror)
(qemuMonitorDrivePivot): New passthroughs.
* src/qemu/qemu_monitor.h (qemuMonitorDriveMirror)
(qemuMonitorDrivePivot): Declare them.
2012-09-28 17:29:53 -06:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2012-10-03 15:13:21 -06:00
|
|
|
|
qemu: read backing chain names from qemu
https://bugzilla.redhat.com/show_bug.cgi?id=1199182 documents that
after a series of disk snapshots into existing destination images,
followed by active commits of the top image, it is possible for
qemu 2.2 and earlier to end up tracking a different name for the
image than what it would have had when opening the chain afresh.
That is, when starting with the chain 'a <- b <- c', the name
associated with 'b' is how it was spelled in the metadata of 'c',
but when starting with 'a', taking two snapshots into 'a <- b <- c',
then committing 'c' back into 'b', the name associated with 'b' is
now the name used when taking the first snapshot.
Sadly, older qemu doesn't know how to treat different spellings of
the same filename as identical files (it uses strcmp() instead of
checking for the same inode), which means libvirt's attempt to
commit an image using solely the names learned from qcow2 metadata
fails with a cryptic:
error: internal error: unable to execute QEMU command 'block-commit': Top image file /tmp/images/c/../b/b not found
even though the file exists. Trying to teach libvirt the rules on
which name qemu will expect is not worth the effort (besides, we'd
have to remember it across libvirtd restarts, and track whether a
file was opened via metadata or via snapshot creation for a given
qemu process); it is easier to just always directly ask qemu what
string it expects to see in the first place.
As a safety valve, we validate that any name returned by qemu
still maps to the same local file as we have tracked it, so that
a compromised qemu cannot accidentally cause us to act on an
incorrect file.
* src/qemu/qemu_monitor.h (qemuMonitorDiskNameLookup): New
prototype.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDiskNameLookup):
Likewise.
* src/qemu/qemu_monitor.c (qemuMonitorDiskNameLookup): New function.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONDiskNameLookupOne): Likewise.
* src/qemu/qemu_driver.c (qemuDomainBlockCommit)
(qemuDomainBlockJobImpl): Use it.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-03-11 14:37:04 -06:00
|
|
|
static char *
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONDiskNameLookupOne(virJSONValue *image,
|
|
|
|
virStorageSource *top,
|
|
|
|
virStorageSource *target)
|
qemu: read backing chain names from qemu
https://bugzilla.redhat.com/show_bug.cgi?id=1199182 documents that
after a series of disk snapshots into existing destination images,
followed by active commits of the top image, it is possible for
qemu 2.2 and earlier to end up tracking a different name for the
image than what it would have had when opening the chain afresh.
That is, when starting with the chain 'a <- b <- c', the name
associated with 'b' is how it was spelled in the metadata of 'c',
but when starting with 'a', taking two snapshots into 'a <- b <- c',
then committing 'c' back into 'b', the name associated with 'b' is
now the name used when taking the first snapshot.
Sadly, older qemu doesn't know how to treat different spellings of
the same filename as identical files (it uses strcmp() instead of
checking for the same inode), which means libvirt's attempt to
commit an image using solely the names learned from qcow2 metadata
fails with a cryptic:
error: internal error: unable to execute QEMU command 'block-commit': Top image file /tmp/images/c/../b/b not found
even though the file exists. Trying to teach libvirt the rules on
which name qemu will expect is not worth the effort (besides, we'd
have to remember it across libvirtd restarts, and track whether a
file was opened via metadata or via snapshot creation for a given
qemu process); it is easier to just always directly ask qemu what
string it expects to see in the first place.
As a safety valve, we validate that any name returned by qemu
still maps to the same local file as we have tracked it, so that
a compromised qemu cannot accidentally cause us to act on an
incorrect file.
* src/qemu/qemu_monitor.h (qemuMonitorDiskNameLookup): New
prototype.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDiskNameLookup):
Likewise.
* src/qemu/qemu_monitor.c (qemuMonitorDiskNameLookup): New function.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONDiskNameLookupOne): Likewise.
* src/qemu/qemu_driver.c (qemuDomainBlockCommit)
(qemuDomainBlockJobImpl): Use it.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-03-11 14:37:04 -06:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *backing;
|
qemu: read backing chain names from qemu
https://bugzilla.redhat.com/show_bug.cgi?id=1199182 documents that
after a series of disk snapshots into existing destination images,
followed by active commits of the top image, it is possible for
qemu 2.2 and earlier to end up tracking a different name for the
image than what it would have had when opening the chain afresh.
That is, when starting with the chain 'a <- b <- c', the name
associated with 'b' is how it was spelled in the metadata of 'c',
but when starting with 'a', taking two snapshots into 'a <- b <- c',
then committing 'c' back into 'b', the name associated with 'b' is
now the name used when taking the first snapshot.
Sadly, older qemu doesn't know how to treat different spellings of
the same filename as identical files (it uses strcmp() instead of
checking for the same inode), which means libvirt's attempt to
commit an image using solely the names learned from qcow2 metadata
fails with a cryptic:
error: internal error: unable to execute QEMU command 'block-commit': Top image file /tmp/images/c/../b/b not found
even though the file exists. Trying to teach libvirt the rules on
which name qemu will expect is not worth the effort (besides, we'd
have to remember it across libvirtd restarts, and track whether a
file was opened via metadata or via snapshot creation for a given
qemu process); it is easier to just always directly ask qemu what
string it expects to see in the first place.
As a safety valve, we validate that any name returned by qemu
still maps to the same local file as we have tracked it, so that
a compromised qemu cannot accidentally cause us to act on an
incorrect file.
* src/qemu/qemu_monitor.h (qemuMonitorDiskNameLookup): New
prototype.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDiskNameLookup):
Likewise.
* src/qemu/qemu_monitor.c (qemuMonitorDiskNameLookup): New function.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONDiskNameLookupOne): Likewise.
* src/qemu/qemu_driver.c (qemuDomainBlockCommit)
(qemuDomainBlockJobImpl): Use it.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-03-11 14:37:04 -06:00
|
|
|
char *ret;
|
|
|
|
|
|
|
|
/* The caller will report a generic message if we return NULL
|
|
|
|
* without an error; but in some cases we can improve by reporting
|
|
|
|
* a more specific message. */
|
|
|
|
if (!top || !image)
|
|
|
|
return NULL;
|
|
|
|
if (top != target) {
|
qemu: simplify json parsing
Rather than grabbing an arbitrary JSON value and then checking
if it has the right type, we might as well request the correct
type to begin with.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONIOProcessEvent)
(qemuMonitorJSONCommandWithFd, qemuMonitorJSONHandleGraphics)
(qemuMonitorJSONGetStatus, qemuMonitorJSONExtractCPUInfo)
(qemuMonitorJSONGetVirtType, qemuMonitorJSONGetBalloonInfo)
(qemuMonitorJSONGetMemoryStats)
(qemuMonitorJSONDevGetBlockExtent)
(qemuMonitorJSONGetOneBlockStatsInfo)
(qemuMonitorJSONGetAllBlockStatsInfo)
(qemuMonitorJSONBlockStatsUpdateCapacityOne)
(qemuMonitorJSONBlockStatsUpdateCapacity)
(qemuMonitorJSONGetBlockExtent)
(qemuMonitorJSONGetMigrationStatusReply)
(qemuMonitorJSONGetDumpGuestMemoryCapability)
(qemuMonitorJSONAddFd, qemuMonitorJSONQueryRxFilterParse)
(qemuMonitorJSONExtractChardevInfo)
(qemuMonitorJSONDiskNameLookupOne)
(qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONGetAllBlockJobInfo)
(qemuMonitorJSONBlockIoThrottleInfo, qemuMonitorJSONGetVersion)
(qemuMonitorJSONGetMachines, qemuMonitorJSONGetCPUDefinitions)
(qemuMonitorJSONGetCommands, qemuMonitorJSONGetEvents)
(qemuMonitorJSONGetKVMState, qemuMonitorJSONGetObjectTypes)
(qemuMonitorJSONGetObjectListPaths)
(qemuMonitorJSONGetObjectProps, qemuMonitorJSONGetTargetArch)
(qemuMonitorJSONGetMigrationCapabilities)
(qemuMonitorJSONGetStringArray, qemuMonitorJSONAttachCharDev)
(qemuMonitorJSONGetCPUx86Data, qemuMonitorJSONGetIOThreads)
(qemuMonitorJSONGetMemoryDeviceInfo): Use shorter idioms.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-06-20 13:40:21 -06:00
|
|
|
backing = virJSONValueObjectGetObject(image, "backing-image");
|
qemu: read backing chain names from qemu
https://bugzilla.redhat.com/show_bug.cgi?id=1199182 documents that
after a series of disk snapshots into existing destination images,
followed by active commits of the top image, it is possible for
qemu 2.2 and earlier to end up tracking a different name for the
image than what it would have had when opening the chain afresh.
That is, when starting with the chain 'a <- b <- c', the name
associated with 'b' is how it was spelled in the metadata of 'c',
but when starting with 'a', taking two snapshots into 'a <- b <- c',
then committing 'c' back into 'b', the name associated with 'b' is
now the name used when taking the first snapshot.
Sadly, older qemu doesn't know how to treat different spellings of
the same filename as identical files (it uses strcmp() instead of
checking for the same inode), which means libvirt's attempt to
commit an image using solely the names learned from qcow2 metadata
fails with a cryptic:
error: internal error: unable to execute QEMU command 'block-commit': Top image file /tmp/images/c/../b/b not found
even though the file exists. Trying to teach libvirt the rules on
which name qemu will expect is not worth the effort (besides, we'd
have to remember it across libvirtd restarts, and track whether a
file was opened via metadata or via snapshot creation for a given
qemu process); it is easier to just always directly ask qemu what
string it expects to see in the first place.
As a safety valve, we validate that any name returned by qemu
still maps to the same local file as we have tracked it, so that
a compromised qemu cannot accidentally cause us to act on an
incorrect file.
* src/qemu/qemu_monitor.h (qemuMonitorDiskNameLookup): New
prototype.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDiskNameLookup):
Likewise.
* src/qemu/qemu_monitor.c (qemuMonitorDiskNameLookup): New function.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONDiskNameLookupOne): Likewise.
* src/qemu/qemu_driver.c (qemuDomainBlockCommit)
(qemuDomainBlockJobImpl): Use it.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-03-11 14:37:04 -06:00
|
|
|
return qemuMonitorJSONDiskNameLookupOne(backing, top->backingStore,
|
|
|
|
target);
|
|
|
|
}
|
2019-10-20 13:49:46 +02:00
|
|
|
ret = g_strdup(virJSONValueObjectGetString(image, "filename"));
|
qemu: read backing chain names from qemu
https://bugzilla.redhat.com/show_bug.cgi?id=1199182 documents that
after a series of disk snapshots into existing destination images,
followed by active commits of the top image, it is possible for
qemu 2.2 and earlier to end up tracking a different name for the
image than what it would have had when opening the chain afresh.
That is, when starting with the chain 'a <- b <- c', the name
associated with 'b' is how it was spelled in the metadata of 'c',
but when starting with 'a', taking two snapshots into 'a <- b <- c',
then committing 'c' back into 'b', the name associated with 'b' is
now the name used when taking the first snapshot.
Sadly, older qemu doesn't know how to treat different spellings of
the same filename as identical files (it uses strcmp() instead of
checking for the same inode), which means libvirt's attempt to
commit an image using solely the names learned from qcow2 metadata
fails with a cryptic:
error: internal error: unable to execute QEMU command 'block-commit': Top image file /tmp/images/c/../b/b not found
even though the file exists. Trying to teach libvirt the rules on
which name qemu will expect is not worth the effort (besides, we'd
have to remember it across libvirtd restarts, and track whether a
file was opened via metadata or via snapshot creation for a given
qemu process); it is easier to just always directly ask qemu what
string it expects to see in the first place.
As a safety valve, we validate that any name returned by qemu
still maps to the same local file as we have tracked it, so that
a compromised qemu cannot accidentally cause us to act on an
incorrect file.
* src/qemu/qemu_monitor.h (qemuMonitorDiskNameLookup): New
prototype.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDiskNameLookup):
Likewise.
* src/qemu/qemu_monitor.c (qemuMonitorDiskNameLookup): New function.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONDiskNameLookupOne): Likewise.
* src/qemu/qemu_driver.c (qemuDomainBlockCommit)
(qemuDomainBlockJobImpl): Use it.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-03-11 14:37:04 -06:00
|
|
|
/* Sanity check - the name qemu gave us should resolve to the same
|
|
|
|
file tracked by our target description. */
|
|
|
|
if (virStorageSourceIsLocalStorage(target) &&
|
|
|
|
STRNEQ(ret, target->path) &&
|
|
|
|
!virFileLinkPointsTo(ret, target->path)) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("qemu block name '%s' doesn't match expected '%s'"),
|
|
|
|
ret, target->path);
|
|
|
|
VIR_FREE(ret);
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
char *
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONDiskNameLookup(qemuMonitor *mon,
|
qemu: read backing chain names from qemu
https://bugzilla.redhat.com/show_bug.cgi?id=1199182 documents that
after a series of disk snapshots into existing destination images,
followed by active commits of the top image, it is possible for
qemu 2.2 and earlier to end up tracking a different name for the
image than what it would have had when opening the chain afresh.
That is, when starting with the chain 'a <- b <- c', the name
associated with 'b' is how it was spelled in the metadata of 'c',
but when starting with 'a', taking two snapshots into 'a <- b <- c',
then committing 'c' back into 'b', the name associated with 'b' is
now the name used when taking the first snapshot.
Sadly, older qemu doesn't know how to treat different spellings of
the same filename as identical files (it uses strcmp() instead of
checking for the same inode), which means libvirt's attempt to
commit an image using solely the names learned from qcow2 metadata
fails with a cryptic:
error: internal error: unable to execute QEMU command 'block-commit': Top image file /tmp/images/c/../b/b not found
even though the file exists. Trying to teach libvirt the rules on
which name qemu will expect is not worth the effort (besides, we'd
have to remember it across libvirtd restarts, and track whether a
file was opened via metadata or via snapshot creation for a given
qemu process); it is easier to just always directly ask qemu what
string it expects to see in the first place.
As a safety valve, we validate that any name returned by qemu
still maps to the same local file as we have tracked it, so that
a compromised qemu cannot accidentally cause us to act on an
incorrect file.
* src/qemu/qemu_monitor.h (qemuMonitorDiskNameLookup): New
prototype.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDiskNameLookup):
Likewise.
* src/qemu/qemu_monitor.c (qemuMonitorDiskNameLookup): New function.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONDiskNameLookupOne): Likewise.
* src/qemu/qemu_driver.c (qemuDomainBlockCommit)
(qemuDomainBlockJobImpl): Use it.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-03-11 14:37:04 -06:00
|
|
|
const char *device,
|
2021-03-11 08:16:13 +01:00
|
|
|
virStorageSource *top,
|
|
|
|
virStorageSource *target)
|
qemu: read backing chain names from qemu
https://bugzilla.redhat.com/show_bug.cgi?id=1199182 documents that
after a series of disk snapshots into existing destination images,
followed by active commits of the top image, it is possible for
qemu 2.2 and earlier to end up tracking a different name for the
image than what it would have had when opening the chain afresh.
That is, when starting with the chain 'a <- b <- c', the name
associated with 'b' is how it was spelled in the metadata of 'c',
but when starting with 'a', taking two snapshots into 'a <- b <- c',
then committing 'c' back into 'b', the name associated with 'b' is
now the name used when taking the first snapshot.
Sadly, older qemu doesn't know how to treat different spellings of
the same filename as identical files (it uses strcmp() instead of
checking for the same inode), which means libvirt's attempt to
commit an image using solely the names learned from qcow2 metadata
fails with a cryptic:
error: internal error: unable to execute QEMU command 'block-commit': Top image file /tmp/images/c/../b/b not found
even though the file exists. Trying to teach libvirt the rules on
which name qemu will expect is not worth the effort (besides, we'd
have to remember it across libvirtd restarts, and track whether a
file was opened via metadata or via snapshot creation for a given
qemu process); it is easier to just always directly ask qemu what
string it expects to see in the first place.
As a safety valve, we validate that any name returned by qemu
still maps to the same local file as we have tracked it, so that
a compromised qemu cannot accidentally cause us to act on an
incorrect file.
* src/qemu/qemu_monitor.h (qemuMonitorDiskNameLookup): New
prototype.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDiskNameLookup):
Likewise.
* src/qemu/qemu_monitor.c (qemuMonitorDiskNameLookup): New function.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONDiskNameLookupOne): Likewise.
* src/qemu/qemu_driver.c (qemuDomainBlockCommit)
(qemuDomainBlockJobImpl): Use it.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-03-11 14:37:04 -06:00
|
|
|
{
|
|
|
|
char *ret = NULL;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *devices;
|
qemu: read backing chain names from qemu
https://bugzilla.redhat.com/show_bug.cgi?id=1199182 documents that
after a series of disk snapshots into existing destination images,
followed by active commits of the top image, it is possible for
qemu 2.2 and earlier to end up tracking a different name for the
image than what it would have had when opening the chain afresh.
That is, when starting with the chain 'a <- b <- c', the name
associated with 'b' is how it was spelled in the metadata of 'c',
but when starting with 'a', taking two snapshots into 'a <- b <- c',
then committing 'c' back into 'b', the name associated with 'b' is
now the name used when taking the first snapshot.
Sadly, older qemu doesn't know how to treat different spellings of
the same filename as identical files (it uses strcmp() instead of
checking for the same inode), which means libvirt's attempt to
commit an image using solely the names learned from qcow2 metadata
fails with a cryptic:
error: internal error: unable to execute QEMU command 'block-commit': Top image file /tmp/images/c/../b/b not found
even though the file exists. Trying to teach libvirt the rules on
which name qemu will expect is not worth the effort (besides, we'd
have to remember it across libvirtd restarts, and track whether a
file was opened via metadata or via snapshot creation for a given
qemu process); it is easier to just always directly ask qemu what
string it expects to see in the first place.
As a safety valve, we validate that any name returned by qemu
still maps to the same local file as we have tracked it, so that
a compromised qemu cannot accidentally cause us to act on an
incorrect file.
* src/qemu/qemu_monitor.h (qemuMonitorDiskNameLookup): New
prototype.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDiskNameLookup):
Likewise.
* src/qemu/qemu_monitor.c (qemuMonitorDiskNameLookup): New function.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONDiskNameLookupOne): Likewise.
* src/qemu/qemu_driver.c (qemuDomainBlockCommit)
(qemuDomainBlockJobImpl): Use it.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-03-11 14:37:04 -06:00
|
|
|
size_t i;
|
|
|
|
|
2016-09-27 09:39:21 -04:00
|
|
|
if (!(devices = qemuMonitorJSONQueryBlock(mon)))
|
qemu: read backing chain names from qemu
https://bugzilla.redhat.com/show_bug.cgi?id=1199182 documents that
after a series of disk snapshots into existing destination images,
followed by active commits of the top image, it is possible for
qemu 2.2 and earlier to end up tracking a different name for the
image than what it would have had when opening the chain afresh.
That is, when starting with the chain 'a <- b <- c', the name
associated with 'b' is how it was spelled in the metadata of 'c',
but when starting with 'a', taking two snapshots into 'a <- b <- c',
then committing 'c' back into 'b', the name associated with 'b' is
now the name used when taking the first snapshot.
Sadly, older qemu doesn't know how to treat different spellings of
the same filename as identical files (it uses strcmp() instead of
checking for the same inode), which means libvirt's attempt to
commit an image using solely the names learned from qcow2 metadata
fails with a cryptic:
error: internal error: unable to execute QEMU command 'block-commit': Top image file /tmp/images/c/../b/b not found
even though the file exists. Trying to teach libvirt the rules on
which name qemu will expect is not worth the effort (besides, we'd
have to remember it across libvirtd restarts, and track whether a
file was opened via metadata or via snapshot creation for a given
qemu process); it is easier to just always directly ask qemu what
string it expects to see in the first place.
As a safety valve, we validate that any name returned by qemu
still maps to the same local file as we have tracked it, so that
a compromised qemu cannot accidentally cause us to act on an
incorrect file.
* src/qemu/qemu_monitor.h (qemuMonitorDiskNameLookup): New
prototype.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDiskNameLookup):
Likewise.
* src/qemu/qemu_monitor.c (qemuMonitorDiskNameLookup): New function.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONDiskNameLookupOne): Likewise.
* src/qemu/qemu_driver.c (qemuDomainBlockCommit)
(qemuDomainBlockJobImpl): Use it.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-03-11 14:37:04 -06:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
for (i = 0; i < virJSONValueArraySize(devices); i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *dev;
|
|
|
|
virJSONValue *inserted;
|
|
|
|
virJSONValue *image;
|
qemu: read backing chain names from qemu
https://bugzilla.redhat.com/show_bug.cgi?id=1199182 documents that
after a series of disk snapshots into existing destination images,
followed by active commits of the top image, it is possible for
qemu 2.2 and earlier to end up tracking a different name for the
image than what it would have had when opening the chain afresh.
That is, when starting with the chain 'a <- b <- c', the name
associated with 'b' is how it was spelled in the metadata of 'c',
but when starting with 'a', taking two snapshots into 'a <- b <- c',
then committing 'c' back into 'b', the name associated with 'b' is
now the name used when taking the first snapshot.
Sadly, older qemu doesn't know how to treat different spellings of
the same filename as identical files (it uses strcmp() instead of
checking for the same inode), which means libvirt's attempt to
commit an image using solely the names learned from qcow2 metadata
fails with a cryptic:
error: internal error: unable to execute QEMU command 'block-commit': Top image file /tmp/images/c/../b/b not found
even though the file exists. Trying to teach libvirt the rules on
which name qemu will expect is not worth the effort (besides, we'd
have to remember it across libvirtd restarts, and track whether a
file was opened via metadata or via snapshot creation for a given
qemu process); it is easier to just always directly ask qemu what
string it expects to see in the first place.
As a safety valve, we validate that any name returned by qemu
still maps to the same local file as we have tracked it, so that
a compromised qemu cannot accidentally cause us to act on an
incorrect file.
* src/qemu/qemu_monitor.h (qemuMonitorDiskNameLookup): New
prototype.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDiskNameLookup):
Likewise.
* src/qemu/qemu_monitor.c (qemuMonitorDiskNameLookup): New function.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONDiskNameLookupOne): Likewise.
* src/qemu/qemu_driver.c (qemuDomainBlockCommit)
(qemuDomainBlockJobImpl): Use it.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-03-11 14:37:04 -06:00
|
|
|
const char *thisdev;
|
|
|
|
|
2016-10-03 14:58:59 -04:00
|
|
|
if (!(dev = qemuMonitorJSONGetBlockDev(devices, i)))
|
qemu: read backing chain names from qemu
https://bugzilla.redhat.com/show_bug.cgi?id=1199182 documents that
after a series of disk snapshots into existing destination images,
followed by active commits of the top image, it is possible for
qemu 2.2 and earlier to end up tracking a different name for the
image than what it would have had when opening the chain afresh.
That is, when starting with the chain 'a <- b <- c', the name
associated with 'b' is how it was spelled in the metadata of 'c',
but when starting with 'a', taking two snapshots into 'a <- b <- c',
then committing 'c' back into 'b', the name associated with 'b' is
now the name used when taking the first snapshot.
Sadly, older qemu doesn't know how to treat different spellings of
the same filename as identical files (it uses strcmp() instead of
checking for the same inode), which means libvirt's attempt to
commit an image using solely the names learned from qcow2 metadata
fails with a cryptic:
error: internal error: unable to execute QEMU command 'block-commit': Top image file /tmp/images/c/../b/b not found
even though the file exists. Trying to teach libvirt the rules on
which name qemu will expect is not worth the effort (besides, we'd
have to remember it across libvirtd restarts, and track whether a
file was opened via metadata or via snapshot creation for a given
qemu process); it is easier to just always directly ask qemu what
string it expects to see in the first place.
As a safety valve, we validate that any name returned by qemu
still maps to the same local file as we have tracked it, so that
a compromised qemu cannot accidentally cause us to act on an
incorrect file.
* src/qemu/qemu_monitor.h (qemuMonitorDiskNameLookup): New
prototype.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDiskNameLookup):
Likewise.
* src/qemu/qemu_monitor.c (qemuMonitorDiskNameLookup): New function.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONDiskNameLookupOne): Likewise.
* src/qemu/qemu_driver.c (qemuDomainBlockCommit)
(qemuDomainBlockJobImpl): Use it.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-03-11 14:37:04 -06:00
|
|
|
goto cleanup;
|
|
|
|
|
2016-10-03 15:04:32 -04:00
|
|
|
if (!(thisdev = qemuMonitorJSONGetBlockDevDevice(dev)))
|
qemu: read backing chain names from qemu
https://bugzilla.redhat.com/show_bug.cgi?id=1199182 documents that
after a series of disk snapshots into existing destination images,
followed by active commits of the top image, it is possible for
qemu 2.2 and earlier to end up tracking a different name for the
image than what it would have had when opening the chain afresh.
That is, when starting with the chain 'a <- b <- c', the name
associated with 'b' is how it was spelled in the metadata of 'c',
but when starting with 'a', taking two snapshots into 'a <- b <- c',
then committing 'c' back into 'b', the name associated with 'b' is
now the name used when taking the first snapshot.
Sadly, older qemu doesn't know how to treat different spellings of
the same filename as identical files (it uses strcmp() instead of
checking for the same inode), which means libvirt's attempt to
commit an image using solely the names learned from qcow2 metadata
fails with a cryptic:
error: internal error: unable to execute QEMU command 'block-commit': Top image file /tmp/images/c/../b/b not found
even though the file exists. Trying to teach libvirt the rules on
which name qemu will expect is not worth the effort (besides, we'd
have to remember it across libvirtd restarts, and track whether a
file was opened via metadata or via snapshot creation for a given
qemu process); it is easier to just always directly ask qemu what
string it expects to see in the first place.
As a safety valve, we validate that any name returned by qemu
still maps to the same local file as we have tracked it, so that
a compromised qemu cannot accidentally cause us to act on an
incorrect file.
* src/qemu/qemu_monitor.h (qemuMonitorDiskNameLookup): New
prototype.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDiskNameLookup):
Likewise.
* src/qemu/qemu_monitor.c (qemuMonitorDiskNameLookup): New function.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONDiskNameLookupOne): Likewise.
* src/qemu/qemu_driver.c (qemuDomainBlockCommit)
(qemuDomainBlockJobImpl): Use it.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-03-11 14:37:04 -06:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (STREQ(thisdev, device)) {
|
qemu: simplify json parsing
Rather than grabbing an arbitrary JSON value and then checking
if it has the right type, we might as well request the correct
type to begin with.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONIOProcessEvent)
(qemuMonitorJSONCommandWithFd, qemuMonitorJSONHandleGraphics)
(qemuMonitorJSONGetStatus, qemuMonitorJSONExtractCPUInfo)
(qemuMonitorJSONGetVirtType, qemuMonitorJSONGetBalloonInfo)
(qemuMonitorJSONGetMemoryStats)
(qemuMonitorJSONDevGetBlockExtent)
(qemuMonitorJSONGetOneBlockStatsInfo)
(qemuMonitorJSONGetAllBlockStatsInfo)
(qemuMonitorJSONBlockStatsUpdateCapacityOne)
(qemuMonitorJSONBlockStatsUpdateCapacity)
(qemuMonitorJSONGetBlockExtent)
(qemuMonitorJSONGetMigrationStatusReply)
(qemuMonitorJSONGetDumpGuestMemoryCapability)
(qemuMonitorJSONAddFd, qemuMonitorJSONQueryRxFilterParse)
(qemuMonitorJSONExtractChardevInfo)
(qemuMonitorJSONDiskNameLookupOne)
(qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONGetAllBlockJobInfo)
(qemuMonitorJSONBlockIoThrottleInfo, qemuMonitorJSONGetVersion)
(qemuMonitorJSONGetMachines, qemuMonitorJSONGetCPUDefinitions)
(qemuMonitorJSONGetCommands, qemuMonitorJSONGetEvents)
(qemuMonitorJSONGetKVMState, qemuMonitorJSONGetObjectTypes)
(qemuMonitorJSONGetObjectListPaths)
(qemuMonitorJSONGetObjectProps, qemuMonitorJSONGetTargetArch)
(qemuMonitorJSONGetMigrationCapabilities)
(qemuMonitorJSONGetStringArray, qemuMonitorJSONAttachCharDev)
(qemuMonitorJSONGetCPUx86Data, qemuMonitorJSONGetIOThreads)
(qemuMonitorJSONGetMemoryDeviceInfo): Use shorter idioms.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-06-20 13:40:21 -06:00
|
|
|
if ((inserted = virJSONValueObjectGetObject(dev, "inserted")) &&
|
|
|
|
(image = virJSONValueObjectGetObject(inserted, "image"))) {
|
qemu: read backing chain names from qemu
https://bugzilla.redhat.com/show_bug.cgi?id=1199182 documents that
after a series of disk snapshots into existing destination images,
followed by active commits of the top image, it is possible for
qemu 2.2 and earlier to end up tracking a different name for the
image than what it would have had when opening the chain afresh.
That is, when starting with the chain 'a <- b <- c', the name
associated with 'b' is how it was spelled in the metadata of 'c',
but when starting with 'a', taking two snapshots into 'a <- b <- c',
then committing 'c' back into 'b', the name associated with 'b' is
now the name used when taking the first snapshot.
Sadly, older qemu doesn't know how to treat different spellings of
the same filename as identical files (it uses strcmp() instead of
checking for the same inode), which means libvirt's attempt to
commit an image using solely the names learned from qcow2 metadata
fails with a cryptic:
error: internal error: unable to execute QEMU command 'block-commit': Top image file /tmp/images/c/../b/b not found
even though the file exists. Trying to teach libvirt the rules on
which name qemu will expect is not worth the effort (besides, we'd
have to remember it across libvirtd restarts, and track whether a
file was opened via metadata or via snapshot creation for a given
qemu process); it is easier to just always directly ask qemu what
string it expects to see in the first place.
As a safety valve, we validate that any name returned by qemu
still maps to the same local file as we have tracked it, so that
a compromised qemu cannot accidentally cause us to act on an
incorrect file.
* src/qemu/qemu_monitor.h (qemuMonitorDiskNameLookup): New
prototype.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDiskNameLookup):
Likewise.
* src/qemu/qemu_monitor.c (qemuMonitorDiskNameLookup): New function.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONDiskNameLookupOne): Likewise.
* src/qemu/qemu_driver.c (qemuDomainBlockCommit)
(qemuDomainBlockJobImpl): Use it.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-03-11 14:37:04 -06:00
|
|
|
ret = qemuMonitorJSONDiskNameLookupOne(image, top, target);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* Guarantee an error when returning NULL, but don't override a
|
|
|
|
* more specific error if one was already generated. */
|
2018-05-05 13:04:21 +01:00
|
|
|
if (!ret && virGetLastErrorCode() == VIR_ERR_OK)
|
qemu: read backing chain names from qemu
https://bugzilla.redhat.com/show_bug.cgi?id=1199182 documents that
after a series of disk snapshots into existing destination images,
followed by active commits of the top image, it is possible for
qemu 2.2 and earlier to end up tracking a different name for the
image than what it would have had when opening the chain afresh.
That is, when starting with the chain 'a <- b <- c', the name
associated with 'b' is how it was spelled in the metadata of 'c',
but when starting with 'a', taking two snapshots into 'a <- b <- c',
then committing 'c' back into 'b', the name associated with 'b' is
now the name used when taking the first snapshot.
Sadly, older qemu doesn't know how to treat different spellings of
the same filename as identical files (it uses strcmp() instead of
checking for the same inode), which means libvirt's attempt to
commit an image using solely the names learned from qcow2 metadata
fails with a cryptic:
error: internal error: unable to execute QEMU command 'block-commit': Top image file /tmp/images/c/../b/b not found
even though the file exists. Trying to teach libvirt the rules on
which name qemu will expect is not worth the effort (besides, we'd
have to remember it across libvirtd restarts, and track whether a
file was opened via metadata or via snapshot creation for a given
qemu process); it is easier to just always directly ask qemu what
string it expects to see in the first place.
As a safety valve, we validate that any name returned by qemu
still maps to the same local file as we have tracked it, so that
a compromised qemu cannot accidentally cause us to act on an
incorrect file.
* src/qemu/qemu_monitor.h (qemuMonitorDiskNameLookup): New
prototype.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDiskNameLookup):
Likewise.
* src/qemu/qemu_monitor.c (qemuMonitorDiskNameLookup): New function.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONDiskNameLookupOne): Likewise.
* src/qemu/qemu_driver.c (qemuDomainBlockCommit)
(qemuDomainBlockJobImpl): Use it.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-03-11 14:37:04 -06:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("unable to find backing name for device %s"),
|
|
|
|
device);
|
|
|
|
|
|
|
|
cleanup:
|
2016-09-27 09:39:21 -04:00
|
|
|
virJSONValueFree(devices);
|
qemu: read backing chain names from qemu
https://bugzilla.redhat.com/show_bug.cgi?id=1199182 documents that
after a series of disk snapshots into existing destination images,
followed by active commits of the top image, it is possible for
qemu 2.2 and earlier to end up tracking a different name for the
image than what it would have had when opening the chain afresh.
That is, when starting with the chain 'a <- b <- c', the name
associated with 'b' is how it was spelled in the metadata of 'c',
but when starting with 'a', taking two snapshots into 'a <- b <- c',
then committing 'c' back into 'b', the name associated with 'b' is
now the name used when taking the first snapshot.
Sadly, older qemu doesn't know how to treat different spellings of
the same filename as identical files (it uses strcmp() instead of
checking for the same inode), which means libvirt's attempt to
commit an image using solely the names learned from qcow2 metadata
fails with a cryptic:
error: internal error: unable to execute QEMU command 'block-commit': Top image file /tmp/images/c/../b/b not found
even though the file exists. Trying to teach libvirt the rules on
which name qemu will expect is not worth the effort (besides, we'd
have to remember it across libvirtd restarts, and track whether a
file was opened via metadata or via snapshot creation for a given
qemu process); it is easier to just always directly ask qemu what
string it expects to see in the first place.
As a safety valve, we validate that any name returned by qemu
still maps to the same local file as we have tracked it, so that
a compromised qemu cannot accidentally cause us to act on an
incorrect file.
* src/qemu/qemu_monitor.h (qemuMonitorDiskNameLookup): New
prototype.
* src/qemu/qemu_monitor_json.h (qemuMonitorJSONDiskNameLookup):
Likewise.
* src/qemu/qemu_monitor.c (qemuMonitorDiskNameLookup): New function.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONDiskNameLookupOne): Likewise.
* src/qemu/qemu_driver.c (qemuDomainBlockCommit)
(qemuDomainBlockJobImpl): Use it.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-03-11 14:37:04 -06:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONArbitraryCommand(qemuMonitor *mon,
|
2010-04-16 22:12:45 -04:00
|
|
|
const char *cmd_str,
|
2019-09-19 17:53:45 +02:00
|
|
|
char **reply_str)
|
2010-04-16 22:12:45 -04:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = NULL;
|
|
|
|
virJSONValue *reply = NULL;
|
2010-04-16 22:12:45 -04:00
|
|
|
int ret = -1;
|
|
|
|
|
2019-09-19 17:53:45 +02:00
|
|
|
if (!(cmd = virJSONValueFromString(cmd_str)))
|
|
|
|
goto cleanup;
|
2010-04-16 22:12:45 -04:00
|
|
|
|
2019-09-19 17:53:45 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2010-04-16 22:12:45 -04:00
|
|
|
|
2019-09-19 17:53:45 +02:00
|
|
|
if (!(*reply_str = virJSONValueToString(reply, false)))
|
|
|
|
goto cleanup;
|
2010-04-16 22:12:45 -04:00
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2010-04-16 22:12:45 -04:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2011-05-10 16:26:06 +08:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONInjectNMI(qemuMonitor *mon)
|
2011-05-10 16:26:06 +08:00
|
|
|
{
|
2016-05-04 10:28:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2011-05-10 16:26:06 +08:00
|
|
|
|
|
|
|
cmd = qemuMonitorJSONMakeCommand("inject-nmi", NULL);
|
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2011-05-10 16:26:06 +08:00
|
|
|
goto cleanup;
|
|
|
|
|
2018-05-22 13:13:03 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2011-05-10 16:26:06 +08:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2011-05-10 16:26:06 +08:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2011-04-01 08:23:58 +02:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONSendKey(qemuMonitor *mon,
|
2011-07-21 15:55:56 +08:00
|
|
|
unsigned int holdtime,
|
|
|
|
unsigned int *keycodes,
|
|
|
|
unsigned int nkeycodes)
|
|
|
|
{
|
2013-04-11 14:33:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = NULL;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *keys = NULL;
|
|
|
|
virJSONValue *key = NULL;
|
Convert 'int i' to 'size_t i' in src/qemu files
Convert the type of loop iterators named 'i', 'j', k',
'ii', 'jj', 'kk', to be 'size_t' instead of 'int' or
'unsigned int', also santizing 'ii', 'jj', 'kk' to use
the normal 'i', 'j', 'k' naming
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2013-07-08 15:09:33 +01:00
|
|
|
size_t i;
|
2013-04-11 14:33:43 +02:00
|
|
|
|
|
|
|
/* create the key data array */
|
2020-01-31 08:18:36 +01:00
|
|
|
keys = virJSONValueNewArray();
|
2013-04-11 14:33:43 +02:00
|
|
|
|
|
|
|
for (i = 0; i < nkeycodes; i++) {
|
|
|
|
if (keycodes[i] > 0xffff) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED,
|
Convert 'int i' to 'size_t i' in src/qemu files
Convert the type of loop iterators named 'i', 'j', k',
'ii', 'jj', 'kk', to be 'size_t' instead of 'int' or
'unsigned int', also santizing 'ii', 'jj', 'kk' to use
the normal 'i', 'j', 'k' naming
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2013-07-08 15:09:33 +01:00
|
|
|
_("keycode %zu is invalid: 0x%X"), i, keycodes[i]);
|
2013-04-11 14:33:43 +02:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* create single key object */
|
2020-03-04 10:04:33 +01:00
|
|
|
key = virJSONValueNewObject();
|
2013-04-11 14:33:43 +02:00
|
|
|
|
|
|
|
/* Union KeyValue has two types, use the generic one */
|
|
|
|
if (virJSONValueObjectAppendString(key, "type", "number") < 0)
|
2013-07-04 12:14:12 +02:00
|
|
|
goto cleanup;
|
2013-04-11 14:33:43 +02:00
|
|
|
|
|
|
|
/* with the keycode */
|
|
|
|
if (virJSONValueObjectAppendNumberInt(key, "data", keycodes[i]) < 0)
|
2013-07-04 12:14:12 +02:00
|
|
|
goto cleanup;
|
2013-04-11 14:33:43 +02:00
|
|
|
|
2021-02-11 17:57:45 +01:00
|
|
|
if (virJSONValueArrayAppend(keys, &key) < 0)
|
2013-07-04 12:14:12 +02:00
|
|
|
goto cleanup;
|
2013-04-11 14:33:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
cmd = qemuMonitorJSONMakeCommand("send-key",
|
2018-03-30 11:12:57 +02:00
|
|
|
"a:keys", &keys,
|
2014-06-03 11:19:51 +02:00
|
|
|
"p:hold-time", holdtime,
|
2014-05-13 17:28:45 +02:00
|
|
|
NULL);
|
2013-04-11 14:33:43 +02:00
|
|
|
if (!cmd)
|
|
|
|
goto cleanup;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2013-04-11 14:33:43 +02:00
|
|
|
goto cleanup;
|
|
|
|
|
2018-05-22 13:13:03 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2013-04-11 14:33:43 +02:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2013-04-11 14:33:43 +02:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
virJSONValueFree(keys);
|
|
|
|
virJSONValueFree(key);
|
|
|
|
return ret;
|
2011-07-21 15:55:56 +08:00
|
|
|
}
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONScreendump(qemuMonitor *mon,
|
2018-05-17 13:53:34 +02:00
|
|
|
const char *device,
|
|
|
|
unsigned int head,
|
2011-04-01 08:23:58 +02:00
|
|
|
const char *file)
|
|
|
|
{
|
2016-05-04 10:28:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2011-04-01 08:23:58 +02:00
|
|
|
|
|
|
|
cmd = qemuMonitorJSONMakeCommand("screendump",
|
|
|
|
"s:filename", file,
|
2018-05-17 13:53:34 +02:00
|
|
|
"S:device", device,
|
|
|
|
"p:head", head,
|
2011-04-01 08:23:58 +02:00
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2011-04-01 08:23:58 +02:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2011-04-01 08:23:58 +02:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2011-04-01 08:23:58 +02:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2011-07-22 13:39:37 +08:00
|
|
|
|
2015-05-22 13:33:49 +02:00
|
|
|
|
2014-08-27 13:29:14 -06:00
|
|
|
static int
|
2020-10-22 19:04:18 +02:00
|
|
|
qemuMonitorJSONParseBlockJobInfo(GHashTable *blockJobs,
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *entry,
|
2019-06-11 16:42:53 +02:00
|
|
|
bool rawjobname)
|
2011-07-22 13:39:37 +08:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorBlockJobInfo *info = NULL;
|
2015-05-22 13:33:49 +02:00
|
|
|
const char *device;
|
2011-07-22 13:39:37 +08:00
|
|
|
const char *type;
|
|
|
|
|
2015-05-22 13:33:49 +02:00
|
|
|
if (!(device = virJSONValueObjectGetString(entry, "device"))) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("entry was missing 'device'"));
|
2011-07-22 13:39:37 +08:00
|
|
|
return -1;
|
|
|
|
}
|
2019-06-11 16:42:53 +02:00
|
|
|
|
|
|
|
if (!rawjobname)
|
|
|
|
device = qemuAliasDiskDriveSkipPrefix(device);
|
2011-07-22 13:39:37 +08:00
|
|
|
|
2020-10-05 12:26:34 +02:00
|
|
|
info = g_new0(qemuMonitorBlockJobInfo, 1);
|
2020-10-04 21:51:15 +02:00
|
|
|
|
|
|
|
if (virHashAddEntry(blockJobs, device, info) < 0) {
|
2015-05-22 13:33:49 +02:00
|
|
|
VIR_FREE(info);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(type = virJSONValueObjectGetString(entry, "type"))) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("entry was missing 'type'"));
|
2011-07-22 13:39:37 +08:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (STREQ(type, "stream"))
|
|
|
|
info->type = VIR_DOMAIN_BLOCK_JOB_TYPE_PULL;
|
2012-10-03 15:13:21 -06:00
|
|
|
else if (STREQ(type, "commit"))
|
|
|
|
info->type = VIR_DOMAIN_BLOCK_JOB_TYPE_COMMIT;
|
2012-10-12 14:06:10 -06:00
|
|
|
else if (STREQ(type, "mirror"))
|
|
|
|
info->type = VIR_DOMAIN_BLOCK_JOB_TYPE_COPY;
|
2019-10-18 15:10:33 +02:00
|
|
|
else if (STREQ(type, "backup"))
|
|
|
|
info->type = VIR_DOMAIN_BLOCK_JOB_TYPE_BACKUP;
|
2011-07-22 13:39:37 +08:00
|
|
|
else
|
|
|
|
info->type = VIR_DOMAIN_BLOCK_JOB_TYPE_UNKNOWN;
|
|
|
|
|
2015-05-22 13:33:49 +02:00
|
|
|
if (virJSONValueObjectGetNumberUlong(entry, "speed", &info->bandwidth) < 0) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("entry was missing 'speed'"));
|
2011-07-22 13:39:37 +08:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberUlong(entry, "offset", &info->cur) < 0) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("entry was missing 'offset'"));
|
2011-07-22 13:39:37 +08:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberUlong(entry, "len", &info->end) < 0) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("entry was missing 'len'"));
|
2011-07-22 13:39:37 +08:00
|
|
|
return -1;
|
|
|
|
}
|
2015-05-22 13:33:49 +02:00
|
|
|
|
2020-12-04 16:07:58 +01:00
|
|
|
if (virJSONValueObjectGetBoolean(entry, "ready", &info->ready) == 0)
|
|
|
|
info->ready_present = true;
|
2015-07-15 15:11:02 +02:00
|
|
|
|
2015-05-22 13:33:49 +02:00
|
|
|
return 0;
|
2011-07-22 13:39:37 +08:00
|
|
|
}
|
|
|
|
|
2020-10-22 19:04:18 +02:00
|
|
|
GHashTable *
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetAllBlockJobInfo(qemuMonitor *mon,
|
2019-06-11 16:42:53 +02:00
|
|
|
bool rawjobname)
|
2011-07-22 13:39:37 +08:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = NULL;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *data;
|
2018-04-19 17:29:02 -04:00
|
|
|
size_t nr_results;
|
Convert 'int i' to 'size_t i' in src/qemu files
Convert the type of loop iterators named 'i', 'j', k',
'ii', 'jj', 'kk', to be 'size_t' instead of 'int' or
'unsigned int', also santizing 'ii', 'jj', 'kk' to use
the normal 'i', 'j', 'k' naming
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2013-07-08 15:09:33 +01:00
|
|
|
size_t i;
|
2020-10-22 19:04:18 +02:00
|
|
|
GHashTable *blockJobs = NULL;
|
2011-07-22 13:39:37 +08:00
|
|
|
|
2014-08-27 13:29:14 -06:00
|
|
|
cmd = qemuMonitorJSONMakeCommand("query-block-jobs", NULL);
|
|
|
|
if (!cmd)
|
2015-05-22 13:33:49 +02:00
|
|
|
return NULL;
|
2014-08-27 13:29:14 -06:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2011-07-22 13:39:37 +08:00
|
|
|
|
qemu: simplify json parsing
Rather than grabbing an arbitrary JSON value and then checking
if it has the right type, we might as well request the correct
type to begin with.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONIOProcessEvent)
(qemuMonitorJSONCommandWithFd, qemuMonitorJSONHandleGraphics)
(qemuMonitorJSONGetStatus, qemuMonitorJSONExtractCPUInfo)
(qemuMonitorJSONGetVirtType, qemuMonitorJSONGetBalloonInfo)
(qemuMonitorJSONGetMemoryStats)
(qemuMonitorJSONDevGetBlockExtent)
(qemuMonitorJSONGetOneBlockStatsInfo)
(qemuMonitorJSONGetAllBlockStatsInfo)
(qemuMonitorJSONBlockStatsUpdateCapacityOne)
(qemuMonitorJSONBlockStatsUpdateCapacity)
(qemuMonitorJSONGetBlockExtent)
(qemuMonitorJSONGetMigrationStatusReply)
(qemuMonitorJSONGetDumpGuestMemoryCapability)
(qemuMonitorJSONAddFd, qemuMonitorJSONQueryRxFilterParse)
(qemuMonitorJSONExtractChardevInfo)
(qemuMonitorJSONDiskNameLookupOne)
(qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONGetAllBlockJobInfo)
(qemuMonitorJSONBlockIoThrottleInfo, qemuMonitorJSONGetVersion)
(qemuMonitorJSONGetMachines, qemuMonitorJSONGetCPUDefinitions)
(qemuMonitorJSONGetCommands, qemuMonitorJSONGetEvents)
(qemuMonitorJSONGetKVMState, qemuMonitorJSONGetObjectTypes)
(qemuMonitorJSONGetObjectListPaths)
(qemuMonitorJSONGetObjectProps, qemuMonitorJSONGetTargetArch)
(qemuMonitorJSONGetMigrationCapabilities)
(qemuMonitorJSONGetStringArray, qemuMonitorJSONAttachCharDev)
(qemuMonitorJSONGetCPUx86Data, qemuMonitorJSONGetIOThreads)
(qemuMonitorJSONGetMemoryDeviceInfo): Use shorter idioms.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-06-20 13:40:21 -06:00
|
|
|
if ((data = virJSONValueObjectGetArray(reply, "return")) == NULL) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("reply was missing return data"));
|
2014-08-27 13:29:14 -06:00
|
|
|
goto cleanup;
|
2011-07-22 13:39:37 +08:00
|
|
|
}
|
|
|
|
|
2018-04-19 17:29:02 -04:00
|
|
|
nr_results = virJSONValueArraySize(data);
|
2020-10-21 12:08:22 +02:00
|
|
|
if (!(blockJobs = virHashNew(g_free)))
|
2015-05-22 13:33:49 +02:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
for (i = 0; i < nr_results; i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *entry = virJSONValueArrayGet(data, i);
|
2011-08-02 13:17:04 -06:00
|
|
|
if (!entry) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("missing array element"));
|
2015-05-22 13:33:49 +02:00
|
|
|
goto error;
|
2011-08-02 13:17:04 -06:00
|
|
|
}
|
2019-06-11 16:42:53 +02:00
|
|
|
if (qemuMonitorJSONParseBlockJobInfo(blockJobs, entry, rawjobname) < 0)
|
2015-05-22 13:33:49 +02:00
|
|
|
goto error;
|
2011-07-22 13:39:37 +08:00
|
|
|
}
|
|
|
|
|
2014-08-27 13:29:14 -06:00
|
|
|
cleanup:
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
2015-05-22 13:33:49 +02:00
|
|
|
return blockJobs;
|
|
|
|
|
|
|
|
error:
|
|
|
|
virHashFree(blockJobs);
|
|
|
|
blockJobs = NULL;
|
|
|
|
goto cleanup;
|
2011-07-22 13:39:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-03-31 17:13:21 +02:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBlockJobError(virJSONValue *cmd,
|
|
|
|
virJSONValue *reply,
|
2018-08-14 13:11:05 +02:00
|
|
|
const char *jobname)
|
2015-03-31 17:13:21 +02:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *error;
|
2015-03-31 17:13:21 +02:00
|
|
|
|
2018-08-14 12:47:05 +02:00
|
|
|
if ((error = virJSONValueObjectGet(reply, "error")) &&
|
|
|
|
(qemuMonitorJSONErrorIsClass(error, "DeviceNotActive"))) {
|
2015-03-31 17:13:21 +02:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID,
|
2018-08-14 13:11:05 +02:00
|
|
|
_("No active block job '%s'"), jobname);
|
2018-08-14 12:47:05 +02:00
|
|
|
return -1;
|
2015-03-31 17:13:21 +02:00
|
|
|
}
|
|
|
|
|
2018-08-14 12:47:05 +02:00
|
|
|
return qemuMonitorJSONCheckError(cmd, reply);
|
2015-03-31 17:13:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
blockjob: fix block-stream bandwidth race
With RHEL 6.2, virDomainBlockPull(dom, dev, bandwidth, 0) has a race
with non-zero bandwidth: there is a window between the block_stream
and block_job_set_speed monitor commands where an unlimited amount
of data was let through, defeating the point of a throttle.
This race was first identified in commit a9d3495e, and libvirt was
able to reduce the size of the window for that race. In the meantime,
the qemu developers decided to fix things properly; per this message:
https://lists.gnu.org/archive/html/qemu-devel/2012-04/msg03793.html
the fix will be in qemu 1.1, and changes block-job-set-speed to use
a different parameter name, as well as adding a new optional parameter
to block-stream, which eliminates the race altogether.
Since our documentation already mentioned that we can refuse a non-zero
bandwidth for some hypervisors, I think the best solution is to do
just that for RHEL 6.2 qemu, so that the race is obvious to the user
(anyone using stock RHEL 6.2 binaries won't have this patch, and anyone
building their own libvirt with this patch for RHEL can also rebuild
qemu to get the modern semantics, so it is no real loss in behavior).
Meanwhile the code must be fixed to honor actual qemu 1.1 naming.
Rename the parameter to 'modern', since the naming difference now
covers more than just 'async' block-job-cancel. And while at it,
fix an unchecked integer overflow.
* src/qemu/qemu_monitor.h (enum BLOCK_JOB_CMD): Drop unused value,
rename enum to match conventions.
* src/qemu/qemu_monitor.c (qemuMonitorBlockJob): Reflect enum rename.
* src/qemu_qemu_monitor_json.h (qemuMonitorJSONBlockJob): Likewise.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONBlockJob): Likewise,
and support difference between RHEL 6.2 and qemu 1.1 block pull.
* src/qemu/qemu_driver.c (qemuDomainBlockJobImpl): Reject
bandwidth during pull with too-old qemu.
* src/libvirt.c (virDomainBlockPull, virDomainBlockRebase):
Document this.
2012-04-25 16:49:44 -06:00
|
|
|
/* speed is in bytes/sec */
|
2012-02-18 09:20:01 -07:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBlockStream(qemuMonitor *mon,
|
2015-04-01 11:45:35 +02:00
|
|
|
const char *device,
|
2018-08-15 13:13:53 +02:00
|
|
|
const char *jobname,
|
|
|
|
bool persistjob,
|
2015-04-01 11:45:35 +02:00
|
|
|
const char *base,
|
2018-08-15 13:13:53 +02:00
|
|
|
const char *baseNode,
|
2015-04-01 11:45:35 +02:00
|
|
|
const char *backingName,
|
2017-09-13 15:40:46 +02:00
|
|
|
unsigned long long speed)
|
2011-07-22 13:39:37 +08:00
|
|
|
{
|
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = NULL;
|
|
|
|
virJSONValue *reply = NULL;
|
2018-08-15 13:13:53 +02:00
|
|
|
virTristateBool autofinalize = VIR_TRISTATE_BOOL_ABSENT;
|
|
|
|
virTristateBool autodismiss = VIR_TRISTATE_BOOL_ABSENT;
|
|
|
|
|
|
|
|
if (persistjob) {
|
|
|
|
autofinalize = VIR_TRISTATE_BOOL_YES;
|
|
|
|
autodismiss = VIR_TRISTATE_BOOL_NO;
|
|
|
|
}
|
2014-05-16 17:51:21 +02:00
|
|
|
|
2018-08-14 12:51:15 +02:00
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("block-stream",
|
2015-04-01 11:45:35 +02:00
|
|
|
"s:device", device,
|
2018-08-15 13:13:53 +02:00
|
|
|
"S:job-id", jobname,
|
2015-04-01 11:45:35 +02:00
|
|
|
"Y:speed", speed,
|
|
|
|
"S:base", base,
|
2018-08-15 13:13:53 +02:00
|
|
|
"S:base-node", baseNode,
|
2015-04-01 11:45:35 +02:00
|
|
|
"S:backing-file", backingName,
|
2018-08-15 13:13:53 +02:00
|
|
|
"T:auto-finalize", autofinalize,
|
|
|
|
"T:auto-dismiss", autodismiss,
|
2015-04-01 11:45:35 +02:00
|
|
|
NULL)))
|
2011-07-22 13:39:37 +08:00
|
|
|
return -1;
|
|
|
|
|
2015-03-31 17:13:21 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2011-07-22 13:39:37 +08:00
|
|
|
|
2018-08-14 12:54:16 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
2015-03-31 17:13:21 +02:00
|
|
|
goto cleanup;
|
2014-03-05 15:08:56 +01:00
|
|
|
|
2015-03-31 17:13:21 +02:00
|
|
|
ret = 0;
|
2011-07-22 13:39:37 +08:00
|
|
|
|
2015-03-31 17:13:21 +02:00
|
|
|
cleanup:
|
2011-07-22 13:39:37 +08:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2011-10-21 09:00:13 +01:00
|
|
|
|
2015-04-01 09:47:04 +02:00
|
|
|
|
2015-04-01 10:40:06 +02:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBlockJobCancel(qemuMonitor *mon,
|
2021-04-20 14:14:51 +02:00
|
|
|
const char *jobname,
|
|
|
|
bool force)
|
2015-04-01 10:40:06 +02:00
|
|
|
{
|
2021-04-20 16:16:56 +02:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2015-04-01 10:40:06 +02:00
|
|
|
|
2018-08-14 12:51:15 +02:00
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("block-job-cancel",
|
2018-08-14 13:11:05 +02:00
|
|
|
"s:device", jobname,
|
2021-04-20 14:14:51 +02:00
|
|
|
"B:force", force,
|
2015-04-01 10:40:06 +02:00
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2021-04-20 16:16:56 +02:00
|
|
|
return -1;
|
2015-04-01 10:40:06 +02:00
|
|
|
|
2018-08-14 13:11:05 +02:00
|
|
|
if (qemuMonitorJSONBlockJobError(cmd, reply, jobname) < 0)
|
2021-04-20 16:16:56 +02:00
|
|
|
return -1;
|
2015-04-01 10:40:06 +02:00
|
|
|
|
2021-04-20 16:16:56 +02:00
|
|
|
return 0;
|
2015-04-01 10:40:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-04-01 09:47:04 +02:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBlockJobSetSpeed(qemuMonitor *mon,
|
2018-08-14 13:11:05 +02:00
|
|
|
const char *jobname,
|
2017-09-13 15:40:46 +02:00
|
|
|
unsigned long long speed)
|
2015-04-01 09:47:04 +02:00
|
|
|
{
|
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2015-04-01 09:47:04 +02:00
|
|
|
|
2018-08-14 12:51:15 +02:00
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("block-job-set-speed",
|
2018-08-14 13:11:05 +02:00
|
|
|
"s:device", jobname,
|
2017-09-13 15:40:46 +02:00
|
|
|
"J:speed", speed,
|
2015-04-01 09:47:04 +02:00
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2018-08-14 13:11:05 +02:00
|
|
|
if (qemuMonitorJSONBlockJobError(cmd, reply, jobname) < 0)
|
2015-04-01 09:47:04 +02:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-08-14 13:02:43 +02:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONDrivePivot(qemuMonitor *mon,
|
2018-08-14 13:11:05 +02:00
|
|
|
const char *jobname)
|
2018-08-14 13:02:43 +02:00
|
|
|
{
|
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2018-08-14 13:02:43 +02:00
|
|
|
|
|
|
|
cmd = qemuMonitorJSONMakeCommand("block-job-complete",
|
2018-08-14 13:11:05 +02:00
|
|
|
"s:device", jobname,
|
2018-08-14 13:02:43 +02:00
|
|
|
NULL);
|
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2018-08-14 13:11:05 +02:00
|
|
|
if (qemuMonitorJSONBlockJobError(cmd, reply, jobname) < 0)
|
2018-08-14 13:02:43 +02:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-08-16 11:48:41 +02:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONJobDismiss(qemuMonitor *mon,
|
2018-08-16 11:48:41 +02:00
|
|
|
const char *jobname)
|
|
|
|
{
|
2019-10-15 14:47:50 +02:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2018-08-16 11:48:41 +02:00
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("job-dismiss",
|
|
|
|
"s:id", jobname,
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONBlockJobError(cmd, reply, jobname) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-08-16 11:48:41 +02:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONJobComplete(qemuMonitor *mon,
|
2018-08-16 11:48:41 +02:00
|
|
|
const char *jobname)
|
|
|
|
{
|
2019-10-15 14:47:50 +02:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2018-08-16 11:48:41 +02:00
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("job-complete",
|
|
|
|
"s:id", jobname,
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONBlockJobError(cmd, reply, jobname) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONOpenGraphics(qemuMonitor *mon,
|
2011-10-21 09:00:13 +01:00
|
|
|
const char *protocol,
|
|
|
|
const char *fdname,
|
|
|
|
bool skipauth)
|
|
|
|
{
|
2016-05-04 10:28:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2011-10-21 09:00:13 +01:00
|
|
|
|
|
|
|
cmd = qemuMonitorJSONMakeCommand("add_client",
|
|
|
|
"s:protocol", protocol,
|
|
|
|
"s:fdname", fdname,
|
|
|
|
"b:skipauth", skipauth,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2011-10-21 09:00:13 +01:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2011-10-21 09:00:13 +01:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2011-10-21 09:00:13 +01:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2011-11-15 17:02:45 +08:00
|
|
|
|
|
|
|
|
2017-11-03 13:09:47 +01:00
|
|
|
#define GET_THROTTLE_STATS_OPTIONAL(FIELD, STORE) \
|
|
|
|
if (virJSONValueObjectGetNumberUlong(inserted, \
|
|
|
|
FIELD, \
|
|
|
|
&reply->STORE) < 0) { \
|
|
|
|
reply->STORE = 0; \
|
2014-10-29 13:16:03 +01:00
|
|
|
}
|
2017-11-03 13:09:47 +01:00
|
|
|
#define GET_THROTTLE_STATS(FIELD, STORE) \
|
|
|
|
if (virJSONValueObjectGetNumberUlong(inserted, \
|
|
|
|
FIELD, \
|
|
|
|
&reply->STORE) < 0) { \
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, \
|
|
|
|
_("block_io_throttle field '%s' missing " \
|
|
|
|
"in qemu's output"), \
|
|
|
|
#STORE); \
|
2019-10-21 15:18:59 -03:00
|
|
|
return -1; \
|
2012-08-09 09:58:05 +02:00
|
|
|
}
|
2011-11-15 17:02:45 +08:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBlockIoThrottleInfo(virJSONValue *io_throttle,
|
2018-07-25 15:14:43 +02:00
|
|
|
const char *drivealias,
|
|
|
|
const char *qdevid,
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainBlockIoTuneInfo *reply)
|
2011-11-15 17:02:45 +08:00
|
|
|
{
|
Convert 'int i' to 'size_t i' in src/qemu files
Convert the type of loop iterators named 'i', 'j', k',
'ii', 'jj', 'kk', to be 'size_t' instead of 'int' or
'unsigned int', also santizing 'ii', 'jj', 'kk' to use
the normal 'i', 'j', 'k' naming
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2013-07-08 15:09:33 +01:00
|
|
|
size_t i;
|
2012-08-09 09:58:05 +02:00
|
|
|
bool found = false;
|
2011-11-15 17:02:45 +08:00
|
|
|
|
|
|
|
for (i = 0; i < virJSONValueArraySize(io_throttle); i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *temp_dev = virJSONValueArrayGet(io_throttle, i);
|
|
|
|
virJSONValue *inserted;
|
2018-07-25 15:14:43 +02:00
|
|
|
const char *current_drive;
|
|
|
|
const char *current_qdev;
|
2011-11-15 17:02:45 +08:00
|
|
|
|
2018-03-29 20:30:05 +02:00
|
|
|
if (!temp_dev || virJSONValueGetType(temp_dev) != VIR_JSON_TYPE_OBJECT) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2012-08-09 09:58:05 +02:00
|
|
|
_("block_io_throttle device entry "
|
|
|
|
"was not in expected format"));
|
2019-10-21 15:18:59 -03:00
|
|
|
return -1;
|
2011-11-15 17:02:45 +08:00
|
|
|
}
|
|
|
|
|
2018-07-25 15:14:43 +02:00
|
|
|
current_qdev = virJSONValueObjectGetString(temp_dev, "qdev");
|
|
|
|
current_drive = virJSONValueObjectGetString(temp_dev, "device");
|
|
|
|
|
|
|
|
if (!current_drive && !current_qdev) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2012-08-09 09:58:05 +02:00
|
|
|
_("block_io_throttle device entry "
|
|
|
|
"was not in expected format"));
|
2019-10-21 15:18:59 -03:00
|
|
|
return -1;
|
2011-11-15 17:02:45 +08:00
|
|
|
}
|
|
|
|
|
2018-08-22 13:20:52 +02:00
|
|
|
if ((drivealias && current_drive && STRNEQ(current_drive, drivealias)) ||
|
|
|
|
(qdevid && current_qdev && STRNEQ(current_qdev, qdevid)))
|
2011-11-15 17:02:45 +08:00
|
|
|
continue;
|
|
|
|
|
2012-08-09 09:58:05 +02:00
|
|
|
found = true;
|
qemu: simplify json parsing
Rather than grabbing an arbitrary JSON value and then checking
if it has the right type, we might as well request the correct
type to begin with.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONIOProcessEvent)
(qemuMonitorJSONCommandWithFd, qemuMonitorJSONHandleGraphics)
(qemuMonitorJSONGetStatus, qemuMonitorJSONExtractCPUInfo)
(qemuMonitorJSONGetVirtType, qemuMonitorJSONGetBalloonInfo)
(qemuMonitorJSONGetMemoryStats)
(qemuMonitorJSONDevGetBlockExtent)
(qemuMonitorJSONGetOneBlockStatsInfo)
(qemuMonitorJSONGetAllBlockStatsInfo)
(qemuMonitorJSONBlockStatsUpdateCapacityOne)
(qemuMonitorJSONBlockStatsUpdateCapacity)
(qemuMonitorJSONGetBlockExtent)
(qemuMonitorJSONGetMigrationStatusReply)
(qemuMonitorJSONGetDumpGuestMemoryCapability)
(qemuMonitorJSONAddFd, qemuMonitorJSONQueryRxFilterParse)
(qemuMonitorJSONExtractChardevInfo)
(qemuMonitorJSONDiskNameLookupOne)
(qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONGetAllBlockJobInfo)
(qemuMonitorJSONBlockIoThrottleInfo, qemuMonitorJSONGetVersion)
(qemuMonitorJSONGetMachines, qemuMonitorJSONGetCPUDefinitions)
(qemuMonitorJSONGetCommands, qemuMonitorJSONGetEvents)
(qemuMonitorJSONGetKVMState, qemuMonitorJSONGetObjectTypes)
(qemuMonitorJSONGetObjectListPaths)
(qemuMonitorJSONGetObjectProps, qemuMonitorJSONGetTargetArch)
(qemuMonitorJSONGetMigrationCapabilities)
(qemuMonitorJSONGetStringArray, qemuMonitorJSONAttachCharDev)
(qemuMonitorJSONGetCPUx86Data, qemuMonitorJSONGetIOThreads)
(qemuMonitorJSONGetMemoryDeviceInfo): Use shorter idioms.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-06-20 13:40:21 -06:00
|
|
|
if (!(inserted = virJSONValueObjectGetObject(temp_dev, "inserted"))) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2012-08-09 09:58:05 +02:00
|
|
|
_("block_io_throttle inserted entry "
|
|
|
|
"was not in expected format"));
|
2019-10-21 15:18:59 -03:00
|
|
|
return -1;
|
2011-11-15 17:02:45 +08:00
|
|
|
}
|
2012-08-09 09:58:05 +02:00
|
|
|
GET_THROTTLE_STATS("bps", total_bytes_sec);
|
|
|
|
GET_THROTTLE_STATS("bps_rd", read_bytes_sec);
|
|
|
|
GET_THROTTLE_STATS("bps_wr", write_bytes_sec);
|
|
|
|
GET_THROTTLE_STATS("iops", total_iops_sec);
|
|
|
|
GET_THROTTLE_STATS("iops_rd", read_iops_sec);
|
|
|
|
GET_THROTTLE_STATS("iops_wr", write_iops_sec);
|
2016-05-20 08:30:45 +02:00
|
|
|
GET_THROTTLE_STATS_OPTIONAL("bps_max", total_bytes_sec_max);
|
|
|
|
GET_THROTTLE_STATS_OPTIONAL("bps_rd_max", read_bytes_sec_max);
|
|
|
|
GET_THROTTLE_STATS_OPTIONAL("bps_wr_max", write_bytes_sec_max);
|
|
|
|
GET_THROTTLE_STATS_OPTIONAL("iops_max", total_iops_sec_max);
|
|
|
|
GET_THROTTLE_STATS_OPTIONAL("iops_rd_max", read_iops_sec_max);
|
|
|
|
GET_THROTTLE_STATS_OPTIONAL("iops_wr_max", write_iops_sec_max);
|
|
|
|
GET_THROTTLE_STATS_OPTIONAL("iops_size", size_iops_sec);
|
2016-10-31 17:22:59 -04:00
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
reply->group_name = g_strdup(virJSONValueObjectGetString(inserted, "group"));
|
2016-10-31 17:22:59 -04:00
|
|
|
|
2016-09-18 12:02:50 -04:00
|
|
|
GET_THROTTLE_STATS_OPTIONAL("bps_max_length", total_bytes_sec_max_length);
|
|
|
|
GET_THROTTLE_STATS_OPTIONAL("bps_rd_max_length", read_bytes_sec_max_length);
|
|
|
|
GET_THROTTLE_STATS_OPTIONAL("bps_wr_max_length", write_bytes_sec_max_length);
|
|
|
|
GET_THROTTLE_STATS_OPTIONAL("iops_max_length", total_iops_sec_max_length);
|
|
|
|
GET_THROTTLE_STATS_OPTIONAL("iops_rd_max_length", read_iops_sec_max_length);
|
|
|
|
GET_THROTTLE_STATS_OPTIONAL("iops_wr_max_length", write_iops_sec_max_length);
|
2011-11-15 17:02:45 +08:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!found) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("cannot find throttling info for device '%s'"),
|
2018-07-25 15:14:43 +02:00
|
|
|
drivealias ? drivealias : qdevid);
|
2019-10-21 15:18:59 -03:00
|
|
|
return -1;
|
2011-11-15 17:02:45 +08:00
|
|
|
}
|
|
|
|
|
2019-10-21 15:18:59 -03:00
|
|
|
return 0;
|
2011-11-15 17:02:45 +08:00
|
|
|
}
|
2012-08-09 09:58:05 +02:00
|
|
|
#undef GET_THROTTLE_STATS
|
2014-10-29 13:16:03 +01:00
|
|
|
#undef GET_THROTTLE_STATS_OPTIONAL
|
2011-11-15 17:02:45 +08:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONSetBlockIoThrottle(qemuMonitor *mon,
|
2018-07-25 15:14:43 +02:00
|
|
|
const char *drivealias,
|
|
|
|
const char *qomid,
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainBlockIoTuneInfo *info,
|
2016-09-18 12:02:50 -04:00
|
|
|
bool supportMaxOptions,
|
2016-10-31 17:22:59 -04:00
|
|
|
bool supportGroupNameOption,
|
2016-09-18 12:02:50 -04:00
|
|
|
bool supportMaxLengthOptions)
|
2011-11-15 17:02:45 +08:00
|
|
|
{
|
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = NULL;
|
|
|
|
virJSONValue *result = NULL;
|
|
|
|
virJSONValue *args = NULL;
|
2018-07-25 15:14:43 +02:00
|
|
|
const char *errdev = drivealias;
|
|
|
|
|
|
|
|
if (!errdev)
|
|
|
|
errdev = qomid;
|
2011-11-15 17:02:45 +08:00
|
|
|
|
2016-11-07 14:46:09 -05:00
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("block_set_io_throttle", NULL)))
|
2011-11-15 17:02:45 +08:00
|
|
|
return -1;
|
|
|
|
|
2016-11-07 14:46:09 -05:00
|
|
|
if (virJSONValueObjectCreate(&args,
|
2018-07-25 15:14:43 +02:00
|
|
|
"S:device", drivealias,
|
|
|
|
"S:id", qomid,
|
2016-11-07 14:46:09 -05:00
|
|
|
"U:bps", info->total_bytes_sec,
|
|
|
|
"U:bps_rd", info->read_bytes_sec,
|
|
|
|
"U:bps_wr", info->write_bytes_sec,
|
|
|
|
"U:iops", info->total_iops_sec,
|
|
|
|
"U:iops_rd", info->read_iops_sec,
|
|
|
|
"U:iops_wr", info->write_iops_sec,
|
|
|
|
NULL) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (supportMaxOptions &&
|
|
|
|
virJSONValueObjectAdd(args,
|
|
|
|
"U:bps_max", info->total_bytes_sec_max,
|
|
|
|
"U:bps_rd_max", info->read_bytes_sec_max,
|
|
|
|
"U:bps_wr_max", info->write_bytes_sec_max,
|
|
|
|
"U:iops_max", info->total_iops_sec_max,
|
|
|
|
"U:iops_rd_max", info->read_iops_sec_max,
|
|
|
|
"U:iops_wr_max", info->write_iops_sec_max,
|
|
|
|
"U:iops_size", info->size_iops_sec,
|
|
|
|
NULL) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2016-10-31 17:22:59 -04:00
|
|
|
if (supportGroupNameOption &&
|
|
|
|
virJSONValueObjectAdd(args,
|
2017-01-24 15:50:00 +01:00
|
|
|
"S:group", info->group_name,
|
2016-10-31 17:22:59 -04:00
|
|
|
NULL) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2016-11-07 14:46:09 -05:00
|
|
|
if (supportMaxLengthOptions &&
|
|
|
|
virJSONValueObjectAdd(args,
|
|
|
|
"P:bps_max_length",
|
|
|
|
info->total_bytes_sec_max_length,
|
|
|
|
"P:bps_rd_max_length",
|
|
|
|
info->read_bytes_sec_max_length,
|
|
|
|
"P:bps_wr_max_length",
|
|
|
|
info->write_bytes_sec_max_length,
|
|
|
|
"P:iops_max_length",
|
|
|
|
info->total_iops_sec_max_length,
|
|
|
|
"P:iops_rd_max_length",
|
|
|
|
info->read_iops_sec_max_length,
|
|
|
|
"P:iops_wr_max_length",
|
|
|
|
info->write_iops_sec_max_length,
|
|
|
|
NULL) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2021-02-11 17:57:45 +01:00
|
|
|
if (virJSONValueObjectAppend(cmd, "arguments", &args) < 0)
|
2016-11-07 14:46:09 -05:00
|
|
|
goto cleanup;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &result) < 0)
|
|
|
|
goto cleanup;
|
2011-11-15 17:02:45 +08:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (virJSONValueObjectHasKey(result, "error")) {
|
2016-10-06 14:13:27 -04:00
|
|
|
if (qemuMonitorJSONHasError(result, "DeviceNotActive")) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID,
|
2018-07-25 15:14:43 +02:00
|
|
|
_("No active operation on device: %s"), errdev);
|
2016-10-06 14:13:27 -04:00
|
|
|
} else if (qemuMonitorJSONHasError(result, "NotSupported")) {
|
2012-07-18 16:22:03 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID,
|
2018-07-25 15:14:43 +02:00
|
|
|
_("Operation is not supported for device: %s"), errdev);
|
2016-10-06 14:13:27 -04:00
|
|
|
} else {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *error = virJSONValueObjectGet(result, "error");
|
2016-10-06 14:13:27 -04:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("unable to execute '%s', unexpected error: '%s'"),
|
|
|
|
qemuMonitorJSONCommandName(cmd),
|
|
|
|
qemuMonitorJSONStringifyError(error));
|
|
|
|
}
|
2016-05-04 10:28:43 +02:00
|
|
|
goto cleanup;
|
2011-11-15 17:02:45 +08:00
|
|
|
}
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2011-11-15 17:02:45 +08:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(result);
|
2016-11-07 14:46:09 -05:00
|
|
|
virJSONValueFree(args);
|
2011-11-15 17:02:45 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONGetBlockIoThrottle(qemuMonitor *mon,
|
2018-07-25 15:14:43 +02:00
|
|
|
const char *drivealias,
|
|
|
|
const char *qdevid,
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainBlockIoTuneInfo *reply)
|
2011-11-15 17:02:45 +08:00
|
|
|
{
|
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *devices = NULL;
|
2011-11-15 17:02:45 +08:00
|
|
|
|
2018-07-25 15:48:09 +02:00
|
|
|
if (!(devices = qemuMonitorJSONQueryBlock(mon)))
|
2011-11-15 17:02:45 +08:00
|
|
|
return -1;
|
|
|
|
|
2018-07-25 15:14:43 +02:00
|
|
|
ret = qemuMonitorJSONBlockIoThrottleInfo(devices, drivealias, qdevid, reply);
|
2018-07-25 15:48:09 +02:00
|
|
|
virJSONValueFree(devices);
|
2011-11-15 17:02:45 +08:00
|
|
|
return ret;
|
|
|
|
}
|
2012-02-10 13:33:52 +01:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONSystemWakeup(qemuMonitor *mon)
|
2012-02-10 13:33:52 +01:00
|
|
|
{
|
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = NULL;
|
|
|
|
virJSONValue *reply = NULL;
|
2012-02-10 13:33:52 +01:00
|
|
|
|
|
|
|
cmd = qemuMonitorJSONMakeCommand("system_wakeup", NULL);
|
2014-11-13 15:25:30 +01:00
|
|
|
if (!cmd)
|
2012-02-10 13:33:52 +01:00
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2012-02-10 13:33:52 +01:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2012-02-10 13:33:52 +01:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2012-02-10 13:33:52 +01:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2012-08-15 15:04:09 +01:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONGetVersion(qemuMonitor *mon,
|
2012-08-15 15:04:09 +01:00
|
|
|
int *major,
|
|
|
|
int *minor,
|
|
|
|
int *micro,
|
|
|
|
char **package)
|
|
|
|
{
|
2016-05-04 10:28:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *data;
|
|
|
|
virJSONValue *qemu;
|
2012-08-15 15:04:09 +01:00
|
|
|
|
|
|
|
*major = *minor = *micro = 0;
|
|
|
|
if (package)
|
|
|
|
*package = NULL;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-version", NULL)))
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2012-08-15 15:04:09 +01:00
|
|
|
goto cleanup;
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_OBJECT) < 0)
|
2016-05-04 10:28:43 +02:00
|
|
|
goto cleanup;
|
2012-08-15 15:04:09 +01:00
|
|
|
|
2016-11-21 00:10:06 +01:00
|
|
|
data = virJSONValueObjectGetObject(reply, "return");
|
2012-08-15 15:04:09 +01:00
|
|
|
|
qemu: simplify json parsing
Rather than grabbing an arbitrary JSON value and then checking
if it has the right type, we might as well request the correct
type to begin with.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONIOProcessEvent)
(qemuMonitorJSONCommandWithFd, qemuMonitorJSONHandleGraphics)
(qemuMonitorJSONGetStatus, qemuMonitorJSONExtractCPUInfo)
(qemuMonitorJSONGetVirtType, qemuMonitorJSONGetBalloonInfo)
(qemuMonitorJSONGetMemoryStats)
(qemuMonitorJSONDevGetBlockExtent)
(qemuMonitorJSONGetOneBlockStatsInfo)
(qemuMonitorJSONGetAllBlockStatsInfo)
(qemuMonitorJSONBlockStatsUpdateCapacityOne)
(qemuMonitorJSONBlockStatsUpdateCapacity)
(qemuMonitorJSONGetBlockExtent)
(qemuMonitorJSONGetMigrationStatusReply)
(qemuMonitorJSONGetDumpGuestMemoryCapability)
(qemuMonitorJSONAddFd, qemuMonitorJSONQueryRxFilterParse)
(qemuMonitorJSONExtractChardevInfo)
(qemuMonitorJSONDiskNameLookupOne)
(qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONGetAllBlockJobInfo)
(qemuMonitorJSONBlockIoThrottleInfo, qemuMonitorJSONGetVersion)
(qemuMonitorJSONGetMachines, qemuMonitorJSONGetCPUDefinitions)
(qemuMonitorJSONGetCommands, qemuMonitorJSONGetEvents)
(qemuMonitorJSONGetKVMState, qemuMonitorJSONGetObjectTypes)
(qemuMonitorJSONGetObjectListPaths)
(qemuMonitorJSONGetObjectProps, qemuMonitorJSONGetTargetArch)
(qemuMonitorJSONGetMigrationCapabilities)
(qemuMonitorJSONGetStringArray, qemuMonitorJSONAttachCharDev)
(qemuMonitorJSONGetCPUx86Data, qemuMonitorJSONGetIOThreads)
(qemuMonitorJSONGetMemoryDeviceInfo): Use shorter idioms.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-06-20 13:40:21 -06:00
|
|
|
if (!(qemu = virJSONValueObjectGetObject(data, "qemu"))) {
|
2012-08-15 15:04:09 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-version reply was missing 'qemu' data"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberInt(qemu, "major", major) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-version reply was missing 'major' version"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
if (virJSONValueObjectGetNumberInt(qemu, "minor", minor) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-version reply was missing 'minor' version"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
if (virJSONValueObjectGetNumberInt(qemu, "micro", micro) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-version reply was missing 'micro' version"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (package) {
|
|
|
|
const char *tmp;
|
|
|
|
if (!(tmp = virJSONValueObjectGetString(data, "package"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-version reply was missing 'package' version"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2019-10-20 13:49:46 +02:00
|
|
|
*package = g_strdup(tmp);
|
2012-08-15 15:04:09 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2012-08-15 15:04:09 +01:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2012-08-15 16:18:41 +01:00
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONGetMachines(qemuMonitor *mon,
|
|
|
|
qemuMonitorMachineInfo ***machines)
|
2012-08-15 16:18:41 +01:00
|
|
|
{
|
2016-05-04 10:28:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *data;
|
|
|
|
qemuMonitorMachineInfo **infolist = NULL;
|
2018-04-19 17:29:02 -04:00
|
|
|
size_t n = 0;
|
2012-08-15 16:18:41 +01:00
|
|
|
size_t i;
|
|
|
|
|
|
|
|
*machines = NULL;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-machines", NULL)))
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2012-08-15 16:18:41 +01:00
|
|
|
goto cleanup;
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_ARRAY) < 0)
|
2016-05-04 10:28:43 +02:00
|
|
|
goto cleanup;
|
2012-08-15 16:18:41 +01:00
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
data = virJSONValueObjectGetArray(reply, "return");
|
|
|
|
n = virJSONValueArraySize(data);
|
2012-08-15 16:18:41 +01:00
|
|
|
|
2013-04-26 11:47:23 -06:00
|
|
|
/* null-terminated list */
|
2021-03-11 08:16:13 +01:00
|
|
|
infolist = g_new0(qemuMonitorMachineInfo *, n + 1);
|
2012-08-15 16:18:41 +01:00
|
|
|
|
2013-05-21 15:21:20 +08:00
|
|
|
for (i = 0; i < n; i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *child = virJSONValueArrayGet(data, i);
|
2012-08-15 16:18:41 +01:00
|
|
|
const char *tmp;
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorMachineInfo *info;
|
2012-08-15 16:18:41 +01:00
|
|
|
|
2020-10-05 12:26:34 +02:00
|
|
|
info = g_new0(qemuMonitorMachineInfo, 1);
|
2012-08-15 16:18:41 +01:00
|
|
|
|
|
|
|
infolist[i] = info;
|
|
|
|
|
|
|
|
if (!(tmp = virJSONValueObjectGetString(child, "name"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-machines reply data was missing 'name'"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
info->name = g_strdup(tmp);
|
2012-08-15 16:18:41 +01:00
|
|
|
|
|
|
|
if (virJSONValueObjectHasKey(child, "is-default") &&
|
|
|
|
virJSONValueObjectGetBoolean(child, "is-default", &info->isDefault) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-machines reply has malformed 'is-default' data"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectHasKey(child, "alias")) {
|
|
|
|
if (!(tmp = virJSONValueObjectGetString(child, "alias"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-machines reply has malformed 'alias' data"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2019-10-20 13:49:46 +02:00
|
|
|
info->alias = g_strdup(tmp);
|
2012-08-15 16:18:41 +01:00
|
|
|
}
|
2013-06-26 17:46:35 +02:00
|
|
|
if (virJSONValueObjectHasKey(child, "cpu-max") &&
|
|
|
|
virJSONValueObjectGetNumberUint(child, "cpu-max", &info->maxCpus) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-machines reply has malformed 'cpu-max' data"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2016-07-29 09:45:19 +02:00
|
|
|
|
|
|
|
ignore_value(virJSONValueObjectGetBoolean(child, "hotpluggable-cpus",
|
|
|
|
&info->hotplugCpus));
|
2019-07-18 19:21:55 +02:00
|
|
|
|
|
|
|
if (virJSONValueObjectHasKey(child, "default-cpu-type")) {
|
|
|
|
if (!(tmp = virJSONValueObjectGetString(child, "default-cpu-type"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-machines reply has malformed "
|
|
|
|
"'default-cpu-type' data"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2019-12-11 14:00:27 +01:00
|
|
|
info->defaultCPU = g_strdup(tmp);
|
2019-07-18 19:21:55 +02:00
|
|
|
}
|
2020-05-14 13:11:01 +02:00
|
|
|
|
|
|
|
if (virJSONValueObjectHasKey(child, "numa-mem-supported")) {
|
|
|
|
if (virJSONValueObjectGetBoolean(child, "numa-mem-supported", &info->numaMemSupported) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("qemu-machines reply has malformed "
|
|
|
|
"'numa-mem-supported' data"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
info->numaMemSupported = true;
|
|
|
|
}
|
2020-05-25 19:13:43 +02:00
|
|
|
|
|
|
|
if (virJSONValueObjectHasKey(child, "default-ram-id")) {
|
|
|
|
if (!(tmp = virJSONValueObjectGetString(child, "default-ram-id"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-machines reply has malformed "
|
|
|
|
"'default-ram-id' data"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
info->defaultRAMid = g_strdup(tmp);
|
|
|
|
}
|
2021-01-22 12:16:23 +00:00
|
|
|
|
|
|
|
if (virJSONValueObjectHasKey(child, "deprecated") &&
|
|
|
|
virJSONValueObjectGetBoolean(child, "deprecated", &info->deprecated) < 0)
|
|
|
|
goto cleanup;
|
2012-08-15 16:18:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
ret = n;
|
2021-02-23 14:58:29 +01:00
|
|
|
*machines = g_steal_pointer(&infolist);
|
2012-08-15 16:18:41 +01:00
|
|
|
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2016-05-16 07:47:26 -04:00
|
|
|
if (infolist) {
|
2013-05-21 15:21:20 +08:00
|
|
|
for (i = 0; i < n; i++)
|
2012-08-15 16:18:41 +01:00
|
|
|
qemuMonitorMachineInfoFree(infolist[i]);
|
|
|
|
VIR_FREE(infolist);
|
|
|
|
}
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2012-08-20 15:58:20 +01:00
|
|
|
|
|
|
|
|
2016-04-21 12:51:01 +02:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetCPUDefinitions(qemuMonitor *mon,
|
|
|
|
qemuMonitorCPUDefs **cpuDefs)
|
2012-08-20 15:58:20 +01:00
|
|
|
{
|
2019-09-24 13:42:00 +02:00
|
|
|
g_autoptr(qemuMonitorCPUDefs) defs = NULL;
|
2019-09-24 13:20:49 +02:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *data;
|
2019-09-24 13:42:00 +02:00
|
|
|
size_t ncpus;
|
2012-08-20 15:58:20 +01:00
|
|
|
size_t i;
|
|
|
|
|
2019-09-24 13:42:00 +02:00
|
|
|
*cpuDefs = NULL;
|
2012-08-20 15:58:20 +01:00
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-cpu-definitions", NULL)))
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2019-09-24 13:42:00 +02:00
|
|
|
return -1;
|
2012-08-20 15:58:20 +01:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
/* Urgh, some QEMU architectures have the query-cpu-definitions
|
|
|
|
* command, but return 'GenericError' with string "Not supported",
|
|
|
|
* instead of simply omitting the command entirely :-(
|
|
|
|
*/
|
2019-09-24 13:42:00 +02:00
|
|
|
if (qemuMonitorJSONHasError(reply, "GenericError"))
|
|
|
|
return 0;
|
2012-09-28 15:55:54 +01:00
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_ARRAY) < 0)
|
2019-09-24 13:42:00 +02:00
|
|
|
return -1;
|
2012-08-20 15:58:20 +01:00
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
data = virJSONValueObjectGetArray(reply, "return");
|
2019-09-24 13:42:00 +02:00
|
|
|
ncpus = virJSONValueArraySize(data);
|
2012-08-20 15:58:20 +01:00
|
|
|
|
2019-09-24 13:42:00 +02:00
|
|
|
if (!(defs = qemuMonitorCPUDefsNew(ncpus)))
|
|
|
|
return -1;
|
2012-08-20 15:58:20 +01:00
|
|
|
|
2019-09-24 13:42:00 +02:00
|
|
|
for (i = 0; i < defs->ncpus; i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *child = virJSONValueArrayGet(data, i);
|
2012-08-20 15:58:20 +01:00
|
|
|
const char *tmp;
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorCPUDefInfo *cpu = defs->cpus + i;
|
2012-08-20 15:58:20 +01:00
|
|
|
|
|
|
|
if (!(tmp = virJSONValueObjectGetString(child, "name"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-cpu-definitions reply data was missing 'name'"));
|
2019-09-24 13:42:00 +02:00
|
|
|
return -1;
|
2012-08-20 15:58:20 +01:00
|
|
|
}
|
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
cpu->name = g_strdup(tmp);
|
2016-04-21 13:08:12 +02:00
|
|
|
|
2019-09-19 20:47:37 +02:00
|
|
|
if ((tmp = virJSONValueObjectGetString(child, "typename")) && *tmp)
|
|
|
|
cpu->type = g_strdup(tmp);
|
|
|
|
|
2016-04-21 13:08:12 +02:00
|
|
|
if (virJSONValueObjectHasKey(child, "unavailable-features")) {
|
2020-12-01 11:47:30 +01:00
|
|
|
if (!(cpu->blockers = virJSONValueObjectGetStringArray(child,
|
|
|
|
"unavailable-features")))
|
2019-09-24 13:42:00 +02:00
|
|
|
return -1;
|
2016-04-21 13:08:12 +02:00
|
|
|
|
2020-12-01 11:47:30 +01:00
|
|
|
if (g_strv_length(cpu->blockers) == 0) {
|
2019-10-29 14:36:11 +01:00
|
|
|
cpu->usable = VIR_DOMCAPS_CPU_USABLE_YES;
|
2020-12-01 11:47:30 +01:00
|
|
|
g_clear_pointer(&cpu->blockers, g_strfreev);
|
2017-09-20 10:45:49 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-10-29 14:36:11 +01:00
|
|
|
cpu->usable = VIR_DOMCAPS_CPU_USABLE_NO;
|
2016-04-21 13:08:12 +02:00
|
|
|
}
|
2021-01-22 11:15:08 +00:00
|
|
|
|
|
|
|
if (virJSONValueObjectHasKey(child, "deprecated") &&
|
|
|
|
virJSONValueObjectGetBoolean(child, "deprecated", &cpu->deprecated) < 0)
|
|
|
|
return -1;
|
2012-08-20 15:58:20 +01:00
|
|
|
}
|
|
|
|
|
2019-09-24 13:42:00 +02:00
|
|
|
*cpuDefs = g_steal_pointer(&defs);
|
|
|
|
return 0;
|
2012-08-20 15:58:20 +01:00
|
|
|
}
|
2012-08-22 10:25:20 +01:00
|
|
|
|
2017-02-22 16:01:30 +01:00
|
|
|
|
|
|
|
VIR_ENUM_IMPL(qemuMonitorCPUProperty,
|
|
|
|
QEMU_MONITOR_CPU_PROPERTY_LAST,
|
2019-01-20 11:30:15 -05:00
|
|
|
"boolean", "string", "number",
|
|
|
|
);
|
2017-02-22 16:01:30 +01:00
|
|
|
|
2016-12-18 14:22:26 -05:00
|
|
|
static int
|
|
|
|
qemuMonitorJSONParseCPUModelProperty(const char *key,
|
|
|
|
virJSONValue *value,
|
|
|
|
void *opaque)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorCPUModelInfo *machine_model = opaque;
|
|
|
|
qemuMonitorCPUProperty *prop;
|
2016-12-18 14:22:26 -05:00
|
|
|
|
2017-02-22 16:01:30 +01:00
|
|
|
prop = machine_model->props + machine_model->nprops;
|
2017-01-11 14:35:52 +01:00
|
|
|
|
2018-04-25 14:42:34 +02:00
|
|
|
switch ((virJSONType)virJSONValueGetType(value)) {
|
2017-02-22 16:01:30 +01:00
|
|
|
case VIR_JSON_TYPE_STRING:
|
2019-10-20 13:49:46 +02:00
|
|
|
prop->value.string = g_strdup(virJSONValueGetString(value));
|
2017-02-22 16:01:30 +01:00
|
|
|
prop->type = QEMU_MONITOR_CPU_PROPERTY_STRING;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_JSON_TYPE_NUMBER:
|
|
|
|
/* Ignore numbers which cannot be parsed as unsigned long long */
|
|
|
|
if (virJSONValueGetNumberLong(value, &prop->value.number) < 0)
|
|
|
|
return 0;
|
|
|
|
prop->type = QEMU_MONITOR_CPU_PROPERTY_NUMBER;
|
|
|
|
break;
|
2016-12-18 14:22:26 -05:00
|
|
|
|
2017-02-22 16:01:30 +01:00
|
|
|
case VIR_JSON_TYPE_BOOLEAN:
|
|
|
|
virJSONValueGetBoolean(value, &prop->value.boolean);
|
|
|
|
prop->type = QEMU_MONITOR_CPU_PROPERTY_BOOLEAN;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_JSON_TYPE_OBJECT:
|
|
|
|
case VIR_JSON_TYPE_ARRAY:
|
|
|
|
case VIR_JSON_TYPE_NULL:
|
|
|
|
return 0;
|
|
|
|
}
|
2016-12-18 14:22:26 -05:00
|
|
|
|
|
|
|
machine_model->nprops++;
|
2019-10-20 13:49:46 +02:00
|
|
|
prop->name = g_strdup(key);
|
2017-02-22 16:01:30 +01:00
|
|
|
|
2016-12-18 14:22:26 -05:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-09-19 16:24:52 -04:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static virJSONValue *
|
|
|
|
qemuMonitorJSONMakeCPUModel(virCPUDef *cpu,
|
2019-09-19 16:24:52 -04:00
|
|
|
bool migratable)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *model = virJSONValueNewObject();
|
|
|
|
virJSONValue *props = NULL;
|
2019-09-19 16:24:55 -04:00
|
|
|
size_t i;
|
2019-09-19 16:24:52 -04:00
|
|
|
|
2019-09-19 16:24:54 -04:00
|
|
|
if (virJSONValueObjectAppendString(model, "name", cpu->model) < 0)
|
2019-09-19 16:24:52 -04:00
|
|
|
goto error;
|
|
|
|
|
2019-09-19 16:24:55 -04:00
|
|
|
if (cpu->nfeatures || !migratable) {
|
2020-03-04 10:04:33 +01:00
|
|
|
props = virJSONValueNewObject();
|
2019-09-19 16:24:55 -04:00
|
|
|
|
|
|
|
for (i = 0; i < cpu->nfeatures; i++) {
|
|
|
|
char *name = cpu->features[i].name;
|
|
|
|
bool enabled = false;
|
|
|
|
|
|
|
|
/* policy may be reported as -1 if the CPU def is a host model */
|
|
|
|
if (cpu->features[i].policy == VIR_CPU_FEATURE_REQUIRE ||
|
|
|
|
cpu->features[i].policy == VIR_CPU_FEATURE_FORCE ||
|
|
|
|
cpu->features[i].policy == -1)
|
|
|
|
enabled = true;
|
|
|
|
|
|
|
|
if (virJSONValueObjectAppendBoolean(props, name, enabled) < 0)
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!migratable &&
|
|
|
|
virJSONValueObjectAppendBoolean(props, "migratable", false) < 0) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2021-02-11 17:57:45 +01:00
|
|
|
if (virJSONValueObjectAppend(model, "props", &props) < 0)
|
2019-09-19 16:24:52 -04:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
return model;
|
|
|
|
|
|
|
|
error:
|
|
|
|
virJSONValueFree(model);
|
|
|
|
virJSONValueFree(props);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONParseCPUModelData(virJSONValue *data,
|
2019-09-19 16:24:57 -04:00
|
|
|
const char *cmd_name,
|
2019-09-19 16:24:56 -04:00
|
|
|
bool fail_no_props,
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue **cpu_model,
|
|
|
|
virJSONValue **cpu_props,
|
2019-09-19 16:24:52 -04:00
|
|
|
const char **cpu_name)
|
|
|
|
{
|
|
|
|
if (!(*cpu_model = virJSONValueObjectGetObject(data, "model"))) {
|
2019-09-19 16:24:57 -04:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("%s reply data was missing 'model'"), cmd_name);
|
2019-09-19 16:24:52 -04:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(*cpu_name = virJSONValueObjectGetString(*cpu_model, "name"))) {
|
2019-09-19 16:24:57 -04:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("%s reply data was missing 'name'"), cmd_name);
|
2019-09-19 16:24:52 -04:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2019-09-19 16:24:56 -04:00
|
|
|
if (!(*cpu_props = virJSONValueObjectGetObject(*cpu_model, "props")) &&
|
|
|
|
fail_no_props) {
|
2019-09-19 16:24:57 -04:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("%s reply data was missing 'props'"), cmd_name);
|
2019-09-19 16:24:52 -04:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
qemuMonitorJSONParseCPUModel(const char *cpu_name,
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cpu_props,
|
|
|
|
qemuMonitorCPUModelInfo **model_info)
|
2019-09-19 16:24:52 -04:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorCPUModelInfo *machine_model = NULL;
|
2019-09-19 16:24:52 -04:00
|
|
|
int ret = -1;
|
|
|
|
|
2020-10-05 12:26:34 +02:00
|
|
|
machine_model = g_new0(qemuMonitorCPUModelInfo, 1);
|
2019-10-20 13:49:46 +02:00
|
|
|
machine_model->name = g_strdup(cpu_name);
|
2019-09-19 16:24:52 -04:00
|
|
|
|
2019-09-19 16:24:56 -04:00
|
|
|
if (cpu_props) {
|
|
|
|
size_t nprops = virJSONValueObjectKeysNumber(cpu_props);
|
2019-09-19 16:24:52 -04:00
|
|
|
|
2020-10-05 12:26:34 +02:00
|
|
|
machine_model->props = g_new0(qemuMonitorCPUProperty, nprops);
|
2019-09-19 16:24:56 -04:00
|
|
|
|
|
|
|
if (virJSONValueObjectForeachKeyValue(cpu_props,
|
|
|
|
qemuMonitorJSONParseCPUModelProperty,
|
|
|
|
machine_model) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2019-09-19 16:24:52 -04:00
|
|
|
|
2019-10-16 13:43:18 +02:00
|
|
|
*model_info = g_steal_pointer(&machine_model);
|
2019-09-19 16:24:52 -04:00
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
qemuMonitorCPUModelInfoFree(machine_model);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-12-18 14:22:26 -05:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetCPUModelExpansion(qemuMonitor *mon,
|
2017-01-31 13:44:00 +01:00
|
|
|
qemuMonitorCPUModelExpansionType type,
|
2021-03-11 08:16:13 +01:00
|
|
|
virCPUDef *cpu,
|
2017-03-29 10:58:41 +02:00
|
|
|
bool migratable,
|
2019-09-19 16:24:56 -04:00
|
|
|
bool fail_no_props,
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorCPUModelInfo **model_info)
|
2016-12-18 14:22:26 -05:00
|
|
|
{
|
2019-10-15 14:47:50 +02:00
|
|
|
g_autoptr(virJSONValue) model = NULL;
|
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *data;
|
|
|
|
virJSONValue *cpu_model;
|
|
|
|
virJSONValue *cpu_props = NULL;
|
2019-09-19 16:24:53 -04:00
|
|
|
const char *cpu_name = "";
|
2017-01-31 13:44:00 +01:00
|
|
|
const char *typeStr = "";
|
2016-12-18 14:22:26 -05:00
|
|
|
|
|
|
|
*model_info = NULL;
|
|
|
|
|
2019-09-19 16:24:54 -04:00
|
|
|
if (!(model = qemuMonitorJSONMakeCPUModel(cpu, migratable)))
|
2019-09-19 16:24:53 -04:00
|
|
|
return -1;
|
2016-12-18 14:22:26 -05:00
|
|
|
|
2017-02-23 13:53:51 +01:00
|
|
|
retry:
|
2017-01-31 13:44:00 +01:00
|
|
|
switch (type) {
|
|
|
|
case QEMU_MONITOR_CPU_MODEL_EXPANSION_STATIC:
|
2017-02-23 13:53:51 +01:00
|
|
|
case QEMU_MONITOR_CPU_MODEL_EXPANSION_STATIC_FULL:
|
2017-01-31 13:44:00 +01:00
|
|
|
typeStr = "static";
|
|
|
|
break;
|
2017-02-23 13:53:51 +01:00
|
|
|
|
|
|
|
case QEMU_MONITOR_CPU_MODEL_EXPANSION_FULL:
|
|
|
|
typeStr = "full";
|
|
|
|
break;
|
2017-01-31 13:44:00 +01:00
|
|
|
}
|
|
|
|
|
2016-12-18 14:22:26 -05:00
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-cpu-model-expansion",
|
2017-01-31 13:44:00 +01:00
|
|
|
"s:type", typeStr,
|
2018-03-30 11:12:57 +02:00
|
|
|
"a:model", &model,
|
2016-12-18 14:22:26 -05:00
|
|
|
NULL)))
|
2019-09-19 16:24:53 -04:00
|
|
|
return -1;
|
2016-12-18 14:22:26 -05:00
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2019-09-19 16:24:53 -04:00
|
|
|
return -1;
|
2016-12-18 14:22:26 -05:00
|
|
|
|
2017-01-12 11:18:11 -05:00
|
|
|
/* Even though query-cpu-model-expansion is advertised by query-commands it
|
|
|
|
* may just return GenericError if it is not implemented for the requested
|
|
|
|
* guest architecture or it is not supported in the host environment.
|
|
|
|
*/
|
2019-09-19 16:24:53 -04:00
|
|
|
if (qemuMonitorJSONHasError(reply, "GenericError"))
|
|
|
|
return 0;
|
2017-01-12 11:18:11 -05:00
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_OBJECT) < 0)
|
2019-09-19 16:24:53 -04:00
|
|
|
return -1;
|
2016-12-18 14:22:26 -05:00
|
|
|
|
|
|
|
data = virJSONValueObjectGetObject(reply, "return");
|
|
|
|
|
2019-09-19 16:24:57 -04:00
|
|
|
if (qemuMonitorJSONParseCPUModelData(data, "query-cpu-model-expansion",
|
2019-09-19 16:24:56 -04:00
|
|
|
fail_no_props, &cpu_model, &cpu_props,
|
|
|
|
&cpu_name) < 0)
|
2019-09-19 16:24:53 -04:00
|
|
|
return -1;
|
2016-12-18 14:22:26 -05:00
|
|
|
|
2017-02-23 13:53:51 +01:00
|
|
|
/* QEMU_MONITOR_CPU_MODEL_EXPANSION_STATIC_FULL requests "full" expansion
|
|
|
|
* on the result of the initial "static" expansion.
|
|
|
|
*/
|
|
|
|
if (type == QEMU_MONITOR_CPU_MODEL_EXPANSION_STATIC_FULL) {
|
|
|
|
if (!(model = virJSONValueCopy(cpu_model)))
|
2019-09-19 16:24:53 -04:00
|
|
|
return -1;
|
2017-02-23 13:53:51 +01:00
|
|
|
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
type = QEMU_MONITOR_CPU_MODEL_EXPANSION_FULL;
|
|
|
|
goto retry;
|
|
|
|
}
|
|
|
|
|
2019-09-19 16:24:53 -04:00
|
|
|
return qemuMonitorJSONParseCPUModel(cpu_name, cpu_props, model_info);
|
2016-12-18 14:22:26 -05:00
|
|
|
}
|
|
|
|
|
2012-08-22 10:25:20 +01:00
|
|
|
|
2019-09-19 16:24:58 -04:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetCPUModelBaseline(qemuMonitor *mon,
|
|
|
|
virCPUDef *cpu_a,
|
|
|
|
virCPUDef *cpu_b,
|
|
|
|
qemuMonitorCPUModelInfo **baseline)
|
2019-09-19 16:24:58 -04:00
|
|
|
{
|
2019-10-15 14:47:50 +02:00
|
|
|
g_autoptr(virJSONValue) model_a = NULL;
|
|
|
|
g_autoptr(virJSONValue) model_b = NULL;
|
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *data;
|
|
|
|
virJSONValue *cpu_model;
|
|
|
|
virJSONValue *cpu_props = NULL;
|
2019-09-19 16:24:58 -04:00
|
|
|
const char *cpu_name = "";
|
|
|
|
|
|
|
|
if (!(model_a = qemuMonitorJSONMakeCPUModel(cpu_a, true)) ||
|
|
|
|
!(model_b = qemuMonitorJSONMakeCPUModel(cpu_b, true)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-cpu-model-baseline",
|
|
|
|
"a:modela", &model_a,
|
|
|
|
"a:modelb", &model_b,
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_OBJECT) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
data = virJSONValueObjectGetObject(reply, "return");
|
|
|
|
|
|
|
|
if (qemuMonitorJSONParseCPUModelData(data, "query-cpu-model-baseline",
|
|
|
|
false, &cpu_model, &cpu_props,
|
|
|
|
&cpu_name) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return qemuMonitorJSONParseCPUModel(cpu_name, cpu_props, baseline);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-09-19 16:25:02 -04:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetCPUModelComparison(qemuMonitor *mon,
|
|
|
|
virCPUDef *cpu_a,
|
|
|
|
virCPUDef *cpu_b,
|
2019-09-19 16:25:02 -04:00
|
|
|
char **result)
|
|
|
|
{
|
2019-10-15 14:47:50 +02:00
|
|
|
g_autoptr(virJSONValue) model_a = NULL;
|
|
|
|
g_autoptr(virJSONValue) model_b = NULL;
|
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2019-09-19 16:25:02 -04:00
|
|
|
const char *data_result;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *data;
|
2019-09-19 16:25:02 -04:00
|
|
|
|
|
|
|
if (!(model_a = qemuMonitorJSONMakeCPUModel(cpu_a, true)) ||
|
|
|
|
!(model_b = qemuMonitorJSONMakeCPUModel(cpu_b, true)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-cpu-model-comparison",
|
|
|
|
"a:modela", &model_a,
|
|
|
|
"a:modelb", &model_b,
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
data = virJSONValueObjectGetObject(reply, "return");
|
|
|
|
|
|
|
|
if (!(data_result = virJSONValueObjectGetString(data, "result"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-cpu-model-comparison reply data was missing "
|
|
|
|
"'result'"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2019-10-20 12:55:05 +02:00
|
|
|
*result = g_strdup(data_result);
|
|
|
|
return 0;
|
2019-09-19 16:25:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONGetCommands(qemuMonitor *mon,
|
2012-08-22 10:25:20 +01:00
|
|
|
char ***commands)
|
|
|
|
{
|
2021-06-14 16:19:11 +02:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *data;
|
2021-06-14 16:19:11 +02:00
|
|
|
g_auto(GStrv) commandlist = NULL;
|
2018-04-19 17:29:02 -04:00
|
|
|
size_t n = 0;
|
2012-08-22 10:25:20 +01:00
|
|
|
size_t i;
|
|
|
|
|
|
|
|
*commands = NULL;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-commands", NULL)))
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2021-06-14 16:19:11 +02:00
|
|
|
return -1;
|
2012-08-22 10:25:20 +01:00
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_ARRAY) < 0)
|
2021-06-14 16:19:11 +02:00
|
|
|
return -1;
|
2012-08-22 10:25:20 +01:00
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
data = virJSONValueObjectGetArray(reply, "return");
|
|
|
|
n = virJSONValueArraySize(data);
|
2012-08-22 10:25:20 +01:00
|
|
|
|
2013-04-26 11:47:23 -06:00
|
|
|
/* null-terminated list */
|
2020-10-05 12:26:34 +02:00
|
|
|
commandlist = g_new0(char *, n + 1);
|
2012-08-22 10:25:20 +01:00
|
|
|
|
2013-05-21 15:21:20 +08:00
|
|
|
for (i = 0; i < n; i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *child = virJSONValueArrayGet(data, i);
|
2012-08-22 10:25:20 +01:00
|
|
|
const char *tmp;
|
|
|
|
|
|
|
|
if (!(tmp = virJSONValueObjectGetString(child, "name"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-commands reply data was missing 'name'"));
|
2021-06-14 16:19:11 +02:00
|
|
|
return -1;
|
2012-08-22 10:25:20 +01:00
|
|
|
}
|
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
commandlist[i] = g_strdup(tmp);
|
2012-08-22 10:25:20 +01:00
|
|
|
}
|
|
|
|
|
2021-02-23 14:58:29 +01:00
|
|
|
*commands = g_steal_pointer(&commandlist);
|
2021-06-14 16:19:11 +02:00
|
|
|
return n;
|
2012-08-22 10:25:20 +01:00
|
|
|
}
|
2012-08-22 10:48:41 +01:00
|
|
|
|
|
|
|
|
2020-11-30 17:54:25 +01:00
|
|
|
static int
|
|
|
|
qemuMonitorJSONGetCommandLineOptionsWorker(size_t pos G_GNUC_UNUSED,
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *item,
|
2020-11-30 17:54:25 +01:00
|
|
|
void *opaque)
|
|
|
|
{
|
|
|
|
const char *name = virJSONValueObjectGetString(item, "option");
|
|
|
|
g_autoptr(virJSONValue) parameters = NULL;
|
|
|
|
GHashTable *options = opaque;
|
|
|
|
|
|
|
|
if (!name ||
|
|
|
|
virJSONValueObjectRemoveKey(item, "parameters", ¶meters) <= 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("reply data was missing 'option' name or parameters"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_hash_table_insert(options, g_strdup(name), parameters);
|
|
|
|
parameters = NULL;
|
|
|
|
|
2020-12-07 09:32:27 +01:00
|
|
|
return 1;
|
2020-11-30 17:54:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
GHashTable *
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetCommandLineOptions(qemuMonitor *mon)
|
2020-11-30 17:54:25 +01:00
|
|
|
{
|
|
|
|
g_autoptr(GHashTable) ret = virHashNew(virJSONValueHashFree);
|
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-command-line-options", NULL)))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_ARRAY) < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (virJSONValueArrayForeachSteal(virJSONValueObjectGetArray(reply, "return"),
|
|
|
|
qemuMonitorJSONGetCommandLineOptionsWorker,
|
|
|
|
ret) < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
return g_steal_pointer(&ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONGetKVMState(qemuMonitor *mon,
|
Make non-KVM machines work with QMP probing
When there is no 'qemu-kvm' binary and the emulator used for a machine
is, for example, 'qemu-system-x86_64' that, by default, runs without
kvm enabled, libvirt still supplies '-no-kvm' option to this process,
even though it does not recognize such option (making the start of a
domain fail in that case).
This patch fixes building a command-line for QEMU machines without KVM
acceleration and is based on following assumptions:
- QEMU_CAPS_KVM flag means that QEMU is running KVM accelerated
machines by default (without explicitly requesting that using a
command-line option). It is the closest to the truth according to
the code with the only exception being the comment next to the
flag, so it's fixed in this patch as well.
- QEMU_CAPS_ENABLE_KVM flag means that QEMU is, by default, running
without KVM acceleration and in case we need KVM acceleration it
needs to be explicitly instructed to do so. This is partially
true for the past (this option essentially means that QEMU
recognizes the '-enable-kvm' option, even though it's almost the
same).
2012-10-31 08:31:49 +01:00
|
|
|
bool *enabled,
|
|
|
|
bool *present)
|
|
|
|
{
|
2016-05-04 10:28:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = NULL;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *data = NULL;
|
Make non-KVM machines work with QMP probing
When there is no 'qemu-kvm' binary and the emulator used for a machine
is, for example, 'qemu-system-x86_64' that, by default, runs without
kvm enabled, libvirt still supplies '-no-kvm' option to this process,
even though it does not recognize such option (making the start of a
domain fail in that case).
This patch fixes building a command-line for QEMU machines without KVM
acceleration and is based on following assumptions:
- QEMU_CAPS_KVM flag means that QEMU is running KVM accelerated
machines by default (without explicitly requesting that using a
command-line option). It is the closest to the truth according to
the code with the only exception being the comment next to the
flag, so it's fixed in this patch as well.
- QEMU_CAPS_ENABLE_KVM flag means that QEMU is, by default, running
without KVM acceleration and in case we need KVM acceleration it
needs to be explicitly instructed to do so. This is partially
true for the past (this option essentially means that QEMU
recognizes the '-enable-kvm' option, even though it's almost the
same).
2012-10-31 08:31:49 +01:00
|
|
|
|
|
|
|
/* Safe defaults */
|
|
|
|
*enabled = *present = false;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-kvm", NULL)))
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
Make non-KVM machines work with QMP probing
When there is no 'qemu-kvm' binary and the emulator used for a machine
is, for example, 'qemu-system-x86_64' that, by default, runs without
kvm enabled, libvirt still supplies '-no-kvm' option to this process,
even though it does not recognize such option (making the start of a
domain fail in that case).
This patch fixes building a command-line for QEMU machines without KVM
acceleration and is based on following assumptions:
- QEMU_CAPS_KVM flag means that QEMU is running KVM accelerated
machines by default (without explicitly requesting that using a
command-line option). It is the closest to the truth according to
the code with the only exception being the comment next to the
flag, so it's fixed in this patch as well.
- QEMU_CAPS_ENABLE_KVM flag means that QEMU is, by default, running
without KVM acceleration and in case we need KVM acceleration it
needs to be explicitly instructed to do so. This is partially
true for the past (this option essentially means that QEMU
recognizes the '-enable-kvm' option, even though it's almost the
same).
2012-10-31 08:31:49 +01:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONHasError(reply, "CommandNotFound")) {
|
|
|
|
ret = 0;
|
|
|
|
goto cleanup;
|
Make non-KVM machines work with QMP probing
When there is no 'qemu-kvm' binary and the emulator used for a machine
is, for example, 'qemu-system-x86_64' that, by default, runs without
kvm enabled, libvirt still supplies '-no-kvm' option to this process,
even though it does not recognize such option (making the start of a
domain fail in that case).
This patch fixes building a command-line for QEMU machines without KVM
acceleration and is based on following assumptions:
- QEMU_CAPS_KVM flag means that QEMU is running KVM accelerated
machines by default (without explicitly requesting that using a
command-line option). It is the closest to the truth according to
the code with the only exception being the comment next to the
flag, so it's fixed in this patch as well.
- QEMU_CAPS_ENABLE_KVM flag means that QEMU is, by default, running
without KVM acceleration and in case we need KVM acceleration it
needs to be explicitly instructed to do so. This is partially
true for the past (this option essentially means that QEMU
recognizes the '-enable-kvm' option, even though it's almost the
same).
2012-10-31 08:31:49 +01:00
|
|
|
}
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_OBJECT) < 0)
|
Make non-KVM machines work with QMP probing
When there is no 'qemu-kvm' binary and the emulator used for a machine
is, for example, 'qemu-system-x86_64' that, by default, runs without
kvm enabled, libvirt still supplies '-no-kvm' option to this process,
even though it does not recognize such option (making the start of a
domain fail in that case).
This patch fixes building a command-line for QEMU machines without KVM
acceleration and is based on following assumptions:
- QEMU_CAPS_KVM flag means that QEMU is running KVM accelerated
machines by default (without explicitly requesting that using a
command-line option). It is the closest to the truth according to
the code with the only exception being the comment next to the
flag, so it's fixed in this patch as well.
- QEMU_CAPS_ENABLE_KVM flag means that QEMU is, by default, running
without KVM acceleration and in case we need KVM acceleration it
needs to be explicitly instructed to do so. This is partially
true for the past (this option essentially means that QEMU
recognizes the '-enable-kvm' option, even though it's almost the
same).
2012-10-31 08:31:49 +01:00
|
|
|
goto cleanup;
|
|
|
|
|
2016-11-21 00:10:06 +01:00
|
|
|
data = virJSONValueObjectGetObject(reply, "return");
|
Make non-KVM machines work with QMP probing
When there is no 'qemu-kvm' binary and the emulator used for a machine
is, for example, 'qemu-system-x86_64' that, by default, runs without
kvm enabled, libvirt still supplies '-no-kvm' option to this process,
even though it does not recognize such option (making the start of a
domain fail in that case).
This patch fixes building a command-line for QEMU machines without KVM
acceleration and is based on following assumptions:
- QEMU_CAPS_KVM flag means that QEMU is running KVM accelerated
machines by default (without explicitly requesting that using a
command-line option). It is the closest to the truth according to
the code with the only exception being the comment next to the
flag, so it's fixed in this patch as well.
- QEMU_CAPS_ENABLE_KVM flag means that QEMU is, by default, running
without KVM acceleration and in case we need KVM acceleration it
needs to be explicitly instructed to do so. This is partially
true for the past (this option essentially means that QEMU
recognizes the '-enable-kvm' option, even though it's almost the
same).
2012-10-31 08:31:49 +01:00
|
|
|
|
|
|
|
if (virJSONValueObjectGetBoolean(data, "enabled", enabled) < 0 ||
|
|
|
|
virJSONValueObjectGetBoolean(data, "present", present) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-kvm replied unexpected data"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
Make non-KVM machines work with QMP probing
When there is no 'qemu-kvm' binary and the emulator used for a machine
is, for example, 'qemu-system-x86_64' that, by default, runs without
kvm enabled, libvirt still supplies '-no-kvm' option to this process,
even though it does not recognize such option (making the start of a
domain fail in that case).
This patch fixes building a command-line for QEMU machines without KVM
acceleration and is based on following assumptions:
- QEMU_CAPS_KVM flag means that QEMU is running KVM accelerated
machines by default (without explicitly requesting that using a
command-line option). It is the closest to the truth according to
the code with the only exception being the comment next to the
flag, so it's fixed in this patch as well.
- QEMU_CAPS_ENABLE_KVM flag means that QEMU is, by default, running
without KVM acceleration and in case we need KVM acceleration it
needs to be explicitly instructed to do so. This is partially
true for the past (this option essentially means that QEMU
recognizes the '-enable-kvm' option, even though it's almost the
same).
2012-10-31 08:31:49 +01:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-06-14 16:20:50 +02:00
|
|
|
int
|
|
|
|
qemuMonitorJSONGetObjectTypes(qemuMonitor *mon,
|
|
|
|
char ***types)
|
2012-08-22 10:48:41 +01:00
|
|
|
{
|
2021-06-14 16:20:50 +02:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *data;
|
2021-06-14 16:20:50 +02:00
|
|
|
g_auto(GStrv) typelist = NULL;
|
2018-04-19 17:29:02 -04:00
|
|
|
size_t n = 0;
|
2012-08-22 10:48:41 +01:00
|
|
|
size_t i;
|
|
|
|
|
|
|
|
*types = NULL;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("qom-list-types", NULL)))
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2021-06-14 16:20:50 +02:00
|
|
|
return -1;
|
2012-08-22 10:48:41 +01:00
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_ARRAY) < 0)
|
2021-06-14 16:20:50 +02:00
|
|
|
return -1;
|
2012-08-22 10:48:41 +01:00
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
data = virJSONValueObjectGetArray(reply, "return");
|
|
|
|
n = virJSONValueArraySize(data);
|
2012-08-22 10:48:41 +01:00
|
|
|
|
2013-04-26 11:47:23 -06:00
|
|
|
/* null-terminated list */
|
2020-10-05 12:26:34 +02:00
|
|
|
typelist = g_new0(char *, n + 1);
|
2012-08-22 10:48:41 +01:00
|
|
|
|
2013-05-21 15:21:20 +08:00
|
|
|
for (i = 0; i < n; i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *child = virJSONValueArrayGet(data, i);
|
2012-08-22 10:48:41 +01:00
|
|
|
const char *tmp;
|
|
|
|
|
|
|
|
if (!(tmp = virJSONValueObjectGetString(child, "name"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("qom-list-types reply data was missing 'name'"));
|
2021-06-14 16:20:50 +02:00
|
|
|
return -1;
|
2012-08-22 10:48:41 +01:00
|
|
|
}
|
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
typelist[i] = g_strdup(tmp);
|
2012-08-22 10:48:41 +01:00
|
|
|
}
|
|
|
|
|
2021-02-23 14:58:29 +01:00
|
|
|
*types = g_steal_pointer(&typelist);
|
2021-06-14 16:20:50 +02:00
|
|
|
return n;
|
2012-08-22 10:48:41 +01:00
|
|
|
}
|
2012-08-22 10:48:41 +01:00
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONGetObjectListPaths(qemuMonitor *mon,
|
2013-06-24 13:51:56 -04:00
|
|
|
const char *path,
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONListPath ***paths)
|
2013-06-24 13:51:56 -04:00
|
|
|
{
|
2016-05-04 10:28:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *data;
|
|
|
|
qemuMonitorJSONListPath **pathlist = NULL;
|
2018-04-19 17:29:02 -04:00
|
|
|
size_t n = 0;
|
2013-06-24 13:51:56 -04:00
|
|
|
size_t i;
|
|
|
|
|
|
|
|
*paths = NULL;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("qom-list",
|
|
|
|
"s:path", path,
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2013-06-24 13:51:56 -04:00
|
|
|
goto cleanup;
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_ARRAY) < 0)
|
2016-05-04 10:28:43 +02:00
|
|
|
goto cleanup;
|
2013-06-24 13:51:56 -04:00
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
data = virJSONValueObjectGetArray(reply, "return");
|
|
|
|
n = virJSONValueArraySize(data);
|
2013-06-24 13:51:56 -04:00
|
|
|
|
|
|
|
/* null-terminated list */
|
2021-03-11 08:16:13 +01:00
|
|
|
pathlist = g_new0(qemuMonitorJSONListPath *, n + 1);
|
2013-06-24 13:51:56 -04:00
|
|
|
|
|
|
|
for (i = 0; i < n; i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *child = virJSONValueArrayGet(data, i);
|
2013-06-24 13:51:56 -04:00
|
|
|
const char *tmp;
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONListPath *info;
|
2013-06-24 13:51:56 -04:00
|
|
|
|
2020-10-05 12:26:34 +02:00
|
|
|
info = g_new0(qemuMonitorJSONListPath, 1);
|
2013-06-24 13:51:56 -04:00
|
|
|
|
|
|
|
pathlist[i] = info;
|
|
|
|
|
|
|
|
if (!(tmp = virJSONValueObjectGetString(child, "name"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("qom-list reply data was missing 'name'"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
info->name = g_strdup(tmp);
|
2013-06-24 13:51:56 -04:00
|
|
|
|
|
|
|
if (virJSONValueObjectHasKey(child, "type")) {
|
|
|
|
if (!(tmp = virJSONValueObjectGetString(child, "type"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("qom-list reply has malformed 'type' data"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2019-10-20 13:49:46 +02:00
|
|
|
info->type = g_strdup(tmp);
|
2013-06-24 13:51:56 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = n;
|
2021-02-23 14:58:29 +01:00
|
|
|
*paths = g_steal_pointer(&pathlist);
|
2013-06-24 13:51:56 -04:00
|
|
|
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2016-05-16 07:47:26 -04:00
|
|
|
if (pathlist) {
|
2013-06-24 13:51:56 -04:00
|
|
|
for (i = 0; i < n; i++)
|
|
|
|
qemuMonitorJSONListPathFree(pathlist[i]);
|
|
|
|
VIR_FREE(pathlist);
|
|
|
|
}
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
void qemuMonitorJSONListPathFree(qemuMonitorJSONListPath *paths)
|
2013-06-24 13:51:56 -04:00
|
|
|
{
|
|
|
|
if (!paths)
|
|
|
|
return;
|
2021-02-03 14:36:01 -05:00
|
|
|
g_free(paths->name);
|
|
|
|
g_free(paths->type);
|
|
|
|
g_free(paths);
|
2013-06-24 13:51:56 -04:00
|
|
|
}
|
|
|
|
|
2013-07-03 14:15:07 -04:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONGetObjectProperty(qemuMonitor *mon,
|
2013-07-03 14:15:07 -04:00
|
|
|
const char *path,
|
|
|
|
const char *property,
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONObjectProperty *prop)
|
2013-07-03 14:15:07 -04:00
|
|
|
{
|
2016-05-04 10:28:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *data;
|
2013-07-03 14:15:07 -04:00
|
|
|
const char *tmp;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("qom-get",
|
|
|
|
"s:path", path,
|
|
|
|
"s:property", property,
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2013-07-03 14:15:07 -04:00
|
|
|
goto cleanup;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2013-07-03 14:15:07 -04:00
|
|
|
|
2016-05-03 10:55:41 +02:00
|
|
|
data = virJSONValueObjectGet(reply, "return");
|
2013-07-03 14:15:07 -04:00
|
|
|
|
|
|
|
switch ((qemuMonitorJSONObjectPropertyType) prop->type) {
|
|
|
|
/* Simple cases of boolean, int, long, uint, ulong, double, and string
|
|
|
|
* will receive return value as part of {"return": xxx} statement
|
|
|
|
*/
|
|
|
|
case QEMU_MONITOR_OBJECT_PROPERTY_BOOLEAN:
|
|
|
|
ret = virJSONValueGetBoolean(data, &prop->val.b);
|
|
|
|
break;
|
|
|
|
case QEMU_MONITOR_OBJECT_PROPERTY_INT:
|
|
|
|
ret = virJSONValueGetNumberInt(data, &prop->val.iv);
|
|
|
|
break;
|
|
|
|
case QEMU_MONITOR_OBJECT_PROPERTY_LONG:
|
|
|
|
ret = virJSONValueGetNumberLong(data, &prop->val.l);
|
|
|
|
break;
|
|
|
|
case QEMU_MONITOR_OBJECT_PROPERTY_UINT:
|
|
|
|
ret = virJSONValueGetNumberUint(data, &prop->val.ui);
|
|
|
|
break;
|
|
|
|
case QEMU_MONITOR_OBJECT_PROPERTY_ULONG:
|
|
|
|
ret = virJSONValueGetNumberUlong(data, &prop->val.ul);
|
|
|
|
break;
|
|
|
|
case QEMU_MONITOR_OBJECT_PROPERTY_DOUBLE:
|
|
|
|
ret = virJSONValueGetNumberDouble(data, &prop->val.d);
|
|
|
|
break;
|
|
|
|
case QEMU_MONITOR_OBJECT_PROPERTY_STRING:
|
|
|
|
tmp = virJSONValueGetString(data);
|
2019-10-20 13:49:46 +02:00
|
|
|
if (tmp)
|
|
|
|
prop->val.str = g_strdup(tmp);
|
2013-07-03 14:15:07 -04:00
|
|
|
if (tmp)
|
|
|
|
ret = 0;
|
|
|
|
break;
|
|
|
|
case QEMU_MONITOR_OBJECT_PROPERTY_LAST:
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("qom-get invalid object property type %d"),
|
|
|
|
prop->type);
|
|
|
|
goto cleanup;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ret == -1) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("qom-get reply was missing return data"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2013-07-03 14:15:07 -04:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-06-17 23:36:53 +02:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetStringListProperty(qemuMonitor *mon,
|
2019-06-17 23:36:53 +02:00
|
|
|
const char *path,
|
|
|
|
const char *property,
|
|
|
|
char ***strList)
|
|
|
|
{
|
2019-10-15 14:47:50 +02:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2019-06-17 23:36:53 +02:00
|
|
|
|
|
|
|
*strList = NULL;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("qom-get",
|
|
|
|
"s:path", path,
|
|
|
|
"s:property", property,
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_ARRAY) < 0)
|
|
|
|
return -1;
|
|
|
|
|
2020-12-01 11:47:30 +01:00
|
|
|
if (!(*strList = virJSONValueObjectGetStringArray(reply, "return")))
|
|
|
|
return -1;
|
2019-06-17 23:36:53 +02:00
|
|
|
|
2021-06-14 16:22:30 +02:00
|
|
|
return 0;
|
2019-06-17 23:36:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-11-03 13:09:47 +01:00
|
|
|
#define MAKE_SET_CMD(STRING, VALUE) \
|
|
|
|
cmd = qemuMonitorJSONMakeCommand("qom-set", \
|
|
|
|
"s:path", path, \
|
|
|
|
"s:property", property, \
|
|
|
|
STRING, VALUE, \
|
2013-07-03 14:42:40 -04:00
|
|
|
NULL)
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONSetObjectProperty(qemuMonitor *mon,
|
2013-07-03 14:42:40 -04:00
|
|
|
const char *path,
|
|
|
|
const char *property,
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONObjectProperty *prop)
|
2013-07-03 14:42:40 -04:00
|
|
|
{
|
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = NULL;
|
|
|
|
virJSONValue *reply = NULL;
|
2013-07-03 14:42:40 -04:00
|
|
|
|
|
|
|
switch ((qemuMonitorJSONObjectPropertyType) prop->type) {
|
|
|
|
/* Simple cases of boolean, int, long, uint, ulong, double, and string
|
|
|
|
* will receive return value as part of {"return": xxx} statement
|
|
|
|
*/
|
|
|
|
case QEMU_MONITOR_OBJECT_PROPERTY_BOOLEAN:
|
|
|
|
MAKE_SET_CMD("b:value", prop->val.b);
|
|
|
|
break;
|
|
|
|
case QEMU_MONITOR_OBJECT_PROPERTY_INT:
|
|
|
|
MAKE_SET_CMD("i:value", prop->val.iv);
|
|
|
|
break;
|
|
|
|
case QEMU_MONITOR_OBJECT_PROPERTY_LONG:
|
|
|
|
MAKE_SET_CMD("I:value", prop->val.l);
|
|
|
|
break;
|
|
|
|
case QEMU_MONITOR_OBJECT_PROPERTY_UINT:
|
|
|
|
MAKE_SET_CMD("u:value", prop->val.ui);
|
|
|
|
break;
|
|
|
|
case QEMU_MONITOR_OBJECT_PROPERTY_ULONG:
|
|
|
|
MAKE_SET_CMD("U:value", prop->val.ul);
|
|
|
|
break;
|
|
|
|
case QEMU_MONITOR_OBJECT_PROPERTY_DOUBLE:
|
|
|
|
MAKE_SET_CMD("d:value", prop->val.d);
|
|
|
|
break;
|
|
|
|
case QEMU_MONITOR_OBJECT_PROPERTY_STRING:
|
|
|
|
MAKE_SET_CMD("s:value", prop->val.str);
|
|
|
|
break;
|
|
|
|
case QEMU_MONITOR_OBJECT_PROPERTY_LAST:
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("qom-set invalid object property type %d"),
|
|
|
|
prop->type);
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
}
|
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2013-07-03 14:42:40 -04:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2013-07-03 14:42:40 -04:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
#undef MAKE_SET_CMD
|
|
|
|
|
|
|
|
|
2018-04-20 09:20:36 +02:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONParsePropsList(virJSONValue *cmd,
|
|
|
|
virJSONValue *reply,
|
2019-06-17 21:51:20 +02:00
|
|
|
const char *type,
|
2018-04-20 09:20:36 +02:00
|
|
|
char ***props)
|
2012-08-22 10:48:41 +01:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *data;
|
2021-06-14 10:24:25 +02:00
|
|
|
g_auto(GStrv) proplist = NULL;
|
2018-04-19 17:29:02 -04:00
|
|
|
size_t n = 0;
|
2019-06-17 21:51:20 +02:00
|
|
|
size_t count = 0;
|
2012-08-22 10:48:41 +01:00
|
|
|
size_t i;
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_ARRAY) < 0)
|
2021-06-14 10:24:25 +02:00
|
|
|
return -1;
|
2012-08-22 10:48:41 +01:00
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
data = virJSONValueObjectGetArray(reply, "return");
|
|
|
|
n = virJSONValueArraySize(data);
|
2012-08-22 10:48:41 +01:00
|
|
|
|
2013-04-26 11:47:23 -06:00
|
|
|
/* null-terminated list */
|
2020-10-05 12:26:34 +02:00
|
|
|
proplist = g_new0(char *, n + 1);
|
2012-08-22 10:48:41 +01:00
|
|
|
|
2013-05-21 15:21:20 +08:00
|
|
|
for (i = 0; i < n; i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *child = virJSONValueArrayGet(data, i);
|
2012-08-22 10:48:41 +01:00
|
|
|
const char *tmp;
|
|
|
|
|
2019-06-17 21:51:20 +02:00
|
|
|
if (type &&
|
|
|
|
STRNEQ_NULLABLE(virJSONValueObjectGetString(child, "type"), type))
|
|
|
|
continue;
|
|
|
|
|
2012-08-22 10:48:41 +01:00
|
|
|
if (!(tmp = virJSONValueObjectGetString(child, "name"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2018-04-20 09:20:36 +02:00
|
|
|
_("reply data was missing 'name'"));
|
2021-06-14 10:24:25 +02:00
|
|
|
return -1;
|
2012-08-22 10:48:41 +01:00
|
|
|
}
|
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
proplist[count++] = g_strdup(tmp);
|
2012-08-22 10:48:41 +01:00
|
|
|
}
|
|
|
|
|
2021-02-23 14:58:29 +01:00
|
|
|
*props = g_steal_pointer(&proplist);
|
2021-06-14 10:24:25 +02:00
|
|
|
return count;
|
2018-04-20 09:20:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-05-06 19:53:43 +02:00
|
|
|
static int
|
|
|
|
qemuMonitorJSONGetDevicePropsWorker(size_t pos G_GNUC_UNUSED,
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *item,
|
2020-05-06 19:53:43 +02:00
|
|
|
void *opaque)
|
|
|
|
{
|
|
|
|
const char *name = virJSONValueObjectGetString(item, "name");
|
2020-10-22 19:04:18 +02:00
|
|
|
GHashTable *devices = opaque;
|
2020-05-06 19:53:43 +02:00
|
|
|
|
|
|
|
if (!name) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("reply data was missing 'name'"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virHashAddEntry(devices, name, item) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-22 19:04:18 +02:00
|
|
|
GHashTable *
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetDeviceProps(qemuMonitor *mon,
|
2020-05-06 19:53:43 +02:00
|
|
|
const char *device)
|
2018-04-20 09:20:36 +02:00
|
|
|
{
|
2020-10-22 19:04:18 +02:00
|
|
|
g_autoptr(GHashTable) props = virHashNew(virJSONValueHashFree);
|
2020-05-06 17:40:47 +02:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2018-04-20 09:20:36 +02:00
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("device-list-properties",
|
|
|
|
"s:typename", device,
|
|
|
|
NULL)))
|
2020-05-06 19:53:43 +02:00
|
|
|
return NULL;
|
2018-04-20 09:20:36 +02:00
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2020-05-06 19:53:43 +02:00
|
|
|
return NULL;
|
2018-04-20 09:20:36 +02:00
|
|
|
|
2020-05-06 19:53:43 +02:00
|
|
|
/* return empty hash */
|
2020-05-06 17:40:47 +02:00
|
|
|
if (qemuMonitorJSONHasError(reply, "DeviceNotFound"))
|
2020-05-06 19:53:43 +02:00
|
|
|
return g_steal_pointer(&props);
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_ARRAY) < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (virJSONValueArrayForeachSteal(virJSONValueObjectGetArray(reply, "return"),
|
|
|
|
qemuMonitorJSONGetDevicePropsWorker,
|
|
|
|
props) < 0)
|
|
|
|
return NULL;
|
2018-04-20 09:20:36 +02:00
|
|
|
|
2020-05-06 19:53:43 +02:00
|
|
|
return g_steal_pointer(&props);
|
2018-04-12 15:04:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetObjectProps(qemuMonitor *mon,
|
2018-04-12 15:04:07 +02:00
|
|
|
const char *object,
|
|
|
|
char ***props)
|
|
|
|
{
|
2021-06-14 10:25:53 +02:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2018-04-12 15:04:07 +02:00
|
|
|
|
|
|
|
*props = NULL;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("qom-list-properties",
|
|
|
|
"s:typename", object,
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2021-06-14 10:25:53 +02:00
|
|
|
return -1;
|
2018-04-12 15:04:07 +02:00
|
|
|
|
2021-06-14 10:25:53 +02:00
|
|
|
if (qemuMonitorJSONHasError(reply, "DeviceNotFound"))
|
|
|
|
return 0;
|
2018-04-12 15:04:07 +02:00
|
|
|
|
2021-06-14 10:25:53 +02:00
|
|
|
return qemuMonitorJSONParsePropsList(cmd, reply, NULL, props);
|
2012-08-22 10:48:41 +01:00
|
|
|
}
|
2012-08-22 10:48:41 +01:00
|
|
|
|
|
|
|
|
|
|
|
char *
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetTargetArch(qemuMonitor *mon)
|
2012-08-22 10:48:41 +01:00
|
|
|
{
|
|
|
|
char *ret = NULL;
|
|
|
|
const char *arch;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *data;
|
2012-08-22 10:48:41 +01:00
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-target", NULL)))
|
|
|
|
return NULL;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2012-08-22 10:48:41 +01:00
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_OBJECT) < 0)
|
2012-08-22 10:48:41 +01:00
|
|
|
goto cleanup;
|
|
|
|
|
2016-11-21 00:10:06 +01:00
|
|
|
data = virJSONValueObjectGetObject(reply, "return");
|
2012-08-22 10:48:41 +01:00
|
|
|
|
|
|
|
if (!(arch = virJSONValueObjectGetString(data, "arch"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-target reply was missing arch data"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2019-10-18 13:27:03 +02:00
|
|
|
ret = g_strdup(arch);
|
2012-08-22 10:48:41 +01:00
|
|
|
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2012-08-22 10:48:41 +01:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2013-01-14 12:45:20 +01:00
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetMigrationCapabilities(qemuMonitor *mon,
|
2014-09-11 14:11:54 +02:00
|
|
|
char ***capabilities)
|
2013-01-14 12:45:20 +01:00
|
|
|
{
|
2021-06-14 10:29:36 +02:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *caps;
|
2021-06-14 10:29:36 +02:00
|
|
|
g_auto(GStrv) list = NULL;
|
Convert 'int i' to 'size_t i' in src/qemu files
Convert the type of loop iterators named 'i', 'j', k',
'ii', 'jj', 'kk', to be 'size_t' instead of 'int' or
'unsigned int', also santizing 'ii', 'jj', 'kk' to use
the normal 'i', 'j', 'k' naming
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2013-07-08 15:09:33 +01:00
|
|
|
size_t i;
|
2018-04-19 17:29:02 -04:00
|
|
|
size_t n;
|
2014-09-11 14:11:54 +02:00
|
|
|
|
|
|
|
*capabilities = NULL;
|
2013-01-14 12:45:20 +01:00
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-migrate-capabilities",
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2021-06-14 10:29:36 +02:00
|
|
|
return -1;
|
2013-01-14 12:45:20 +01:00
|
|
|
|
2021-06-14 10:29:36 +02:00
|
|
|
if (qemuMonitorJSONHasError(reply, "CommandNotFound"))
|
|
|
|
return 0;
|
2013-01-14 12:45:20 +01:00
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_ARRAY) < 0)
|
2021-06-14 10:29:36 +02:00
|
|
|
return -1;
|
2013-01-14 12:45:20 +01:00
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
caps = virJSONValueObjectGetArray(reply, "return");
|
|
|
|
n = virJSONValueArraySize(caps);
|
2013-01-14 12:45:20 +01:00
|
|
|
|
2020-10-05 12:26:34 +02:00
|
|
|
list = g_new0(char *, n + 1);
|
2014-09-11 14:11:54 +02:00
|
|
|
|
|
|
|
for (i = 0; i < n; i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cap = virJSONValueArrayGet(caps, i);
|
2013-01-14 12:45:20 +01:00
|
|
|
const char *name;
|
|
|
|
|
2018-03-29 20:30:05 +02:00
|
|
|
if (!cap || virJSONValueGetType(cap) != VIR_JSON_TYPE_OBJECT) {
|
2013-01-14 12:45:20 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("missing entry in migration capabilities list"));
|
2021-06-14 10:29:36 +02:00
|
|
|
return -1;
|
2013-01-14 12:45:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!(name = virJSONValueObjectGetString(cap, "capability"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("missing migration capability name"));
|
2021-06-14 10:29:36 +02:00
|
|
|
return -1;
|
2013-01-14 12:45:20 +01:00
|
|
|
}
|
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
list[i] = g_strdup(name);
|
2013-01-14 12:45:20 +01:00
|
|
|
}
|
|
|
|
|
2021-02-23 14:58:29 +01:00
|
|
|
*capabilities = g_steal_pointer(&list);
|
2021-06-14 10:29:36 +02:00
|
|
|
return n;
|
2013-01-14 12:45:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-03-01 09:26:07 +01:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONSetMigrationCapabilities(qemuMonitor *mon,
|
|
|
|
virJSONValue **caps)
|
2018-03-01 09:26:07 +01:00
|
|
|
{
|
2020-11-30 15:27:44 +01:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2018-03-01 09:26:07 +01:00
|
|
|
|
2020-11-30 15:27:44 +01:00
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("migrate-set-capabilities",
|
|
|
|
"a:capabilities", caps,
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
2018-03-01 09:26:07 +01:00
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2020-11-30 15:27:44 +01:00
|
|
|
return -1;
|
2018-03-01 09:26:07 +01:00
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
2020-11-30 15:27:44 +01:00
|
|
|
return -1;
|
2018-03-01 09:26:07 +01:00
|
|
|
|
2020-11-30 15:27:44 +01:00
|
|
|
return 0;
|
2018-03-01 09:26:07 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-03-08 18:24:18 +01:00
|
|
|
/**
|
|
|
|
* qemuMonitorJSONGetGICCapabilities:
|
|
|
|
* @mon: QEMU JSON monitor
|
|
|
|
* @capabilities: where to store the GIC capabilities
|
|
|
|
*
|
|
|
|
* Use @mon to obtain information about the GIC capabilities for the
|
|
|
|
* corresponding QEMU binary, and store them in @capabilities.
|
|
|
|
*
|
|
|
|
* If the QEMU binary has no GIC capabilities, or if GIC capabilities could
|
|
|
|
* not be determined due to the lack of 'query-gic-capabilities' QMP command,
|
|
|
|
* a NULL pointer will be returned instead of an empty array.
|
|
|
|
*
|
|
|
|
* Returns: the number of GIC capabilities obtained from the monitor,
|
|
|
|
* <0 on failure
|
|
|
|
*/
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetGICCapabilities(qemuMonitor *mon,
|
2016-03-08 18:24:18 +01:00
|
|
|
virGICCapability **capabilities)
|
|
|
|
{
|
2016-05-04 10:28:43 +02:00
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *caps;
|
2016-03-08 18:24:18 +01:00
|
|
|
virGICCapability *list = NULL;
|
|
|
|
size_t i;
|
2018-04-19 17:29:02 -04:00
|
|
|
size_t n;
|
2016-03-08 18:24:18 +01:00
|
|
|
|
|
|
|
*capabilities = NULL;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-gic-capabilities",
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2016-03-08 18:24:18 +01:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
/* If the 'query-gic-capabilities' QMP command was not available
|
|
|
|
* we simply successfully return zero capabilities.
|
|
|
|
* This is the case for QEMU <2.6 and all non-ARM architectures */
|
|
|
|
if (qemuMonitorJSONHasError(reply, "CommandNotFound")) {
|
|
|
|
ret = 0;
|
|
|
|
goto cleanup;
|
2016-03-08 18:24:18 +01:00
|
|
|
}
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_ARRAY) < 0)
|
2016-03-08 18:24:18 +01:00
|
|
|
goto cleanup;
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
caps = virJSONValueObjectGetArray(reply, "return");
|
|
|
|
n = virJSONValueArraySize(caps);
|
2016-03-08 18:24:18 +01:00
|
|
|
|
|
|
|
/* If the returned array was empty we have to return successfully */
|
|
|
|
if (n == 0) {
|
|
|
|
ret = 0;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2020-10-05 12:26:34 +02:00
|
|
|
list = g_new0(virGICCapability, n);
|
2016-03-08 18:24:18 +01:00
|
|
|
|
|
|
|
for (i = 0; i < n; i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cap = virJSONValueArrayGet(caps, i);
|
2016-03-08 18:24:18 +01:00
|
|
|
int version;
|
|
|
|
bool kernel;
|
|
|
|
bool emulated;
|
|
|
|
|
2018-03-29 20:30:05 +02:00
|
|
|
if (!cap || virJSONValueGetType(cap) != VIR_JSON_TYPE_OBJECT) {
|
2016-03-08 18:24:18 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("missing entry in GIC capabilities list"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberInt(cap, "version", &version) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("missing GIC version"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetBoolean(cap, "kernel", &kernel) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("missing in-kernel GIC information"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetBoolean(cap, "emulated", &emulated) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("missing emulated GIC information"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
list[i].version = version;
|
|
|
|
if (kernel)
|
|
|
|
list[i].implementation |= VIR_GIC_IMPLEMENTATION_KERNEL;
|
|
|
|
if (emulated)
|
|
|
|
list[i].implementation |= VIR_GIC_IMPLEMENTATION_EMULATED;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = n;
|
2021-02-23 14:58:29 +01:00
|
|
|
*capabilities = g_steal_pointer(&list);
|
2016-03-08 18:24:18 +01:00
|
|
|
|
|
|
|
cleanup:
|
2016-05-16 07:47:26 -04:00
|
|
|
VIR_FREE(list);
|
2016-03-08 18:24:18 +01:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2018-06-08 09:40:51 -05:00
|
|
|
|
2018-08-14 15:54:46 +02:00
|
|
|
/**
|
|
|
|
* qemuMonitorJSONGetSEVCapabilities:
|
|
|
|
* @mon: qemu monitor object
|
|
|
|
* @capabilities: pointer to pointer to a SEV capability structure to be filled
|
|
|
|
*
|
|
|
|
* This function queries and fills in AMD's SEV platform-specific data.
|
|
|
|
* Note that from QEMU's POV both -object sev-guest and query-sev-capabilities
|
|
|
|
* can be present even if SEV is not available, which basically leaves us with
|
|
|
|
* checking for JSON "GenericError" in order to differentiate between
|
|
|
|
* compiled-in support and actual SEV support on the platform.
|
|
|
|
*
|
|
|
|
* Returns -1 on error, 0 if SEV is not supported, and 1 if SEV is supported on
|
|
|
|
* the platform.
|
|
|
|
*/
|
2018-06-08 09:40:51 -05:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetSEVCapabilities(qemuMonitor *mon,
|
2018-06-08 09:40:51 -05:00
|
|
|
virSEVCapability **capabilities)
|
|
|
|
{
|
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *caps;
|
2018-08-15 17:02:06 +02:00
|
|
|
const char *pdh = NULL;
|
|
|
|
const char *cert_chain = NULL;
|
|
|
|
unsigned int cbitpos;
|
|
|
|
unsigned int reduced_phys_bits;
|
2019-10-15 14:47:50 +02:00
|
|
|
g_autoptr(virSEVCapability) capability = NULL;
|
2018-06-08 09:40:51 -05:00
|
|
|
|
|
|
|
*capabilities = NULL;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-sev-capabilities",
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2018-08-14 15:54:46 +02:00
|
|
|
/* QEMU has only compiled-in support of SEV */
|
2018-06-14 09:30:27 +02:00
|
|
|
if (qemuMonitorJSONHasError(reply, "GenericError")) {
|
|
|
|
ret = 0;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2018-06-08 09:40:51 -05:00
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
caps = virJSONValueObjectGetObject(reply, "return");
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberUint(caps, "cbitpos", &cbitpos) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-sev-capabilities reply was missing"
|
|
|
|
" 'cbitpos' field"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberUint(caps, "reduced-phys-bits",
|
2018-08-15 17:02:06 +02:00
|
|
|
&reduced_phys_bits) < 0) {
|
2018-06-08 09:40:51 -05:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-sev-capabilities reply was missing"
|
|
|
|
" 'reduced-phys-bits' field"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(pdh = virJSONValueObjectGetString(caps, "pdh"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-sev-capabilities reply was missing"
|
|
|
|
" 'pdh' field"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(cert_chain = virJSONValueObjectGetString(caps, "cert-chain"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-sev-capabilities reply was missing"
|
|
|
|
" 'cert-chain' field"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2020-10-05 12:26:34 +02:00
|
|
|
capability = g_new0(virSEVCapability, 1);
|
2018-06-08 09:40:51 -05:00
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
capability->pdh = g_strdup(pdh);
|
2018-06-08 09:40:51 -05:00
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
capability->cert_chain = g_strdup(cert_chain);
|
2018-06-08 09:40:51 -05:00
|
|
|
|
|
|
|
capability->cbitpos = cbitpos;
|
|
|
|
capability->reduced_phys_bits = reduced_phys_bits;
|
2019-10-16 13:43:18 +02:00
|
|
|
*capabilities = g_steal_pointer(&capability);
|
2018-08-14 15:54:46 +02:00
|
|
|
ret = 1;
|
2018-06-08 09:40:51 -05:00
|
|
|
cleanup:
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static virJSONValue *
|
2013-03-12 19:18:22 +01:00
|
|
|
qemuMonitorJSONBuildInetSocketAddress(const char *host,
|
|
|
|
const char *port)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *addr = NULL;
|
|
|
|
virJSONValue *data = NULL;
|
2013-03-12 19:18:22 +01:00
|
|
|
|
2016-09-26 17:19:43 +02:00
|
|
|
if (virJSONValueObjectCreate(&data, "s:host", host,
|
|
|
|
"s:port", port, NULL) < 0)
|
|
|
|
return NULL;
|
2013-03-12 19:18:22 +01:00
|
|
|
|
2016-09-26 17:19:43 +02:00
|
|
|
if (virJSONValueObjectCreate(&addr, "s:type", "inet",
|
2018-03-30 11:12:57 +02:00
|
|
|
"a:data", &data, NULL) < 0) {
|
2016-09-26 17:19:43 +02:00
|
|
|
virJSONValueFree(data);
|
|
|
|
return NULL;
|
|
|
|
}
|
2013-03-12 19:18:22 +01:00
|
|
|
|
|
|
|
return addr;
|
|
|
|
}
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static virJSONValue *
|
2013-03-12 19:48:04 +01:00
|
|
|
qemuMonitorJSONBuildUnixSocketAddress(const char *path)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *addr = NULL;
|
|
|
|
virJSONValue *data = NULL;
|
2013-03-12 19:48:04 +01:00
|
|
|
|
2016-09-26 17:19:43 +02:00
|
|
|
if (virJSONValueObjectCreate(&data, "s:path", path, NULL) < 0)
|
|
|
|
return NULL;
|
2013-03-12 19:48:04 +01:00
|
|
|
|
2016-09-26 17:19:43 +02:00
|
|
|
if (virJSONValueObjectCreate(&addr, "s:type", "unix",
|
2018-03-30 11:12:57 +02:00
|
|
|
"a:data", &data, NULL) < 0) {
|
2016-09-26 17:19:43 +02:00
|
|
|
virJSONValueFree(data);
|
|
|
|
return NULL;
|
|
|
|
}
|
2013-03-12 19:48:04 +01:00
|
|
|
|
|
|
|
return addr;
|
|
|
|
}
|
|
|
|
|
2012-11-22 16:08:52 +01:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONNBDServerStart(qemuMonitor *mon,
|
2019-04-09 16:21:38 -05:00
|
|
|
const virStorageNetHostDef *server,
|
2018-02-21 14:18:15 +01:00
|
|
|
const char *tls_alias)
|
2012-11-22 16:08:52 +01:00
|
|
|
{
|
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = NULL;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *addr = NULL;
|
2012-11-22 16:08:52 +01:00
|
|
|
char *port_str = NULL;
|
|
|
|
|
2019-04-09 16:21:38 -05:00
|
|
|
switch ((virStorageNetHostTransport)server->transport) {
|
|
|
|
case VIR_STORAGE_NET_HOST_TRANS_TCP:
|
2019-10-22 15:26:14 +02:00
|
|
|
port_str = g_strdup_printf("%u", server->port);
|
2019-04-09 16:21:38 -05:00
|
|
|
addr = qemuMonitorJSONBuildInetSocketAddress(server->name, port_str);
|
|
|
|
break;
|
|
|
|
case VIR_STORAGE_NET_HOST_TRANS_UNIX:
|
|
|
|
addr = qemuMonitorJSONBuildUnixSocketAddress(server->socket);
|
|
|
|
break;
|
|
|
|
case VIR_STORAGE_NET_HOST_TRANS_RDMA:
|
|
|
|
case VIR_STORAGE_NET_HOST_TRANS_LAST:
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("invalid server address"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
if (!addr)
|
2018-06-25 21:48:50 +08:00
|
|
|
goto cleanup;
|
2012-11-22 16:08:52 +01:00
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("nbd-server-start",
|
2018-03-30 11:12:57 +02:00
|
|
|
"a:addr", &addr,
|
2018-02-21 14:18:15 +01:00
|
|
|
"S:tls-creds", tls_alias,
|
2012-11-22 16:08:52 +01:00
|
|
|
NULL)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2012-11-22 16:08:52 +01:00
|
|
|
VIR_FREE(port_str);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(addr);
|
|
|
|
return ret;
|
|
|
|
}
|
2012-11-22 16:17:13 +01:00
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONNBDServerAdd(qemuMonitor *mon,
|
2012-11-22 16:17:13 +01:00
|
|
|
const char *deviceID,
|
2019-06-05 21:25:05 -05:00
|
|
|
const char *export,
|
|
|
|
bool writable,
|
|
|
|
const char *bitmap)
|
2012-11-22 16:17:13 +01:00
|
|
|
{
|
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2012-11-22 16:17:13 +01:00
|
|
|
|
2019-06-05 21:25:05 -05:00
|
|
|
/* Note: bitmap must be NULL if QEMU_CAPS_NBD_BITMAP is lacking */
|
2012-11-22 16:17:13 +01:00
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("nbd-server-add",
|
|
|
|
"s:device", deviceID,
|
2019-06-05 21:25:05 -05:00
|
|
|
"S:name", export,
|
2012-11-22 16:17:13 +01:00
|
|
|
"b:writable", writable,
|
2019-06-05 21:25:05 -05:00
|
|
|
"S:bitmap", bitmap,
|
2012-11-22 16:17:13 +01:00
|
|
|
NULL)))
|
|
|
|
return ret;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2012-11-22 16:17:13 +01:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2012-11-22 16:17:13 +01:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2012-11-22 16:17:13 +01:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2013-01-31 14:47:49 +01:00
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONNBDServerStop(qemuMonitor *mon)
|
2013-01-31 14:47:49 +01:00
|
|
|
{
|
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2013-01-31 14:47:49 +01:00
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("nbd-server-stop",
|
|
|
|
NULL)))
|
|
|
|
return ret;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2013-01-31 14:47:49 +01:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2013-01-31 14:47:49 +01:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2013-01-31 14:47:49 +01:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2013-04-12 16:55:45 -04:00
|
|
|
|
|
|
|
|
2020-10-14 11:33:06 +02:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBlockExportAdd(qemuMonitor *mon,
|
|
|
|
virJSONValue **props)
|
2020-10-14 11:33:06 +02:00
|
|
|
{
|
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
|
|
|
|
2020-11-30 15:34:56 +01:00
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommandInternal("block-export-add", props)))
|
2020-10-14 11:33:06 +02:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-04-12 16:55:45 -04:00
|
|
|
static int
|
2021-06-14 16:20:50 +02:00
|
|
|
qemuMonitorJSONGetStringArray(qemuMonitor *mon,
|
|
|
|
const char *qmpCmd,
|
2013-04-12 16:55:45 -04:00
|
|
|
char ***array)
|
|
|
|
{
|
2021-06-14 16:20:50 +02:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2013-04-12 16:55:45 -04:00
|
|
|
|
|
|
|
*array = NULL;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand(qmpCmd, NULL)))
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2021-06-14 16:20:50 +02:00
|
|
|
return -1;
|
2013-04-16 07:05:21 -04:00
|
|
|
|
2021-06-14 16:20:50 +02:00
|
|
|
if (qemuMonitorJSONHasError(reply, "CommandNotFound"))
|
|
|
|
return 0;
|
2013-04-12 16:55:45 -04:00
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_ARRAY) < 0)
|
2021-06-14 16:20:50 +02:00
|
|
|
return -1;
|
2013-04-12 16:55:45 -04:00
|
|
|
|
2020-11-20 22:09:44 +04:00
|
|
|
if (!(*array = virJSONValueObjectGetStringArray(reply, "return")))
|
2021-06-14 16:20:50 +02:00
|
|
|
return -1;
|
2013-04-12 16:55:45 -04:00
|
|
|
|
2021-06-14 16:41:09 +02:00
|
|
|
return 0;
|
2013-04-12 16:55:45 -04:00
|
|
|
}
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONGetTPMModels(qemuMonitor *mon,
|
2013-04-12 16:55:45 -04:00
|
|
|
char ***tpmmodels)
|
|
|
|
{
|
|
|
|
return qemuMonitorJSONGetStringArray(mon, "query-tpm-models", tpmmodels);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int qemuMonitorJSONGetTPMTypes(qemuMonitor *mon,
|
2013-04-12 16:55:45 -04:00
|
|
|
char ***tpmtypes)
|
|
|
|
{
|
|
|
|
return qemuMonitorJSONGetStringArray(mon, "query-tpm-types", tpmtypes);
|
|
|
|
}
|
2013-03-12 19:48:04 +01:00
|
|
|
|
2017-09-13 08:33:40 +08:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBuildChrChardevReconnect(virJSONValue *object,
|
2017-09-13 08:33:40 +08:00
|
|
|
const virDomainChrSourceReconnectDef *def)
|
|
|
|
{
|
|
|
|
if (def->enabled != VIR_TRISTATE_BOOL_YES)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return virJSONValueObjectAppendNumberUint(object, "reconnect", def->timeout);
|
|
|
|
}
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static virJSONValue *
|
2013-03-12 19:48:04 +01:00
|
|
|
qemuMonitorJSONAttachCharDevCommand(const char *chrID,
|
2013-10-08 11:07:53 -06:00
|
|
|
const virDomainChrSourceDef *chr)
|
2013-03-12 19:48:04 +01:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *ret = NULL;
|
|
|
|
virJSONValue *backend = virJSONValueNewObject();
|
|
|
|
virJSONValue *data = virJSONValueNewObject();
|
|
|
|
virJSONValue *addr = NULL;
|
2013-03-12 19:48:04 +01:00
|
|
|
const char *backend_type = NULL;
|
2017-05-19 12:56:48 +02:00
|
|
|
const char *host;
|
|
|
|
const char *port;
|
2016-06-13 12:30:34 -04:00
|
|
|
char *tlsalias = NULL;
|
2013-03-12 19:48:04 +01:00
|
|
|
bool telnet;
|
|
|
|
|
2018-04-25 14:42:34 +02:00
|
|
|
switch ((virDomainChrType)chr->type) {
|
2013-03-12 19:48:04 +01:00
|
|
|
case VIR_DOMAIN_CHR_TYPE_NULL:
|
|
|
|
backend_type = "null";
|
|
|
|
break;
|
|
|
|
|
2021-02-01 11:59:48 +01:00
|
|
|
case VIR_DOMAIN_CHR_TYPE_VC:
|
|
|
|
backend_type = "vc";
|
|
|
|
break;
|
|
|
|
|
2013-03-12 19:48:04 +01:00
|
|
|
case VIR_DOMAIN_CHR_TYPE_PTY:
|
|
|
|
backend_type = "pty";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_CHR_TYPE_FILE:
|
|
|
|
backend_type = "file";
|
|
|
|
if (virJSONValueObjectAppendString(data, "out", chr->data.file.path) < 0)
|
2017-06-13 17:55:36 +02:00
|
|
|
goto cleanup;
|
2020-12-02 14:43:25 +03:00
|
|
|
if (virJSONValueObjectAdd(data,
|
|
|
|
"T:append", chr->data.file.append,
|
|
|
|
NULL) < 0)
|
|
|
|
goto cleanup;
|
2013-03-12 19:48:04 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_CHR_TYPE_DEV:
|
|
|
|
backend_type = STRPREFIX(chrID, "parallel") ? "parallel" : "serial";
|
|
|
|
if (virJSONValueObjectAppendString(data, "device",
|
|
|
|
chr->data.file.path) < 0)
|
2017-06-13 17:55:36 +02:00
|
|
|
goto cleanup;
|
2013-03-12 19:48:04 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_CHR_TYPE_TCP:
|
|
|
|
backend_type = "socket";
|
|
|
|
addr = qemuMonitorJSONBuildInetSocketAddress(chr->data.tcp.host,
|
|
|
|
chr->data.tcp.service);
|
|
|
|
if (!addr ||
|
2021-02-11 17:57:45 +01:00
|
|
|
virJSONValueObjectAppend(data, "addr", &addr) < 0)
|
2017-06-13 17:55:36 +02:00
|
|
|
goto cleanup;
|
2013-03-12 19:48:04 +01:00
|
|
|
|
|
|
|
telnet = chr->data.tcp.protocol == VIR_DOMAIN_CHR_TCP_PROTOCOL_TELNET;
|
|
|
|
|
2019-04-15 18:51:48 +02:00
|
|
|
if (chr->data.tcp.listen &&
|
|
|
|
virJSONValueObjectAppendBoolean(data, "wait", false) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (virJSONValueObjectAppendBoolean(data, "telnet", telnet) < 0 ||
|
2013-03-12 19:48:04 +01:00
|
|
|
virJSONValueObjectAppendBoolean(data, "server", chr->data.tcp.listen) < 0)
|
2017-06-13 17:55:36 +02:00
|
|
|
goto cleanup;
|
2016-06-13 12:30:34 -04:00
|
|
|
if (chr->data.tcp.tlscreds) {
|
2017-02-16 14:59:06 -05:00
|
|
|
if (!(tlsalias = qemuAliasTLSObjFromSrcAlias(chrID)))
|
2017-06-13 17:55:36 +02:00
|
|
|
goto cleanup;
|
2016-06-13 12:30:34 -04:00
|
|
|
|
|
|
|
if (virJSONValueObjectAppendString(data, "tls-creds", tlsalias) < 0)
|
2017-06-13 17:55:36 +02:00
|
|
|
goto cleanup;
|
2016-06-13 12:30:34 -04:00
|
|
|
}
|
2017-09-13 08:33:40 +08:00
|
|
|
|
|
|
|
if (qemuMonitorJSONBuildChrChardevReconnect(data, &chr->data.tcp.reconnect) < 0)
|
|
|
|
goto cleanup;
|
2013-03-12 19:48:04 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_CHR_TYPE_UDP:
|
2016-09-27 16:01:55 +02:00
|
|
|
backend_type = "udp";
|
2017-09-26 13:56:36 +02:00
|
|
|
host = chr->data.udp.connectHost;
|
|
|
|
if (!host)
|
|
|
|
host = "";
|
|
|
|
addr = qemuMonitorJSONBuildInetSocketAddress(host,
|
2013-03-12 19:48:04 +01:00
|
|
|
chr->data.udp.connectService);
|
|
|
|
if (!addr ||
|
2021-02-11 17:57:45 +01:00
|
|
|
virJSONValueObjectAppend(data, "remote", &addr) < 0)
|
2017-06-13 17:55:36 +02:00
|
|
|
goto cleanup;
|
2016-09-27 16:01:55 +02:00
|
|
|
|
2017-05-19 12:56:48 +02:00
|
|
|
host = chr->data.udp.bindHost;
|
|
|
|
port = chr->data.udp.bindService;
|
|
|
|
if (host || port) {
|
|
|
|
if (!host)
|
|
|
|
host = "";
|
|
|
|
if (!port)
|
|
|
|
port = "";
|
|
|
|
addr = qemuMonitorJSONBuildInetSocketAddress(host, port);
|
2016-09-27 16:01:55 +02:00
|
|
|
if (!addr ||
|
2021-02-11 17:57:45 +01:00
|
|
|
virJSONValueObjectAppend(data, "local", &addr) < 0)
|
2017-06-13 17:55:36 +02:00
|
|
|
goto cleanup;
|
2016-09-27 16:01:55 +02:00
|
|
|
}
|
2013-03-12 19:48:04 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_CHR_TYPE_UNIX:
|
|
|
|
backend_type = "socket";
|
|
|
|
addr = qemuMonitorJSONBuildUnixSocketAddress(chr->data.nix.path);
|
|
|
|
|
|
|
|
if (!addr ||
|
2021-02-11 17:57:45 +01:00
|
|
|
virJSONValueObjectAppend(data, "addr", &addr) < 0)
|
2017-06-13 17:55:36 +02:00
|
|
|
goto cleanup;
|
2013-03-12 19:48:04 +01:00
|
|
|
|
2019-04-15 18:51:48 +02:00
|
|
|
if (chr->data.nix.listen &&
|
|
|
|
virJSONValueObjectAppendBoolean(data, "wait", false) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (virJSONValueObjectAppendBoolean(data, "server", chr->data.nix.listen) < 0)
|
2017-06-13 17:55:36 +02:00
|
|
|
goto cleanup;
|
2017-09-13 08:33:40 +08:00
|
|
|
|
|
|
|
if (qemuMonitorJSONBuildChrChardevReconnect(data, &chr->data.nix.reconnect) < 0)
|
|
|
|
goto cleanup;
|
2013-03-12 19:48:04 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_CHR_TYPE_SPICEVMC:
|
2016-06-08 15:51:27 +02:00
|
|
|
backend_type = "spicevmc";
|
|
|
|
|
|
|
|
if (virJSONValueObjectAppendString(data, "type",
|
|
|
|
virDomainChrSpicevmcTypeToString(chr->data.spicevmc)) < 0)
|
2017-06-13 17:55:36 +02:00
|
|
|
goto cleanup;
|
2016-06-08 15:51:27 +02:00
|
|
|
break;
|
|
|
|
|
2014-02-10 11:18:16 +01:00
|
|
|
case VIR_DOMAIN_CHR_TYPE_SPICEPORT:
|
2013-03-12 19:48:04 +01:00
|
|
|
case VIR_DOMAIN_CHR_TYPE_PIPE:
|
|
|
|
case VIR_DOMAIN_CHR_TYPE_STDIO:
|
2014-03-15 16:30:01 +04:00
|
|
|
case VIR_DOMAIN_CHR_TYPE_NMDM:
|
2013-03-12 19:48:04 +01:00
|
|
|
case VIR_DOMAIN_CHR_TYPE_LAST:
|
2014-11-10 16:52:49 +01:00
|
|
|
if (virDomainChrTypeToString(chr->type)) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED,
|
|
|
|
_("Hotplug unsupported for char device type '%s'"),
|
|
|
|
virDomainChrTypeToString(chr->type));
|
|
|
|
} else {
|
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED,
|
|
|
|
_("Hotplug unsupported for char device type '%d'"),
|
|
|
|
chr->type);
|
|
|
|
}
|
2017-06-13 17:55:36 +02:00
|
|
|
goto cleanup;
|
2013-03-12 19:48:04 +01:00
|
|
|
}
|
|
|
|
|
2020-12-02 14:44:38 +03:00
|
|
|
if (chr->logfile &&
|
|
|
|
virJSONValueObjectAdd(data,
|
|
|
|
"s:logfile", chr->logfile,
|
|
|
|
"T:logappend", chr->logappend,
|
|
|
|
NULL) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2013-03-12 19:48:04 +01:00
|
|
|
if (virJSONValueObjectAppendString(backend, "type", backend_type) < 0 ||
|
2021-02-11 17:57:45 +01:00
|
|
|
virJSONValueObjectAppend(backend, "data", &data) < 0)
|
2017-06-13 17:55:36 +02:00
|
|
|
goto cleanup;
|
2013-03-12 19:48:04 +01:00
|
|
|
|
|
|
|
if (!(ret = qemuMonitorJSONMakeCommand("chardev-add",
|
|
|
|
"s:id", chrID,
|
2018-03-30 11:12:57 +02:00
|
|
|
"a:backend", &backend,
|
2013-03-12 19:48:04 +01:00
|
|
|
NULL)))
|
2017-06-13 17:55:36 +02:00
|
|
|
goto cleanup;
|
2013-03-12 19:48:04 +01:00
|
|
|
|
2017-06-13 17:55:36 +02:00
|
|
|
cleanup:
|
2016-06-13 12:30:34 -04:00
|
|
|
VIR_FREE(tlsalias);
|
2013-03-12 19:48:04 +01:00
|
|
|
virJSONValueFree(addr);
|
|
|
|
virJSONValueFree(data);
|
|
|
|
virJSONValueFree(backend);
|
2017-06-13 17:55:36 +02:00
|
|
|
return ret;
|
2013-03-12 19:48:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONAttachCharDev(qemuMonitor *mon,
|
2013-03-12 19:48:04 +01:00
|
|
|
const char *chrID,
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainChrSourceDef *chr)
|
2013-03-12 19:48:04 +01:00
|
|
|
{
|
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2013-03-12 19:48:04 +01:00
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONAttachCharDevCommand(chrID, chr)))
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (chr->type == VIR_DOMAIN_CHR_TYPE_PTY) {
|
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_OBJECT) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
} else {
|
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2013-03-12 19:48:04 +01:00
|
|
|
|
|
|
|
if (chr->type == VIR_DOMAIN_CHR_TYPE_PTY) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *data = virJSONValueObjectGetObject(reply, "return");
|
2013-03-12 19:48:04 +01:00
|
|
|
const char *path;
|
|
|
|
|
|
|
|
if (!(path = virJSONValueObjectGetString(data, "pty"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("chardev-add reply was missing pty path"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
chr->data.file.path = g_strdup(path);
|
2013-03-12 19:48:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2013-03-12 19:48:04 +01:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2013-03-12 19:57:48 +01:00
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONDetachCharDev(qemuMonitor *mon,
|
2013-03-12 19:57:48 +01:00
|
|
|
const char *chrID)
|
|
|
|
{
|
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2013-03-12 19:57:48 +01:00
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("chardev-remove",
|
|
|
|
"s:id", chrID,
|
|
|
|
NULL)))
|
|
|
|
return ret;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2013-03-12 19:57:48 +01:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2013-03-12 19:57:48 +01:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2013-03-12 19:57:48 +01:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2013-07-19 15:01:38 +02:00
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetDeviceAliases(qemuMonitor *mon,
|
2013-07-19 15:01:38 +02:00
|
|
|
char ***aliases)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONListPath **paths = NULL;
|
2013-07-19 15:01:38 +02:00
|
|
|
char **alias;
|
|
|
|
int ret = -1;
|
|
|
|
size_t i;
|
|
|
|
int n;
|
|
|
|
|
|
|
|
*aliases = NULL;
|
|
|
|
|
|
|
|
n = qemuMonitorJSONGetObjectListPaths(mon, "/machine/peripheral", &paths);
|
|
|
|
if (n < 0)
|
|
|
|
return -1;
|
|
|
|
|
2020-10-05 12:26:34 +02:00
|
|
|
*aliases = g_new0(char *, n + 1);
|
2013-07-19 15:01:38 +02:00
|
|
|
|
|
|
|
alias = *aliases;
|
|
|
|
for (i = 0; i < n; i++) {
|
|
|
|
if (STRPREFIX(paths[i]->type, "child<")) {
|
2021-03-24 10:32:58 +01:00
|
|
|
*alias = g_steal_pointer(&paths[i]->name);
|
2013-07-19 15:01:38 +02:00
|
|
|
alias++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
for (i = 0; i < n; i++)
|
|
|
|
qemuMonitorJSONListPathFree(paths[i]);
|
|
|
|
VIR_FREE(paths);
|
|
|
|
return ret;
|
|
|
|
}
|
2013-07-22 13:07:23 +02:00
|
|
|
|
|
|
|
|
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONParseCPUx86FeatureWord(virJSONValue *data,
|
2013-07-22 13:07:23 +02:00
|
|
|
virCPUx86CPUID *cpuid)
|
|
|
|
{
|
|
|
|
const char *reg;
|
2016-05-20 09:48:21 +02:00
|
|
|
unsigned long long eax_in;
|
2016-05-20 10:59:13 +02:00
|
|
|
unsigned long long ecx_in = 0;
|
2013-07-22 13:07:23 +02:00
|
|
|
unsigned long long features;
|
|
|
|
|
|
|
|
memset(cpuid, 0, sizeof(*cpuid));
|
|
|
|
|
|
|
|
if (!(reg = virJSONValueObjectGetString(data, "cpuid-register"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("missing cpuid-register in CPU data"));
|
|
|
|
return -1;
|
|
|
|
}
|
2016-05-20 09:48:21 +02:00
|
|
|
if (virJSONValueObjectGetNumberUlong(data, "cpuid-input-eax", &eax_in) < 0) {
|
2013-07-22 13:07:23 +02:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("missing or invalid cpuid-input-eax in CPU data"));
|
|
|
|
return -1;
|
|
|
|
}
|
2016-05-20 10:59:13 +02:00
|
|
|
ignore_value(virJSONValueObjectGetNumberUlong(data, "cpuid-input-ecx",
|
|
|
|
&ecx_in));
|
2013-07-22 13:07:23 +02:00
|
|
|
if (virJSONValueObjectGetNumberUlong(data, "features", &features) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("missing or invalid features in CPU data"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2016-05-20 09:48:21 +02:00
|
|
|
cpuid->eax_in = eax_in;
|
2016-05-20 10:59:13 +02:00
|
|
|
cpuid->ecx_in = ecx_in;
|
2013-07-22 13:07:23 +02:00
|
|
|
if (STREQ(reg, "EAX")) {
|
|
|
|
cpuid->eax = features;
|
|
|
|
} else if (STREQ(reg, "EBX")) {
|
|
|
|
cpuid->ebx = features;
|
|
|
|
} else if (STREQ(reg, "ECX")) {
|
|
|
|
cpuid->ecx = features;
|
|
|
|
} else if (STREQ(reg, "EDX")) {
|
|
|
|
cpuid->edx = features;
|
|
|
|
} else {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("unknown CPU register '%s'"), reg);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static virCPUData *
|
|
|
|
qemuMonitorJSONParseCPUx86Features(virJSONValue *data)
|
2016-06-03 16:59:59 +02:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virCPUData *cpudata = NULL;
|
2019-03-13 17:01:19 +01:00
|
|
|
virCPUx86DataItem item = { 0 };
|
2016-06-03 16:59:59 +02:00
|
|
|
size_t i;
|
|
|
|
|
2017-02-02 12:19:13 +01:00
|
|
|
if (!(cpudata = virCPUDataNew(VIR_ARCH_X86_64)))
|
|
|
|
goto error;
|
|
|
|
|
2019-03-14 15:44:27 +01:00
|
|
|
item.type = VIR_CPU_X86_DATA_CPUID;
|
2018-04-19 17:29:02 -04:00
|
|
|
for (i = 0; i < virJSONValueArraySize(data); i++) {
|
2016-06-03 16:59:59 +02:00
|
|
|
if (qemuMonitorJSONParseCPUx86FeatureWord(virJSONValueArrayGet(data, i),
|
2019-03-14 15:44:27 +01:00
|
|
|
&item.data.cpuid) < 0 ||
|
2019-03-14 22:02:44 +01:00
|
|
|
virCPUx86DataAdd(cpudata, &item) < 0)
|
2017-02-02 12:19:13 +01:00
|
|
|
goto error;
|
2016-06-03 16:59:59 +02:00
|
|
|
}
|
|
|
|
|
2017-02-02 12:19:13 +01:00
|
|
|
return cpudata;
|
2016-06-03 16:59:59 +02:00
|
|
|
|
2017-02-02 12:19:13 +01:00
|
|
|
error:
|
2017-02-02 15:37:40 +01:00
|
|
|
virCPUDataFree(cpudata);
|
2017-02-02 12:19:13 +01:00
|
|
|
return NULL;
|
2016-06-03 16:59:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetCPUx86Data(qemuMonitor *mon,
|
2013-11-11 14:47:08 +01:00
|
|
|
const char *property,
|
2021-03-11 08:16:13 +01:00
|
|
|
virCPUData **cpudata)
|
2013-07-22 13:07:23 +02:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = NULL;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *data;
|
2013-11-11 14:47:08 +01:00
|
|
|
int ret = -1;
|
2013-07-22 13:07:23 +02:00
|
|
|
|
2016-06-03 16:59:59 +02:00
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("qom-get",
|
2013-11-11 16:34:53 +01:00
|
|
|
"s:path", QOM_CPU_PATH,
|
2016-06-03 16:59:59 +02:00
|
|
|
"s:property", property,
|
2013-11-11 16:34:53 +01:00
|
|
|
NULL)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_ARRAY) < 0)
|
2013-11-11 16:34:53 +01:00
|
|
|
goto cleanup;
|
|
|
|
|
qemu: simplify json parsing
Rather than grabbing an arbitrary JSON value and then checking
if it has the right type, we might as well request the correct
type to begin with.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONIOProcessEvent)
(qemuMonitorJSONCommandWithFd, qemuMonitorJSONHandleGraphics)
(qemuMonitorJSONGetStatus, qemuMonitorJSONExtractCPUInfo)
(qemuMonitorJSONGetVirtType, qemuMonitorJSONGetBalloonInfo)
(qemuMonitorJSONGetMemoryStats)
(qemuMonitorJSONDevGetBlockExtent)
(qemuMonitorJSONGetOneBlockStatsInfo)
(qemuMonitorJSONGetAllBlockStatsInfo)
(qemuMonitorJSONBlockStatsUpdateCapacityOne)
(qemuMonitorJSONBlockStatsUpdateCapacity)
(qemuMonitorJSONGetBlockExtent)
(qemuMonitorJSONGetMigrationStatusReply)
(qemuMonitorJSONGetDumpGuestMemoryCapability)
(qemuMonitorJSONAddFd, qemuMonitorJSONQueryRxFilterParse)
(qemuMonitorJSONExtractChardevInfo)
(qemuMonitorJSONDiskNameLookupOne)
(qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONGetAllBlockJobInfo)
(qemuMonitorJSONBlockIoThrottleInfo, qemuMonitorJSONGetVersion)
(qemuMonitorJSONGetMachines, qemuMonitorJSONGetCPUDefinitions)
(qemuMonitorJSONGetCommands, qemuMonitorJSONGetEvents)
(qemuMonitorJSONGetKVMState, qemuMonitorJSONGetObjectTypes)
(qemuMonitorJSONGetObjectListPaths)
(qemuMonitorJSONGetObjectProps, qemuMonitorJSONGetTargetArch)
(qemuMonitorJSONGetMigrationCapabilities)
(qemuMonitorJSONGetStringArray, qemuMonitorJSONAttachCharDev)
(qemuMonitorJSONGetCPUx86Data, qemuMonitorJSONGetIOThreads)
(qemuMonitorJSONGetMemoryDeviceInfo): Use shorter idioms.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-06-20 13:40:21 -06:00
|
|
|
data = virJSONValueObjectGetArray(reply, "return");
|
2017-02-02 12:19:13 +01:00
|
|
|
if (!(*cpudata = qemuMonitorJSONParseCPUx86Features(data)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = 0;
|
2013-11-11 16:34:53 +01:00
|
|
|
|
2016-06-03 16:59:59 +02:00
|
|
|
cleanup:
|
2013-11-11 16:34:53 +01:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
2016-06-03 16:59:59 +02:00
|
|
|
return ret;
|
|
|
|
}
|
2013-11-11 16:34:53 +01:00
|
|
|
|
2016-06-03 16:59:59 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Returns -1 on error, 0 if QEMU does not support reporting CPUID features
|
|
|
|
* of a guest CPU, and 1 if the feature is supported.
|
|
|
|
*/
|
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONCheckCPUx86(qemuMonitor *mon)
|
2016-06-03 16:59:59 +02:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd = NULL;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *data;
|
2016-06-03 16:59:59 +02:00
|
|
|
size_t i;
|
2018-04-19 17:29:02 -04:00
|
|
|
size_t n;
|
2016-06-03 16:59:59 +02:00
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("qom-list",
|
2013-07-22 13:07:23 +02:00
|
|
|
"s:path", QOM_CPU_PATH,
|
|
|
|
NULL)))
|
2013-11-11 16:34:53 +01:00
|
|
|
goto cleanup;
|
2013-07-22 13:07:23 +02:00
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2016-06-03 16:59:59 +02:00
|
|
|
if ((data = virJSONValueObjectGet(reply, "error"))) {
|
|
|
|
const char *klass = virJSONValueObjectGetString(data, "class");
|
|
|
|
if (STREQ_NULLABLE(klass, "DeviceNotFound") ||
|
|
|
|
STREQ_NULLABLE(klass, "CommandNotFound")) {
|
|
|
|
ret = 0;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_ARRAY) < 0)
|
2013-07-22 13:07:23 +02:00
|
|
|
goto cleanup;
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
data = virJSONValueObjectGetArray(reply, "return");
|
|
|
|
n = virJSONValueArraySize(data);
|
2013-07-22 13:07:23 +02:00
|
|
|
|
|
|
|
for (i = 0; i < n; i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *element = virJSONValueArrayGet(data, i);
|
2016-06-03 16:59:59 +02:00
|
|
|
if (STREQ_NULLABLE(virJSONValueObjectGetString(element, "name"),
|
|
|
|
"feature-words"))
|
|
|
|
break;
|
2013-07-22 13:07:23 +02:00
|
|
|
}
|
|
|
|
|
2016-06-03 16:59:59 +02:00
|
|
|
if (i == n)
|
|
|
|
ret = 0;
|
|
|
|
else
|
|
|
|
ret = 1;
|
2013-11-11 14:47:08 +01:00
|
|
|
|
2014-03-25 07:49:44 +01:00
|
|
|
cleanup:
|
2013-07-22 13:07:23 +02:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
2019-06-17 16:56:32 +02:00
|
|
|
* qemuMonitorJSONGetGuestCPUx86:
|
2013-07-22 13:07:23 +02:00
|
|
|
* @mon: Pointer to the monitor
|
2013-11-11 14:47:08 +01:00
|
|
|
* @data: returns the cpu data of the guest
|
2017-03-13 11:00:48 +01:00
|
|
|
* @disabled: returns the CPU data for features which were disabled by QEMU
|
2013-07-22 13:07:23 +02:00
|
|
|
*
|
|
|
|
* Retrieve the definition of the guest CPU from a running qemu instance.
|
|
|
|
*
|
2013-11-11 14:47:08 +01:00
|
|
|
* Returns 0 on success, -2 if guest doesn't support this feature,
|
|
|
|
* -1 on other errors.
|
2013-07-22 13:07:23 +02:00
|
|
|
*/
|
2013-11-11 14:47:08 +01:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetGuestCPUx86(qemuMonitor *mon,
|
|
|
|
virCPUData **data,
|
|
|
|
virCPUData **disabled)
|
2013-07-22 13:07:23 +02:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virCPUData *cpuEnabled = NULL;
|
|
|
|
virCPUData *cpuDisabled = NULL;
|
2016-06-03 16:59:59 +02:00
|
|
|
int rc;
|
|
|
|
|
2019-06-17 16:56:32 +02:00
|
|
|
if ((rc = qemuMonitorJSONCheckCPUx86(mon)) < 0)
|
|
|
|
return -1;
|
|
|
|
else if (!rc)
|
|
|
|
return -2;
|
2017-03-13 11:00:48 +01:00
|
|
|
|
2019-06-17 16:56:32 +02:00
|
|
|
if (qemuMonitorJSONGetCPUx86Data(mon, "feature-words",
|
|
|
|
&cpuEnabled) < 0)
|
|
|
|
goto error;
|
2017-03-13 11:00:48 +01:00
|
|
|
|
2019-06-17 16:56:32 +02:00
|
|
|
if (disabled &&
|
|
|
|
qemuMonitorJSONGetCPUx86Data(mon, "filtered-features",
|
|
|
|
&cpuDisabled) < 0)
|
|
|
|
goto error;
|
2017-03-13 10:23:37 +01:00
|
|
|
|
2019-06-17 16:56:32 +02:00
|
|
|
*data = cpuEnabled;
|
|
|
|
if (disabled)
|
|
|
|
*disabled = cpuDisabled;
|
|
|
|
return 0;
|
2017-03-13 11:00:48 +01:00
|
|
|
|
|
|
|
error:
|
|
|
|
virCPUDataFree(cpuEnabled);
|
|
|
|
virCPUDataFree(cpuDisabled);
|
|
|
|
return -1;
|
2013-07-22 13:07:23 +02:00
|
|
|
}
|
2014-08-13 14:28:24 +02:00
|
|
|
|
2019-06-17 23:36:53 +02:00
|
|
|
|
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetCPUProperties(qemuMonitor *mon,
|
2019-06-17 23:36:53 +02:00
|
|
|
char ***props)
|
|
|
|
{
|
2019-10-15 14:47:50 +02:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2019-06-17 23:36:53 +02:00
|
|
|
|
|
|
|
*props = NULL;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("qom-list",
|
|
|
|
"s:path", QOM_CPU_PATH,
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONHasError(reply, "DeviceNotFound"))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return qemuMonitorJSONParsePropsList(cmd, reply, "bool", props);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetCPUData(qemuMonitor *mon,
|
2019-06-17 23:36:53 +02:00
|
|
|
qemuMonitorCPUFeatureTranslationCallback translate,
|
|
|
|
void *opaque,
|
2021-03-11 08:16:13 +01:00
|
|
|
virCPUData *data)
|
2019-06-17 23:36:53 +02:00
|
|
|
{
|
|
|
|
qemuMonitorJSONObjectProperty prop = { .type = QEMU_MONITOR_OBJECT_PROPERTY_BOOLEAN };
|
2020-12-01 09:21:32 +01:00
|
|
|
g_auto(GStrv) props = NULL;
|
2019-06-17 23:36:53 +02:00
|
|
|
char **p;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONGetCPUProperties(mon, &props) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
for (p = props; p && *p; p++) {
|
|
|
|
const char *name = *p;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONGetObjectProperty(mon, QOM_CPU_PATH, name, &prop) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (!prop.val.b)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (translate)
|
|
|
|
name = translate(name, opaque);
|
|
|
|
|
|
|
|
if (virCPUDataAddFeature(data, name) < 0)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetCPUDataDisabled(qemuMonitor *mon,
|
2019-06-17 23:36:53 +02:00
|
|
|
qemuMonitorCPUFeatureTranslationCallback translate,
|
|
|
|
void *opaque,
|
2021-03-11 08:16:13 +01:00
|
|
|
virCPUData *data)
|
2019-06-17 23:36:53 +02:00
|
|
|
{
|
2020-12-01 09:21:32 +01:00
|
|
|
g_auto(GStrv) props = NULL;
|
2019-06-17 23:36:53 +02:00
|
|
|
char **p;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONGetStringListProperty(mon, QOM_CPU_PATH,
|
|
|
|
"unavailable-features", &props) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
for (p = props; p && *p; p++) {
|
|
|
|
const char *name = *p;
|
|
|
|
|
|
|
|
if (translate)
|
|
|
|
name = translate(name, opaque);
|
|
|
|
|
|
|
|
if (virCPUDataAddFeature(data, name) < 0)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* qemuMonitorJSONGetGuestCPU:
|
|
|
|
* @mon: Pointer to the monitor
|
|
|
|
* @arch: CPU architecture
|
|
|
|
* @translate: callback for translating CPU feature names from QEMU to libvirt
|
|
|
|
* @opaque: data for @translate callback
|
|
|
|
* @enabled: returns the CPU data for all enabled features
|
|
|
|
* @disabled: returns the CPU data for features which we asked for
|
|
|
|
* (either explicitly or via a named CPU model) but QEMU disabled them
|
|
|
|
*
|
|
|
|
* Retrieve the definition of the guest CPU from a running QEMU instance.
|
|
|
|
*
|
|
|
|
* Returns 0 on success, -1 on error.
|
|
|
|
*/
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetGuestCPU(qemuMonitor *mon,
|
2019-06-17 23:36:53 +02:00
|
|
|
virArch arch,
|
|
|
|
qemuMonitorCPUFeatureTranslationCallback translate,
|
|
|
|
void *opaque,
|
2021-03-11 08:16:13 +01:00
|
|
|
virCPUData **enabled,
|
|
|
|
virCPUData **disabled)
|
2019-06-17 23:36:53 +02:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virCPUData *cpuEnabled = NULL;
|
|
|
|
virCPUData *cpuDisabled = NULL;
|
2019-06-17 23:36:53 +02:00
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
if (!(cpuEnabled = virCPUDataNew(arch)) ||
|
|
|
|
!(cpuDisabled = virCPUDataNew(arch)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONGetCPUData(mon, translate, opaque, cpuEnabled) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (disabled &&
|
|
|
|
qemuMonitorJSONGetCPUDataDisabled(mon, translate, opaque, cpuDisabled) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2019-10-16 13:43:18 +02:00
|
|
|
*enabled = g_steal_pointer(&cpuEnabled);
|
2019-06-17 23:36:53 +02:00
|
|
|
if (disabled)
|
2019-10-16 13:43:18 +02:00
|
|
|
*disabled = g_steal_pointer(&cpuDisabled);
|
2019-06-17 23:36:53 +02:00
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virCPUDataFree(cpuEnabled);
|
|
|
|
virCPUDataFree(cpuDisabled);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-08-13 14:28:24 +02:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONRTCResetReinjection(qemuMonitor *mon)
|
2014-08-13 14:28:24 +02:00
|
|
|
{
|
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2014-08-13 14:28:24 +02:00
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("rtc-reset-reinjection",
|
|
|
|
NULL)))
|
|
|
|
return ret;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2014-08-13 14:28:24 +02:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
2014-08-13 14:28:24 +02:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2014-08-13 14:28:24 +02:00
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2014-08-29 16:23:11 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Query and parse returned array of data such as:
|
|
|
|
*
|
|
|
|
* {u'return': [{u'id': u'iothread1', u'thread-id': 30992}, \
|
|
|
|
* {u'id': u'iothread2', u'thread-id': 30993}]}
|
|
|
|
*/
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetIOThreads(qemuMonitor *mon,
|
|
|
|
qemuMonitorIOThreadInfo ***iothreads,
|
2020-12-02 12:34:24 -05:00
|
|
|
int *niothreads)
|
2014-08-29 16:23:11 -04:00
|
|
|
{
|
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *data;
|
|
|
|
qemuMonitorIOThreadInfo **infolist = NULL;
|
2018-04-19 17:29:02 -04:00
|
|
|
size_t n = 0;
|
2014-08-29 16:23:11 -04:00
|
|
|
size_t i;
|
|
|
|
|
|
|
|
*iothreads = NULL;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-iothreads", NULL)))
|
|
|
|
return ret;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2014-08-29 16:23:11 -04:00
|
|
|
goto cleanup;
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_ARRAY) < 0)
|
2016-05-04 10:28:43 +02:00
|
|
|
goto cleanup;
|
2014-08-29 16:23:11 -04:00
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
data = virJSONValueObjectGetArray(reply, "return");
|
|
|
|
n = virJSONValueArraySize(data);
|
2014-08-29 16:23:11 -04:00
|
|
|
|
|
|
|
/* null-terminated list */
|
2021-03-11 08:16:13 +01:00
|
|
|
infolist = g_new0(qemuMonitorIOThreadInfo *, n + 1);
|
2014-08-29 16:23:11 -04:00
|
|
|
|
|
|
|
for (i = 0; i < n; i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *child = virJSONValueArrayGet(data, i);
|
2014-08-29 16:23:11 -04:00
|
|
|
const char *tmp;
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorIOThreadInfo *info;
|
2014-08-29 16:23:11 -04:00
|
|
|
|
|
|
|
if (!(tmp = virJSONValueObjectGetString(child, "id"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-iothreads reply data was missing 'id'"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2015-04-27 14:16:54 -04:00
|
|
|
if (!STRPREFIX(tmp, "iothread"))
|
|
|
|
continue;
|
|
|
|
|
2020-10-05 12:26:34 +02:00
|
|
|
info = g_new0(qemuMonitorIOThreadInfo, 1);
|
2014-08-29 16:23:11 -04:00
|
|
|
|
2015-04-27 14:16:54 -04:00
|
|
|
infolist[i] = info;
|
|
|
|
|
|
|
|
if (virStrToLong_ui(tmp + strlen("iothread"),
|
|
|
|
NULL, 10, &info->iothread_id) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("failed to find iothread id for '%s'"),
|
|
|
|
tmp);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2014-08-29 16:23:11 -04:00
|
|
|
if (virJSONValueObjectGetNumberInt(child, "thread-id",
|
|
|
|
&info->thread_id) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-iothreads reply has malformed "
|
|
|
|
"'thread-id' data"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2018-10-03 07:38:34 -04:00
|
|
|
|
|
|
|
/* Fetch poll values (since QEMU 2.9 ) if available. QEMU
|
|
|
|
* stores these values as int64_t's; however, the qapi type
|
|
|
|
* is an int. The qapi/misc.json also mis-describes the grow
|
|
|
|
* and shrink values as pure add/remove values. The source
|
|
|
|
* util/aio-posix.c function aio_poll uses them as a factor
|
|
|
|
* or divisor in it's calculation. We will fetch and store
|
|
|
|
* them as defined in our structures. */
|
|
|
|
if (virJSONValueObjectGetNumberUlong(child, "poll-max-ns",
|
|
|
|
&info->poll_max_ns) == 0 &&
|
|
|
|
virJSONValueObjectGetNumberUint(child, "poll-grow",
|
|
|
|
&info->poll_grow) == 0 &&
|
|
|
|
virJSONValueObjectGetNumberUint(child, "poll-shrink",
|
|
|
|
&info->poll_shrink) == 0)
|
|
|
|
info->poll_valid = true;
|
2014-08-29 16:23:11 -04:00
|
|
|
}
|
|
|
|
|
2020-12-02 12:34:24 -05:00
|
|
|
*niothreads = n;
|
2021-02-23 14:58:29 +01:00
|
|
|
*iothreads = g_steal_pointer(&infolist);
|
2020-12-02 12:34:24 -05:00
|
|
|
ret = 0;
|
2014-08-29 16:23:11 -04:00
|
|
|
|
|
|
|
cleanup:
|
2016-05-16 07:47:26 -04:00
|
|
|
if (infolist) {
|
2014-08-29 16:23:11 -04:00
|
|
|
for (i = 0; i < n; i++)
|
2015-04-28 06:32:52 -04:00
|
|
|
VIR_FREE(infolist[i]);
|
2014-08-29 16:23:11 -04:00
|
|
|
VIR_FREE(infolist);
|
|
|
|
}
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2015-01-19 13:21:09 +01:00
|
|
|
|
|
|
|
|
2018-10-03 18:13:14 -04:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONSetIOThread(qemuMonitor *mon,
|
|
|
|
qemuMonitorIOThreadInfo *iothreadInfo)
|
2018-10-03 18:13:14 -04:00
|
|
|
{
|
|
|
|
int ret = -1;
|
|
|
|
char *path = NULL;
|
|
|
|
qemuMonitorJSONObjectProperty prop;
|
|
|
|
|
2019-10-22 15:26:14 +02:00
|
|
|
path = g_strdup_printf("/objects/iothread%u", iothreadInfo->iothread_id);
|
2018-10-03 18:13:14 -04:00
|
|
|
|
|
|
|
#define VIR_IOTHREAD_SET_PROP(propName, propVal) \
|
|
|
|
if (iothreadInfo->set_##propVal) { \
|
|
|
|
memset(&prop, 0, sizeof(qemuMonitorJSONObjectProperty)); \
|
|
|
|
prop.type = QEMU_MONITOR_OBJECT_PROPERTY_INT; \
|
|
|
|
prop.val.iv = iothreadInfo->propVal; \
|
|
|
|
if (qemuMonitorJSONSetObjectProperty(mon, path, propName, &prop) < 0) \
|
|
|
|
goto cleanup; \
|
|
|
|
}
|
|
|
|
|
|
|
|
VIR_IOTHREAD_SET_PROP("poll-max-ns", poll_max_ns);
|
|
|
|
VIR_IOTHREAD_SET_PROP("poll-grow", poll_grow);
|
|
|
|
VIR_IOTHREAD_SET_PROP("poll-shrink", poll_shrink);
|
|
|
|
|
|
|
|
#undef VIR_IOTHREAD_SET_PROP
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(path);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-01-19 13:21:09 +01:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetMemoryDeviceInfo(qemuMonitor *mon,
|
2020-10-22 19:04:18 +02:00
|
|
|
GHashTable *info)
|
2015-01-19 13:21:09 +01:00
|
|
|
{
|
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *data = NULL;
|
|
|
|
qemuMonitorMemoryDeviceInfo *meminfo = NULL;
|
2015-01-19 13:21:09 +01:00
|
|
|
size_t i;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-memory-devices", NULL)))
|
|
|
|
return -1;
|
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
2015-01-19 13:21:09 +01:00
|
|
|
|
2016-05-04 10:28:43 +02:00
|
|
|
if (qemuMonitorJSONHasError(reply, "CommandNotFound")) {
|
|
|
|
ret = -2;
|
|
|
|
goto cleanup;
|
2015-01-19 13:21:09 +01:00
|
|
|
}
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_ARRAY) < 0)
|
2015-01-19 13:21:09 +01:00
|
|
|
goto cleanup;
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
data = virJSONValueObjectGetArray(reply, "return");
|
2015-01-19 13:21:09 +01:00
|
|
|
|
2018-04-19 17:29:02 -04:00
|
|
|
for (i = 0; i < virJSONValueArraySize(data); i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *elem = virJSONValueArrayGet(data, i);
|
2015-01-19 13:21:09 +01:00
|
|
|
const char *type;
|
|
|
|
|
|
|
|
if (!(type = virJSONValueObjectGetString(elem, "type"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-memory-devices reply data doesn't contain "
|
|
|
|
"enum type discriminator"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* dimm memory devices */
|
|
|
|
if (STREQ(type, "dimm")) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *dimminfo;
|
2015-01-19 13:21:09 +01:00
|
|
|
const char *devalias;
|
|
|
|
|
qemu: simplify json parsing
Rather than grabbing an arbitrary JSON value and then checking
if it has the right type, we might as well request the correct
type to begin with.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONIOProcessEvent)
(qemuMonitorJSONCommandWithFd, qemuMonitorJSONHandleGraphics)
(qemuMonitorJSONGetStatus, qemuMonitorJSONExtractCPUInfo)
(qemuMonitorJSONGetVirtType, qemuMonitorJSONGetBalloonInfo)
(qemuMonitorJSONGetMemoryStats)
(qemuMonitorJSONDevGetBlockExtent)
(qemuMonitorJSONGetOneBlockStatsInfo)
(qemuMonitorJSONGetAllBlockStatsInfo)
(qemuMonitorJSONBlockStatsUpdateCapacityOne)
(qemuMonitorJSONBlockStatsUpdateCapacity)
(qemuMonitorJSONGetBlockExtent)
(qemuMonitorJSONGetMigrationStatusReply)
(qemuMonitorJSONGetDumpGuestMemoryCapability)
(qemuMonitorJSONAddFd, qemuMonitorJSONQueryRxFilterParse)
(qemuMonitorJSONExtractChardevInfo)
(qemuMonitorJSONDiskNameLookupOne)
(qemuMonitorJSONDiskNameLookup)
(qemuMonitorJSONGetAllBlockJobInfo)
(qemuMonitorJSONBlockIoThrottleInfo, qemuMonitorJSONGetVersion)
(qemuMonitorJSONGetMachines, qemuMonitorJSONGetCPUDefinitions)
(qemuMonitorJSONGetCommands, qemuMonitorJSONGetEvents)
(qemuMonitorJSONGetKVMState, qemuMonitorJSONGetObjectTypes)
(qemuMonitorJSONGetObjectListPaths)
(qemuMonitorJSONGetObjectProps, qemuMonitorJSONGetTargetArch)
(qemuMonitorJSONGetMigrationCapabilities)
(qemuMonitorJSONGetStringArray, qemuMonitorJSONAttachCharDev)
(qemuMonitorJSONGetCPUx86Data, qemuMonitorJSONGetIOThreads)
(qemuMonitorJSONGetMemoryDeviceInfo): Use shorter idioms.
Signed-off-by: Eric Blake <eblake@redhat.com>
2015-06-20 13:40:21 -06:00
|
|
|
if (!(dimminfo = virJSONValueObjectGetObject(elem, "data"))) {
|
2015-01-19 13:21:09 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-memory-devices reply data doesn't "
|
|
|
|
"contain enum data"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(devalias = virJSONValueObjectGetString(dimminfo, "id"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("dimm memory info data is missing 'id'"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2020-10-05 12:26:34 +02:00
|
|
|
meminfo = g_new0(qemuMonitorMemoryDeviceInfo, 1);
|
2015-01-19 13:21:09 +01:00
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberUlong(dimminfo, "addr",
|
|
|
|
&meminfo->address) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("malformed/missing addr in dimm memory info"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberUint(dimminfo, "slot",
|
|
|
|
&meminfo->slot) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("malformed/missing slot in dimm memory info"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetBoolean(dimminfo, "hotplugged",
|
|
|
|
&meminfo->hotplugged) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("malformed/missing hotplugged in dimm memory info"));
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetBoolean(dimminfo, "hotpluggable",
|
|
|
|
&meminfo->hotpluggable) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("malformed/missing hotpluggable in dimm memory info"));
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virHashAddEntry(info, devalias, meminfo) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
meminfo = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(meminfo);
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2015-06-03 12:22:53 +02:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
2016-07-25 10:47:00 +02:00
|
|
|
* Search for a QOM object link by alias and name.
|
|
|
|
*
|
|
|
|
* For @alias and @name, this function tries to find QOM object named @name
|
|
|
|
* with id @alias in /machine/peripheral.
|
|
|
|
*
|
|
|
|
* Returns:
|
|
|
|
* 0 - Found
|
|
|
|
* -1 - Error - bail out
|
|
|
|
* -2 - Not found
|
|
|
|
*/
|
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONFindObjectPathByAlias(qemuMonitor *mon,
|
2016-07-25 10:47:00 +02:00
|
|
|
const char *name,
|
|
|
|
const char *alias,
|
|
|
|
char **path)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONListPath **paths = NULL;
|
2016-07-25 10:47:00 +02:00
|
|
|
char *child = NULL;
|
|
|
|
int npaths;
|
|
|
|
int ret = -1;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
npaths = qemuMonitorJSONGetObjectListPaths(mon, "/machine/peripheral", &paths);
|
|
|
|
if (npaths < 0)
|
|
|
|
return -1;
|
|
|
|
|
2019-10-22 15:26:14 +02:00
|
|
|
child = g_strdup_printf("child<%s>", name);
|
2016-07-25 10:47:00 +02:00
|
|
|
|
|
|
|
for (i = 0; i < npaths; i++) {
|
|
|
|
if (STREQ(paths[i]->name, alias) && STREQ(paths[i]->type, child)) {
|
2019-10-22 15:26:14 +02:00
|
|
|
*path = g_strdup_printf("/machine/peripheral/%s", alias);
|
2016-07-25 10:47:00 +02:00
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = -2;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
for (i = 0; i < npaths; i++)
|
|
|
|
qemuMonitorJSONListPathFree(paths[i]);
|
|
|
|
VIR_FREE(paths);
|
|
|
|
VIR_FREE(child);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Recursively search for a QOM object link only by name.
|
2015-06-03 12:22:53 +02:00
|
|
|
*
|
2015-06-03 12:40:08 +02:00
|
|
|
* For @name, this function finds the first QOM object
|
|
|
|
* named @name, recursively going through all the "child<>"
|
|
|
|
* entries, starting from @curpath.
|
2015-06-03 12:22:53 +02:00
|
|
|
*
|
|
|
|
* Returns:
|
|
|
|
* 0 - Found
|
2015-06-03 12:40:08 +02:00
|
|
|
* -1 - Error - bail out
|
2015-06-03 12:22:53 +02:00
|
|
|
* -2 - Not found
|
|
|
|
*/
|
2015-06-03 12:40:08 +02:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONFindObjectPathByName(qemuMonitor *mon,
|
2016-07-25 10:47:00 +02:00
|
|
|
const char *curpath,
|
|
|
|
const char *name,
|
|
|
|
char **path)
|
2015-06-03 12:22:53 +02:00
|
|
|
{
|
|
|
|
ssize_t i, npaths = 0;
|
|
|
|
int ret = -2;
|
|
|
|
char *nextpath = NULL;
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONListPath **paths = NULL;
|
2015-06-03 12:22:53 +02:00
|
|
|
|
2015-06-03 12:40:08 +02:00
|
|
|
VIR_DEBUG("Searching for '%s' Object Path starting at '%s'", name, curpath);
|
2015-06-03 12:22:53 +02:00
|
|
|
|
|
|
|
npaths = qemuMonitorJSONGetObjectListPaths(mon, curpath, &paths);
|
|
|
|
if (npaths < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
for (i = 0; i < npaths && ret == -2; i++) {
|
|
|
|
|
2015-06-03 12:40:08 +02:00
|
|
|
if (STREQ_NULLABLE(paths[i]->type, name)) {
|
|
|
|
VIR_DEBUG("Path to '%s' is '%s/%s'", name, curpath, paths[i]->name);
|
2015-06-03 12:22:53 +02:00
|
|
|
ret = 0;
|
2019-10-22 15:26:14 +02:00
|
|
|
*path = g_strdup_printf("%s/%s", curpath, paths[i]->name);
|
2015-06-03 12:22:53 +02:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Type entries that begin with "child<" are a branch that can be
|
|
|
|
* traversed looking for more entries
|
|
|
|
*/
|
|
|
|
if (paths[i]->type && STRPREFIX(paths[i]->type, "child<")) {
|
2019-10-22 15:26:14 +02:00
|
|
|
nextpath = g_strdup_printf("%s/%s", curpath, paths[i]->name);
|
2015-06-03 12:22:53 +02:00
|
|
|
|
2016-07-25 10:47:00 +02:00
|
|
|
ret = qemuMonitorJSONFindObjectPathByName(mon, nextpath, name, path);
|
2015-06-03 12:22:53 +02:00
|
|
|
VIR_FREE(nextpath);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
for (i = 0; i < npaths; i++)
|
|
|
|
qemuMonitorJSONListPathFree(paths[i]);
|
|
|
|
VIR_FREE(paths);
|
|
|
|
VIR_FREE(nextpath);
|
2015-06-03 12:40:08 +02:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Recursively search for a QOM object link.
|
|
|
|
*
|
2016-07-25 10:47:00 +02:00
|
|
|
* For @name and @alias, this function finds the first QOM object.
|
|
|
|
* The search is done at first by @alias and @name and if nothing was found
|
|
|
|
* it continues recursively only with @name.
|
2015-06-03 12:40:08 +02:00
|
|
|
*
|
|
|
|
* Returns:
|
|
|
|
* 0 - Found
|
|
|
|
* -1 - Error
|
|
|
|
* -2 - Not found
|
|
|
|
*/
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONFindLinkPath(qemuMonitor *mon,
|
2015-06-03 12:40:08 +02:00
|
|
|
const char *name,
|
2016-07-25 10:47:00 +02:00
|
|
|
const char *alias,
|
2015-06-03 12:40:08 +02:00
|
|
|
char **path)
|
|
|
|
{
|
|
|
|
char *linkname = NULL;
|
|
|
|
int ret = -1;
|
|
|
|
|
2016-07-25 10:47:00 +02:00
|
|
|
if (alias) {
|
|
|
|
ret = qemuMonitorJSONFindObjectPathByAlias(mon, name, alias, path);
|
|
|
|
if (ret == -1 || ret == 0)
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2019-10-22 15:26:14 +02:00
|
|
|
linkname = g_strdup_printf("link<%s>", name);
|
2015-06-03 12:40:08 +02:00
|
|
|
|
2016-07-25 10:47:00 +02:00
|
|
|
ret = qemuMonitorJSONFindObjectPathByName(mon, "/", linkname, path);
|
2015-06-03 12:40:08 +02:00
|
|
|
VIR_FREE(linkname);
|
2015-06-03 12:22:53 +02:00
|
|
|
return ret;
|
|
|
|
}
|
2015-10-20 22:51:48 +02:00
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONMigrateIncoming(qemuMonitor *mon,
|
2015-10-20 22:51:48 +02:00
|
|
|
const char *uri)
|
|
|
|
{
|
2020-07-13 06:49:44 -03:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2015-10-20 22:51:48 +02:00
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("migrate-incoming",
|
|
|
|
"s:uri", uri,
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2020-07-13 06:49:44 -03:00
|
|
|
return -1;
|
2015-10-20 22:51:48 +02:00
|
|
|
|
2020-07-13 06:49:44 -03:00
|
|
|
return qemuMonitorJSONCheckError(cmd, reply);
|
2015-10-20 22:51:48 +02:00
|
|
|
}
|
2014-12-01 16:59:54 +01:00
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONMigrateStartPostCopy(qemuMonitor *mon)
|
2014-12-01 16:59:54 +01:00
|
|
|
{
|
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2014-12-01 16:59:54 +01:00
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("migrate-start-postcopy", NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = qemuMonitorJSONCheckError(cmd, reply);
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2016-04-29 16:01:47 +02:00
|
|
|
|
2017-10-20 09:17:09 +02:00
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONMigrateContinue(qemuMonitor *mon,
|
2017-10-20 09:17:09 +02:00
|
|
|
qemuMonitorMigrationStatus status)
|
|
|
|
{
|
|
|
|
const char *statusStr = qemuMonitorMigrationStatusTypeToString(status);
|
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2017-10-20 09:17:09 +02:00
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("migrate-continue",
|
|
|
|
"s:state", statusStr,
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = qemuMonitorJSONCheckError(cmd, reply);
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-04-29 16:01:47 +02:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetRTCTime(qemuMonitor *mon,
|
2016-04-29 16:01:47 +02:00
|
|
|
struct tm *tm)
|
|
|
|
{
|
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *data;
|
2016-04-29 16:01:47 +02:00
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("qom-get",
|
|
|
|
"s:path", "/machine",
|
|
|
|
"s:property", "rtc-time",
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_OBJECT) < 0)
|
2016-04-29 16:01:47 +02:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
data = virJSONValueObjectGet(reply, "return");
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberInt(data, "tm_year", &tm->tm_year) < 0 ||
|
|
|
|
virJSONValueObjectGetNumberInt(data, "tm_mon", &tm->tm_mon) < 0 ||
|
|
|
|
virJSONValueObjectGetNumberInt(data, "tm_mday", &tm->tm_mday) < 0 ||
|
|
|
|
virJSONValueObjectGetNumberInt(data, "tm_hour", &tm->tm_hour) < 0 ||
|
|
|
|
virJSONValueObjectGetNumberInt(data, "tm_min", &tm->tm_min) < 0 ||
|
|
|
|
virJSONValueObjectGetNumberInt(data, "tm_sec", &tm->tm_sec) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("qemu returned malformed time"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2016-07-08 13:52:11 +02:00
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
qemuMonitorQueryHotpluggableCpusFree(struct qemuMonitorQueryHotpluggableCpusEntry *entries,
|
|
|
|
size_t nentries)
|
|
|
|
{
|
|
|
|
struct qemuMonitorQueryHotpluggableCpusEntry *entry;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
if (!entries)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (i = 0; i < nentries; i++) {
|
|
|
|
entry = entries + i;
|
|
|
|
|
2021-02-03 14:36:01 -05:00
|
|
|
g_free(entry->type);
|
|
|
|
g_free(entry->qom_path);
|
|
|
|
g_free(entry->alias);
|
2019-08-29 14:47:10 +02:00
|
|
|
virJSONValueFree(entry->props);
|
2016-07-08 13:52:11 +02:00
|
|
|
}
|
|
|
|
|
2021-02-03 14:36:01 -05:00
|
|
|
g_free(entries);
|
2016-07-08 13:52:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* [{
|
|
|
|
* "props": {
|
|
|
|
* "core-id": 0,
|
|
|
|
* "thread-id": 0,
|
|
|
|
* "socket-id": 0
|
|
|
|
* },
|
|
|
|
* "vcpus-count": 1,
|
|
|
|
* "qom-path": "/machine/unattached/device[0]",
|
|
|
|
* "type": "qemu64-x86_64-cpu"
|
|
|
|
* },
|
|
|
|
* {...}
|
|
|
|
* ]
|
|
|
|
*/
|
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONProcessHotpluggableCpusReply(virJSONValue *vcpu,
|
2016-07-08 13:52:11 +02:00
|
|
|
struct qemuMonitorQueryHotpluggableCpusEntry *entry)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *props;
|
2016-07-08 13:52:11 +02:00
|
|
|
const char *tmp;
|
|
|
|
|
|
|
|
if (!(tmp = virJSONValueObjectGetString(vcpu, "type"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-hotpluggable-cpus didn't return device type"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
entry->type = g_strdup(tmp);
|
2016-07-08 13:52:11 +02:00
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberUint(vcpu, "vcpus-count", &entry->vcpus) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-hotpluggable-cpus didn't return vcpus-count"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(props = virJSONValueObjectGetObject(vcpu, "props"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-hotpluggable-cpus didn't return device props"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2019-08-29 14:47:10 +02:00
|
|
|
if (!(entry->props = virJSONValueCopy(props)))
|
|
|
|
return -1;
|
|
|
|
|
2016-07-08 13:52:11 +02:00
|
|
|
entry->node_id = -1;
|
|
|
|
entry->socket_id = -1;
|
2020-03-13 16:43:26 +00:00
|
|
|
entry->die_id = -1;
|
2016-07-08 13:52:11 +02:00
|
|
|
entry->core_id = -1;
|
|
|
|
entry->thread_id = -1;
|
|
|
|
|
|
|
|
ignore_value(virJSONValueObjectGetNumberInt(props, "node-id", &entry->node_id));
|
|
|
|
ignore_value(virJSONValueObjectGetNumberInt(props, "socket-id", &entry->socket_id));
|
2020-03-13 16:43:26 +00:00
|
|
|
ignore_value(virJSONValueObjectGetNumberInt(props, "die-id", &entry->die_id));
|
2016-07-08 13:52:11 +02:00
|
|
|
ignore_value(virJSONValueObjectGetNumberInt(props, "core-id", &entry->core_id));
|
|
|
|
ignore_value(virJSONValueObjectGetNumberInt(props, "thread-id", &entry->thread_id));
|
|
|
|
|
|
|
|
if (entry->node_id == -1 && entry->socket_id == -1 &&
|
|
|
|
entry->core_id == -1 && entry->thread_id == -1) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-hotpluggable-cpus entry doesn't report "
|
|
|
|
"topology information"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* qom path is not present unless the vCPU is online */
|
|
|
|
if ((tmp = virJSONValueObjectGetString(vcpu, "qom-path"))) {
|
2019-10-20 13:49:46 +02:00
|
|
|
entry->qom_path = g_strdup(tmp);
|
2016-07-08 13:52:11 +02:00
|
|
|
|
|
|
|
/* alias is the part after last slash having a "vcpu" prefix */
|
2019-10-20 13:49:46 +02:00
|
|
|
if ((tmp = strrchr(tmp, '/')) && STRPREFIX(tmp + 1, "vcpu"))
|
|
|
|
entry->alias = g_strdup(tmp + 1);
|
2016-07-08 13:52:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
qemuMonitorQueryHotpluggableCpusEntrySort(const void *p1,
|
|
|
|
const void *p2)
|
|
|
|
{
|
|
|
|
const struct qemuMonitorQueryHotpluggableCpusEntry *a = p1;
|
|
|
|
const struct qemuMonitorQueryHotpluggableCpusEntry *b = p2;
|
|
|
|
|
|
|
|
if (a->socket_id != b->socket_id)
|
|
|
|
return a->socket_id - b->socket_id;
|
|
|
|
|
2020-03-13 16:43:26 +00:00
|
|
|
if (a->die_id != b->die_id)
|
|
|
|
return a->die_id - b->die_id;
|
|
|
|
|
2016-07-08 13:52:11 +02:00
|
|
|
if (a->core_id != b->core_id)
|
|
|
|
return a->core_id - b->core_id;
|
|
|
|
|
|
|
|
return a->thread_id - b->thread_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetHotpluggableCPUs(qemuMonitor *mon,
|
2016-07-08 13:52:11 +02:00
|
|
|
struct qemuMonitorQueryHotpluggableCpusEntry **entries,
|
|
|
|
size_t *nentries)
|
|
|
|
{
|
|
|
|
struct qemuMonitorQueryHotpluggableCpusEntry *info = NULL;
|
2018-04-19 17:29:02 -04:00
|
|
|
size_t ninfo = 0;
|
2016-07-08 13:52:11 +02:00
|
|
|
int ret = -1;
|
|
|
|
size_t i;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *data;
|
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *vcpu;
|
2016-07-08 13:52:11 +02:00
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-hotpluggable-cpus", NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_ARRAY) < 0)
|
2016-07-08 13:52:11 +02:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
data = virJSONValueObjectGet(reply, "return");
|
2018-03-28 12:45:21 +02:00
|
|
|
ninfo = virJSONValueArraySize(data);
|
2016-07-08 13:52:11 +02:00
|
|
|
|
2020-10-05 12:26:34 +02:00
|
|
|
info = g_new0(struct qemuMonitorQueryHotpluggableCpusEntry, ninfo);
|
2016-07-08 13:52:11 +02:00
|
|
|
|
|
|
|
for (i = 0; i < ninfo; i++) {
|
|
|
|
vcpu = virJSONValueArrayGet(data, i);
|
|
|
|
|
|
|
|
if (qemuMonitorJSONProcessHotpluggableCpusReply(vcpu, info + i) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
qsort(info, ninfo, sizeof(*info), qemuMonitorQueryHotpluggableCpusEntrySort);
|
|
|
|
|
2019-10-16 13:43:18 +02:00
|
|
|
*entries = g_steal_pointer(&info);
|
2016-07-08 13:52:11 +02:00
|
|
|
*nentries = ninfo;
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
qemuMonitorQueryHotpluggableCpusFree(info, ninfo);
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2016-10-17 14:20:42 +02:00
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *
|
|
|
|
qemuMonitorJSONQueryQMPSchema(qemuMonitor *mon)
|
2016-10-17 14:20:42 +02:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *ret = NULL;
|
2016-10-17 14:20:42 +02:00
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-qmp-schema", NULL)))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_ARRAY) < 0)
|
2016-10-17 14:20:42 +02:00
|
|
|
goto cleanup;
|
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
ret = virJSONValueObjectStealArray(reply, "return");
|
2016-10-17 14:20:42 +02:00
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
2017-02-23 13:50:24 +01:00
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONSetBlockThreshold(qemuMonitor *mon,
|
2017-02-23 13:50:24 +01:00
|
|
|
const char *nodename,
|
|
|
|
unsigned long long threshold)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2017-02-23 13:50:24 +01:00
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("block-set-write-threshold",
|
|
|
|
"s:node-name", nodename,
|
|
|
|
"U:write-threshold", threshold,
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
2017-02-24 14:59:40 +01:00
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *
|
|
|
|
qemuMonitorJSONQueryNamedBlockNodes(qemuMonitor *mon,
|
2020-01-21 16:42:49 +01:00
|
|
|
bool flat)
|
2017-02-24 14:59:40 +01:00
|
|
|
{
|
2020-01-21 16:33:12 +01:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2017-02-24 14:59:40 +01:00
|
|
|
|
2020-01-21 16:42:49 +01:00
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-named-block-nodes",
|
|
|
|
"B:flat", flat,
|
|
|
|
NULL)))
|
2017-02-24 14:59:40 +01:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2020-01-21 16:33:12 +01:00
|
|
|
return NULL;
|
2017-02-24 14:59:40 +01:00
|
|
|
|
2018-03-28 12:45:21 +02:00
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_ARRAY) < 0)
|
2020-01-21 16:33:12 +01:00
|
|
|
return NULL;
|
2017-02-24 14:59:40 +01:00
|
|
|
|
2020-01-21 16:33:12 +01:00
|
|
|
return virJSONValueObjectStealArray(reply, "return");
|
2017-02-24 14:59:40 +01:00
|
|
|
}
|
2017-09-01 13:39:15 +02:00
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONSetWatchdogAction(qemuMonitor *mon,
|
2017-09-01 13:39:15 +02:00
|
|
|
const char *action)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2017-09-01 13:39:15 +02:00
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("watchdog-set-action",
|
|
|
|
"s:action", action,
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2018-02-20 18:04:47 +01:00
|
|
|
|
|
|
|
|
2018-09-03 14:45:16 +02:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBlockdevCreate(qemuMonitor *mon,
|
2018-09-03 14:45:16 +02:00
|
|
|
const char *jobname,
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *props)
|
2018-09-03 14:45:16 +02:00
|
|
|
{
|
2019-10-15 14:47:50 +02:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2018-09-03 14:45:16 +02:00
|
|
|
|
|
|
|
cmd = qemuMonitorJSONMakeCommand("blockdev-create",
|
|
|
|
"s:job-id", jobname,
|
|
|
|
"a:options", &props,
|
|
|
|
NULL);
|
|
|
|
virJSONValueFree(props);
|
|
|
|
if (!cmd)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-02-20 18:04:47 +01:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBlockdevAdd(qemuMonitor *mon,
|
|
|
|
virJSONValue **props)
|
2018-02-20 18:04:47 +01:00
|
|
|
{
|
2020-02-06 13:03:07 +01:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2018-02-20 18:04:47 +01:00
|
|
|
|
2020-11-30 15:34:56 +01:00
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommandInternal("blockdev-add", props)))
|
2018-02-20 18:04:47 +01:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2020-02-06 13:03:07 +01:00
|
|
|
return -1;
|
2018-02-20 18:04:47 +01:00
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
2020-02-06 13:03:07 +01:00
|
|
|
return -1;
|
2018-02-20 18:04:47 +01:00
|
|
|
|
2020-02-06 13:03:07 +01:00
|
|
|
return 0;
|
2018-02-20 18:04:47 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-02-13 09:00:37 +01:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBlockdevReopen(qemuMonitor *mon,
|
|
|
|
virJSONValue **props)
|
2020-02-13 09:00:37 +01:00
|
|
|
{
|
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
|
|
|
|
2020-11-30 15:34:56 +01:00
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommandInternal("blockdev-reopen", props)))
|
2020-02-13 09:00:37 +01:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-02-20 18:04:47 +01:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBlockdevDel(qemuMonitor *mon,
|
2018-02-20 18:04:47 +01:00
|
|
|
const char *nodename)
|
|
|
|
{
|
2020-02-06 13:03:07 +01:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2018-02-20 18:04:47 +01:00
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("blockdev-del",
|
|
|
|
"s:node-name", nodename,
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
2020-02-06 13:03:07 +01:00
|
|
|
return -1;
|
2018-02-20 18:04:47 +01:00
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
2020-02-06 13:03:07 +01:00
|
|
|
return -1;
|
2018-02-20 18:04:47 +01:00
|
|
|
|
2020-02-06 13:03:07 +01:00
|
|
|
return 0;
|
2018-02-20 18:04:47 +01:00
|
|
|
}
|
2018-06-08 09:41:01 -05:00
|
|
|
|
2018-07-12 12:11:31 +02:00
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBlockdevTrayOpen(qemuMonitor *mon,
|
2018-07-12 12:11:31 +02:00
|
|
|
const char *id,
|
|
|
|
bool force)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2018-07-12 12:11:31 +02:00
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("blockdev-open-tray",
|
|
|
|
"s:id", id,
|
|
|
|
"b:force", force, NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBlockdevTrayClose(qemuMonitor *mon,
|
2018-07-12 12:11:31 +02:00
|
|
|
const char *id)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2018-07-12 12:11:31 +02:00
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("blockdev-close-tray",
|
|
|
|
"s:id", id, NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBlockdevMediumRemove(qemuMonitor *mon,
|
2018-07-12 12:11:31 +02:00
|
|
|
const char *id)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2018-07-12 12:11:31 +02:00
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("blockdev-remove-medium",
|
|
|
|
"s:id", id, NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBlockdevMediumInsert(qemuMonitor *mon,
|
2018-07-12 12:11:31 +02:00
|
|
|
const char *id,
|
|
|
|
const char *nodename)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2018-07-12 12:11:31 +02:00
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("blockdev-insert-medium",
|
|
|
|
"s:id", id,
|
|
|
|
"s:node-name", nodename,
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-06-08 09:41:01 -05:00
|
|
|
/**
|
|
|
|
* The function is used to retrieve the measurement of a SEV guest.
|
|
|
|
* The measurement is signature of the memory contents that was encrypted
|
|
|
|
* through the SEV launch flow.
|
|
|
|
*
|
|
|
|
* A example JSON output:
|
|
|
|
*
|
|
|
|
* { "execute" : "query-sev-launch-measure" }
|
|
|
|
* { "return" : { "data" : "4l8LXeNlSPUDlXPJG5966/8%YZ" } }
|
|
|
|
*/
|
|
|
|
char *
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetSEVMeasurement(qemuMonitor *mon)
|
2018-06-08 09:41:01 -05:00
|
|
|
{
|
|
|
|
const char *tmp;
|
|
|
|
char *measurement = NULL;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
|
|
|
virJSONValue *data;
|
2018-06-08 09:41:01 -05:00
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-sev-launch-measure", NULL)))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_OBJECT) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
data = virJSONValueObjectGetObject(reply, "return");
|
|
|
|
|
|
|
|
if (!(tmp = virJSONValueObjectGetString(data, "data")))
|
|
|
|
goto cleanup;
|
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
measurement = g_strdup(tmp);
|
2018-06-08 09:41:01 -05:00
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return measurement;
|
|
|
|
}
|
2018-07-03 12:07:30 +02:00
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Example return data
|
|
|
|
*
|
|
|
|
* "return": [
|
|
|
|
* { "connected": true, "id": "pr-helper0" }
|
|
|
|
* ]
|
|
|
|
*/
|
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONExtractPRManagerInfo(virJSONValue *reply,
|
2020-10-22 19:04:18 +02:00
|
|
|
GHashTable *info)
|
2018-07-03 12:07:30 +02:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorPRManagerInfo *entry = NULL;
|
|
|
|
virJSONValue *data;
|
2018-07-03 12:07:30 +02:00
|
|
|
int ret = -1;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
data = virJSONValueObjectGetArray(reply, "return");
|
|
|
|
|
|
|
|
for (i = 0; i < virJSONValueArraySize(data); i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *prManager = virJSONValueArrayGet(data, i);
|
2018-07-03 12:07:30 +02:00
|
|
|
const char *alias;
|
|
|
|
|
|
|
|
if (!(alias = virJSONValueObjectGetString(prManager, "id")))
|
|
|
|
goto malformed;
|
|
|
|
|
2020-10-05 12:26:34 +02:00
|
|
|
entry = g_new0(qemuMonitorPRManagerInfo, 1);
|
2018-07-03 12:07:30 +02:00
|
|
|
|
|
|
|
if (virJSONValueObjectGetBoolean(prManager,
|
|
|
|
"connected",
|
|
|
|
&entry->connected) < 0) {
|
|
|
|
goto malformed;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virHashAddEntry(info, alias, entry) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
entry = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(entry);
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
malformed:
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("malformed prManager reply"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetPRManagerInfo(qemuMonitor *mon,
|
2020-10-22 19:04:18 +02:00
|
|
|
GHashTable *info)
|
2018-07-03 12:07:30 +02:00
|
|
|
{
|
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2018-07-03 12:07:30 +02:00
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-pr-managers",
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_ARRAY) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = qemuMonitorJSONExtractPRManagerInfo(reply, info);
|
|
|
|
cleanup:
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
}
|
2019-04-24 18:16:28 -03:00
|
|
|
|
|
|
|
|
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONExtractCurrentMachineInfo(virJSONValue *reply,
|
|
|
|
qemuMonitorCurrentMachineInfo *info)
|
2019-04-24 18:16:28 -03:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *data;
|
2019-04-24 18:16:28 -03:00
|
|
|
|
|
|
|
data = virJSONValueObjectGetObject(reply, "return");
|
|
|
|
if (!data)
|
|
|
|
goto malformed;
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetBoolean(data, "wakeup-suspend-support",
|
|
|
|
&info->wakeupSuspendSupport) < 0)
|
|
|
|
goto malformed;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
malformed:
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("malformed qemu-current-machine reply"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetCurrentMachineInfo(qemuMonitor *mon,
|
|
|
|
qemuMonitorCurrentMachineInfo *info)
|
2019-04-24 18:16:28 -03:00
|
|
|
{
|
|
|
|
int ret = -1;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *cmd;
|
|
|
|
virJSONValue *reply = NULL;
|
2019-04-24 18:16:28 -03:00
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-current-machine",
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_OBJECT) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = qemuMonitorJSONExtractCurrentMachineInfo(reply, info);
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virJSONValueFree(cmd);
|
|
|
|
virJSONValueFree(reply);
|
|
|
|
return ret;
|
|
|
|
}
|
2019-06-05 21:25:05 -05:00
|
|
|
|
|
|
|
|
2019-09-26 15:53:39 +02:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONTransactionBitmapAdd(virJSONValue *actions,
|
2019-09-26 15:53:39 +02:00
|
|
|
const char *node,
|
|
|
|
const char *name,
|
|
|
|
bool persistent,
|
2019-11-28 16:03:16 +01:00
|
|
|
bool disabled,
|
|
|
|
unsigned long long granularity)
|
2019-09-26 15:53:39 +02:00
|
|
|
{
|
|
|
|
return qemuMonitorJSONTransactionAdd(actions,
|
|
|
|
"block-dirty-bitmap-add",
|
|
|
|
"s:node", node,
|
|
|
|
"s:name", name,
|
|
|
|
"b:persistent", persistent,
|
|
|
|
"b:disabled", disabled,
|
2019-11-28 16:03:16 +01:00
|
|
|
"P:granularity", granularity,
|
2019-09-26 15:53:39 +02:00
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONTransactionBitmapRemove(virJSONValue *actions,
|
2019-09-26 15:53:39 +02:00
|
|
|
const char *node,
|
|
|
|
const char *name)
|
|
|
|
{
|
|
|
|
return qemuMonitorJSONTransactionAdd(actions,
|
|
|
|
"block-dirty-bitmap-remove",
|
|
|
|
"s:node", node,
|
|
|
|
"s:name", name,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-02-08 16:58:53 +01:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONBitmapRemove(qemuMonitor *mon,
|
2021-02-08 16:58:53 +01:00
|
|
|
const char *node,
|
|
|
|
const char *name)
|
|
|
|
{
|
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("block-dirty-bitmap-remove",
|
|
|
|
"s:node", node,
|
|
|
|
"s:name", name,
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-09-26 15:53:39 +02:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONTransactionBitmapEnable(virJSONValue *actions,
|
2019-09-26 15:53:39 +02:00
|
|
|
const char *node,
|
|
|
|
const char *name)
|
|
|
|
{
|
|
|
|
return qemuMonitorJSONTransactionAdd(actions,
|
|
|
|
"block-dirty-bitmap-enable",
|
|
|
|
"s:node", node,
|
|
|
|
"s:name", name,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONTransactionBitmapDisable(virJSONValue *actions,
|
2019-09-26 15:53:39 +02:00
|
|
|
const char *node,
|
|
|
|
const char *name)
|
|
|
|
{
|
|
|
|
return qemuMonitorJSONTransactionAdd(actions,
|
|
|
|
"block-dirty-bitmap-disable",
|
|
|
|
"s:node", node,
|
|
|
|
"s:name", name,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONTransactionBitmapMerge(virJSONValue *actions,
|
2019-09-26 15:53:39 +02:00
|
|
|
const char *node,
|
|
|
|
const char *target,
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue **sources)
|
2019-09-26 15:53:39 +02:00
|
|
|
{
|
|
|
|
return qemuMonitorJSONTransactionAdd(actions,
|
|
|
|
"block-dirty-bitmap-merge",
|
|
|
|
"s:node", node,
|
|
|
|
"s:target", target,
|
|
|
|
"a:bitmaps", sources,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-10-07 16:19:34 +02:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONTransactionBitmapMergeSourceAddBitmap(virJSONValue *sources,
|
2019-10-07 16:19:34 +02:00
|
|
|
const char *sourcenode,
|
|
|
|
const char *sourcebitmap)
|
|
|
|
{
|
|
|
|
g_autoptr(virJSONValue) sourceobj = NULL;
|
|
|
|
|
|
|
|
if (virJSONValueObjectCreate(&sourceobj,
|
|
|
|
"s:node", sourcenode,
|
|
|
|
"s:name", sourcebitmap,
|
|
|
|
NULL) < 0)
|
|
|
|
return -1;
|
|
|
|
|
2021-02-11 17:57:45 +01:00
|
|
|
if (virJSONValueArrayAppend(sources, &sourceobj) < 0)
|
2019-10-07 16:19:34 +02:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-09-26 16:03:46 +02:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONTransactionSnapshotLegacy(virJSONValue *actions,
|
2019-09-26 16:03:46 +02:00
|
|
|
const char *device,
|
|
|
|
const char *path,
|
|
|
|
const char *format,
|
|
|
|
bool existing)
|
|
|
|
{
|
|
|
|
const char *mode = NULL;
|
|
|
|
|
|
|
|
if (existing)
|
|
|
|
mode = "existing";
|
|
|
|
|
|
|
|
return qemuMonitorJSONTransactionAdd(actions,
|
|
|
|
"blockdev-snapshot-sync",
|
|
|
|
"s:device", device,
|
|
|
|
"s:snapshot-file", path,
|
|
|
|
"s:format", format,
|
|
|
|
"S:mode", mode,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONTransactionSnapshotBlockdev(virJSONValue *actions,
|
2019-09-26 16:03:46 +02:00
|
|
|
const char *node,
|
|
|
|
const char *overlay)
|
|
|
|
{
|
|
|
|
return qemuMonitorJSONTransactionAdd(actions,
|
|
|
|
"blockdev-snapshot",
|
|
|
|
"s:node", node,
|
|
|
|
"s:overlay", overlay,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
|
2019-09-27 17:28:48 +02:00
|
|
|
VIR_ENUM_DECL(qemuMonitorTransactionBackupSyncMode);
|
|
|
|
VIR_ENUM_IMPL(qemuMonitorTransactionBackupSyncMode,
|
|
|
|
QEMU_MONITOR_TRANSACTION_BACKUP_SYNC_MODE_LAST,
|
|
|
|
"none",
|
|
|
|
"incremental",
|
|
|
|
"full");
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONTransactionBackup(virJSONValue *actions,
|
2019-09-27 17:28:48 +02:00
|
|
|
const char *device,
|
|
|
|
const char *jobname,
|
|
|
|
const char *target,
|
|
|
|
const char *bitmap,
|
|
|
|
qemuMonitorTransactionBackupSyncMode syncmode)
|
|
|
|
{
|
|
|
|
const char *syncmodestr = qemuMonitorTransactionBackupSyncModeTypeToString(syncmode);
|
|
|
|
|
|
|
|
return qemuMonitorJSONTransactionAdd(actions,
|
|
|
|
"blockdev-backup",
|
|
|
|
"s:device", device,
|
|
|
|
"s:job-id", jobname,
|
|
|
|
"s:target", target,
|
|
|
|
"s:sync", syncmodestr,
|
|
|
|
"S:bitmap", bitmap,
|
|
|
|
"T:auto-finalize", VIR_TRISTATE_BOOL_YES,
|
|
|
|
"T:auto-dismiss", VIR_TRISTATE_BOOL_NO,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
|
2019-09-26 16:03:46 +02:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static qemuMonitorJobInfo *
|
|
|
|
qemuMonitorJSONGetJobInfoOne(virJSONValue *data)
|
2018-12-04 17:58:38 +01:00
|
|
|
{
|
|
|
|
const char *id = virJSONValueObjectGetString(data, "id");
|
|
|
|
const char *type = virJSONValueObjectGetString(data, "type");
|
|
|
|
const char *status = virJSONValueObjectGetString(data, "status");
|
|
|
|
const char *errmsg = virJSONValueObjectGetString(data, "error");
|
|
|
|
int tmp;
|
2019-10-15 14:47:50 +02:00
|
|
|
g_autoptr(qemuMonitorJobInfo) job = NULL;
|
2018-12-04 17:58:38 +01:00
|
|
|
|
2020-10-05 12:26:34 +02:00
|
|
|
job = g_new0(qemuMonitorJobInfo, 1);
|
2018-12-04 17:58:38 +01:00
|
|
|
|
|
|
|
if ((tmp = qemuMonitorJobTypeFromString(type)) < 0)
|
|
|
|
tmp = QEMU_MONITOR_JOB_TYPE_UNKNOWN;
|
|
|
|
|
|
|
|
job->type = tmp;
|
|
|
|
|
|
|
|
if ((tmp = qemuMonitorJobStatusTypeFromString(status)) < 0)
|
|
|
|
tmp = QEMU_MONITOR_JOB_STATUS_UNKNOWN;
|
|
|
|
|
|
|
|
job->status = tmp;
|
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
job->id = g_strdup(id);
|
|
|
|
job->error = g_strdup(errmsg);
|
2018-12-04 17:58:38 +01:00
|
|
|
|
2019-11-25 21:28:14 +01:00
|
|
|
/* failure to fetch progress stats is not fatal */
|
|
|
|
ignore_value(virJSONValueObjectGetNumberUlong(data, "current-progress",
|
|
|
|
&job->progressCurrent));
|
|
|
|
ignore_value(virJSONValueObjectGetNumberUlong(data, "total-progress",
|
|
|
|
&job->progressTotal));
|
|
|
|
|
2019-10-17 10:10:10 +02:00
|
|
|
return g_steal_pointer(&job);
|
2018-12-04 17:58:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetJobInfo(qemuMonitor *mon,
|
|
|
|
qemuMonitorJobInfo ***jobs,
|
2018-12-04 17:58:38 +01:00
|
|
|
size_t *njobs)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *data;
|
2019-10-15 14:47:50 +02:00
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2018-12-04 17:58:38 +01:00
|
|
|
size_t i;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-jobs", NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_ARRAY) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
data = virJSONValueObjectGetArray(reply, "return");
|
|
|
|
|
|
|
|
for (i = 0; i < virJSONValueArraySize(data); i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJobInfo *job = NULL;
|
2018-12-04 17:58:38 +01:00
|
|
|
|
|
|
|
if (!(job = qemuMonitorJSONGetJobInfoOne(virJSONValueArrayGet(data, i))))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (VIR_APPEND_ELEMENT(*jobs, *njobs, job) < 0)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2020-07-15 22:33:07 +02:00
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONGetCPUMigratable(qemuMonitor *mon,
|
2020-07-15 22:33:07 +02:00
|
|
|
bool *migratable)
|
|
|
|
{
|
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("qom-get",
|
|
|
|
"s:path", QOM_CPU_PATH,
|
|
|
|
"s:property", "migratable",
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONHasError(reply, "GenericError"))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckReply(cmd, reply, VIR_JSON_TYPE_BOOLEAN) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return virJSONValueGetBoolean(virJSONValueObjectGet(reply, "return"),
|
|
|
|
migratable);
|
|
|
|
}
|
2021-03-16 20:32:46 +08:00
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONStartDirtyRateCalc(qemuMonitor *mon,
|
2021-03-16 20:32:46 +08:00
|
|
|
int seconds)
|
|
|
|
{
|
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("calc-dirty-rate",
|
|
|
|
"i:calc-time", seconds,
|
|
|
|
NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2021-03-16 20:32:49 +08:00
|
|
|
|
|
|
|
VIR_ENUM_DECL(qemuMonitorDirtyRateStatus);
|
|
|
|
VIR_ENUM_IMPL(qemuMonitorDirtyRateStatus,
|
|
|
|
VIR_DOMAIN_DIRTYRATE_LAST,
|
|
|
|
"unstarted",
|
|
|
|
"measuring",
|
|
|
|
"measured");
|
|
|
|
|
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONExtractDirtyRateInfo(virJSONValue *data,
|
|
|
|
qemuMonitorDirtyRateInfo *info)
|
2021-03-16 20:32:49 +08:00
|
|
|
{
|
|
|
|
const char *statusstr;
|
|
|
|
int status;
|
|
|
|
|
|
|
|
if (!(statusstr = virJSONValueObjectGetString(data, "status"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-dirty-rate reply was missing 'status' data"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((status = qemuMonitorDirtyRateStatusTypeFromString(statusstr)) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Unknown dirty rate status: %s"), statusstr);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
info->status = status;
|
|
|
|
|
|
|
|
/* `query-dirty-rate` replies `dirty-rate` data only if the status of the latest
|
|
|
|
* calculation is `measured`.
|
|
|
|
*/
|
|
|
|
if ((info->status == VIR_DOMAIN_DIRTYRATE_MEASURED) &&
|
|
|
|
(virJSONValueObjectGetNumberLong(data, "dirty-rate", &info->dirtyRate) < 0)) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-dirty-rate reply was missing 'dirty-rate' data"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberLong(data, "start-time", &info->startTime) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-dirty-rate reply was missing 'start-time' data"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectGetNumberInt(data, "calc-time", &info->calcTime) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-dirty-rate reply was missing 'calc-time' data"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuMonitorJSONQueryDirtyRate(qemuMonitor *mon,
|
|
|
|
qemuMonitorDirtyRateInfo *info)
|
2021-03-16 20:32:49 +08:00
|
|
|
{
|
|
|
|
g_autoptr(virJSONValue) cmd = NULL;
|
|
|
|
g_autoptr(virJSONValue) reply = NULL;
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *data = NULL;
|
2021-03-16 20:32:49 +08:00
|
|
|
|
|
|
|
if (!(cmd = qemuMonitorJSONMakeCommand("query-dirty-rate", NULL)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCommand(mon, cmd, &reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuMonitorJSONCheckError(cmd, reply) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (!(data = virJSONValueObjectGetObject(reply, "return"))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("query-dirty-rate reply was missing 'return' data"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return qemuMonitorJSONExtractDirtyRateInfo(data, info);
|
|
|
|
}
|