mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2024-12-28 16:45:26 +00:00
3b4df5d350
In few places we have the following code pattern: int ret; ... /* @ret is not accessed here */ ret = f(...); return ret; This pattern can be written less verbose: ... return f(...); This patch was generated with following coccinelle spatch: @@ type T; constant C; expression f; identifier ret; @@ -T ret = C; ... when != ret -ret = f; -return ret; +return f; Afterwards I needed to fix a few places, e.g. comment in virDomainNetIPParseXML() was removed too because coccinelle thinks it refers to @ret while in fact it doesn't. Also in few places it replaced @ret declaration with a few spaces instead of removing the line. But nothing terribly wrong. Signed-off-by: Michal Privoznik <mprivozn@redhat.com> Reviewed-by: Daniel Henrique Barboza <danielhb413@gmail.com>
419 lines
10 KiB
C
419 lines
10 KiB
C
/*
|
|
* 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
|
|
* <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include "qemu_vhost_user.h"
|
|
#include "qemu_interop_config.h"
|
|
#include "virjson.h"
|
|
#include "virlog.h"
|
|
#include "virstring.h"
|
|
#include "viralloc.h"
|
|
#include "virenum.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;
|
|
typedef qemuVhostUserGPU *qemuVhostUserGPUPtr;
|
|
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)
|
|
{
|
|
VIR_FREE(features);
|
|
}
|
|
|
|
|
|
G_DEFINE_AUTOPTR_CLEANUP_FUNC(qemuVhostUserGPUFeature, qemuVhostUserGPUFeatureFree);
|
|
|
|
|
|
void
|
|
qemuVhostUserFree(qemuVhostUserPtr vu)
|
|
{
|
|
if (!vu)
|
|
return;
|
|
|
|
if (vu->type == QEMU_VHOST_USER_TYPE_GPU)
|
|
VIR_FREE(vu->capabilities.gpu.features);
|
|
|
|
VIR_FREE(vu->binary);
|
|
|
|
VIR_FREE(vu);
|
|
}
|
|
|
|
|
|
/* 1MiB should be enough for everybody (TM) */
|
|
#define DOCUMENT_SIZE (1024 * 1024)
|
|
|
|
|
|
static int
|
|
qemuVhostUserTypeParse(const char *path,
|
|
virJSONValuePtr doc,
|
|
qemuVhostUserPtr 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,
|
|
virJSONValuePtr doc,
|
|
qemuVhostUserPtr 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;
|
|
}
|
|
|
|
|
|
qemuVhostUserPtr
|
|
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;
|
|
}
|
|
|
|
if (VIR_ALLOC(vu) < 0)
|
|
return NULL;
|
|
|
|
if (qemuVhostUserTypeParse(path, doc, vu) < 0)
|
|
return NULL;
|
|
|
|
if (qemuVhostUserBinaryParse(path, doc, vu) < 0)
|
|
return NULL;
|
|
|
|
return g_steal_pointer(&vu);
|
|
}
|
|
|
|
|
|
char *
|
|
qemuVhostUserFormat(qemuVhostUserPtr vu)
|
|
{
|
|
g_autoptr(virJSONValue) doc = NULL;
|
|
|
|
if (!vu)
|
|
return NULL;
|
|
|
|
if (!(doc = virJSONValueNewObject()))
|
|
return NULL;
|
|
|
|
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,
|
|
qemuVhostUserPtr **vhostuserRet,
|
|
char ***pathsRet)
|
|
{
|
|
VIR_AUTOSTRINGLIST paths = NULL;
|
|
size_t npaths;
|
|
qemuVhostUserPtr *vus = NULL;
|
|
size_t i;
|
|
|
|
if (qemuVhostUserFetchConfigs(&paths, privileged) < 0)
|
|
return -1;
|
|
|
|
npaths = virStringListLength((const char **)paths);
|
|
|
|
if (VIR_ALLOC_N(vus, npaths) < 0)
|
|
return -1;
|
|
|
|
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(qemuVhostUserPtr vu,
|
|
virJSONValuePtr doc)
|
|
{
|
|
qemuVhostUserGPUPtr gpu = &vu->capabilities.gpu;
|
|
virJSONValuePtr featuresJSON;
|
|
size_t nfeatures;
|
|
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);
|
|
if (VIR_ALLOC_N(features, nfeatures) < 0)
|
|
return -1;
|
|
|
|
for (i = 0; i < nfeatures; i++) {
|
|
virJSONValuePtr item = virJSONValueArrayGet(featuresJSON, i);
|
|
const char *tmpStr = virJSONValueGetString(item);
|
|
int tmp;
|
|
|
|
if ((tmp = qemuVhostUserGPUFeatureTypeFromString(tmpStr)) <= 0) {
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
_("unknown feature %s"),
|
|
tmpStr);
|
|
continue;
|
|
}
|
|
|
|
features[i] = tmp;
|
|
}
|
|
|
|
gpu->features = g_steal_pointer(&features);
|
|
gpu->nfeatures = nfeatures;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
static bool
|
|
qemuVhostUserGPUHasFeature(qemuVhostUserGPUPtr gpu,
|
|
qemuVhostUserGPUFeature feature)
|
|
{
|
|
size_t i;
|
|
|
|
for (i = 0; i < gpu->nfeatures; i++) {
|
|
if (gpu->features[i] == feature)
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
int
|
|
qemuVhostUserFillDomainGPU(virQEMUDriverPtr driver,
|
|
virDomainVideoDefPtr video)
|
|
{
|
|
qemuVhostUserPtr *vus = NULL;
|
|
qemuVhostUserPtr 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 && VIR_ALLOC(video->driver) < 0)
|
|
goto end;
|
|
|
|
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 && VIR_ALLOC(video->accel) < 0)
|
|
goto end;
|
|
|
|
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;
|
|
}
|