2022-12-15 14:19:16 -05:00
|
|
|
/*
|
|
|
|
* qemu_passt.c: QEMU passt support
|
|
|
|
*
|
|
|
|
* Copyright (C) 2022 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_dbus.h"
|
|
|
|
#include "qemu_extdevice.h"
|
|
|
|
#include "qemu_security.h"
|
|
|
|
#include "qemu_passt.h"
|
|
|
|
#include "virenum.h"
|
|
|
|
#include "virerror.h"
|
|
|
|
#include "virjson.h"
|
|
|
|
#include "virlog.h"
|
|
|
|
#include "virpidfile.h"
|
|
|
|
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_NONE
|
|
|
|
|
|
|
|
VIR_LOG_INIT("qemu.passt");
|
|
|
|
|
|
|
|
|
|
|
|
static char *
|
|
|
|
qemuPasstCreatePidFilename(virDomainObj *vm,
|
|
|
|
virDomainNetDef *net)
|
|
|
|
{
|
|
|
|
qemuDomainObjPrivate *priv = vm->privateData;
|
|
|
|
virQEMUDriver *driver = priv->driver;
|
|
|
|
g_autofree char *shortName = virDomainDefGetShortName(vm->def);
|
|
|
|
g_autoptr(virQEMUDriverConfig) cfg = virQEMUDriverGetConfig(driver);
|
|
|
|
g_autofree char *name = NULL;
|
|
|
|
|
|
|
|
name = g_strdup_printf("%s-%s-passt", shortName, net->info.alias);
|
|
|
|
|
|
|
|
return virPidFileBuildPath(cfg->passtStateDir, name);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static char *
|
|
|
|
qemuPasstCreateSocketPath(virDomainObj *vm,
|
|
|
|
virDomainNetDef *net)
|
|
|
|
{
|
|
|
|
qemuDomainObjPrivate *priv = vm->privateData;
|
|
|
|
virQEMUDriver *driver = priv->driver;
|
|
|
|
g_autofree char *shortName = virDomainDefGetShortName(vm->def);
|
|
|
|
g_autoptr(virQEMUDriverConfig) cfg = virQEMUDriverGetConfig(driver);
|
|
|
|
|
|
|
|
return g_strdup_printf("%s/%s-%s.socket", cfg->passtStateDir,
|
|
|
|
shortName, net->info.alias);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
qemuPasstGetPid(virDomainObj *vm,
|
|
|
|
virDomainNetDef *net,
|
|
|
|
pid_t *pid)
|
|
|
|
{
|
|
|
|
g_autofree char *pidfile = qemuPasstCreatePidFilename(vm, net);
|
|
|
|
|
2023-02-16 11:46:55 +01:00
|
|
|
return virPidFileReadPath(pidfile, pid);
|
2022-12-15 14:19:16 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
qemuPasstAddNetProps(virDomainObj *vm,
|
|
|
|
virDomainNetDef *net,
|
|
|
|
virJSONValue **netprops)
|
|
|
|
{
|
|
|
|
g_autofree char *passtSocketName = qemuPasstCreateSocketPath(vm, net);
|
|
|
|
g_autoptr(virJSONValue) addrprops = NULL;
|
2023-02-20 18:26:51 -05:00
|
|
|
qemuDomainObjPrivate *priv = vm->privateData;
|
|
|
|
virQEMUCaps *qemuCaps = priv->qemuCaps;
|
2022-12-15 14:19:16 -05:00
|
|
|
|
|
|
|
if (virJSONValueObjectAdd(&addrprops,
|
|
|
|
"s:type", "unix",
|
|
|
|
"s:path", passtSocketName,
|
|
|
|
NULL) < 0) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virJSONValueObjectAdd(netprops,
|
|
|
|
"s:type", "stream",
|
|
|
|
"a:addr", &addrprops,
|
|
|
|
"b:server", false,
|
|
|
|
NULL) < 0) {
|
|
|
|
return -1;
|
|
|
|
}
|
2023-02-20 18:26:51 -05:00
|
|
|
|
|
|
|
/* a narrow range of QEMU releases support -netdev stream, but
|
|
|
|
* don't support its "reconnect" option
|
|
|
|
*/
|
|
|
|
if (virQEMUCapsGet(qemuCaps, QEMU_CAPS_NETDEV_STREAM_RECONNECT) &&
|
|
|
|
virJSONValueObjectAdd(netprops, "u:reconnect", 5, NULL) < 0) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2022-12-15 14:19:16 -05:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-02-16 12:07:42 +01:00
|
|
|
static void
|
2023-02-21 20:19:07 +01:00
|
|
|
qemuPasstKill(const char *pidfile, const char *passtSocketName)
|
2022-12-15 14:19:16 -05:00
|
|
|
{
|
|
|
|
virErrorPtr orig_err;
|
2023-02-16 11:46:55 +01:00
|
|
|
pid_t pid = 0;
|
2022-12-15 14:19:16 -05:00
|
|
|
|
|
|
|
virErrorPreserveLast(&orig_err);
|
|
|
|
|
2023-02-16 11:46:55 +01:00
|
|
|
ignore_value(virPidFileReadPath(pidfile, &pid));
|
|
|
|
if (pid != 0)
|
|
|
|
virProcessKillPainfully(pid, true);
|
|
|
|
unlink(pidfile);
|
2022-12-15 14:19:16 -05:00
|
|
|
|
2023-02-21 20:19:07 +01:00
|
|
|
unlink(passtSocketName);
|
|
|
|
|
2022-12-15 14:19:16 -05:00
|
|
|
virErrorRestore(&orig_err);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-02-16 12:07:42 +01:00
|
|
|
void
|
|
|
|
qemuPasstStop(virDomainObj *vm,
|
|
|
|
virDomainNetDef *net)
|
|
|
|
{
|
|
|
|
g_autofree char *pidfile = qemuPasstCreatePidFilename(vm, net);
|
2023-02-21 20:19:07 +01:00
|
|
|
g_autofree char *passtSocketName = qemuPasstCreateSocketPath(vm, net);
|
2023-02-16 12:07:42 +01:00
|
|
|
|
2023-02-21 20:19:07 +01:00
|
|
|
qemuPasstKill(pidfile, passtSocketName);
|
2023-02-16 12:07:42 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-12-15 14:19:16 -05:00
|
|
|
int
|
|
|
|
qemuPasstSetupCgroup(virDomainObj *vm,
|
|
|
|
virDomainNetDef *net,
|
|
|
|
virCgroup *cgroup)
|
|
|
|
{
|
|
|
|
pid_t pid = (pid_t) -1;
|
|
|
|
|
2023-02-13 16:05:04 +01:00
|
|
|
if (qemuPasstGetPid(vm, net, &pid) < 0 || pid <= 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Could not get process ID of passt"));
|
2022-12-15 14:19:16 -05:00
|
|
|
return -1;
|
2023-02-13 16:05:04 +01:00
|
|
|
}
|
2022-12-15 14:19:16 -05:00
|
|
|
|
|
|
|
return virCgroupAddProcess(cgroup, pid);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
qemuPasstStart(virDomainObj *vm,
|
|
|
|
virDomainNetDef *net)
|
|
|
|
{
|
|
|
|
qemuDomainObjPrivate *priv = vm->privateData;
|
|
|
|
virQEMUDriver *driver = priv->driver;
|
|
|
|
g_autofree char *passtSocketName = qemuPasstCreateSocketPath(vm, net);
|
|
|
|
g_autoptr(virCommand) cmd = NULL;
|
|
|
|
g_autofree char *pidfile = qemuPasstCreatePidFilename(vm, net);
|
|
|
|
char macaddr[VIR_MAC_STRING_BUFLEN];
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
cmd = virCommandNew(PASST);
|
|
|
|
|
|
|
|
virCommandClearCaps(cmd);
|
|
|
|
|
|
|
|
virCommandAddArgList(cmd,
|
|
|
|
"--one-off",
|
|
|
|
"--socket", passtSocketName,
|
|
|
|
"--mac-addr", virMacAddrFormat(&net->mac, macaddr),
|
2023-02-16 11:46:55 +01:00
|
|
|
"--pid", pidfile,
|
2022-12-15 14:19:16 -05:00
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (net->mtu) {
|
|
|
|
virCommandAddArg(cmd, "--mtu");
|
|
|
|
virCommandAddArgFormat(cmd, "%u", net->mtu);
|
|
|
|
}
|
|
|
|
|
2023-01-12 23:42:16 -05:00
|
|
|
if (net->sourceDev)
|
|
|
|
virCommandAddArgList(cmd, "--interface", net->sourceDev, NULL);
|
2022-12-15 14:19:16 -05:00
|
|
|
|
|
|
|
if (net->backend.logFile)
|
|
|
|
virCommandAddArgList(cmd, "--log-file", net->backend.logFile, NULL);
|
|
|
|
|
|
|
|
/* Add IP address info */
|
|
|
|
for (i = 0; i < net->guestIP.nips; i++) {
|
|
|
|
const virNetDevIPAddr *ip = net->guestIP.ips[i];
|
|
|
|
g_autofree char *addr = NULL;
|
|
|
|
|
|
|
|
/* validation has already limited us to
|
|
|
|
* a single IPv4 and single IPv6 address
|
|
|
|
*/
|
|
|
|
if (!(addr = virSocketAddrFormat(&ip->address)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
virCommandAddArgList(cmd, "--address", addr, NULL);
|
|
|
|
|
|
|
|
if (ip->prefix && VIR_SOCKET_ADDR_IS_FAMILY(&ip->address, AF_INET)) {
|
|
|
|
/* validation already made sure no prefix is
|
|
|
|
* specified for IPv6 (not allowed by passt)
|
|
|
|
*/
|
|
|
|
virCommandAddArg(cmd, "--netmask");
|
|
|
|
virCommandAddArgFormat(cmd, "%u", ip->prefix);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Add port forwarding info */
|
|
|
|
|
|
|
|
for (i = 0; i < net->nPortForwards; i++) {
|
|
|
|
virDomainNetPortForward *pf = net->portForwards[i];
|
|
|
|
g_auto(virBuffer) buf = VIR_BUFFER_INITIALIZER;
|
|
|
|
|
|
|
|
if (pf->proto == VIR_DOMAIN_NET_PROTO_TCP) {
|
|
|
|
virCommandAddArg(cmd, "--tcp-ports");
|
|
|
|
} else if (pf->proto == VIR_DOMAIN_NET_PROTO_UDP) {
|
|
|
|
virCommandAddArg(cmd, "--udp-ports");
|
|
|
|
} else {
|
|
|
|
/* validation guarantees this will never happen */
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Invalid portForward proto value %u"), pf->proto);
|
2023-01-11 10:40:10 +01:00
|
|
|
return -1;
|
2022-12-15 14:19:16 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
if (VIR_SOCKET_ADDR_VALID(&pf->address)) {
|
|
|
|
g_autofree char *addr = NULL;
|
|
|
|
|
|
|
|
if (!(addr = virSocketAddrFormat(&pf->address)))
|
2023-01-11 10:40:10 +01:00
|
|
|
return -1;
|
2022-12-15 14:19:16 -05:00
|
|
|
|
|
|
|
virBufferAddStr(&buf, addr);
|
|
|
|
|
|
|
|
if (pf->dev)
|
|
|
|
virBufferAsprintf(&buf, "%%%s", pf->dev);
|
|
|
|
|
|
|
|
virBufferAddChar(&buf, '/');
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!pf->nRanges) {
|
|
|
|
virBufferAddLit(&buf, "all");
|
|
|
|
} else {
|
|
|
|
size_t r;
|
|
|
|
|
|
|
|
for (r = 0; r < pf->nRanges; r++) {
|
|
|
|
virDomainNetPortForwardRange *range = pf->ranges[r];
|
|
|
|
|
|
|
|
if (r > 0)
|
|
|
|
virBufferAddChar(&buf, ',');
|
|
|
|
|
|
|
|
if (range->exclude == VIR_TRISTATE_BOOL_YES)
|
|
|
|
virBufferAddChar(&buf, '~');
|
|
|
|
|
|
|
|
virBufferAsprintf(&buf, "%u", range->start);
|
|
|
|
|
|
|
|
if (range->end)
|
|
|
|
virBufferAsprintf(&buf, "-%u", range->end);
|
|
|
|
if (range->to) {
|
|
|
|
virBufferAsprintf(&buf, ":%u", range->to);
|
|
|
|
if (range->end) {
|
|
|
|
virBufferAsprintf(&buf, "-%u",
|
|
|
|
range->end + range->to - range->start);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
virCommandAddArg(cmd, virBufferCurrentContent(&buf));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (qemuExtDeviceLogCommand(driver, vm, cmd, "passt") < 0)
|
2023-01-11 10:40:10 +01:00
|
|
|
return -1;
|
2022-12-15 14:19:16 -05:00
|
|
|
|
2023-03-01 15:34:32 -05:00
|
|
|
if (qemuSecurityCommandRun(driver, vm, cmd, -1, -1, false, NULL) < 0)
|
2022-12-15 14:19:16 -05:00
|
|
|
goto error;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
error:
|
2023-02-21 20:19:07 +01:00
|
|
|
qemuPasstKill(pidfile, passtSocketName);
|
2022-12-15 14:19:16 -05:00
|
|
|
return -1;
|
|
|
|
}
|