/* * qemu_vhost_user.c: QEMU vhost-user * * Copyright (C) 2019 Red Hat, Inc. * * 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 "qemu_vhost_user.h" #include "qemu_interop_config.h" #include "virjson.h" #include "virlog.h" #include "viralloc.h" #include "virenum.h" #include "virutil.h" #define VIR_FROM_THIS VIR_FROM_QEMU VIR_LOG_INIT("qemu.qemu_vhost_user"); typedef enum { QEMU_VHOST_USER_TYPE_NONE = 0, QEMU_VHOST_USER_TYPE_9P, QEMU_VHOST_USER_TYPE_BALLOON, QEMU_VHOST_USER_TYPE_BLOCK, QEMU_VHOST_USER_TYPE_CAIF, QEMU_VHOST_USER_TYPE_CONSOLE, QEMU_VHOST_USER_TYPE_CRYPTO, QEMU_VHOST_USER_TYPE_GPU, QEMU_VHOST_USER_TYPE_INPUT, QEMU_VHOST_USER_TYPE_NET, QEMU_VHOST_USER_TYPE_RNG, QEMU_VHOST_USER_TYPE_RPMSG, QEMU_VHOST_USER_TYPE_RPROC_SERIAL, QEMU_VHOST_USER_TYPE_SCSI, QEMU_VHOST_USER_TYPE_VSOCK, QEMU_VHOST_USER_TYPE_FS, QEMU_VHOST_USER_TYPE_LAST } qemuVhostUserType; VIR_ENUM_DECL(qemuVhostUserType); VIR_ENUM_IMPL(qemuVhostUserType, QEMU_VHOST_USER_TYPE_LAST, "", "9p", "balloon", "block", "caif", "console", "crypto", "gpu", "input", "net", "rng", "rpmsg", "rproc-serial", "scsi", "vsock", "fs", ); typedef enum { QEMU_VHOST_USER_GPU_FEATURE_NONE = 0, QEMU_VHOST_USER_GPU_FEATURE_VIRGL, QEMU_VHOST_USER_GPU_FEATURE_RENDER_NODE, QEMU_VHOST_USER_GPU_FEATURE_LAST } qemuVhostUserGPUFeature; VIR_ENUM_DECL(qemuVhostUserGPUFeature); VIR_ENUM_IMPL(qemuVhostUserGPUFeature, QEMU_VHOST_USER_GPU_FEATURE_LAST, "", "virgl", "render-node", ); typedef struct _qemuVhostUserGPU qemuVhostUserGPU; struct _qemuVhostUserGPU { size_t nfeatures; qemuVhostUserGPUFeature *features; }; struct _qemuVhostUser { /* Description intentionally not parsed. */ qemuVhostUserType type; char *binary; /* Tags intentionally not parsed. */ union { qemuVhostUserGPU gpu; } capabilities; }; static void qemuVhostUserGPUFeatureFree(qemuVhostUserGPUFeature *features) { g_free(features); } G_DEFINE_AUTOPTR_CLEANUP_FUNC(qemuVhostUserGPUFeature, qemuVhostUserGPUFeatureFree); void qemuVhostUserFree(qemuVhostUser *vu) { if (!vu) return; if (vu->type == QEMU_VHOST_USER_TYPE_GPU) g_free(vu->capabilities.gpu.features); g_free(vu->binary); g_free(vu); } /* 1MiB should be enough for everybody (TM) */ #define DOCUMENT_SIZE (1024 * 1024) static int qemuVhostUserTypeParse(const char *path, virJSONValue *doc, qemuVhostUser *vu) { const char *type = virJSONValueObjectGetString(doc, "type"); int tmp; VIR_DEBUG("vhost-user description path '%s' type : %s", path, type); if ((tmp = qemuVhostUserTypeTypeFromString(type)) <= 0) { virReportError(VIR_ERR_INTERNAL_ERROR, _("unknown vhost-user type: '%s'"), type); return -1; } vu->type = tmp; return 0; } static int qemuVhostUserBinaryParse(const char *path, virJSONValue *doc, qemuVhostUser *vu) { const char *binary = virJSONValueObjectGetString(doc, "binary"); VIR_DEBUG("vhost-user description path '%s' binary : %s", path, binary); vu->binary = g_strdup(binary); return 0; } qemuVhostUser * qemuVhostUserParse(const char *path) { g_autofree char *cont = NULL; g_autoptr(virJSONValue) doc = NULL; g_autoptr(qemuVhostUser) vu = NULL; if (virFileReadAll(path, DOCUMENT_SIZE, &cont) < 0) return NULL; if (!(doc = virJSONValueFromString(cont))) { virReportError(VIR_ERR_INTERNAL_ERROR, _("unable to parse json file '%s'"), path); return NULL; } vu = g_new0(qemuVhostUser, 1); if (qemuVhostUserTypeParse(path, doc, vu) < 0) return NULL; if (qemuVhostUserBinaryParse(path, doc, vu) < 0) return NULL; return g_steal_pointer(&vu); } char * qemuVhostUserFormat(qemuVhostUser *vu) { g_autoptr(virJSONValue) doc = NULL; if (!vu) return NULL; doc = virJSONValueNewObject(); if (virJSONValueObjectAppendString(doc, "type", qemuVhostUserTypeTypeToString(vu->type)) < 0) return NULL; if (virJSONValueObjectAppendString(doc, "binary", vu->binary) < 0) return NULL; return virJSONValueToString(doc, true); } int qemuVhostUserFetchConfigs(char ***configs, bool privileged) { return qemuInteropFetchConfigs("vhost-user", configs, privileged); } static ssize_t qemuVhostUserFetchParsedConfigs(bool privileged, qemuVhostUser ***vhostuserRet, char ***pathsRet) { g_auto(GStrv) paths = NULL; size_t npaths; qemuVhostUser **vus = NULL; size_t i; if (qemuVhostUserFetchConfigs(&paths, privileged) < 0) return -1; if (!paths) return 0; npaths = g_strv_length(paths); vus = g_new0(qemuVhostUser *, npaths); for (i = 0; i < npaths; i++) { if (!(vus[i] = qemuVhostUserParse(paths[i]))) goto error; } *vhostuserRet = g_steal_pointer(&vus); if (pathsRet) *pathsRet = g_steal_pointer(&paths); return npaths; error: while (i > 0) qemuVhostUserFree(vus[--i]); VIR_FREE(vus); return -1; } static int qemuVhostUserGPUFillCapabilities(qemuVhostUser *vu, virJSONValue *doc) { qemuVhostUserGPU *gpu = &vu->capabilities.gpu; virJSONValue *featuresJSON; size_t nfeatures; size_t nparsed = 0; size_t i; g_autoptr(qemuVhostUserGPUFeature) features = NULL; if (!(featuresJSON = virJSONValueObjectGetArray(doc, "features"))) { virReportError(VIR_ERR_INTERNAL_ERROR, _("failed to get features from '%s'"), vu->binary); return -1; } nfeatures = virJSONValueArraySize(featuresJSON); features = g_new0(qemuVhostUserGPUFeature, nfeatures); for (i = 0; i < nfeatures; i++) { virJSONValue *item = virJSONValueArrayGet(featuresJSON, i); const char *tmpStr = virJSONValueGetString(item); int tmp; if ((tmp = qemuVhostUserGPUFeatureTypeFromString(tmpStr)) <= 0) { VIR_DEBUG("ignoring unknown QEMU vhost-user feature '%s'", tmpStr); continue; } features[nparsed] = tmp; nparsed++; } gpu->features = g_steal_pointer(&features); gpu->nfeatures = nparsed; return 0; } static bool qemuVhostUserGPUHasFeature(qemuVhostUserGPU *gpu, qemuVhostUserGPUFeature feature) { size_t i; for (i = 0; i < gpu->nfeatures; i++) { if (gpu->features[i] == feature) return true; } return false; } int qemuVhostUserFillDomainGPU(virQEMUDriver *driver, virDomainVideoDef *video) { qemuVhostUser **vus = NULL; qemuVhostUser *vu = NULL; ssize_t nvus = 0; ssize_t i; int ret = -1; if ((nvus = qemuVhostUserFetchParsedConfigs(driver->privileged, &vus, NULL)) < 0) goto end; for (i = 0; i < nvus; i++) { g_autoptr(virJSONValue) doc = NULL; g_autofree char *output = NULL; g_autoptr(virCommand) cmd = NULL; vu = vus[i]; if (vu->type != QEMU_VHOST_USER_TYPE_GPU) continue; cmd = virCommandNewArgList(vu->binary, "--print-capabilities", NULL); virCommandSetOutputBuffer(cmd, &output); if (virCommandRun(cmd, NULL) < 0) continue; if (!(doc = virJSONValueFromString(output))) { virReportError(VIR_ERR_INTERNAL_ERROR, _("unable to parse json capabilities '%s'"), vu->binary); continue; } if (qemuVhostUserGPUFillCapabilities(vu, doc) < 0) continue; if (video->accel) { if (video->accel->accel3d && !qemuVhostUserGPUHasFeature(&vu->capabilities.gpu, QEMU_VHOST_USER_GPU_FEATURE_VIRGL)) continue; if (video->accel->rendernode && !qemuVhostUserGPUHasFeature(&vu->capabilities.gpu, QEMU_VHOST_USER_GPU_FEATURE_RENDER_NODE)) continue; } if (!video->driver) video->driver = g_new0(virDomainVideoDriverDef, 1); VIR_FREE(video->driver->vhost_user_binary); video->driver->vhost_user_binary = g_strdup(vu->binary); break; } if (i == nvus) { virReportError(VIR_ERR_OPERATION_FAILED, "%s", _("Unable to find a satisfying vhost-user-gpu")); goto end; } if (!video->accel) video->accel = g_new0(virDomainVideoAccelDef, 1); if (!video->accel->rendernode && qemuVhostUserGPUHasFeature(&vu->capabilities.gpu, QEMU_VHOST_USER_GPU_FEATURE_RENDER_NODE)) { video->accel->rendernode = virHostGetDRMRenderNode(); if (!video->accel->rendernode) goto end; } ret = 0; end: for (i = 0; i < nvus; i++) qemuVhostUserFree(vus[i]); VIR_FREE(vus); return ret; } int qemuVhostUserFillDomainFS(virQEMUDriver *driver, virDomainFSDef *fs) { qemuVhostUser **vus = NULL; ssize_t nvus = 0; ssize_t i; int ret = -1; if ((nvus = qemuVhostUserFetchParsedConfigs(driver->privileged, &vus, NULL)) < 0) goto end; for (i = 0; i < nvus; i++) { qemuVhostUser *vu = vus[i]; if (vu->type != QEMU_VHOST_USER_TYPE_FS) continue; fs->binary = g_strdup(vu->binary); break; } if (i == nvus) { virReportError(VIR_ERR_OPERATION_FAILED, "%s", _("Unable to find a satisfying virtiofsd")); goto end; } ret = 0; end: for (i = 0; i < nvus; i++) qemuVhostUserFree(vus[i]); g_free(vus); return ret; }