bhyve: add <os firmware='efi'> support

Implement "<os firmware='efi'>" support for bhyve driver.
As there are not really lot of options, try to find
"BHYVE_UEFI.fd" firmware which is installed by the
sysutils/uefi-edk2-bhyve FreeBSD port.

If not found, just use the first found firmware
in the firmwares directory (which is configurable via
config file).

Signed-off-by: Roman Bogorodskiy <bogorodskiy@gmail.com>
Reviewed-by: Michal Privoznik <mprivozn@redhat.com>
This commit is contained in:
Roman Bogorodskiy 2021-02-22 13:46:13 +04:00
parent dd0529b319
commit 63bed955df
17 changed files with 257 additions and 12 deletions

View File

@ -14,6 +14,7 @@
@SRCDIR@src/bhyve/bhyve_command.c @SRCDIR@src/bhyve/bhyve_command.c
@SRCDIR@src/bhyve/bhyve_domain.c @SRCDIR@src/bhyve/bhyve_domain.c
@SRCDIR@src/bhyve/bhyve_driver.c @SRCDIR@src/bhyve/bhyve_driver.c
@SRCDIR@src/bhyve/bhyve_firmware.c
@SRCDIR@src/bhyve/bhyve_monitor.c @SRCDIR@src/bhyve/bhyve_monitor.c
@SRCDIR@src/bhyve/bhyve_parse_command.c @SRCDIR@src/bhyve/bhyve_parse_command.c
@SRCDIR@src/bhyve/bhyve_process.c @SRCDIR@src/bhyve/bhyve_process.c

View File

@ -64,6 +64,9 @@ bhyveDomainDefNeedsISAController(virDomainDefPtr def)
if (def->os.bootloader == NULL && def->os.loader) if (def->os.bootloader == NULL && def->os.loader)
return true; return true;
if (def->os.firmware == VIR_DOMAIN_OS_DEF_FIRMWARE_EFI)
return true;
if (def->nserials || def->nconsoles) if (def->nserials || def->nconsoles)
return true; return true;
@ -230,6 +233,8 @@ virDomainDefParserConfig virBhyveDriverDomainDefParserConfig = {
.domainPostParseCallback = bhyveDomainDefPostParse, .domainPostParseCallback = bhyveDomainDefPostParse,
.assignAddressesCallback = bhyveDomainDefAssignAddresses, .assignAddressesCallback = bhyveDomainDefAssignAddresses,
.deviceValidateCallback = bhyveDomainDeviceDefValidate, .deviceValidateCallback = bhyveDomainDeviceDefValidate,
.features = VIR_DOMAIN_DEF_FEATURE_FW_AUTOSELECT,
}; };
static void static void

View File

