mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-25 05:55:17 +00:00
0268270b0f
Allow callers to request XML validation against the schema. All callers for now pass 'false'. Signed-off-by: Peter Krempa <pkrempa@redhat.com> Reviewed-by: Jonathon Jongsma <jjongsma@redhat.com>
643 lines
19 KiB
C
643 lines
19 KiB
C
/*
|
|
* domain_driver.c: general functions shared between hypervisor drivers
|
|
*
|
|
* Copyright IBM Corp. 2020
|
|
*
|
|
* 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 "domain_driver.h"
|
|
#include "viralloc.h"
|
|
#include "virstring.h"
|
|
#include "vircrypto.h"
|
|
#include "virutil.h"
|
|
#include "virhostdev.h"
|
|
#include "viraccessapicheck.h"
|
|
#include "datatypes.h"
|
|
#include "driver.h"
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_DOMAIN
|
|
|
|
char *
|
|
virDomainDriverGenerateRootHash(const char *drivername,
|
|
const char *root)
|
|
{
|
|
g_autofree char *hash = NULL;
|
|
|
|
if (virCryptoHashString(VIR_CRYPTO_HASH_SHA256, root, &hash) < 0)
|
|
return NULL;
|
|
|
|
/* When two embed drivers start two domains with the same @name and @id
|
|
* we would generate a non-unique name. Include parts of hashed @root
|
|
* which guarantees uniqueness. The first 8 characters of SHA256 ought
|
|
* to be enough for anybody. */
|
|
return g_strdup_printf("%s-embed-%.8s", drivername, hash);
|
|
}
|
|
|
|
|
|
#define HOSTNAME_CHARS \
|
|
"abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789-"
|
|
|
|
static void
|
|
virDomainMachineNameAppendValid(virBuffer *buf,
|
|
const char *name)
|
|
{
|
|
bool skip = true;
|
|
|
|
for (; *name; name++) {
|
|
if (strlen(virBufferCurrentContent(buf)) >= 64)
|
|
break;
|
|
|
|
if (*name == '.' || *name == '-') {
|
|
if (!skip) {
|
|
virBufferAddChar(buf, *name);
|
|
skip = true;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
if (!strchr(HOSTNAME_CHARS, *name))
|
|
continue;
|
|
|
|
virBufferAddChar(buf, *name);
|
|
skip = false;
|
|
}
|
|
|
|
/* trailing dashes or dots are not allowed */
|
|
virBufferTrimChars(buf, "-.");
|
|
}
|
|
|
|
#undef HOSTNAME_CHARS
|
|
|
|
char *
|
|
virDomainDriverGenerateMachineName(const char *drivername,
|
|
const char *root,
|
|
int id,
|
|
const char *name,
|
|
bool privileged)
|
|
{
|
|
g_auto(virBuffer) buf = VIR_BUFFER_INITIALIZER;
|
|
|
|
if (root) {
|
|
g_autofree char *hash = NULL;
|
|
|
|
if (!(hash = virDomainDriverGenerateRootHash(drivername, root)))
|
|
return NULL;
|
|
|
|
virBufferAsprintf(&buf, "%s-", hash);
|
|
} else {
|
|
virBufferAsprintf(&buf, "%s-", drivername);
|
|
if (!privileged) {
|
|
|
|
g_autofree char *username = NULL;
|
|
if (!(username = virGetUserName(geteuid())))
|
|
return NULL;
|
|
|
|
virBufferAsprintf(&buf, "%s-", username);
|
|
}
|
|
}
|
|
|
|
virBufferAsprintf(&buf, "%d-", id);
|
|
virDomainMachineNameAppendValid(&buf, name);
|
|
|
|
return virBufferContentAndReset(&buf);
|
|
}
|
|
|
|
|
|
/* Modify dest_array to reflect all blkio device changes described in
|
|
* src_array. */
|
|
int
|
|
virDomainDriverMergeBlkioDevice(virBlkioDevice **dest_array,
|
|
size_t *dest_size,
|
|
virBlkioDevice *src_array,
|
|
size_t src_size,
|
|
const char *type)
|
|
{
|
|
size_t i, j;
|
|
virBlkioDevice *dest;
|
|
virBlkioDevice *src;
|
|
|
|
for (i = 0; i < src_size; i++) {
|
|
bool found = false;
|
|
|
|
src = &src_array[i];
|
|
for (j = 0; j < *dest_size; j++) {
|
|
dest = &(*dest_array)[j];
|
|
if (STREQ(src->path, dest->path)) {
|
|
found = true;
|
|
|
|
if (STREQ(type, VIR_DOMAIN_BLKIO_DEVICE_WEIGHT)) {
|
|
dest->weight = src->weight;
|
|
} else if (STREQ(type, VIR_DOMAIN_BLKIO_DEVICE_READ_IOPS)) {
|
|
dest->riops = src->riops;
|
|
} else if (STREQ(type, VIR_DOMAIN_BLKIO_DEVICE_WRITE_IOPS)) {
|
|
dest->wiops = src->wiops;
|
|
} else if (STREQ(type, VIR_DOMAIN_BLKIO_DEVICE_READ_BPS)) {
|
|
dest->rbps = src->rbps;
|
|
} else if (STREQ(type, VIR_DOMAIN_BLKIO_DEVICE_WRITE_BPS)) {
|
|
dest->wbps = src->wbps;
|
|
} else {
|
|
virReportError(VIR_ERR_INVALID_ARG, _("Unknown parameter %s"),
|
|
type);
|
|
return -1;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
if (!found) {
|
|
if (!src->weight && !src->riops && !src->wiops && !src->rbps && !src->wbps)
|
|
continue;
|
|
VIR_EXPAND_N(*dest_array, *dest_size, 1);
|
|
dest = &(*dest_array)[*dest_size - 1];
|
|
|
|
if (STREQ(type, VIR_DOMAIN_BLKIO_DEVICE_WEIGHT)) {
|
|
dest->weight = src->weight;
|
|
} else if (STREQ(type, VIR_DOMAIN_BLKIO_DEVICE_READ_IOPS)) {
|
|
dest->riops = src->riops;
|
|
} else if (STREQ(type, VIR_DOMAIN_BLKIO_DEVICE_WRITE_IOPS)) {
|
|
dest->wiops = src->wiops;
|
|
} else if (STREQ(type, VIR_DOMAIN_BLKIO_DEVICE_READ_BPS)) {
|
|
dest->rbps = src->rbps;
|
|
} else if (STREQ(type, VIR_DOMAIN_BLKIO_DEVICE_WRITE_BPS)) {
|
|
dest->wbps = src->wbps;
|
|
} else {
|
|
*dest_size = *dest_size - 1;
|
|
return -1;
|
|
}
|
|
|
|
dest->path = g_steal_pointer(&src->path);
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* blkioDeviceStr in the form of /device/path,weight,/device/path,weight
|
|
* for example, /dev/disk/by-path/pci-0000:00:1f.2-scsi-0:0:0:0,800
|
|
*/
|
|
int
|
|
virDomainDriverParseBlkioDeviceStr(char *blkioDeviceStr, const char *type,
|
|
virBlkioDevice **dev, size_t *size)
|
|
{
|
|
char *temp;
|
|
int ndevices = 0;
|
|
int nsep = 0;
|
|
size_t i;
|
|
virBlkioDevice *result = NULL;
|
|
|
|
*dev = NULL;
|
|
*size = 0;
|
|
|
|
if (STREQ(blkioDeviceStr, ""))
|
|
return 0;
|
|
|
|
temp = blkioDeviceStr;
|
|
while (temp) {
|
|
temp = strchr(temp, ',');
|
|
if (temp) {
|
|
temp++;
|
|
nsep++;
|
|
}
|
|
}
|
|
|
|
/* A valid string must have even number of fields, hence an odd
|
|
* number of commas. */
|
|
if (!(nsep & 1))
|
|
goto parse_error;
|
|
|
|
ndevices = (nsep + 1) / 2;
|
|
|
|
result = g_new0(virBlkioDevice, ndevices);
|
|
|
|
i = 0;
|
|
temp = blkioDeviceStr;
|
|
while (temp) {
|
|
char *p = temp;
|
|
|
|
/* device path */
|
|
p = strchr(p, ',');
|
|
if (!p)
|
|
goto parse_error;
|
|
|
|
result[i].path = g_strndup(temp, p - temp);
|
|
|
|
/* value */
|
|
temp = p + 1;
|
|
|
|
if (STREQ(type, VIR_DOMAIN_BLKIO_DEVICE_WEIGHT)) {
|
|
if (virStrToLong_uip(temp, &p, 10, &result[i].weight) < 0)
|
|
goto number_error;
|
|
} else if (STREQ(type, VIR_DOMAIN_BLKIO_DEVICE_READ_IOPS)) {
|
|
if (virStrToLong_uip(temp, &p, 10, &result[i].riops) < 0)
|
|
goto number_error;
|
|
} else if (STREQ(type, VIR_DOMAIN_BLKIO_DEVICE_WRITE_IOPS)) {
|
|
if (virStrToLong_uip(temp, &p, 10, &result[i].wiops) < 0)
|
|
goto number_error;
|
|
} else if (STREQ(type, VIR_DOMAIN_BLKIO_DEVICE_READ_BPS)) {
|
|
if (virStrToLong_ullp(temp, &p, 10, &result[i].rbps) < 0)
|
|
goto number_error;
|
|
} else if (STREQ(type, VIR_DOMAIN_BLKIO_DEVICE_WRITE_BPS)) {
|
|
if (virStrToLong_ullp(temp, &p, 10, &result[i].wbps) < 0)
|
|
goto number_error;
|
|
} else {
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
_("unknown parameter '%s'"), type);
|
|
goto cleanup;
|
|
}
|
|
|
|
i++;
|
|
|
|
if (*p == '\0')
|
|
break;
|
|
else if (*p != ',')
|
|
goto parse_error;
|
|
temp = p + 1;
|
|
}
|
|
|
|
if (!i)
|
|
VIR_FREE(result);
|
|
|
|
*dev = result;
|
|
*size = i;
|
|
|
|
return 0;
|
|
|
|
parse_error:
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
_("unable to parse blkio device '%s' '%s'"),
|
|
type, blkioDeviceStr);
|
|
goto cleanup;
|
|
|
|
number_error:
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
_("invalid value '%s' for parameter '%s' of device '%s'"),
|
|
temp, type, result[i].path);
|
|
|
|
cleanup:
|
|
if (result) {
|
|
virBlkioDeviceArrayClear(result, ndevices);
|
|
VIR_FREE(result);
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
|
|
int
|
|
virDomainDriverSetupPersistentDefBlkioParams(virDomainDef *persistentDef,
|
|
virTypedParameterPtr params,
|
|
int nparams)
|
|
{
|
|
size_t i;
|
|
int ret = 0;
|
|
|
|
for (i = 0; i < nparams; i++) {
|
|
virTypedParameterPtr param = ¶ms[i];
|
|
|
|
if (STREQ(param->field, VIR_DOMAIN_BLKIO_WEIGHT)) {
|
|
persistentDef->blkio.weight = param->value.ui;
|
|
} else if (STREQ(param->field, VIR_DOMAIN_BLKIO_DEVICE_WEIGHT) ||
|
|
STREQ(param->field, VIR_DOMAIN_BLKIO_DEVICE_READ_IOPS) ||
|
|
STREQ(param->field, VIR_DOMAIN_BLKIO_DEVICE_WRITE_IOPS) ||
|
|
STREQ(param->field, VIR_DOMAIN_BLKIO_DEVICE_READ_BPS) ||
|
|
STREQ(param->field, VIR_DOMAIN_BLKIO_DEVICE_WRITE_BPS)) {
|
|
virBlkioDevice *devices = NULL;
|
|
size_t ndevices;
|
|
|
|
if (virDomainDriverParseBlkioDeviceStr(param->value.s,
|
|
param->field,
|
|
&devices,
|
|
&ndevices) < 0) {
|
|
ret = -1;
|
|
continue;
|
|
}
|
|
|
|
if (virDomainDriverMergeBlkioDevice(&persistentDef->blkio.devices,
|
|
&persistentDef->blkio.ndevices,
|
|
devices, ndevices,
|
|
param->field) < 0)
|
|
ret = -1;
|
|
|
|
virBlkioDeviceArrayClear(devices, ndevices);
|
|
g_free(devices);
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
int
|
|
virDomainDriverNodeDeviceGetPCIInfo(virNodeDeviceDef *def,
|
|
virPCIDeviceAddress *devAddr)
|
|
{
|
|
virNodeDevCapsDef *cap;
|
|
|
|
cap = def->caps;
|
|
while (cap) {
|
|
if (cap->data.type == VIR_NODE_DEV_CAP_PCI_DEV) {
|
|
devAddr->domain = cap->data.pci_dev.domain;
|
|
devAddr->bus = cap->data.pci_dev.bus;
|
|
devAddr->slot = cap->data.pci_dev.slot;
|
|
devAddr->function = cap->data.pci_dev.function;
|
|
break;
|
|
}
|
|
|
|
cap = cap->next;
|
|
}
|
|
|
|
if (!cap) {
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
_("device %s is not a PCI device"), def->name);
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int
|
|
virDomainDriverNodeDeviceReset(virNodeDevicePtr dev,
|
|
virHostdevManager *hostdevMgr)
|
|
{
|
|
g_autoptr(virPCIDevice) pci = NULL;
|
|
virPCIDeviceAddress devAddr = { 0 };
|
|
g_autoptr(virNodeDeviceDef) def = NULL;
|
|
g_autofree char *xml = NULL;
|
|
g_autoptr(virConnect) nodeconn = NULL;
|
|
g_autoptr(virNodeDevice) nodedev = NULL;
|
|
|
|
if (!(nodeconn = virGetConnectNodeDev()))
|
|
return -1;
|
|
|
|
/* 'dev' is associated with virConnectPtr, so for split
|
|
* daemons, we need to get a copy that is associated with
|
|
* the virnodedevd daemon. */
|
|
if (!(nodedev = virNodeDeviceLookupByName(
|
|
nodeconn, virNodeDeviceGetName(dev))))
|
|
return -1;
|
|
|
|
xml = virNodeDeviceGetXMLDesc(nodedev, 0);
|
|
if (!xml)
|
|
return -1;
|
|
|
|
def = virNodeDeviceDefParse(xml, NULL, EXISTING_DEVICE, NULL, NULL, NULL, false);
|
|
if (!def)
|
|
return -1;
|
|
|
|
/* ACL check must happen against original 'dev',
|
|
* not the new 'nodedev' we acquired */
|
|
if (virNodeDeviceResetEnsureACL(dev->conn, def) < 0)
|
|
return -1;
|
|
|
|
if (virDomainDriverNodeDeviceGetPCIInfo(def, &devAddr) < 0)
|
|
return -1;
|
|
|
|
pci = virPCIDeviceNew(&devAddr);
|
|
if (!pci)
|
|
return -1;
|
|
|
|
return virHostdevPCINodeDeviceReset(hostdevMgr, pci);
|
|
}
|
|
|
|
|
|
int
|
|
virDomainDriverNodeDeviceReAttach(virNodeDevicePtr dev,
|
|
virHostdevManager *hostdevMgr)
|
|
{
|
|
g_autoptr(virPCIDevice) pci = NULL;
|
|
virPCIDeviceAddress devAddr = { 0 };
|
|
g_autoptr(virNodeDeviceDef) def = NULL;
|
|
g_autofree char *xml = NULL;
|
|
g_autoptr(virConnect) nodeconn = NULL;
|
|
g_autoptr(virNodeDevice) nodedev = NULL;
|
|
|
|
if (!(nodeconn = virGetConnectNodeDev()))
|
|
return -1;
|
|
|
|
/* 'dev' is associated with virConnectPtr, so for split
|
|
* daemons, we need to get a copy that is associated with
|
|
* the virnodedevd daemon. */
|
|
if (!(nodedev = virNodeDeviceLookupByName(
|
|
nodeconn, virNodeDeviceGetName(dev))))
|
|
return -1;
|
|
|
|
xml = virNodeDeviceGetXMLDesc(nodedev, 0);
|
|
if (!xml)
|
|
return -1;
|
|
|
|
def = virNodeDeviceDefParse(xml, NULL, EXISTING_DEVICE, NULL, NULL, NULL, false);
|
|
if (!def)
|
|
return -1;
|
|
|
|
/* ACL check must happen against original 'dev',
|
|
* not the new 'nodedev' we acquired */
|
|
if (virNodeDeviceReAttachEnsureACL(dev->conn, def) < 0)
|
|
return -1;
|
|
|
|
if (virDomainDriverNodeDeviceGetPCIInfo(def, &devAddr) < 0)
|
|
return -1;
|
|
|
|
pci = virPCIDeviceNew(&devAddr);
|
|
if (!pci)
|
|
return -1;
|
|
|
|
return virHostdevPCINodeDeviceReAttach(hostdevMgr, pci);
|
|
}
|
|
|
|
int
|
|
virDomainDriverNodeDeviceDetachFlags(virNodeDevicePtr dev,
|
|
virHostdevManager *hostdevMgr,
|
|
const char *driverName)
|
|
{
|
|
g_autoptr(virPCIDevice) pci = NULL;
|
|
virPCIDeviceAddress devAddr = { 0 };
|
|
g_autoptr(virNodeDeviceDef) def = NULL;
|
|
g_autofree char *xml = NULL;
|
|
g_autoptr(virConnect) nodeconn = NULL;
|
|
g_autoptr(virNodeDevice) nodedev = NULL;
|
|
|
|
if (!driverName)
|
|
return -1;
|
|
|
|
if (!(nodeconn = virGetConnectNodeDev()))
|
|
return -1;
|
|
|
|
/* 'dev' is associated with virConnectPtr, so for split
|
|
* daemons, we need to get a copy that is associated with
|
|
* the virnodedevd daemon. */
|
|
if (!(nodedev = virNodeDeviceLookupByName(nodeconn,
|
|
virNodeDeviceGetName(dev))))
|
|
return -1;
|
|
|
|
xml = virNodeDeviceGetXMLDesc(nodedev, 0);
|
|
if (!xml)
|
|
return -1;
|
|
|
|
def = virNodeDeviceDefParse(xml, NULL, EXISTING_DEVICE, NULL, NULL, NULL, false);
|
|
if (!def)
|
|
return -1;
|
|
|
|
/* ACL check must happen against original 'dev',
|
|
* not the new 'nodedev' we acquired */
|
|
if (virNodeDeviceDetachFlagsEnsureACL(dev->conn, def) < 0)
|
|
return -1;
|
|
|
|
if (virDomainDriverNodeDeviceGetPCIInfo(def, &devAddr) < 0)
|
|
return -1;
|
|
|
|
pci = virPCIDeviceNew(&devAddr);
|
|
if (!pci)
|
|
return -1;
|
|
|
|
if (STREQ(driverName, "vfio"))
|
|
virPCIDeviceSetStubDriver(pci, VIR_PCI_STUB_DRIVER_VFIO);
|
|
else if (STREQ(driverName, "xen"))
|
|
virPCIDeviceSetStubDriver(pci, VIR_PCI_STUB_DRIVER_XEN);
|
|
|
|
return virHostdevPCINodeDeviceDetach(hostdevMgr, pci);
|
|
}
|
|
|
|
/**
|
|
* virDomainDriverAddIOThreadCheck:
|
|
* @def: domain definition
|
|
* @iothread_id: iothread id
|
|
*
|
|
* Returns -1 if an IOThread is already using the given iothread id
|
|
*/
|
|
int
|
|
virDomainDriverAddIOThreadCheck(virDomainDef *def,
|
|
unsigned int iothread_id)
|
|
{
|
|
if (virDomainIOThreadIDFind(def, iothread_id)) {
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
_("an IOThread is already using iothread_id '%u'"),
|
|
iothread_id);
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* virDomainDriverDelIOThreadCheck:
|
|
* @def: domain definition
|
|
* @iothread_id: iothread id
|
|
*
|
|
* Returns -1 if there is no IOThread using the given iothread id
|
|
*/
|
|
int
|
|
virDomainDriverDelIOThreadCheck(virDomainDef *def,
|
|
unsigned int iothread_id)
|
|
{
|
|
size_t i;
|
|
|
|
if (!virDomainIOThreadIDFind(def, iothread_id)) {
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
_("cannot find IOThread '%u' in iothreadids list"),
|
|
iothread_id);
|
|
return -1;
|
|
}
|
|
|
|
for (i = 0; i < def->ndisks; i++) {
|
|
if (def->disks[i]->iothread == iothread_id) {
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
_("cannot remove IOThread %u since it "
|
|
"is being used by disk '%s'"),
|
|
iothread_id, def->disks[i]->dst);
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < def->ncontrollers; i++) {
|
|
if (def->controllers[i]->iothread == iothread_id) {
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
_("cannot remove IOThread '%u' since it "
|
|
"is being used by controller"),
|
|
iothread_id);
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* virDomainDriverGetIOThreadsConfig:
|
|
* @targetDef: domain definition
|
|
* @info: information about the IOThread in a domain
|
|
* @bitmap_size: generate bitmap with bitmap_size, 0 for getting the size
|
|
* from host
|
|
*
|
|
* Returns the number of IOThreads in the given domain or -1 in case of error
|
|
*/
|
|
int
|
|
virDomainDriverGetIOThreadsConfig(virDomainDef *targetDef,
|
|
virDomainIOThreadInfoPtr **info,
|
|
unsigned int bitmap_size)
|
|
{
|
|
virDomainIOThreadInfoPtr *info_ret = NULL;
|
|
size_t i;
|
|
int ret = -1;
|
|
|
|
if (targetDef->niothreadids == 0)
|
|
return 0;
|
|
|
|
info_ret = g_new0(virDomainIOThreadInfoPtr, targetDef->niothreadids);
|
|
|
|
for (i = 0; i < targetDef->niothreadids; i++) {
|
|
g_autoptr(virBitmap) bitmap = NULL;
|
|
virBitmap *cpumask = NULL;
|
|
info_ret[i] = g_new0(virDomainIOThreadInfo, 1);
|
|
|
|
/* IOThread ID's are taken from the iothreadids list */
|
|
info_ret[i]->iothread_id = targetDef->iothreadids[i]->iothread_id;
|
|
|
|
cpumask = targetDef->iothreadids[i]->cpumask;
|
|
if (!cpumask) {
|
|
if (targetDef->cpumask) {
|
|
cpumask = targetDef->cpumask;
|
|
} else {
|
|
if (bitmap_size) {
|
|
if (!(bitmap = virBitmapNew(bitmap_size)))
|
|
goto cleanup;
|
|
virBitmapSetAll(bitmap);
|
|
} else {
|
|
if (!(bitmap = virHostCPUGetAvailableCPUsBitmap()))
|
|
goto cleanup;
|
|
}
|
|
cpumask = bitmap;
|
|
}
|
|
}
|
|
if (virBitmapToData(cpumask, &info_ret[i]->cpumap,
|
|
&info_ret[i]->cpumaplen) < 0)
|
|
goto cleanup;
|
|
}
|
|
|
|
*info = g_steal_pointer(&info_ret);
|
|
ret = targetDef->niothreadids;
|
|
|
|
cleanup:
|
|
if (info_ret) {
|
|
for (i = 0; i < targetDef->niothreadids; i++)
|
|
virDomainIOThreadInfoFree(info_ret[i]);
|
|
VIR_FREE(info_ret);
|
|
}
|
|
|
|
return ret;
|
|
}
|