mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-18 02:25:18 +00:00
915b637257
The current <audio> element only allows an "OSS" audio backend, as this is all that BHyve needed. This is now extended to cover most QEMU audio backends. These backends all have a variety of attributes they support, but this initial impl does the bare minimum, relying on built-in defaults for everything. The only QEMU backend omitted is "dsound" since the libvirt QEMU driver is not built on Windows platforms. The SDL audio driver names are based on the SDL 2.0 drivers. It is not intended to support SDL 1.2 drivers. Reviewed-by: Michal Privoznik <mprivozn@redhat.com> Signed-off-by: Daniel P. Berrangé <berrange@redhat.com>
1119 lines
36 KiB
C
1119 lines
36 KiB
C
/*
|
|
* bhyve_command.c: bhyve command generation
|
|
*
|
|
* 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>
|
|
|
|
#include "bhyve_capabilities.h"
|
|
#include "bhyve_command.h"
|
|
#include "bhyve_domain.h"
|
|
#include "bhyve_conf.h"
|
|
#include "bhyve_driver.h"
|
|
#include "datatypes.h"
|
|
#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
|
|
|
|
VIR_LOG_INIT("bhyve.bhyve_command");
|
|
|
|
static int
|
|
bhyveBuildNetArgStr(const virDomainDef *def,
|
|
virDomainNetDefPtr net,
|
|
bhyveConnPtr driver,
|
|
virCommandPtr cmd,
|
|
bool dryRun)
|
|
{
|
|
char macaddr[VIR_MAC_STRING_BUFLEN];
|
|
char *realifname = NULL;
|
|
char *brname = NULL;
|
|
char *nic_model = NULL;
|
|
int ret = -1;
|
|
virDomainNetType actualType = virDomainNetGetActualType(net);
|
|
|
|
if (net->model == VIR_DOMAIN_NET_MODEL_VIRTIO) {
|
|
nic_model = g_strdup("virtio-net");
|
|
} else if (net->model == VIR_DOMAIN_NET_MODEL_E1000) {
|
|
if ((bhyveDriverGetBhyveCaps(driver) & BHYVE_CAP_NET_E1000) != 0) {
|
|
nic_model = g_strdup("e1000");
|
|
} else {
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
_("NIC model 'e1000' is not supported "
|
|
"by given bhyve binary"));
|
|
return -1;
|
|
}
|
|
} else {
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
_("NIC model is not supported"));
|
|
return -1;
|
|
}
|
|
|
|
if (actualType == VIR_DOMAIN_NET_TYPE_BRIDGE) {
|
|
brname = g_strdup(virDomainNetGetActualBridgeName(net));
|
|
} else {
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
_("Network type %d is not supported"),
|
|
virDomainNetGetActualType(net));
|
|
goto cleanup;
|
|
}
|
|
|
|
if (!dryRun) {
|
|
if (virNetDevTapCreateInBridgePort(brname, &net->ifname, &net->mac,
|
|
def->uuid, NULL, NULL, 0,
|
|
virDomainNetGetActualVirtPortProfile(net),
|
|
virDomainNetGetActualVlan(net),
|
|
virDomainNetGetActualPortOptionsIsolated(net),
|
|
NULL, 0, NULL,
|
|
VIR_NETDEV_TAP_CREATE_IFUP | VIR_NETDEV_TAP_CREATE_PERSIST) < 0) {
|
|
goto cleanup;
|
|
}
|
|
|
|
realifname = virNetDevTapGetRealDeviceName(net->ifname);
|
|
|
|
if (realifname == NULL)
|
|
goto cleanup;
|
|
|
|
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
|
|
*/
|
|
if (virNetDevSetOnline(net->ifname, true) != 0)
|
|
goto cleanup;
|
|
} else {
|
|
realifname = g_strdup("tap0");
|
|
}
|
|
|
|
|
|
virCommandAddArg(cmd, "-s");
|
|
virCommandAddArgFormat(cmd, "%d:0,%s,%s,mac=%s",
|
|
net->info.addr.pci.slot, nic_model,
|
|
realifname, virMacAddrFormat(&net->mac, macaddr));
|
|
|
|
ret = 0;
|
|
cleanup:
|
|
if (ret < 0)
|
|
VIR_FREE(net->ifname);
|
|
VIR_FREE(brname);
|
|
VIR_FREE(realifname);
|
|
VIR_FREE(nic_model);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int
|
|
bhyveBuildConsoleArgStr(const virDomainDef *def, virCommandPtr cmd)
|
|
{
|
|
virDomainChrDefPtr chr = NULL;
|
|
|
|
if (!def->nserials)
|
|
return 0;
|
|
|
|
chr = def->serials[0];
|
|
|
|
if (chr->source->type != VIR_DOMAIN_CHR_TYPE_NMDM) {
|
|
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",
|
|
chr->target.port + 1, chr->source->data.file.path);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
bhyveBuildAHCIControllerArgStr(const virDomainDef *def,
|
|
virDomainControllerDefPtr controller,
|
|
bhyveConnPtr driver,
|
|
virCommandPtr cmd)
|
|
{
|
|
g_auto(virBuffer) buf = VIR_BUFFER_INITIALIZER;
|
|
const char *disk_source;
|
|
size_t i;
|
|
|
|
for (i = 0; i < def->ndisks; i++) {
|
|
g_auto(virBuffer) device = VIR_BUFFER_INITIALIZER;
|
|
virDomainDiskDefPtr disk = def->disks[i];
|
|
|
|
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"));
|
|
return -1;
|
|
}
|
|
|
|
if (virDomainDiskTranslateSourcePool(disk) < 0)
|
|
return -1;
|
|
|
|
disk_source = virDomainDiskGetSource(disk);
|
|
|
|
if ((disk->device == VIR_DOMAIN_DISK_DEVICE_CDROM) &&
|
|
(disk_source == NULL)) {
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
_("cdrom device without source path "
|
|
"not supported"));
|
|
return -1;
|
|
}
|
|
|
|
switch (disk->device) {
|
|
case VIR_DOMAIN_DISK_DEVICE_DISK:
|
|
if ((bhyveDriverGetBhyveCaps(driver) & BHYVE_CAP_AHCI32SLOT))
|
|
virBufferAsprintf(&device, ",hd:%s", disk_source);
|
|
else
|
|
virBufferAsprintf(&device, "-hd,%s", disk_source);
|
|
break;
|
|
case VIR_DOMAIN_DISK_DEVICE_CDROM:
|
|
if ((bhyveDriverGetBhyveCaps(driver) & BHYVE_CAP_AHCI32SLOT))
|
|
virBufferAsprintf(&device, ",cd:%s", disk_source);
|
|
else
|
|
virBufferAsprintf(&device, "-cd,%s", disk_source);
|
|
break;
|
|
default:
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
_("unsupported disk device"));
|
|
return -1;
|
|
}
|
|
virBufferAddBuffer(&buf, &device);
|
|
}
|
|
|
|
virCommandAddArg(cmd, "-s");
|
|
virCommandAddArgFormat(cmd, "%d:0,ahci%s",
|
|
controller->info.addr.pci.slot,
|
|
virBufferCurrentContent(&buf));
|
|
|
|
return 0;
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
static int
|
|
bhyveBuildVirtIODiskArgStr(const virDomainDef *def G_GNUC_UNUSED,
|
|
virDomainDiskDefPtr disk,
|
|
virCommandPtr cmd)
|
|
{
|
|
const char *disk_source;
|
|
|
|
if (virDomainDiskTranslateSourcePool(disk) < 0)
|
|
return -1;
|
|
|
|
if (disk->device != VIR_DOMAIN_DISK_DEVICE_DISK) {
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
_("unsupported disk device"));
|
|
return -1;
|
|
}
|
|
|
|
if ((virDomainDiskGetType(disk) != VIR_STORAGE_TYPE_FILE) &&
|
|
(virDomainDiskGetType(disk) != VIR_STORAGE_TYPE_VOLUME)) {
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
_("unsupported disk type"));
|
|
return -1;
|
|
}
|
|
|
|
disk_source = virDomainDiskGetSource(disk);
|
|
|
|
virCommandAddArg(cmd, "-s");
|
|
virCommandAddArgFormat(cmd, "%d:0,virtio-blk,%s",
|
|
disk->info.addr.pci.slot,
|
|
disk_source);
|
|
|
|
return 0;
|
|
}
|
|
|
|
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,
|
|
unsigned *nusbcontrollers,
|
|
unsigned *nisacontrollers)
|
|
{
|
|
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;
|
|
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;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
bhyveBuildGraphicsArgStr(const virDomainDef *def,
|
|
virDomainGraphicsDefPtr graphics,
|
|
virDomainVideoDefPtr video,
|
|
bhyveConnPtr driver,
|
|
virCommandPtr cmd,
|
|
bool dryRun)
|
|
{
|
|
g_auto(virBuffer) opt = VIR_BUFFER_INITIALIZER;
|
|
virDomainGraphicsListenDefPtr glisten = NULL;
|
|
bool escapeAddr;
|
|
unsigned short port;
|
|
|
|
if (!(bhyveDriverGetBhyveCaps(driver) & BHYVE_CAP_LPC_BOOTROM) ||
|
|
def->os.bootloader ||
|
|
!def->os.loader) {
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
_("Graphics are only supported"
|
|
" when booting using UEFI"));
|
|
return -1;
|
|
}
|
|
|
|
if (!(bhyveDriverGetBhyveCaps(driver) & BHYVE_CAP_FBUF)) {
|
|
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"));
|
|
return -1;
|
|
}
|
|
|
|
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]"));
|
|
return -1;
|
|
}
|
|
|
|
if (glisten->address) {
|
|
escapeAddr = strchr(glisten->address, ':') != NULL;
|
|
if (escapeAddr)
|
|
virBufferAsprintf(&opt, "[%s]", glisten->address);
|
|
else
|
|
virBufferAdd(&opt, glisten->address, -1);
|
|
}
|
|
|
|
if (!dryRun) {
|
|
if (graphics->data.vnc.autoport) {
|
|
if (virPortAllocatorAcquire(driver->remotePorts, &port) < 0)
|
|
return -1;
|
|
graphics->data.vnc.port = port;
|
|
} else {
|
|
if (virPortAllocatorSetUsed(graphics->data.vnc.port) < 0)
|
|
VIR_WARN("Failed to mark VNC port '%d' as used by '%s'",
|
|
graphics->data.vnc.port, def->name);
|
|
}
|
|
}
|
|
|
|
virBufferAsprintf(&opt, ":%d", graphics->data.vnc.port);
|
|
break;
|
|
case VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_SOCKET:
|
|
case VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_NONE:
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
_("Unsupported listen type"));
|
|
return -1;
|
|
case VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_LAST:
|
|
default:
|
|
virReportEnumRangeError(virDomainGraphicsListenType, glisten->type);
|
|
return -1;
|
|
}
|
|
|
|
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."));
|
|
}
|
|
|
|
if (video->res)
|
|
virBufferAsprintf(&opt, ",w=%d,h=%d", video->res->x, video->res->y);
|
|
|
|
if (video->driver)
|
|
virBufferAsprintf(&opt, ",vga=%s",
|
|
virDomainVideoVGAConfTypeToString(video->driver->vgaconf));
|
|
|
|
virCommandAddArg(cmd, "-s");
|
|
virCommandAddArgBuffer(cmd, &opt);
|
|
return 0;
|
|
|
|
}
|
|
|
|
static int
|
|
bhyveBuildSoundArgStr(const virDomainDef *def G_GNUC_UNUSED,
|
|
virDomainSoundDefPtr sound,
|
|
virDomainAudioDefPtr audio,
|
|
bhyveConnPtr driver,
|
|
virCommandPtr cmd)
|
|
{
|
|
g_auto(virBuffer) params = VIR_BUFFER_INITIALIZER;
|
|
|
|
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");
|
|
|
|
if (audio) {
|
|
switch ((virDomainAudioType) audio->type) {
|
|
case VIR_DOMAIN_AUDIO_TYPE_OSS:
|
|
if (audio->backend.oss.input.dev)
|
|
virBufferAsprintf(¶ms, ",play=%s",
|
|
audio->backend.oss.input.dev);
|
|
|
|
if (audio->backend.oss.output.dev)
|
|
virBufferAsprintf(¶ms, ",rec=%s",
|
|
audio->backend.oss.output.dev);
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_AUDIO_TYPE_NONE:
|
|
case VIR_DOMAIN_AUDIO_TYPE_ALSA:
|
|
case VIR_DOMAIN_AUDIO_TYPE_COREAUDIO:
|
|
case VIR_DOMAIN_AUDIO_TYPE_JACK:
|
|
case VIR_DOMAIN_AUDIO_TYPE_PULSEAUDIO:
|
|
case VIR_DOMAIN_AUDIO_TYPE_SDL:
|
|
case VIR_DOMAIN_AUDIO_TYPE_SPICE:
|
|
case VIR_DOMAIN_AUDIO_TYPE_FILE:
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
_("unsupported audio backend '%s'"),
|
|
virDomainAudioTypeTypeToString(audio->type));
|
|
return -1;
|
|
case VIR_DOMAIN_AUDIO_TYPE_LAST:
|
|
default:
|
|
virReportEnumRangeError(virDomainAudioType, audio->type);
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
virCommandAddArgFormat(cmd, "%d:%d,hda%s",
|
|
sound->info.addr.pci.slot,
|
|
sound->info.addr.pci.function,
|
|
virBufferCurrentContent(¶ms));
|
|
|
|
return 0;
|
|
}
|
|
|
|
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,
|
|
fs->src->path,
|
|
virBufferCurrentContent(¶ms));
|
|
|
|
return 0;
|
|
}
|
|
|
|
virCommandPtr
|
|
virBhyveProcessBuildBhyveCmd(bhyveConnPtr driver, virDomainDefPtr def,
|
|
bool dryRun)
|
|
{
|
|
/*
|
|
* /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);
|
|
size_t i;
|
|
unsigned nusbcontrollers = 0;
|
|
unsigned nisacontrollers = 0;
|
|
unsigned nvcpus = virDomainDefGetVcpus(def);
|
|
|
|
/* CPUs */
|
|
virCommandAddArg(cmd, "-c");
|
|
if (def->cpu && def->cpu->sockets) {
|
|
if (def->cpu->dies != 1) {
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
_("Only 1 die per socket is supported"));
|
|
goto error;
|
|
}
|
|
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;
|
|
}
|
|
|
|
if ((bhyveDriverGetBhyveCaps(driver) & BHYVE_CAP_CPUTOPOLOGY) != 0) {
|
|
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);
|
|
}
|
|
|
|
/* Memory */
|
|
virCommandAddArg(cmd, "-m");
|
|
virCommandAddArgFormat(cmd, "%llu",
|
|
VIR_DIV_UP(virDomainDefGetMemoryInitial(def), 1024));
|
|
|
|
if (def->mem.locked)
|
|
virCommandAddArg(cmd, "-S"); /* Wire guest memory */
|
|
|
|
/* Options */
|
|
if (def->features[VIR_DOMAIN_FEATURE_ACPI] == VIR_TRISTATE_SWITCH_ON)
|
|
virCommandAddArg(cmd, "-A"); /* Create an ACPI table */
|
|
if (def->features[VIR_DOMAIN_FEATURE_APIC] == VIR_TRISTATE_SWITCH_ON)
|
|
virCommandAddArg(cmd, "-I"); /* Present ioapic to the guest */
|
|
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");
|
|
}
|
|
|
|
switch (def->clock.offset) {
|
|
case VIR_DOMAIN_CLOCK_OFFSET_LOCALTIME:
|
|
/* used by default in bhyve */
|
|
break;
|
|
case VIR_DOMAIN_CLOCK_OFFSET_UTC:
|
|
if ((bhyveDriverGetBhyveCaps(driver) & BHYVE_CAP_RTC_UTC) != 0) {
|
|
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;
|
|
}
|
|
|
|
/* 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 */
|
|
|
|
virCommandAddArgList(cmd, "-s", "0:0,hostbridge", NULL);
|
|
|
|
if (def->os.bootloader == NULL &&
|
|
def->os.loader) {
|
|
if ((bhyveDriverGetBhyveCaps(driver) & BHYVE_CAP_LPC_BOOTROM)) {
|
|
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;
|
|
}
|
|
}
|
|
|
|
/* Devices */
|
|
for (i = 0; i < def->ncontrollers; i++) {
|
|
if (bhyveBuildControllerArgStr(def, def->controllers[i], driver, cmd,
|
|
&nusbcontrollers, &nisacontrollers) < 0)
|
|
goto error;
|
|
}
|
|
for (i = 0; i < def->nnets; i++) {
|
|
if (bhyveBuildNetArgStr(def, def->nets[i], driver, cmd, dryRun) < 0)
|
|
goto error;
|
|
}
|
|
for (i = 0; i < def->ndisks; i++) {
|
|
if (bhyveBuildDiskArgStr(def, def->disks[i], cmd) < 0)
|
|
goto error;
|
|
}
|
|
|
|
if (def->ngraphics && def->nvideos) {
|
|
if (def->ngraphics == 1 && def->nvideos == 1) {
|
|
if (bhyveBuildGraphicsArgStr(def, def->graphics[0], def->videos[0],
|
|
driver, cmd, dryRun) < 0)
|
|
goto error;
|
|
} else {
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
_("Multiple graphics devices are not supported"));
|
|
goto error;
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < def->nsounds; i++) {
|
|
if (bhyveBuildSoundArgStr(def, def->sounds[i],
|
|
virDomainDefFindAudioForSound(def, def->sounds[i]),
|
|
driver, cmd) < 0)
|
|
goto error;
|
|
}
|
|
|
|
for (i = 0; i < def->nfss; i++) {
|
|
if (bhyveBuildFSArgStr(def, def->fss[i], cmd) < 0)
|
|
goto error;
|
|
}
|
|
|
|
if (bhyveBuildConsoleArgStr(def, cmd) < 0)
|
|
goto error;
|
|
|
|
if (def->namespaceData) {
|
|
bhyveDomainCmdlineDefPtr bhyvecmd;
|
|
|
|
VIR_WARN("Booting the guest using command line pass-through feature, "
|
|
"which could potentially cause inconsistent state and "
|
|
"upgrade issues");
|
|
|
|
bhyvecmd = def->namespaceData;
|
|
for (i = 0; i < bhyvecmd->num_args; i++)
|
|
virCommandAddArg(cmd, bhyvecmd->args[i]);
|
|
}
|
|
|
|
virCommandAddArg(cmd, def->name);
|
|
|
|
return cmd;
|
|
|
|
error:
|
|
virCommandFree(cmd);
|
|
return NULL;
|
|
}
|
|
|
|
virCommandPtr
|
|
virBhyveProcessBuildDestroyCmd(bhyveConnPtr driver G_GNUC_UNUSED,
|
|
virDomainDefPtr def)
|
|
{
|
|
virCommandPtr cmd = virCommandNew(BHYVECTL);
|
|
|
|
virCommandAddArg(cmd, "--destroy");
|
|
virCommandAddArgPair(cmd, "--vm", def->name);
|
|
|
|
return cmd;
|
|
}
|
|
|
|
static void
|
|
virAppendBootloaderArgs(virCommandPtr cmd, virDomainDefPtr def)
|
|
{
|
|
char **blargs;
|
|
|
|
/* XXX: Handle quoted? */
|
|
blargs = g_strsplit(def->os.bootloaderArgs, " ", 0);
|
|
virCommandAddArgSet(cmd, (const char * const *)blargs);
|
|
g_strfreev(blargs);
|
|
}
|
|
|
|
static virCommandPtr
|
|
virBhyveProcessBuildBhyveloadCmd(virDomainDefPtr def, virDomainDiskDefPtr disk)
|
|
{
|
|
virCommandPtr cmd;
|
|
|
|
cmd = virCommandNew(BHYVELOAD);
|
|
|
|
if (def->os.bootloaderArgs == NULL) {
|
|
VIR_DEBUG("bhyveload with default arguments");
|
|
|
|
/* Memory (MB) */
|
|
virCommandAddArg(cmd, "-m");
|
|
virCommandAddArgFormat(cmd, "%llu",
|
|
VIR_DIV_UP(virDomainDefGetMemoryInitial(def), 1024));
|
|
|
|
/* 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");
|
|
return NULL;
|
|
}
|
|
|
|
VIR_DEBUG("custom loader '%s' with arguments", def->os.bootloader);
|
|
|
|
cmd = virCommandNew(def->os.bootloader);
|
|
virAppendBootloaderArgs(cmd, def);
|
|
return cmd;
|
|
}
|
|
|
|
static bool
|
|
virBhyveUsableDisk(virDomainDiskDefPtr disk)
|
|
{
|
|
if (virDomainDiskTranslateSourcePool(disk) < 0)
|
|
return false;
|
|
|
|
if ((disk->device != VIR_DOMAIN_DISK_DEVICE_DISK) &&
|
|
(disk->device != VIR_DOMAIN_DISK_DEVICE_CDROM)) {
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
_("unsupported disk device"));
|
|
return false;
|
|
}
|
|
|
|
if ((virDomainDiskGetType(disk) != VIR_STORAGE_TYPE_FILE) &&
|
|
(virDomainDiskGetType(disk) != VIR_STORAGE_TYPE_VOLUME)) {
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
_("unsupported disk type"));
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
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));
|
|
}
|
|
|
|
static virCommandPtr
|
|
virBhyveProcessBuildGrubbhyveCmd(virDomainDefPtr def,
|
|
bhyveConnPtr driver,
|
|
const char *devmap_file,
|
|
char **devicesmap_out)
|
|
{
|
|
virDomainDiskDefPtr hdd, cd, userdef, diskdef;
|
|
virCommandPtr cmd;
|
|
unsigned int best_idx = UINT_MAX;
|
|
size_t i;
|
|
|
|
if (def->os.bootloaderArgs != NULL)
|
|
return virBhyveProcessBuildCustomLoaderCmd(def);
|
|
|
|
/* 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;
|
|
for (i = 0; i < def->ndisks; i++) {
|
|
if (!virBhyveUsableDisk(def->disks[i]))
|
|
continue;
|
|
|
|
diskdef = def->disks[i];
|
|
|
|
if (diskdef->info.bootIndex && diskdef->info.bootIndex < best_idx) {
|
|
userdef = diskdef;
|
|
best_idx = userdef->info.bootIndex;
|
|
continue;
|
|
}
|
|
|
|
if (cd == NULL &&
|
|
def->disks[i]->device == VIR_DOMAIN_DISK_DEVICE_CDROM) {
|
|
cd = diskdef;
|
|
VIR_INFO("Picking %s as CD", virDomainDiskGetSource(cd));
|
|
}
|
|
|
|
if (hdd == NULL &&
|
|
def->disks[i]->device == VIR_DOMAIN_DISK_DEVICE_DISK) {
|
|
hdd = diskdef;
|
|
VIR_INFO("Picking %s as HDD", virDomainDiskGetSource(hdd));
|
|
}
|
|
}
|
|
|
|
cmd = virCommandNew(def->os.bootloader);
|
|
|
|
VIR_DEBUG("grub-bhyve with default arguments");
|
|
|
|
if (devicesmap_out != NULL) {
|
|
g_auto(virBuffer) devicemap = VIR_BUFFER_INITIALIZER;
|
|
|
|
/* Grub device.map (just for boot) */
|
|
if (userdef != NULL) {
|
|
virBhyveFormatGrubDevice(&devicemap, userdef);
|
|
} else {
|
|
if (hdd != NULL)
|
|
virBhyveFormatGrubDevice(&devicemap, hdd);
|
|
|
|
if (cd != NULL)
|
|
virBhyveFormatGrubDevice(&devicemap, cd);
|
|
}
|
|
|
|
*devicesmap_out = virBufferContentAndReset(&devicemap);
|
|
}
|
|
|
|
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) {
|
|
virCommandAddArg(cmd, "cd");
|
|
} else {
|
|
virCommandAddArg(cmd, "hd0,msdos1");
|
|
}
|
|
|
|
virCommandAddArg(cmd, "--device-map");
|
|
virCommandAddArg(cmd, devmap_file);
|
|
|
|
/* Memory in MB */
|
|
virCommandAddArg(cmd, "--memory");
|
|
virCommandAddArgFormat(cmd, "%llu",
|
|
VIR_DIV_UP(virDomainDefGetMemoryInitial(def), 1024));
|
|
|
|
if ((bhyveDriverGetGrubCaps(driver) & BHYVE_GRUB_CAP_CONSDEV) != 0 &&
|
|
def->nserials > 0) {
|
|
virDomainChrDefPtr chr;
|
|
|
|
chr = def->serials[0];
|
|
|
|
if (chr->source->type != VIR_DOMAIN_CHR_TYPE_NMDM) {
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
_("only nmdm console types are supported"));
|
|
return NULL;
|
|
}
|
|
|
|
virCommandAddArg(cmd, "--cons-dev");
|
|
virCommandAddArg(cmd, chr->source->data.file.path);
|
|
}
|
|
|
|
/* VM name */
|
|
virCommandAddArg(cmd, def->name);
|
|
|
|
return cmd;
|
|
}
|
|
|
|
static virDomainDiskDefPtr
|
|
virBhyveGetBootDisk(virDomainDefPtr def)
|
|
{
|
|
size_t i;
|
|
virDomainDiskDefPtr match = NULL;
|
|
int boot_dev = -1;
|
|
|
|
if (def->ndisks < 1) {
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
_("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"));
|
|
return NULL;
|
|
} 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++) {
|
|
if (!virBhyveUsableDisk(def->disks[i]))
|
|
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++) {
|
|
if (!virBhyveUsableDisk(def->disks[i]))
|
|
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;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* If user didn't explicitly specify boot priority,
|
|
* just return the first usable disk */
|
|
if ((match == NULL) && (first_usable_disk_index >= 0))
|
|
return def->disks[first_usable_disk_index];
|
|
}
|
|
|
|
return match;
|
|
}
|
|
|
|
virCommandPtr
|
|
virBhyveProcessBuildLoadCmd(bhyveConnPtr driver, virDomainDefPtr def,
|
|
const char *devmap_file, char **devicesmap_out)
|
|
{
|
|
virDomainDiskDefPtr disk = NULL;
|
|
|
|
if (def->os.bootloader == NULL) {
|
|
disk = virBhyveGetBootDisk(def);
|
|
|
|
if (disk == NULL)
|
|
return NULL;
|
|
|
|
return virBhyveProcessBuildBhyveloadCmd(def, disk);
|
|
} else if (strstr(def->os.bootloader, "grub-bhyve") != NULL) {
|
|
return virBhyveProcessBuildGrubbhyveCmd(def, driver, devmap_file,
|
|
devicesmap_out);
|
|
} else {
|
|
return virBhyveProcessBuildCustomLoaderCmd(def);
|
|
}
|
|
}
|