2012-01-10 17:31:21 +00:00
|
|
|
/*
|
|
|
|
* virt-host-validate-common.c: Sanity check helper APIs
|
|
|
|
*
|
2014-10-28 12:38:04 -06:00
|
|
|
* Copyright (C) 2012, 2014 Red Hat, Inc.
|
2012-01-10 17:31:21 +00:00
|
|
|
*
|
|
|
|
* 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
|
2012-09-20 16:30:55 -06:00
|
|
|
* License along with this library. If not, see
|
2012-07-21 18:06:23 +08:00
|
|
|
* <http://www.gnu.org/licenses/>.
|
2012-01-10 17:31:21 +00:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <sys/utsname.h>
|
2015-10-07 17:46:18 +01:00
|
|
|
#include <sys/stat.h>
|
2012-01-10 17:31:21 +00:00
|
|
|
|
2023-04-05 10:57:45 +02:00
|
|
|
#include "viracpi.h"
|
2012-12-12 18:06:53 +00:00
|
|
|
#include "viralloc.h"
|
2018-09-29 21:37:22 +02:00
|
|
|
#include "vircgroup.h"
|
2012-01-10 17:31:21 +00:00
|
|
|
#include "virfile.h"
|
|
|
|
#include "virt-host-validate-common.h"
|
2013-04-03 12:36:23 +02:00
|
|
|
#include "virstring.h"
|
2017-08-17 19:18:05 +05:30
|
|
|
#include "virarch.h"
|
2020-02-16 22:59:15 +01:00
|
|
|
#include "virutil.h"
|
2024-06-24 09:31:09 +02:00
|
|
|
#include "virhostcpu.h"
|
2012-01-10 17:31:21 +00:00
|
|
|
|
2013-06-07 17:10:28 +02:00
|
|
|
#define VIR_FROM_THIS VIR_FROM_NONE
|
|
|
|
|
2019-03-16 14:20:32 -04:00
|
|
|
VIR_ENUM_IMPL(virHostValidateCPUFlag,
|
|
|
|
VIR_HOST_VALIDATE_CPU_FLAG_LAST,
|
2016-03-29 16:38:28 +02:00
|
|
|
"vmx",
|
2016-05-03 08:10:54 +02:00
|
|
|
"svm",
|
2020-06-15 10:28:09 +02:00
|
|
|
"sie",
|
2020-06-15 10:28:10 +02:00
|
|
|
"158",
|
|
|
|
"sev");
|
2016-03-29 16:38:28 +02:00
|
|
|
|
2012-01-10 17:31:21 +00:00
|
|
|
|
2015-10-07 17:02:31 +01:00
|
|
|
int virHostValidateDeviceExists(const char *hvname,
|
|
|
|
const char *dev_name,
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateLevel level,
|
2015-10-07 17:02:31 +01:00
|
|
|
const char *hint)
|
2012-01-10 17:31:21 +00:00
|
|
|
{
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateCheck(hvname, _("Checking if device '%1$s' exists"), dev_name);
|
2015-10-07 17:02:31 +01:00
|
|
|
|
|
|
|
if (access(dev_name, F_OK) < 0) {
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateFail(level, "%s", hint);
|
|
|
|
return VIR_VALIDATE_FAILURE(level);
|
2015-10-07 17:02:31 +01:00
|
|
|
}
|
|
|
|
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidatePass();
|
2015-10-07 17:02:31 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int virHostValidateDeviceAccessible(const char *hvname,
|
|
|
|
const char *dev_name,
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateLevel level,
|
2015-10-07 17:02:31 +01:00
|
|
|
const char *hint)
|
|
|
|
{
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateCheck(hvname, _("Checking if device '%1$s' is accessible"), dev_name);
|
2012-01-10 17:31:21 +00:00
|
|
|
|
2012-03-29 17:08:27 -06:00
|
|
|
if (access(dev_name, R_OK|W_OK) < 0) {
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateFail(level, "%s", hint);
|
|
|
|
return VIR_VALIDATE_FAILURE(level);
|
2012-01-10 17:31:21 +00:00
|
|
|
}
|
|
|
|
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidatePass();
|
2012-01-10 17:31:21 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-10-07 16:58:39 +01:00
|
|
|
int virHostValidateNamespace(const char *hvname,
|
|
|
|
const char *ns_name,
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateLevel level,
|
2015-10-07 16:58:39 +01:00
|
|
|
const char *hint)
|
|
|
|
{
|
|
|
|
char nspath[100];
|
|
|
|
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateCheck(hvname, _("Checking for namespace '%1$s'"), ns_name);
|
2020-08-03 17:27:58 +02:00
|
|
|
|
2019-11-13 14:53:42 +01:00
|
|
|
g_snprintf(nspath, sizeof(nspath), "/proc/self/ns/%s", ns_name);
|
2015-10-07 16:58:39 +01:00
|
|
|
|
|
|
|
if (access(nspath, F_OK) < 0) {
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateFail(level, "%s", hint);
|
|
|
|
return VIR_VALIDATE_FAILURE(level);
|
2015-10-07 16:58:39 +01:00
|
|
|
}
|
|
|
|
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidatePass();
|
2015-10-07 16:58:39 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
virBitmap *virHostValidateGetCPUFlags(void)
|
2012-01-10 17:31:21 +00:00
|
|
|
{
|
2016-03-29 16:38:28 +02:00
|
|
|
FILE *fp;
|
2021-03-11 08:16:13 +01:00
|
|
|
virBitmap *flags = NULL;
|
2024-07-23 10:31:27 +02:00
|
|
|
g_autofree char *line = NULL;
|
|
|
|
size_t linelen = 0;
|
2012-01-10 17:31:21 +00:00
|
|
|
|
2016-03-29 16:38:28 +02:00
|
|
|
if (!(fp = fopen("/proc/cpuinfo", "r")))
|
|
|
|
return NULL;
|
|
|
|
|
2020-10-01 16:56:48 +02:00
|
|
|
flags = virBitmapNew(VIR_HOST_VALIDATE_CPU_FLAG_LAST);
|
2012-01-10 17:31:21 +00:00
|
|
|
|
2024-07-23 10:31:27 +02:00
|
|
|
while (getline(&line, &linelen, fp) > 0) {
|
2016-03-29 16:38:28 +02:00
|
|
|
char *start;
|
2021-03-23 06:52:54 +01:00
|
|
|
g_auto(GStrv) tokens = NULL;
|
|
|
|
GStrv next;
|
2012-01-10 17:31:21 +00:00
|
|
|
|
2016-05-03 08:10:52 +02:00
|
|
|
/* The line we're interested in is marked differently depending
|
|
|
|
* on the architecture, so check possible prefixes */
|
|
|
|
if (!STRPREFIX(line, "flags") &&
|
|
|
|
!STRPREFIX(line, "Features") &&
|
2020-06-15 10:28:09 +02:00
|
|
|
!STRPREFIX(line, "features") &&
|
|
|
|
!STRPREFIX(line, "facilities"))
|
2016-03-29 16:38:28 +02:00
|
|
|
continue;
|
|
|
|
|
2024-07-23 10:31:27 +02:00
|
|
|
/* getline() may include the trailing newline in the output
|
|
|
|
* buffer, so we need to clean that up ourselves. */
|
|
|
|
virStringTrimOptionalNewline(line);
|
2016-03-29 16:38:28 +02:00
|
|
|
|
|
|
|
/* Skip to the separator */
|
|
|
|
if (!(start = strchr(line, ':')))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/* Split the line using " " as a delimiter. The first token
|
|
|
|
* will always be ":", but that's okay */
|
2021-03-23 06:52:54 +01:00
|
|
|
if (!(tokens = g_strsplit(start, " ", 0)))
|
2016-03-29 16:38:28 +02:00
|
|
|
continue;
|
|
|
|
|
|
|
|
/* Go through all flags and check whether one of those we
|
|
|
|
* might want to check for later on is present; if that's
|
|
|
|
* the case, set the relevant bit in the bitmap */
|
2021-03-23 06:52:54 +01:00
|
|
|
for (next = tokens; *next; next++) {
|
2016-03-29 16:38:28 +02:00
|
|
|
int value;
|
|
|
|
|
2021-03-23 06:52:54 +01:00
|
|
|
if ((value = virHostValidateCPUFlagTypeFromString(*next)) >= 0)
|
2016-03-29 16:38:28 +02:00
|
|
|
ignore_value(virBitmapSetBit(flags, value));
|
2012-01-10 17:31:21 +00:00
|
|
|
}
|
2024-07-23 10:31:27 +02:00
|
|
|
}
|
2012-01-10 17:31:21 +00:00
|
|
|
|
|
|
|
VIR_FORCE_FCLOSE(fp);
|
|
|
|
|
2016-03-29 16:38:28 +02:00
|
|
|
return flags;
|
2012-01-10 17:31:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int virHostValidateLinuxKernel(const char *hvname,
|
|
|
|
int version,
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateLevel level,
|
2012-01-10 17:31:21 +00:00
|
|
|
const char *hint)
|
|
|
|
{
|
|
|
|
struct utsname uts;
|
2023-03-07 16:18:51 +01:00
|
|
|
unsigned long long thisversion;
|
2012-01-10 17:31:21 +00:00
|
|
|
|
|
|
|
uname(&uts);
|
|
|
|
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateCheck(hvname, _("Checking for Linux >= %1$d.%2$d.%3$d"),
|
|
|
|
((version >> 16) & 0xff),
|
|
|
|
((version >> 8) & 0xff),
|
|
|
|
(version & 0xff));
|
2012-01-10 17:31:21 +00:00
|
|
|
|
|
|
|
if (STRNEQ(uts.sysname, "Linux")) {
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateFail(level, "%s", hint);
|
|
|
|
return VIR_VALIDATE_FAILURE(level);
|
2012-01-10 17:31:21 +00:00
|
|
|
}
|
|
|
|
|
2020-02-24 00:47:18 +01:00
|
|
|
if (virStringParseVersion(&thisversion, uts.release, true) < 0) {
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateFail(level, "%s", hint);
|
|
|
|
return VIR_VALIDATE_FAILURE(level);
|
2012-01-10 17:31:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (thisversion < version) {
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateFail(level, "%s", hint);
|
|
|
|
return VIR_VALIDATE_FAILURE(level);
|
2012-01-10 17:31:21 +00:00
|
|
|
} else {
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidatePass();
|
2012-01-10 17:31:21 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
2015-10-07 17:36:37 +01:00
|
|
|
|
2018-09-29 21:37:22 +02:00
|
|
|
#ifdef __linux__
|
|
|
|
int virHostValidateCGroupControllers(const char *hvname,
|
|
|
|
int controllers,
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateLevel level)
|
2015-10-07 17:36:37 +01:00
|
|
|
{
|
2020-09-22 12:52:38 +02:00
|
|
|
g_autoptr(virCgroup) group = NULL;
|
2018-09-29 21:37:22 +02:00
|
|
|
int ret = 0;
|
|
|
|
size_t i;
|
2015-10-07 17:36:37 +01:00
|
|
|
|
2021-06-08 09:17:52 +02:00
|
|
|
if (virCgroupNew("/", -1, &group) < 0) {
|
|
|
|
fprintf(stderr, "Unable to initialize cgroups: %s\n",
|
|
|
|
virGetLastErrorMessage());
|
2024-06-06 17:07:24 +01:00
|
|
|
return VIR_VALIDATE_FAILURE(level);
|
2021-06-08 09:17:52 +02:00
|
|
|
}
|
2015-10-07 17:36:37 +01:00
|
|
|
|
2018-09-29 21:37:22 +02:00
|
|
|
for (i = 0; i < VIR_CGROUP_CONTROLLER_LAST; i++) {
|
|
|
|
int flag = 1 << i;
|
|
|
|
const char *cg_name = virCgroupControllerTypeToString(i);
|
2016-04-04 16:25:15 +02:00
|
|
|
|
2018-09-29 21:37:22 +02:00
|
|
|
if (!(controllers & flag))
|
2015-10-07 17:36:37 +01:00
|
|
|
continue;
|
|
|
|
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateCheck(hvname, _("Checking for cgroup '%1$s' controller support"), cg_name);
|
2015-10-07 17:36:37 +01:00
|
|
|
|
2018-09-29 21:37:22 +02:00
|
|
|
if (!virCgroupHasController(group, i)) {
|
2024-06-06 17:07:24 +01:00
|
|
|
ret = VIR_VALIDATE_FAILURE(level);
|
|
|
|
virValidateFail(level, "Enable '%s' in kernel Kconfig file or "
|
|
|
|
"mount/enable cgroup controller in your system",
|
|
|
|
cg_name);
|
2018-09-29 21:37:22 +02:00
|
|
|
} else {
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidatePass();
|
2016-04-04 16:25:15 +02:00
|
|
|
}
|
2015-10-07 17:36:37 +01:00
|
|
|
}
|
|
|
|
|
2018-09-29 21:37:22 +02:00
|
|
|
return ret;
|
2015-10-07 17:36:37 +01:00
|
|
|
}
|
2018-09-29 21:37:22 +02:00
|
|
|
#else /* !__linux__ */
|
2019-10-14 14:44:29 +02:00
|
|
|
int virHostValidateCGroupControllers(const char *hvname G_GNUC_UNUSED,
|
|
|
|
int controllers G_GNUC_UNUSED,
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateLevel level)
|
2015-10-12 14:33:04 +01:00
|
|
|
{
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateFail(level, "%s", "This platform does not support cgroups");
|
|
|
|
return VIR_VALIDATE_FAILURE(level);
|
2015-10-12 14:33:04 +01:00
|
|
|
}
|
2018-09-29 21:37:22 +02:00
|
|
|
#endif /* !__linux__ */
|
2015-10-07 17:46:18 +01:00
|
|
|
|
|
|
|
int virHostValidateIOMMU(const char *hvname,
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateLevel level)
|
2015-10-07 17:46:18 +01:00
|
|
|
{
|
2021-12-07 16:34:00 +01:00
|
|
|
g_autoptr(virBitmap) flags = NULL;
|
2015-10-07 17:46:18 +01:00
|
|
|
struct stat sb;
|
|
|
|
const char *bootarg = NULL;
|
2019-03-01 12:10:26 +01:00
|
|
|
bool isAMD = false, isIntel = false;
|
|
|
|
virArch arch = virArchFromHost();
|
|
|
|
struct dirent *dent;
|
|
|
|
int rc;
|
|
|
|
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateCheck(hvname, "%s", _("Checking for device assignment IOMMU support"));
|
2021-06-08 22:16:42 +02:00
|
|
|
|
2016-03-29 16:38:28 +02:00
|
|
|
flags = virHostValidateGetCPUFlags();
|
|
|
|
|
|
|
|
if (flags && virBitmapIsBitSet(flags, VIR_HOST_VALIDATE_CPU_FLAG_VMX))
|
2015-10-07 17:46:18 +01:00
|
|
|
isIntel = true;
|
2016-03-29 16:38:28 +02:00
|
|
|
else if (flags && virBitmapIsBitSet(flags, VIR_HOST_VALIDATE_CPU_FLAG_SVM))
|
2015-10-07 17:46:18 +01:00
|
|
|
isAMD = true;
|
|
|
|
|
|
|
|
if (isIntel) {
|
|
|
|
if (access("/sys/firmware/acpi/tables/DMAR", F_OK) == 0) {
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidatePass();
|
2015-10-07 17:46:18 +01:00
|
|
|
bootarg = "intel_iommu=on";
|
|
|
|
} else {
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateFail(level,
|
|
|
|
"No ACPI DMAR table found, IOMMU either "
|
|
|
|
"disabled in BIOS or not supported by this "
|
|
|
|
"hardware platform");
|
|
|
|
return VIR_VALIDATE_FAILURE(level);
|
2015-10-07 17:46:18 +01:00
|
|
|
}
|
|
|
|
} else if (isAMD) {
|
|
|
|
if (access("/sys/firmware/acpi/tables/IVRS", F_OK) == 0) {
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidatePass();
|
2015-10-07 17:46:18 +01:00
|
|
|
bootarg = "iommu=pt iommu=1";
|
|
|
|
} else {
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateFail(level,
|
|
|
|
"No ACPI IVRS table found, IOMMU either "
|
|
|
|
"disabled in BIOS or not supported by this "
|
|
|
|
"hardware platform");
|
|
|
|
return VIR_VALIDATE_FAILURE(level);
|
2015-10-07 17:46:18 +01:00
|
|
|
}
|
2019-03-01 12:10:26 +01:00
|
|
|
} else if (ARCH_IS_PPC64(arch)) {
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidatePass();
|
2019-03-01 12:10:26 +01:00
|
|
|
} else if (ARCH_IS_S390(arch)) {
|
2020-10-25 17:50:51 -04:00
|
|
|
g_autoptr(DIR) dir = NULL;
|
2020-10-25 17:56:22 -04:00
|
|
|
|
2019-03-01 12:10:26 +01:00
|
|
|
/* On s390x, we skip the IOMMU check if there are no PCI
|
|
|
|
* devices (which is quite usual on s390x). If there are
|
|
|
|
* no PCI devices the directory is still there but is
|
|
|
|
* empty. */
|
2021-12-17 16:20:37 +01:00
|
|
|
if (!virDirOpen(&dir, "/sys/bus/pci/devices")) {
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateFail(VIR_VALIDATE_NOTE,
|
|
|
|
"Skipped - PCI support disabled");
|
|
|
|
return VIR_VALIDATE_FAILURE(VIR_VALIDATE_NOTE);
|
2021-12-17 16:20:37 +01:00
|
|
|
}
|
2019-03-01 12:10:26 +01:00
|
|
|
rc = virDirRead(dir, &dent, NULL);
|
2021-12-17 16:20:37 +01:00
|
|
|
if (rc <= 0) {
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateFail(VIR_VALIDATE_NOTE,
|
|
|
|
"Skipped - No PCI devices are online");
|
|
|
|
return VIR_VALIDATE_FAILURE(VIR_VALIDATE_NOTE);
|
2021-12-17 16:20:37 +01:00
|
|
|
}
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidatePass();
|
2023-03-22 16:33:32 +01:00
|
|
|
} else if (ARCH_IS_ARM(arch)) {
|
2023-04-05 10:57:45 +02:00
|
|
|
if (access("/sys/firmware/acpi/tables/IORT", F_OK) != 0) {
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateFail(level,
|
|
|
|
"No ACPI IORT table found, IOMMU not "
|
|
|
|
"supported by this hardware platform");
|
|
|
|
return VIR_VALIDATE_FAILURE(level);
|
2023-04-05 10:57:45 +02:00
|
|
|
} else {
|
|
|
|
rc = virAcpiHasSMMU();
|
|
|
|
if (rc < 0) {
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateFail(level,
|
|
|
|
"Failed to parse ACPI IORT table");
|
|
|
|
return VIR_VALIDATE_FAILURE(level);
|
2023-04-05 10:57:45 +02:00
|
|
|
} else if (rc == 0) {
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateFail(level,
|
|
|
|
"No SMMU found");
|
|
|
|
return VIR_VALIDATE_FAILURE(level);
|
2023-04-05 10:57:45 +02:00
|
|
|
} else {
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidatePass();
|
2023-04-05 10:57:45 +02:00
|
|
|
}
|
2023-03-22 16:33:32 +01:00
|
|
|
}
|
2015-10-07 17:46:18 +01:00
|
|
|
} else {
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateFail(level,
|
|
|
|
"Unknown if this platform has IOMMU support");
|
|
|
|
return VIR_VALIDATE_FAILURE(level);
|
2015-10-07 17:46:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* We can only check on newer kernels with iommu groups & vfio */
|
|
|
|
if (stat("/sys/kernel/iommu_groups", &sb) < 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (!S_ISDIR(sb.st_mode))
|
|
|
|
return 0;
|
|
|
|
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateCheck(hvname, "%s", _("Checking if IOMMU is enabled by kernel"));
|
2015-10-07 17:46:18 +01:00
|
|
|
if (sb.st_nlink <= 2) {
|
2019-03-01 12:10:26 +01:00
|
|
|
if (bootarg)
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateFail(level,
|
|
|
|
"IOMMU appears to be disabled in kernel. "
|
|
|
|
"Add %s to kernel cmdline arguments", bootarg);
|
2017-08-17 19:18:05 +05:30
|
|
|
else
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateFail(level, "IOMMU capability not compiled into kernel.");
|
|
|
|
return VIR_VALIDATE_FAILURE(level);
|
2015-10-07 17:46:18 +01:00
|
|
|
}
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidatePass();
|
2015-10-07 17:46:18 +01:00
|
|
|
return 0;
|
|
|
|
}
|
2020-03-13 14:48:03 -03:00
|
|
|
|
|
|
|
|
|
|
|
bool virHostKernelModuleIsLoaded(const char *module)
|
|
|
|
{
|
|
|
|
FILE *fp;
|
|
|
|
bool ret = false;
|
|
|
|
|
|
|
|
if (!(fp = fopen("/proc/modules", "r")))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
do {
|
|
|
|
char line[1024];
|
|
|
|
|
|
|
|
if (!fgets(line, sizeof(line), fp))
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (STRPREFIX(line, module)) {
|
|
|
|
ret = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
} while (1);
|
|
|
|
|
|
|
|
VIR_FORCE_FCLOSE(fp);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
2020-06-15 10:28:09 +02:00
|
|
|
|
|
|
|
|
2024-06-24 09:22:16 +02:00
|
|
|
static int
|
2024-06-24 09:31:09 +02:00
|
|
|
virHostValidateAMDSev(const char *hvname,
|
|
|
|
virValidateLevel level)
|
2024-06-24 09:22:16 +02:00
|
|
|
{
|
|
|
|
g_autofree char *mod_value = NULL;
|
2024-06-24 09:31:09 +02:00
|
|
|
uint32_t eax, ebx;
|
2024-06-24 09:22:16 +02:00
|
|
|
|
|
|
|
if (virFileReadValueString(&mod_value, "/sys/module/kvm_amd/parameters/sev") < 0) {
|
|
|
|
virValidateFail(level, "AMD Secure Encrypted Virtualization not "
|
|
|
|
"supported by the currently used kernel");
|
|
|
|
return VIR_VALIDATE_FAILURE(level);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mod_value[0] != '1' && mod_value[0] != 'Y' && mod_value[0] != 'y') {
|
|
|
|
virValidateFail(level,
|
|
|
|
"AMD Secure Encrypted Virtualization appears to be "
|
|
|
|
"disabled in kernel. Add kvm_amd.sev=1 "
|
|
|
|
"to the kernel cmdline arguments");
|
|
|
|
return VIR_VALIDATE_FAILURE(level);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!virFileExists("/dev/sev")) {
|
|
|
|
virValidateFail(level,
|
|
|
|
"AMD Secure Encrypted Virtualization appears to be "
|
|
|
|
"disabled in firmware.");
|
|
|
|
return VIR_VALIDATE_FAILURE(level);
|
|
|
|
}
|
|
|
|
|
2024-06-24 09:31:09 +02:00
|
|
|
virValidatePass();
|
|
|
|
|
|
|
|
virValidateCheck(hvname, "%s",
|
|
|
|
_("Checking for AMD Secure Encrypted Virtualization-Encrypted State (SEV-ES)"));
|
|
|
|
|
|
|
|
virHostCPUX86GetCPUID(0x8000001F, 0, &eax, &ebx, NULL, NULL);
|
|
|
|
|
|
|
|
if (eax & (1U << 3)) {
|
|
|
|
virValidatePass();
|
|
|
|
} else {
|
|
|
|
virValidateFail(level,
|
|
|
|
"AMD SEV-ES is not supported");
|
|
|
|
return VIR_VALIDATE_FAILURE(level);
|
|
|
|
}
|
|
|
|
|
|
|
|
virValidateCheck(hvname, "%s",
|
|
|
|
_("Checking for AMD Secure Encrypted Virtualization-Secure Nested Paging (SEV-SNP)"));
|
|
|
|
|
|
|
|
if (eax & (1U << 4)) {
|
|
|
|
virValidatePass();
|
|
|
|
} else {
|
|
|
|
virValidateFail(level,
|
|
|
|
"AMD SEV-SNP is not supported");
|
|
|
|
return VIR_VALIDATE_FAILURE(level);
|
|
|
|
}
|
|
|
|
|
2024-06-24 09:22:16 +02:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-06-15 10:28:09 +02:00
|
|
|
int virHostValidateSecureGuests(const char *hvname,
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateLevel level)
|
2020-06-15 10:28:09 +02:00
|
|
|
{
|
2021-12-07 16:34:00 +01:00
|
|
|
g_autoptr(virBitmap) flags = NULL;
|
2020-06-15 10:28:09 +02:00
|
|
|
bool hasFac158 = false;
|
2020-06-15 10:28:10 +02:00
|
|
|
bool hasAMDSev = false;
|
2020-06-15 10:28:09 +02:00
|
|
|
virArch arch = virArchFromHost();
|
|
|
|
g_autofree char *cmdline = NULL;
|
|
|
|
static const char *kIBMValues[] = {"y", "Y", "on", "ON", "oN", "On", "1"};
|
|
|
|
|
|
|
|
flags = virHostValidateGetCPUFlags();
|
|
|
|
|
|
|
|
if (flags && virBitmapIsBitSet(flags, VIR_HOST_VALIDATE_CPU_FLAG_FACILITY_158))
|
|
|
|
hasFac158 = true;
|
2020-06-15 10:28:10 +02:00
|
|
|
else if (flags && virBitmapIsBitSet(flags, VIR_HOST_VALIDATE_CPU_FLAG_SEV))
|
|
|
|
hasAMDSev = true;
|
2020-06-15 10:28:09 +02:00
|
|
|
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateCheck(hvname, "%s", _("Checking for secure guest support"));
|
2020-06-15 10:28:09 +02:00
|
|
|
if (ARCH_IS_S390(arch)) {
|
|
|
|
if (hasFac158) {
|
|
|
|
if (!virFileIsDir("/sys/firmware/uv")) {
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateFail(level, "IBM Secure Execution not supported by "
|
|
|
|
"the currently used kernel");
|
|
|
|
return VIR_VALIDATE_FAILURE(level);
|
2020-06-15 10:28:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* we're prefix matching rather than equality matching here, because
|
|
|
|
* kernel would treat even something like prot_virt='yFOO' as
|
|
|
|
* enabled
|
|
|
|
*/
|
2021-06-08 09:21:27 +02:00
|
|
|
if (virFileReadValueString(&cmdline, "/proc/cmdline") >= 0 &&
|
|
|
|
virKernelCmdlineMatchParam(cmdline, "prot_virt", kIBMValues,
|
2020-06-15 10:28:09 +02:00
|
|
|
G_N_ELEMENTS(kIBMValues),
|
|
|
|
VIR_KERNEL_CMDLINE_FLAGS_SEARCH_FIRST |
|
|
|
|
VIR_KERNEL_CMDLINE_FLAGS_CMP_PREFIX)) {
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidatePass();
|
2020-06-15 10:28:09 +02:00
|
|
|
return 1;
|
|
|
|
} else {
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateFail(level,
|
|
|
|
"IBM Secure Execution appears to be disabled "
|
|
|
|
"in kernel. Add prot_virt=1 to kernel cmdline "
|
|
|
|
"arguments");
|
|
|
|
return VIR_VALIDATE_FAILURE(level);
|
2020-06-15 10:28:09 +02:00
|
|
|
}
|
|
|
|
} else {
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateFail(level, "Hardware or firmware does not provide "
|
|
|
|
"support for IBM Secure Execution");
|
|
|
|
return VIR_VALIDATE_FAILURE(level);
|
2020-06-15 10:28:09 +02:00
|
|
|
}
|
2020-06-15 10:28:10 +02:00
|
|
|
} else if (hasAMDSev) {
|
2024-07-11 09:32:40 +02:00
|
|
|
return virHostValidateAMDSev(hvname, level);
|
2020-06-15 10:28:09 +02:00
|
|
|
}
|
|
|
|
|
2024-06-06 17:07:24 +01:00
|
|
|
virValidateFail(level,
|
|
|
|
"Unknown if this platform has Secure Guest support");
|
|
|
|
return VIR_VALIDATE_FAILURE(level);
|
2020-06-15 10:28:09 +02:00
|
|
|
}
|