2014-02-18 14:08:10 +04:00
|
|
|
/*
|
2014-03-07 14:38:51 +01:00
|
|
|
* bhyve_command.c: bhyve command generation
|
2014-02-18 14:08:10 +04:00
|
|
|
*
|
|
|
|
* Copyright (C) 2014 Roman Bogorodskiy
|
|
|
|
*
|
|
|
|
* 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>
|
|
|
|
|
2014-11-08 11:48:35 -05:00
|
|
|
#include "bhyve_capabilities.h"
|
2014-02-18 14:08:10 +04:00
|
|
|
#include "bhyve_command.h"
|
2014-11-08 11:48:30 -05:00
|
|
|
#include "bhyve_domain.h"
|
2019-01-17 19:07:20 +04:00
|
|
|
#include "bhyve_conf.h"
|
2014-11-08 11:48:35 -05:00
|
|
|
#include "bhyve_driver.h"
|
2014-11-08 11:48:30 -05:00
|
|
|
#include "datatypes.h"
|
2014-02-18 14:08:10 +04:00
|
|
|
#include "viralloc.h"
|
|
|
|
#include "virfile.h"
|
|
|
|
#include "virstring.h"
|
|
|
|
#include "virlog.h"
|
|
|
|
#include "virnetdev.h"
|
|
|
|
#include "virnetdevbridge.h"
|
|
|
|
#include "virnetdevtap.h"
|
|
|
|
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_BHYVE
|
|
|
|
|
2014-02-28 12:16:17 +00:00
|
|
|
VIR_LOG_INIT("bhyve.bhyve_command");
|
|
|
|
|
2014-02-18 14:08:10 +04:00
|
|
|
static int
|
2020-02-24 01:46:20 -05:00
|
|
|
bhyveBuildNetArgStr(const virDomainDef *def,
|
2014-04-12 23:37:53 +04:00
|
|
|
virDomainNetDefPtr net,
|
2020-02-24 01:46:20 -05:00
|
|
|
bhyveConnPtr driver,
|
2014-04-12 23:37:53 +04:00
|
|
|
virCommandPtr cmd,
|
|
|
|
bool dryRun)
|
2014-02-18 14:08:10 +04:00
|
|
|
{
|
2014-04-12 23:37:53 +04:00
|
|
|
char macaddr[VIR_MAC_STRING_BUFLEN];
|
2014-02-18 14:08:10 +04:00
|
|
|
char *realifname = NULL;
|
2014-04-12 23:37:53 +04:00
|
|
|
char *brname = NULL;
|
2016-08-27 15:30:34 +03:00
|
|
|
char *nic_model = NULL;
|
2016-11-21 18:43:20 +03:00
|
|
|
int ret = -1;
|
2016-09-23 17:04:53 +02:00
|
|
|
virDomainNetType actualType = virDomainNetGetActualType(net);
|
2014-02-18 14:08:10 +04:00
|
|
|
|
2019-01-17 19:59:21 -05:00
|
|
|
if (net->model == VIR_DOMAIN_NET_MODEL_VIRTIO) {
|
2019-10-20 13:49:46 +02:00
|
|
|
nic_model = g_strdup("virtio-net");
|
2019-01-17 19:59:21 -05:00
|
|
|
} else if (net->model == VIR_DOMAIN_NET_MODEL_E1000) {
|
2020-02-22 01:38:41 -05:00
|
|
|
if ((bhyveDriverGetBhyveCaps(driver) & BHYVE_CAP_NET_E1000) != 0) {
|
2019-10-20 13:49:46 +02:00
|
|
|
nic_model = g_strdup("e1000");
|
2016-08-27 15:30:34 +03:00
|
|
|
} else {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("NIC model 'e1000' is not supported "
|
|
|
|
"by given bhyve binary"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
} else {
|
2019-01-17 19:59:21 -05:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("NIC model is not supported"));
|
2016-08-27 15:30:34 +03:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2014-04-12 23:37:53 +04:00
|
|
|
if (actualType == VIR_DOMAIN_NET_TYPE_BRIDGE) {
|
2019-10-20 13:49:46 +02:00
|
|
|
brname = g_strdup(virDomainNetGetActualBridgeName(net));
|
2014-04-12 23:37:53 +04:00
|
|
|
} else {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("Network type %d is not supported"),
|
|
|
|
virDomainNetGetActualType(net));
|
2016-11-21 18:43:20 +03:00
|
|
|
goto cleanup;
|
2014-02-18 14:08:10 +04:00
|
|
|
}
|
|
|
|
|
2014-04-12 23:37:53 +04:00
|
|
|
if (!dryRun) {
|
|
|
|
if (virNetDevTapCreateInBridgePort(brname, &net->ifname, &net->mac,
|
2014-09-11 17:15:24 +02:00
|
|
|
def->uuid, NULL, NULL, 0,
|
2014-04-12 23:37:53 +04:00
|
|
|
virDomainNetGetActualVirtPortProfile(net),
|
|
|
|
virDomainNetGetActualVlan(net),
|
2020-02-13 12:57:47 -05:00
|
|
|
virDomainNetGetActualPortOptionsIsolated(net),
|
2017-04-07 17:54:12 +02:00
|
|
|
NULL, 0, NULL,
|
2014-04-12 23:37:53 +04:00
|
|
|
VIR_NETDEV_TAP_CREATE_IFUP | VIR_NETDEV_TAP_CREATE_PERSIST) < 0) {
|
2016-11-21 18:43:20 +03:00
|
|
|
goto cleanup;
|
2014-02-18 14:08:10 +04:00
|
|
|
}
|
|
|
|
|
2014-04-13 13:27:03 +04:00
|
|
|
realifname = virNetDevTapGetRealDeviceName(net->ifname);
|
|
|
|
|
2016-11-21 18:43:20 +03:00
|
|
|
if (realifname == NULL)
|
|
|
|
goto cleanup;
|
2014-02-18 14:08:10 +04:00
|
|
|
|
2014-04-13 13:27:03 +04:00
|
|
|
VIR_DEBUG("%s -> %s", net->ifname, realifname);
|
|
|
|
/* hack on top of other hack: we need to set
|
|
|
|
* interface to 'UP' again after re-opening to find its
|
|
|
|
* name
|
|
|
|
*/
|
2016-11-21 18:43:20 +03:00
|
|
|
if (virNetDevSetOnline(net->ifname, true) != 0)
|
|
|
|
goto cleanup;
|
2014-04-13 13:27:03 +04:00
|
|
|
} else {
|
2019-10-20 13:49:46 +02:00
|
|
|
realifname = g_strdup("tap0");
|
2014-02-18 14:08:10 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
virCommandAddArg(cmd, "-s");
|
2016-08-27 15:30:34 +03:00
|
|
|
virCommandAddArgFormat(cmd, "%d:0,%s,%s,mac=%s",
|
|
|
|
net->info.addr.pci.slot, nic_model,
|
2014-03-20 09:39:21 +01:00
|
|
|
realifname, virMacAddrFormat(&net->mac, macaddr));
|
2016-11-21 18:43:20 +03:00
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
|
|
if (ret < 0)
|
|
|
|
VIR_FREE(net->ifname);
|
|
|
|
VIR_FREE(brname);
|
2014-04-12 23:37:53 +04:00
|
|
|
VIR_FREE(realifname);
|
2016-08-27 15:30:34 +03:00
|
|
|
VIR_FREE(nic_model);
|
2014-02-18 14:08:10 +04:00
|
|
|
|
2016-11-21 18:43:20 +03:00
|
|
|
return ret;
|
2014-02-18 14:08:10 +04:00
|
|
|
}
|
|
|
|
|
2014-03-15 16:30:01 +04:00
|
|
|
static int
|
|
|
|
bhyveBuildConsoleArgStr(const virDomainDef *def, virCommandPtr cmd)
|
|
|
|
{
|
|
|
|
virDomainChrDefPtr chr = NULL;
|
|
|
|
|
|
|
|
if (!def->nserials)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
chr = def->serials[0];
|
|
|
|
|
2016-10-21 07:45:54 -04:00
|
|
|
if (chr->source->type != VIR_DOMAIN_CHR_TYPE_NMDM) {
|
2014-03-15 16:30:01 +04:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("only nmdm console types are supported"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* bhyve supports only two ports: com1 and com2 */
|
|
|
|
if (chr->target.port > 2) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("only two serial ports are supported"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
virCommandAddArg(cmd, "-l");
|
|
|
|
virCommandAddArgFormat(cmd, "com%d,%s",
|
2016-10-21 07:45:54 -04:00
|
|
|
chr->target.port + 1, chr->source->data.file.path);
|
2014-03-15 16:30:01 +04:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-02-18 14:08:10 +04:00
|
|
|
static int
|
bhyve: fix SATA address allocation
As bhyve for a long time didn't have a notion of the explicit SATA
controller and created a controller for each drive, the bhyve driver
in libvirt acted in a similar way and didn't care about the SATA
controllers and assigned PCI addresses to drives directly, as
the generated command will look like this anyway:
2:0,ahci-hd,somedisk.img
This no longer makes sense because:
1. After commit c07d1c1c4f it's not possible to assign
PCI addresses to disks
2. Bhyve now supports multiple disk drives for a controller,
so it's going away from 1:1 controller:disk mapping, so
the controller object starts to make more sense now
So, this patch does the following:
- Assign PCI address to SATA controllers (previously we didn't do this)
- Assign disk addresses instead of PCI addresses for disks. Now, when
building a bhyve command, we take PCI address not from the disk
itself but from its controller
- Assign addresses at XML parsing time using the
assignAddressesCallback. This is done mainly for being able to
verify address allocation via xml2xml tests
- Adjust existing bhyvexml2{xml,argv} tests to chase the new
address allocation
This patch is largely based on work of Fabian Freyer.
2017-01-05 16:51:25 +04:00
|
|
|
bhyveBuildAHCIControllerArgStr(const virDomainDef *def,
|
|
|
|
virDomainControllerDefPtr controller,
|
2020-02-22 01:38:41 -05:00
|
|
|
bhyveConnPtr driver,
|
bhyve: fix SATA address allocation
As bhyve for a long time didn't have a notion of the explicit SATA
controller and created a controller for each drive, the bhyve driver
in libvirt acted in a similar way and didn't care about the SATA
controllers and assigned PCI addresses to drives directly, as
the generated command will look like this anyway:
2:0,ahci-hd,somedisk.img
This no longer makes sense because:
1. After commit c07d1c1c4f it's not possible to assign
PCI addresses to disks
2. Bhyve now supports multiple disk drives for a controller,
so it's going away from 1:1 controller:disk mapping, so
the controller object starts to make more sense now
So, this patch does the following:
- Assign PCI address to SATA controllers (previously we didn't do this)
- Assign disk addresses instead of PCI addresses for disks. Now, when
building a bhyve command, we take PCI address not from the disk
itself but from its controller
- Assign addresses at XML parsing time using the
assignAddressesCallback. This is done mainly for being able to
verify address allocation via xml2xml tests
- Adjust existing bhyvexml2{xml,argv} tests to chase the new
address allocation
This patch is largely based on work of Fabian Freyer.
2017-01-05 16:51:25 +04:00
|
|
|
virCommandPtr cmd)
|
2014-02-18 14:08:10 +04:00
|
|
|
{
|
2020-07-02 18:03:45 -04:00
|
|
|
g_auto(virBuffer) buf = VIR_BUFFER_INITIALIZER;
|
2014-07-19 19:15:26 +04:00
|
|
|
const char *disk_source;
|
bhyve: fix SATA address allocation
As bhyve for a long time didn't have a notion of the explicit SATA
controller and created a controller for each drive, the bhyve driver
in libvirt acted in a similar way and didn't care about the SATA
controllers and assigned PCI addresses to drives directly, as
the generated command will look like this anyway:
2:0,ahci-hd,somedisk.img
This no longer makes sense because:
1. After commit c07d1c1c4f it's not possible to assign
PCI addresses to disks
2. Bhyve now supports multiple disk drives for a controller,
so it's going away from 1:1 controller:disk mapping, so
the controller object starts to make more sense now
So, this patch does the following:
- Assign PCI address to SATA controllers (previously we didn't do this)
- Assign disk addresses instead of PCI addresses for disks. Now, when
building a bhyve command, we take PCI address not from the disk
itself but from its controller
- Assign addresses at XML parsing time using the
assignAddressesCallback. This is done mainly for being able to
verify address allocation via xml2xml tests
- Adjust existing bhyvexml2{xml,argv} tests to chase the new
address allocation
This patch is largely based on work of Fabian Freyer.
2017-01-05 16:51:25 +04:00
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < def->ndisks; i++) {
|
2020-07-02 18:03:45 -04:00
|
|
|
g_auto(virBuffer) device = VIR_BUFFER_INITIALIZER;
|
bhyve: fix SATA address allocation
As bhyve for a long time didn't have a notion of the explicit SATA
controller and created a controller for each drive, the bhyve driver
in libvirt acted in a similar way and didn't care about the SATA
controllers and assigned PCI addresses to drives directly, as
the generated command will look like this anyway:
2:0,ahci-hd,somedisk.img
This no longer makes sense because:
1. After commit c07d1c1c4f it's not possible to assign
PCI addresses to disks
2. Bhyve now supports multiple disk drives for a controller,
so it's going away from 1:1 controller:disk mapping, so
the controller object starts to make more sense now
So, this patch does the following:
- Assign PCI address to SATA controllers (previously we didn't do this)
- Assign disk addresses instead of PCI addresses for disks. Now, when
building a bhyve command, we take PCI address not from the disk
itself but from its controller
- Assign addresses at XML parsing time using the
assignAddressesCallback. This is done mainly for being able to
verify address allocation via xml2xml tests
- Adjust existing bhyvexml2{xml,argv} tests to chase the new
address allocation
This patch is largely based on work of Fabian Freyer.
2017-01-05 16:51:25 +04:00
|
|
|
virDomainDiskDefPtr disk = def->disks[i];
|
2020-07-02 18:03:45 -04:00
|
|
|
|
bhyve: fix SATA address allocation
As bhyve for a long time didn't have a notion of the explicit SATA
controller and created a controller for each drive, the bhyve driver
in libvirt acted in a similar way and didn't care about the SATA
controllers and assigned PCI addresses to drives directly, as
the generated command will look like this anyway:
2:0,ahci-hd,somedisk.img
This no longer makes sense because:
1. After commit c07d1c1c4f it's not possible to assign
PCI addresses to disks
2. Bhyve now supports multiple disk drives for a controller,
so it's going away from 1:1 controller:disk mapping, so
the controller object starts to make more sense now
So, this patch does the following:
- Assign PCI address to SATA controllers (previously we didn't do this)
- Assign disk addresses instead of PCI addresses for disks. Now, when
building a bhyve command, we take PCI address not from the disk
itself but from its controller
- Assign addresses at XML parsing time using the
assignAddressesCallback. This is done mainly for being able to
verify address allocation via xml2xml tests
- Adjust existing bhyvexml2{xml,argv} tests to chase the new
address allocation
This patch is largely based on work of Fabian Freyer.
2017-01-05 16:51:25 +04:00
|
|
|
if (disk->bus != VIR_DOMAIN_DISK_BUS_SATA)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (disk->info.addr.drive.controller != controller->idx)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
VIR_DEBUG("disk %zu controller %d", i, controller->idx);
|
|
|
|
|
|
|
|
if ((virDomainDiskGetType(disk) != VIR_STORAGE_TYPE_FILE) &&
|
|
|
|
(virDomainDiskGetType(disk) != VIR_STORAGE_TYPE_VOLUME)) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("unsupported disk type"));
|
2020-07-02 23:19:10 -04:00
|
|
|
return -1;
|
bhyve: fix SATA address allocation
As bhyve for a long time didn't have a notion of the explicit SATA
controller and created a controller for each drive, the bhyve driver
in libvirt acted in a similar way and didn't care about the SATA
controllers and assigned PCI addresses to drives directly, as
the generated command will look like this anyway:
2:0,ahci-hd,somedisk.img
This no longer makes sense because:
1. After commit c07d1c1c4f it's not possible to assign
PCI addresses to disks
2. Bhyve now supports multiple disk drives for a controller,
so it's going away from 1:1 controller:disk mapping, so
the controller object starts to make more sense now
So, this patch does the following:
- Assign PCI address to SATA controllers (previously we didn't do this)
- Assign disk addresses instead of PCI addresses for disks. Now, when
building a bhyve command, we take PCI address not from the disk
itself but from its controller
- Assign addresses at XML parsing time using the
assignAddressesCallback. This is done mainly for being able to
verify address allocation via xml2xml tests
- Adjust existing bhyvexml2{xml,argv} tests to chase the new
address allocation
This patch is largely based on work of Fabian Freyer.
2017-01-05 16:51:25 +04:00
|
|
|
}
|
|
|
|
|
2018-02-19 14:19:41 +01:00
|
|
|
if (virDomainDiskTranslateSourcePool(disk) < 0)
|
2020-07-02 23:19:10 -04:00
|
|
|
return -1;
|
bhyve: fix SATA address allocation
As bhyve for a long time didn't have a notion of the explicit SATA
controller and created a controller for each drive, the bhyve driver
in libvirt acted in a similar way and didn't care about the SATA
controllers and assigned PCI addresses to drives directly, as
the generated command will look like this anyway:
2:0,ahci-hd,somedisk.img
This no longer makes sense because:
1. After commit c07d1c1c4f it's not possible to assign
PCI addresses to disks
2. Bhyve now supports multiple disk drives for a controller,
so it's going away from 1:1 controller:disk mapping, so
the controller object starts to make more sense now
So, this patch does the following:
- Assign PCI address to SATA controllers (previously we didn't do this)
- Assign disk addresses instead of PCI addresses for disks. Now, when
building a bhyve command, we take PCI address not from the disk
itself but from its controller
- Assign addresses at XML parsing time using the
assignAddressesCallback. This is done mainly for being able to
verify address allocation via xml2xml tests
- Adjust existing bhyvexml2{xml,argv} tests to chase the new
address allocation
This patch is largely based on work of Fabian Freyer.
2017-01-05 16:51:25 +04:00
|
|
|
|
|
|
|
disk_source = virDomainDiskGetSource(disk);
|
|
|
|
|
|
|
|
if ((disk->device == VIR_DOMAIN_DISK_DEVICE_CDROM) &&
|
|
|
|
(disk_source == NULL)) {
|
2020-02-22 01:38:43 -05:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("cdrom device without source path "
|
|
|
|
"not supported"));
|
2020-07-02 23:19:10 -04:00
|
|
|
return -1;
|
bhyve: fix SATA address allocation
As bhyve for a long time didn't have a notion of the explicit SATA
controller and created a controller for each drive, the bhyve driver
in libvirt acted in a similar way and didn't care about the SATA
controllers and assigned PCI addresses to drives directly, as
the generated command will look like this anyway:
2:0,ahci-hd,somedisk.img
This no longer makes sense because:
1. After commit c07d1c1c4f it's not possible to assign
PCI addresses to disks
2. Bhyve now supports multiple disk drives for a controller,
so it's going away from 1:1 controller:disk mapping, so
the controller object starts to make more sense now
So, this patch does the following:
- Assign PCI address to SATA controllers (previously we didn't do this)
- Assign disk addresses instead of PCI addresses for disks. Now, when
building a bhyve command, we take PCI address not from the disk
itself but from its controller
- Assign addresses at XML parsing time using the
assignAddressesCallback. This is done mainly for being able to
verify address allocation via xml2xml tests
- Adjust existing bhyvexml2{xml,argv} tests to chase the new
address allocation
This patch is largely based on work of Fabian Freyer.
2017-01-05 16:51:25 +04:00
|
|
|
}
|
2014-02-18 14:08:10 +04:00
|
|
|
|
2014-07-19 19:15:26 +04:00
|
|
|
switch (disk->device) {
|
|
|
|
case VIR_DOMAIN_DISK_DEVICE_DISK:
|
2020-02-22 01:38:41 -05:00
|
|
|
if ((bhyveDriverGetBhyveCaps(driver) & BHYVE_CAP_AHCI32SLOT))
|
bhyve: fix SATA address allocation
As bhyve for a long time didn't have a notion of the explicit SATA
controller and created a controller for each drive, the bhyve driver
in libvirt acted in a similar way and didn't care about the SATA
controllers and assigned PCI addresses to drives directly, as
the generated command will look like this anyway:
2:0,ahci-hd,somedisk.img
This no longer makes sense because:
1. After commit c07d1c1c4f it's not possible to assign
PCI addresses to disks
2. Bhyve now supports multiple disk drives for a controller,
so it's going away from 1:1 controller:disk mapping, so
the controller object starts to make more sense now
So, this patch does the following:
- Assign PCI address to SATA controllers (previously we didn't do this)
- Assign disk addresses instead of PCI addresses for disks. Now, when
building a bhyve command, we take PCI address not from the disk
itself but from its controller
- Assign addresses at XML parsing time using the
assignAddressesCallback. This is done mainly for being able to
verify address allocation via xml2xml tests
- Adjust existing bhyvexml2{xml,argv} tests to chase the new
address allocation
This patch is largely based on work of Fabian Freyer.
2017-01-05 16:51:25 +04:00
|
|
|
virBufferAsprintf(&device, ",hd:%s", disk_source);
|
|
|
|
else
|
|
|
|
virBufferAsprintf(&device, "-hd,%s", disk_source);
|
2014-07-19 19:15:26 +04:00
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_DISK_DEVICE_CDROM:
|
2020-02-22 01:38:41 -05:00
|
|
|
if ((bhyveDriverGetBhyveCaps(driver) & BHYVE_CAP_AHCI32SLOT))
|
bhyve: fix SATA address allocation
As bhyve for a long time didn't have a notion of the explicit SATA
controller and created a controller for each drive, the bhyve driver
in libvirt acted in a similar way and didn't care about the SATA
controllers and assigned PCI addresses to drives directly, as
the generated command will look like this anyway:
2:0,ahci-hd,somedisk.img
This no longer makes sense because:
1. After commit c07d1c1c4f it's not possible to assign
PCI addresses to disks
2. Bhyve now supports multiple disk drives for a controller,
so it's going away from 1:1 controller:disk mapping, so
the controller object starts to make more sense now
So, this patch does the following:
- Assign PCI address to SATA controllers (previously we didn't do this)
- Assign disk addresses instead of PCI addresses for disks. Now, when
building a bhyve command, we take PCI address not from the disk
itself but from its controller
- Assign addresses at XML parsing time using the
assignAddressesCallback. This is done mainly for being able to
verify address allocation via xml2xml tests
- Adjust existing bhyvexml2{xml,argv} tests to chase the new
address allocation
This patch is largely based on work of Fabian Freyer.
2017-01-05 16:51:25 +04:00
|
|
|
virBufferAsprintf(&device, ",cd:%s", disk_source);
|
|
|
|
else
|
|
|
|
virBufferAsprintf(&device, "-cd,%s", disk_source);
|
2014-07-19 19:15:26 +04:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("unsupported disk device"));
|
2020-07-02 23:19:10 -04:00
|
|
|
return -1;
|
2014-07-19 19:15:26 +04:00
|
|
|
}
|
bhyve: fix SATA address allocation
As bhyve for a long time didn't have a notion of the explicit SATA
controller and created a controller for each drive, the bhyve driver
in libvirt acted in a similar way and didn't care about the SATA
controllers and assigned PCI addresses to drives directly, as
the generated command will look like this anyway:
2:0,ahci-hd,somedisk.img
This no longer makes sense because:
1. After commit c07d1c1c4f it's not possible to assign
PCI addresses to disks
2. Bhyve now supports multiple disk drives for a controller,
so it's going away from 1:1 controller:disk mapping, so
the controller object starts to make more sense now
So, this patch does the following:
- Assign PCI address to SATA controllers (previously we didn't do this)
- Assign disk addresses instead of PCI addresses for disks. Now, when
building a bhyve command, we take PCI address not from the disk
itself but from its controller
- Assign addresses at XML parsing time using the
assignAddressesCallback. This is done mainly for being able to
verify address allocation via xml2xml tests
- Adjust existing bhyvexml2{xml,argv} tests to chase the new
address allocation
This patch is largely based on work of Fabian Freyer.
2017-01-05 16:51:25 +04:00
|
|
|
virBufferAddBuffer(&buf, &device);
|
|
|
|
}
|
|
|
|
|
|
|
|
virCommandAddArg(cmd, "-s");
|
|
|
|
virCommandAddArgFormat(cmd, "%d:0,ahci%s",
|
|
|
|
controller->info.addr.pci.slot,
|
|
|
|
virBufferCurrentContent(&buf));
|
|
|
|
|
2020-07-02 23:19:10 -04:00
|
|
|
return 0;
|
bhyve: fix SATA address allocation
As bhyve for a long time didn't have a notion of the explicit SATA
controller and created a controller for each drive, the bhyve driver
in libvirt acted in a similar way and didn't care about the SATA
controllers and assigned PCI addresses to drives directly, as
the generated command will look like this anyway:
2:0,ahci-hd,somedisk.img
This no longer makes sense because:
1. After commit c07d1c1c4f it's not possible to assign
PCI addresses to disks
2. Bhyve now supports multiple disk drives for a controller,
so it's going away from 1:1 controller:disk mapping, so
the controller object starts to make more sense now
So, this patch does the following:
- Assign PCI address to SATA controllers (previously we didn't do this)
- Assign disk addresses instead of PCI addresses for disks. Now, when
building a bhyve command, we take PCI address not from the disk
itself but from its controller
- Assign addresses at XML parsing time using the
assignAddressesCallback. This is done mainly for being able to
verify address allocation via xml2xml tests
- Adjust existing bhyvexml2{xml,argv} tests to chase the new
address allocation
This patch is largely based on work of Fabian Freyer.
2017-01-05 16:51:25 +04:00
|
|
|
}
|
|
|
|
|
2017-03-20 17:58:51 +04:00
|
|
|
static int
|
|
|
|
bhyveBuildUSBControllerArgStr(const virDomainDef *def,
|
|
|
|
virDomainControllerDefPtr controller,
|
|
|
|
virCommandPtr cmd)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
int ndevices = 0;
|
|
|
|
|
|
|
|
for (i = 0; i < def->ninputs; i++) {
|
|
|
|
virDomainInputDefPtr input = def->inputs[i];
|
|
|
|
|
|
|
|
if (input->bus != VIR_DOMAIN_INPUT_BUS_USB) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("only USB input devices are supported"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (input->type != VIR_DOMAIN_INPUT_TYPE_TABLET) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("only tablet input devices are supported"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
ndevices++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ndevices != 1) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("only single input device is supported"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
virCommandAddArg(cmd, "-s");
|
|
|
|
virCommandAddArgFormat(cmd, "%d:%d,xhci,tablet",
|
|
|
|
controller->info.addr.pci.slot,
|
|
|
|
controller->info.addr.pci.function);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
bhyve: fix SATA address allocation
As bhyve for a long time didn't have a notion of the explicit SATA
controller and created a controller for each drive, the bhyve driver
in libvirt acted in a similar way and didn't care about the SATA
controllers and assigned PCI addresses to drives directly, as
the generated command will look like this anyway:
2:0,ahci-hd,somedisk.img
This no longer makes sense because:
1. After commit c07d1c1c4f it's not possible to assign
PCI addresses to disks
2. Bhyve now supports multiple disk drives for a controller,
so it's going away from 1:1 controller:disk mapping, so
the controller object starts to make more sense now
So, this patch does the following:
- Assign PCI address to SATA controllers (previously we didn't do this)
- Assign disk addresses instead of PCI addresses for disks. Now, when
building a bhyve command, we take PCI address not from the disk
itself but from its controller
- Assign addresses at XML parsing time using the
assignAddressesCallback. This is done mainly for being able to
verify address allocation via xml2xml tests
- Adjust existing bhyvexml2{xml,argv} tests to chase the new
address allocation
This patch is largely based on work of Fabian Freyer.
2017-01-05 16:51:25 +04:00
|
|
|
static int
|
2019-10-14 14:45:33 +02:00
|
|
|
bhyveBuildVirtIODiskArgStr(const virDomainDef *def G_GNUC_UNUSED,
|
2020-02-22 01:38:43 -05:00
|
|
|
virDomainDiskDefPtr disk,
|
|
|
|
virCommandPtr cmd)
|
bhyve: fix SATA address allocation
As bhyve for a long time didn't have a notion of the explicit SATA
controller and created a controller for each drive, the bhyve driver
in libvirt acted in a similar way and didn't care about the SATA
controllers and assigned PCI addresses to drives directly, as
the generated command will look like this anyway:
2:0,ahci-hd,somedisk.img
This no longer makes sense because:
1. After commit c07d1c1c4f it's not possible to assign
PCI addresses to disks
2. Bhyve now supports multiple disk drives for a controller,
so it's going away from 1:1 controller:disk mapping, so
the controller object starts to make more sense now
So, this patch does the following:
- Assign PCI address to SATA controllers (previously we didn't do this)
- Assign disk addresses instead of PCI addresses for disks. Now, when
building a bhyve command, we take PCI address not from the disk
itself but from its controller
- Assign addresses at XML parsing time using the
assignAddressesCallback. This is done mainly for being able to
verify address allocation via xml2xml tests
- Adjust existing bhyvexml2{xml,argv} tests to chase the new
address allocation
This patch is largely based on work of Fabian Freyer.
2017-01-05 16:51:25 +04:00
|
|
|
{
|
|
|
|
const char *disk_source;
|
|
|
|
|
2018-02-19 14:19:41 +01:00
|
|
|
if (virDomainDiskTranslateSourcePool(disk) < 0)
|
bhyve: fix SATA address allocation
As bhyve for a long time didn't have a notion of the explicit SATA
controller and created a controller for each drive, the bhyve driver
in libvirt acted in a similar way and didn't care about the SATA
controllers and assigned PCI addresses to drives directly, as
the generated command will look like this anyway:
2:0,ahci-hd,somedisk.img
This no longer makes sense because:
1. After commit c07d1c1c4f it's not possible to assign
PCI addresses to disks
2. Bhyve now supports multiple disk drives for a controller,
so it's going away from 1:1 controller:disk mapping, so
the controller object starts to make more sense now
So, this patch does the following:
- Assign PCI address to SATA controllers (previously we didn't do this)
- Assign disk addresses instead of PCI addresses for disks. Now, when
building a bhyve command, we take PCI address not from the disk
itself but from its controller
- Assign addresses at XML parsing time using the
assignAddressesCallback. This is done mainly for being able to
verify address allocation via xml2xml tests
- Adjust existing bhyvexml2{xml,argv} tests to chase the new
address allocation
This patch is largely based on work of Fabian Freyer.
2017-01-05 16:51:25 +04:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (disk->device != VIR_DOMAIN_DISK_DEVICE_DISK) {
|
2014-02-18 14:08:10 +04:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
bhyve: fix SATA address allocation
As bhyve for a long time didn't have a notion of the explicit SATA
controller and created a controller for each drive, the bhyve driver
in libvirt acted in a similar way and didn't care about the SATA
controllers and assigned PCI addresses to drives directly, as
the generated command will look like this anyway:
2:0,ahci-hd,somedisk.img
This no longer makes sense because:
1. After commit c07d1c1c4f it's not possible to assign
PCI addresses to disks
2. Bhyve now supports multiple disk drives for a controller,
so it's going away from 1:1 controller:disk mapping, so
the controller object starts to make more sense now
So, this patch does the following:
- Assign PCI address to SATA controllers (previously we didn't do this)
- Assign disk addresses instead of PCI addresses for disks. Now, when
building a bhyve command, we take PCI address not from the disk
itself but from its controller
- Assign addresses at XML parsing time using the
assignAddressesCallback. This is done mainly for being able to
verify address allocation via xml2xml tests
- Adjust existing bhyvexml2{xml,argv} tests to chase the new
address allocation
This patch is largely based on work of Fabian Freyer.
2017-01-05 16:51:25 +04:00
|
|
|
_("unsupported disk device"));
|
2014-02-18 14:08:10 +04:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2014-08-14 20:15:57 +04:00
|
|
|
if ((virDomainDiskGetType(disk) != VIR_STORAGE_TYPE_FILE) &&
|
|
|
|
(virDomainDiskGetType(disk) != VIR_STORAGE_TYPE_VOLUME)) {
|
2014-02-18 14:08:10 +04:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("unsupported disk type"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2014-07-19 19:15:26 +04:00
|
|
|
disk_source = virDomainDiskGetSource(disk);
|
|
|
|
|
2014-02-18 14:08:10 +04:00
|
|
|
virCommandAddArg(cmd, "-s");
|
bhyve: fix SATA address allocation
As bhyve for a long time didn't have a notion of the explicit SATA
controller and created a controller for each drive, the bhyve driver
in libvirt acted in a similar way and didn't care about the SATA
controllers and assigned PCI addresses to drives directly, as
the generated command will look like this anyway:
2:0,ahci-hd,somedisk.img
This no longer makes sense because:
1. After commit c07d1c1c4f it's not possible to assign
PCI addresses to disks
2. Bhyve now supports multiple disk drives for a controller,
so it's going away from 1:1 controller:disk mapping, so
the controller object starts to make more sense now
So, this patch does the following:
- Assign PCI address to SATA controllers (previously we didn't do this)
- Assign disk addresses instead of PCI addresses for disks. Now, when
building a bhyve command, we take PCI address not from the disk
itself but from its controller
- Assign addresses at XML parsing time using the
assignAddressesCallback. This is done mainly for being able to
verify address allocation via xml2xml tests
- Adjust existing bhyvexml2{xml,argv} tests to chase the new
address allocation
This patch is largely based on work of Fabian Freyer.
2017-01-05 16:51:25 +04:00
|
|
|
virCommandAddArgFormat(cmd, "%d:0,virtio-blk,%s",
|
|
|
|
disk->info.addr.pci.slot,
|
2014-07-19 19:15:26 +04:00
|
|
|
disk_source);
|
2014-02-18 14:08:10 +04:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-02-24 01:46:20 -05:00
|
|
|
static int
|
|
|
|
bhyveBuildDiskArgStr(const virDomainDef *def,
|
|
|
|
virDomainDiskDefPtr disk,
|
|
|
|
virCommandPtr cmd)
|
|
|
|
{
|
|
|
|
switch (disk->bus) {
|
|
|
|
case VIR_DOMAIN_DISK_BUS_SATA:
|
|
|
|
/* Handled by bhyveBuildAHCIControllerArgStr() */
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_DISK_BUS_VIRTIO:
|
|
|
|
if (bhyveBuildVirtIODiskArgStr(def, disk, cmd) < 0)
|
|
|
|
return -1;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("unsupported disk device"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
bhyveBuildControllerArgStr(const virDomainDef *def,
|
|
|
|
virDomainControllerDefPtr controller,
|
|
|
|
bhyveConnPtr driver,
|
|
|
|
virCommandPtr cmd,
|
2019-02-17 17:04:00 +04:00
|
|
|
unsigned *nusbcontrollers,
|
|
|
|
unsigned *nisacontrollers)
|
2020-02-24 01:46:20 -05:00
|
|
|
{
|
|
|
|
switch (controller->type) {
|
|
|
|
case VIR_DOMAIN_CONTROLLER_TYPE_PCI:
|
|
|
|
if (controller->model != VIR_DOMAIN_CONTROLLER_MODEL_PCI_ROOT) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("unsupported PCI controller model: "
|
|
|
|
"only PCI root supported"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_CONTROLLER_TYPE_SATA:
|
|
|
|
if (bhyveBuildAHCIControllerArgStr(def, controller, driver, cmd) < 0)
|
|
|
|
return -1;
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_CONTROLLER_TYPE_USB:
|
|
|
|
if (++*nusbcontrollers > 1) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("only single USB controller is supported"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bhyveBuildUSBControllerArgStr(def, controller, cmd) < 0)
|
|
|
|
return -1;
|
|
|
|
break;
|
2019-02-17 17:04:00 +04:00
|
|
|
case VIR_DOMAIN_CONTROLLER_TYPE_ISA:
|
|
|
|
if (++*nisacontrollers > 1) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
"%s", _("only single ISA controller is supported"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
virCommandAddArg(cmd, "-s");
|
|
|
|
virCommandAddArgFormat(cmd, "%d:0,lpc",
|
|
|
|
controller->info.addr.pci.slot);
|
|
|
|
break;
|
2020-02-24 01:46:20 -05:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-07-16 21:03:33 +00:00
|
|
|
static int
|
2017-05-08 13:36:43 +03:00
|
|
|
bhyveBuildGraphicsArgStr(const virDomainDef *def,
|
2016-07-16 21:03:33 +00:00
|
|
|
virDomainGraphicsDefPtr graphics,
|
|
|
|
virDomainVideoDefPtr video,
|
2020-02-22 01:38:41 -05:00
|
|
|
bhyveConnPtr driver,
|
2017-05-08 13:36:43 +03:00
|
|
|
virCommandPtr cmd,
|
|
|
|
bool dryRun)
|
2016-07-16 21:03:33 +00:00
|
|
|
{
|
2020-07-02 18:03:45 -04:00
|
|
|
g_auto(virBuffer) opt = VIR_BUFFER_INITIALIZER;
|
2016-07-16 21:03:33 +00:00
|
|
|
virDomainGraphicsListenDefPtr glisten = NULL;
|
|
|
|
bool escapeAddr;
|
2017-05-08 13:36:43 +03:00
|
|
|
unsigned short port;
|
|
|
|
|
2020-02-22 01:38:41 -05:00
|
|
|
if (!(bhyveDriverGetBhyveCaps(driver) & BHYVE_CAP_LPC_BOOTROM) ||
|
2016-07-16 21:03:33 +00:00
|
|
|
def->os.bootloader ||
|
|
|
|
!def->os.loader) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("Graphics are only supported"
|
|
|
|
" when booting using UEFI"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2020-02-22 01:38:41 -05:00
|
|
|
if (!(bhyveDriverGetBhyveCaps(driver) & BHYVE_CAP_FBUF)) {
|
2016-07-16 21:03:33 +00:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("Bhyve version does not support framebuffer"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (graphics->type != VIR_DOMAIN_GRAPHICS_TYPE_VNC) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("Only VNC supported"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(glisten = virDomainGraphicsGetListen(graphics, 0))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Missing listen element"));
|
2020-07-02 23:19:10 -04:00
|
|
|
return -1;
|
2016-07-16 21:03:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virBufferAsprintf(&opt, "%d:%d,fbuf", video->info.addr.pci.slot, video->info.addr.pci.function);
|
|
|
|
|
|
|
|
switch (glisten->type) {
|
|
|
|
case VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_ADDRESS:
|
|
|
|
case VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_NETWORK:
|
|
|
|
virBufferAddLit(&opt, ",tcp=");
|
|
|
|
|
|
|
|
if (!graphics->data.vnc.autoport &&
|
|
|
|
(graphics->data.vnc.port < 5900 ||
|
|
|
|
graphics->data.vnc.port > 65535)) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("vnc port must be in range [5900,65535]"));
|
2020-07-02 23:19:10 -04:00
|
|
|
return -1;
|
2016-07-16 21:03:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (glisten->address) {
|
|
|
|
escapeAddr = strchr(glisten->address, ':') != NULL;
|
|
|
|
if (escapeAddr)
|
|
|
|
virBufferAsprintf(&opt, "[%s]", glisten->address);
|
|
|
|
else
|
|
|
|
virBufferAdd(&opt, glisten->address, -1);
|
|
|
|
}
|
|
|
|
|
2017-05-08 13:36:43 +03:00
|
|
|
if (!dryRun) {
|
|
|
|
if (graphics->data.vnc.autoport) {
|
|
|
|
if (virPortAllocatorAcquire(driver->remotePorts, &port) < 0)
|
|
|
|
return -1;
|
|
|
|
graphics->data.vnc.port = port;
|
|
|
|
} else {
|
2018-02-06 12:09:10 +03:00
|
|
|
if (virPortAllocatorSetUsed(graphics->data.vnc.port) < 0)
|
2017-05-08 13:36:43 +03:00
|
|
|
VIR_WARN("Failed to mark VNC port '%d' as used by '%s'",
|
|
|
|
graphics->data.vnc.port, def->name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-07-16 21:03:33 +00:00
|
|
|
virBufferAsprintf(&opt, ":%d", graphics->data.vnc.port);
|
|
|
|
break;
|
2018-02-22 09:19:51 +00:00
|
|
|
case VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_SOCKET:
|
|
|
|
case VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_NONE:
|
2016-07-16 21:03:33 +00:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("Unsupported listen type"));
|
2020-07-02 23:19:10 -04:00
|
|
|
return -1;
|
2018-02-22 09:19:51 +00:00
|
|
|
case VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_LAST:
|
|
|
|
default:
|
|
|
|
virReportEnumRangeError(virDomainGraphicsListenType, glisten->type);
|
2020-07-02 23:19:10 -04:00
|
|
|
return -1;
|
2016-07-16 21:03:33 +00:00
|
|
|
}
|
|
|
|
|
2020-05-06 13:35:55 +00:00
|
|
|
if (graphics->data.vnc.auth.passwd) {
|
|
|
|
if (!(bhyveDriverGetBhyveCaps(driver) & BHYVE_CAP_VNC_PASSWORD)) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("VNC Password authentication not supported "
|
|
|
|
"by bhyve"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strchr(graphics->data.vnc.auth.passwd, ',')) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("Password may not contain ',' character"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
virBufferAsprintf(&opt, ",password=%s", graphics->data.vnc.auth.passwd);
|
|
|
|
} else {
|
|
|
|
if (!(bhyveDriverGetBhyveCaps(driver) & BHYVE_CAP_VNC_PASSWORD))
|
|
|
|
VIR_WARN("%s", _("Security warning: VNC auth is not supported."));
|
|
|
|
else
|
|
|
|
VIR_WARN("%s", _("Security warning: VNC is used without authentication."));
|
|
|
|
}
|
|
|
|
|
2020-05-06 13:35:53 +00:00
|
|
|
if (video->res)
|
|
|
|
virBufferAsprintf(&opt, ",w=%d,h=%d", video->res->x, video->res->y);
|
|
|
|
|
2017-05-09 14:48:30 +04:00
|
|
|
if (video->driver)
|
|
|
|
virBufferAsprintf(&opt, ",vga=%s",
|
|
|
|
virDomainVideoVGAConfTypeToString(video->driver->vgaconf));
|
|
|
|
|
2016-07-16 21:03:33 +00:00
|
|
|
virCommandAddArg(cmd, "-s");
|
|
|
|
virCommandAddArgBuffer(cmd, &opt);
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2020-07-14 18:44:13 +04:00
|
|
|
static int
|
|
|
|
bhyveBuildSoundArgStr(const virDomainDef *def G_GNUC_UNUSED,
|
|
|
|
virDomainSoundDefPtr sound,
|
2020-07-18 15:23:28 +04:00
|
|
|
virDomainAudioDefPtr audio,
|
2020-07-14 18:44:13 +04:00
|
|
|
bhyveConnPtr driver,
|
|
|
|
virCommandPtr cmd)
|
|
|
|
{
|
2020-07-18 15:23:28 +04:00
|
|
|
g_auto(virBuffer) params = VIR_BUFFER_INITIALIZER;
|
|
|
|
|
2020-07-14 18:44:13 +04:00
|
|
|
if (!(bhyveDriverGetBhyveCaps(driver) & BHYVE_CAP_SOUND_HDA)) {
|
|
|
|
/* Currently, bhyve only supports "hda" sound devices, so
|
|
|
|
if it's not supported, sound devices are not supported at all */
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("Sound devices emulation is not supported "
|
|
|
|
"by given bhyve binary"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sound->model != VIR_DOMAIN_SOUND_MODEL_ICH7) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("Sound device model is not supported"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
virCommandAddArg(cmd, "-s");
|
2020-07-18 15:23:28 +04:00
|
|
|
|
|
|
|
if (audio) {
|
|
|
|
switch ((virDomainAudioType) audio->type) {
|
|
|
|
case VIR_DOMAIN_AUDIO_TYPE_OSS:
|
2021-03-02 14:59:28 +00:00
|
|
|
if (audio->backend.oss.input.dev)
|
2020-07-18 15:23:28 +04:00
|
|
|
virBufferAsprintf(¶ms, ",play=%s",
|
2021-03-02 14:59:28 +00:00
|
|
|
audio->backend.oss.input.dev);
|
2020-07-18 15:23:28 +04:00
|
|
|
|
2021-03-02 14:59:28 +00:00
|
|
|
if (audio->backend.oss.output.dev)
|
2020-07-18 15:23:28 +04:00
|
|
|
virBufferAsprintf(¶ms, ",rec=%s",
|
2021-03-02 14:59:28 +00:00
|
|
|
audio->backend.oss.output.dev);
|
2020-07-18 15:23:28 +04:00
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_AUDIO_TYPE_LAST:
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("unsupported audio backend '%s'"),
|
|
|
|
virDomainAudioTypeTypeToString(audio->type));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
virCommandAddArgFormat(cmd, "%d:%d,hda%s",
|
2020-07-14 18:44:13 +04:00
|
|
|
sound->info.addr.pci.slot,
|
2020-07-18 15:23:28 +04:00
|
|
|
sound->info.addr.pci.function,
|
|
|
|
virBufferCurrentContent(¶ms));
|
|
|
|
|
2020-07-14 18:44:13 +04:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-10-07 20:07:04 +04:00
|
|
|
static int
|
|
|
|
bhyveBuildFSArgStr(const virDomainDef *def G_GNUC_UNUSED,
|
|
|
|
virDomainFSDefPtr fs,
|
|
|
|
virCommandPtr cmd)
|
|
|
|
{
|
|
|
|
g_auto(virBuffer) params = VIR_BUFFER_INITIALIZER;
|
|
|
|
|
|
|
|
switch ((virDomainFSType) fs->type) {
|
|
|
|
case VIR_DOMAIN_FS_TYPE_MOUNT:
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_FS_TYPE_BLOCK:
|
|
|
|
case VIR_DOMAIN_FS_TYPE_FILE:
|
|
|
|
case VIR_DOMAIN_FS_TYPE_TEMPLATE:
|
|
|
|
case VIR_DOMAIN_FS_TYPE_RAM:
|
|
|
|
case VIR_DOMAIN_FS_TYPE_BIND:
|
|
|
|
case VIR_DOMAIN_FS_TYPE_VOLUME:
|
|
|
|
case VIR_DOMAIN_FS_TYPE_LAST:
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("unsupported filesystem type '%s'"),
|
|
|
|
virDomainFSTypeToString(fs->type));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (fs->fsdriver) {
|
|
|
|
case VIR_DOMAIN_FS_DRIVER_TYPE_DEFAULT:
|
|
|
|
/* The only supported driver by bhyve currently */
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_FS_DRIVER_TYPE_VIRTIOFS:
|
|
|
|
case VIR_DOMAIN_FS_DRIVER_TYPE_PATH:
|
|
|
|
case VIR_DOMAIN_FS_DRIVER_TYPE_HANDLE:
|
|
|
|
case VIR_DOMAIN_FS_DRIVER_TYPE_LOOP:
|
|
|
|
case VIR_DOMAIN_FS_DRIVER_TYPE_NBD:
|
|
|
|
case VIR_DOMAIN_FS_DRIVER_TYPE_PLOOP:
|
|
|
|
case VIR_DOMAIN_FS_DRIVER_TYPE_LAST:
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("unsupported filesystem driver '%s'"),
|
|
|
|
virDomainFSDriverTypeToString(fs->fsdriver));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (fs->accessmode) {
|
|
|
|
case VIR_DOMAIN_FS_ACCESSMODE_PASSTHROUGH:
|
|
|
|
/* This is the only supported mode for now, does not need specific configuration */
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_FS_ACCESSMODE_MAPPED:
|
|
|
|
case VIR_DOMAIN_FS_ACCESSMODE_SQUASH:
|
|
|
|
case VIR_DOMAIN_FS_ACCESSMODE_LAST:
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("unsupported filesystem accessmode '%s'"),
|
|
|
|
virDomainFSAccessModeTypeToString(fs->accessmode));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fs->readonly)
|
|
|
|
virBufferAddLit(¶ms, ",ro");
|
|
|
|
|
|
|
|
virCommandAddArg(cmd, "-s");
|
|
|
|
virCommandAddArgFormat(cmd, "%d:%d,virtio-9p,%s=%s%s",
|
|
|
|
fs->info.addr.pci.slot,
|
|
|
|
fs->info.addr.pci.function,
|
|
|
|
fs->dst,
|
2020-10-10 08:08:07 +04:00
|
|
|
fs->src->path,
|
2020-10-07 20:07:04 +04:00
|
|
|
virBufferCurrentContent(¶ms));
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-02-18 14:08:10 +04:00
|
|
|
virCommandPtr
|
2020-02-24 01:46:20 -05:00
|
|
|
virBhyveProcessBuildBhyveCmd(bhyveConnPtr driver, virDomainDefPtr def,
|
|
|
|
bool dryRun)
|
2014-02-18 14:08:10 +04:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
* /usr/sbin/bhyve -c 2 -m 256 -AI -H -P \
|
|
|
|
* -s 0:0,hostbridge \
|
|
|
|
* -s 1:0,virtio-net,tap0 \
|
|
|
|
* -s 2:0,ahci-hd,${IMG} \
|
|
|
|
* -S 31,uart,stdio \
|
|
|
|
* vm0
|
|
|
|
*/
|
|
|
|
virCommandPtr cmd = virCommandNew(BHYVE);
|
2020-02-24 01:46:20 -05:00
|
|
|
size_t i;
|
|
|
|
unsigned nusbcontrollers = 0;
|
2019-02-17 17:04:00 +04:00
|
|
|
unsigned nisacontrollers = 0;
|
2020-02-24 01:46:20 -05:00
|
|
|
unsigned nvcpus = virDomainDefGetVcpus(def);
|
2014-02-18 14:08:10 +04:00
|
|
|
|
|
|
|
/* CPUs */
|
|
|
|
virCommandAddArg(cmd, "-c");
|
2018-05-21 18:53:36 +04:00
|
|
|
if (def->cpu && def->cpu->sockets) {
|
2020-01-17 12:31:16 +00:00
|
|
|
if (def->cpu->dies != 1) {
|
2019-12-16 11:16:51 +00:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("Only 1 die per socket is supported"));
|
2020-01-17 12:31:16 +00:00
|
|
|
goto error;
|
2019-12-16 11:16:51 +00:00
|
|
|
}
|
2018-05-21 18:53:36 +04:00
|
|
|
if (nvcpus != def->cpu->sockets * def->cpu->cores * def->cpu->threads) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("Invalid CPU topology: total number of vCPUs "
|
|
|
|
"must equal the product of sockets, cores, "
|
|
|
|
"and threads"));
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2020-02-22 01:38:41 -05:00
|
|
|
if ((bhyveDriverGetBhyveCaps(driver) & BHYVE_CAP_CPUTOPOLOGY) != 0) {
|
2018-05-21 18:53:36 +04:00
|
|
|
virCommandAddArgFormat(cmd, "cpus=%d,sockets=%d,cores=%d,threads=%d",
|
|
|
|
nvcpus,
|
|
|
|
def->cpu->sockets,
|
|
|
|
def->cpu->cores,
|
|
|
|
def->cpu->threads);
|
|
|
|
} else {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("Installed bhyve binary does not support "
|
|
|
|
"defining CPU topology"));
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
virCommandAddArgFormat(cmd, "%d", nvcpus);
|
|
|
|
}
|
2014-02-18 14:08:10 +04:00
|
|
|
|
|
|
|
/* Memory */
|
|
|
|
virCommandAddArg(cmd, "-m");
|
|
|
|
virCommandAddArgFormat(cmd, "%llu",
|
2015-02-17 18:01:09 +01:00
|
|
|
VIR_DIV_UP(virDomainDefGetMemoryInitial(def), 1024));
|
2014-02-18 14:08:10 +04:00
|
|
|
|
2018-05-09 03:01:35 +02:00
|
|
|
if (def->mem.locked)
|
|
|
|
virCommandAddArg(cmd, "-S"); /* Wire guest memory */
|
|
|
|
|
2014-02-18 14:08:10 +04:00
|
|
|
/* Options */
|
2014-06-27 17:18:53 +02:00
|
|
|
if (def->features[VIR_DOMAIN_FEATURE_ACPI] == VIR_TRISTATE_SWITCH_ON)
|
2014-02-18 14:08:10 +04:00
|
|
|
virCommandAddArg(cmd, "-A"); /* Create an ACPI table */
|
2014-06-27 17:18:53 +02:00
|
|
|
if (def->features[VIR_DOMAIN_FEATURE_APIC] == VIR_TRISTATE_SWITCH_ON)
|
2014-02-18 14:08:10 +04:00
|
|
|
virCommandAddArg(cmd, "-I"); /* Present ioapic to the guest */
|
2019-01-25 19:27:58 +04:00
|
|
|
if (def->features[VIR_DOMAIN_FEATURE_MSRS] == VIR_TRISTATE_SWITCH_ON) {
|
|
|
|
if (def->msrs_features[VIR_DOMAIN_MSRS_UNKNOWN] == VIR_DOMAIN_MSRS_UNKNOWN_IGNORE)
|
|
|
|
virCommandAddArg(cmd, "-w");
|
|
|
|
}
|
2014-02-18 14:08:10 +04:00
|
|
|
|
2015-07-19 11:20:35 +03:00
|
|
|
switch (def->clock.offset) {
|
|
|
|
case VIR_DOMAIN_CLOCK_OFFSET_LOCALTIME:
|
|
|
|
/* used by default in bhyve */
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_CLOCK_OFFSET_UTC:
|
2020-02-22 01:38:41 -05:00
|
|
|
if ((bhyveDriverGetBhyveCaps(driver) & BHYVE_CAP_RTC_UTC) != 0) {
|
2015-07-19 11:20:35 +03:00
|
|
|
virCommandAddArg(cmd, "-u");
|
|
|
|
} else {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("Installed bhyve binary does not support "
|
|
|
|
"UTC clock"));
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("unsupported clock offset '%s'"),
|
|
|
|
virDomainClockOffsetTypeToString(def->clock.offset));
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2014-02-18 14:08:10 +04:00
|
|
|
/* Clarification about -H and -P flags from Peter Grehan:
|
|
|
|
* -H and -P flags force the guest to exit when it executes IA32 HLT and PAUSE
|
|
|
|
* instructions respectively.
|
|
|
|
*
|
|
|
|
* For the HLT exit, bhyve uses that to infer that the guest is idling and can
|
|
|
|
* be put to sleep until an external event arrives. If this option is not used,
|
|
|
|
* the guest will always use 100% of CPU on the host.
|
|
|
|
*
|
|
|
|
* The PAUSE exit is most useful when there are large numbers of guest VMs running,
|
|
|
|
* since it forces the guest to exit when it spins on a lock acquisition.
|
|
|
|
*/
|
|
|
|
virCommandAddArg(cmd, "-H"); /* vmexit from guest on hlt */
|
|
|
|
virCommandAddArg(cmd, "-P"); /* vmexit from guest on pause */
|
|
|
|
|
2014-03-15 19:17:14 +04:00
|
|
|
virCommandAddArgList(cmd, "-s", "0:0,hostbridge", NULL);
|
2016-06-30 02:29:15 +00:00
|
|
|
|
|
|
|
if (def->os.bootloader == NULL &&
|
|
|
|
def->os.loader) {
|
2020-02-22 01:38:41 -05:00
|
|
|
if ((bhyveDriverGetBhyveCaps(driver) & BHYVE_CAP_LPC_BOOTROM)) {
|
2016-06-30 02:29:15 +00:00
|
|
|
virCommandAddArg(cmd, "-l");
|
|
|
|
virCommandAddArgFormat(cmd, "bootrom,%s", def->os.loader->path);
|
|
|
|
} else {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("Installed bhyve binary does not support "
|
|
|
|
"UEFI loader"));
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-02-18 14:08:10 +04:00
|
|
|
/* Devices */
|
bhyve: fix SATA address allocation
As bhyve for a long time didn't have a notion of the explicit SATA
controller and created a controller for each drive, the bhyve driver
in libvirt acted in a similar way and didn't care about the SATA
controllers and assigned PCI addresses to drives directly, as
the generated command will look like this anyway:
2:0,ahci-hd,somedisk.img
This no longer makes sense because:
1. After commit c07d1c1c4f it's not possible to assign
PCI addresses to disks
2. Bhyve now supports multiple disk drives for a controller,
so it's going away from 1:1 controller:disk mapping, so
the controller object starts to make more sense now
So, this patch does the following:
- Assign PCI address to SATA controllers (previously we didn't do this)
- Assign disk addresses instead of PCI addresses for disks. Now, when
building a bhyve command, we take PCI address not from the disk
itself but from its controller
- Assign addresses at XML parsing time using the
assignAddressesCallback. This is done mainly for being able to
verify address allocation via xml2xml tests
- Adjust existing bhyvexml2{xml,argv} tests to chase the new
address allocation
This patch is largely based on work of Fabian Freyer.
2017-01-05 16:51:25 +04:00
|
|
|
for (i = 0; i < def->ncontrollers; i++) {
|
2020-02-24 01:46:20 -05:00
|
|
|
if (bhyveBuildControllerArgStr(def, def->controllers[i], driver, cmd,
|
2019-02-17 17:04:00 +04:00
|
|
|
&nusbcontrollers, &nisacontrollers) < 0)
|
2020-02-24 01:46:20 -05:00
|
|
|
goto error;
|
bhyve: fix SATA address allocation
As bhyve for a long time didn't have a notion of the explicit SATA
controller and created a controller for each drive, the bhyve driver
in libvirt acted in a similar way and didn't care about the SATA
controllers and assigned PCI addresses to drives directly, as
the generated command will look like this anyway:
2:0,ahci-hd,somedisk.img
This no longer makes sense because:
1. After commit c07d1c1c4f it's not possible to assign
PCI addresses to disks
2. Bhyve now supports multiple disk drives for a controller,
so it's going away from 1:1 controller:disk mapping, so
the controller object starts to make more sense now
So, this patch does the following:
- Assign PCI address to SATA controllers (previously we didn't do this)
- Assign disk addresses instead of PCI addresses for disks. Now, when
building a bhyve command, we take PCI address not from the disk
itself but from its controller
- Assign addresses at XML parsing time using the
assignAddressesCallback. This is done mainly for being able to
verify address allocation via xml2xml tests
- Adjust existing bhyvexml2{xml,argv} tests to chase the new
address allocation
This patch is largely based on work of Fabian Freyer.
2017-01-05 16:51:25 +04:00
|
|
|
}
|
2014-04-12 23:37:53 +04:00
|
|
|
for (i = 0; i < def->nnets; i++) {
|
2020-02-24 01:46:20 -05:00
|
|
|
if (bhyveBuildNetArgStr(def, def->nets[i], driver, cmd, dryRun) < 0)
|
2014-04-12 23:37:53 +04:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
for (i = 0; i < def->ndisks; i++) {
|
2020-02-24 01:46:20 -05:00
|
|
|
if (bhyveBuildDiskArgStr(def, def->disks[i], cmd) < 0)
|
2014-04-12 23:37:53 +04:00
|
|
|
goto error;
|
|
|
|
}
|
2016-06-30 02:29:15 +00:00
|
|
|
|
2016-07-16 21:03:33 +00:00
|
|
|
if (def->ngraphics && def->nvideos) {
|
|
|
|
if (def->ngraphics == 1 && def->nvideos == 1) {
|
2017-05-08 13:36:43 +03:00
|
|
|
if (bhyveBuildGraphicsArgStr(def, def->graphics[0], def->videos[0],
|
2020-02-22 01:38:41 -05:00
|
|
|
driver, cmd, dryRun) < 0)
|
2016-07-16 21:03:33 +00:00
|
|
|
goto error;
|
|
|
|
} else {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("Multiple graphics devices are not supported"));
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-14 18:44:13 +04:00
|
|
|
for (i = 0; i < def->nsounds; i++) {
|
2020-07-18 15:23:28 +04:00
|
|
|
if (bhyveBuildSoundArgStr(def, def->sounds[i],
|
|
|
|
virDomainDefFindAudioForSound(def, def->sounds[i]),
|
|
|
|
driver, cmd) < 0)
|
2020-07-14 18:44:13 +04:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2020-10-07 20:07:04 +04:00
|
|
|
for (i = 0; i < def->nfss; i++) {
|
|
|
|
if (bhyveBuildFSArgStr(def, def->fss[i], cmd) < 0)
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2014-04-13 13:27:03 +04:00
|
|
|
if (bhyveBuildConsoleArgStr(def, cmd) < 0)
|
2014-03-15 16:30:01 +04:00
|
|
|
goto error;
|
2016-06-30 02:29:15 +00:00
|
|
|
|
2019-01-17 19:07:20 +04:00
|
|
|
if (def->namespaceData) {
|
|
|
|
bhyveDomainCmdlineDefPtr bhyvecmd;
|
|
|
|
|
2019-01-31 16:37:01 +04:00
|
|
|
VIR_WARN("Booting the guest using command line pass-through feature, "
|
|
|
|
"which could potentially cause inconsistent state and "
|
|
|
|
"upgrade issues");
|
|
|
|
|
2019-01-17 19:07:20 +04:00
|
|
|
bhyvecmd = def->namespaceData;
|
|
|
|
for (i = 0; i < bhyvecmd->num_args; i++)
|
|
|
|
virCommandAddArg(cmd, bhyvecmd->args[i]);
|
|
|
|
}
|
|
|
|
|
2014-04-13 13:27:03 +04:00
|
|
|
virCommandAddArg(cmd, def->name);
|
2014-02-18 14:08:10 +04:00
|
|
|
|
|
|
|
return cmd;
|
|
|
|
|
2014-03-25 07:57:22 +01:00
|
|
|
error:
|
2014-02-18 14:08:10 +04:00
|
|
|
virCommandFree(cmd);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
virCommandPtr
|
2019-10-14 14:45:33 +02:00
|
|
|
virBhyveProcessBuildDestroyCmd(bhyveConnPtr driver G_GNUC_UNUSED,
|
2014-04-13 13:27:03 +04:00
|
|
|
virDomainDefPtr def)
|
2014-02-18 14:08:10 +04:00
|
|
|
{
|
|
|
|
virCommandPtr cmd = virCommandNew(BHYVECTL);
|
|
|
|
|
|
|
|
virCommandAddArg(cmd, "--destroy");
|
2014-04-13 13:27:03 +04:00
|
|
|
virCommandAddArgPair(cmd, "--vm", def->name);
|
2014-02-18 14:08:10 +04:00
|
|
|
|
|
|
|
return cmd;
|
|
|
|
}
|
|
|
|
|
2014-11-08 11:48:30 -05:00
|
|
|
static void
|
|
|
|
virAppendBootloaderArgs(virCommandPtr cmd, virDomainDefPtr def)
|
|
|
|
{
|
|
|
|
char **blargs;
|
|
|
|
|
|
|
|
/* XXX: Handle quoted? */
|
2021-02-05 18:35:07 +01:00
|
|
|
blargs = g_strsplit(def->os.bootloaderArgs, " ", 0);
|
2014-11-08 11:48:30 -05:00
|
|
|
virCommandAddArgSet(cmd, (const char * const *)blargs);
|
2020-08-02 19:36:03 +02:00
|
|
|
g_strfreev(blargs);
|
2014-11-08 11:48:30 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
static virCommandPtr
|
|
|
|
virBhyveProcessBuildBhyveloadCmd(virDomainDefPtr def, virDomainDiskDefPtr disk)
|
2014-02-18 14:08:10 +04:00
|
|
|
{
|
|
|
|
virCommandPtr cmd;
|
|
|
|
|
2014-11-08 11:48:30 -05:00
|
|
|
cmd = virCommandNew(BHYVELOAD);
|
|
|
|
|
|
|
|
if (def->os.bootloaderArgs == NULL) {
|
|
|
|
VIR_DEBUG("bhyveload with default arguments");
|
|
|
|
|
|
|
|
/* Memory (MB) */
|
|
|
|
virCommandAddArg(cmd, "-m");
|
|
|
|
virCommandAddArgFormat(cmd, "%llu",
|
2015-02-17 18:01:09 +01:00
|
|
|
VIR_DIV_UP(virDomainDefGetMemoryInitial(def), 1024));
|
2014-11-08 11:48:30 -05:00
|
|
|
|
|
|
|
/* Image path */
|
|
|
|
virCommandAddArg(cmd, "-d");
|
|
|
|
virCommandAddArg(cmd, virDomainDiskGetSource(disk));
|
|
|
|
|
|
|
|
/* VM name */
|
|
|
|
virCommandAddArg(cmd, def->name);
|
|
|
|
} else {
|
|
|
|
VIR_DEBUG("bhyveload with arguments");
|
|
|
|
virAppendBootloaderArgs(cmd, def);
|
|
|
|
}
|
|
|
|
|
|
|
|
return cmd;
|
|
|
|
}
|
|
|
|
|
|
|
|
static virCommandPtr
|
|
|
|
virBhyveProcessBuildCustomLoaderCmd(virDomainDefPtr def)
|
|
|
|
{
|
|
|
|
virCommandPtr cmd;
|
|
|
|
|
|
|
|
if (def->os.bootloaderArgs == NULL) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("Custom loader requires explicit %s configuration"),
|
|
|
|
"bootloader_args");
|
2014-02-18 14:08:10 +04:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2014-11-08 11:48:30 -05:00
|
|
|
VIR_DEBUG("custom loader '%s' with arguments", def->os.bootloader);
|
|
|
|
|
|
|
|
cmd = virCommandNew(def->os.bootloader);
|
|
|
|
virAppendBootloaderArgs(cmd, def);
|
|
|
|
return cmd;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool
|
2018-02-19 14:19:41 +01:00
|
|
|
virBhyveUsableDisk(virDomainDiskDefPtr disk)
|
2014-11-08 11:48:30 -05:00
|
|
|
{
|
2018-02-19 14:19:41 +01:00
|
|
|
if (virDomainDiskTranslateSourcePool(disk) < 0)
|
2014-11-08 11:48:30 -05:00
|
|
|
return false;
|
2014-08-14 20:15:57 +04:00
|
|
|
|
2014-07-19 19:15:26 +04:00
|
|
|
if ((disk->device != VIR_DOMAIN_DISK_DEVICE_DISK) &&
|
|
|
|
(disk->device != VIR_DOMAIN_DISK_DEVICE_CDROM)) {
|
2014-02-18 14:08:10 +04:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("unsupported disk device"));
|
2014-11-08 11:48:30 -05:00
|
|
|
return false;
|
2014-02-18 14:08:10 +04:00
|
|
|
}
|
|
|
|
|
2014-08-14 20:15:57 +04:00
|
|
|
if ((virDomainDiskGetType(disk) != VIR_STORAGE_TYPE_FILE) &&
|
|
|
|
(virDomainDiskGetType(disk) != VIR_STORAGE_TYPE_VOLUME)) {
|
2014-02-18 14:08:10 +04:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("unsupported disk type"));
|
2014-11-08 11:48:30 -05:00
|
|
|
return false;
|
2014-02-18 14:08:10 +04:00
|
|
|
}
|
|
|
|
|
2014-11-08 11:48:30 -05:00
|
|
|
return true;
|
|
|
|
}
|
2014-02-18 14:08:10 +04:00
|
|
|
|
2014-11-12 17:31:53 -05:00
|
|
|
static void
|
|
|
|
virBhyveFormatGrubDevice(virBufferPtr devicemap, virDomainDiskDefPtr def)
|
|
|
|
{
|
|
|
|
if (def->device == VIR_DOMAIN_DISK_DEVICE_CDROM)
|
|
|
|
virBufferAsprintf(devicemap, "(cd) %s\n",
|
|
|
|
virDomainDiskGetSource(def));
|
|
|
|
else
|
|
|
|
virBufferAsprintf(devicemap, "(hd0) %s\n",
|
|
|
|
virDomainDiskGetSource(def));
|
|
|
|
}
|
|
|
|
|
2014-11-08 11:48:30 -05:00
|
|
|
static virCommandPtr
|
|
|
|
virBhyveProcessBuildGrubbhyveCmd(virDomainDefPtr def,
|
2020-02-22 01:38:41 -05:00
|
|
|
bhyveConnPtr driver,
|
2014-11-08 11:48:30 -05:00
|
|
|
const char *devmap_file,
|
|
|
|
char **devicesmap_out)
|
|
|
|
{
|
2014-11-12 17:31:53 -05:00
|
|
|
virDomainDiskDefPtr hdd, cd, userdef, diskdef;
|
2014-11-08 11:48:30 -05:00
|
|
|
virCommandPtr cmd;
|
2016-03-29 14:31:37 +02:00
|
|
|
unsigned int best_idx = UINT_MAX;
|
2014-11-08 11:48:30 -05:00
|
|
|
size_t i;
|
|
|
|
|
|
|
|
if (def->os.bootloaderArgs != NULL)
|
|
|
|
return virBhyveProcessBuildCustomLoaderCmd(def);
|
|
|
|
|
2014-11-12 17:31:53 -05:00
|
|
|
/* Search disk list for CD or HDD device. We'll respect <boot order=''> if
|
|
|
|
* present and otherwise pick the first CD or failing that HDD we come
|
|
|
|
* across. */
|
|
|
|
cd = hdd = userdef = NULL;
|
2014-11-08 11:48:30 -05:00
|
|
|
for (i = 0; i < def->ndisks; i++) {
|
2018-02-19 14:19:41 +01:00
|
|
|
if (!virBhyveUsableDisk(def->disks[i]))
|
2014-11-08 11:48:30 -05:00
|
|
|
continue;
|
|
|
|
|
2014-11-12 17:31:53 -05:00
|
|
|
diskdef = def->disks[i];
|
|
|
|
|
|
|
|
if (diskdef->info.bootIndex && diskdef->info.bootIndex < best_idx) {
|
|
|
|
userdef = diskdef;
|
|
|
|
best_idx = userdef->info.bootIndex;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2014-11-08 11:48:30 -05:00
|
|
|
if (cd == NULL &&
|
|
|
|
def->disks[i]->device == VIR_DOMAIN_DISK_DEVICE_CDROM) {
|
2014-11-12 17:31:53 -05:00
|
|
|
cd = diskdef;
|
|
|
|
VIR_INFO("Picking %s as CD", virDomainDiskGetSource(cd));
|
2014-11-08 11:48:30 -05:00
|
|
|
}
|
|
|
|
|
2014-11-12 17:31:53 -05:00
|
|
|
if (hdd == NULL &&
|
2014-11-08 11:48:30 -05:00
|
|
|
def->disks[i]->device == VIR_DOMAIN_DISK_DEVICE_DISK) {
|
2014-11-12 17:31:53 -05:00
|
|
|
hdd = diskdef;
|
|
|
|
VIR_INFO("Picking %s as HDD", virDomainDiskGetSource(hdd));
|
2014-11-08 11:48:30 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cmd = virCommandNew(def->os.bootloader);
|
|
|
|
|
|
|
|
VIR_DEBUG("grub-bhyve with default arguments");
|
|
|
|
|
|
|
|
if (devicesmap_out != NULL) {
|
2020-07-02 18:03:45 -04:00
|
|
|
g_auto(virBuffer) devicemap = VIR_BUFFER_INITIALIZER;
|
|
|
|
|
2014-11-08 11:48:30 -05:00
|
|
|
/* Grub device.map (just for boot) */
|
2014-11-12 17:31:53 -05:00
|
|
|
if (userdef != NULL) {
|
|
|
|
virBhyveFormatGrubDevice(&devicemap, userdef);
|
|
|
|
} else {
|
|
|
|
if (hdd != NULL)
|
|
|
|
virBhyveFormatGrubDevice(&devicemap, hdd);
|
2014-11-08 11:48:30 -05:00
|
|
|
|
2014-11-12 17:31:53 -05:00
|
|
|
if (cd != NULL)
|
|
|
|
virBhyveFormatGrubDevice(&devicemap, cd);
|
|
|
|
}
|
2014-11-08 11:48:30 -05:00
|
|
|
|
|
|
|
*devicesmap_out = virBufferContentAndReset(&devicemap);
|
|
|
|
}
|
|
|
|
|
2014-11-12 17:31:53 -05:00
|
|
|
virCommandAddArg(cmd, "--root");
|
|
|
|
if (userdef != NULL) {
|
|
|
|
if (userdef->device == VIR_DOMAIN_DISK_DEVICE_CDROM)
|
|
|
|
virCommandAddArg(cmd, "cd");
|
|
|
|
else
|
|
|
|
virCommandAddArg(cmd, "hd0,msdos1");
|
|
|
|
} else if (cd != NULL) {
|
2014-11-08 11:48:30 -05:00
|
|
|
virCommandAddArg(cmd, "cd");
|
|
|
|
} else {
|
|
|
|
virCommandAddArg(cmd, "hd0,msdos1");
|
|
|
|
}
|
|
|
|
|
|
|
|
virCommandAddArg(cmd, "--device-map");
|
|
|
|
virCommandAddArg(cmd, devmap_file);
|
|
|
|
|
|
|
|
/* Memory in MB */
|
|
|
|
virCommandAddArg(cmd, "--memory");
|
2014-02-18 14:08:10 +04:00
|
|
|
virCommandAddArgFormat(cmd, "%llu",
|
2015-02-17 18:01:09 +01:00
|
|
|
VIR_DIV_UP(virDomainDefGetMemoryInitial(def), 1024));
|
2014-02-18 14:08:10 +04:00
|
|
|
|
2020-02-22 01:38:41 -05:00
|
|
|
if ((bhyveDriverGetGrubCaps(driver) & BHYVE_GRUB_CAP_CONSDEV) != 0 &&
|
2014-11-08 11:48:35 -05:00
|
|
|
def->nserials > 0) {
|
|
|
|
virDomainChrDefPtr chr;
|
|
|
|
|
|
|
|
chr = def->serials[0];
|
|
|
|
|
2016-10-21 07:45:54 -04:00
|
|
|
if (chr->source->type != VIR_DOMAIN_CHR_TYPE_NMDM) {
|
2014-11-08 11:48:35 -05:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("only nmdm console types are supported"));
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
virCommandAddArg(cmd, "--cons-dev");
|
2016-10-21 07:45:54 -04:00
|
|
|
virCommandAddArg(cmd, chr->source->data.file.path);
|
2014-11-08 11:48:35 -05:00
|
|
|
}
|
|
|
|
|
2014-02-18 14:08:10 +04:00
|
|
|
/* VM name */
|
2014-04-13 13:27:03 +04:00
|
|
|
virCommandAddArg(cmd, def->name);
|
2014-02-18 14:08:10 +04:00
|
|
|
|
|
|
|
return cmd;
|
|
|
|
}
|
2014-11-08 11:48:30 -05:00
|
|
|
|
2015-12-13 06:17:15 +03:00
|
|
|
static virDomainDiskDefPtr
|
2018-02-19 14:19:41 +01:00
|
|
|
virBhyveGetBootDisk(virDomainDefPtr def)
|
2014-11-08 11:48:30 -05:00
|
|
|
{
|
2015-12-13 06:17:15 +03:00
|
|
|
size_t i;
|
|
|
|
virDomainDiskDefPtr match = NULL;
|
|
|
|
int boot_dev = -1;
|
2014-11-08 11:48:30 -05:00
|
|
|
|
|
|
|
if (def->ndisks < 1) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
2015-12-13 06:17:15 +03:00
|
|
|
_("Domain should have at least one disk defined"));
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (def->os.nBootDevs > 1) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("Only one boot device is supported"));
|
2014-11-08 11:48:30 -05:00
|
|
|
return NULL;
|
2015-12-13 06:17:15 +03:00
|
|
|
} else if (def->os.nBootDevs == 1) {
|
|
|
|
switch (def->os.bootDevs[0]) {
|
|
|
|
case VIR_DOMAIN_BOOT_CDROM:
|
|
|
|
boot_dev = VIR_DOMAIN_DISK_DEVICE_CDROM;
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_BOOT_DISK:
|
|
|
|
boot_dev = VIR_DOMAIN_DISK_DEVICE_DISK;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("Cannot boot from device %s"),
|
|
|
|
virDomainBootTypeToString(def->os.bootDevs[0]));
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (boot_dev != -1) {
|
|
|
|
/* If boot_dev is set, we return the first device of
|
|
|
|
* the request type */
|
|
|
|
for (i = 0; i < def->ndisks; i++) {
|
2018-02-19 14:19:41 +01:00
|
|
|
if (!virBhyveUsableDisk(def->disks[i]))
|
2015-12-13 06:17:15 +03:00
|
|
|
continue;
|
|
|
|
|
|
|
|
if (def->disks[i]->device == boot_dev) {
|
|
|
|
match = def->disks[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (match == NULL) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("Cannot find boot device of requested type %s"),
|
|
|
|
virDomainBootTypeToString(def->os.bootDevs[0]));
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* Otherwise, if boot_dev is not set, we try to find if bootIndex
|
|
|
|
* is set for individual device. However, as bhyve does not support
|
|
|
|
* specifying real boot priority for devices, we allow only single
|
|
|
|
* device with boot priority set.
|
|
|
|
*/
|
|
|
|
int first_usable_disk_index = -1;
|
|
|
|
|
|
|
|
for (i = 0; i < def->ndisks; i++) {
|
2018-02-19 14:19:41 +01:00
|
|
|
if (!virBhyveUsableDisk(def->disks[i]))
|
2015-12-13 06:17:15 +03:00
|
|
|
continue;
|
|
|
|
else
|
|
|
|
first_usable_disk_index = i;
|
|
|
|
|
|
|
|
if (def->disks[i]->info.bootIndex > 0) {
|
|
|
|
if (match == NULL) {
|
|
|
|
match = def->disks[i];
|
|
|
|
} else {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("Only one boot device is supported"));
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-04 19:08:14 +02:00
|
|
|
/* If user didn't explicitly specify boot priority,
|
2015-12-13 06:17:15 +03:00
|
|
|
* just return the first usable disk */
|
|
|
|
if ((match == NULL) && (first_usable_disk_index >= 0))
|
|
|
|
return def->disks[first_usable_disk_index];
|
2014-11-08 11:48:30 -05:00
|
|
|
}
|
|
|
|
|
2015-12-13 06:17:15 +03:00
|
|
|
return match;
|
|
|
|
}
|
|
|
|
|
|
|
|
virCommandPtr
|
2020-02-22 01:38:41 -05:00
|
|
|
virBhyveProcessBuildLoadCmd(bhyveConnPtr driver, virDomainDefPtr def,
|
2015-12-13 06:17:15 +03:00
|
|
|
const char *devmap_file, char **devicesmap_out)
|
|
|
|
{
|
|
|
|
virDomainDiskDefPtr disk = NULL;
|
|
|
|
|
2014-11-08 11:48:30 -05:00
|
|
|
if (def->os.bootloader == NULL) {
|
2018-02-19 14:19:41 +01:00
|
|
|
disk = virBhyveGetBootDisk(def);
|
2014-11-08 11:48:30 -05:00
|
|
|
|
2015-12-13 06:17:15 +03:00
|
|
|
if (disk == NULL)
|
2014-11-08 11:48:30 -05:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
return virBhyveProcessBuildBhyveloadCmd(def, disk);
|
|
|
|
} else if (strstr(def->os.bootloader, "grub-bhyve") != NULL) {
|
2020-02-22 01:38:41 -05:00
|
|
|
return virBhyveProcessBuildGrubbhyveCmd(def, driver, devmap_file,
|
2014-11-08 11:48:30 -05:00
|
|
|
devicesmap_out);
|
|
|
|
} else {
|
|
|
|
return virBhyveProcessBuildCustomLoaderCmd(def);
|
|
|
|
}
|
|
|
|
}
|