@ -0,0 +1,91 @@
/*
* bhyve_firmware.c: bhyve firmware management
*
* Copyright (C) 2021 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 <dirent.h>
#include "viralloc.h"
#include "virlog.h"
#include "virfile.h"
#include "bhyve_conf.h"
#include "bhyve_firmware.h"
#define VIR_FROM_THIS VIR_FROM_BHYVE
VIR_LOG_INIT("bhyve.bhyve_firmware");
#define BHYVE_DEFAULT_FIRMWARE "BHYVE_UEFI.fd"
int
bhyveFirmwareFillDomain(bhyveConnPtr driver,
virDomainDefPtr def,
unsigned int flags)
{
g_autoptr(DIR) dir = NULL;
g_autoptr(virBhyveDriverConfig) cfg = virBhyveDriverGetConfig(driver);
const char *firmware_dir = cfg->firmwareDir;
struct dirent *entry;
g_autofree char *matching_firmware = NULL;
g_autofree char *first_found = NULL;
virCheckFlags(0, -1);
if (def->os.firmware == VIR_DOMAIN_OS_DEF_FIRMWARE_NONE)
return 0;
if (virDirOpenIfExists(&dir, firmware_dir) > 0) {
while ((virDirRead(dir, &entry, firmware_dir)) > 0) {
if (g_str_has_prefix(entry->d_name, "."))
continue;
if (STREQ(entry->d_name, BHYVE_DEFAULT_FIRMWARE)) {
matching_firmware = g_strdup(entry->d_name);
break;
}
if (!first_found)
first_found = g_strdup(entry->d_name);
}
}
if (!matching_firmware) {
if (!first_found) {
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
_("no firmwares found in %s"),
firmware_dir);
return -1;
} else {
matching_firmware = g_steal_pointer(&first_found);
}
}
if (!def->os.loader)
def->os.loader = g_new0(virDomainLoaderDef, 1);
def->os.loader->type = VIR_DOMAIN_LOADER_TYPE_PFLASH;
def->os.loader->readonly = VIR_TRISTATE_BOOL_YES;
VIR_FREE(def->os.loader->path);
def->os.loader->path = g_build_filename(firmware_dir, matching_firmware, NULL);
return 0;
}

View File

@ -0,0 +1,30 @@
/*
* bhyve_firmware.h: bhyve firmware management
*
* Copyright (C) 2021 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/>.
*
*/
#pragma once
#include "domain_conf.h"
#include "bhyve_utils.h"
int
bhyveFirmwareFillDomain(bhyveConnPtr driver,
virDomainDefPtr def,
unsigned int flags);

View File

@ -33,6 +33,7 @@
#include "bhyve_device.h" #include "bhyve_device.h"
#include "bhyve_driver.h" #include "bhyve_driver.h"
#include "bhyve_command.h" #include "bhyve_command.h"
#include "bhyve_firmware.h"
#include "bhyve_monitor.h" #include "bhyve_monitor.h"
#include "bhyve_process.h" #include "bhyve_process.h"
#include "datatypes.h" #include "datatypes.h"
@ -251,6 +252,17 @@ virBhyveProcessStartImpl(bhyveConnPtr driver,
return ret; return ret;
} }
int
bhyveProcessPrepareDomain(bhyveConnPtr driver,
virDomainObjPtr vm,
unsigned int flags)
{
if (bhyveFirmwareFillDomain(driver, vm->def, flags) < 0)
return -1;
return 0;
}
int int
virBhyveProcessStart(virConnectPtr conn, virBhyveProcessStart(virConnectPtr conn,
virDomainObjPtr vm, virDomainObjPtr vm,
@ -268,6 +280,9 @@ virBhyveProcessStart(virConnectPtr conn,
conn, bhyveProcessAutoDestroy) < 0) conn, bhyveProcessAutoDestroy) < 0)
return -1; return -1;
if (bhyveProcessPrepareDomain(driver, vm, flags) < 0)
return -1;
return virBhyveProcessStartImpl(driver, vm, reason); return virBhyveProcessStartImpl(driver, vm, reason);
} }

View File

