mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2024-10-07 14:55:46 +00:00
26ae4e482a
There is no "node driver" as there was before, drivers have to do their own ACL checking anyway, so they all specify their functions and nodeinfo is basically just extending conf/capablities. Hence moving the code to src/conf/ is the right way to go. Also that way we can de-duplicate some code that is in virsysfs and/or virhostcpu that got duplicated during the virhostcpu.c split. And Some cleanup is done throughout the changes, like adding the vir* prefix etc. Signed-off-by: Martin Kletzander <mkletzan@redhat.com>
343 lines
9.2 KiB
C
343 lines
9.2 KiB
C
/*
|
|
* bhyve_capabilities.c: bhyve capabilities module
|
|
*
|
|
* Copyright (C) 2014 Roman Bogorodskiy
|
|
* Copyright (C) 2014 Semihalf
|
|
* Copyright (C) 2016 Fabian Freyer
|
|
*
|
|
* 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 <sys/utsname.h>
|
|
#include <dirent.h>
|
|
#include <stdio.h>
|
|
#include <sys/types.h>
|
|
|
|
#include "viralloc.h"
|
|
#include "virfile.h"
|
|
#include "virlog.h"
|
|
#include "virstring.h"
|
|
#include "cpu/cpu.h"
|
|
#include "domain_conf.h"
|
|
#include "vircommand.h"
|
|
#include "bhyve_capabilities.h"
|
|
#include "bhyve_conf.h"
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_BHYVE
|
|
|
|
VIR_LOG_INIT("bhyve.bhyve_capabilities");
|
|
|
|
|
|
virCapsPtr
|
|
virBhyveCapsBuild(void)
|
|
{
|
|
virCapsPtr caps;
|
|
virCapsGuestPtr guest;
|
|
|
|
if ((caps = virCapabilitiesNew(virArchFromHost(),
|
|
false, false)) == NULL)
|
|
return NULL;
|
|
|
|
if ((guest = virCapabilitiesAddGuest(caps, VIR_DOMAIN_OSTYPE_HVM,
|
|
VIR_ARCH_X86_64,
|
|
"bhyve",
|
|
NULL, 0, NULL)) == NULL)
|
|
goto error;
|
|
|
|
if (virCapabilitiesAddGuestDomain(guest, VIR_DOMAIN_VIRT_BHYVE,
|
|
NULL, NULL, 0, NULL) == NULL)
|
|
goto error;
|
|
|
|
if (!(caps->host.cpu = virCPUProbeHost(caps->host.arch)))
|
|
VIR_WARN("Failed to get host CPU");
|
|
|
|
return caps;
|
|
|
|
error:
|
|
virObjectUnref(caps);
|
|
return NULL;
|
|
}
|
|
|
|
int
|
|
virBhyveDomainCapsFill(virDomainCapsPtr caps,
|
|
unsigned int bhyvecaps,
|
|
virDomainCapsStringValuesPtr firmwares)
|
|
{
|
|
caps->disk.supported = true;
|
|
VIR_DOMAIN_CAPS_ENUM_SET(caps->disk.diskDevice,
|
|
VIR_DOMAIN_DISK_DEVICE_DISK,
|
|
VIR_DOMAIN_DISK_DEVICE_CDROM);
|
|
|
|
VIR_DOMAIN_CAPS_ENUM_SET(caps->disk.bus,
|
|
VIR_DOMAIN_DISK_BUS_SATA,
|
|
VIR_DOMAIN_DISK_BUS_VIRTIO);
|
|
|
|
caps->os.supported = true;
|
|
|
|
if (bhyvecaps & BHYVE_CAP_LPC_BOOTROM) {
|
|
caps->os.loader.supported = true;
|
|
VIR_DOMAIN_CAPS_ENUM_SET(caps->os.loader.type,
|
|
VIR_DOMAIN_LOADER_TYPE_PFLASH);
|
|
VIR_DOMAIN_CAPS_ENUM_SET(caps->os.loader.readonly,
|
|
VIR_TRISTATE_BOOL_YES);
|
|
|
|
caps->os.loader.values.values = firmwares->values;
|
|
caps->os.loader.values.nvalues = firmwares->nvalues;
|
|
}
|
|
|
|
|
|
if (bhyvecaps & BHYVE_CAP_FBUF) {
|
|
caps->graphics.supported = true;
|
|
caps->video.supported = true;
|
|
VIR_DOMAIN_CAPS_ENUM_SET(caps->graphics.type, VIR_DOMAIN_GRAPHICS_TYPE_VNC);
|
|
VIR_DOMAIN_CAPS_ENUM_SET(caps->video.modelType, VIR_DOMAIN_VIDEO_TYPE_GOP);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
virDomainCapsPtr
|
|
virBhyveDomainCapsBuild(bhyveConnPtr conn,
|
|
const char *emulatorbin,
|
|
const char *machine,
|
|
virArch arch,
|
|
virDomainVirtType virttype)
|
|
{
|
|
virDomainCapsPtr caps = NULL;
|
|
unsigned int bhyve_caps = 0;
|
|
DIR *dir;
|
|
struct dirent *entry;
|
|
size_t firmwares_alloc = 0;
|
|
virBhyveDriverConfigPtr cfg = virBhyveDriverGetConfig(conn);
|
|
const char *firmware_dir = cfg->firmwareDir;
|
|
virDomainCapsStringValuesPtr firmwares = NULL;
|
|
|
|
if (!(caps = virDomainCapsNew(emulatorbin, machine, arch, virttype)))
|
|
goto cleanup;
|
|
|
|
if (virBhyveProbeCaps(&bhyve_caps)) {
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
_("failed probing capabilities"));
|
|
goto cleanup;
|
|
}
|
|
|
|
if (VIR_ALLOC(firmwares) < 0)
|
|
goto cleanup;
|
|
|
|
if (virDirOpenIfExists(&dir, firmware_dir) > 0) {
|
|
while ((virDirRead(dir, &entry, firmware_dir)) > 0) {
|
|
if (VIR_RESIZE_N(firmwares->values,
|
|
firmwares_alloc, firmwares->nvalues, 1) < 0)
|
|
goto cleanup;
|
|
|
|
if (virAsprintf(
|
|
&firmwares->values[firmwares->nvalues],
|
|
"%s/%s", firmware_dir, entry->d_name) < 0)
|
|
goto cleanup;
|
|
|
|
firmwares->nvalues++;
|
|
}
|
|
} else {
|
|
VIR_WARN("Cannot open firmware directory %s", firmware_dir);
|
|
}
|
|
|
|
if (virBhyveDomainCapsFill(caps, bhyve_caps, firmwares) < 0)
|
|
goto cleanup;
|
|
|
|
cleanup:
|
|
VIR_FREE(firmwares);
|
|
VIR_DIR_CLOSE(dir);
|
|
virObjectUnref(cfg);
|
|
return caps;
|
|
}
|
|
|
|
int
|
|
virBhyveProbeGrubCaps(virBhyveGrubCapsFlags *caps)
|
|
{
|
|
char *binary, *help;
|
|
virCommandPtr cmd;
|
|
int ret, exit;
|
|
|
|
ret = 0;
|
|
*caps = 0;
|
|
cmd = NULL;
|
|
help = NULL;
|
|
|
|
binary = virFindFileInPath("grub-bhyve");
|
|
if (binary == NULL)
|
|
goto out;
|
|
if (!virFileIsExecutable(binary))
|
|
goto out;
|
|
|
|
cmd = virCommandNew(binary);
|
|
virCommandAddArg(cmd, "--help");
|
|
virCommandSetOutputBuffer(cmd, &help);
|
|
if (virCommandRun(cmd, &exit) < 0) {
|
|
ret = -1;
|
|
goto out;
|
|
}
|
|
|
|
if (strstr(help, "--cons-dev") != NULL)
|
|
*caps |= BHYVE_GRUB_CAP_CONSDEV;
|
|
|
|
out:
|
|
VIR_FREE(help);
|
|
virCommandFree(cmd);
|
|
VIR_FREE(binary);
|
|
return ret;
|
|
}
|
|
|
|
static int
|
|
bhyveProbeCapsDeviceHelper(unsigned int *caps,
|
|
char *binary,
|
|
const char *bus,
|
|
const char *device,
|
|
const char *errormsg,
|
|
unsigned int flag)
|
|
{
|
|
char *error;
|
|
virCommandPtr cmd = NULL;
|
|
int ret = -1, exit;
|
|
|
|
cmd = virCommandNew(binary);
|
|
virCommandAddArgList(cmd, bus, device, NULL);
|
|
virCommandSetErrorBuffer(cmd, &error);
|
|
if (virCommandRun(cmd, &exit) < 0)
|
|
goto cleanup;
|
|
|
|
if (strstr(error, errormsg) == NULL)
|
|
*caps |= flag;
|
|
|
|
ret = 0;
|
|
cleanup:
|
|
VIR_FREE(error);
|
|
virCommandFree(cmd);
|
|
return ret;
|
|
}
|
|
|
|
static int
|
|
bhyveProbeCapsRTC_UTC(unsigned int *caps, char *binary)
|
|
{
|
|
char *help;
|
|
virCommandPtr cmd = NULL;
|
|
int ret = 0, exit;
|
|
|
|
cmd = virCommandNew(binary);
|
|
virCommandAddArg(cmd, "-h");
|
|
virCommandSetErrorBuffer(cmd, &help);
|
|
if (virCommandRun(cmd, &exit) < 0) {
|
|
ret = -1;
|
|
goto out;
|
|
}
|
|
|
|
if (strstr(help, "-u:") != NULL)
|
|
*caps |= BHYVE_CAP_RTC_UTC;
|
|
|
|
out:
|
|
VIR_FREE(help);
|
|
virCommandFree(cmd);
|
|
return ret;
|
|
}
|
|
|
|
static int
|
|
bhyveProbeCapsAHCI32Slot(unsigned int *caps, char *binary)
|
|
{
|
|
return bhyveProbeCapsDeviceHelper(caps, binary,
|
|
"-s",
|
|
"0,ahci",
|
|
"pci slot 0:0: unknown device \"ahci\"",
|
|
BHYVE_CAP_AHCI32SLOT);
|
|
}
|
|
|
|
|
|
static int
|
|
bhyveProbeCapsNetE1000(unsigned int *caps, char *binary)
|
|
{
|
|
return bhyveProbeCapsDeviceHelper(caps, binary,
|
|
"-s",
|
|
"0,e1000",
|
|
"pci slot 0:0: unknown device \"e1000\"",
|
|
BHYVE_CAP_NET_E1000);
|
|
}
|
|
|
|
static int
|
|
bhyveProbeCapsLPC_Bootrom(unsigned int *caps, char *binary)
|
|
{
|
|
return bhyveProbeCapsDeviceHelper(caps, binary,
|
|
"-l",
|
|
"bootrom",
|
|
"bhyve: invalid lpc device configuration 'bootrom'",
|
|
BHYVE_CAP_LPC_BOOTROM);
|
|
}
|
|
|
|
|
|
static int
|
|
bhyveProbeCapsFramebuffer(unsigned int *caps, char *binary)
|
|
{
|
|
return bhyveProbeCapsDeviceHelper(caps, binary,
|
|
"-s",
|
|
"0,fbuf",
|
|
"pci slot 0:0: unknown device \"fbuf\"",
|
|
BHYVE_CAP_FBUF);
|
|
}
|
|
|
|
|
|
static int
|
|
bhyveProbeCapsXHCIController(unsigned int *caps, char *binary)
|
|
{
|
|
return bhyveProbeCapsDeviceHelper(caps, binary,
|
|
"-s",
|
|
"0,xhci",
|
|
"pci slot 0:0: unknown device \"xhci\"",
|
|
BHYVE_CAP_FBUF);
|
|
}
|
|
|
|
|
|
int
|
|
virBhyveProbeCaps(unsigned int *caps)
|
|
{
|
|
char *binary;
|
|
int ret = 0;
|
|
|
|
binary = virFindFileInPath("bhyve");
|
|
if (binary == NULL)
|
|
goto out;
|
|
if (!virFileIsExecutable(binary))
|
|
goto out;
|
|
|
|
if ((ret = bhyveProbeCapsRTC_UTC(caps, binary)))
|
|
goto out;
|
|
|
|
if ((ret = bhyveProbeCapsAHCI32Slot(caps, binary)))
|
|
goto out;
|
|
|
|
if ((ret = bhyveProbeCapsNetE1000(caps, binary)))
|
|
goto out;
|
|
|
|
if ((ret = bhyveProbeCapsLPC_Bootrom(caps, binary)))
|
|
goto out;
|
|
|
|
if ((ret = bhyveProbeCapsFramebuffer(caps, binary)))
|
|
goto out;
|
|
|
|
if ((ret = bhyveProbeCapsXHCIController(caps, binary)))
|
|
goto out;
|
|
|
|
out:
|
|
VIR_FREE(binary);
|
|
return ret;
|
|
}
|