/* * bhyve_capabilities.c: bhyve capabilities module * * Copyright (C) 2014 Roman Bogorodskiy * Copyright (C) 2014 Semihalf * Copyright (C) 2020 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 * . * */ #include #include #include #include #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 = VIR_TRISTATE_BOOL_YES; caps->disk.diskDevice.report = true; caps->disk.bus.report = true; caps->disk.model.report = 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 = VIR_TRISTATE_BOOL_YES; caps->os.loader.supported = VIR_TRISTATE_BOOL_NO; if (bhyvecaps & BHYVE_CAP_LPC_BOOTROM) { caps->os.loader.type.report = true; caps->os.loader.readonly.report = true; caps->os.loader.supported = VIR_TRISTATE_BOOL_YES; 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; } caps->graphics.supported = VIR_TRISTATE_BOOL_NO; caps->video.supported = VIR_TRISTATE_BOOL_NO; if (bhyvecaps & BHYVE_CAP_FBUF) { caps->graphics.supported = VIR_TRISTATE_BOOL_YES; caps->graphics.type.report = true; caps->video.supported = VIR_TRISTATE_BOOL_YES; caps->video.modelType.report = 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); } caps->hostdev.supported = VIR_TRISTATE_BOOL_NO; caps->features[VIR_DOMAIN_CAPS_FEATURE_IOTHREADS] = VIR_TRISTATE_BOOL_NO; caps->features[VIR_DOMAIN_CAPS_FEATURE_VMCOREINFO] = VIR_TRISTATE_BOOL_NO; caps->features[VIR_DOMAIN_CAPS_FEATURE_GENID] = VIR_TRISTATE_BOOL_NO; caps->gic.supported = VIR_TRISTATE_BOOL_NO; 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; } firmwares = g_new0(virDomainCapsStringValues, 1); 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; firmwares->values[firmwares->nvalues] = g_strdup_printf("%s/%s", firmware_dir, entry->d_name); 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; 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 bhyveProbeCapsFromHelp(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; /* "-c vcpus" was there before CPU topology support was introduced, * then it became * "-c [[cpus=]numcpus][,sockets=n][,cores=n][,threads=n] */ if (strstr(help, "-c vcpus") == NULL) *caps |= BHYVE_CAP_CPUTOPOLOGY; 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); } static int bhyveProbeCapsSoundHda(unsigned int *caps, char *binary) { return bhyveProbeCapsDeviceHelper(caps, binary, "-s", "0,hda", "pci slot 0:0: unknown device \"hda\"", BHYVE_CAP_SOUND_HDA); } static int bhyveProbeCapsVNCPassword(unsigned int *caps, char *binary) { return bhyveProbeCapsDeviceHelper(caps, binary, "-s", "0,fbuf,password=", "Invalid fbuf emulation \"password\"", BHYVE_CAP_VNC_PASSWORD); } static int bhyveProbeCapsVirtio9p(unsigned int *caps, char *binary) { return bhyveProbeCapsDeviceHelper(caps, binary, "-s", "0,virtio-9p", "pci slot 0:0: unknown device \"hda\"", BHYVE_CAP_VIRTIO_9P); } int virBhyveProbeCaps(unsigned int *caps) { char *binary; int ret = 0; binary = virFindFileInPath("bhyve"); if (binary == NULL) goto out; if ((ret = bhyveProbeCapsFromHelp(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; if ((ret = bhyveProbeCapsSoundHda(caps, binary))) goto out; if ((ret = bhyveProbeCapsVNCPassword(caps, binary))) goto out; if ((ret = bhyveProbeCapsVirtio9p(caps, binary))) goto out; out: VIR_FREE(binary); return ret; }