2016-05-31 17:35:14 -04:00
|
|
|
/*
|
|
|
|
* virqemu.c: utilities for working with qemu and its tools
|
|
|
|
*
|
|
|
|
* Copyright (C) 2016 Red Hat, Inc.
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
* License along with this library. If not, see
|
|
|
|
* <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include "virbuffer.h"
|
|
|
|
#include "virerror.h"
|
|
|
|
#include "virlog.h"
|
|
|
|
#include "virqemu.h"
|
2016-07-22 17:19:28 +02:00
|
|
|
#include "virstring.h"
|
|
|
|
#include "viralloc.h"
|
2021-04-08 13:35:33 +02:00
|
|
|
#include "virbitmap.h"
|
2016-05-31 17:35:14 -04:00
|
|
|
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_NONE
|
|
|
|
|
|
|
|
VIR_LOG_INIT("util.qemu");
|
|
|
|
|
2016-07-22 17:19:28 +02:00
|
|
|
struct virQEMUCommandLineJSONIteratorData {
|
|
|
|
const char *prefix;
|
2021-03-11 08:16:13 +01:00
|
|
|
virBuffer *buf;
|
2020-05-14 09:41:48 +02:00
|
|
|
const char *skipKey;
|
2016-07-22 17:50:03 +02:00
|
|
|
virQEMUBuildCommandLineJSONArrayFormatFunc arrayFunc;
|
2016-07-22 17:19:28 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
virQEMUBuildCommandLineJSONRecurse(const char *key,
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *value,
|
|
|
|
virBuffer *buf,
|
2020-05-14 09:41:48 +02:00
|
|
|
const char *skipKey,
|
2016-07-22 17:50:03 +02:00
|
|
|
virQEMUBuildCommandLineJSONArrayFormatFunc arrayFunc,
|
2016-07-22 17:19:28 +02:00
|
|
|
bool nested);
|
|
|
|
|
2016-07-22 17:50:03 +02:00
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
virQEMUBuildCommandLineJSONArrayBitmap(const char *key,
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *array,
|
|
|
|
virBuffer *buf,
|
2021-02-16 12:36:15 +00:00
|
|
|
const char *skipKey G_GNUC_UNUSED)
|
2016-07-22 17:50:03 +02:00
|
|
|
{
|
|
|
|
ssize_t pos = -1;
|
|
|
|
ssize_t end;
|
2021-04-08 13:35:33 +02:00
|
|
|
g_autoptr(virBitmap) bitmap = virBitmapNew(0);
|
|
|
|
size_t i;
|
2016-07-22 17:50:03 +02:00
|
|
|
|
2021-04-08 13:35:33 +02:00
|
|
|
for (i = 0; i < virJSONValueArraySize(array); i++) {
|
|
|
|
virJSONValue *member = virJSONValueArrayGet(array, i);
|
|
|
|
unsigned long long value;
|
|
|
|
|
|
|
|
if (virJSONValueGetNumberUlong(member, &value) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (virBitmapSetBitExpand(bitmap, value) < 0)
|
|
|
|
return -1;
|
|
|
|
}
|
2016-07-22 17:50:03 +02:00
|
|
|
|
|
|
|
while ((pos = virBitmapNextSetBit(bitmap, pos)) > -1) {
|
|
|
|
if ((end = virBitmapNextClearBit(bitmap, pos)) < 0)
|
|
|
|
end = virBitmapLastSetBit(bitmap) + 1;
|
|
|
|
|
|
|
|
if (end - 1 > pos) {
|
2016-07-25 14:59:19 +02:00
|
|
|
virBufferAsprintf(buf, "%s=%zd-%zd,", key, pos, end - 1);
|
2016-07-22 17:50:03 +02:00
|
|
|
pos = end;
|
|
|
|
} else {
|
2016-07-25 14:59:19 +02:00
|
|
|
virBufferAsprintf(buf, "%s=%zd,", key, pos);
|
2016-07-22 17:50:03 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-07-25 19:47:40 +02:00
|
|
|
int
|
|
|
|
virQEMUBuildCommandLineJSONArrayNumbered(const char *key,
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *array,
|
|
|
|
virBuffer *buf,
|
2021-02-16 12:36:15 +00:00
|
|
|
const char *skipKey)
|
2016-07-25 19:47:40 +02:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *member;
|
2016-07-25 19:47:40 +02:00
|
|
|
size_t i;
|
|
|
|
|
2016-10-10 16:08:39 -04:00
|
|
|
for (i = 0; i < virJSONValueArraySize(array); i++) {
|
2019-10-15 15:16:31 +02:00
|
|
|
g_autofree char *prefix = NULL;
|
2016-07-25 19:47:40 +02:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
member = virJSONValueArrayGet((virJSONValue *) array, i);
|
2019-10-22 15:26:14 +02:00
|
|
|
prefix = g_strdup_printf("%s.%zu", key, i);
|
2016-07-25 19:47:40 +02:00
|
|
|
|
2021-02-16 12:36:15 +00:00
|
|
|
if (virQEMUBuildCommandLineJSONRecurse(prefix, member, buf, skipKey,
|
2016-07-25 19:47:40 +02:00
|
|
|
virQEMUBuildCommandLineJSONArrayNumbered,
|
|
|
|
true) < 0)
|
2018-07-28 23:31:46 +05:30
|
|
|
return 0;
|
2016-07-25 19:47:40 +02:00
|
|
|
}
|
|
|
|
|
2018-07-28 23:31:46 +05:30
|
|
|
return 0;
|
2016-07-25 19:47:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-05-15 10:59:40 +02:00
|
|
|
/**
|
2020-07-09 12:42:21 +08:00
|
|
|
* This array converter is for quirky cases where the QMP schema mandates an
|
2020-05-15 10:59:40 +02:00
|
|
|
* array of objects with only one attribute 'str' which needs to be formatted as
|
|
|
|
* repeated key-value pairs without the 'str' being printed:
|
|
|
|
*
|
|
|
|
* 'guestfwd': [
|
|
|
|
* { "str": "tcp:10.0.2.1:4600-chardev:charchannel0" },
|
|
|
|
* { "str": "...."},
|
|
|
|
* ]
|
|
|
|
*
|
|
|
|
* guestfwd=tcp:10.0.2.1:4600-chardev:charchannel0,guestfwd=...
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
virQEMUBuildCommandLineJSONArrayObjectsStr(const char *key,
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *array,
|
|
|
|
virBuffer *buf,
|
2021-02-16 12:36:15 +00:00
|
|
|
const char *skipKey G_GNUC_UNUSED)
|
2020-05-15 10:59:40 +02:00
|
|
|
{
|
|
|
|
g_auto(virBuffer) tmp = VIR_BUFFER_INITIALIZER;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < virJSONValueArraySize(array); i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *member = virJSONValueArrayGet(array, i);
|
2020-05-15 10:59:40 +02:00
|
|
|
const char *str = virJSONValueObjectGetString(member, "str");
|
|
|
|
|
|
|
|
if (!str)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
virBufferAsprintf(&tmp, "%s=%s,", key, str);
|
|
|
|
}
|
|
|
|
|
|
|
|
virBufferAddBuffer(buf, &tmp);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-07-22 17:19:28 +02:00
|
|
|
/* internal iterator to handle nested object formatting */
|
|
|
|
static int
|
|
|
|
virQEMUBuildCommandLineJSONIterate(const char *key,
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *value,
|
2016-07-22 17:19:28 +02:00
|
|
|
void *opaque)
|
|
|
|
{
|
|
|
|
struct virQEMUCommandLineJSONIteratorData *data = opaque;
|
|
|
|
|
2020-05-14 09:41:48 +02:00
|
|
|
if (STREQ_NULLABLE(key, data->skipKey))
|
|
|
|
return 0;
|
|
|
|
|
2016-07-22 17:19:28 +02:00
|
|
|
if (data->prefix) {
|
2019-10-15 15:16:31 +02:00
|
|
|
g_autofree char *tmpkey = NULL;
|
2018-07-28 23:31:46 +05:30
|
|
|
|
2019-10-22 15:26:14 +02:00
|
|
|
tmpkey = g_strdup_printf("%s.%s", data->prefix, key);
|
2016-07-22 17:19:28 +02:00
|
|
|
|
2018-07-28 23:31:46 +05:30
|
|
|
return virQEMUBuildCommandLineJSONRecurse(tmpkey, value, data->buf,
|
2021-02-16 12:36:15 +00:00
|
|
|
data->skipKey,
|
2018-09-19 16:38:14 +08:00
|
|
|
data->arrayFunc, false);
|
2016-07-22 17:19:28 +02:00
|
|
|
} else {
|
2018-07-28 23:31:46 +05:30
|
|
|
return virQEMUBuildCommandLineJSONRecurse(key, value, data->buf,
|
2021-02-16 12:36:15 +00:00
|
|
|
data->skipKey,
|
2018-09-19 16:38:14 +08:00
|
|
|
data->arrayFunc, false);
|
2016-07-22 17:19:28 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-31 17:35:14 -04:00
|
|
|
|
|
|
|
static int
|
2016-07-22 15:54:57 +02:00
|
|
|
virQEMUBuildCommandLineJSONRecurse(const char *key,
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *value,
|
|
|
|
virBuffer *buf,
|
2020-05-14 09:41:48 +02:00
|
|
|
const char *skipKey,
|
2016-07-22 17:50:03 +02:00
|
|
|
virQEMUBuildCommandLineJSONArrayFormatFunc arrayFunc,
|
2016-07-22 15:54:57 +02:00
|
|
|
bool nested)
|
2016-05-31 17:35:14 -04:00
|
|
|
{
|
2021-02-16 12:36:15 +00:00
|
|
|
struct virQEMUCommandLineJSONIteratorData data = { key, buf, skipKey, arrayFunc };
|
2018-03-29 20:30:05 +02:00
|
|
|
virJSONType type = virJSONValueGetType(value);
|
2021-03-11 08:16:13 +01:00
|
|
|
virJSONValue *elem;
|
2018-03-29 20:41:07 +02:00
|
|
|
bool tmp;
|
2016-05-31 17:35:14 -04:00
|
|
|
size_t i;
|
|
|
|
|
2018-03-29 20:30:05 +02:00
|
|
|
if (!key && type != VIR_JSON_TYPE_OBJECT) {
|
2016-07-22 17:19:28 +02:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("only JSON objects can be top level"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2018-03-29 20:30:05 +02:00
|
|
|
switch (type) {
|
2016-05-31 17:35:14 -04:00
|
|
|
case VIR_JSON_TYPE_STRING:
|
2016-07-25 14:59:19 +02:00
|
|
|
virBufferAsprintf(buf, "%s=", key);
|
2018-03-29 20:41:07 +02:00
|
|
|
virQEMUBuildBufferEscapeComma(buf, virJSONValueGetString(value));
|
2016-07-25 14:59:19 +02:00
|
|
|
virBufferAddLit(buf, ",");
|
2016-05-31 17:35:14 -04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_JSON_TYPE_NUMBER:
|
2018-03-29 20:41:07 +02:00
|
|
|
virBufferAsprintf(buf, "%s=%s,", key, virJSONValueGetNumberString(value));
|
2016-05-31 17:35:14 -04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_JSON_TYPE_BOOLEAN:
|
2018-03-29 20:41:07 +02:00
|
|
|
virJSONValueGetBoolean(value, &tmp);
|
2021-02-16 12:36:15 +00:00
|
|
|
if (tmp)
|
|
|
|
virBufferAsprintf(buf, "%s=on,", key);
|
|
|
|
else
|
|
|
|
virBufferAsprintf(buf, "%s=off,", key);
|
2016-05-31 17:35:14 -04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_JSON_TYPE_ARRAY:
|
|
|
|
if (nested) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2016-07-22 15:54:57 +02:00
|
|
|
_("nested JSON array to commandline conversion is "
|
|
|
|
"not supported"));
|
2016-05-31 17:35:14 -04:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2021-02-16 12:36:15 +00:00
|
|
|
if (!arrayFunc || arrayFunc(key, value, buf, skipKey) < 0) {
|
2016-05-31 17:35:14 -04:00
|
|
|
/* fallback, treat the array as a non-bitmap, adding the key
|
|
|
|
* for each member */
|
|
|
|
for (i = 0; i < virJSONValueArraySize(value); i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
elem = virJSONValueArrayGet((virJSONValue *)value, i);
|
2016-05-31 17:35:14 -04:00
|
|
|
|
|
|
|
/* recurse to avoid duplicating code */
|
2020-05-14 09:41:48 +02:00
|
|
|
if (virQEMUBuildCommandLineJSONRecurse(key, elem, buf, skipKey,
|
2021-02-16 12:36:15 +00:00
|
|
|
arrayFunc, true) < 0)
|
2016-05-31 17:35:14 -04:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_JSON_TYPE_OBJECT:
|
2016-07-22 17:19:28 +02:00
|
|
|
if (virJSONValueObjectForeachKeyValue(value,
|
|
|
|
virQEMUBuildCommandLineJSONIterate,
|
|
|
|
&data) < 0)
|
|
|
|
return -1;
|
|
|
|
break;
|
|
|
|
|
2016-05-31 17:35:14 -04:00
|
|
|
case VIR_JSON_TYPE_NULL:
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2016-07-22 17:19:28 +02:00
|
|
|
_("NULL JSON type can't be converted to commandline"));
|
2016-05-31 17:35:14 -04:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-07-22 15:54:57 +02:00
|
|
|
/**
|
|
|
|
* virQEMUBuildCommandLineJSON:
|
|
|
|
* @value: json object containing the value
|
|
|
|
* @buf: otuput buffer
|
2020-05-14 09:41:48 +02:00
|
|
|
* @skipKey: name of key that will be handled separately by caller
|
2016-07-22 17:50:03 +02:00
|
|
|
* @arrayFunc: array formatter function to allow for different syntax
|
2016-07-22 15:54:57 +02:00
|
|
|
*
|
|
|
|
* Formats JSON value object into command line parameters suitable for use with
|
|
|
|
* qemu.
|
|
|
|
*
|
|
|
|
* Returns 0 on success -1 on error.
|
|
|
|
*/
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
virQEMUBuildCommandLineJSON(virJSONValue *value,
|
|
|
|
virBuffer *buf,
|
2020-05-14 09:41:48 +02:00
|
|
|
const char *skipKey,
|
2016-07-22 17:50:03 +02:00
|
|
|
virQEMUBuildCommandLineJSONArrayFormatFunc array)
|
2016-07-22 15:54:57 +02:00
|
|
|
{
|
2021-02-16 12:36:15 +00:00
|
|
|
if (virQEMUBuildCommandLineJSONRecurse(NULL, value, buf, skipKey, array, false) < 0)
|
2016-07-25 14:59:19 +02:00
|
|
|
return -1;
|
|
|
|
|
2020-02-02 20:26:38 +01:00
|
|
|
virBufferTrim(buf, ",");
|
2016-07-25 14:59:19 +02:00
|
|
|
|
|
|
|
return 0;
|
2016-05-31 17:35:14 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-05-14 10:10:16 +02:00
|
|
|
/**
|
|
|
|
* virQEMUBuildNetdevCommandlineFromJSON:
|
|
|
|
* @props: JSON properties describing a netdev
|
2020-05-15 14:24:21 +02:00
|
|
|
* @rawjson: don't transform to commandline args, but just stringify json
|
2020-05-14 10:10:16 +02:00
|
|
|
*
|
|
|
|
* Converts @props into arguments for -netdev including all the quirks and
|
|
|
|
* differences between the monitor and command line syntax.
|
2020-05-15 14:24:21 +02:00
|
|
|
*
|
|
|
|
* @rawjson is meant for testing of the schema in the xml2argvtest
|
2020-05-14 10:10:16 +02:00
|
|
|
*/
|
|
|
|
char *
|
2021-03-11 08:16:13 +01:00
|
|
|
virQEMUBuildNetdevCommandlineFromJSON(virJSONValue *props,
|
2020-05-15 14:24:21 +02:00
|
|
|
bool rawjson)
|
2020-05-14 10:10:16 +02:00
|
|
|
{
|
|
|
|
const char *type = virJSONValueObjectGetString(props, "type");
|
|
|
|
g_auto(virBuffer) buf = VIR_BUFFER_INITIALIZER;
|
|
|
|
|
2020-05-15 14:24:21 +02:00
|
|
|
if (rawjson)
|
|
|
|
return virJSONValueToString(props, false);
|
|
|
|
|
2020-05-14 10:10:16 +02:00
|
|
|
virBufferAsprintf(&buf, "%s,", type);
|
|
|
|
|
2021-02-16 12:36:15 +00:00
|
|
|
if (virQEMUBuildCommandLineJSON(props, &buf, "type",
|
2020-05-15 10:59:40 +02:00
|
|
|
virQEMUBuildCommandLineJSONArrayObjectsStr) < 0)
|
2020-05-14 10:10:16 +02:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
return virBufferContentAndReset(&buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-07-25 19:51:18 +02:00
|
|
|
char *
|
2021-03-11 08:16:13 +01:00
|
|
|
virQEMUBuildDriveCommandlineFromJSON(virJSONValue *srcdef)
|
2016-07-25 19:51:18 +02:00
|
|
|
{
|
2020-07-02 22:30:20 -04:00
|
|
|
g_auto(virBuffer) buf = VIR_BUFFER_INITIALIZER;
|
2016-07-25 19:51:18 +02:00
|
|
|
|
2021-02-16 12:36:15 +00:00
|
|
|
if (virQEMUBuildCommandLineJSON(srcdef, &buf, NULL,
|
2016-07-25 19:51:18 +02:00
|
|
|
virQEMUBuildCommandLineJSONArrayNumbered) < 0)
|
2020-07-02 23:20:00 -04:00
|
|
|
return NULL;
|
2016-07-25 19:51:18 +02:00
|
|
|
|
2020-07-02 23:20:00 -04:00
|
|
|
return virBufferContentAndReset(&buf);
|
2016-07-25 19:51:18 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-07-15 07:02:05 -04:00
|
|
|
/**
|
|
|
|
* virQEMUBuildBufferEscapeComma:
|
|
|
|
* @buf: buffer to append the escaped string
|
|
|
|
* @str: the string to escape
|
|
|
|
*
|
|
|
|
* qemu requires that any values passed on the command line which contain
|
|
|
|
* a ',' must escape it using an extra ',' as the escape character
|
|
|
|
*/
|
|
|
|
void
|
2021-03-11 08:16:13 +01:00
|
|
|
virQEMUBuildBufferEscapeComma(virBuffer *buf, const char *str)
|
2016-07-15 07:02:05 -04:00
|
|
|
{
|
|
|
|
virBufferEscape(buf, ',', ",", "%s", str);
|
|
|
|
}
|