@ -23,6 +23,11 @@
#include "bhyve_utils.h" #include "bhyve_utils.h"
int
bhyveProcessPrepareDomain(bhyveConnPtr driver,
virDomainObjPtr vm,
unsigned int flags);
int virBhyveProcessStart(virConnectPtr conn, int virBhyveProcessStart(virConnectPtr conn,
virDomainObjPtr vm, virDomainObjPtr vm,
virDomainRunningReason reason, virDomainRunningReason reason,

View File

@ -43,6 +43,8 @@ struct _virBhyveDriverConfig {
char *firmwareDir; char *firmwareDir;
}; };
G_DEFINE_AUTOPTR_CLEANUP_FUNC(virBhyveDriverConfig, virObjectUnref);
struct _bhyveConn { struct _bhyveConn {
virMutex lock; virMutex lock;

View File

@ -2,6 +2,7 @@ bhyve_sources = files(
'bhyve_capabilities.c', 'bhyve_capabilities.c',
'bhyve_command.c', 'bhyve_command.c',
'bhyve_conf.c', 'bhyve_conf.c',
'bhyve_firmware.c',
'bhyve_parse_command.c', 'bhyve_parse_command.c',
'bhyve_device.c', 'bhyve_device.c',
'bhyve_domain.c', 'bhyve_domain.c',

View File

View File

@ -0,0 +1,11 @@
/usr/sbin/bhyve \
-c 1 \
-m 214 \
-u \
-H \
-P \
-s 0:0,hostbridge \
-l bootrom,fakefirmwaredir/BHYVE_UEFI.fd \
-s 1:0,lpc \
-s 2:0,ahci,hd:/tmp/freebsd.img \
-s 3:0,virtio-net,faketapdev,mac=52:54:00:00:00:00 bhyve

View File

@ -0,0 +1 @@
dummy

View File

@ -0,0 +1,22 @@
<domain type='bhyve'>
<name>bhyve</name>
<uuid>df3be7e7-a104-11e3-aeb0-50e5492bd3dc</uuid>
<memory>219136</memory>
<vcpu>1</vcpu>
<os firmware='efi'>
<type>hvm</type>
</os>
<devices>
<disk type='file'>
<driver name='file' type='raw'/>
<source file='/tmp/freebsd.img'/>
<target dev='hda' bus='sata'/>
<address type='drive' controller='0' bus='0' target='2' unit='0'/>
</disk>
<interface type='bridge'>
<model type='virtio'/>
<source bridge="virbr0"/>
<address type='pci' domain='0x0000' bus='0x00' slot='0x03' function='0x0'/>
</interface>
</devices>
</domain>

View File

@ -1,13 +1,46 @@
#include <config.h> #include <config.h>
#include <dirent.h>
#include "viralloc.h" #include "viralloc.h"
#include "virstring.h" #include "virstring.h"
#include "virnetdev.h" #include "virnetdev.h"
#include "virnetdevtap.h" #include "virnetdevtap.h"
#include "virmock.h"
#include "internal.h" #include "internal.h"
#define VIR_FROM_THIS VIR_FROM_BHYVE #define VIR_FROM_THIS VIR_FROM_BHYVE
static DIR * (*real_opendir)(const char *name);
static void
init_syms(void)
{
VIR_MOCK_REAL_INIT(opendir);
}
#define FAKEFIRMWAREDIR abs_srcdir "/bhyvefirmwaredata/three_firmwares"
#define FAKEFIRMWAREEMPTYDIR abs_srcdir "/bhyvefirmwaredata/empty"
DIR *
opendir(const char *path)
{
init_syms();
g_autofree char *path_override = NULL;
if (STREQ(path, "fakefirmwaredir")) {
path_override = g_strdup(FAKEFIRMWAREDIR);
} else if (STREQ(path, "fakefirmwareemptydir")) {
path_override = g_strdup(FAKEFIRMWAREEMPTYDIR);
}
if (!path_override)
path_override = g_strdup(path);
return real_opendir(path_override);
}
void virMacAddrGenerate(const unsigned char prefix[VIR_MAC_PREFIX_BUFLEN], void virMacAddrGenerate(const unsigned char prefix[VIR_MAC_PREFIX_BUFLEN],
virMacAddrPtr addr) virMacAddrPtr addr)
{ {

View File

@ -7,17 +7,20 @@
# include "datatypes.h" # include "datatypes.h"
# include "bhyve/bhyve_capabilities.h" # include "bhyve/bhyve_capabilities.h"
# include "bhyve/bhyve_conf.h"
# include "bhyve/bhyve_domain.h" # include "bhyve/bhyve_domain.h"
# include "bhyve/bhyve_utils.h" # include "bhyve/bhyve_utils.h"
# include "bhyve/bhyve_command.h" # include "bhyve/bhyve_command.h"
# include "bhyve/bhyve_process.h"
# define VIR_FROM_THIS VIR_FROM_BHYVE # define VIR_FROM_THIS VIR_FROM_BHYVE
static bhyveConn driver; static bhyveConn driver;
typedef enum { typedef enum {
FLAG_EXPECT_FAILURE = 1 << 0, FLAG_EXPECT_FAILURE = 1 << 0,
FLAG_EXPECT_PARSE_ERROR = 1 << 1, FLAG_EXPECT_PARSE_ERROR = 1 << 1,
FLAG_EXPECT_PREPARE_ERROR = 1 << 2,
} virBhyveXMLToArgvTestFlags; } virBhyveXMLToArgvTestFlags;
static int testCompareXMLToArgvFiles(const char *xml, static int testCompareXMLToArgvFiles(const char *xml,
@ -29,7 +32,7 @@ static int testCompareXMLToArgvFiles(const char *xml,
g_autofree char *actualargv = NULL; g_autofree char *actualargv = NULL;
g_autofree char *actualld = NULL; g_autofree char *actualld = NULL;
g_autofree char *actualdm = NULL; g_autofree char *actualdm = NULL;
g_autoptr(virDomainDef) vmdef = NULL; g_autoptr(virDomainObj) vm = NULL;
g_autoptr(virCommand) cmd = NULL; g_autoptr(virCommand) cmd = NULL;
g_autoptr(virCommand) ldcmd = NULL; g_autoptr(virCommand) ldcmd = NULL;
g_autoptr(virConnect) conn = NULL; g_autoptr(virConnect) conn = NULL;
@ -38,8 +41,11 @@ static int testCompareXMLToArgvFiles(const char *xml,
if (!(conn = virGetConnect())) if (!(conn = virGetConnect()))
goto out; goto out;
if (!(vmdef = virDomainDefParseFile(xml, driver.xmlopt, if (!(vm = virDomainObjNew(driver.xmlopt)))
NULL, VIR_DOMAIN_DEF_PARSE_INACTIVE))) { return -1;
if (!(vm->def = virDomainDefParseFile(xml, driver.xmlopt,
NULL, VIR_DOMAIN_DEF_PARSE_INACTIVE))) {
if (flags & FLAG_EXPECT_PARSE_ERROR) { if (flags & FLAG_EXPECT_PARSE_ERROR) {
ret = 0; ret = 0;
} else if (flags & FLAG_EXPECT_FAILURE) { } else if (flags & FLAG_EXPECT_FAILURE) {
@ -54,11 +60,20 @@ static int testCompareXMLToArgvFiles(const char *xml,
conn->privateData = &driver; conn->privateData = &driver;
cmd = virBhyveProcessBuildBhyveCmd(&driver, vmdef, false); if (bhyveProcessPrepareDomain(&driver, vm, 0) < 0) {
if (vmdef->os.loader) if (flags & FLAG_EXPECT_PREPARE_ERROR) {
ret = 0;
VIR_TEST_DEBUG("Got expected error: %s",
virGetLastErrorMessage());
}
goto out;
}
cmd = virBhyveProcessBuildBhyveCmd(&driver, vm->def, false);
if (vm->def->os.loader)
ldcmd = virCommandNew("dummy"); ldcmd = virCommandNew("dummy");
else else
ldcmd = virBhyveProcessBuildLoadCmd(&driver, vmdef, "<device.map>", ldcmd = virBhyveProcessBuildLoadCmd(&driver, vm->def, "<device.map>",
&actualdm); &actualdm);
if ((cmd == NULL) || (ldcmd == NULL)) { if ((cmd == NULL) || (ldcmd == NULL)) {
@ -94,10 +109,10 @@ static int testCompareXMLToArgvFiles(const char *xml,
ret = 0; ret = 0;
out: out:
if (vmdef && if (vm && vm->def &&
vmdef->ngraphics == 1 && vm->def->ngraphics == 1 &&
vmdef->graphics[0]->type == VIR_DOMAIN_GRAPHICS_TYPE_VNC) vm->def->graphics[0]->type == VIR_DOMAIN_GRAPHICS_TYPE_VNC)
virPortAllocatorRelease(vmdef->graphics[0]->data.vnc.port); virPortAllocatorRelease(vm->def->graphics[0]->data.vnc.port);
return ret; return ret;
} }
@ -132,6 +147,8 @@ static int
mymain(void) mymain(void)
{ {
int ret = 0; int ret = 0;
g_autofree char *fakefirmwaredir = g_strdup("fakefirmwaredir");
g_autofree char *fakefirmwareemptydir = g_strdup("fakefirmwareemptydir");
if ((driver.caps = virBhyveCapsBuild()) == NULL) if ((driver.caps = virBhyveCapsBuild()) == NULL)
return EXIT_FAILURE; return EXIT_FAILURE;
@ -142,6 +159,10 @@ mymain(void)
if (!(driver.remotePorts = virPortAllocatorRangeNew("display", 5900, 65535))) if (!(driver.remotePorts = virPortAllocatorRangeNew("display", 5900, 65535)))
return EXIT_FAILURE; return EXIT_FAILURE;
if (!(driver.config = virBhyveDriverConfigNew()))
return EXIT_FAILURE;
driver.config->firmwareDir = fakefirmwaredir;
# define DO_TEST_FULL(name, flags) \ # define DO_TEST_FULL(name, flags) \
do { \ do { \
@ -162,6 +183,9 @@ mymain(void)
# define DO_TEST_PARSE_ERROR(name) \ # define DO_TEST_PARSE_ERROR(name) \
DO_TEST_FULL(name, FLAG_EXPECT_PARSE_ERROR) DO_TEST_FULL(name, FLAG_EXPECT_PARSE_ERROR)
# define DO_TEST_PREPARE_ERROR(name) \
DO_TEST_FULL(name, FLAG_EXPECT_PREPARE_ERROR)
driver.grubcaps = BHYVE_GRUB_CAP_CONSDEV; driver.grubcaps = BHYVE_GRUB_CAP_CONSDEV;
driver.bhyvecaps = BHYVE_CAP_RTC_UTC | BHYVE_CAP_AHCI32SLOT | \ driver.bhyvecaps = BHYVE_CAP_RTC_UTC | BHYVE_CAP_AHCI32SLOT | \
BHYVE_CAP_NET_E1000 | BHYVE_CAP_LPC_BOOTROM | \ BHYVE_CAP_NET_E1000 | BHYVE_CAP_LPC_BOOTROM | \
@ -209,6 +233,9 @@ mymain(void)
DO_TEST("sound"); DO_TEST("sound");
DO_TEST("isa-controller"); DO_TEST("isa-controller");
DO_TEST_FAILURE("isa-multiple-controllers"); DO_TEST_FAILURE("isa-multiple-controllers");
DO_TEST("firmware-efi");
driver.config->firmwareDir = fakefirmwareemptydir;
DO_TEST_PREPARE_ERROR("firmware-efi");
DO_TEST("fs-9p"); DO_TEST("fs-9p");
DO_TEST("fs-9p-readonly"); DO_TEST("fs-9p-readonly");
DO_TEST_FAILURE("fs-9p-unsupported-type"); DO_TEST_FAILURE("fs-9p-unsupported-type");
@ -267,6 +294,7 @@ mymain(void)
virObjectUnref(driver.caps); virObjectUnref(driver.caps);
virObjectUnref(driver.xmlopt); virObjectUnref(driver.xmlopt);
virPortAllocatorRangeFree(driver.remotePorts); virPortAllocatorRangeFree(driver.remotePorts);
virObjectUnref(driver.config);
return ret == 0 ? EXIT_SUCCESS : EXIT_FAILURE; return ret == 0 ? EXIT_SUCCESS : EXIT_FAILURE;
} }