2011-02-21 14:40:10 +01:00
|
|
|
/*
|
|
|
|
* xen_xm.c: Xen XM parsing functions
|
|
|
|
*
|
2014-03-24 12:14:48 -06:00
|
|
|
* Copyright (C) 2006-2007, 2009-2014 Red Hat, Inc.
|
2011-02-21 14:40:10 +01:00
|
|
|
* Copyright (C) 2011 Univention GmbH
|
|
|
|
* 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/>.
|
2011-02-21 14:40:10 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include "internal.h"
|
2012-12-13 18:21:53 +00:00
|
|
|
#include "virerror.h"
|
2012-12-12 16:35:35 +00:00
|
|
|
#include "virconf.h"
|
2012-12-12 18:06:53 +00:00
|
|
|
#include "viralloc.h"
|
2011-02-21 14:40:10 +01:00
|
|
|
#include "xenxs_private.h"
|
|
|
|
#include "xen_xm.h"
|
2011-04-14 18:05:14 +02:00
|
|
|
#include "domain_conf.h"
|
2022-07-07 10:03:33 +02:00
|
|
|
#include "domain_postparse.h"
|
2014-08-15 20:52:15 -06:00
|
|
|
#include "xen_common.h"
|
2014-08-07 21:32:54 +03:00
|
|
|
|
2016-01-20 11:36:26 -07:00
|
|
|
#define VIR_FROM_THIS VIR_FROM_XENXM
|
2014-08-07 21:32:54 +03:00
|
|
|
|
2015-03-19 18:28:49 -06:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
xenParseXMOS(virConf *conf, virDomainDef *def)
|
2015-03-19 18:28:49 -06:00
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
2015-04-16 20:11:06 -04:00
|
|
|
if (def->os.type == VIR_DOMAIN_OSTYPE_HVM) {
|
2019-10-15 15:16:31 +02:00
|
|
|
g_autofree char *boot = NULL;
|
2015-03-19 18:28:49 -06:00
|
|
|
|
2023-01-31 17:05:44 +01:00
|
|
|
def->os.loader = virDomainLoaderDefNew();
|
2023-05-16 19:50:50 +02:00
|
|
|
def->os.loader->format = VIR_STORAGE_FILE_RAW;
|
2020-10-06 16:45:20 +02:00
|
|
|
|
|
|
|
if (xenConfigCopyString(conf, "kernel", &def->os.loader->path) < 0)
|
2015-03-19 18:28:49 -06:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (xenConfigGetString(conf, "boot", &boot, "c") < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
for (i = 0; i < VIR_DOMAIN_BOOT_LAST && boot[i]; i++) {
|
|
|
|
switch (boot[i]) {
|
|
|
|
case 'a':
|
|
|
|
def->os.bootDevs[i] = VIR_DOMAIN_BOOT_FLOPPY;
|
|
|
|
break;
|
|
|
|
case 'd':
|
|
|
|
def->os.bootDevs[i] = VIR_DOMAIN_BOOT_CDROM;
|
|
|
|
break;
|
|
|
|
case 'n':
|
|
|
|
def->os.bootDevs[i] = VIR_DOMAIN_BOOT_NET;
|
|
|
|
break;
|
|
|
|
case 'c':
|
|
|
|
default:
|
|
|
|
def->os.bootDevs[i] = VIR_DOMAIN_BOOT_DISK;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
def->os.nBootDevs++;
|
|
|
|
}
|
|
|
|
} else {
|
2019-10-15 15:16:31 +02:00
|
|
|
g_autofree char *extra = NULL;
|
|
|
|
g_autofree char *root = NULL;
|
2015-03-19 18:28:49 -06:00
|
|
|
|
|
|
|
if (xenConfigCopyStringOpt(conf, "bootloader", &def->os.bootloader) < 0)
|
|
|
|
return -1;
|
|
|
|
if (xenConfigCopyStringOpt(conf, "bootargs", &def->os.bootloaderArgs) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (xenConfigCopyStringOpt(conf, "kernel", &def->os.kernel) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (xenConfigCopyStringOpt(conf, "ramdisk", &def->os.initrd) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (xenConfigGetString(conf, "extra", &extra, NULL) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (xenConfigGetString(conf, "root", &root, NULL) < 0)
|
|
|
|
return -1;
|
|
|
|
|
2018-05-12 17:45:02 +02:00
|
|
|
if (root && extra) {
|
2019-10-22 15:26:14 +02:00
|
|
|
def->os.cmdline = g_strdup_printf("root=%s %s", root, extra);
|
2018-05-12 17:45:02 +02:00
|
|
|
} else if (root) {
|
2019-10-22 15:26:14 +02:00
|
|
|
def->os.cmdline = g_strdup_printf("root=%s", root);
|
2018-05-12 17:45:02 +02:00
|
|
|
} else if (extra) {
|
2019-10-20 13:49:46 +02:00
|
|
|
def->os.cmdline = g_strdup(extra);
|
2015-03-19 18:28:49 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static virDomainDiskDef *
|
2018-05-28 00:28:20 +02:00
|
|
|
xenParseXMDisk(char *entry, int hvm)
|
2014-03-18 09:14:16 +01:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainDiskDef *disk = NULL;
|
2018-05-28 00:28:20 +02:00
|
|
|
char *head;
|
|
|
|
char *offset;
|
|
|
|
char *tmp;
|
|
|
|
const char *src;
|
2011-02-21 14:40:10 +01:00
|
|
|
|
2018-05-28 00:28:20 +02:00
|
|
|
if (!(disk = virDomainDiskDefNew(NULL)))
|
|
|
|
return NULL;
|
2011-02-21 14:40:10 +01:00
|
|
|
|
2018-05-28 00:28:20 +02:00
|
|
|
head = entry;
|
|
|
|
/*
|
|
|
|
* Disks have 3 components, SOURCE,DEST-DEVICE,MODE
|
|
|
|
* eg, phy:/dev/HostVG/XenGuest1,xvda,w
|
|
|
|
* The SOURCE is usually prefixed with a driver type,
|
|
|
|
* and optionally driver sub-type
|
|
|
|
* The DEST-DEVICE is optionally post-fixed with disk type
|
|
|
|
*/
|
|
|
|
|
2020-01-24 21:30:04 +01:00
|
|
|
/* Extract the source file path */
|
2018-05-28 00:28:20 +02:00
|
|
|
if (!(offset = strchr(head, ',')))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (offset == head) {
|
|
|
|
/* No source file given, eg CDROM with no media */
|
2020-11-05 01:23:41 -05:00
|
|
|
virDomainDiskSetSource(disk, NULL);
|
2018-05-28 00:28:20 +02:00
|
|
|
} else {
|
2019-10-24 19:41:34 +02:00
|
|
|
tmp = g_strndup(head, offset - head);
|
2011-02-21 14:40:10 +01:00
|
|
|
|
2020-11-05 01:23:41 -05:00
|
|
|
virDomainDiskSetSource(disk, tmp);
|
2018-05-28 00:28:20 +02:00
|
|
|
VIR_FREE(tmp);
|
|
|
|
}
|
|
|
|
|
|
|
|
head = offset + 1;
|
|
|
|
/* Remove legacy ioemu: junk */
|
|
|
|
if (STRPREFIX(head, "ioemu:"))
|
|
|
|
head = head + 6;
|
|
|
|
|
|
|
|
/* Extract the dest device name */
|
|
|
|
if (!(offset = strchr(head, ',')))
|
|
|
|
goto error;
|
|
|
|
|
2021-03-02 12:08:28 +01:00
|
|
|
disk->dst = g_strndup(head, offset - head);
|
2011-02-21 14:40:10 +01:00
|
|
|
|
2018-05-28 00:28:20 +02:00
|
|
|
head = offset + 1;
|
|
|
|
/* Extract source driver type */
|
|
|
|
src = virDomainDiskGetSource(disk);
|
|
|
|
if (src) {
|
|
|
|
size_t len;
|
|
|
|
/* The main type phy:, file:, tap: ... */
|
|
|
|
if ((tmp = strchr(src, ':')) != NULL) {
|
|
|
|
len = tmp - src;
|
2019-10-24 23:34:40 +02:00
|
|
|
tmp = g_strndup(src, len);
|
2018-05-28 00:28:20 +02:00
|
|
|
|
2020-11-05 22:32:37 -05:00
|
|
|
virDomainDiskSetDriver(disk, tmp);
|
2018-05-28 00:28:20 +02:00
|
|
|
VIR_FREE(tmp);
|
2011-02-21 14:40:10 +01:00
|
|
|
|
2018-05-28 00:28:20 +02:00
|
|
|
/* Strip the prefix we found off the source file name */
|
2020-11-05 01:23:41 -05:00
|
|
|
virDomainDiskSetSource(disk, src + len + 1);
|
2011-02-21 14:40:10 +01:00
|
|
|
|
2018-05-28 00:28:20 +02:00
|
|
|
src = virDomainDiskGetSource(disk);
|
|
|
|
}
|
2014-08-07 21:32:55 +03:00
|
|
|
|
2018-05-28 00:28:20 +02:00
|
|
|
/* And the sub-type for tap:XXX: type */
|
|
|
|
if (STREQ_NULLABLE(virDomainDiskGetDriver(disk), "tap") ||
|
|
|
|
STREQ_NULLABLE(virDomainDiskGetDriver(disk), "tap2")) {
|
|
|
|
char *driverType;
|
2014-08-07 21:32:55 +03:00
|
|
|
|
2018-05-28 00:28:20 +02:00
|
|
|
if (!(tmp = strchr(src, ':')))
|
|
|
|
goto error;
|
|
|
|
len = tmp - src;
|
|
|
|
|
2019-10-24 23:34:40 +02:00
|
|
|
driverType = g_strndup(src, len);
|
2018-05-28 00:28:20 +02:00
|
|
|
|
|
|
|
if (STREQ(driverType, "aio"))
|
|
|
|
virDomainDiskSetFormat(disk, VIR_STORAGE_FILE_RAW);
|
|
|
|
else
|
|
|
|
virDomainDiskSetFormat(disk,
|
|
|
|
virStorageFileFormatTypeFromString(driverType));
|
|
|
|
VIR_FREE(driverType);
|
|
|
|
if (virDomainDiskGetFormat(disk) <= 0) {
|
2012-07-18 14:17:20 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
2023-03-09 12:11:59 +01:00
|
|
|
_("Unknown driver type %1$s"),
|
2018-05-28 00:28:20 +02:00
|
|
|
src);
|
|
|
|
goto error;
|
2011-02-21 14:40:10 +01:00
|
|
|
}
|
|
|
|
|
2018-05-28 00:28:20 +02:00
|
|
|
/* Strip the prefix we found off the source file name */
|
2020-11-05 01:23:41 -05:00
|
|
|
virDomainDiskSetSource(disk, src + len + 1);
|
2014-03-24 12:14:48 -06:00
|
|
|
src = virDomainDiskGetSource(disk);
|
2018-05-28 00:28:20 +02:00
|
|
|
}
|
|
|
|
}
|
2011-02-21 14:40:10 +01:00
|
|
|
|
2018-05-28 00:28:20 +02:00
|
|
|
/* No source, or driver name, so fix to phy: */
|
2020-11-05 22:32:37 -05:00
|
|
|
if (!virDomainDiskGetDriver(disk))
|
|
|
|
virDomainDiskSetDriver(disk, "phy");
|
2018-05-28 00:28:20 +02:00
|
|
|
|
|
|
|
/* phy: type indicates a block device */
|
|
|
|
virDomainDiskSetType(disk,
|
|
|
|
STREQ(virDomainDiskGetDriver(disk), "phy") ?
|
|
|
|
VIR_STORAGE_TYPE_BLOCK :
|
|
|
|
VIR_STORAGE_TYPE_FILE);
|
|
|
|
|
|
|
|
/* Check for a :cdrom/:disk postfix */
|
|
|
|
disk->device = VIR_DOMAIN_DISK_DEVICE_DISK;
|
|
|
|
if ((tmp = strchr(disk->dst, ':')) != NULL) {
|
|
|
|
if (STREQ(tmp, ":cdrom"))
|
|
|
|
disk->device = VIR_DOMAIN_DISK_DEVICE_CDROM;
|
|
|
|
tmp[0] = '\0';
|
|
|
|
}
|
2011-02-21 14:40:10 +01:00
|
|
|
|
2018-05-28 00:28:20 +02:00
|
|
|
if (STRPREFIX(disk->dst, "xvd") || !hvm)
|
|
|
|
disk->bus = VIR_DOMAIN_DISK_BUS_XEN;
|
|
|
|
else if (STRPREFIX(disk->dst, "sd"))
|
|
|
|
disk->bus = VIR_DOMAIN_DISK_BUS_SCSI;
|
|
|
|
else
|
|
|
|
disk->bus = VIR_DOMAIN_DISK_BUS_IDE;
|
2011-02-21 14:40:10 +01:00
|
|
|
|
2018-05-28 00:28:20 +02:00
|
|
|
if (STREQ(head, "r") || STREQ(head, "ro"))
|
|
|
|
disk->src->readonly = true;
|
|
|
|
else if (STREQ(head, "w!") || STREQ(head, "!"))
|
|
|
|
disk->src->shared = true;
|
2011-02-21 14:40:10 +01:00
|
|
|
|
2018-05-28 00:28:20 +02:00
|
|
|
return disk;
|
2011-02-21 14:40:10 +01:00
|
|
|
|
2018-05-28 00:28:20 +02:00
|
|
|
error:
|
|
|
|
virDomainDiskDefFree(disk);
|
|
|
|
return NULL;
|
|
|
|
}
|
2011-02-21 14:40:10 +01:00
|
|
|
|
2018-05-28 00:28:20 +02:00
|
|
|
|
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
xenParseXMDiskList(virConf *conf, virDomainDef *def)
|
2018-05-28 00:28:20 +02:00
|
|
|
{
|
2021-08-03 15:08:54 +02:00
|
|
|
g_auto(GStrv) disks = NULL;
|
|
|
|
GStrv entries;
|
2018-05-28 00:28:20 +02:00
|
|
|
int hvm = def->os.type == VIR_DOMAIN_OSTYPE_HVM;
|
2018-05-28 00:28:21 +02:00
|
|
|
int rc;
|
2018-05-28 00:28:20 +02:00
|
|
|
|
2018-05-28 00:28:21 +02:00
|
|
|
rc = virConfGetValueStringList(conf, "disk", false, &disks);
|
|
|
|
if (rc <= 0)
|
|
|
|
return rc;
|
2018-05-28 00:28:20 +02:00
|
|
|
|
2018-05-28 00:28:21 +02:00
|
|
|
for (entries = disks; *entries; entries++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainDiskDef *disk;
|
2018-05-28 00:28:21 +02:00
|
|
|
char *entry = *entries;
|
2018-05-28 00:28:20 +02:00
|
|
|
|
2018-05-28 00:28:21 +02:00
|
|
|
if (!(disk = xenParseXMDisk(entry, hvm)))
|
2018-05-28 00:28:20 +02:00
|
|
|
continue;
|
|
|
|
|
|
|
|
/* Maintain list in sorted order according to target device name */
|
2021-08-03 14:14:20 +02:00
|
|
|
VIR_APPEND_ELEMENT(def->disks, def->ndisks, disk);
|
2011-02-21 14:40:10 +01:00
|
|
|
}
|
|
|
|
|
2021-08-03 15:08:54 +02:00
|
|
|
return 0;
|
2014-08-07 21:32:55 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-08-07 21:32:56 +03:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
xenFormatXMDisk(virConfValue *list,
|
|
|
|
virDomainDiskDef *disk)
|
2014-08-07 21:32:56 +03:00
|
|
|
{
|
2020-07-02 18:21:30 -04:00
|
|
|
g_auto(virBuffer) buf = VIR_BUFFER_INITIALIZER;
|
2021-03-11 08:16:13 +01:00
|
|
|
virConfValue *val;
|
|
|
|
virConfValue *tmp;
|
2014-08-15 20:52:15 -06:00
|
|
|
const char *src = virDomainDiskGetSource(disk);
|
|
|
|
int format = virDomainDiskGetFormat(disk);
|
|
|
|
const char *driver = virDomainDiskGetDriver(disk);
|
2014-08-07 21:32:56 +03:00
|
|
|
|
2014-08-15 20:52:15 -06:00
|
|
|
if (src) {
|
|
|
|
if (format) {
|
|
|
|
const char *type;
|
2014-08-07 21:32:56 +03:00
|
|
|
|
2014-08-15 20:52:15 -06:00
|
|
|
if (format == VIR_STORAGE_FILE_RAW)
|
|
|
|
type = "aio";
|
|
|
|
else
|
|
|
|
type = virStorageFileFormatTypeToString(format);
|
2016-05-24 11:56:08 +01:00
|
|
|
|
|
|
|
if (driver) {
|
|
|
|
virBufferAsprintf(&buf, "%s:", driver);
|
2016-05-19 08:24:55 +02:00
|
|
|
if (STREQ(driver, "tap") || STREQ(driver, "tap2"))
|
2016-05-24 11:56:08 +01:00
|
|
|
virBufferAsprintf(&buf, "%s:", type);
|
|
|
|
}
|
2014-08-07 21:32:56 +03:00
|
|
|
} else {
|
2014-08-15 20:52:15 -06:00
|
|
|
switch (virDomainDiskGetType(disk)) {
|
|
|
|
case VIR_STORAGE_TYPE_FILE:
|
|
|
|
virBufferAddLit(&buf, "file:");
|
|
|
|
break;
|
|
|
|
case VIR_STORAGE_TYPE_BLOCK:
|
|
|
|
virBufferAddLit(&buf, "phy:");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
2023-03-09 12:11:59 +01:00
|
|
|
_("unsupported disk type %1$s"),
|
2014-08-15 20:52:15 -06:00
|
|
|
virStorageTypeToString(virDomainDiskGetType(disk)));
|
2020-07-02 23:19:45 -04:00
|
|
|
return -1;
|
2014-08-07 21:32:56 +03:00
|
|
|
}
|
|
|
|
}
|
2014-08-15 20:52:15 -06:00
|
|
|
virBufferAdd(&buf, src, -1);
|
2014-08-07 21:32:56 +03:00
|
|
|
}
|
2014-08-15 20:52:15 -06:00
|
|
|
virBufferAddLit(&buf, ",");
|
2014-08-07 21:32:56 +03:00
|
|
|
|
2014-08-15 20:52:15 -06:00
|
|
|
virBufferAdd(&buf, disk->dst, -1);
|
|
|
|
if (disk->device == VIR_DOMAIN_DISK_DEVICE_CDROM)
|
|
|
|
virBufferAddLit(&buf, ":cdrom");
|
2014-08-07 21:32:56 +03:00
|
|
|
|
2014-08-15 20:52:15 -06:00
|
|
|
if (disk->src->readonly)
|
|
|
|
virBufferAddLit(&buf, ",r");
|
|
|
|
else if (disk->src->shared)
|
|
|
|
virBufferAddLit(&buf, ",!");
|
|
|
|
else
|
|
|
|
virBufferAddLit(&buf, ",w");
|
|
|
|
if (disk->transient) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("transient disks not supported yet"));
|
2014-08-12 00:21:28 +03:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2020-09-23 20:44:29 +02:00
|
|
|
val = g_new0(virConfValue, 1);
|
2014-08-15 20:52:15 -06:00
|
|
|
val->type = VIR_CONF_STRING;
|
|
|
|
val->str = virBufferContentAndReset(&buf);
|
|
|
|
tmp = list->list;
|
|
|
|
while (tmp && tmp->next)
|
|
|
|
tmp = tmp->next;
|
|
|
|
if (tmp)
|
|
|
|
tmp->next = val;
|
|
|
|
else
|
|
|
|
list->list = val;
|
2014-08-12 00:21:28 +03:00
|
|
|
|
|
|
|
return 0;
|
2014-08-12 00:21:29 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-08-12 00:21:30 +03:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
xenFormatXMDisks(virConf *conf, virDomainDef *def)
|
2014-08-12 00:21:30 +03:00
|
|
|
{
|
2022-01-14 10:28:52 +01:00
|
|
|
g_autoptr(virConfValue) diskVal = NULL;
|
2014-08-12 00:21:30 +03:00
|
|
|
size_t i = 0;
|
|
|
|
|
2020-09-23 20:43:09 +02:00
|
|
|
diskVal = g_new0(virConfValue, 1);
|
2014-08-12 00:21:30 +03:00
|
|
|
|
|
|
|
diskVal->type = VIR_CONF_LIST;
|
|
|
|
diskVal->list = NULL;
|
|
|
|
|
|
|
|
for (i = 0; i < def->ndisks; i++) {
|
|
|
|
if (def->disks[i]->device == VIR_DOMAIN_DISK_DEVICE_FLOPPY)
|
|
|
|
continue;
|
|
|
|
|
2015-12-04 19:11:39 -07:00
|
|
|
if (xenFormatXMDisk(diskVal, def->disks[i]) < 0)
|
2022-01-14 13:49:54 +01:00
|
|
|
return -1;
|
2014-08-12 00:21:30 +03:00
|
|
|
}
|
|
|
|
|
2022-01-14 10:11:42 +01:00
|
|
|
if (diskVal->list != NULL &&
|
|
|
|
virConfSetValue(conf, "disk", &diskVal) < 0)
|
2022-01-14 13:49:54 +01:00
|
|
|
return -1;
|
2014-08-12 00:21:30 +03:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-03-06 15:36:13 +01:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
xenParseXMInputDevs(virConf *conf, virDomainDef *def)
|
2015-03-06 15:36:13 +01:00
|
|
|
{
|
2019-10-15 15:16:31 +02:00
|
|
|
g_autofree char *str = NULL;
|
2015-03-06 15:36:13 +01:00
|
|
|
|
2015-04-16 20:11:06 -04:00
|
|
|
if (def->os.type == VIR_DOMAIN_OSTYPE_HVM) {
|
2015-03-06 15:36:13 +01:00
|
|
|
if (xenConfigGetString(conf, "usbdevice", &str, NULL) < 0)
|
|
|
|
return -1;
|
|
|
|
if (str &&
|
|
|
|
(STREQ(str, "tablet") ||
|
|
|
|
STREQ(str, "mouse") ||
|
|
|
|
STREQ(str, "keyboard"))) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainInputDef *input;
|
2020-09-23 20:43:09 +02:00
|
|
|
input = g_new0(virDomainInputDef, 1);
|
2015-03-06 15:36:13 +01:00
|
|
|
|
|
|
|
input->bus = VIR_DOMAIN_INPUT_BUS_USB;
|
|
|
|
if (STREQ(str, "mouse"))
|
|
|
|
input->type = VIR_DOMAIN_INPUT_TYPE_MOUSE;
|
|
|
|
else if (STREQ(str, "tablet"))
|
|
|
|
input->type = VIR_DOMAIN_INPUT_TYPE_TABLET;
|
|
|
|
else if (STREQ(str, "keyboard"))
|
|
|
|
input->type = VIR_DOMAIN_INPUT_TYPE_KBD;
|
2021-08-03 14:14:20 +02:00
|
|
|
VIR_APPEND_ELEMENT(def->inputs, def->ninputs, input);
|
2015-03-06 15:36:13 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-08-15 20:52:15 -06:00
|
|
|
/*
|
|
|
|
* Convert an XM config record into a virDomainDef object.
|
|
|
|
*/
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainDef *
|
|
|
|
xenParseXM(virConf *conf,
|
|
|
|
virCaps *caps,
|
|
|
|
virDomainXMLOption *xmlopt)
|
2014-08-12 00:21:34 +03:00
|
|
|
{
|
2021-11-29 09:07:44 +01:00
|
|
|
g_autoptr(virDomainDef) def = NULL;
|
2014-08-12 00:21:34 +03:00
|
|
|
|
2021-08-03 11:00:48 +02:00
|
|
|
if (!(def = virDomainDefNew(xmlopt)))
|
2014-08-15 20:52:15 -06:00
|
|
|
return NULL;
|
2014-08-12 00:21:34 +03:00
|
|
|
|
2014-08-15 20:52:15 -06:00
|
|
|
def->virtType = VIR_DOMAIN_VIRT_XEN;
|
|
|
|
def->id = -1;
|
2014-08-13 14:46:16 -06:00
|
|
|
|
2016-06-29 14:55:24 +02:00
|
|
|
if (xenParseConfigCommon(conf, def, caps, XEN_CONFIG_FORMAT_XM,
|
|
|
|
xmlopt) < 0)
|
2021-11-02 12:29:02 +01:00
|
|
|
return NULL;
|
2014-08-13 14:46:16 -06:00
|
|
|
|
2015-03-19 18:28:49 -06:00
|
|
|
if (xenParseXMOS(conf, def) < 0)
|
2021-11-02 12:29:02 +01:00
|
|
|
return NULL;
|
2015-03-19 18:28:49 -06:00
|
|
|
|
2018-05-28 00:28:20 +02:00
|
|
|
if (xenParseXMDiskList(conf, def) < 0)
|
2021-11-02 12:29:02 +01:00
|
|
|
return NULL;
|
2014-08-13 14:46:16 -06:00
|
|
|
|
2015-03-06 15:36:13 +01:00
|
|
|
if (xenParseXMInputDevs(conf, def) < 0)
|
2021-11-02 12:29:02 +01:00
|
|
|
return NULL;
|
2015-03-06 15:36:13 +01:00
|
|
|
|
2019-11-27 12:08:10 +00:00
|
|
|
if (virDomainDefPostParse(def, VIR_DOMAIN_DEF_PARSE_ABI_UPDATE,
|
2016-09-22 16:41:33 +02:00
|
|
|
xmlopt, NULL) < 0)
|
2021-11-02 12:29:02 +01:00
|
|
|
return NULL;
|
2015-11-28 05:33:55 +01:00
|
|
|
|
2021-11-29 09:07:44 +01:00
|
|
|
return g_steal_pointer(&def);
|
2014-08-12 00:21:35 +03:00
|
|
|
}
|
|
|
|
|
2015-03-19 18:28:49 -06:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
xenFormatXMOS(virConf *conf, virDomainDef *def)
|
2015-03-19 18:28:49 -06:00
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
2015-04-16 20:11:06 -04:00
|
|
|
if (def->os.type == VIR_DOMAIN_OSTYPE_HVM) {
|
2015-03-19 18:28:49 -06:00
|
|
|
char boot[VIR_DOMAIN_BOOT_LAST+1];
|
|
|
|
if (xenConfigSetString(conf, "builder", "hvm") < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (def->os.loader && def->os.loader->path &&
|
|
|
|
xenConfigSetString(conf, "kernel", def->os.loader->path) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
for (i = 0; i < def->os.nBootDevs; i++) {
|
|
|
|
switch (def->os.bootDevs[i]) {
|
|
|
|
case VIR_DOMAIN_BOOT_FLOPPY:
|
|
|
|
boot[i] = 'a';
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_BOOT_CDROM:
|
|
|
|
boot[i] = 'd';
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_BOOT_NET:
|
|
|
|
boot[i] = 'n';
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_BOOT_DISK:
|
|
|
|
default:
|
|
|
|
boot[i] = 'c';
|
|
|
|
break;
|
2022-01-20 21:57:49 +01:00
|
|
|
case VIR_DOMAIN_BOOT_LAST:
|
|
|
|
break;
|
2015-03-19 18:28:49 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!def->os.nBootDevs) {
|
|
|
|
boot[0] = 'c';
|
|
|
|
boot[1] = '\0';
|
|
|
|
} else {
|
|
|
|
boot[def->os.nBootDevs] = '\0';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (xenConfigSetString(conf, "boot", boot) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
/* XXX floppy disks */
|
|
|
|
} else {
|
|
|
|
if (def->os.bootloader &&
|
|
|
|
xenConfigSetString(conf, "bootloader", def->os.bootloader) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (def->os.bootloaderArgs &&
|
|
|
|
xenConfigSetString(conf, "bootargs", def->os.bootloaderArgs) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (def->os.kernel &&
|
|
|
|
xenConfigSetString(conf, "kernel", def->os.kernel) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (def->os.initrd &&
|
|
|
|
xenConfigSetString(conf, "ramdisk", def->os.initrd) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (def->os.cmdline &&
|
|
|
|
xenConfigSetString(conf, "extra", def->os.cmdline) < 0)
|
|
|
|
return -1;
|
|
|
|
} /* !hvm */
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-03-06 15:36:13 +01:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
xenFormatXMInputDevs(virConf *conf, virDomainDef *def)
|
2015-03-06 15:36:13 +01:00
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
const char *devtype;
|
|
|
|
|
2015-04-16 20:11:06 -04:00
|
|
|
if (def->os.type == VIR_DOMAIN_OSTYPE_HVM) {
|
2015-03-06 15:36:13 +01:00
|
|
|
for (i = 0; i < def->ninputs; i++) {
|
|
|
|
if (def->inputs[i]->bus == VIR_DOMAIN_INPUT_BUS_USB) {
|
|
|
|
if (xenConfigSetInt(conf, "usb", 1) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
switch (def->inputs[i]->type) {
|
|
|
|
case VIR_DOMAIN_INPUT_TYPE_MOUSE:
|
|
|
|
devtype = "mouse";
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_INPUT_TYPE_TABLET:
|
|
|
|
devtype = "tablet";
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_INPUT_TYPE_KBD:
|
|
|
|
devtype = "keyboard";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (xenConfigSetString(conf, "usbdevice", devtype) < 0)
|
|
|
|
return -1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-08-12 00:21:35 +03:00
|
|
|
|
|
|
|
/* Computing the vcpu_avail bitmask works because MAX_VIRT_CPUS is
|
|
|
|
either 32, or 64 on a platform where long is big enough. */
|
2020-01-09 10:39:55 +00:00
|
|
|
G_STATIC_ASSERT(MAX_VIRT_CPUS <= sizeof(1UL) * CHAR_BIT);
|
2014-08-12 00:21:35 +03:00
|
|
|
|
2014-08-15 20:52:15 -06:00
|
|
|
/*
|
2015-03-19 16:53:00 +01:00
|
|
|
* Convert a virDomainDef object into an XM config record.
|
2014-08-15 20:52:15 -06:00
|
|
|
*/
|
2021-03-11 08:16:13 +01:00
|
|
|
virConf *
|
2014-08-12 00:21:35 +03:00
|
|
|
xenFormatXM(virConnectPtr conn,
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainDef *def)
|
2014-08-12 00:21:35 +03:00
|
|
|
{
|
2019-10-15 14:47:50 +02:00
|
|
|
g_autoptr(virConf) conf = NULL;
|
2014-08-12 00:21:35 +03:00
|
|
|
|
|
|
|
if (!(conf = virConfNew()))
|
2019-09-09 17:56:26 +02:00
|
|
|
return NULL;
|
2014-08-12 00:21:35 +03:00
|
|
|
|
2016-05-17 17:34:45 +08:00
|
|
|
if (xenFormatConfigCommon(conf, def, conn, XEN_CONFIG_FORMAT_XM) < 0)
|
2019-09-09 17:56:26 +02:00
|
|
|
return NULL;
|
2014-08-12 00:21:35 +03:00
|
|
|
|
2015-03-19 18:28:49 -06:00
|
|
|
if (xenFormatXMOS(conf, def) < 0)
|
2019-09-09 17:56:26 +02:00
|
|
|
return NULL;
|
2015-03-19 18:28:49 -06:00
|
|
|
|
2015-12-04 19:11:39 -07:00
|
|
|
if (xenFormatXMDisks(conf, def) < 0)
|
2019-09-09 17:56:26 +02:00
|
|
|
return NULL;
|
2011-02-21 14:40:11 +01:00
|
|
|
|
2015-03-06 15:36:13 +01:00
|
|
|
if (xenFormatXMInputDevs(conf, def) < 0)
|
2019-09-09 17:56:26 +02:00
|
|
|
return NULL;
|
2011-02-21 14:40:11 +01:00
|
|
|
|
2019-10-16 13:35:54 +02:00
|
|
|
return g_steal_pointer(&conf);
|
2011-02-25 15:41:12 +01:00
|
|
|
}
|