2016-02-16 10:24:35 -05:00
|
|
|
/*
|
|
|
|
* qemu_alias.c: QEMU alias manipulation
|
|
|
|
*
|
|
|
|
* Copyright (C) 2006-2016 Red Hat, Inc.
|
|
|
|
* 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
|
|
|
|
* License along with this library. If not, see
|
|
|
|
* <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include "qemu_alias.h"
|
|
|
|
#include "virlog.h"
|
|
|
|
#include "virstring.h"
|
2020-02-16 22:59:28 +01:00
|
|
|
#include "virutil.h"
|
2016-02-16 10:24:35 -05:00
|
|
|
|
2016-06-29 13:34:00 -04:00
|
|
|
#define QEMU_DRIVE_HOST_PREFIX "drive-"
|
|
|
|
|
2016-02-16 10:24:35 -05:00
|
|
|
#define VIR_FROM_THIS VIR_FROM_QEMU
|
|
|
|
|
|
|
|
VIR_LOG_INIT("qemu.qemu_alias");
|
|
|
|
|
|
|
|
int
|
|
|
|
qemuDomainDeviceAliasIndex(const virDomainDeviceInfo *info,
|
|
|
|
const char *prefix)
|
|
|
|
{
|
|
|
|
int idx;
|
|
|
|
|
|
|
|
if (!info->alias)
|
|
|
|
return -1;
|
|
|
|
if (!STRPREFIX(info->alias, prefix))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (virStrToLong_i(info->alias + strlen(prefix), NULL, 10, &idx) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static ssize_t
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuGetNextChrDevIndex(virDomainDef *def,
|
|
|
|
virDomainChrDef *chr,
|
2016-02-16 10:24:35 -05:00
|
|
|
const char *prefix)
|
|
|
|
{
|
|
|
|
const virDomainChrDef **arrPtr;
|
|
|
|
size_t cnt;
|
|
|
|
size_t i;
|
|
|
|
ssize_t idx = 0;
|
|
|
|
const char *prefix2 = NULL;
|
|
|
|
|
|
|
|
if (chr->deviceType == VIR_DOMAIN_CHR_DEVICE_TYPE_CONSOLE)
|
|
|
|
prefix2 = "serial";
|
|
|
|
|
|
|
|
virDomainChrGetDomainPtrs(def, chr->deviceType, &arrPtr, &cnt);
|
|
|
|
|
|
|
|
for (i = 0; i < cnt; i++) {
|
|
|
|
ssize_t thisidx;
|
|
|
|
if (((thisidx = qemuDomainDeviceAliasIndex(&arrPtr[i]->info, prefix)) < 0) &&
|
|
|
|
(prefix2 &&
|
2017-10-20 13:24:41 +02:00
|
|
|
(thisidx = qemuDomainDeviceAliasIndex(&arrPtr[i]->info, prefix2)) < 0))
|
|
|
|
continue;
|
2016-02-16 10:24:35 -05:00
|
|
|
if (thisidx >= idx)
|
|
|
|
idx = thisidx + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAssignDeviceChrAlias(virDomainDef *def,
|
|
|
|
virDomainChrDef *chr,
|
2016-02-16 10:24:35 -05:00
|
|
|
ssize_t idx)
|
|
|
|
{
|
|
|
|
const char *prefix = NULL;
|
|
|
|
|
2017-10-14 10:16:20 -07:00
|
|
|
if (chr->info.alias)
|
|
|
|
return 0;
|
|
|
|
|
2018-04-25 14:42:34 +02:00
|
|
|
switch ((virDomainChrDeviceType)chr->deviceType) {
|
2016-02-16 10:24:35 -05:00
|
|
|
case VIR_DOMAIN_CHR_DEVICE_TYPE_PARALLEL:
|
|
|
|
prefix = "parallel";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_CHR_DEVICE_TYPE_SERIAL:
|
|
|
|
prefix = "serial";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_CHR_DEVICE_TYPE_CONSOLE:
|
|
|
|
prefix = "console";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_CHR_DEVICE_TYPE_CHANNEL:
|
|
|
|
prefix = "channel";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_CHR_DEVICE_TYPE_LAST:
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (idx == -1 && (idx = qemuGetNextChrDevIndex(def, chr, prefix)) < 0)
|
|
|
|
return -1;
|
|
|
|
|
2019-10-22 15:26:14 +02:00
|
|
|
chr->info.alias = g_strdup_printf("%s%zd", prefix, idx);
|
|
|
|
return 0;
|
2016-02-16 10:24:35 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-11-24 17:12:54 +01:00
|
|
|
void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAssignDeviceControllerAlias(virDomainDef *domainDef,
|
|
|
|
virDomainControllerDef *controller)
|
2016-02-16 10:24:35 -05:00
|
|
|
{
|
|
|
|
const char *prefix = virDomainControllerTypeToString(controller->type);
|
|
|
|
|
2017-10-14 10:16:20 -07:00
|
|
|
if (controller->info.alias)
|
2021-11-24 17:12:54 +01:00
|
|
|
return;
|
2017-10-14 10:16:20 -07:00
|
|
|
|
2016-02-16 10:24:35 -05:00
|
|
|
if (controller->type == VIR_DOMAIN_CONTROLLER_TYPE_PCI) {
|
2021-03-30 15:09:27 +02:00
|
|
|
if (!virQEMUCapsHasPCIMultiBus(domainDef)) {
|
2016-02-16 10:24:35 -05:00
|
|
|
/* qemus that don't support multiple PCI buses have
|
|
|
|
* hardcoded the name of their single PCI controller as
|
|
|
|
* "pci".
|
|
|
|
*/
|
2019-10-20 12:55:05 +02:00
|
|
|
controller->info.alias = g_strdup("pci");
|
2021-11-24 17:12:54 +01:00
|
|
|
return;
|
2021-11-23 18:20:10 +01:00
|
|
|
}
|
|
|
|
if (controller->model == VIR_DOMAIN_CONTROLLER_MODEL_PCIE_ROOT) {
|
2016-02-16 10:24:35 -05:00
|
|
|
/* The pcie-root controller on Q35 machinetypes uses a
|
|
|
|
* different naming convention ("pcie.0"), because it is
|
|
|
|
* hardcoded that way in qemu.
|
|
|
|
*/
|
2019-10-22 15:26:14 +02:00
|
|
|
controller->info.alias = g_strdup_printf("pcie.%d", controller->idx);
|
2021-11-24 17:12:54 +01:00
|
|
|
return;
|
2016-02-16 10:24:35 -05:00
|
|
|
}
|
|
|
|
/* All other PCI controllers use the consistent "pci.%u"
|
|
|
|
* (including the hardcoded pci-root controller on
|
|
|
|
* multibus-capable qemus).
|
|
|
|
*/
|
2019-10-22 15:26:14 +02:00
|
|
|
controller->info.alias = g_strdup_printf("pci.%d", controller->idx);
|
2021-11-24 17:12:54 +01:00
|
|
|
return;
|
2021-11-23 18:20:10 +01:00
|
|
|
}
|
|
|
|
if (controller->type == VIR_DOMAIN_CONTROLLER_TYPE_IDE) {
|
2016-02-16 10:24:35 -05:00
|
|
|
/* for any machine based on e.g. I440FX or G3Beige, the
|
|
|
|
* first (and currently only) IDE controller is an integrated
|
|
|
|
* controller hardcoded with id "ide"
|
|
|
|
*/
|
2017-04-18 12:43:58 +02:00
|
|
|
if (qemuDomainHasBuiltinIDE(domainDef) &&
|
2019-10-20 12:55:05 +02:00
|
|
|
controller->idx == 0) {
|
|
|
|
controller->info.alias = g_strdup("ide");
|
2021-11-24 17:12:54 +01:00
|
|
|
return;
|
2019-10-20 12:55:05 +02:00
|
|
|
}
|
2016-02-16 10:24:35 -05:00
|
|
|
} else if (controller->type == VIR_DOMAIN_CONTROLLER_TYPE_SATA) {
|
|
|
|
/* for any Q35 machine, the first SATA controller is the
|
|
|
|
* integrated one, and it too is hardcoded with id "ide"
|
|
|
|
*/
|
2019-10-20 12:55:05 +02:00
|
|
|
if (qemuDomainIsQ35(domainDef) && controller->idx == 0) {
|
|
|
|
controller->info.alias = g_strdup("ide");
|
2021-11-24 17:12:54 +01:00
|
|
|
return;
|
2019-10-20 12:55:05 +02:00
|
|
|
}
|
2016-02-16 10:24:35 -05:00
|
|
|
} else if (controller->type == VIR_DOMAIN_CONTROLLER_TYPE_USB) {
|
|
|
|
/* first USB device is "usb", others are normal "usb%d" */
|
2019-10-20 12:55:05 +02:00
|
|
|
if (controller->idx == 0) {
|
|
|
|
controller->info.alias = g_strdup("usb");
|
2021-11-24 17:12:54 +01:00
|
|
|
return;
|
2019-10-20 12:55:05 +02:00
|
|
|
}
|
2020-11-18 16:33:16 +00:00
|
|
|
} else if (controller->type == VIR_DOMAIN_CONTROLLER_TYPE_SCSI) {
|
|
|
|
if (controller->model == VIR_DOMAIN_CONTROLLER_MODEL_SCSI_NCR53C90 &&
|
|
|
|
controller->idx == 0) {
|
|
|
|
controller->info.alias = g_strdup("scsi");
|
2021-11-24 17:12:54 +01:00
|
|
|
return;
|
2020-11-18 16:33:16 +00:00
|
|
|
}
|
2016-02-16 10:24:35 -05:00
|
|
|
}
|
|
|
|
/* all other controllers use the default ${type}${index} naming
|
|
|
|
* scheme for alias/id.
|
|
|
|
*/
|
2019-10-22 15:26:14 +02:00
|
|
|
controller->info.alias = g_strdup_printf("%s%d", prefix, controller->idx);
|
2016-02-16 10:24:35 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-05-15 17:24:54 -04:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAssignDeviceDiskAlias(virDomainDef *def,
|
|
|
|
virDomainDiskDef *disk,
|
|
|
|
virQEMUCaps *qemuCaps)
|
2016-02-16 10:24:35 -05:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuDomainDiskPrivate *diskPriv = QEMU_DOMAIN_DISK_PRIVATE(disk);
|
2016-02-16 10:24:35 -05:00
|
|
|
const char *prefix = virDomainDiskBusTypeToString(disk->bus);
|
|
|
|
int controllerModel = -1;
|
|
|
|
|
2018-06-26 16:55:19 +02:00
|
|
|
if (!disk->info.alias) {
|
|
|
|
if (disk->info.type == VIR_DOMAIN_DEVICE_ADDRESS_TYPE_DRIVE) {
|
|
|
|
if (disk->bus == VIR_DOMAIN_DISK_BUS_SCSI) {
|
|
|
|
controllerModel = qemuDomainFindSCSIControllerModel(def,
|
|
|
|
&disk->info);
|
|
|
|
if (controllerModel < 0)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (disk->bus != VIR_DOMAIN_DISK_BUS_SCSI ||
|
|
|
|
controllerModel == VIR_DOMAIN_CONTROLLER_MODEL_SCSI_LSILOGIC) {
|
2019-10-22 15:26:14 +02:00
|
|
|
disk->info.alias = g_strdup_printf("%s%d-%d-%d", prefix,
|
|
|
|
disk->info.addr.drive.controller,
|
|
|
|
disk->info.addr.drive.bus,
|
|
|
|
disk->info.addr.drive.unit);
|
2018-06-26 16:55:19 +02:00
|
|
|
} else {
|
2019-10-22 15:26:14 +02:00
|
|
|
disk->info.alias = g_strdup_printf("%s%d-%d-%d-%d", prefix,
|
|
|
|
disk->info.addr.drive.controller,
|
|
|
|
disk->info.addr.drive.bus,
|
|
|
|
disk->info.addr.drive.target,
|
|
|
|
disk->info.addr.drive.unit);
|
2018-06-26 16:55:19 +02:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
int idx = virDiskNameToIndex(disk->dst);
|
2019-10-22 15:26:14 +02:00
|
|
|
disk->info.alias = g_strdup_printf("%s-disk%d", prefix, idx);
|
2016-02-16 10:24:35 -05:00
|
|
|
}
|
2018-06-26 16:55:19 +02:00
|
|
|
}
|
2016-02-16 10:24:35 -05:00
|
|
|
|
2018-06-26 16:55:19 +02:00
|
|
|
/* For -blockdev we need to know the qom names of the disk which are based
|
|
|
|
* on the alias in qemu. While certain disk types use just the alias, some
|
|
|
|
* need the full path into /machine/peripheral as a historical artifact.
|
|
|
|
*/
|
2019-08-16 17:01:10 +02:00
|
|
|
if (!diskPriv->qomName &&
|
|
|
|
virQEMUCapsGet(qemuCaps, QEMU_CAPS_BLOCKDEV)) {
|
2018-06-26 16:55:19 +02:00
|
|
|
switch ((virDomainDiskBus) disk->bus) {
|
|
|
|
case VIR_DOMAIN_DISK_BUS_FDC:
|
|
|
|
case VIR_DOMAIN_DISK_BUS_IDE:
|
|
|
|
case VIR_DOMAIN_DISK_BUS_SATA:
|
|
|
|
case VIR_DOMAIN_DISK_BUS_SCSI:
|
2019-10-20 13:49:46 +02:00
|
|
|
diskPriv->qomName = g_strdup(disk->info.alias);
|
2018-06-26 16:55:19 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_DISK_BUS_VIRTIO:
|
2019-10-22 15:26:14 +02:00
|
|
|
diskPriv->qomName = g_strdup_printf("/machine/peripheral/%s/virtio-backend",
|
|
|
|
disk->info.alias);
|
2018-06-26 16:55:19 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_DISK_BUS_USB:
|
2019-10-22 15:26:14 +02:00
|
|
|
diskPriv->qomName = g_strdup_printf("/machine/peripheral/%s/%s.0/legacy[0]",
|
|
|
|
disk->info.alias, disk->info.alias);
|
2018-06-26 16:55:19 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_DISK_BUS_XEN:
|
|
|
|
case VIR_DOMAIN_DISK_BUS_UML:
|
|
|
|
case VIR_DOMAIN_DISK_BUS_SD:
|
2021-04-15 16:58:19 +02:00
|
|
|
case VIR_DOMAIN_DISK_BUS_NONE:
|
2018-06-26 16:55:19 +02:00
|
|
|
case VIR_DOMAIN_DISK_BUS_LAST:
|
|
|
|
break;
|
2016-02-16 10:24:35 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-11-24 17:12:54 +01:00
|
|
|
void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAssignDeviceHostdevAlias(virDomainDef *def,
|
2016-04-01 10:40:23 -04:00
|
|
|
char **alias,
|
2016-02-16 10:24:35 -05:00
|
|
|
int idx)
|
|
|
|
{
|
2017-10-14 10:16:20 -07:00
|
|
|
if (*alias)
|
2021-11-24 17:12:54 +01:00
|
|
|
return;
|
2017-10-14 10:16:20 -07:00
|
|
|
|
2016-02-16 10:24:35 -05:00
|
|
|
if (idx == -1) {
|
|
|
|
size_t i;
|
qemu: fix alias name for <interface type='hostdev'>
Starting with commit f8e712fe, if you start a domain that has an
<interface type='hostdev' (or that has <interface type='network'>
where the network is a pool of devices for hostdev assignment), when
you later try to add *another* interface (of any kind) with hotplug,
the function qemuAssignDeviceNetAlias() fails as soon as it sees a
"hostdevN" alias in the list of interfaces), causing the attach to
fail.
This is because (starting with f8e712fe) the device alias names are
assigned during the new function qemuProcessPrepareDomain(), which is
called *before* networkAllocateActualDevice() (which is called from
qemuProcessPrepareHost(), which is called from
qemuProcessLaunch()). Prior to that commit,
networkAllocateActualDevice() was called first.
The problem with this is that the alias for interfaces that are really
a hostdev (<interface type='hostdev'>) is of the form "hostdevN" (just
like other hostdevs), while other interfaces are "netN". But if you
don't know that the interface is going to be a hostdev at the time you
assign the alias name, you can't name it differently. (As far as I've
seen so far, the change in name by itself wouldn't have been a problem
(other than just an outwardly noticeable change in behavior) except
for the abovementioned failure to attach/detach new interfaces.
Rather than take the chance that there may be other not-yet-revealed
problems associated with changing the alias name, this patch changes
the way that aliases are assigned to restore the old behavior.
Old: In the past, assigning an alias to an interface was skipped if it
was seen that the interface was type='hostdev' - we knew that the
hostdev part of the interface was also in the list of hostdevs (that's
part of what happens in networkAllocateActualDevice()) and it would be
assigned when all the other hostdev aliases were assigned.
New: When assigning an alias to an interface, we haven't yet called
networkAllocateActualDevice() to construct the hostdev part of the
interface, so we can't just wait for the loop that creates aliases for
all the hostdevs (there's nothing on that list for this device
yet!). Instead we handle it immediately in the loop creating interface
aliases, by calling the new function networkGetActualType() to
determine if it is going to be hostdev, and if so calling
qemuAssignDeviceHostdevAlias() instead.
Some adjustments have to be made to both
qemuAssignDeviceHostdevAlias() and to qemuAssignDeviceNetAlias() to
accommodate this. In both of them, an error return from
qemuDomainDeviceAliasIndex() is no longer considered an error; instead
it's just ignored (because it almost certainly means that the alias
string for the device was "net" when we expected "hostdev" or vice
versa). in qemuAssignDeviceHostdevAlias() we have to look at all
interface aliases for hostdevN in addition to looking at all hostdev
aliases (this wasn't necessary in the past, because both the interface
entry and the hostdev entry for the device already pointed at the
device info; no longer the case since the hostdev entry hasn't yet
been setup).
Fortunately the buggy behavior hasn't yet been in any official release
of libvirt.
2016-04-01 13:18:57 -04:00
|
|
|
|
2016-02-16 10:24:35 -05:00
|
|
|
idx = 0;
|
|
|
|
for (i = 0; i < def->nhostdevs; i++) {
|
|
|
|
int thisidx;
|
qemu: fix alias name for <interface type='hostdev'>
Starting with commit f8e712fe, if you start a domain that has an
<interface type='hostdev' (or that has <interface type='network'>
where the network is a pool of devices for hostdev assignment), when
you later try to add *another* interface (of any kind) with hotplug,
the function qemuAssignDeviceNetAlias() fails as soon as it sees a
"hostdevN" alias in the list of interfaces), causing the attach to
fail.
This is because (starting with f8e712fe) the device alias names are
assigned during the new function qemuProcessPrepareDomain(), which is
called *before* networkAllocateActualDevice() (which is called from
qemuProcessPrepareHost(), which is called from
qemuProcessLaunch()). Prior to that commit,
networkAllocateActualDevice() was called first.
The problem with this is that the alias for interfaces that are really
a hostdev (<interface type='hostdev'>) is of the form "hostdevN" (just
like other hostdevs), while other interfaces are "netN". But if you
don't know that the interface is going to be a hostdev at the time you
assign the alias name, you can't name it differently. (As far as I've
seen so far, the change in name by itself wouldn't have been a problem
(other than just an outwardly noticeable change in behavior) except
for the abovementioned failure to attach/detach new interfaces.
Rather than take the chance that there may be other not-yet-revealed
problems associated with changing the alias name, this patch changes
the way that aliases are assigned to restore the old behavior.
Old: In the past, assigning an alias to an interface was skipped if it
was seen that the interface was type='hostdev' - we knew that the
hostdev part of the interface was also in the list of hostdevs (that's
part of what happens in networkAllocateActualDevice()) and it would be
assigned when all the other hostdev aliases were assigned.
New: When assigning an alias to an interface, we haven't yet called
networkAllocateActualDevice() to construct the hostdev part of the
interface, so we can't just wait for the loop that creates aliases for
all the hostdevs (there's nothing on that list for this device
yet!). Instead we handle it immediately in the loop creating interface
aliases, by calling the new function networkGetActualType() to
determine if it is going to be hostdev, and if so calling
qemuAssignDeviceHostdevAlias() instead.
Some adjustments have to be made to both
qemuAssignDeviceHostdevAlias() and to qemuAssignDeviceNetAlias() to
accommodate this. In both of them, an error return from
qemuDomainDeviceAliasIndex() is no longer considered an error; instead
it's just ignored (because it almost certainly means that the alias
string for the device was "net" when we expected "hostdev" or vice
versa). in qemuAssignDeviceHostdevAlias() we have to look at all
interface aliases for hostdevN in addition to looking at all hostdev
aliases (this wasn't necessary in the past, because both the interface
entry and the hostdev entry for the device already pointed at the
device info; no longer the case since the hostdev entry hasn't yet
been setup).
Fortunately the buggy behavior hasn't yet been in any official release
of libvirt.
2016-04-01 13:18:57 -04:00
|
|
|
|
|
|
|
if ((thisidx = qemuDomainDeviceAliasIndex(def->hostdevs[i]->info, "hostdev")) < 0)
|
|
|
|
continue; /* error just means the alias wasn't "hostdevN", but something else */
|
|
|
|
if (thisidx >= idx)
|
|
|
|
idx = thisidx + 1;
|
|
|
|
}
|
|
|
|
/* network interfaces can also have a hostdevN alias */
|
|
|
|
for (i = 0; i < def->nnets; i++) {
|
|
|
|
int thisidx;
|
|
|
|
|
|
|
|
if ((thisidx = qemuDomainDeviceAliasIndex(&def->nets[i]->info, "hostdev")) < 0)
|
|
|
|
continue;
|
2016-02-16 10:24:35 -05:00
|
|
|
if (thisidx >= idx)
|
|
|
|
idx = thisidx + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-22 15:26:14 +02:00
|
|
|
*alias = g_strdup_printf("hostdev%d", idx);
|
2016-02-16 10:24:35 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-11-24 17:12:54 +01:00
|
|
|
void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAssignDeviceNetAlias(virDomainDef *def,
|
|
|
|
virDomainNetDef *net,
|
2016-02-16 10:24:35 -05:00
|
|
|
int idx)
|
|
|
|
{
|
2017-10-14 10:16:20 -07:00
|
|
|
if (net->info.alias)
|
2021-11-24 17:12:54 +01:00
|
|
|
return;
|
2017-10-14 10:16:20 -07:00
|
|
|
|
qemu: fix alias name for <interface type='hostdev'>
Starting with commit f8e712fe, if you start a domain that has an
<interface type='hostdev' (or that has <interface type='network'>
where the network is a pool of devices for hostdev assignment), when
you later try to add *another* interface (of any kind) with hotplug,
the function qemuAssignDeviceNetAlias() fails as soon as it sees a
"hostdevN" alias in the list of interfaces), causing the attach to
fail.
This is because (starting with f8e712fe) the device alias names are
assigned during the new function qemuProcessPrepareDomain(), which is
called *before* networkAllocateActualDevice() (which is called from
qemuProcessPrepareHost(), which is called from
qemuProcessLaunch()). Prior to that commit,
networkAllocateActualDevice() was called first.
The problem with this is that the alias for interfaces that are really
a hostdev (<interface type='hostdev'>) is of the form "hostdevN" (just
like other hostdevs), while other interfaces are "netN". But if you
don't know that the interface is going to be a hostdev at the time you
assign the alias name, you can't name it differently. (As far as I've
seen so far, the change in name by itself wouldn't have been a problem
(other than just an outwardly noticeable change in behavior) except
for the abovementioned failure to attach/detach new interfaces.
Rather than take the chance that there may be other not-yet-revealed
problems associated with changing the alias name, this patch changes
the way that aliases are assigned to restore the old behavior.
Old: In the past, assigning an alias to an interface was skipped if it
was seen that the interface was type='hostdev' - we knew that the
hostdev part of the interface was also in the list of hostdevs (that's
part of what happens in networkAllocateActualDevice()) and it would be
assigned when all the other hostdev aliases were assigned.
New: When assigning an alias to an interface, we haven't yet called
networkAllocateActualDevice() to construct the hostdev part of the
interface, so we can't just wait for the loop that creates aliases for
all the hostdevs (there's nothing on that list for this device
yet!). Instead we handle it immediately in the loop creating interface
aliases, by calling the new function networkGetActualType() to
determine if it is going to be hostdev, and if so calling
qemuAssignDeviceHostdevAlias() instead.
Some adjustments have to be made to both
qemuAssignDeviceHostdevAlias() and to qemuAssignDeviceNetAlias() to
accommodate this. In both of them, an error return from
qemuDomainDeviceAliasIndex() is no longer considered an error; instead
it's just ignored (because it almost certainly means that the alias
string for the device was "net" when we expected "hostdev" or vice
versa). in qemuAssignDeviceHostdevAlias() we have to look at all
interface aliases for hostdevN in addition to looking at all hostdev
aliases (this wasn't necessary in the past, because both the interface
entry and the hostdev entry for the device already pointed at the
device info; no longer the case since the hostdev entry hasn't yet
been setup).
Fortunately the buggy behavior hasn't yet been in any official release
of libvirt.
2016-04-01 13:18:57 -04:00
|
|
|
/* <interface type='hostdev'> uses "hostdevN" as the alias
|
|
|
|
* We must use "-1" as the index because the caller doesn't know
|
|
|
|
* that we're now looking for a unique hostdevN rather than netN
|
|
|
|
*/
|
2021-11-24 17:12:54 +01:00
|
|
|
if (virDomainNetResolveActualType(net) == VIR_DOMAIN_NET_TYPE_HOSTDEV) {
|
|
|
|
qemuAssignDeviceHostdevAlias(def, &net->info.alias, -1);
|
|
|
|
return;
|
|
|
|
}
|
qemu: fix alias name for <interface type='hostdev'>
Starting with commit f8e712fe, if you start a domain that has an
<interface type='hostdev' (or that has <interface type='network'>
where the network is a pool of devices for hostdev assignment), when
you later try to add *another* interface (of any kind) with hotplug,
the function qemuAssignDeviceNetAlias() fails as soon as it sees a
"hostdevN" alias in the list of interfaces), causing the attach to
fail.
This is because (starting with f8e712fe) the device alias names are
assigned during the new function qemuProcessPrepareDomain(), which is
called *before* networkAllocateActualDevice() (which is called from
qemuProcessPrepareHost(), which is called from
qemuProcessLaunch()). Prior to that commit,
networkAllocateActualDevice() was called first.
The problem with this is that the alias for interfaces that are really
a hostdev (<interface type='hostdev'>) is of the form "hostdevN" (just
like other hostdevs), while other interfaces are "netN". But if you
don't know that the interface is going to be a hostdev at the time you
assign the alias name, you can't name it differently. (As far as I've
seen so far, the change in name by itself wouldn't have been a problem
(other than just an outwardly noticeable change in behavior) except
for the abovementioned failure to attach/detach new interfaces.
Rather than take the chance that there may be other not-yet-revealed
problems associated with changing the alias name, this patch changes
the way that aliases are assigned to restore the old behavior.
Old: In the past, assigning an alias to an interface was skipped if it
was seen that the interface was type='hostdev' - we knew that the
hostdev part of the interface was also in the list of hostdevs (that's
part of what happens in networkAllocateActualDevice()) and it would be
assigned when all the other hostdev aliases were assigned.
New: When assigning an alias to an interface, we haven't yet called
networkAllocateActualDevice() to construct the hostdev part of the
interface, so we can't just wait for the loop that creates aliases for
all the hostdevs (there's nothing on that list for this device
yet!). Instead we handle it immediately in the loop creating interface
aliases, by calling the new function networkGetActualType() to
determine if it is going to be hostdev, and if so calling
qemuAssignDeviceHostdevAlias() instead.
Some adjustments have to be made to both
qemuAssignDeviceHostdevAlias() and to qemuAssignDeviceNetAlias() to
accommodate this. In both of them, an error return from
qemuDomainDeviceAliasIndex() is no longer considered an error; instead
it's just ignored (because it almost certainly means that the alias
string for the device was "net" when we expected "hostdev" or vice
versa). in qemuAssignDeviceHostdevAlias() we have to look at all
interface aliases for hostdevN in addition to looking at all hostdev
aliases (this wasn't necessary in the past, because both the interface
entry and the hostdev entry for the device already pointed at the
device info; no longer the case since the hostdev entry hasn't yet
been setup).
Fortunately the buggy behavior hasn't yet been in any official release
of libvirt.
2016-04-01 13:18:57 -04:00
|
|
|
|
2016-02-16 10:24:35 -05:00
|
|
|
if (idx == -1) {
|
|
|
|
size_t i;
|
qemu: fix alias name for <interface type='hostdev'>
Starting with commit f8e712fe, if you start a domain that has an
<interface type='hostdev' (or that has <interface type='network'>
where the network is a pool of devices for hostdev assignment), when
you later try to add *another* interface (of any kind) with hotplug,
the function qemuAssignDeviceNetAlias() fails as soon as it sees a
"hostdevN" alias in the list of interfaces), causing the attach to
fail.
This is because (starting with f8e712fe) the device alias names are
assigned during the new function qemuProcessPrepareDomain(), which is
called *before* networkAllocateActualDevice() (which is called from
qemuProcessPrepareHost(), which is called from
qemuProcessLaunch()). Prior to that commit,
networkAllocateActualDevice() was called first.
The problem with this is that the alias for interfaces that are really
a hostdev (<interface type='hostdev'>) is of the form "hostdevN" (just
like other hostdevs), while other interfaces are "netN". But if you
don't know that the interface is going to be a hostdev at the time you
assign the alias name, you can't name it differently. (As far as I've
seen so far, the change in name by itself wouldn't have been a problem
(other than just an outwardly noticeable change in behavior) except
for the abovementioned failure to attach/detach new interfaces.
Rather than take the chance that there may be other not-yet-revealed
problems associated with changing the alias name, this patch changes
the way that aliases are assigned to restore the old behavior.
Old: In the past, assigning an alias to an interface was skipped if it
was seen that the interface was type='hostdev' - we knew that the
hostdev part of the interface was also in the list of hostdevs (that's
part of what happens in networkAllocateActualDevice()) and it would be
assigned when all the other hostdev aliases were assigned.
New: When assigning an alias to an interface, we haven't yet called
networkAllocateActualDevice() to construct the hostdev part of the
interface, so we can't just wait for the loop that creates aliases for
all the hostdevs (there's nothing on that list for this device
yet!). Instead we handle it immediately in the loop creating interface
aliases, by calling the new function networkGetActualType() to
determine if it is going to be hostdev, and if so calling
qemuAssignDeviceHostdevAlias() instead.
Some adjustments have to be made to both
qemuAssignDeviceHostdevAlias() and to qemuAssignDeviceNetAlias() to
accommodate this. In both of them, an error return from
qemuDomainDeviceAliasIndex() is no longer considered an error; instead
it's just ignored (because it almost certainly means that the alias
string for the device was "net" when we expected "hostdev" or vice
versa). in qemuAssignDeviceHostdevAlias() we have to look at all
interface aliases for hostdevN in addition to looking at all hostdev
aliases (this wasn't necessary in the past, because both the interface
entry and the hostdev entry for the device already pointed at the
device info; no longer the case since the hostdev entry hasn't yet
been setup).
Fortunately the buggy behavior hasn't yet been in any official release
of libvirt.
2016-04-01 13:18:57 -04:00
|
|
|
|
2016-02-16 10:24:35 -05:00
|
|
|
idx = 0;
|
|
|
|
for (i = 0; i < def->nnets; i++) {
|
|
|
|
int thisidx;
|
|
|
|
|
qemu: fix alias name for <interface type='hostdev'>
Starting with commit f8e712fe, if you start a domain that has an
<interface type='hostdev' (or that has <interface type='network'>
where the network is a pool of devices for hostdev assignment), when
you later try to add *another* interface (of any kind) with hotplug,
the function qemuAssignDeviceNetAlias() fails as soon as it sees a
"hostdevN" alias in the list of interfaces), causing the attach to
fail.
This is because (starting with f8e712fe) the device alias names are
assigned during the new function qemuProcessPrepareDomain(), which is
called *before* networkAllocateActualDevice() (which is called from
qemuProcessPrepareHost(), which is called from
qemuProcessLaunch()). Prior to that commit,
networkAllocateActualDevice() was called first.
The problem with this is that the alias for interfaces that are really
a hostdev (<interface type='hostdev'>) is of the form "hostdevN" (just
like other hostdevs), while other interfaces are "netN". But if you
don't know that the interface is going to be a hostdev at the time you
assign the alias name, you can't name it differently. (As far as I've
seen so far, the change in name by itself wouldn't have been a problem
(other than just an outwardly noticeable change in behavior) except
for the abovementioned failure to attach/detach new interfaces.
Rather than take the chance that there may be other not-yet-revealed
problems associated with changing the alias name, this patch changes
the way that aliases are assigned to restore the old behavior.
Old: In the past, assigning an alias to an interface was skipped if it
was seen that the interface was type='hostdev' - we knew that the
hostdev part of the interface was also in the list of hostdevs (that's
part of what happens in networkAllocateActualDevice()) and it would be
assigned when all the other hostdev aliases were assigned.
New: When assigning an alias to an interface, we haven't yet called
networkAllocateActualDevice() to construct the hostdev part of the
interface, so we can't just wait for the loop that creates aliases for
all the hostdevs (there's nothing on that list for this device
yet!). Instead we handle it immediately in the loop creating interface
aliases, by calling the new function networkGetActualType() to
determine if it is going to be hostdev, and if so calling
qemuAssignDeviceHostdevAlias() instead.
Some adjustments have to be made to both
qemuAssignDeviceHostdevAlias() and to qemuAssignDeviceNetAlias() to
accommodate this. In both of them, an error return from
qemuDomainDeviceAliasIndex() is no longer considered an error; instead
it's just ignored (because it almost certainly means that the alias
string for the device was "net" when we expected "hostdev" or vice
versa). in qemuAssignDeviceHostdevAlias() we have to look at all
interface aliases for hostdevN in addition to looking at all hostdev
aliases (this wasn't necessary in the past, because both the interface
entry and the hostdev entry for the device already pointed at the
device info; no longer the case since the hostdev entry hasn't yet
been setup).
Fortunately the buggy behavior hasn't yet been in any official release
of libvirt.
2016-04-01 13:18:57 -04:00
|
|
|
if ((thisidx = qemuDomainDeviceAliasIndex(&def->nets[i]->info, "net")) < 0)
|
|
|
|
continue; /* failure could be due to "hostdevN" */
|
2016-02-16 10:24:35 -05:00
|
|
|
if (thisidx >= idx)
|
|
|
|
idx = thisidx + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-22 15:26:14 +02:00
|
|
|
net->info.alias = g_strdup_printf("net%d", idx);
|
2016-02-16 10:24:35 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-11-24 17:12:54 +01:00
|
|
|
void
|
2021-08-17 17:24:08 +02:00
|
|
|
qemuAssignDeviceFSAlias(virDomainDef *def,
|
|
|
|
virDomainFSDef *fss)
|
2017-09-20 18:17:32 +02:00
|
|
|
{
|
2021-08-17 17:24:08 +02:00
|
|
|
size_t i;
|
|
|
|
int maxidx = 0;
|
|
|
|
|
2017-10-14 10:16:20 -07:00
|
|
|
if (fss->info.alias)
|
2021-11-24 17:12:54 +01:00
|
|
|
return;
|
2017-10-14 10:16:20 -07:00
|
|
|
|
2021-08-17 17:24:08 +02:00
|
|
|
for (i = 0; i < def->nfss; i++) {
|
|
|
|
int idx;
|
|
|
|
|
|
|
|
if ((idx = qemuDomainDeviceAliasIndex(&def->fss[i]->info, "fs")) >= maxidx)
|
|
|
|
maxidx = idx + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
fss->info.alias = g_strdup_printf("fs%d", maxidx);
|
2017-09-20 18:17:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-11-24 17:12:53 +01:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAssignDeviceSoundAlias(virDomainSoundDef *sound,
|
2017-09-20 18:17:32 +02:00
|
|
|
int idx)
|
|
|
|
{
|
2021-11-24 17:12:53 +01:00
|
|
|
if (!sound->info.alias)
|
|
|
|
sound->info.alias = g_strdup_printf("sound%d", idx);
|
2017-09-20 18:17:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-11-24 17:12:53 +01:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAssignDeviceVideoAlias(virDomainVideoDef *video,
|
2017-09-20 18:17:32 +02:00
|
|
|
int idx)
|
|
|
|
{
|
2021-11-24 17:12:53 +01:00
|
|
|
if (!video->info.alias)
|
|
|
|
video->info.alias = g_strdup_printf("video%d", idx);
|
2017-09-20 18:17:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-11-24 17:12:53 +01:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAssignDeviceHubAlias(virDomainHubDef *hub,
|
2017-09-20 18:17:32 +02:00
|
|
|
int idx)
|
|
|
|
{
|
2021-11-24 17:12:53 +01:00
|
|
|
if (!hub->info.alias)
|
|
|
|
hub->info.alias = g_strdup_printf("hub%d", idx);
|
2017-09-20 18:17:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-11-24 17:12:53 +01:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAssignDeviceSmartcardAlias(virDomainSmartcardDef *smartcard,
|
2017-09-20 18:17:32 +02:00
|
|
|
int idx)
|
|
|
|
{
|
2021-11-24 17:12:53 +01:00
|
|
|
if (!smartcard->info.alias)
|
|
|
|
smartcard->info.alias = g_strdup_printf("smartcard%d", idx);
|
2017-09-20 18:17:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-11-24 17:12:53 +01:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAssignDeviceMemballoonAlias(virDomainMemballoonDef *memballoon,
|
2017-09-20 18:17:32 +02:00
|
|
|
int idx)
|
|
|
|
{
|
2021-11-24 17:12:53 +01:00
|
|
|
if (!memballoon->info.alias)
|
|
|
|
memballoon->info.alias = g_strdup_printf("balloon%d", idx);
|
2017-09-20 18:17:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-11-24 17:12:53 +01:00
|
|
|
static void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAssignDeviceTPMAlias(virDomainTPMDef *tpm,
|
2017-09-20 18:17:32 +02:00
|
|
|
int idx)
|
|
|
|
{
|
2021-11-24 17:12:53 +01:00
|
|
|
if (!tpm->info.alias)
|
|
|
|
tpm->info.alias = g_strdup_printf("tpm%d", idx);
|
2017-09-20 18:17:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-11-24 17:12:54 +01:00
|
|
|
void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAssignDeviceRedirdevAlias(virDomainDef *def,
|
|
|
|
virDomainRedirdevDef *redirdev,
|
2016-02-16 10:24:35 -05:00
|
|
|
int idx)
|
|
|
|
{
|
2017-10-14 10:16:20 -07:00
|
|
|
if (redirdev->info.alias)
|
2021-11-24 17:12:54 +01:00
|
|
|
return;
|
2017-10-14 10:16:20 -07:00
|
|
|
|
2016-02-16 10:24:35 -05:00
|
|
|
if (idx == -1) {
|
|
|
|
size_t i;
|
|
|
|
idx = 0;
|
|
|
|
for (i = 0; i < def->nredirdevs; i++) {
|
|
|
|
int thisidx;
|
2017-10-20 13:24:41 +02:00
|
|
|
if ((thisidx = qemuDomainDeviceAliasIndex(&def->redirdevs[i]->info, "redir")) < 0)
|
|
|
|
continue;
|
2016-02-16 10:24:35 -05:00
|
|
|
if (thisidx >= idx)
|
|
|
|
idx = thisidx + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-22 15:26:14 +02:00
|
|
|
redirdev->info.alias = g_strdup_printf("redir%d", idx);
|
2016-02-16 10:24:35 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-11-24 17:12:54 +01:00
|
|
|
void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAssignDeviceRNGAlias(virDomainDef *def,
|
|
|
|
virDomainRNGDef *rng)
|
2016-02-16 10:24:35 -05:00
|
|
|
{
|
2016-04-06 17:32:12 +02:00
|
|
|
size_t i;
|
|
|
|
int maxidx = 0;
|
|
|
|
int idx;
|
|
|
|
|
2017-10-14 10:16:20 -07:00
|
|
|
if (rng->info.alias)
|
2021-11-24 17:12:54 +01:00
|
|
|
return;
|
2017-10-14 10:16:20 -07:00
|
|
|
|
2016-04-06 17:32:12 +02:00
|
|
|
for (i = 0; i < def->nrngs; i++) {
|
|
|
|
if ((idx = qemuDomainDeviceAliasIndex(&def->rngs[i]->info, "rng")) >= maxidx)
|
|
|
|
maxidx = idx + 1;
|
|
|
|
}
|
|
|
|
|
2019-10-22 15:26:14 +02:00
|
|
|
rng->info.alias = g_strdup_printf("rng%d", maxidx);
|
2016-04-06 17:32:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-31 20:50:22 +01:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuDeviceMemoryGetAliasID(virDomainDef *def,
|
|
|
|
virDomainMemoryDef *mem,
|
2020-10-31 20:50:22 +01:00
|
|
|
bool oldAlias,
|
|
|
|
const char *prefix)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
int maxidx = 0;
|
|
|
|
|
2021-01-19 10:16:55 +01:00
|
|
|
/* virtio-pmem and virtio-mem go onto PCI bus and thus DIMM address is not
|
|
|
|
* valid */
|
|
|
|
if (!oldAlias &&
|
|
|
|
mem->model != VIR_DOMAIN_MEMORY_MODEL_VIRTIO_PMEM &&
|
|
|
|
mem->model != VIR_DOMAIN_MEMORY_MODEL_VIRTIO_MEM)
|
2020-10-31 20:50:22 +01:00
|
|
|
return mem->info.addr.dimm.slot;
|
|
|
|
|
|
|
|
for (i = 0; i < def->nmems; i++) {
|
|
|
|
int idx;
|
|
|
|
if ((idx = qemuDomainDeviceAliasIndex(&def->mems[i]->info, prefix)) >= maxidx)
|
|
|
|
maxidx = idx + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return maxidx;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-11-01 06:21:36 +01:00
|
|
|
/**
|
|
|
|
* qemuAssignDeviceMemoryAlias:
|
|
|
|
* @def: domain definition. Necessary only if @oldAlias is true.
|
|
|
|
* @mem: memory device definition
|
|
|
|
* @oldAlias: Generate the alias according to the order of the device in @def
|
|
|
|
* rather than according to the slot number for legacy reasons.
|
|
|
|
*
|
|
|
|
* Generates alias for a memory device according to slot number if @oldAlias is
|
|
|
|
* false or according to order in @def->mems otherwise.
|
|
|
|
*
|
|
|
|
* Returns 0 on success, -1 on error.
|
|
|
|
*/
|
2016-04-06 17:32:12 +02:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAssignDeviceMemoryAlias(virDomainDef *def,
|
|
|
|
virDomainMemoryDef *mem,
|
2016-11-01 06:21:36 +01:00
|
|
|
bool oldAlias)
|
2016-04-06 17:32:12 +02:00
|
|
|
{
|
2020-10-31 20:50:22 +01:00
|
|
|
const char *prefix = NULL;
|
|
|
|
int idx = 0;
|
2016-07-29 11:02:25 +02:00
|
|
|
|
2017-10-14 10:16:20 -07:00
|
|
|
if (mem->info.alias)
|
|
|
|
return 0;
|
|
|
|
|
2020-10-31 20:50:22 +01:00
|
|
|
switch (mem->model) {
|
|
|
|
case VIR_DOMAIN_MEMORY_MODEL_DIMM:
|
2016-07-29 11:02:25 +02:00
|
|
|
prefix = "dimm";
|
2020-10-31 20:50:22 +01:00
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_MEMORY_MODEL_NVDIMM:
|
2016-07-29 11:02:25 +02:00
|
|
|
prefix = "nvdimm";
|
2020-10-31 20:50:22 +01:00
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_MEMORY_MODEL_VIRTIO_PMEM:
|
|
|
|
prefix = "virtiopmem";
|
|
|
|
break;
|
conf: Introduce virtio-mem <memory/> model
The virtio-mem is paravirtualized mechanism of adding/removing
memory to/from a VM. A virtio-mem-pci device is split into blocks
of equal size which are then exposed (all or only a requested
portion of them) to the guest kernel to use as regular memory.
Therefore, the device has two important attributes:
1) block-size, which defines the size of a block
2) requested-size, which defines how much memory (in bytes)
is the device requested to expose to the guest.
The 'block-size' is configured on command line and immutable
throughout device's lifetime. The 'requested-size' can be set on
the command line too, but also is adjustable via monitor. In
fact, that is how management software places its requests to
change the memory allocation. If it wants to give more memory to
the guest it changes 'requested-size' to a bigger value, and if it
wants to shrink guest memory it changes the 'requested-size' to a
smaller value. Note, value of zero means that guest should
release all memory offered by the device. Of course, guest has to
cooperate. Therefore, there is a third attribute 'size' which is
read only and reflects how much memory the guest still has. This
can be different to 'requested-size', obviously. Because of name
clash, I've named it 'current' and it is dealt with in future
commits (it is a runtime information anyway).
In the backend, memory for virtio-mem is backed by usual objects:
memory-backend-{ram,file,memfd} and their size puts the cap on
the amount of memory that a virtio-mem device can offer to a
guest. But we are already able to express this info using <size/>
under <target/>.
Therefore, we need only two more elements to cover 'block-size'
and 'requested-size' attributes. This is the XML I've came up
with:
<memory model='virtio-mem'>
<source>
<nodemask>1-3</nodemask>
<pagesize unit='KiB'>2048</pagesize>
</source>
<target>
<size unit='KiB'>2097152</size>
<node>0</node>
<block unit='KiB'>2048</block>
<requested unit='KiB'>1048576</requested>
</target>
<address type='pci' domain='0x0000' bus='0x00' slot='0x04' function='0x0'/>
</memory>
I hope by now it is obvious that:
1) 'requested-size' must be an integer multiple of
'block-size', and
2) virtio-mem-pci device goes onto PCI bus and thus needs PCI
address.
Then there is a limitation that the minimal 'block-size' is
transparent huge page size (I'll leave this without explanation).
Signed-off-by: Michal Privoznik <mprivozn@redhat.com>
Reviewed-by: Ján Tomko <jtomko@redhat.com>
2021-01-18 16:13:12 +01:00
|
|
|
case VIR_DOMAIN_MEMORY_MODEL_VIRTIO_MEM:
|
2021-01-19 10:16:55 +01:00
|
|
|
prefix = "virtiomem";
|
|
|
|
break;
|
2020-10-31 20:50:22 +01:00
|
|
|
case VIR_DOMAIN_MEMORY_MODEL_NONE:
|
|
|
|
case VIR_DOMAIN_MEMORY_MODEL_LAST:
|
|
|
|
default:
|
|
|
|
virReportEnumRangeError(virDomainMemoryModel, mem->model);
|
|
|
|
return -1;
|
|
|
|
break;
|
2016-04-06 17:32:12 +02:00
|
|
|
}
|
|
|
|
|
2020-10-31 20:50:22 +01:00
|
|
|
idx = qemuDeviceMemoryGetAliasID(def, mem, oldAlias, prefix);
|
|
|
|
mem->info.alias = g_strdup_printf("%s%d", prefix, idx);
|
2016-02-16 10:24:35 -05:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-11-24 17:12:54 +01:00
|
|
|
void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAssignDeviceShmemAlias(virDomainDef *def,
|
|
|
|
virDomainShmemDef *shmem,
|
2016-09-12 15:48:41 +02:00
|
|
|
int idx)
|
|
|
|
{
|
2017-10-14 10:16:20 -07:00
|
|
|
if (shmem->info.alias)
|
2021-11-24 17:12:54 +01:00
|
|
|
return;
|
2017-10-14 10:16:20 -07:00
|
|
|
|
2016-09-12 15:48:41 +02:00
|
|
|
if (idx == -1) {
|
|
|
|
size_t i;
|
|
|
|
idx = 0;
|
|
|
|
for (i = 0; i < def->nshmems; i++) {
|
|
|
|
int thisidx;
|
|
|
|
|
|
|
|
if ((thisidx = qemuDomainDeviceAliasIndex(&def->shmems[i]->info,
|
2017-10-20 13:24:41 +02:00
|
|
|
"shmem")) < 0)
|
|
|
|
continue;
|
2016-09-12 15:48:41 +02:00
|
|
|
|
|
|
|
if (thisidx >= idx)
|
|
|
|
idx = thisidx + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-22 15:26:14 +02:00
|
|
|
shmem->info.alias = g_strdup_printf("shmem%d", idx);
|
2016-09-12 15:48:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-11-24 17:12:53 +01:00
|
|
|
void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAssignDeviceWatchdogAlias(virDomainWatchdogDef *watchdog)
|
2017-09-01 13:39:15 +02:00
|
|
|
{
|
|
|
|
/* Currently, there's just one watchdog per domain */
|
|
|
|
|
2021-11-24 17:12:53 +01:00
|
|
|
if (!watchdog->info.alias)
|
|
|
|
watchdog->info.alias = g_strdup("watchdog0");
|
2017-10-04 11:09:22 +02:00
|
|
|
}
|
|
|
|
|
2021-11-24 17:12:53 +01:00
|
|
|
|
|
|
|
void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAssignDeviceInputAlias(virDomainDef *def,
|
|
|
|
virDomainInputDef *input,
|
2017-10-04 11:09:22 +02:00
|
|
|
int idx)
|
|
|
|
{
|
2018-03-13 13:01:11 +01:00
|
|
|
if (input->info.alias)
|
2021-11-24 17:12:53 +01:00
|
|
|
return;
|
2018-03-13 13:01:11 +01:00
|
|
|
|
2017-10-04 11:09:22 +02:00
|
|
|
if (idx == -1) {
|
|
|
|
int thisidx;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < def->ninputs; i++) {
|
|
|
|
if ((thisidx = qemuDomainDeviceAliasIndex(&def->inputs[i]->info, "input")) >= idx)
|
|
|
|
idx = thisidx + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-22 15:26:14 +02:00
|
|
|
input->info.alias = g_strdup_printf("input%d", idx);
|
2017-09-01 13:39:15 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-11-24 17:12:53 +01:00
|
|
|
void
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAssignDeviceVsockAlias(virDomainVsockDef *vsock)
|
2018-05-22 15:57:47 +02:00
|
|
|
{
|
2021-11-24 17:12:53 +01:00
|
|
|
if (!vsock->info.alias)
|
|
|
|
vsock->info.alias = g_strdup("vsock0");
|
2018-05-22 15:57:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-07-21 11:04:10 +02:00
|
|
|
static void
|
|
|
|
qemuAssignDeviceIOMMUAlias(virDomainIOMMUDef *iommu)
|
|
|
|
{
|
|
|
|
if (!iommu->info.alias)
|
|
|
|
iommu->info.alias = g_strdup("iommu0");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-02-16 10:24:35 -05:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
qemuAssignDeviceAliases(virDomainDef *def, virQEMUCaps *qemuCaps)
|
2016-02-16 10:24:35 -05:00
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < def->ndisks; i++) {
|
2018-06-26 16:55:19 +02:00
|
|
|
if (qemuAssignDeviceDiskAlias(def, def->disks[i], qemuCaps) < 0)
|
2016-02-16 10:24:35 -05:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
for (i = 0; i < def->nnets; i++) {
|
2021-11-24 17:12:54 +01:00
|
|
|
qemuAssignDeviceNetAlias(def, def->nets[i], -1);
|
2016-02-16 10:24:35 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < def->nfss; i++) {
|
2021-11-24 17:12:54 +01:00
|
|
|
qemuAssignDeviceFSAlias(def, def->fss[i]);
|
2016-02-16 10:24:35 -05:00
|
|
|
}
|
|
|
|
for (i = 0; i < def->nsounds; i++) {
|
2021-11-24 17:12:53 +01:00
|
|
|
qemuAssignDeviceSoundAlias(def->sounds[i], i);
|
2016-02-16 10:24:35 -05:00
|
|
|
}
|
|
|
|
for (i = 0; i < def->nhostdevs; i++) {
|
qemu: fix alias name for <interface type='hostdev'>
Starting with commit f8e712fe, if you start a domain that has an
<interface type='hostdev' (or that has <interface type='network'>
where the network is a pool of devices for hostdev assignment), when
you later try to add *another* interface (of any kind) with hotplug,
the function qemuAssignDeviceNetAlias() fails as soon as it sees a
"hostdevN" alias in the list of interfaces), causing the attach to
fail.
This is because (starting with f8e712fe) the device alias names are
assigned during the new function qemuProcessPrepareDomain(), which is
called *before* networkAllocateActualDevice() (which is called from
qemuProcessPrepareHost(), which is called from
qemuProcessLaunch()). Prior to that commit,
networkAllocateActualDevice() was called first.
The problem with this is that the alias for interfaces that are really
a hostdev (<interface type='hostdev'>) is of the form "hostdevN" (just
like other hostdevs), while other interfaces are "netN". But if you
don't know that the interface is going to be a hostdev at the time you
assign the alias name, you can't name it differently. (As far as I've
seen so far, the change in name by itself wouldn't have been a problem
(other than just an outwardly noticeable change in behavior) except
for the abovementioned failure to attach/detach new interfaces.
Rather than take the chance that there may be other not-yet-revealed
problems associated with changing the alias name, this patch changes
the way that aliases are assigned to restore the old behavior.
Old: In the past, assigning an alias to an interface was skipped if it
was seen that the interface was type='hostdev' - we knew that the
hostdev part of the interface was also in the list of hostdevs (that's
part of what happens in networkAllocateActualDevice()) and it would be
assigned when all the other hostdev aliases were assigned.
New: When assigning an alias to an interface, we haven't yet called
networkAllocateActualDevice() to construct the hostdev part of the
interface, so we can't just wait for the loop that creates aliases for
all the hostdevs (there's nothing on that list for this device
yet!). Instead we handle it immediately in the loop creating interface
aliases, by calling the new function networkGetActualType() to
determine if it is going to be hostdev, and if so calling
qemuAssignDeviceHostdevAlias() instead.
Some adjustments have to be made to both
qemuAssignDeviceHostdevAlias() and to qemuAssignDeviceNetAlias() to
accommodate this. In both of them, an error return from
qemuDomainDeviceAliasIndex() is no longer considered an error; instead
it's just ignored (because it almost certainly means that the alias
string for the device was "net" when we expected "hostdev" or vice
versa). in qemuAssignDeviceHostdevAlias() we have to look at all
interface aliases for hostdevN in addition to looking at all hostdev
aliases (this wasn't necessary in the past, because both the interface
entry and the hostdev entry for the device already pointed at the
device info; no longer the case since the hostdev entry hasn't yet
been setup).
Fortunately the buggy behavior hasn't yet been in any official release
of libvirt.
2016-04-01 13:18:57 -04:00
|
|
|
/* we can't start assigning at 0, since netdevs may have used
|
|
|
|
* up some hostdevN entries already. Also if the HostdevDef is
|
|
|
|
* linked to a NetDef, they will share an info and the alias
|
|
|
|
* will already be set, so don't try to set it again.
|
|
|
|
*/
|
2021-11-24 17:12:54 +01:00
|
|
|
qemuAssignDeviceHostdevAlias(def, &def->hostdevs[i]->info->alias, -1);
|
2016-02-16 10:24:35 -05:00
|
|
|
}
|
|
|
|
for (i = 0; i < def->nredirdevs; i++) {
|
2021-11-24 17:12:54 +01:00
|
|
|
qemuAssignDeviceRedirdevAlias(def, def->redirdevs[i], i);
|
2016-02-16 10:24:35 -05:00
|
|
|
}
|
|
|
|
for (i = 0; i < def->nvideos; i++) {
|
2021-11-24 17:12:53 +01:00
|
|
|
qemuAssignDeviceVideoAlias(def->videos[i], i);
|
2016-02-16 10:24:35 -05:00
|
|
|
}
|
|
|
|
for (i = 0; i < def->ncontrollers; i++) {
|
2021-11-24 17:12:54 +01:00
|
|
|
qemuAssignDeviceControllerAlias(def, def->controllers[i]);
|
2016-02-16 10:24:35 -05:00
|
|
|
}
|
|
|
|
for (i = 0; i < def->ninputs; i++) {
|
2021-11-24 17:12:53 +01:00
|
|
|
qemuAssignDeviceInputAlias(def, def->inputs[i], i);
|
2016-02-16 10:24:35 -05:00
|
|
|
}
|
|
|
|
for (i = 0; i < def->nparallels; i++) {
|
|
|
|
if (qemuAssignDeviceChrAlias(def, def->parallels[i], i) < 0)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
for (i = 0; i < def->nserials; i++) {
|
|
|
|
if (qemuAssignDeviceChrAlias(def, def->serials[i], i) < 0)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
for (i = 0; i < def->nchannels; i++) {
|
|
|
|
if (qemuAssignDeviceChrAlias(def, def->channels[i], i) < 0)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
for (i = 0; i < def->nconsoles; i++) {
|
|
|
|
if (qemuAssignDeviceChrAlias(def, def->consoles[i], i) < 0)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
for (i = 0; i < def->nhubs; i++) {
|
2021-11-24 17:12:53 +01:00
|
|
|
qemuAssignDeviceHubAlias(def->hubs[i], i);
|
2016-02-16 10:24:35 -05:00
|
|
|
}
|
|
|
|
for (i = 0; i < def->nshmems; i++) {
|
2021-11-24 17:12:54 +01:00
|
|
|
qemuAssignDeviceShmemAlias(def, def->shmems[i], i);
|
2016-02-16 10:24:35 -05:00
|
|
|
}
|
|
|
|
for (i = 0; i < def->nsmartcards; i++) {
|
2021-11-24 17:12:53 +01:00
|
|
|
qemuAssignDeviceSmartcardAlias(def->smartcards[i], i);
|
2016-02-16 10:24:35 -05:00
|
|
|
}
|
|
|
|
if (def->watchdog) {
|
2021-11-24 17:12:53 +01:00
|
|
|
qemuAssignDeviceWatchdogAlias(def->watchdog);
|
2016-02-16 10:24:35 -05:00
|
|
|
}
|
2018-03-19 18:14:52 +01:00
|
|
|
if (def->memballoon &&
|
|
|
|
def->memballoon->model != VIR_DOMAIN_MEMBALLOON_MODEL_NONE) {
|
2021-11-24 17:12:53 +01:00
|
|
|
qemuAssignDeviceMemballoonAlias(def->memballoon, 0);
|
2016-02-16 10:24:35 -05:00
|
|
|
}
|
|
|
|
for (i = 0; i < def->nrngs; i++) {
|
2021-11-24 17:12:54 +01:00
|
|
|
qemuAssignDeviceRNGAlias(def, def->rngs[i]);
|
2016-02-16 10:24:35 -05:00
|
|
|
}
|
2020-06-10 15:11:47 -03:00
|
|
|
for (i = 0; i < def->ntpms; i++) {
|
2021-11-24 17:12:53 +01:00
|
|
|
qemuAssignDeviceTPMAlias(def->tpms[i], i);
|
2016-02-16 10:24:35 -05:00
|
|
|
}
|
|
|
|
for (i = 0; i < def->nmems; i++) {
|
2020-10-31 20:50:22 +01:00
|
|
|
if (qemuAssignDeviceMemoryAlias(def, def->mems[i], false) < 0)
|
2016-02-16 10:24:35 -05:00
|
|
|
return -1;
|
|
|
|
}
|
2018-05-22 15:57:47 +02:00
|
|
|
if (def->vsock) {
|
2021-11-24 17:12:53 +01:00
|
|
|
qemuAssignDeviceVsockAlias(def->vsock);
|
2018-05-22 15:57:47 +02:00
|
|
|
}
|
2022-07-21 11:04:10 +02:00
|
|
|
if (def->iommu)
|
|
|
|
qemuAssignDeviceIOMMUAlias(def->iommu);
|
2016-02-16 10:24:35 -05:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2016-02-26 16:29:58 +01:00
|
|
|
|
|
|
|
|
2018-05-31 11:55:24 +02:00
|
|
|
/* qemuAliasDiskDriveFromDisk
|
2016-06-29 12:37:39 -04:00
|
|
|
* @disk: Pointer to a disk definition
|
|
|
|
*
|
|
|
|
* Generate and return an alias for the device disk '-drive'
|
|
|
|
*
|
|
|
|
* Returns NULL with error or a string containing the alias
|
|
|
|
*/
|
2016-02-26 16:29:58 +01:00
|
|
|
char *
|
2018-05-31 11:55:24 +02:00
|
|
|
qemuAliasDiskDriveFromDisk(const virDomainDiskDef *disk)
|
2016-02-26 16:29:58 +01:00
|
|
|
{
|
|
|
|
if (!disk->info.alias) {
|
2016-06-29 12:37:39 -04:00
|
|
|
virReportError(VIR_ERR_INVALID_ARG, "%s",
|
|
|
|
_("disk does not have an alias"));
|
2016-02-26 16:29:58 +01:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2021-11-23 17:34:36 +01:00
|
|
|
return g_strdup_printf("%s%s", QEMU_DRIVE_HOST_PREFIX, disk->info.alias);
|
2016-02-26 16:29:58 +01:00
|
|
|
}
|
2016-03-29 18:23:02 -04:00
|
|
|
|
|
|
|
|
2016-06-29 13:34:00 -04:00
|
|
|
/* qemuAliasDiskDriveSkipPrefix:
|
|
|
|
* @dev_name: Pointer to a const char string
|
|
|
|
*
|
|
|
|
* If the QEMU_DRIVE_HOST_PREFIX exists in the input string, then
|
|
|
|
* increment the pointer and return it
|
|
|
|
*/
|
|
|
|
const char *
|
|
|
|
qemuAliasDiskDriveSkipPrefix(const char *dev_name)
|
|
|
|
{
|
|
|
|
if (STRPREFIX(dev_name, QEMU_DRIVE_HOST_PREFIX))
|
|
|
|
dev_name += strlen(QEMU_DRIVE_HOST_PREFIX);
|
|
|
|
return dev_name;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-07-18 13:19:02 -04:00
|
|
|
/* qemuAliasFromHostdev
|
|
|
|
* @hostdev: Pointer to host device
|
|
|
|
*
|
|
|
|
* Generate and return a string containing a drive alias
|
|
|
|
*/
|
|
|
|
char *
|
|
|
|
qemuAliasFromHostdev(const virDomainHostdevDef *hostdev)
|
|
|
|
{
|
|
|
|
if (!hostdev->info->alias) {
|
|
|
|
virReportError(VIR_ERR_INVALID_ARG, "%s",
|
|
|
|
_("hostdev does not have an alias"));
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2021-11-23 17:34:36 +01:00
|
|
|
return g_strdup_printf("%s-%s",
|
|
|
|
virDomainDeviceAddressTypeToString(hostdev->info->type),
|
|
|
|
hostdev->info->alias);
|
2016-07-18 13:19:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-03-29 18:23:02 -04:00
|
|
|
/* qemuDomainGetMasterKeyAlias:
|
|
|
|
*
|
|
|
|
* Generate and return the masterKey alias
|
|
|
|
*
|
|
|
|
* Returns NULL or a string containing the master key alias
|
|
|
|
*/
|
|
|
|
char *
|
|
|
|
qemuDomainGetMasterKeyAlias(void)
|
|
|
|
{
|
2021-10-22 10:56:01 +02:00
|
|
|
return g_strdup("masterKey0");
|
2016-03-29 18:23:02 -04:00
|
|
|
}
|
qemu: Utilize qemu secret objects for RBD auth/secret
https://bugzilla.redhat.com/show_bug.cgi?id=1182074
If they're available and we need to pass secrets to qemu, then use the
qemu domain secret object in order to pass the secrets for RBD volumes
instead of passing the base64 encoded secret on the command line.
The goal is to make AES secrets the default and have no user interaction
required in order to allow using the AES mechanism. If the mechanism
is not available, then fall back to the current plain mechanism using
a base64 encoded secret.
New APIs:
qemu_domain.c:
qemuDomainGetSecretAESAlias:
Generate/return the secret object alias for an AES Secret Info type.
This will be called from qemuDomainSecretAESSetup.
qemuDomainSecretAESSetup: (private)
This API handles the details of the generation of the AES secret
and saves the pieces that need to be passed to qemu in order for
the secret to be decrypted. The encrypted secret based upon the
domain master key, an initialization vector (16 byte random value),
and the stored secret. Finally, the requirement from qemu is the IV
and encrypted secret are to be base64 encoded.
qemu_command.c:
qemuBuildSecretInfoProps: (private)
Generate/return a JSON properties object for the AES secret to
be used by both the command building and eventually the hotplug
code in order to add the secret object. Code was designed so that
in the future perhaps hotplug could use it if it made sense.
qemuBuildObjectSecretCommandLine (private)
Generate and add to the command line the -object secret for the
secret. This will be required for the subsequent RBD reference
to the object.
qemuBuildDiskSecinfoCommandLine (private)
Handle adding the AES secret object.
Adjustments:
qemu_domain.c:
The qemuDomainSecretSetup was altered to call either the AES or Plain
Setup functions based upon whether AES secrets are possible (we have
the encryption API) or not, we have secrets, and of course if the
protocol source is RBD.
qemu_command.c:
Adjust the qemuBuildRBDSecinfoURI API's in order to generate the
specific command options for an AES secret, such as:
-object secret,id=$alias,keyid=$masterKey,data=$base64encodedencrypted,
format=base64
-drive file=rbd:pool/image:id=myname:auth_supported=cephx\;none:\
mon_host=mon1.example.org\:6321,password-secret=$alias,...
where the 'id=' value is the secret object alias generated by
concatenating the disk alias and "-aesKey0". The 'keyid= $masterKey'
is the master key shared with qemu, and the -drive syntax will
reference that alias as the 'password-secret'. For the -drive
syntax, the 'id=myname' is kept to define the username, while the
'key=$base64 encoded secret' is removed.
While according to the syntax described for qemu commit '60390a21'
or as seen in the email archive:
https://lists.gnu.org/archive/html/qemu-devel/2016-01/msg04083.html
it is possible to pass a plaintext password via a file, the qemu
commit 'ac1d8878' describes the more feature rich 'keyid=' option
based upon the shared masterKey.
Add tests for checking/comparing output.
NB: For hotplug, since the hotplug code doesn't add command line
arguments, passing the encoded secret directly to the monitor
will suffice.
2016-04-11 11:26:14 -04:00
|
|
|
|
|
|
|
|
2020-03-06 15:36:42 +01:00
|
|
|
/* qemuAliasForSecret:
|
|
|
|
* @parentalias: alias of the parent object
|
|
|
|
* @obj: optional sub-object of the parent device the secret is for
|
|
|
|
*
|
|
|
|
* Generate alias for a secret object used by @parentalias device or one of
|
|
|
|
* the dependencies of the device described by @obj.
|
|
|
|
*/
|
|
|
|
char *
|
|
|
|
qemuAliasForSecret(const char *parentalias,
|
|
|
|
const char *obj)
|
|
|
|
{
|
|
|
|
if (obj)
|
|
|
|
return g_strdup_printf("%s-%s-secret0", parentalias, obj);
|
2021-11-23 18:20:10 +01:00
|
|
|
return g_strdup_printf("%s-secret0", parentalias);
|
2020-03-06 15:36:42 +01:00
|
|
|
}
|
|
|
|
|
2017-02-16 14:59:06 -05:00
|
|
|
/* qemuAliasTLSObjFromSrcAlias
|
|
|
|
* @srcAlias: Pointer to a source alias string
|
2016-06-09 18:30:55 -04:00
|
|
|
*
|
|
|
|
* Generate and return a string to be used as the TLS object alias
|
|
|
|
*/
|
|
|
|
char *
|
2017-02-16 14:59:06 -05:00
|
|
|
qemuAliasTLSObjFromSrcAlias(const char *srcAlias)
|
2016-06-09 18:30:55 -04:00
|
|
|
{
|
2021-10-22 10:56:01 +02:00
|
|
|
return g_strdup_printf("obj%s_tls0", srcAlias);
|
2016-06-09 18:30:55 -04:00
|
|
|
}
|
2016-10-18 16:37:23 +02:00
|
|
|
|
|
|
|
|
|
|
|
/* qemuAliasChardevFromDevAlias:
|
|
|
|
* @devAlias: pointer do device alias
|
|
|
|
*
|
|
|
|
* Generate and return a string to be used as chardev alias.
|
|
|
|
*/
|
|
|
|
char *
|
|
|
|
qemuAliasChardevFromDevAlias(const char *devAlias)
|
|
|
|
{
|
2021-10-22 10:56:01 +02:00
|
|
|
return g_strdup_printf("char%s", devAlias);
|
2016-10-18 16:37:23 +02:00
|
|
|
}
|
2018-04-18 16:55:14 +02:00
|
|
|
|
|
|
|
|
|
|
|
const char *
|
|
|
|
qemuDomainGetManagedPRAlias(void)
|
|
|
|
{
|
|
|
|
return "pr-helper0";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
char *
|
2018-05-11 14:11:36 +02:00
|
|
|
qemuDomainGetUnmanagedPRAlias(const char *parentalias)
|
2018-04-18 16:55:14 +02:00
|
|
|
{
|
2021-10-22 10:56:01 +02:00
|
|
|
return g_strdup_printf("pr-helper-%s", parentalias);
|
2018-04-18 16:55:14 +02:00
|
|
|
}
|
2020-02-25 10:55:11 +01:00
|
|
|
|
|
|
|
|
|
|
|
const char *
|
|
|
|
qemuDomainGetDBusVMStateAlias(void)
|
|
|
|
{
|
|
|
|
return "dbus-vmstate0";
|
|
|
|
}
|
2021-02-01 11:59:03 +01:00
|
|
|
|
|
|
|
char *
|
|
|
|
qemuDomainGetVhostUserChrAlias(const char *devalias)
|
|
|
|
{
|
|
|
|
return g_strdup_printf("chr-vu-%s", devalias);
|
|
|
|
}
|