2013-09-03 23:43:00 -06:00
|
|
|
/*
|
|
|
|
* libxl_conf.c: libxl configuration management
|
|
|
|
*
|
2016-06-07 11:39:34 -04:00
|
|
|
* Copyright (C) 2012-2014, 2016 Red Hat, Inc.
|
2013-09-03 23:43:00 -06:00
|
|
|
* Copyright (c) 2011-2013 SUSE LINUX Products GmbH, Nuernberg, Germany.
|
2012-02-02 15:47:04 -07:00
|
|
|
* Copyright (C) 2011 Univention GmbH.
|
2011-02-10 15:42:34 -07: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/>.
|
2011-02-10 15:42:34 -07:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include <libxl.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
|
|
|
|
#include "internal.h"
|
2012-12-12 17:59:27 +00:00
|
|
|
#include "virlog.h"
|
2012-12-13 18:21:53 +00:00
|
|
|
#include "virerror.h"
|
2011-02-10 15:42:34 -07:00
|
|
|
#include "datatypes.h"
|
2013-09-09 09:15:15 -06:00
|
|
|
#include "virconf.h"
|
2011-07-19 12:32:58 -06:00
|
|
|
#include "virfile.h"
|
2021-05-07 16:53:40 +01:00
|
|
|
#include "viridentity.h"
|
2012-11-26 09:28:56 -07:00
|
|
|
#include "virstring.h"
|
2012-12-12 18:06:53 +00:00
|
|
|
#include "viralloc.h"
|
2012-12-13 18:01:25 +00:00
|
|
|
#include "viruuid.h"
|
2014-09-19 10:32:43 -06:00
|
|
|
#include "vircommand.h"
|
2017-07-05 11:19:43 +01:00
|
|
|
#include "virsocketaddr.h"
|
2021-03-25 17:26:08 +01:00
|
|
|
#include "libxl_api_wrapper.h"
|
2013-08-30 11:18:31 -06:00
|
|
|
#include "libxl_domain.h"
|
2011-02-10 15:42:34 -07:00
|
|
|
#include "libxl_conf.h"
|
2012-11-26 09:28:56 -07:00
|
|
|
#include "libxl_utils.h"
|
2012-12-13 15:25:48 +00:00
|
|
|
#include "virstoragefile.h"
|
2019-10-24 18:00:55 +02:00
|
|
|
#include "virsecret.h"
|
2017-04-24 15:06:59 +02:00
|
|
|
#include "cpu/cpu.h"
|
2017-12-07 03:27:45 +01:00
|
|
|
#include "xen_common.h"
|
2018-04-12 03:03:24 +02:00
|
|
|
#include "xen_xl.h"
|
2016-06-28 14:20:02 -06:00
|
|
|
#include "virnetdevvportprofile.h"
|
2019-04-01 12:14:26 +02:00
|
|
|
#include "virenum.h"
|
2021-02-01 12:50:00 +01:00
|
|
|
#include "virsecureerase.h"
|
2011-02-10 15:42:34 -07:00
|
|
|
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_LIBXL
|
|
|
|
|
2014-02-28 12:16:17 +00:00
|
|
|
VIR_LOG_INIT("libxl.libxl_conf");
|
|
|
|
|
2011-02-10 15:42:34 -07:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
static virClass *libxlDriverConfigClass;
|
2013-08-30 14:52:14 -06:00
|
|
|
static void libxlDriverConfigDispose(void *obj);
|
|
|
|
|
|
|
|
static int libxlConfigOnceInit(void)
|
|
|
|
{
|
2018-04-17 17:42:33 +02:00
|
|
|
if (!VIR_CLASS_NEW(libxlDriverConfig, virClassForObject()))
|
2013-08-30 14:52:14 -06:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-01-20 12:23:29 -05:00
|
|
|
VIR_ONCE_GLOBAL_INIT(libxlConfig);
|
2013-08-30 14:52:14 -06:00
|
|
|
|
|
|
|
static void
|
|
|
|
libxlDriverConfigDispose(void *obj)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlDriverConfig *cfg = obj;
|
2013-08-30 14:52:14 -06:00
|
|
|
|
|
|
|
virObjectUnref(cfg->caps);
|
|
|
|
libxl_ctx_free(cfg->ctx);
|
2018-04-12 03:03:19 +02:00
|
|
|
if (cfg->logger)
|
|
|
|
libxlLoggerFree(cfg->logger);
|
2013-08-30 14:52:14 -06:00
|
|
|
|
2021-02-03 15:13:46 -05:00
|
|
|
g_free(cfg->configBaseDir);
|
|
|
|
g_free(cfg->configDir);
|
|
|
|
g_free(cfg->autostartDir);
|
|
|
|
g_free(cfg->logDir);
|
|
|
|
g_free(cfg->stateDir);
|
|
|
|
g_free(cfg->libDir);
|
|
|
|
g_free(cfg->saveDir);
|
|
|
|
g_free(cfg->autoDumpDir);
|
|
|
|
g_free(cfg->lockManagerName);
|
|
|
|
g_free(cfg->channelDir);
|
2016-05-18 08:24:37 -06:00
|
|
|
virFirmwareFreeList(cfg->firmwares, cfg->nfirmwares);
|
2013-08-30 14:52:14 -06:00
|
|
|
}
|
|
|
|
|
2014-09-03 14:14:50 -06:00
|
|
|
|
|
|
|
static libxl_action_on_shutdown
|
|
|
|
libxlActionFromVirLifecycle(virDomainLifecycleAction action)
|
|
|
|
{
|
|
|
|
switch (action) {
|
2017-10-10 14:32:11 +02:00
|
|
|
case VIR_DOMAIN_LIFECYCLE_ACTION_DESTROY:
|
2014-09-03 14:14:50 -06:00
|
|
|
return LIBXL_ACTION_ON_SHUTDOWN_DESTROY;
|
|
|
|
|
2017-10-10 14:32:11 +02:00
|
|
|
case VIR_DOMAIN_LIFECYCLE_ACTION_RESTART:
|
|
|
|
return LIBXL_ACTION_ON_SHUTDOWN_RESTART;
|
2014-09-03 14:14:50 -06:00
|
|
|
|
2017-10-10 14:32:11 +02:00
|
|
|
case VIR_DOMAIN_LIFECYCLE_ACTION_RESTART_RENAME:
|
2014-09-03 14:14:50 -06:00
|
|
|
return LIBXL_ACTION_ON_SHUTDOWN_RESTART_RENAME;
|
|
|
|
|
2017-10-10 14:32:11 +02:00
|
|
|
case VIR_DOMAIN_LIFECYCLE_ACTION_PRESERVE:
|
2014-09-03 14:14:50 -06:00
|
|
|
return LIBXL_ACTION_ON_SHUTDOWN_PRESERVE;
|
|
|
|
|
2017-10-10 14:51:38 +02:00
|
|
|
case VIR_DOMAIN_LIFECYCLE_ACTION_COREDUMP_DESTROY:
|
2014-09-03 14:14:50 -06:00
|
|
|
return LIBXL_ACTION_ON_SHUTDOWN_COREDUMP_DESTROY;
|
|
|
|
|
2017-10-10 14:51:38 +02:00
|
|
|
case VIR_DOMAIN_LIFECYCLE_ACTION_COREDUMP_RESTART:
|
2014-09-03 14:14:50 -06:00
|
|
|
return LIBXL_ACTION_ON_SHUTDOWN_COREDUMP_RESTART;
|
|
|
|
|
2017-10-10 14:51:38 +02:00
|
|
|
case VIR_DOMAIN_LIFECYCLE_ACTION_LAST:
|
2014-09-03 14:14:50 -06:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-02-10 15:42:34 -07:00
|
|
|
static int
|
2013-08-30 14:34:10 -06:00
|
|
|
libxlMakeDomCreateInfo(libxl_ctx *ctx,
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainDef *def,
|
2012-11-26 09:28:56 -07:00
|
|
|
libxl_domain_create_info *c_info)
|
2011-02-10 15:42:34 -07:00
|
|
|
{
|
|
|
|
char uuidstr[VIR_UUID_STRING_BUFLEN];
|
|
|
|
|
2018-11-26 20:34:37 +01:00
|
|
|
if (def->os.type == VIR_DOMAIN_OSTYPE_HVM ||
|
|
|
|
def->os.type == VIR_DOMAIN_OSTYPE_XENPVH) {
|
2020-09-01 13:27:44 +02:00
|
|
|
#ifdef WITH_XEN_PVH
|
2018-11-26 20:34:37 +01:00
|
|
|
c_info->type = def->os.type == VIR_DOMAIN_OSTYPE_HVM ?
|
|
|
|
LIBXL_DOMAIN_TYPE_HVM : LIBXL_DOMAIN_TYPE_PVH;
|
|
|
|
#else
|
|
|
|
if (def->os.type == VIR_DOMAIN_OSTYPE_XENPVH) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("PVH guest os type not supported"));
|
|
|
|
return -1;
|
|
|
|
}
|
2012-11-26 09:28:56 -07:00
|
|
|
c_info->type = LIBXL_DOMAIN_TYPE_HVM;
|
2018-11-26 20:34:37 +01:00
|
|
|
#endif
|
2016-02-22 20:06:57 -07:00
|
|
|
switch ((virTristateSwitch) def->features[VIR_DOMAIN_FEATURE_HAP]) {
|
|
|
|
case VIR_TRISTATE_SWITCH_OFF:
|
|
|
|
libxl_defbool_set(&c_info->hap, false);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_TRISTATE_SWITCH_ON:
|
|
|
|
libxl_defbool_set(&c_info->hap, true);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_TRISTATE_SWITCH_ABSENT:
|
|
|
|
case VIR_TRISTATE_SWITCH_LAST:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else {
|
2012-11-26 09:28:56 -07:00
|
|
|
c_info->type = LIBXL_DOMAIN_TYPE_PV;
|
2016-02-22 20:06:57 -07:00
|
|
|
}
|
2011-02-10 15:42:34 -07:00
|
|
|
|
2020-04-15 17:18:25 -06:00
|
|
|
#ifdef LIBXL_HAVE_CREATEINFO_PASSTHROUGH
|
|
|
|
if (def->features[VIR_DOMAIN_FEATURE_XEN] == VIR_TRISTATE_SWITCH_ON) {
|
|
|
|
switch ((virTristateSwitch) def->xen_features[VIR_DOMAIN_XEN_PASSTHROUGH]) {
|
|
|
|
case VIR_TRISTATE_SWITCH_ON:
|
|
|
|
if (def->xen_passthrough_mode == VIR_DOMAIN_XEN_PASSTHROUGH_MODE_SYNC_PT)
|
|
|
|
c_info->passthrough = LIBXL_PASSTHROUGH_SYNC_PT;
|
|
|
|
else if (def->xen_passthrough_mode == VIR_DOMAIN_XEN_PASSTHROUGH_MODE_SHARE_PT)
|
|
|
|
c_info->passthrough = LIBXL_PASSTHROUGH_SHARE_PT;
|
|
|
|
else
|
|
|
|
c_info->passthrough = LIBXL_PASSTHROUGH_ENABLED;
|
|
|
|
break;
|
|
|
|
case VIR_TRISTATE_SWITCH_OFF:
|
|
|
|
c_info->passthrough = LIBXL_PASSTHROUGH_DISABLED;
|
|
|
|
break;
|
|
|
|
case VIR_TRISTATE_SWITCH_ABSENT:
|
|
|
|
case VIR_TRISTATE_SWITCH_LAST:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
c_info->name = g_strdup(def->name);
|
2011-02-10 15:42:34 -07:00
|
|
|
|
2012-11-26 09:28:56 -07:00
|
|
|
if (def->nseclabels &&
|
|
|
|
def->seclabels[0]->type == VIR_DOMAIN_SECLABEL_STATIC) {
|
2013-08-30 14:34:10 -06:00
|
|
|
if (libxl_flask_context_to_sid(ctx,
|
2012-11-26 09:28:56 -07:00
|
|
|
def->seclabels[0]->label,
|
|
|
|
strlen(def->seclabels[0]->label),
|
|
|
|
&c_info->ssidref)) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("libxenlight failed to resolve security label '%s'"),
|
|
|
|
def->seclabels[0]->label);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-02-10 15:42:34 -07:00
|
|
|
virUUIDFormat(def->uuid, uuidstr);
|
2012-10-17 10:23:12 +01:00
|
|
|
if (libxl_uuid_from_string(&c_info->uuid, uuidstr)) {
|
2012-07-18 14:32:10 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("libxenlight failed to parse UUID '%s'"), uuidstr);
|
2011-02-10 15:42:34 -07:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
2014-03-25 07:51:17 +01:00
|
|
|
error:
|
2012-11-26 09:28:56 -07:00
|
|
|
libxl_domain_create_info_dispose(c_info);
|
2011-02-10 15:42:34 -07:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2013-07-26 11:30:50 +08:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlMakeChrdevStr(virDomainChrDef *def, char **buf)
|
2013-07-26 11:30:50 +08:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainChrSourceDef *srcdef = def->source;
|
2016-10-21 07:45:54 -04:00
|
|
|
const char *type = virDomainChrTypeToString(srcdef->type);
|
2013-07-26 11:30:50 +08:00
|
|
|
|
|
|
|
if (!type) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
"%s", _("unknown chrdev type"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2016-10-21 07:45:54 -04:00
|
|
|
switch (srcdef->type) {
|
2013-07-26 11:30:50 +08:00
|
|
|
case VIR_DOMAIN_CHR_TYPE_NULL:
|
|
|
|
case VIR_DOMAIN_CHR_TYPE_STDIO:
|
|
|
|
case VIR_DOMAIN_CHR_TYPE_VC:
|
|
|
|
case VIR_DOMAIN_CHR_TYPE_PTY:
|
2019-10-20 13:49:46 +02:00
|
|
|
*buf = g_strdup(type);
|
2013-07-26 11:30:50 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_CHR_TYPE_FILE:
|
|
|
|
case VIR_DOMAIN_CHR_TYPE_PIPE:
|
2019-10-22 15:26:14 +02:00
|
|
|
*buf = g_strdup_printf("%s:%s", type, srcdef->data.file.path);
|
2013-07-26 11:30:50 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_CHR_TYPE_DEV:
|
2019-10-20 13:49:46 +02:00
|
|
|
*buf = g_strdup(srcdef->data.file.path);
|
2013-07-26 11:30:50 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_CHR_TYPE_UDP: {
|
2016-10-21 07:45:54 -04:00
|
|
|
const char *connectHost = srcdef->data.udp.connectHost;
|
|
|
|
const char *bindHost = srcdef->data.udp.bindHost;
|
|
|
|
const char *bindService = srcdef->data.udp.bindService;
|
2013-07-26 11:30:50 +08:00
|
|
|
|
|
|
|
if (connectHost == NULL)
|
|
|
|
connectHost = "";
|
|
|
|
if (bindHost == NULL)
|
|
|
|
bindHost = "";
|
|
|
|
if (bindService == NULL)
|
|
|
|
bindService = "0";
|
|
|
|
|
2019-10-22 15:26:14 +02:00
|
|
|
*buf = g_strdup_printf("udp:%s:%s@%s:%s", connectHost,
|
|
|
|
srcdef->data.udp.connectService, bindHost, bindService);
|
2013-07-26 11:30:50 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case VIR_DOMAIN_CHR_TYPE_TCP: {
|
|
|
|
const char *prefix;
|
|
|
|
|
2016-10-21 07:45:54 -04:00
|
|
|
if (srcdef->data.tcp.protocol == VIR_DOMAIN_CHR_TCP_PROTOCOL_TELNET)
|
2013-07-26 11:30:50 +08:00
|
|
|
prefix = "telnet";
|
|
|
|
else
|
|
|
|
prefix = "tcp";
|
|
|
|
|
2019-10-22 15:26:14 +02:00
|
|
|
*buf = g_strdup_printf("%s:%s:%s%s", prefix, srcdef->data.tcp.host,
|
|
|
|
srcdef->data.tcp.service,
|
|
|
|
srcdef->data.tcp.listen ? ",server,nowait" : "");
|
2013-07-26 11:30:50 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case VIR_DOMAIN_CHR_TYPE_UNIX:
|
2019-10-22 15:26:14 +02:00
|
|
|
*buf = g_strdup_printf("unix:%s%s", srcdef->data.nix.path,
|
|
|
|
srcdef->data.nix.listen ? ",server,nowait" : "");
|
2013-07-26 11:30:50 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("unsupported chardev '%s'"), type);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-05-05 16:06:32 +02:00
|
|
|
static int
|
|
|
|
libxlSetVcpuAffinities(virDomainDef *def,
|
|
|
|
libxl_ctx *ctx,
|
|
|
|
libxl_domain_build_info *b_info)
|
|
|
|
{
|
|
|
|
libxl_bitmap *vcpu_affinity_array;
|
|
|
|
unsigned int vcpuid;
|
|
|
|
unsigned int vcpu_idx = 0;
|
|
|
|
virDomainVcpuDef *vcpu;
|
|
|
|
bool has_vcpu_pin = false;
|
|
|
|
|
|
|
|
/* Get highest vcpuid with cpumask */
|
|
|
|
for (vcpuid = 0; vcpuid < b_info->max_vcpus; vcpuid++) {
|
|
|
|
vcpu = virDomainDefGetVcpu(def, vcpuid);
|
|
|
|
if (!vcpu)
|
|
|
|
continue;
|
|
|
|
if (!vcpu->cpumask)
|
|
|
|
continue;
|
|
|
|
vcpu_idx = vcpuid;
|
|
|
|
has_vcpu_pin = true;
|
|
|
|
}
|
|
|
|
/* Nothing to do */
|
|
|
|
if (!has_vcpu_pin)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
/* Adjust index */
|
|
|
|
vcpu_idx++;
|
|
|
|
|
|
|
|
b_info->num_vcpu_hard_affinity = vcpu_idx;
|
|
|
|
/* Will be released by libxl_domain_config_dispose */
|
|
|
|
b_info->vcpu_hard_affinity = g_new0(libxl_bitmap, vcpu_idx);
|
|
|
|
vcpu_affinity_array = b_info->vcpu_hard_affinity;
|
|
|
|
|
|
|
|
for (vcpuid = 0; vcpuid < vcpu_idx; vcpuid++) {
|
|
|
|
libxl_bitmap *map = &vcpu_affinity_array[vcpuid];
|
|
|
|
libxl_bitmap_init(map);
|
|
|
|
/* libxl owns the bitmap */
|
|
|
|
if (libxl_cpu_bitmap_alloc(ctx, map, 0))
|
|
|
|
return -1;
|
|
|
|
vcpu = virDomainDefGetVcpu(def, vcpuid);
|
|
|
|
/* Apply the given mask, or allow unhandled vcpus to run anywhere */
|
|
|
|
if (vcpu && vcpu->cpumask)
|
|
|
|
virBitmapToDataBuf(vcpu->cpumask, map->map, map->size);
|
|
|
|
else
|
|
|
|
libxl_bitmap_set_any(map);
|
|
|
|
}
|
|
|
|
libxl_defbool_set(&b_info->numa_placement, false);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-10 15:42:34 -07:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlMakeDomBuildInfo(virDomainDef *def,
|
|
|
|
libxlDriverConfig *cfg,
|
|
|
|
virCaps *caps,
|
2014-05-30 14:46:35 +01:00
|
|
|
libxl_domain_config *d_config)
|
2011-02-10 15:42:34 -07:00
|
|
|
{
|
2018-02-20 16:51:27 -07:00
|
|
|
virDomainClockDef clock = def->clock;
|
2018-04-12 03:03:20 +02:00
|
|
|
libxl_ctx *ctx = cfg->ctx;
|
2011-02-10 15:42:34 -07:00
|
|
|
libxl_domain_build_info *b_info = &d_config->b_info;
|
2018-11-26 20:34:37 +01:00
|
|
|
bool hvm = def->os.type == VIR_DOMAIN_OSTYPE_HVM;
|
|
|
|
bool pvh = def->os.type == VIR_DOMAIN_OSTYPE_XENPVH;
|
Convert 'int i' to 'size_t i' in src/libxl/ files
Convert the type of loop iterators named 'i', 'j', k',
'ii', 'jj', 'kk', to be 'size_t' instead of 'int' or
'unsigned int', also santizing 'ii', 'jj', 'kk' to use
the normal 'i', 'j', 'k' naming
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2013-07-08 15:09:33 +01:00
|
|
|
size_t i;
|
2016-09-23 14:11:50 +02:00
|
|
|
size_t nusbdevice = 0;
|
2011-02-10 15:42:34 -07:00
|
|
|
|
2018-11-26 20:34:37 +01:00
|
|
|
if (hvm) {
|
2012-11-26 09:28:56 -07:00
|
|
|
libxl_domain_build_info_init_type(b_info, LIBXL_DOMAIN_TYPE_HVM);
|
2018-11-26 20:34:37 +01:00
|
|
|
} else if (pvh) {
|
2020-09-01 13:27:44 +02:00
|
|
|
#ifdef WITH_XEN_PVH
|
2018-11-26 20:34:37 +01:00
|
|
|
libxl_domain_build_info_init_type(b_info, LIBXL_DOMAIN_TYPE_PVH);
|
|
|
|
#else
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("PVH guest os type not supported"));
|
|
|
|
return -1;
|
|
|
|
#endif
|
|
|
|
} else {
|
2012-11-26 09:28:56 -07:00
|
|
|
libxl_domain_build_info_init_type(b_info, LIBXL_DOMAIN_TYPE_PV);
|
2018-11-26 20:34:37 +01:00
|
|
|
}
|
2013-07-24 13:43:43 +02:00
|
|
|
|
2015-10-19 19:21:24 +02:00
|
|
|
b_info->max_vcpus = virDomainDefGetVcpusMax(def);
|
|
|
|
if (libxl_cpu_bitmap_alloc(ctx, &b_info->avail_vcpus, b_info->max_vcpus))
|
2014-10-10 15:39:46 -06:00
|
|
|
return -1;
|
2013-07-24 13:43:43 +02:00
|
|
|
libxl_bitmap_set_none(&b_info->avail_vcpus);
|
2015-10-22 14:59:03 +02:00
|
|
|
for (i = 0; i < virDomainDefGetVcpus(def); i++)
|
2013-07-24 13:43:43 +02:00
|
|
|
libxl_bitmap_set((&b_info->avail_vcpus), i);
|
|
|
|
|
2021-05-05 16:06:32 +02:00
|
|
|
if (libxlSetVcpuAffinities(def, ctx, b_info))
|
|
|
|
return -1;
|
|
|
|
|
2018-02-20 16:51:27 -07:00
|
|
|
switch ((virDomainClockOffsetType) clock.offset) {
|
|
|
|
case VIR_DOMAIN_CLOCK_OFFSET_VARIABLE:
|
|
|
|
if (clock.data.variable.basis == VIR_DOMAIN_CLOCK_BASIS_LOCALTIME)
|
|
|
|
libxl_defbool_set(&b_info->localtime, true);
|
|
|
|
b_info->rtc_timeoffset = clock.data.variable.adjustment;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_CLOCK_OFFSET_LOCALTIME:
|
|
|
|
libxl_defbool_set(&b_info->localtime, true);
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* Nothing to do since UTC is the default in libxl */
|
|
|
|
case VIR_DOMAIN_CLOCK_OFFSET_UTC:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_CLOCK_OFFSET_TIMEZONE:
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("unsupported clock offset '%s'"),
|
|
|
|
virDomainClockOffsetTypeToString(clock.offset));
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_CLOCK_OFFSET_LAST:
|
|
|
|
default:
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("unexpected clock offset '%d'"), clock.offset);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < clock.ntimers; i++) {
|
|
|
|
switch ((virDomainTimerNameType) clock.timers[i]->name) {
|
2017-01-16 15:37:40 -07:00
|
|
|
case VIR_DOMAIN_TIMER_NAME_TSC:
|
2018-02-20 16:51:27 -07:00
|
|
|
switch (clock.timers[i]->mode) {
|
2011-02-10 15:42:34 -07:00
|
|
|
case VIR_DOMAIN_TIMER_MODE_NATIVE:
|
2017-01-19 16:21:34 -07:00
|
|
|
b_info->tsc_mode = LIBXL_TSC_MODE_NATIVE;
|
2011-02-10 15:42:34 -07:00
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_TIMER_MODE_PARAVIRT:
|
2017-01-19 16:21:34 -07:00
|
|
|
b_info->tsc_mode = LIBXL_TSC_MODE_NATIVE_PARAVIRT;
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_TIMER_MODE_EMULATE:
|
|
|
|
b_info->tsc_mode = LIBXL_TSC_MODE_ALWAYS_EMULATE;
|
2011-02-10 15:42:34 -07:00
|
|
|
break;
|
|
|
|
default:
|
2017-01-19 16:21:34 -07:00
|
|
|
b_info->tsc_mode = LIBXL_TSC_MODE_DEFAULT;
|
2017-01-16 15:37:40 -07:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_TIMER_NAME_HPET:
|
|
|
|
if (!hvm) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("unsupported timer type (name) '%s'"),
|
2018-02-20 16:51:27 -07:00
|
|
|
virDomainTimerNameTypeToString(clock.timers[i]->name));
|
2017-01-16 15:37:40 -07:00
|
|
|
return -1;
|
|
|
|
}
|
2018-02-20 16:51:27 -07:00
|
|
|
if (clock.timers[i]->present == 1)
|
2017-01-16 15:37:40 -07:00
|
|
|
libxl_defbool_set(&b_info->u.hvm.hpet, 1);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_TIMER_NAME_PLATFORM:
|
|
|
|
case VIR_DOMAIN_TIMER_NAME_KVMCLOCK:
|
|
|
|
case VIR_DOMAIN_TIMER_NAME_HYPERVCLOCK:
|
|
|
|
case VIR_DOMAIN_TIMER_NAME_RTC:
|
|
|
|
case VIR_DOMAIN_TIMER_NAME_PIT:
|
2020-02-06 16:54:45 +01:00
|
|
|
case VIR_DOMAIN_TIMER_NAME_ARMVTIMER:
|
2017-01-16 15:37:40 -07:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("unsupported timer type (name) '%s'"),
|
2018-02-20 16:51:27 -07:00
|
|
|
virDomainTimerNameTypeToString(clock.timers[i]->name));
|
2017-01-16 15:37:40 -07:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_TIMER_NAME_LAST:
|
|
|
|
break;
|
2011-02-10 15:42:34 -07:00
|
|
|
}
|
|
|
|
}
|
2017-01-16 15:37:40 -07:00
|
|
|
|
2018-02-22 11:52:56 -07:00
|
|
|
if (def->cputune.sharesSpecified)
|
|
|
|
b_info->sched_params.weight = def->cputune.shares;
|
|
|
|
|
2018-02-21 18:48:38 -07:00
|
|
|
/* Xen requires the memory sizes to be rounded to 1MiB increments */
|
|
|
|
virDomainDefSetMemoryTotal(def,
|
|
|
|
VIR_ROUND_UP(virDomainDefGetMemoryInitial(def), 1024));
|
|
|
|
def->mem.cur_balloon = VIR_ROUND_UP(def->mem.cur_balloon, 1024);
|
2015-02-17 18:01:09 +01:00
|
|
|
b_info->max_memkb = virDomainDefGetMemoryInitial(def);
|
2011-02-10 15:42:34 -07:00
|
|
|
b_info->target_memkb = def->mem.cur_balloon;
|
2019-03-07 15:16:09 -07:00
|
|
|
|
|
|
|
for (i = 0; i < def->ncontrollers; i++) {
|
2020-04-07 17:15:04 -06:00
|
|
|
if (def->controllers[i]->type == VIR_DOMAIN_CONTROLLER_TYPE_XENBUS) {
|
|
|
|
if (def->controllers[i]->opts.xenbusopts.maxEventChannels > 0)
|
|
|
|
b_info->event_channels = def->controllers[i]->opts.xenbusopts.maxEventChannels;
|
|
|
|
|
|
|
|
#ifdef LIBXL_HAVE_BUILDINFO_GRANT_LIMITS
|
|
|
|
if (def->controllers[i]->opts.xenbusopts.maxGrantFrames > 0)
|
|
|
|
b_info->max_grant_frames = def->controllers[i]->opts.xenbusopts.maxGrantFrames;
|
2019-03-07 15:16:09 -07:00
|
|
|
#endif
|
2020-04-07 17:15:04 -06:00
|
|
|
}
|
|
|
|
}
|
2019-03-07 15:16:09 -07:00
|
|
|
|
2018-11-26 20:34:37 +01:00
|
|
|
if (hvm || pvh) {
|
2017-04-24 15:06:59 +02:00
|
|
|
if (caps &&
|
|
|
|
def->cpu && def->cpu->mode == (VIR_CPU_MODE_HOST_PASSTHROUGH)) {
|
|
|
|
bool hasHwVirt = false;
|
|
|
|
bool svm = false, vmx = false;
|
2018-04-12 03:03:24 +02:00
|
|
|
char xlCPU[32];
|
2017-04-24 15:06:59 +02:00
|
|
|
|
2018-04-12 03:03:22 +02:00
|
|
|
/* enable nested HVM only if global nested_hvm option enable it and
|
2020-01-24 21:30:04 +01:00
|
|
|
* host support it */
|
2017-04-24 15:06:59 +02:00
|
|
|
if (ARCH_IS_X86(def->os.arch)) {
|
|
|
|
vmx = virCPUCheckFeature(caps->host.arch, caps->host.cpu, "vmx");
|
|
|
|
svm = virCPUCheckFeature(caps->host.arch, caps->host.cpu, "svm");
|
2018-04-12 03:03:22 +02:00
|
|
|
hasHwVirt = cfg->nested_hvm && (vmx | svm);
|
2017-04-24 15:06:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (def->cpu->nfeatures) {
|
|
|
|
for (i = 0; i < def->cpu->nfeatures; i++) {
|
|
|
|
|
|
|
|
switch (def->cpu->features[i].policy) {
|
|
|
|
|
|
|
|
case VIR_CPU_FEATURE_DISABLE:
|
|
|
|
case VIR_CPU_FEATURE_FORBID:
|
|
|
|
if ((vmx && STREQ(def->cpu->features[i].name, "vmx")) ||
|
2018-04-12 03:03:24 +02:00
|
|
|
(svm && STREQ(def->cpu->features[i].name, "svm"))) {
|
2017-04-24 15:06:59 +02:00
|
|
|
hasHwVirt = false;
|
2018-04-12 03:03:24 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-11-13 14:53:42 +01:00
|
|
|
g_snprintf(xlCPU,
|
|
|
|
sizeof(xlCPU),
|
|
|
|
"%s=0",
|
|
|
|
xenTranslateCPUFeature(
|
|
|
|
def->cpu->features[i].name,
|
|
|
|
false));
|
2018-04-12 03:03:24 +02:00
|
|
|
if (libxl_cpuid_parse_config(&b_info->cpuid, xlCPU)) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("unsupported cpu feature '%s'"),
|
|
|
|
def->cpu->features[i].name);
|
|
|
|
return -1;
|
|
|
|
}
|
2017-04-24 15:06:59 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_CPU_FEATURE_FORCE:
|
|
|
|
case VIR_CPU_FEATURE_REQUIRE:
|
2018-04-12 03:03:22 +02:00
|
|
|
if ((vmx && STREQ(def->cpu->features[i].name, "vmx")) ||
|
2018-04-12 03:03:24 +02:00
|
|
|
(svm && STREQ(def->cpu->features[i].name, "svm"))) {
|
2018-04-12 03:03:22 +02:00
|
|
|
hasHwVirt = true;
|
2018-04-12 03:03:24 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-11-13 14:53:42 +01:00
|
|
|
g_snprintf(xlCPU,
|
|
|
|
sizeof(xlCPU),
|
|
|
|
"%s=1",
|
|
|
|
xenTranslateCPUFeature(
|
|
|
|
def->cpu->features[i].name, false));
|
2018-04-12 03:03:24 +02:00
|
|
|
if (libxl_cpuid_parse_config(&b_info->cpuid, xlCPU)) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("unsupported cpu feature '%s'"),
|
|
|
|
def->cpu->features[i].name);
|
|
|
|
return -1;
|
|
|
|
}
|
2018-04-12 03:03:22 +02:00
|
|
|
break;
|
2017-04-24 15:06:59 +02:00
|
|
|
case VIR_CPU_FEATURE_OPTIONAL:
|
|
|
|
case VIR_CPU_FEATURE_LAST:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-09-19 02:50:32 +02:00
|
|
|
#ifdef LIBXL_HAVE_BUILDINFO_NESTED_HVM
|
|
|
|
libxl_defbool_set(&b_info->nested_hvm, hasHwVirt);
|
|
|
|
#else
|
|
|
|
if (hvm) {
|
|
|
|
libxl_defbool_set(&b_info->u.hvm.nested_hvm, hasHwVirt);
|
|
|
|
} else {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("unsupported nested HVM setting for %s machine on this Xen version"),
|
|
|
|
def->os.machine);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
#endif
|
2017-04-24 15:06:59 +02:00
|
|
|
}
|
|
|
|
|
2018-04-12 03:03:21 +02:00
|
|
|
if (def->cpu && def->cpu->mode == VIR_CPU_MODE_CUSTOM) {
|
|
|
|
VIR_WARN("Ignoring CPU with mode=custom, update your config to "
|
|
|
|
"mode=host-passthrough to avoid risk of changed guest "
|
|
|
|
"semantics when mode=custom is supported in the future");
|
|
|
|
}
|
2018-11-26 20:34:36 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (hvm) {
|
|
|
|
char bootorder[VIR_DOMAIN_BOOT_LAST + 1];
|
|
|
|
|
|
|
|
libxl_defbool_set(&b_info->u.hvm.pae,
|
|
|
|
def->features[VIR_DOMAIN_FEATURE_PAE] ==
|
|
|
|
VIR_TRISTATE_SWITCH_ON);
|
2020-09-10 06:18:01 +02:00
|
|
|
#ifdef LIBXL_HAVE_BUILDINFO_APIC
|
|
|
|
libxl_defbool_set(&b_info->apic,
|
|
|
|
def->features[VIR_DOMAIN_FEATURE_APIC] ==
|
|
|
|
VIR_TRISTATE_SWITCH_ON);
|
|
|
|
/*
|
|
|
|
* Strictly speaking b_info->acpi was introduced earlier (Xen 4.8), but
|
|
|
|
* there is no separate #define in libxl.h.
|
|
|
|
*/
|
|
|
|
libxl_defbool_set(&b_info->acpi,
|
|
|
|
def->features[VIR_DOMAIN_FEATURE_ACPI] ==
|
|
|
|
VIR_TRISTATE_SWITCH_ON);
|
|
|
|
#else
|
2018-11-26 20:34:36 +01:00
|
|
|
libxl_defbool_set(&b_info->u.hvm.apic,
|
|
|
|
def->features[VIR_DOMAIN_FEATURE_APIC] ==
|
|
|
|
VIR_TRISTATE_SWITCH_ON);
|
|
|
|
libxl_defbool_set(&b_info->u.hvm.acpi,
|
|
|
|
def->features[VIR_DOMAIN_FEATURE_ACPI] ==
|
|
|
|
VIR_TRISTATE_SWITCH_ON);
|
2020-09-10 06:18:01 +02:00
|
|
|
#endif
|
2018-04-12 03:03:21 +02:00
|
|
|
|
2019-09-15 21:43:22 +02:00
|
|
|
/* copy SLIC table path to acpi_firmware */
|
2019-10-20 13:49:46 +02:00
|
|
|
if (def->os.slic_table)
|
|
|
|
b_info->u.hvm.acpi_firmware = g_strdup(def->os.slic_table);
|
2019-09-15 21:43:22 +02:00
|
|
|
|
2015-05-04 13:50:10 -06:00
|
|
|
if (def->nsounds > 0) {
|
|
|
|
/*
|
|
|
|
* Use first sound device. man xl.cfg(5) describes soundhw as
|
|
|
|
* a single device. From the man page: soundhw=DEVICE
|
|
|
|
*/
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainSoundDef *snd = def->sounds[0];
|
2015-05-04 13:50:10 -06:00
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
b_info->u.hvm.soundhw = g_strdup(virDomainSoundModelTypeToString(snd->model));
|
2015-05-04 13:50:10 -06:00
|
|
|
}
|
|
|
|
|
2013-05-21 15:21:16 +08:00
|
|
|
for (i = 0; i < def->os.nBootDevs; i++) {
|
2012-12-12 14:28:24 -07:00
|
|
|
switch (def->os.bootDevs[i]) {
|
|
|
|
case VIR_DOMAIN_BOOT_FLOPPY:
|
|
|
|
bootorder[i] = 'a';
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
case VIR_DOMAIN_BOOT_DISK:
|
|
|
|
bootorder[i] = 'c';
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_BOOT_CDROM:
|
|
|
|
bootorder[i] = 'd';
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_BOOT_NET:
|
|
|
|
bootorder[i] = 'n';
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (def->os.nBootDevs == 0) {
|
|
|
|
bootorder[0] = 'c';
|
|
|
|
bootorder[1] = '\0';
|
2014-09-03 13:04:59 -06:00
|
|
|
} else {
|
2012-12-12 14:28:24 -07:00
|
|
|
bootorder[def->os.nBootDevs] = '\0';
|
|
|
|
}
|
2019-10-20 13:49:46 +02:00
|
|
|
b_info->u.hvm.boot = g_strdup(bootorder);
|
2011-05-10 13:24:15 +02:00
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
b_info->cmdline = g_strdup(def->os.cmdline);
|
|
|
|
b_info->kernel = g_strdup(def->os.kernel);
|
|
|
|
b_info->ramdisk = g_strdup(def->os.initrd);
|
2015-03-19 22:16:11 -06:00
|
|
|
|
2016-04-20 15:14:35 -06:00
|
|
|
/*
|
|
|
|
* Currently libxl only allows specifying the type of BIOS.
|
2021-06-03 14:04:30 -06:00
|
|
|
* If automatic firmware selection is enabled or the loader
|
|
|
|
* type is PFLASH, we assume OVMF and set libxl_bios_type
|
2016-04-20 15:14:35 -06:00
|
|
|
* to LIBXL_BIOS_TYPE_OVMF. The path to the OVMF firmware is
|
|
|
|
* configured when building Xen using '--with-system-ovmf='. If
|
|
|
|
* not specified, LIBXL_FIRMWARE_DIR/ovmf.bin is used. In the
|
|
|
|
* future, Xen will support a user-specified firmware path. See
|
2020-08-26 00:44:00 +02:00
|
|
|
* https://lists.xenproject.org/archives/html/xen-devel/2016-03/msg01628.html
|
2016-04-20 15:14:35 -06:00
|
|
|
*/
|
2021-06-03 14:04:30 -06:00
|
|
|
if (def->os.firmware == VIR_DOMAIN_OS_DEF_FIRMWARE_EFI) {
|
|
|
|
if (def->os.loader == NULL)
|
|
|
|
def->os.loader = g_new0(virDomainLoaderDef, 1);
|
|
|
|
if (def->os.loader->path == NULL)
|
|
|
|
def->os.loader->path = g_strdup(cfg->firmwares[0]->name);
|
|
|
|
if (def->os.loader->type == VIR_DOMAIN_LOADER_TYPE_NONE)
|
|
|
|
def->os.loader->type = VIR_DOMAIN_LOADER_TYPE_PFLASH;
|
|
|
|
if (def->os.loader->readonly == VIR_TRISTATE_BOOL_ABSENT)
|
|
|
|
def->os.loader->readonly = VIR_TRISTATE_BOOL_YES;
|
2016-04-20 15:14:35 -06:00
|
|
|
b_info->u.hvm.bios = LIBXL_BIOS_TYPE_OVMF;
|
2021-06-03 14:04:30 -06:00
|
|
|
def->os.firmware = VIR_DOMAIN_OS_DEF_FIRMWARE_NONE;
|
|
|
|
} else if (virDomainDefHasOldStyleUEFI(def)) {
|
|
|
|
b_info->u.hvm.bios = LIBXL_BIOS_TYPE_OVMF;
|
|
|
|
}
|
2016-04-20 15:14:35 -06:00
|
|
|
|
2014-09-19 13:03:18 -06:00
|
|
|
if (def->emulator) {
|
|
|
|
if (!virFileExists(def->emulator)) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("emulator '%s' not found"),
|
|
|
|
def->emulator);
|
2014-10-10 15:39:46 -06:00
|
|
|
return -1;
|
2014-09-19 13:03:18 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!virFileIsExecutable(def->emulator)) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("emulator '%s' is not executable"),
|
|
|
|
def->emulator);
|
2014-10-10 15:39:46 -06:00
|
|
|
return -1;
|
2014-09-19 13:03:18 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
VIR_FREE(b_info->device_model);
|
2019-10-20 13:49:46 +02:00
|
|
|
b_info->device_model = g_strdup(def->emulator);
|
2014-09-19 13:03:18 -06:00
|
|
|
|
|
|
|
b_info->device_model_version = libxlDomainGetEmulatorType(def);
|
|
|
|
}
|
|
|
|
|
2013-07-26 11:30:50 +08:00
|
|
|
if (def->nserials) {
|
2016-08-18 10:20:48 +08:00
|
|
|
if (def->nserials == 1) {
|
|
|
|
if (libxlMakeChrdevStr(def->serials[0], &b_info->u.hvm.serial) <
|
|
|
|
0)
|
|
|
|
return -1;
|
|
|
|
} else {
|
2020-09-23 20:43:09 +02:00
|
|
|
b_info->u.hvm.serial_list = *g_new0(libxl_string_list, def->nserials + 1);
|
2016-08-18 10:20:48 +08:00
|
|
|
for (i = 0; i < def->nserials; i++) {
|
|
|
|
if (libxlMakeChrdevStr(def->serials[i],
|
|
|
|
&b_info->u.hvm.serial_list[i]) < 0)
|
|
|
|
{
|
|
|
|
libxl_string_list_dispose(&b_info->u.hvm.serial_list);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
b_info->u.hvm.serial_list[i] = NULL;
|
2013-07-26 11:30:50 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (def->nparallels) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
"%s",
|
|
|
|
_("Parallel devices are not supported by libxl"));
|
2014-10-10 15:39:46 -06:00
|
|
|
return -1;
|
2013-07-26 11:30:50 +08:00
|
|
|
}
|
|
|
|
|
2015-02-10 03:36:47 +01:00
|
|
|
/* Disable VNC and SDL until explicitly enabled */
|
|
|
|
libxl_defbool_set(&b_info->u.hvm.vnc.enable, 0);
|
|
|
|
libxl_defbool_set(&b_info->u.hvm.sdl.enable, 0);
|
|
|
|
|
2016-09-23 14:11:50 +02:00
|
|
|
for (i = 0; i < def->ninputs; i++) {
|
|
|
|
char **usbdevice;
|
|
|
|
|
|
|
|
if (def->inputs[i]->bus != VIR_DOMAIN_INPUT_BUS_USB)
|
|
|
|
continue;
|
|
|
|
|
2021-03-20 00:37:03 +01:00
|
|
|
VIR_EXPAND_N(b_info->u.hvm.usbdevice_list, nusbdevice, 1);
|
2017-01-10 17:09:03 +01:00
|
|
|
usbdevice = &b_info->u.hvm.usbdevice_list[nusbdevice - 1];
|
2016-09-23 14:11:50 +02:00
|
|
|
switch (def->inputs[i]->type) {
|
|
|
|
case VIR_DOMAIN_INPUT_TYPE_MOUSE:
|
|
|
|
VIR_FREE(*usbdevice);
|
2019-10-20 13:49:46 +02:00
|
|
|
*usbdevice = g_strdup("mouse");
|
2016-09-23 14:11:50 +02:00
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_INPUT_TYPE_TABLET:
|
|
|
|
VIR_FREE(*usbdevice);
|
2019-10-20 13:49:46 +02:00
|
|
|
*usbdevice = g_strdup("tablet");
|
2016-09-23 14:11:50 +02:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("Unknown input device type"));
|
|
|
|
return -1;
|
2015-03-06 15:36:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-23 14:11:50 +02:00
|
|
|
/* NULL-terminate usbdevice_list */
|
2021-03-20 00:37:03 +01:00
|
|
|
if (nusbdevice > 0)
|
|
|
|
VIR_EXPAND_N(b_info->u.hvm.usbdevice_list, nusbdevice, 1);
|
2018-11-26 20:34:37 +01:00
|
|
|
} else if (pvh) {
|
2019-10-20 13:49:46 +02:00
|
|
|
b_info->cmdline = g_strdup(def->os.cmdline);
|
|
|
|
b_info->kernel = g_strdup(def->os.kernel);
|
|
|
|
b_info->ramdisk = g_strdup(def->os.initrd);
|
2018-11-26 20:34:37 +01:00
|
|
|
#ifdef LIBXL_HAVE_BUILDINFO_BOOTLOADER
|
2019-10-20 13:49:46 +02:00
|
|
|
b_info->bootloader = g_strdup(def->os.bootloader);
|
2018-11-26 20:34:37 +01:00
|
|
|
if (def->os.bootloaderArgs) {
|
|
|
|
if (!(b_info->bootloader_args =
|
2021-02-05 18:35:07 +01:00
|
|
|
g_strsplit(def->os.bootloaderArgs, " \t\n", 0)))
|
2018-11-26 20:34:37 +01:00
|
|
|
return -1;
|
|
|
|
}
|
2016-09-23 14:11:50 +02:00
|
|
|
#endif
|
2011-02-10 15:42:34 -07:00
|
|
|
} else {
|
2013-05-17 09:30:08 -06:00
|
|
|
/*
|
|
|
|
* For compatibility with the legacy xen toolstack, default to pygrub
|
|
|
|
* if bootloader is not specified AND direct kernel boot is not specified.
|
|
|
|
*/
|
|
|
|
if (def->os.bootloader) {
|
2019-10-20 13:49:46 +02:00
|
|
|
b_info->u.pv.bootloader = g_strdup(def->os.bootloader);
|
2013-05-17 09:30:08 -06:00
|
|
|
} else if (def->os.kernel == NULL) {
|
2019-10-20 13:49:46 +02:00
|
|
|
b_info->u.pv.bootloader = g_strdup(LIBXL_BOOTLOADER_PATH);
|
2013-05-17 09:30:08 -06:00
|
|
|
}
|
2011-02-10 15:42:34 -07:00
|
|
|
if (def->os.bootloaderArgs) {
|
2012-11-26 09:28:56 -07:00
|
|
|
if (!(b_info->u.pv.bootloader_args =
|
2021-02-05 18:35:07 +01:00
|
|
|
g_strsplit(def->os.bootloaderArgs, " \t\n", 0)))
|
2014-10-10 15:39:46 -06:00
|
|
|
return -1;
|
2011-02-10 15:42:34 -07:00
|
|
|
}
|
2019-10-20 13:49:46 +02:00
|
|
|
b_info->u.pv.cmdline = g_strdup(def->os.cmdline);
|
2011-02-10 15:42:34 -07:00
|
|
|
if (def->os.kernel) {
|
2019-10-20 13:49:46 +02:00
|
|
|
/* libxl_init_build_info() sets kernel.path = g_strdup("hvmloader") */
|
2012-11-26 09:28:56 -07:00
|
|
|
VIR_FREE(b_info->u.pv.kernel);
|
2019-10-20 13:49:46 +02:00
|
|
|
b_info->u.pv.kernel = g_strdup(def->os.kernel);
|
2011-02-10 15:42:34 -07:00
|
|
|
}
|
2019-10-20 13:49:46 +02:00
|
|
|
b_info->u.pv.ramdisk = g_strdup(def->os.initrd);
|
2020-04-14 04:37:05 +02:00
|
|
|
|
|
|
|
if (def->features[VIR_DOMAIN_FEATURE_XEN] == VIR_TRISTATE_SWITCH_ON) {
|
|
|
|
switch ((virTristateSwitch) def->xen_features[VIR_DOMAIN_XEN_E820_HOST]) {
|
|
|
|
case VIR_TRISTATE_SWITCH_ON:
|
|
|
|
libxl_defbool_set(&b_info->u.pv.e820_host, true);
|
|
|
|
break;
|
|
|
|
case VIR_TRISTATE_SWITCH_OFF:
|
|
|
|
libxl_defbool_set(&b_info->u.pv.e820_host, false);
|
|
|
|
break;
|
|
|
|
case VIR_TRISTATE_SWITCH_ABSENT:
|
|
|
|
case VIR_TRISTATE_SWITCH_LAST:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2011-02-10 15:42:34 -07:00
|
|
|
}
|
|
|
|
|
2018-04-09 18:15:17 -06:00
|
|
|
/* only the 'xen' balloon device model is supported */
|
|
|
|
if (def->memballoon) {
|
2021-04-27 17:04:34 +02:00
|
|
|
switch (def->memballoon->model) {
|
2018-04-09 18:15:17 -06:00
|
|
|
case VIR_DOMAIN_MEMBALLOON_MODEL_XEN:
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_MEMBALLOON_MODEL_VIRTIO:
|
2019-01-17 12:52:39 -05:00
|
|
|
case VIR_DOMAIN_MEMBALLOON_MODEL_VIRTIO_TRANSITIONAL:
|
|
|
|
case VIR_DOMAIN_MEMBALLOON_MODEL_VIRTIO_NON_TRANSITIONAL:
|
2018-04-09 18:15:17 -06:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("unsupported balloon device model '%s'"),
|
2021-04-27 17:04:34 +02:00
|
|
|
virDomainMemballoonModelTypeToString(def->memballoon->model));
|
2018-04-09 18:15:17 -06:00
|
|
|
return -1;
|
|
|
|
case VIR_DOMAIN_MEMBALLOON_MODEL_NONE:
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
"%s",
|
|
|
|
_("balloon device cannot be disabled"));
|
|
|
|
return -1;
|
|
|
|
case VIR_DOMAIN_MEMBALLOON_MODEL_LAST:
|
|
|
|
default:
|
2021-04-27 17:04:34 +02:00
|
|
|
virReportEnumRangeError(virDomainMemballoonModel, def->memballoon->model);
|
2018-04-09 18:15:17 -06:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-19 02:50:31 +02:00
|
|
|
/* Allow libxl to calculate shadow memory requirements */
|
|
|
|
b_info->shadow_memkb =
|
|
|
|
libxl_get_required_shadow_memory(b_info->max_memkb,
|
|
|
|
b_info->max_vcpus);
|
|
|
|
|
2020-07-30 13:25:20 -06:00
|
|
|
if (def->namespaceData) {
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlDomainXmlNsDef *nsdata = def->namespaceData;
|
2020-07-30 13:25:20 -06:00
|
|
|
|
|
|
|
if (nsdata->num_args > 0)
|
|
|
|
b_info->extra = g_strdupv(nsdata->args);
|
|
|
|
}
|
|
|
|
|
2011-02-10 15:42:34 -07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-11-02 16:47:22 +01:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlMakeVnumaList(virDomainDef *def,
|
2017-11-02 16:47:22 +01:00
|
|
|
libxl_ctx *ctx,
|
|
|
|
libxl_domain_config *d_config)
|
|
|
|
{
|
|
|
|
int ret = -1;
|
|
|
|
size_t i, j;
|
|
|
|
size_t nr_nodes;
|
|
|
|
size_t num_vnuma;
|
|
|
|
bool simulate = false;
|
2021-03-11 08:16:13 +01:00
|
|
|
virBitmap *bitmap = NULL;
|
|
|
|
virDomainNuma *numa = def->numa;
|
2017-11-02 16:47:22 +01:00
|
|
|
libxl_domain_build_info *b_info = &d_config->b_info;
|
|
|
|
libxl_physinfo physinfo;
|
|
|
|
libxl_vnode_info *vnuma_nodes = NULL;
|
|
|
|
|
|
|
|
if (!numa)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
num_vnuma = virDomainNumaGetNodeCount(numa);
|
|
|
|
if (!num_vnuma)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
libxl_physinfo_init(&physinfo);
|
|
|
|
if (libxl_get_physinfo(ctx, &physinfo) < 0) {
|
|
|
|
libxl_physinfo_dispose(&physinfo);
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("libxl_get_physinfo_info failed"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
nr_nodes = physinfo.nr_nodes;
|
|
|
|
libxl_physinfo_dispose(&physinfo);
|
|
|
|
|
|
|
|
if (num_vnuma > nr_nodes) {
|
|
|
|
VIR_WARN("Number of configured numa cells %zu exceeds available physical nodes %zu. All cells will use physical node 0",
|
|
|
|
num_vnuma, nr_nodes);
|
|
|
|
simulate = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* allocate the vnuma_nodes for assignment under b_info.
|
|
|
|
*/
|
2020-09-23 20:43:09 +02:00
|
|
|
vnuma_nodes = g_new0(libxl_vnode_info, num_vnuma);
|
2017-11-02 16:47:22 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
* parse the vnuma vnodes data.
|
|
|
|
*/
|
|
|
|
for (i = 0; i < num_vnuma; i++) {
|
|
|
|
int cpu;
|
|
|
|
libxl_bitmap vcpu_bitmap;
|
|
|
|
libxl_vnode_info *p = &vnuma_nodes[i];
|
|
|
|
|
|
|
|
libxl_vnode_info_init(p);
|
|
|
|
|
|
|
|
/* pnode */
|
|
|
|
p->pnode = simulate ? 0 : i;
|
|
|
|
|
|
|
|
/* memory size */
|
|
|
|
p->memkb = virDomainNumaGetNodeMemorySize(numa, i);
|
|
|
|
|
|
|
|
/* vcpus */
|
|
|
|
bitmap = virDomainNumaGetNodeCpumask(numa, i);
|
|
|
|
if (bitmap == NULL) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("vnuma sibling %zu missing vcpus set"), i);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((cpu = virBitmapNextSetBit(bitmap, -1)) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
libxl_bitmap_init(&vcpu_bitmap);
|
2021-02-24 11:32:10 +01:00
|
|
|
if (libxl_cpu_bitmap_alloc(ctx, &vcpu_bitmap, b_info->max_vcpus))
|
|
|
|
abort();
|
2017-11-02 16:47:22 +01:00
|
|
|
|
|
|
|
do {
|
|
|
|
libxl_bitmap_set(&vcpu_bitmap, cpu);
|
|
|
|
} while ((cpu = virBitmapNextSetBit(bitmap, cpu)) >= 0);
|
|
|
|
|
|
|
|
libxl_bitmap_copy_alloc(ctx, &p->vcpus, &vcpu_bitmap);
|
|
|
|
libxl_bitmap_dispose(&vcpu_bitmap);
|
|
|
|
|
|
|
|
/* vdistances */
|
2020-09-23 20:43:09 +02:00
|
|
|
p->distances = g_new0(uint32_t, num_vnuma);
|
2017-11-02 16:47:22 +01:00
|
|
|
p->num_distances = num_vnuma;
|
|
|
|
|
|
|
|
for (j = 0; j < num_vnuma; j++)
|
|
|
|
p->distances[j] = virDomainNumaGetNodeDistance(numa, i, j);
|
|
|
|
}
|
|
|
|
|
|
|
|
b_info->vnuma_nodes = vnuma_nodes;
|
|
|
|
b_info->num_vnuma_nodes = num_vnuma;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
if (ret) {
|
|
|
|
for (i = 0; i < num_vnuma; i++) {
|
|
|
|
libxl_vnode_info *p = &vnuma_nodes[i];
|
|
|
|
|
|
|
|
VIR_FREE(p->distances);
|
|
|
|
}
|
|
|
|
VIR_FREE(vnuma_nodes);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2014-07-07 17:05:17 +02:00
|
|
|
static int
|
|
|
|
libxlDiskSetDiscard(libxl_device_disk *x_disk, int discard)
|
|
|
|
{
|
|
|
|
if (!x_disk->readwrite)
|
|
|
|
return 0;
|
|
|
|
#if defined(LIBXL_HAVE_LIBXL_DEVICE_DISK_DISCARD_ENABLE)
|
2014-07-11 14:43:47 +01:00
|
|
|
switch ((virDomainDiskDiscard)discard) {
|
2014-07-07 17:05:17 +02:00
|
|
|
case VIR_DOMAIN_DISK_DISCARD_DEFAULT:
|
|
|
|
case VIR_DOMAIN_DISK_DISCARD_LAST:
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_DISK_DISCARD_UNMAP:
|
|
|
|
libxl_defbool_set(&x_disk->discard_enable, true);
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_DISK_DISCARD_IGNORE:
|
|
|
|
libxl_defbool_set(&x_disk->discard_enable, false);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
#else
|
|
|
|
if (discard == VIR_DOMAIN_DISK_DISCARD_DEFAULT)
|
|
|
|
return 0;
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("This version of libxenlight does not support "
|
|
|
|
"disk 'discard' option passing"));
|
|
|
|
return -1;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2016-02-10 20:53:36 -07:00
|
|
|
static char *
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlMakeNetworkDiskSrcStr(virStorageSource *src,
|
2016-02-10 20:53:36 -07:00
|
|
|
const char *username,
|
|
|
|
const char *secret)
|
|
|
|
{
|
2020-07-02 18:21:30 -04:00
|
|
|
g_auto(virBuffer) buf = VIR_BUFFER_INITIALIZER;
|
2016-02-10 20:53:36 -07:00
|
|
|
size_t i;
|
|
|
|
|
|
|
|
switch ((virStorageNetProtocol) src->protocol) {
|
|
|
|
case VIR_STORAGE_NET_PROTOCOL_NBD:
|
|
|
|
case VIR_STORAGE_NET_PROTOCOL_HTTP:
|
|
|
|
case VIR_STORAGE_NET_PROTOCOL_HTTPS:
|
|
|
|
case VIR_STORAGE_NET_PROTOCOL_FTP:
|
|
|
|
case VIR_STORAGE_NET_PROTOCOL_FTPS:
|
|
|
|
case VIR_STORAGE_NET_PROTOCOL_TFTP:
|
|
|
|
case VIR_STORAGE_NET_PROTOCOL_ISCSI:
|
|
|
|
case VIR_STORAGE_NET_PROTOCOL_GLUSTER:
|
|
|
|
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
2014-09-09 17:56:04 +02:00
|
|
|
case VIR_STORAGE_NET_PROTOCOL_SSH:
|
2017-08-29 07:21:51 -04:00
|
|
|
case VIR_STORAGE_NET_PROTOCOL_VXHS:
|
2021-01-06 15:32:26 -06:00
|
|
|
case VIR_STORAGE_NET_PROTOCOL_NFS:
|
2016-02-10 20:53:36 -07:00
|
|
|
case VIR_STORAGE_NET_PROTOCOL_LAST:
|
|
|
|
case VIR_STORAGE_NET_PROTOCOL_NONE:
|
|
|
|
virReportError(VIR_ERR_NO_SUPPORT,
|
|
|
|
_("Unsupported network block protocol '%s'"),
|
|
|
|
virStorageNetProtocolTypeToString(src->protocol));
|
2020-07-02 23:19:45 -04:00
|
|
|
return NULL;
|
2016-02-10 20:53:36 -07:00
|
|
|
|
|
|
|
case VIR_STORAGE_NET_PROTOCOL_RBD:
|
|
|
|
if (strchr(src->path, ':')) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("':' not allowed in RBD source volume name '%s'"),
|
|
|
|
src->path);
|
2020-07-02 23:19:45 -04:00
|
|
|
return NULL;
|
2016-02-10 20:53:36 -07:00
|
|
|
}
|
|
|
|
|
2017-10-23 16:23:45 +02:00
|
|
|
virBufferStrcat(&buf, "rbd:", src->volume, "/", src->path, NULL);
|
2016-02-10 20:53:36 -07:00
|
|
|
|
|
|
|
if (username) {
|
|
|
|
virBufferEscape(&buf, '\\', ":", ":id=%s", username);
|
|
|
|
virBufferEscape(&buf, '\\', ":",
|
|
|
|
":key=%s:auth_supported=cephx\\;none",
|
|
|
|
secret);
|
|
|
|
} else {
|
|
|
|
virBufferAddLit(&buf, ":auth_supported=none");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (src->nhosts > 0) {
|
|
|
|
virBufferAddLit(&buf, ":mon_host=");
|
|
|
|
for (i = 0; i < src->nhosts; i++) {
|
|
|
|
if (i)
|
|
|
|
virBufferAddLit(&buf, "\\;");
|
|
|
|
|
|
|
|
/* assume host containing : is ipv6 */
|
|
|
|
if (strchr(src->hosts[i].name, ':'))
|
|
|
|
virBufferEscape(&buf, '\\', ":", "[%s]",
|
|
|
|
src->hosts[i].name);
|
|
|
|
else
|
|
|
|
virBufferAsprintf(&buf, "%s", src->hosts[i].name);
|
|
|
|
|
|
|
|
if (src->hosts[i].port)
|
2017-07-20 12:45:42 +02:00
|
|
|
virBufferAsprintf(&buf, "\\:%u", src->hosts[i].port);
|
2016-02-10 20:53:36 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (src->configFile)
|
|
|
|
virBufferEscape(&buf, '\\', ":", ":conf=%s", src->configFile);
|
|
|
|
|
2020-07-02 23:19:45 -04:00
|
|
|
return virBufferContentAndReset(&buf);
|
2016-02-10 20:53:36 -07:00
|
|
|
}
|
|
|
|
|
2020-07-02 23:19:45 -04:00
|
|
|
return NULL;
|
2016-02-10 20:53:36 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlMakeNetworkDiskSrc(virStorageSource *src, char **srcstr)
|
2016-02-10 20:53:36 -07:00
|
|
|
{
|
|
|
|
virConnectPtr conn = NULL;
|
2021-02-02 16:49:30 +01:00
|
|
|
g_autofree char *base64secret = NULL;
|
2016-02-10 20:53:36 -07:00
|
|
|
char *username = NULL;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
*srcstr = NULL;
|
|
|
|
if (src->auth && src->protocol == VIR_STORAGE_NET_PROTOCOL_RBD) {
|
2021-02-01 12:50:00 +01:00
|
|
|
g_autofree uint8_t *secret = NULL;
|
|
|
|
size_t secretlen = 0;
|
2021-05-07 16:53:40 +01:00
|
|
|
VIR_IDENTITY_AUTORESTORE virIdentity *oldident = virIdentityElevateCurrent();
|
|
|
|
|
|
|
|
if (!oldident)
|
|
|
|
goto cleanup;
|
2021-02-01 12:50:00 +01:00
|
|
|
|
2016-02-10 20:53:36 -07:00
|
|
|
username = src->auth->username;
|
|
|
|
if (!(conn = virConnectOpen("xen:///system")))
|
|
|
|
goto cleanup;
|
|
|
|
|
2016-05-28 08:43:23 -04:00
|
|
|
if (virSecretGetSecretString(conn, &src->auth->seclookupdef,
|
2016-05-12 11:43:39 -04:00
|
|
|
VIR_SECRET_USAGE_TYPE_CEPH,
|
|
|
|
&secret, &secretlen) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
/* RBD expects an encoded secret */
|
2019-09-16 13:29:20 +01:00
|
|
|
base64secret = g_base64_encode(secret, secretlen);
|
2021-02-01 12:50:00 +01:00
|
|
|
virSecureErase(secret, secretlen);
|
2016-02-10 20:53:36 -07:00
|
|
|
}
|
|
|
|
|
2021-02-02 16:49:30 +01:00
|
|
|
*srcstr = libxlMakeNetworkDiskSrcStr(src, username, base64secret);
|
|
|
|
virSecureEraseString(base64secret);
|
|
|
|
|
|
|
|
if (!*srcstr)
|
2016-02-23 15:28:04 -07:00
|
|
|
goto cleanup;
|
2016-02-10 20:53:36 -07:00
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virObjectUnref(conn);
|
|
|
|
return ret;
|
|
|
|
}
|
2014-09-19 10:32:43 -06:00
|
|
|
|
2011-05-26 23:09:01 +08:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlMakeDisk(virDomainDiskDef *l_disk, libxl_device_disk *x_disk)
|
2011-05-26 23:09:01 +08:00
|
|
|
{
|
2016-05-23 15:56:01 -06:00
|
|
|
const char *driver = virDomainDiskGetDriver(l_disk);
|
|
|
|
int format = virDomainDiskGetFormat(l_disk);
|
2016-02-10 20:53:36 -07:00
|
|
|
int actual_type = virStorageSourceGetActualType(l_disk->src);
|
2014-03-17 14:14:11 -06:00
|
|
|
|
2016-02-10 20:53:36 -07:00
|
|
|
if (actual_type == VIR_STORAGE_TYPE_NETWORK) {
|
2016-05-23 15:56:01 -06:00
|
|
|
if (STRNEQ_NULLABLE(driver, "qemu")) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID, "%s",
|
|
|
|
_("only the 'qemu' driver can be used with network disks"));
|
|
|
|
return -1;
|
|
|
|
}
|
2016-02-10 20:53:36 -07:00
|
|
|
if (libxlMakeNetworkDiskSrc(l_disk->src, &x_disk->pdev_path) < 0)
|
|
|
|
return -1;
|
|
|
|
} else {
|
2019-10-20 13:49:46 +02:00
|
|
|
x_disk->pdev_path = g_strdup(virDomainDiskGetSource(l_disk));
|
2016-02-10 20:53:36 -07:00
|
|
|
}
|
2011-05-26 23:09:01 +08:00
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
x_disk->vdev = g_strdup(l_disk->dst);
|
2011-05-26 23:09:01 +08:00
|
|
|
|
2014-03-17 14:14:11 -06:00
|
|
|
if (driver) {
|
|
|
|
if (STREQ(driver, "tap") || STREQ(driver, "tap2")) {
|
|
|
|
switch (format) {
|
2012-10-15 15:47:42 -06:00
|
|
|
case VIR_STORAGE_FILE_QCOW:
|
2012-11-26 09:28:56 -07:00
|
|
|
x_disk->format = LIBXL_DISK_FORMAT_QCOW;
|
|
|
|
x_disk->backend = LIBXL_DISK_BACKEND_QDISK;
|
2012-10-15 15:47:42 -06:00
|
|
|
break;
|
|
|
|
case VIR_STORAGE_FILE_QCOW2:
|
2012-11-26 09:28:56 -07:00
|
|
|
x_disk->format = LIBXL_DISK_FORMAT_QCOW2;
|
|
|
|
x_disk->backend = LIBXL_DISK_BACKEND_QDISK;
|
2012-10-15 15:47:42 -06:00
|
|
|
break;
|
|
|
|
case VIR_STORAGE_FILE_VHD:
|
2012-11-26 09:28:56 -07:00
|
|
|
x_disk->format = LIBXL_DISK_FORMAT_VHD;
|
|
|
|
x_disk->backend = LIBXL_DISK_BACKEND_TAP;
|
2012-10-15 15:47:42 -06:00
|
|
|
break;
|
|
|
|
case VIR_STORAGE_FILE_RAW:
|
2012-11-26 09:28:56 -07:00
|
|
|
x_disk->format = LIBXL_DISK_FORMAT_RAW;
|
|
|
|
x_disk->backend = LIBXL_DISK_BACKEND_TAP;
|
2012-10-15 15:47:42 -06:00
|
|
|
break;
|
2016-11-14 16:00:02 +01:00
|
|
|
case VIR_STORAGE_FILE_QED:
|
|
|
|
x_disk->format = LIBXL_DISK_FORMAT_QED;
|
|
|
|
x_disk->backend = LIBXL_DISK_BACKEND_QDISK;
|
|
|
|
break;
|
2012-10-15 15:47:42 -06:00
|
|
|
default:
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
2013-05-13 12:14:20 -06:00
|
|
|
_("libxenlight does not support disk format %s "
|
|
|
|
"with disk driver %s"),
|
2014-03-17 14:14:11 -06:00
|
|
|
virStorageFileFormatTypeToString(format),
|
|
|
|
driver);
|
2012-10-15 15:47:42 -06:00
|
|
|
return -1;
|
2011-05-26 23:09:01 +08:00
|
|
|
}
|
2014-03-17 14:14:11 -06:00
|
|
|
} else if (STREQ(driver, "qemu")) {
|
2013-05-13 12:14:21 -06:00
|
|
|
x_disk->backend = LIBXL_DISK_BACKEND_QDISK;
|
2014-03-17 14:14:11 -06:00
|
|
|
switch (format) {
|
2013-05-13 12:14:21 -06:00
|
|
|
case VIR_STORAGE_FILE_QCOW:
|
|
|
|
x_disk->format = LIBXL_DISK_FORMAT_QCOW;
|
|
|
|
break;
|
|
|
|
case VIR_STORAGE_FILE_QCOW2:
|
|
|
|
x_disk->format = LIBXL_DISK_FORMAT_QCOW2;
|
|
|
|
break;
|
2016-11-14 16:00:02 +01:00
|
|
|
case VIR_STORAGE_FILE_QED:
|
|
|
|
x_disk->format = LIBXL_DISK_FORMAT_QED;
|
|
|
|
break;
|
2013-05-13 12:14:21 -06:00
|
|
|
case VIR_STORAGE_FILE_VHD:
|
|
|
|
x_disk->format = LIBXL_DISK_FORMAT_VHD;
|
|
|
|
break;
|
|
|
|
case VIR_STORAGE_FILE_RAW:
|
|
|
|
x_disk->format = LIBXL_DISK_FORMAT_RAW;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("libxenlight does not support disk format %s "
|
|
|
|
"with disk driver %s"),
|
2014-03-17 14:14:11 -06:00
|
|
|
virStorageFileFormatTypeToString(format),
|
|
|
|
driver);
|
2013-05-13 12:14:21 -06:00
|
|
|
return -1;
|
|
|
|
}
|
2014-03-17 14:14:11 -06:00
|
|
|
} else if (STREQ(driver, "file")) {
|
2017-02-07 11:00:33 -07:00
|
|
|
if (format != VIR_STORAGE_FILE_RAW) {
|
2013-05-13 12:14:20 -06:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("libxenlight does not support disk format %s "
|
|
|
|
"with disk driver %s"),
|
2014-03-17 14:14:11 -06:00
|
|
|
virStorageFileFormatTypeToString(format),
|
|
|
|
driver);
|
2013-05-13 12:14:20 -06:00
|
|
|
return -1;
|
|
|
|
}
|
2012-11-26 09:28:56 -07:00
|
|
|
x_disk->format = LIBXL_DISK_FORMAT_RAW;
|
2014-06-19 23:36:25 -06:00
|
|
|
x_disk->backend = LIBXL_DISK_BACKEND_QDISK;
|
2014-03-17 14:14:11 -06:00
|
|
|
} else if (STREQ(driver, "phy")) {
|
2017-02-07 11:00:33 -07:00
|
|
|
if (format != VIR_STORAGE_FILE_RAW) {
|
2013-05-13 12:14:20 -06:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("libxenlight does not support disk format %s "
|
|
|
|
"with disk driver %s"),
|
2014-03-17 14:14:11 -06:00
|
|
|
virStorageFileFormatTypeToString(format),
|
|
|
|
driver);
|
2013-05-13 12:14:20 -06:00
|
|
|
return -1;
|
|
|
|
}
|
2012-11-26 09:28:56 -07:00
|
|
|
x_disk->format = LIBXL_DISK_FORMAT_RAW;
|
|
|
|
x_disk->backend = LIBXL_DISK_BACKEND_PHY;
|
2011-05-26 23:09:01 +08:00
|
|
|
} else {
|
2012-07-18 14:32:10 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("libxenlight does not support disk driver %s"),
|
2014-03-17 14:14:11 -06:00
|
|
|
driver);
|
2011-05-26 23:09:01 +08:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
} else {
|
2013-02-20 13:02:28 -07:00
|
|
|
/*
|
|
|
|
* If driverName is not specified, default to raw as per
|
|
|
|
* xl-disk-configuration.txt in the xen documentation and let
|
|
|
|
* libxl pick a suitable backend.
|
|
|
|
*/
|
2012-11-26 09:28:56 -07:00
|
|
|
x_disk->format = LIBXL_DISK_FORMAT_RAW;
|
2013-02-20 13:02:28 -07:00
|
|
|
x_disk->backend = LIBXL_DISK_BACKEND_UNKNOWN;
|
2011-05-26 23:09:01 +08:00
|
|
|
}
|
|
|
|
|
2012-11-26 09:28:56 -07:00
|
|
|
/* XXX is this right? */
|
|
|
|
x_disk->removable = 1;
|
2014-06-24 15:15:55 +02:00
|
|
|
x_disk->readwrite = !l_disk->src->readonly;
|
2011-05-26 23:09:01 +08:00
|
|
|
x_disk->is_cdrom = l_disk->device == VIR_DOMAIN_DISK_DEVICE_CDROM ? 1 : 0;
|
2014-07-07 17:05:17 +02:00
|
|
|
if (libxlDiskSetDiscard(x_disk, l_disk->discard) < 0)
|
|
|
|
return -1;
|
2014-03-27 17:55:03 +01:00
|
|
|
/* An empty CDROM must have the empty format, otherwise libxl fails. */
|
|
|
|
if (x_disk->is_cdrom && !x_disk->pdev_path)
|
|
|
|
x_disk->format = LIBXL_DISK_FORMAT_EMPTY;
|
2011-08-13 15:42:48 -06:00
|
|
|
if (l_disk->transient) {
|
2012-07-18 14:32:10 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("libxenlight does not support transient disks"));
|
2011-08-13 15:42:48 -06:00
|
|
|
return -1;
|
|
|
|
}
|
2011-05-26 23:09:01 +08:00
|
|
|
|
2015-02-20 04:22:06 +01:00
|
|
|
if (l_disk->domain_name) {
|
|
|
|
#ifdef LIBXL_HAVE_DEVICE_BACKEND_DOMNAME
|
2019-10-20 13:49:46 +02:00
|
|
|
x_disk->backend_domname = g_strdup(l_disk->domain_name);
|
2015-02-20 04:22:06 +01:00
|
|
|
#else
|
|
|
|
virReportError(VIR_ERR_XML_DETAIL, "%s",
|
|
|
|
_("this version of libxenlight does not "
|
|
|
|
"support backend domain name"));
|
|
|
|
return -1;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2011-05-26 23:09:01 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-10 15:42:34 -07:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlMakeDiskList(virDomainDef *def, libxl_domain_config *d_config)
|
2011-02-10 15:42:34 -07:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainDiskDef **l_disks = def->disks;
|
2011-02-10 15:42:34 -07:00
|
|
|
int ndisks = def->ndisks;
|
Convert 'int i' to 'size_t i' in src/libxl/ files
Convert the type of loop iterators named 'i', 'j', k',
'ii', 'jj', 'kk', to be 'size_t' instead of 'int' or
'unsigned int', also santizing 'ii', 'jj', 'kk' to use
the normal 'i', 'j', 'k' naming
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2013-07-08 15:09:33 +01:00
|
|
|
size_t i;
|
2011-02-10 15:42:34 -07:00
|
|
|
|
2021-05-20 12:17:20 +02:00
|
|
|
d_config->disks = g_new0(libxl_device_disk, ndisks);
|
|
|
|
d_config->num_disks = ndisks;
|
2011-02-10 15:42:34 -07:00
|
|
|
|
|
|
|
for (i = 0; i < ndisks; i++) {
|
2021-05-20 12:17:20 +02:00
|
|
|
libxl_device_disk_init(&d_config->disks[i]);
|
|
|
|
if (libxlMakeDisk(l_disks[i], &d_config->disks[i]) < 0)
|
|
|
|
return -1;
|
2011-02-10 15:42:34 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-02-07 12:05:15 -07:00
|
|
|
/*
|
|
|
|
* Update libvirt disk config with libxl disk config.
|
|
|
|
*
|
|
|
|
* This function can be used to update the libvirt disk config with default
|
|
|
|
* values selected by libxl. Currently only the backend type is selected by
|
|
|
|
* libxl when not explicitly specified by the user.
|
|
|
|
*/
|
|
|
|
void
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlUpdateDiskDef(virDomainDiskDef *l_disk, libxl_device_disk *x_disk)
|
2017-02-07 12:05:15 -07:00
|
|
|
{
|
|
|
|
const char *driver = NULL;
|
|
|
|
|
|
|
|
if (virDomainDiskGetDriver(l_disk))
|
|
|
|
return;
|
|
|
|
|
|
|
|
switch (x_disk->backend) {
|
|
|
|
case LIBXL_DISK_BACKEND_QDISK:
|
|
|
|
driver = "qemu";
|
|
|
|
break;
|
|
|
|
case LIBXL_DISK_BACKEND_TAP:
|
|
|
|
driver = "tap";
|
|
|
|
break;
|
|
|
|
case LIBXL_DISK_BACKEND_PHY:
|
|
|
|
driver = "phy";
|
|
|
|
break;
|
|
|
|
case LIBXL_DISK_BACKEND_UNKNOWN:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (driver)
|
2020-11-05 22:32:37 -05:00
|
|
|
virDomainDiskSetDriver(l_disk, driver);
|
2017-02-07 12:05:15 -07:00
|
|
|
}
|
|
|
|
|
2011-05-26 23:09:01 +08:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlMakeNic(virDomainDef *def,
|
|
|
|
virDomainNetDef *l_nic,
|
2016-12-14 10:53:32 +00:00
|
|
|
libxl_device_nic *x_nic,
|
|
|
|
bool attach)
|
2011-05-26 23:09:01 +08:00
|
|
|
{
|
2014-06-10 14:28:07 -06:00
|
|
|
virDomainNetType actual_type = virDomainNetGetActualType(l_nic);
|
2016-02-24 15:40:00 -07:00
|
|
|
virNetworkPtr network = NULL;
|
|
|
|
virConnectPtr conn = NULL;
|
2019-10-01 12:52:03 -04:00
|
|
|
const virNetDevBandwidth *actual_bw;
|
2019-10-01 13:56:35 -04:00
|
|
|
const virNetDevVPortProfile *port_profile;
|
2019-10-01 12:25:47 -04:00
|
|
|
const virNetDevVlan *virt_vlan;
|
2020-07-02 18:21:30 -04:00
|
|
|
g_auto(virBuffer) buf = VIR_BUFFER_INITIALIZER;
|
2016-06-28 14:20:02 -06:00
|
|
|
size_t i;
|
|
|
|
const char *script = NULL;
|
2016-02-24 15:40:00 -07:00
|
|
|
int ret = -1;
|
2014-01-06 11:37:20 -07:00
|
|
|
|
2012-11-26 09:28:56 -07:00
|
|
|
/* TODO: Where is mtu stored?
|
|
|
|
*
|
|
|
|
* x_nics[i].mtu = 1492;
|
|
|
|
*/
|
2011-05-26 23:09:01 +08:00
|
|
|
|
2014-06-10 14:53:55 -06:00
|
|
|
if (l_nic->script && !(actual_type == VIR_DOMAIN_NET_TYPE_BRIDGE ||
|
|
|
|
actual_type == VIR_DOMAIN_NET_TYPE_ETHERNET)) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("specifying a script is only supported with "
|
|
|
|
"interface types bridge and ethernet"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2012-07-17 08:07:59 -04:00
|
|
|
virMacAddrGetRaw(&l_nic->mac, x_nic->mac);
|
2011-05-26 23:09:01 +08:00
|
|
|
|
2016-12-14 10:53:32 +00:00
|
|
|
/*
|
|
|
|
* The nictype field of libxl_device_nic structure tells Xen which type of
|
|
|
|
* NIC device to create for the domain. LIBXL_NIC_TYPE_VIF specifies a
|
|
|
|
* PV NIC. LIBXL_NIC_TYPE_VIF_IOEMU specifies a PV and emulated NIC,
|
|
|
|
* allowing the domain to choose which NIC to use and unplug the unused
|
|
|
|
* one. LIBXL_NIC_TYPE_VIF_IOEMU is only valid for HVM domains. Further,
|
|
|
|
* if hotplugging the NIC, emulated NICs are currently not supported.
|
|
|
|
* Alternatively one could set LIBXL_NIC_TYPE_UNKNOWN and let libxl decide,
|
|
|
|
* but its behaviour might not be consistent across all libvirt supported
|
|
|
|
* versions. The other nictype values are well established already, hence
|
|
|
|
* we manually select our own default and mimic xl/libxl behaviour starting
|
|
|
|
* xen commit 32e9d0f ("libxl: nic type defaults to vif in hotplug for
|
|
|
|
* hvm guest").
|
|
|
|
*/
|
2019-01-17 17:06:28 -05:00
|
|
|
if (virDomainNetGetModelString(l_nic)) {
|
2018-11-26 20:34:37 +01:00
|
|
|
if ((def->os.type == VIR_DOMAIN_OSTYPE_XEN ||
|
|
|
|
def->os.type == VIR_DOMAIN_OSTYPE_XENPVH) &&
|
2019-01-17 19:12:27 -05:00
|
|
|
l_nic->model != VIR_DOMAIN_NET_MODEL_NETFRONT) {
|
2016-12-14 10:53:32 +00:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("only model 'netfront' is supported for "
|
2018-11-26 20:34:37 +01:00
|
|
|
"Xen PV(H) domains"));
|
2016-12-14 10:53:32 +00:00
|
|
|
return -1;
|
|
|
|
}
|
2019-10-20 13:49:46 +02:00
|
|
|
x_nic->model = g_strdup(virDomainNetGetModelString(l_nic));
|
2019-01-17 19:12:27 -05:00
|
|
|
if (l_nic->model == VIR_DOMAIN_NET_MODEL_NETFRONT)
|
2014-01-06 11:37:20 -07:00
|
|
|
x_nic->nictype = LIBXL_NIC_TYPE_VIF;
|
2016-12-14 10:53:32 +00:00
|
|
|
else
|
|
|
|
x_nic->nictype = LIBXL_NIC_TYPE_VIF_IOEMU;
|
|
|
|
} else {
|
|
|
|
if (def->os.type == VIR_DOMAIN_OSTYPE_HVM && !attach)
|
|
|
|
x_nic->nictype = LIBXL_NIC_TYPE_VIF_IOEMU;
|
|
|
|
else
|
|
|
|
x_nic->nictype = LIBXL_NIC_TYPE_VIF;
|
2011-05-26 23:09:01 +08:00
|
|
|
}
|
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
x_nic->ifname = g_strdup(l_nic->ifname);
|
2011-05-26 23:09:01 +08:00
|
|
|
|
2016-06-28 14:20:02 -06:00
|
|
|
port_profile = virDomainNetGetActualVirtPortProfile(l_nic);
|
|
|
|
virt_vlan = virDomainNetGetActualVlan(l_nic);
|
|
|
|
script = l_nic->script;
|
2014-06-10 14:28:07 -06:00
|
|
|
switch (actual_type) {
|
2013-06-13 03:54:03 +02:00
|
|
|
case VIR_DOMAIN_NET_TYPE_BRIDGE:
|
2016-06-28 14:20:02 -06:00
|
|
|
virBufferAddStr(&buf, virDomainNetGetActualBridgeName(l_nic));
|
|
|
|
/*
|
|
|
|
* A bit of special handling if vif will be connected to an
|
|
|
|
* openvswitch bridge
|
|
|
|
*/
|
|
|
|
if (port_profile &&
|
|
|
|
port_profile->virtPortType == VIR_NETDEV_VPORT_PROFILE_OPENVSWITCH) {
|
|
|
|
/*
|
|
|
|
* If a custom script is not specified for openvswitch, use
|
|
|
|
* Xen's vif-openvswitch script
|
|
|
|
*/
|
|
|
|
if (!script)
|
|
|
|
script = "vif-openvswitch";
|
|
|
|
/*
|
|
|
|
* libxl_device_nic->bridge supports an extended format for
|
|
|
|
* specifying VLAN tags and trunks when using openvswitch
|
|
|
|
*
|
|
|
|
* BRIDGE_NAME[.VLAN][:TRUNK:TRUNK]
|
|
|
|
*
|
|
|
|
* See Xen's networking wiki for more details
|
|
|
|
* https://wiki.xenproject.org/wiki/Xen_Networking#Open_vSwitch
|
|
|
|
*/
|
|
|
|
if (virt_vlan && virt_vlan->nTags > 0) {
|
|
|
|
if (virt_vlan->trunk) {
|
|
|
|
for (i = 0; i < virt_vlan->nTags; i++)
|
|
|
|
virBufferAsprintf(&buf, ":%d", virt_vlan->tag[i]);
|
|
|
|
} else {
|
|
|
|
virBufferAsprintf(&buf, ".%d", virt_vlan->tag[0]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-07-04 23:43:52 -04:00
|
|
|
x_nic->bridge = virBufferContentAndReset(&buf);
|
2019-10-15 13:38:21 +02:00
|
|
|
G_GNUC_FALLTHROUGH;
|
2013-06-13 03:54:03 +02:00
|
|
|
case VIR_DOMAIN_NET_TYPE_ETHERNET:
|
2019-10-20 13:49:46 +02:00
|
|
|
x_nic->script = g_strdup(script);
|
2016-06-07 11:39:34 -04:00
|
|
|
if (l_nic->guestIP.nips > 0) {
|
2017-12-07 03:27:45 +01:00
|
|
|
x_nic->ip = xenMakeIPList(&l_nic->guestIP);
|
2015-02-10 03:17:23 +01:00
|
|
|
if (!x_nic->ip)
|
2016-02-24 15:40:00 -07:00
|
|
|
goto cleanup;
|
2015-02-10 03:17:23 +01:00
|
|
|
}
|
2013-06-13 03:54:03 +02:00
|
|
|
break;
|
2014-06-10 14:28:07 -06:00
|
|
|
case VIR_DOMAIN_NET_TYPE_NETWORK:
|
2019-04-30 13:26:25 +01:00
|
|
|
{
|
|
|
|
if (!(conn = virConnectOpen("xen:///system")))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (!(network =
|
|
|
|
virNetworkLookupByName(conn, l_nic->data.network.name))) {
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (l_nic->guestIP.nips > 0) {
|
|
|
|
x_nic->ip = xenMakeIPList(&l_nic->guestIP);
|
|
|
|
if (!x_nic->ip)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2014-06-10 14:28:07 -06:00
|
|
|
|
2019-04-30 13:26:25 +01:00
|
|
|
if (!(x_nic->bridge = virNetworkGetBridgeName(network)))
|
|
|
|
goto cleanup;
|
|
|
|
break;
|
|
|
|
}
|
2014-07-11 19:47:31 +02:00
|
|
|
case VIR_DOMAIN_NET_TYPE_VHOSTUSER:
|
2014-06-10 14:28:07 -06:00
|
|
|
case VIR_DOMAIN_NET_TYPE_USER:
|
|
|
|
case VIR_DOMAIN_NET_TYPE_SERVER:
|
|
|
|
case VIR_DOMAIN_NET_TYPE_CLIENT:
|
|
|
|
case VIR_DOMAIN_NET_TYPE_MCAST:
|
2015-08-29 16:19:10 -04:00
|
|
|
case VIR_DOMAIN_NET_TYPE_UDP:
|
2014-06-10 14:28:07 -06:00
|
|
|
case VIR_DOMAIN_NET_TYPE_INTERNAL:
|
|
|
|
case VIR_DOMAIN_NET_TYPE_DIRECT:
|
|
|
|
case VIR_DOMAIN_NET_TYPE_HOSTDEV:
|
2020-10-14 12:08:25 -05:00
|
|
|
case VIR_DOMAIN_NET_TYPE_VDPA:
|
2014-06-10 14:28:07 -06:00
|
|
|
case VIR_DOMAIN_NET_TYPE_LAST:
|
2014-06-10 14:53:55 -06:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("unsupported interface type %s"),
|
2013-06-13 03:54:03 +02:00
|
|
|
virDomainNetTypeToString(l_nic->type));
|
2016-02-24 15:40:00 -07:00
|
|
|
goto cleanup;
|
2011-05-26 23:09:01 +08:00
|
|
|
}
|
|
|
|
|
2015-02-20 04:22:06 +01:00
|
|
|
if (l_nic->domain_name) {
|
|
|
|
#ifdef LIBXL_HAVE_DEVICE_BACKEND_DOMNAME
|
2019-10-20 13:49:46 +02:00
|
|
|
x_nic->backend_domname = g_strdup(l_nic->domain_name);
|
2015-02-20 04:22:06 +01:00
|
|
|
#else
|
|
|
|
virReportError(VIR_ERR_XML_DETAIL, "%s",
|
|
|
|
_("this version of libxenlight does not "
|
|
|
|
"support backend domain name"));
|
2016-02-24 15:40:00 -07:00
|
|
|
goto cleanup;
|
2015-02-20 04:22:06 +01:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2015-12-28 17:26:58 -07:00
|
|
|
/*
|
|
|
|
* Set bandwidth.
|
|
|
|
* From $xen-sources/docs/misc/xl-network-configuration.markdown:
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* Specifies the rate at which the outgoing traffic will be limited to.
|
|
|
|
* The default if this keyword is not specified is unlimited.
|
|
|
|
*
|
|
|
|
* The rate may be specified as "<RATE>/s" or optionally "<RATE>/s@<INTERVAL>".
|
|
|
|
*
|
|
|
|
* `RATE` is in bytes and can accept suffixes:
|
|
|
|
* GB, MB, KB, B for bytes.
|
|
|
|
* Gb, Mb, Kb, b for bits.
|
|
|
|
* `INTERVAL` is in microseconds and can accept suffixes: ms, us, s.
|
|
|
|
* It determines the frequency at which the vif transmission credit
|
|
|
|
* is replenished. The default is 50ms.
|
|
|
|
|
|
|
|
* Vif rate limiting is credit-based. It means that for "1MB/s@20ms",
|
|
|
|
* the available credit will be equivalent of the traffic you would have
|
|
|
|
* done at "1MB/s" during 20ms. This will results in a credit of 20,000
|
|
|
|
* bytes replenished every 20,000 us.
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* libvirt doesn't support the notion of rate limiting over an interval.
|
|
|
|
* Similar to xl's behavior when interval is not specified, set a default
|
|
|
|
* interval of 50ms and calculate the number of bytes per interval based
|
|
|
|
* on the specified average bandwidth.
|
|
|
|
*/
|
|
|
|
actual_bw = virDomainNetGetActualBandwidth(l_nic);
|
|
|
|
if (actual_bw && actual_bw->out && actual_bw->out->average) {
|
|
|
|
uint64_t bytes_per_sec = actual_bw->out->average * 1024;
|
|
|
|
uint64_t bytes_per_interval =
|
|
|
|
(((uint64_t) bytes_per_sec * 50000UL) / 1000000UL);
|
|
|
|
|
|
|
|
x_nic->rate_bytes_per_interval = bytes_per_interval;
|
|
|
|
x_nic->rate_interval_usecs = 50000UL;
|
|
|
|
}
|
|
|
|
|
2016-02-24 15:40:00 -07:00
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virObjectUnref(network);
|
|
|
|
virObjectUnref(conn);
|
|
|
|
|
|
|
|
return ret;
|
2011-05-26 23:09:01 +08:00
|
|
|
}
|
|
|
|
|
2011-02-10 15:42:34 -07:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlMakeNicList(virDomainDef *def, libxl_domain_config *d_config)
|
2011-02-10 15:42:34 -07:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainNetDef **l_nics = def->nets;
|
2014-05-08 14:44:04 +08:00
|
|
|
size_t nnics = def->nnets;
|
2011-02-10 15:42:34 -07:00
|
|
|
libxl_device_nic *x_nics;
|
2014-05-08 14:44:04 +08:00
|
|
|
size_t i, nvnics = 0;
|
2021-05-26 14:05:05 -06:00
|
|
|
int ret = -1;
|
2011-02-10 15:42:34 -07:00
|
|
|
|
2020-09-23 20:43:09 +02:00
|
|
|
x_nics = g_new0(libxl_device_nic, nnics);
|
2011-02-10 15:42:34 -07:00
|
|
|
|
|
|
|
for (i = 0; i < nnics; i++) {
|
2016-03-10 10:50:41 +08:00
|
|
|
if (virDomainNetGetActualType(l_nics[i]) == VIR_DOMAIN_NET_TYPE_HOSTDEV)
|
2014-05-08 14:44:04 +08:00
|
|
|
continue;
|
|
|
|
|
2021-05-26 14:05:05 -06:00
|
|
|
libxl_device_nic_init(&x_nics[nvnics]);
|
2016-12-14 10:53:32 +00:00
|
|
|
if (libxlMakeNic(def, l_nics[i], &x_nics[nvnics], false))
|
2021-05-26 14:05:05 -06:00
|
|
|
goto out;
|
2014-01-08 11:39:19 +01:00
|
|
|
/*
|
|
|
|
* The devid (at least right now) will not get initialized by
|
|
|
|
* libxl in the setup case but is required for starting the
|
|
|
|
* device-model.
|
|
|
|
*/
|
2014-05-08 14:44:04 +08:00
|
|
|
if (x_nics[nvnics].devid < 0)
|
|
|
|
x_nics[nvnics].devid = nvnics;
|
|
|
|
|
|
|
|
nvnics++;
|
2011-02-10 15:42:34 -07:00
|
|
|
}
|
2021-05-26 14:05:05 -06:00
|
|
|
ret = 0;
|
2011-02-10 15:42:34 -07:00
|
|
|
|
2021-05-26 14:05:05 -06:00
|
|
|
out:
|
2014-05-08 14:44:04 +08:00
|
|
|
VIR_SHRINK_N(x_nics, nnics, nnics - nvnics);
|
2012-11-26 09:28:56 -07:00
|
|
|
d_config->nics = x_nics;
|
2016-02-23 16:07:58 +08:00
|
|
|
d_config->num_nics = nvnics;
|
2011-02-10 15:42:34 -07:00
|
|
|
|
2021-05-26 14:05:05 -06:00
|
|
|
return ret;
|
2011-02-10 15:42:34 -07:00
|
|
|
}
|
|
|
|
|
2011-05-26 23:09:01 +08:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlMakeVfb(virPortAllocatorRange *graphicsports,
|
|
|
|
virDomainGraphicsDef *l_vfb,
|
2012-11-26 09:28:56 -07:00
|
|
|
libxl_device_vfb *x_vfb)
|
2011-05-26 23:09:01 +08:00
|
|
|
{
|
2013-01-11 12:09:53 +00:00
|
|
|
unsigned short port;
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainGraphicsListenDef *glisten = NULL;
|
2011-05-26 23:09:01 +08:00
|
|
|
|
2013-06-13 03:54:10 +02:00
|
|
|
libxl_device_vfb_init(x_vfb);
|
|
|
|
|
2011-05-26 23:09:01 +08:00
|
|
|
switch (l_vfb->type) {
|
|
|
|
case VIR_DOMAIN_GRAPHICS_TYPE_SDL:
|
2012-11-26 09:28:56 -07:00
|
|
|
libxl_defbool_set(&x_vfb->sdl.enable, 1);
|
2015-04-17 17:19:31 +00:00
|
|
|
libxl_defbool_set(&x_vfb->vnc.enable, 0);
|
2015-04-24 14:11:26 -06:00
|
|
|
libxl_defbool_set(&x_vfb->sdl.opengl, 0);
|
2019-10-20 13:49:46 +02:00
|
|
|
x_vfb->sdl.display = g_strdup(l_vfb->data.sdl.display);
|
|
|
|
x_vfb->sdl.xauthority = g_strdup(l_vfb->data.sdl.xauth);
|
2011-05-26 23:09:01 +08:00
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_GRAPHICS_TYPE_VNC:
|
2012-11-26 09:28:56 -07:00
|
|
|
libxl_defbool_set(&x_vfb->vnc.enable, 1);
|
2015-04-17 17:19:31 +00:00
|
|
|
libxl_defbool_set(&x_vfb->sdl.enable, 0);
|
2011-05-26 23:09:01 +08:00
|
|
|
/* driver handles selection of free port */
|
2012-11-26 09:28:56 -07:00
|
|
|
libxl_defbool_set(&x_vfb->vnc.findunused, 0);
|
2011-05-26 23:09:01 +08:00
|
|
|
if (l_vfb->data.vnc.autoport) {
|
2013-01-11 12:09:53 +00:00
|
|
|
|
2014-05-30 14:48:56 +01:00
|
|
|
if (virPortAllocatorAcquire(graphicsports, &port) < 0)
|
2013-01-11 12:09:53 +00:00
|
|
|
return -1;
|
2011-05-26 23:09:01 +08:00
|
|
|
l_vfb->data.vnc.port = port;
|
|
|
|
}
|
2012-11-26 09:28:56 -07:00
|
|
|
x_vfb->vnc.display = l_vfb->data.vnc.port - LIBXL_VNC_PORT_MIN;
|
2011-05-26 23:09:01 +08:00
|
|
|
|
2017-05-19 15:52:04 -06:00
|
|
|
if ((glisten = virDomainGraphicsGetListen(l_vfb, 0))) {
|
|
|
|
if (glisten->address) {
|
2019-10-20 13:49:46 +02:00
|
|
|
/* libxl_device_vfb_init() does g_strdup("127.0.0.1") */
|
2017-05-19 15:52:04 -06:00
|
|
|
VIR_FREE(x_vfb->vnc.listen);
|
2019-10-20 13:49:46 +02:00
|
|
|
x_vfb->vnc.listen = g_strdup(glisten->address);
|
2017-05-19 15:52:04 -06:00
|
|
|
} else {
|
2019-10-20 13:49:46 +02:00
|
|
|
glisten->address = g_strdup(VIR_LOOPBACK_IPV4_ADDR);
|
2017-05-19 15:52:04 -06:00
|
|
|
}
|
2011-05-26 23:09:01 +08:00
|
|
|
}
|
2017-05-19 15:52:04 -06:00
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
x_vfb->vnc.passwd = g_strdup(l_vfb->data.vnc.auth.passwd);
|
|
|
|
x_vfb->keymap = g_strdup(l_vfb->data.vnc.keymap);
|
2011-05-26 23:09:01 +08:00
|
|
|
break;
|
2016-05-02 17:45:23 +02:00
|
|
|
|
|
|
|
case VIR_DOMAIN_GRAPHICS_TYPE_RDP:
|
|
|
|
case VIR_DOMAIN_GRAPHICS_TYPE_DESKTOP:
|
|
|
|
case VIR_DOMAIN_GRAPHICS_TYPE_SPICE:
|
2018-06-30 16:23:01 +02:00
|
|
|
case VIR_DOMAIN_GRAPHICS_TYPE_EGL_HEADLESS:
|
2016-05-02 17:45:23 +02:00
|
|
|
case VIR_DOMAIN_GRAPHICS_TYPE_LAST:
|
|
|
|
break;
|
2011-05-26 23:09:01 +08:00
|
|
|
}
|
2012-11-26 09:28:56 -07:00
|
|
|
|
2011-05-26 23:09:01 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-10 15:42:34 -07:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlMakeVfbList(virPortAllocatorRange *graphicsports,
|
|
|
|
virDomainDef *def,
|
2012-11-26 09:28:56 -07:00
|
|
|
libxl_domain_config *d_config)
|
2011-02-10 15:42:34 -07:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainGraphicsDef **l_vfbs = def->graphics;
|
2011-02-10 15:42:34 -07:00
|
|
|
int nvfbs = def->ngraphics;
|
|
|
|
libxl_device_vfb *x_vfbs;
|
|
|
|
libxl_device_vkb *x_vkbs;
|
Convert 'int i' to 'size_t i' in src/libxl/ files
Convert the type of loop iterators named 'i', 'j', k',
'ii', 'jj', 'kk', to be 'size_t' instead of 'int' or
'unsigned int', also santizing 'ii', 'jj', 'kk' to use
the normal 'i', 'j', 'k' naming
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2013-07-08 15:09:33 +01:00
|
|
|
size_t i;
|
2011-02-10 15:42:34 -07:00
|
|
|
|
|
|
|
if (nvfbs == 0)
|
|
|
|
return 0;
|
|
|
|
|
2020-09-23 20:43:09 +02:00
|
|
|
x_vfbs = g_new0(libxl_device_vfb, nvfbs);
|
|
|
|
x_vkbs = g_new0(libxl_device_vkb, nvfbs);
|
2011-02-10 15:42:34 -07:00
|
|
|
|
|
|
|
for (i = 0; i < nvfbs; i++) {
|
2012-11-26 09:28:56 -07:00
|
|
|
libxl_device_vkb_init(&x_vkbs[i]);
|
2011-02-10 15:42:34 -07:00
|
|
|
|
2014-05-30 14:48:56 +01:00
|
|
|
if (libxlMakeVfb(graphicsports, l_vfbs[i], &x_vfbs[i]) < 0)
|
2011-05-26 23:09:01 +08:00
|
|
|
goto error;
|
2011-02-10 15:42:34 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
d_config->vfbs = x_vfbs;
|
|
|
|
d_config->vkbs = x_vkbs;
|
|
|
|
d_config->num_vfbs = d_config->num_vkbs = nvfbs;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
2014-03-25 07:51:17 +01:00
|
|
|
error:
|
2011-02-10 15:42:34 -07:00
|
|
|
for (i = 0; i < nvfbs; i++) {
|
2012-11-26 09:28:56 -07:00
|
|
|
libxl_device_vfb_dispose(&x_vfbs[i]);
|
|
|
|
libxl_device_vkb_dispose(&x_vkbs[i]);
|
2011-02-10 15:42:34 -07:00
|
|
|
}
|
|
|
|
VIR_FREE(x_vfbs);
|
|
|
|
VIR_FREE(x_vkbs);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2015-04-27 12:32:55 -06:00
|
|
|
/*
|
|
|
|
* Populate vfb info in libxl_domain_build_info struct for HVM domains.
|
|
|
|
* Prior to calling this function, libxlMakeVfbList must be called to
|
|
|
|
* populate libxl_domain_config->vfbs.
|
|
|
|
*/
|
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlMakeBuildInfoVfb(virPortAllocatorRange *graphicsports,
|
|
|
|
virDomainDef *def,
|
2015-04-27 16:27:52 -06:00
|
|
|
libxl_domain_config *d_config)
|
2015-04-27 12:32:55 -06:00
|
|
|
{
|
|
|
|
libxl_domain_build_info *b_info = &d_config->b_info;
|
|
|
|
libxl_device_vfb x_vfb;
|
2015-04-27 16:27:52 -06:00
|
|
|
size_t i;
|
2015-04-27 12:32:55 -06:00
|
|
|
|
|
|
|
if (def->os.type != VIR_DOMAIN_OSTYPE_HVM)
|
|
|
|
return 0;
|
|
|
|
|
2015-04-27 16:27:52 -06:00
|
|
|
if (def->ngraphics == 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Prefer SPICE, otherwise use first libxl_device_vfb device in
|
|
|
|
* libxl_domain_config->vfbs. Prior to calling this function,
|
|
|
|
*/
|
|
|
|
for (i = 0; i < def->ngraphics; i++) {
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainGraphicsDef *l_vfb = def->graphics[i];
|
2015-04-27 16:27:52 -06:00
|
|
|
unsigned short port;
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainGraphicsListenDef *glisten = NULL;
|
2015-04-27 16:27:52 -06:00
|
|
|
|
|
|
|
if (l_vfb->type != VIR_DOMAIN_GRAPHICS_TYPE_SPICE)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
libxl_defbool_set(&b_info->u.hvm.spice.enable, true);
|
|
|
|
|
|
|
|
if (l_vfb->data.spice.autoport) {
|
|
|
|
if (virPortAllocatorAcquire(graphicsports, &port) < 0)
|
|
|
|
return -1;
|
|
|
|
l_vfb->data.spice.port = port;
|
|
|
|
}
|
|
|
|
b_info->u.hvm.spice.port = l_vfb->data.spice.port;
|
|
|
|
|
2017-05-19 15:52:04 -06:00
|
|
|
if ((glisten = virDomainGraphicsGetListen(l_vfb, 0))) {
|
|
|
|
if (glisten->address) {
|
2019-10-20 13:49:46 +02:00
|
|
|
b_info->u.hvm.spice.host = g_strdup(glisten->address);
|
2017-05-19 15:52:04 -06:00
|
|
|
} else {
|
2019-10-20 13:49:46 +02:00
|
|
|
b_info->u.hvm.spice.host = g_strdup(VIR_LOOPBACK_IPV4_ADDR);
|
|
|
|
glisten->address = g_strdup(VIR_LOOPBACK_IPV4_ADDR);
|
2017-05-19 15:52:04 -06:00
|
|
|
}
|
|
|
|
}
|
2015-04-27 16:27:52 -06:00
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
b_info->u.hvm.keymap = g_strdup(l_vfb->data.spice.keymap);
|
2015-04-27 16:27:52 -06:00
|
|
|
|
|
|
|
if (l_vfb->data.spice.auth.passwd) {
|
2019-10-20 13:49:46 +02:00
|
|
|
b_info->u.hvm.spice.passwd = g_strdup(l_vfb->data.spice.auth.passwd);
|
2015-04-27 16:27:52 -06:00
|
|
|
libxl_defbool_set(&b_info->u.hvm.spice.disable_ticketing, false);
|
|
|
|
} else {
|
|
|
|
libxl_defbool_set(&b_info->u.hvm.spice.disable_ticketing, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (l_vfb->data.spice.mousemode) {
|
|
|
|
/* client mouse mode is default in xl.cfg */
|
|
|
|
case VIR_DOMAIN_GRAPHICS_SPICE_MOUSE_MODE_DEFAULT:
|
|
|
|
case VIR_DOMAIN_GRAPHICS_SPICE_MOUSE_MODE_CLIENT:
|
|
|
|
libxl_defbool_set(&b_info->u.hvm.spice.agent_mouse, true);
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_GRAPHICS_SPICE_MOUSE_MODE_SERVER:
|
|
|
|
libxl_defbool_set(&b_info->u.hvm.spice.agent_mouse, false);
|
|
|
|
break;
|
2016-05-02 17:45:23 +02:00
|
|
|
case VIR_DOMAIN_GRAPHICS_SPICE_MOUSE_MODE_LAST:
|
|
|
|
break;
|
2015-04-27 16:27:52 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
if (l_vfb->data.spice.copypaste == VIR_TRISTATE_BOOL_YES) {
|
|
|
|
libxl_defbool_set(&b_info->u.hvm.spice.vdagent, true);
|
|
|
|
libxl_defbool_set(&b_info->u.hvm.spice.clipboard_sharing, true);
|
|
|
|
} else {
|
|
|
|
libxl_defbool_set(&b_info->u.hvm.spice.vdagent, false);
|
|
|
|
libxl_defbool_set(&b_info->u.hvm.spice.clipboard_sharing, false);
|
|
|
|
}
|
|
|
|
|
2015-04-27 12:32:55 -06:00
|
|
|
return 0;
|
2015-04-27 16:27:52 -06:00
|
|
|
}
|
2015-04-27 12:32:55 -06:00
|
|
|
|
|
|
|
x_vfb = d_config->vfbs[0];
|
|
|
|
|
|
|
|
if (libxl_defbool_val(x_vfb.vnc.enable)) {
|
|
|
|
libxl_defbool_set(&b_info->u.hvm.vnc.enable, true);
|
2019-10-20 13:49:46 +02:00
|
|
|
b_info->u.hvm.vnc.listen = g_strdup(x_vfb.vnc.listen);
|
|
|
|
b_info->u.hvm.vnc.passwd = g_strdup(x_vfb.vnc.passwd);
|
2015-04-27 12:32:55 -06:00
|
|
|
b_info->u.hvm.vnc.display = x_vfb.vnc.display;
|
|
|
|
libxl_defbool_set(&b_info->u.hvm.vnc.findunused,
|
|
|
|
libxl_defbool_val(x_vfb.vnc.findunused));
|
|
|
|
} else if (libxl_defbool_val(x_vfb.sdl.enable)) {
|
|
|
|
libxl_defbool_set(&b_info->u.hvm.sdl.enable, true);
|
|
|
|
libxl_defbool_set(&b_info->u.hvm.sdl.opengl,
|
|
|
|
libxl_defbool_val(x_vfb.sdl.opengl));
|
2019-10-20 13:49:46 +02:00
|
|
|
b_info->u.hvm.sdl.display = g_strdup(x_vfb.sdl.display);
|
|
|
|
b_info->u.hvm.sdl.xauthority = g_strdup(x_vfb.sdl.xauthority);
|
2015-04-27 12:32:55 -06:00
|
|
|
}
|
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
b_info->u.hvm.keymap = g_strdup(x_vfb.keymap);
|
2015-04-27 12:32:55 -06:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-09-09 09:15:15 -06:00
|
|
|
/*
|
|
|
|
* Get domain0 autoballoon configuration. Honor user-specified
|
|
|
|
* setting in libxl.conf first. If not specified, autoballooning
|
|
|
|
* is disabled when domain0's memory is set with 'dom0_mem'.
|
|
|
|
* Otherwise autoballooning is enabled.
|
|
|
|
*/
|
2013-09-04 00:21:42 -06:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlGetAutoballoonConf(libxlDriverConfig *cfg,
|
|
|
|
virConf *conf)
|
2013-08-30 11:11:08 -06:00
|
|
|
{
|
2019-11-13 12:59:55 +01:00
|
|
|
g_autoptr(GRegex) regex = NULL;
|
|
|
|
g_autoptr(GError) err = NULL;
|
2013-09-04 00:21:42 -06:00
|
|
|
int res;
|
|
|
|
|
2017-01-30 11:43:15 -07:00
|
|
|
res = virConfGetValueBool(conf, "autoballoon", &cfg->autoballoon);
|
|
|
|
if (res < 0)
|
2016-07-08 13:52:04 +01:00
|
|
|
return -1;
|
2017-01-30 11:43:15 -07:00
|
|
|
else if (res == 1)
|
|
|
|
return 0;
|
2013-09-09 09:15:15 -06:00
|
|
|
|
2019-11-13 12:59:55 +01:00
|
|
|
regex = g_regex_new("(^| )dom0_mem=((|min:|max:)[0-9]+[bBkKmMgG]?,?)+($| )",
|
|
|
|
0, 0, &err);
|
|
|
|
if (!regex) {
|
2013-09-04 00:21:42 -06:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
2019-11-13 12:59:55 +01:00
|
|
|
_("Failed to compile regex %s"), err->message);
|
2013-09-04 00:21:42 -06:00
|
|
|
return -1;
|
|
|
|
}
|
2013-08-30 11:11:08 -06:00
|
|
|
|
2019-11-13 12:59:55 +01:00
|
|
|
cfg->autoballoon = !g_regex_match(regex, cfg->verInfo->commandline, 0, NULL);
|
2013-09-04 00:21:42 -06:00
|
|
|
return 0;
|
2013-08-30 11:11:08 -06:00
|
|
|
}
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlDriverConfig *
|
2013-08-30 14:52:14 -06:00
|
|
|
libxlDriverConfigNew(void)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlDriverConfig *cfg;
|
2013-08-30 14:52:14 -06:00
|
|
|
|
|
|
|
if (libxlConfigInitialize() < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (!(cfg = virObjectNew(libxlDriverConfigClass)))
|
|
|
|
return NULL;
|
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
cfg->configBaseDir = g_strdup(LIBXL_CONFIG_BASE_DIR);
|
|
|
|
cfg->configDir = g_strdup(LIBXL_CONFIG_DIR);
|
|
|
|
cfg->autostartDir = g_strdup(LIBXL_AUTOSTART_DIR);
|
|
|
|
cfg->logDir = g_strdup(LIBXL_LOG_DIR);
|
|
|
|
cfg->stateDir = g_strdup(LIBXL_STATE_DIR);
|
|
|
|
cfg->libDir = g_strdup(LIBXL_LIB_DIR);
|
|
|
|
cfg->saveDir = g_strdup(LIBXL_SAVE_DIR);
|
|
|
|
cfg->autoDumpDir = g_strdup(LIBXL_DUMP_DIR);
|
|
|
|
cfg->channelDir = g_strdup(LIBXL_CHANNEL_DIR);
|
2013-08-30 14:52:14 -06:00
|
|
|
|
2016-05-18 08:24:37 -06:00
|
|
|
#ifdef DEFAULT_LOADER_NVRAM
|
|
|
|
if (virFirmwareParseList(DEFAULT_LOADER_NVRAM,
|
|
|
|
&cfg->firmwares,
|
2021-03-20 00:37:05 +01:00
|
|
|
&cfg->nfirmwares) < 0) {
|
|
|
|
virObjectUnref(cfg);
|
|
|
|
return NULL;
|
|
|
|
}
|
2016-05-18 08:24:37 -06:00
|
|
|
#else
|
2021-03-11 08:16:13 +01:00
|
|
|
cfg->firmwares = g_new0(virFirmware *, 1);
|
2016-05-18 08:24:37 -06:00
|
|
|
cfg->nfirmwares = 1;
|
2020-09-23 20:43:09 +02:00
|
|
|
cfg->firmwares[0] = g_new0(virFirmware, 1);
|
2019-10-20 13:49:46 +02:00
|
|
|
cfg->firmwares[0]->name = g_strdup(LIBXL_FIRMWARE_DIR "/ovmf.bin");
|
2016-05-18 08:24:37 -06:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Always add hvmloader to firmwares */
|
2021-03-20 00:37:05 +01:00
|
|
|
VIR_REALLOC_N(cfg->firmwares, cfg->nfirmwares + 1);
|
2016-05-18 08:24:37 -06:00
|
|
|
cfg->nfirmwares++;
|
2020-09-23 20:43:09 +02:00
|
|
|
cfg->firmwares[cfg->nfirmwares - 1] = g_new0(virFirmware, 1);
|
2019-10-20 13:49:46 +02:00
|
|
|
cfg->firmwares[cfg->nfirmwares - 1]->name = g_strdup(LIBXL_FIRMWARE_DIR "/hvmloader");
|
2016-05-18 08:24:37 -06:00
|
|
|
|
2020-02-22 13:02:18 +01:00
|
|
|
/* defaults for keepalive messages */
|
|
|
|
cfg->keepAliveInterval = 5;
|
|
|
|
cfg->keepAliveCount = 5;
|
|
|
|
|
2013-08-30 14:52:14 -06:00
|
|
|
return cfg;
|
|
|
|
}
|
|
|
|
|
2020-02-22 13:01:38 +01:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlDriverConfigInit(libxlDriverConfig *cfg)
|
2020-02-22 13:01:38 +01:00
|
|
|
{
|
2021-03-25 17:26:08 +01:00
|
|
|
uint64_t free_mem;
|
2020-02-22 13:01:38 +01:00
|
|
|
|
2021-02-26 09:37:10 +01:00
|
|
|
if (g_mkdir_with_parents(cfg->logDir, 0777) < 0) {
|
2020-02-22 13:01:38 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("failed to create log dir '%s': %s"),
|
|
|
|
cfg->logDir,
|
2020-02-26 18:57:34 +01:00
|
|
|
g_strerror(errno));
|
2020-02-22 13:01:38 +01:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
cfg->logger = libxlLoggerNew(cfg->logDir, virLogGetDefaultPriority());
|
|
|
|
if (!cfg->logger) {
|
|
|
|
VIR_ERROR(_("cannot create logger for libxenlight, disabling driver"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (libxl_ctx_alloc(&cfg->ctx, LIBXL_VERSION, 0, (xentoollog_logger *)cfg->logger)) {
|
|
|
|
VIR_ERROR(_("cannot initialize libxenlight context, probably not "
|
|
|
|
"running in a Xen Dom0, disabling driver"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((cfg->verInfo = libxl_get_version_info(cfg->ctx)) == NULL) {
|
|
|
|
VIR_ERROR(_("cannot version information from libxenlight, "
|
|
|
|
"disabling driver"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
cfg->version = (cfg->verInfo->xen_version_major * 1000000) +
|
|
|
|
(cfg->verInfo->xen_version_minor * 1000);
|
|
|
|
|
|
|
|
/* This will fill xenstore info about free and dom0 memory if missing,
|
|
|
|
* should be called before starting first domain */
|
2021-03-25 17:26:08 +01:00
|
|
|
if (libxlGetFreeMemoryWrapper(cfg->ctx, &free_mem)) {
|
2020-02-22 13:01:38 +01:00
|
|
|
VIR_ERROR(_("Unable to configure libxl's memory management parameters"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlDriverConfig *
|
|
|
|
libxlDriverConfigGet(libxlDriverPrivate *driver)
|
2013-08-30 14:52:14 -06:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlDriverConfig *cfg;
|
2013-08-30 14:57:42 -06:00
|
|
|
|
|
|
|
libxlDriverLock(driver);
|
|
|
|
cfg = virObjectRef(driver->config);
|
|
|
|
libxlDriverUnlock(driver);
|
|
|
|
return cfg;
|
2013-08-30 14:52:14 -06:00
|
|
|
}
|
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
int libxlDriverConfigLoadFile(libxlDriverConfig *cfg,
|
2013-09-09 09:15:15 -06:00
|
|
|
const char *filename)
|
|
|
|
{
|
2019-10-15 14:47:50 +02:00
|
|
|
g_autoptr(virConf) conf = NULL;
|
2013-09-09 09:15:15 -06:00
|
|
|
|
|
|
|
/* Check the file is readable before opening it, otherwise
|
|
|
|
* libvirt emits an error.
|
|
|
|
*/
|
|
|
|
if (access(filename, R_OK) == -1) {
|
|
|
|
VIR_INFO("Could not read libxl config file %s", filename);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(conf = virConfReadFile(filename, 0)))
|
2019-09-09 17:56:26 +02:00
|
|
|
return -1;
|
2013-09-09 09:15:15 -06:00
|
|
|
|
|
|
|
/* setup autoballoon */
|
|
|
|
if (libxlGetAutoballoonConf(cfg, conf) < 0)
|
2019-09-09 17:56:26 +02:00
|
|
|
return -1;
|
2013-09-09 09:15:15 -06:00
|
|
|
|
2016-07-08 13:52:04 +01:00
|
|
|
if (virConfGetValueString(conf, "lock_manager", &cfg->lockManagerName) < 0)
|
2019-09-09 17:56:26 +02:00
|
|
|
return -1;
|
2016-02-05 20:45:03 +00:00
|
|
|
|
2016-07-08 13:52:04 +01:00
|
|
|
if (virConfGetValueInt(conf, "keepalive_interval", &cfg->keepAliveInterval) < 0)
|
2019-09-09 17:56:26 +02:00
|
|
|
return -1;
|
2016-02-05 20:45:03 +00:00
|
|
|
|
2016-07-08 13:52:04 +01:00
|
|
|
if (virConfGetValueUInt(conf, "keepalive_count", &cfg->keepAliveCount) < 0)
|
2019-09-09 17:56:26 +02:00
|
|
|
return -1;
|
2016-02-05 20:45:03 +00:00
|
|
|
|
2018-04-12 03:03:22 +02:00
|
|
|
if (virConfGetValueBool(conf, "nested_hvm", &cfg->nested_hvm) < 0)
|
2019-09-09 17:56:26 +02:00
|
|
|
return -1;
|
2013-09-09 09:15:15 -06:00
|
|
|
|
2019-09-09 17:56:26 +02:00
|
|
|
return 0;
|
2013-09-09 09:15:15 -06:00
|
|
|
}
|
|
|
|
|
2017-01-31 17:10:34 -07:00
|
|
|
/*
|
2020-07-09 12:42:21 +08:00
|
|
|
* dom0's maximum memory can be controlled by the user with the 'dom0_mem' Xen
|
2017-01-31 17:10:34 -07:00
|
|
|
* command line parameter. E.g. to set dom0's initial memory to 4G and max
|
|
|
|
* memory to 8G: dom0_mem=4G,max:8G
|
|
|
|
* Supported unit suffixes are [bBkKmMgGtT]. If not specified the default
|
|
|
|
* unit is kilobytes.
|
|
|
|
*
|
|
|
|
* If not constrained by the user, dom0 can effectively use all host memory.
|
|
|
|
* This function returns the configured maximum memory for dom0 in kilobytes,
|
|
|
|
* either the user-specified value or total physical memory as a default.
|
|
|
|
*/
|
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlDriverGetDom0MaxmemConf(libxlDriverConfig *cfg,
|
2017-01-31 17:10:34 -07:00
|
|
|
unsigned long long *maxmem)
|
|
|
|
{
|
|
|
|
char **cmd_tokens = NULL;
|
|
|
|
char **mem_tokens = NULL;
|
|
|
|
size_t i;
|
|
|
|
size_t j;
|
|
|
|
libxl_physinfo physinfo;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
if (cfg->verInfo->commandline == NULL ||
|
2021-02-05 18:35:07 +01:00
|
|
|
!(cmd_tokens = g_strsplit(cfg->verInfo->commandline, " ", 0)))
|
2017-01-31 17:10:34 -07:00
|
|
|
goto physmem;
|
|
|
|
|
|
|
|
for (i = 0; cmd_tokens[i] != NULL; i++) {
|
|
|
|
if (!STRPREFIX(cmd_tokens[i], "dom0_mem="))
|
|
|
|
continue;
|
|
|
|
|
2021-02-05 18:35:07 +01:00
|
|
|
if (!(mem_tokens = g_strsplit(cmd_tokens[i], ",", 0)))
|
2017-01-31 17:10:34 -07:00
|
|
|
break;
|
|
|
|
for (j = 0; mem_tokens[j] != NULL; j++) {
|
|
|
|
if (STRPREFIX(mem_tokens[j], "max:")) {
|
|
|
|
char *p = mem_tokens[j] + 4;
|
|
|
|
unsigned long long multiplier = 1;
|
|
|
|
|
2019-11-18 15:13:11 +01:00
|
|
|
while (g_ascii_isdigit(*p))
|
2017-01-31 17:10:34 -07:00
|
|
|
p++;
|
|
|
|
if (virStrToLong_ull(mem_tokens[j] + 4, &p, 10, maxmem) < 0)
|
|
|
|
break;
|
|
|
|
if (*p) {
|
|
|
|
switch (*p) {
|
|
|
|
case 'm':
|
|
|
|
case 'M':
|
|
|
|
multiplier = 1024;
|
|
|
|
break;
|
|
|
|
case 'g':
|
|
|
|
case 'G':
|
|
|
|
multiplier = 1024 * 1024;
|
|
|
|
break;
|
|
|
|
case 't':
|
|
|
|
case 'T':
|
|
|
|
multiplier = 1024 * 1024 * 1024;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*maxmem = *maxmem * multiplier;
|
|
|
|
ret = 0;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
2020-08-02 19:36:03 +02:00
|
|
|
g_strfreev(mem_tokens);
|
2017-02-15 10:45:27 -07:00
|
|
|
mem_tokens = NULL;
|
2017-01-31 17:10:34 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
physmem:
|
|
|
|
/* No 'max' specified in dom0_mem, so dom0 can use all physical memory */
|
|
|
|
libxl_physinfo_init(&physinfo);
|
|
|
|
if (libxl_get_physinfo(cfg->ctx, &physinfo)) {
|
|
|
|
VIR_WARN("libxl_get_physinfo failed");
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
*maxmem = (physinfo.total_pages * cfg->verInfo->pagesize) / 1024;
|
|
|
|
libxl_physinfo_dispose(&physinfo);
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
2020-08-02 19:36:03 +02:00
|
|
|
g_strfreev(cmd_tokens);
|
|
|
|
g_strfreev(mem_tokens);
|
2017-01-31 17:10:34 -07:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
libxl: channels support
And allow libxl to handle channel element which creates a Xen
console visible to the guest as a low-bandwitdh communication
channel. If type is PTY we also fetch the tty after boot using
libxl_channel_getinfo to fetch the tty path. On socket case,
we autogenerate a path if not specified in the XML. Path autogenerated
is slightly different from qemu driver: qemu stores also on
"channels/target" but it creates then a directory per domain with
each channel target name. libxl doesn't appear to have a clear
definition of private files associated with each domain, so for
simplicity we do it slightly different. On qemu each autogenerated
channel goes like:
channels/target/<domain-name>/<target name>
Whereas for libxl:
channels/target/<domain-name>-<target name>
Should note that if path is not specified it won't persist,
existing only on live XML, unless user had initially specified it.
Since support for libxl channels only came on Xen >= 4.5 we therefore
need to conditionally compile it with LIBXL_HAVE_DEVICE_CHANNEL.
After this patch and having a qemu guest agent:
$ cat domain.xml | grep -a1 channel | head -n 5 | tail -n 4
<channel type='unix'>
<source mode='bind' path='/tmp/channel'/>
<target type='xen' name='org.qemu.guest_agent.0'/>
</channel>
$ virsh create domain.xml
$ echo '{"execute":"guest-network-get-interfaces"}' | socat
stdio,ignoreeof unix-connect:/tmp/channel
{"execute":"guest-network-get-interfaces"}
{"return": [{"name": "lo", "ip-addresses": [{"ip-address-type": "ipv4",
"ip-address": "127.0.0.1", "prefix": 8}, {"ip-address-type": "ipv6",
"ip-address": "::1", "prefix": 128}], "hardware-address":
"00:00:00:00:00:00"}, {"name": "eth0", "ip-addresses":
[{"ip-address-type": "ipv4", "ip-address": "10.100.0.6", "prefix": 24},
{"ip-address-type": "ipv6", "ip-address": "fe80::216:3eff:fe40:88eb",
"prefix": 64}], "hardware-address": "00:16:3e:40:88:eb"}, {"name":
"sit0"}]}
Signed-off-by: Joao Martins <joao.m.martins@oracle.com>
Signed-off-by: Jim Fehlig <jfehlig@suse.com>
2016-09-26 18:33:16 +01:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlPrepareChannel(virDomainChrDef *channel,
|
libxl: channels support
And allow libxl to handle channel element which creates a Xen
console visible to the guest as a low-bandwitdh communication
channel. If type is PTY we also fetch the tty after boot using
libxl_channel_getinfo to fetch the tty path. On socket case,
we autogenerate a path if not specified in the XML. Path autogenerated
is slightly different from qemu driver: qemu stores also on
"channels/target" but it creates then a directory per domain with
each channel target name. libxl doesn't appear to have a clear
definition of private files associated with each domain, so for
simplicity we do it slightly different. On qemu each autogenerated
channel goes like:
channels/target/<domain-name>/<target name>
Whereas for libxl:
channels/target/<domain-name>-<target name>
Should note that if path is not specified it won't persist,
existing only on live XML, unless user had initially specified it.
Since support for libxl channels only came on Xen >= 4.5 we therefore
need to conditionally compile it with LIBXL_HAVE_DEVICE_CHANNEL.
After this patch and having a qemu guest agent:
$ cat domain.xml | grep -a1 channel | head -n 5 | tail -n 4
<channel type='unix'>
<source mode='bind' path='/tmp/channel'/>
<target type='xen' name='org.qemu.guest_agent.0'/>
</channel>
$ virsh create domain.xml
$ echo '{"execute":"guest-network-get-interfaces"}' | socat
stdio,ignoreeof unix-connect:/tmp/channel
{"execute":"guest-network-get-interfaces"}
{"return": [{"name": "lo", "ip-addresses": [{"ip-address-type": "ipv4",
"ip-address": "127.0.0.1", "prefix": 8}, {"ip-address-type": "ipv6",
"ip-address": "::1", "prefix": 128}], "hardware-address":
"00:00:00:00:00:00"}, {"name": "eth0", "ip-addresses":
[{"ip-address-type": "ipv4", "ip-address": "10.100.0.6", "prefix": 24},
{"ip-address-type": "ipv6", "ip-address": "fe80::216:3eff:fe40:88eb",
"prefix": 64}], "hardware-address": "00:16:3e:40:88:eb"}, {"name":
"sit0"}]}
Signed-off-by: Joao Martins <joao.m.martins@oracle.com>
Signed-off-by: Jim Fehlig <jfehlig@suse.com>
2016-09-26 18:33:16 +01:00
|
|
|
const char *channelDir,
|
|
|
|
const char *domainName)
|
|
|
|
{
|
|
|
|
if (channel->targetType == VIR_DOMAIN_CHR_CHANNEL_TARGET_TYPE_XEN &&
|
2016-10-21 07:45:54 -04:00
|
|
|
channel->source->type == VIR_DOMAIN_CHR_TYPE_UNIX &&
|
|
|
|
!channel->source->data.nix.path) {
|
2019-10-22 15:26:14 +02:00
|
|
|
const char *target = channel->target.name;
|
|
|
|
if (!target)
|
|
|
|
target = "unknown.sock";
|
|
|
|
channel->source->data.nix.path = g_strdup_printf("%s/%s-%s", channelDir,
|
|
|
|
domainName,
|
|
|
|
target);
|
libxl: channels support
And allow libxl to handle channel element which creates a Xen
console visible to the guest as a low-bandwitdh communication
channel. If type is PTY we also fetch the tty after boot using
libxl_channel_getinfo to fetch the tty path. On socket case,
we autogenerate a path if not specified in the XML. Path autogenerated
is slightly different from qemu driver: qemu stores also on
"channels/target" but it creates then a directory per domain with
each channel target name. libxl doesn't appear to have a clear
definition of private files associated with each domain, so for
simplicity we do it slightly different. On qemu each autogenerated
channel goes like:
channels/target/<domain-name>/<target name>
Whereas for libxl:
channels/target/<domain-name>-<target name>
Should note that if path is not specified it won't persist,
existing only on live XML, unless user had initially specified it.
Since support for libxl channels only came on Xen >= 4.5 we therefore
need to conditionally compile it with LIBXL_HAVE_DEVICE_CHANNEL.
After this patch and having a qemu guest agent:
$ cat domain.xml | grep -a1 channel | head -n 5 | tail -n 4
<channel type='unix'>
<source mode='bind' path='/tmp/channel'/>
<target type='xen' name='org.qemu.guest_agent.0'/>
</channel>
$ virsh create domain.xml
$ echo '{"execute":"guest-network-get-interfaces"}' | socat
stdio,ignoreeof unix-connect:/tmp/channel
{"execute":"guest-network-get-interfaces"}
{"return": [{"name": "lo", "ip-addresses": [{"ip-address-type": "ipv4",
"ip-address": "127.0.0.1", "prefix": 8}, {"ip-address-type": "ipv6",
"ip-address": "::1", "prefix": 128}], "hardware-address":
"00:00:00:00:00:00"}, {"name": "eth0", "ip-addresses":
[{"ip-address-type": "ipv4", "ip-address": "10.100.0.6", "prefix": 24},
{"ip-address-type": "ipv6", "ip-address": "fe80::216:3eff:fe40:88eb",
"prefix": 64}], "hardware-address": "00:16:3e:40:88:eb"}, {"name":
"sit0"}]}
Signed-off-by: Joao Martins <joao.m.martins@oracle.com>
Signed-off-by: Jim Fehlig <jfehlig@suse.com>
2016-09-26 18:33:16 +01:00
|
|
|
|
2016-10-21 07:45:54 -04:00
|
|
|
channel->source->data.nix.listen = true;
|
libxl: channels support
And allow libxl to handle channel element which creates a Xen
console visible to the guest as a low-bandwitdh communication
channel. If type is PTY we also fetch the tty after boot using
libxl_channel_getinfo to fetch the tty path. On socket case,
we autogenerate a path if not specified in the XML. Path autogenerated
is slightly different from qemu driver: qemu stores also on
"channels/target" but it creates then a directory per domain with
each channel target name. libxl doesn't appear to have a clear
definition of private files associated with each domain, so for
simplicity we do it slightly different. On qemu each autogenerated
channel goes like:
channels/target/<domain-name>/<target name>
Whereas for libxl:
channels/target/<domain-name>-<target name>
Should note that if path is not specified it won't persist,
existing only on live XML, unless user had initially specified it.
Since support for libxl channels only came on Xen >= 4.5 we therefore
need to conditionally compile it with LIBXL_HAVE_DEVICE_CHANNEL.
After this patch and having a qemu guest agent:
$ cat domain.xml | grep -a1 channel | head -n 5 | tail -n 4
<channel type='unix'>
<source mode='bind' path='/tmp/channel'/>
<target type='xen' name='org.qemu.guest_agent.0'/>
</channel>
$ virsh create domain.xml
$ echo '{"execute":"guest-network-get-interfaces"}' | socat
stdio,ignoreeof unix-connect:/tmp/channel
{"execute":"guest-network-get-interfaces"}
{"return": [{"name": "lo", "ip-addresses": [{"ip-address-type": "ipv4",
"ip-address": "127.0.0.1", "prefix": 8}, {"ip-address-type": "ipv6",
"ip-address": "::1", "prefix": 128}], "hardware-address":
"00:00:00:00:00:00"}, {"name": "eth0", "ip-addresses":
[{"ip-address-type": "ipv4", "ip-address": "10.100.0.6", "prefix": 24},
{"ip-address-type": "ipv6", "ip-address": "fe80::216:3eff:fe40:88eb",
"prefix": 64}], "hardware-address": "00:16:3e:40:88:eb"}, {"name":
"sit0"}]}
Signed-off-by: Joao Martins <joao.m.martins@oracle.com>
Signed-off-by: Jim Fehlig <jfehlig@suse.com>
2016-09-26 18:33:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlMakeChannel(virDomainChrDef *l_channel,
|
libxl: channels support
And allow libxl to handle channel element which creates a Xen
console visible to the guest as a low-bandwitdh communication
channel. If type is PTY we also fetch the tty after boot using
libxl_channel_getinfo to fetch the tty path. On socket case,
we autogenerate a path if not specified in the XML. Path autogenerated
is slightly different from qemu driver: qemu stores also on
"channels/target" but it creates then a directory per domain with
each channel target name. libxl doesn't appear to have a clear
definition of private files associated with each domain, so for
simplicity we do it slightly different. On qemu each autogenerated
channel goes like:
channels/target/<domain-name>/<target name>
Whereas for libxl:
channels/target/<domain-name>-<target name>
Should note that if path is not specified it won't persist,
existing only on live XML, unless user had initially specified it.
Since support for libxl channels only came on Xen >= 4.5 we therefore
need to conditionally compile it with LIBXL_HAVE_DEVICE_CHANNEL.
After this patch and having a qemu guest agent:
$ cat domain.xml | grep -a1 channel | head -n 5 | tail -n 4
<channel type='unix'>
<source mode='bind' path='/tmp/channel'/>
<target type='xen' name='org.qemu.guest_agent.0'/>
</channel>
$ virsh create domain.xml
$ echo '{"execute":"guest-network-get-interfaces"}' | socat
stdio,ignoreeof unix-connect:/tmp/channel
{"execute":"guest-network-get-interfaces"}
{"return": [{"name": "lo", "ip-addresses": [{"ip-address-type": "ipv4",
"ip-address": "127.0.0.1", "prefix": 8}, {"ip-address-type": "ipv6",
"ip-address": "::1", "prefix": 128}], "hardware-address":
"00:00:00:00:00:00"}, {"name": "eth0", "ip-addresses":
[{"ip-address-type": "ipv4", "ip-address": "10.100.0.6", "prefix": 24},
{"ip-address-type": "ipv6", "ip-address": "fe80::216:3eff:fe40:88eb",
"prefix": 64}], "hardware-address": "00:16:3e:40:88:eb"}, {"name":
"sit0"}]}
Signed-off-by: Joao Martins <joao.m.martins@oracle.com>
Signed-off-by: Jim Fehlig <jfehlig@suse.com>
2016-09-26 18:33:16 +01:00
|
|
|
libxl_device_channel *x_channel)
|
|
|
|
{
|
|
|
|
libxl_device_channel_init(x_channel);
|
|
|
|
|
|
|
|
if (l_channel->targetType != VIR_DOMAIN_CHR_CHANNEL_TARGET_TYPE_XEN) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("channel target type not supported"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2016-10-21 07:45:54 -04:00
|
|
|
switch (l_channel->source->type) {
|
libxl: channels support
And allow libxl to handle channel element which creates a Xen
console visible to the guest as a low-bandwitdh communication
channel. If type is PTY we also fetch the tty after boot using
libxl_channel_getinfo to fetch the tty path. On socket case,
we autogenerate a path if not specified in the XML. Path autogenerated
is slightly different from qemu driver: qemu stores also on
"channels/target" but it creates then a directory per domain with
each channel target name. libxl doesn't appear to have a clear
definition of private files associated with each domain, so for
simplicity we do it slightly different. On qemu each autogenerated
channel goes like:
channels/target/<domain-name>/<target name>
Whereas for libxl:
channels/target/<domain-name>-<target name>
Should note that if path is not specified it won't persist,
existing only on live XML, unless user had initially specified it.
Since support for libxl channels only came on Xen >= 4.5 we therefore
need to conditionally compile it with LIBXL_HAVE_DEVICE_CHANNEL.
After this patch and having a qemu guest agent:
$ cat domain.xml | grep -a1 channel | head -n 5 | tail -n 4
<channel type='unix'>
<source mode='bind' path='/tmp/channel'/>
<target type='xen' name='org.qemu.guest_agent.0'/>
</channel>
$ virsh create domain.xml
$ echo '{"execute":"guest-network-get-interfaces"}' | socat
stdio,ignoreeof unix-connect:/tmp/channel
{"execute":"guest-network-get-interfaces"}
{"return": [{"name": "lo", "ip-addresses": [{"ip-address-type": "ipv4",
"ip-address": "127.0.0.1", "prefix": 8}, {"ip-address-type": "ipv6",
"ip-address": "::1", "prefix": 128}], "hardware-address":
"00:00:00:00:00:00"}, {"name": "eth0", "ip-addresses":
[{"ip-address-type": "ipv4", "ip-address": "10.100.0.6", "prefix": 24},
{"ip-address-type": "ipv6", "ip-address": "fe80::216:3eff:fe40:88eb",
"prefix": 64}], "hardware-address": "00:16:3e:40:88:eb"}, {"name":
"sit0"}]}
Signed-off-by: Joao Martins <joao.m.martins@oracle.com>
Signed-off-by: Jim Fehlig <jfehlig@suse.com>
2016-09-26 18:33:16 +01:00
|
|
|
case VIR_DOMAIN_CHR_TYPE_PTY:
|
|
|
|
x_channel->connection = LIBXL_CHANNEL_CONNECTION_PTY;
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_CHR_TYPE_UNIX:
|
|
|
|
x_channel->connection = LIBXL_CHANNEL_CONNECTION_SOCKET;
|
2019-10-20 13:49:46 +02:00
|
|
|
x_channel->u.socket.path = g_strdup(l_channel->source->data.nix.path);
|
libxl: channels support
And allow libxl to handle channel element which creates a Xen
console visible to the guest as a low-bandwitdh communication
channel. If type is PTY we also fetch the tty after boot using
libxl_channel_getinfo to fetch the tty path. On socket case,
we autogenerate a path if not specified in the XML. Path autogenerated
is slightly different from qemu driver: qemu stores also on
"channels/target" but it creates then a directory per domain with
each channel target name. libxl doesn't appear to have a clear
definition of private files associated with each domain, so for
simplicity we do it slightly different. On qemu each autogenerated
channel goes like:
channels/target/<domain-name>/<target name>
Whereas for libxl:
channels/target/<domain-name>-<target name>
Should note that if path is not specified it won't persist,
existing only on live XML, unless user had initially specified it.
Since support for libxl channels only came on Xen >= 4.5 we therefore
need to conditionally compile it with LIBXL_HAVE_DEVICE_CHANNEL.
After this patch and having a qemu guest agent:
$ cat domain.xml | grep -a1 channel | head -n 5 | tail -n 4
<channel type='unix'>
<source mode='bind' path='/tmp/channel'/>
<target type='xen' name='org.qemu.guest_agent.0'/>
</channel>
$ virsh create domain.xml
$ echo '{"execute":"guest-network-get-interfaces"}' | socat
stdio,ignoreeof unix-connect:/tmp/channel
{"execute":"guest-network-get-interfaces"}
{"return": [{"name": "lo", "ip-addresses": [{"ip-address-type": "ipv4",
"ip-address": "127.0.0.1", "prefix": 8}, {"ip-address-type": "ipv6",
"ip-address": "::1", "prefix": 128}], "hardware-address":
"00:00:00:00:00:00"}, {"name": "eth0", "ip-addresses":
[{"ip-address-type": "ipv4", "ip-address": "10.100.0.6", "prefix": 24},
{"ip-address-type": "ipv6", "ip-address": "fe80::216:3eff:fe40:88eb",
"prefix": 64}], "hardware-address": "00:16:3e:40:88:eb"}, {"name":
"sit0"}]}
Signed-off-by: Joao Martins <joao.m.martins@oracle.com>
Signed-off-by: Jim Fehlig <jfehlig@suse.com>
2016-09-26 18:33:16 +01:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("channel source type not supported"));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!l_channel->target.name) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID, "%s",
|
|
|
|
_("channel target name missing"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2019-10-20 13:49:46 +02:00
|
|
|
x_channel->name = g_strdup(l_channel->target.name);
|
libxl: channels support
And allow libxl to handle channel element which creates a Xen
console visible to the guest as a low-bandwitdh communication
channel. If type is PTY we also fetch the tty after boot using
libxl_channel_getinfo to fetch the tty path. On socket case,
we autogenerate a path if not specified in the XML. Path autogenerated
is slightly different from qemu driver: qemu stores also on
"channels/target" but it creates then a directory per domain with
each channel target name. libxl doesn't appear to have a clear
definition of private files associated with each domain, so for
simplicity we do it slightly different. On qemu each autogenerated
channel goes like:
channels/target/<domain-name>/<target name>
Whereas for libxl:
channels/target/<domain-name>-<target name>
Should note that if path is not specified it won't persist,
existing only on live XML, unless user had initially specified it.
Since support for libxl channels only came on Xen >= 4.5 we therefore
need to conditionally compile it with LIBXL_HAVE_DEVICE_CHANNEL.
After this patch and having a qemu guest agent:
$ cat domain.xml | grep -a1 channel | head -n 5 | tail -n 4
<channel type='unix'>
<source mode='bind' path='/tmp/channel'/>
<target type='xen' name='org.qemu.guest_agent.0'/>
</channel>
$ virsh create domain.xml
$ echo '{"execute":"guest-network-get-interfaces"}' | socat
stdio,ignoreeof unix-connect:/tmp/channel
{"execute":"guest-network-get-interfaces"}
{"return": [{"name": "lo", "ip-addresses": [{"ip-address-type": "ipv4",
"ip-address": "127.0.0.1", "prefix": 8}, {"ip-address-type": "ipv6",
"ip-address": "::1", "prefix": 128}], "hardware-address":
"00:00:00:00:00:00"}, {"name": "eth0", "ip-addresses":
[{"ip-address-type": "ipv4", "ip-address": "10.100.0.6", "prefix": 24},
{"ip-address-type": "ipv6", "ip-address": "fe80::216:3eff:fe40:88eb",
"prefix": 64}], "hardware-address": "00:16:3e:40:88:eb"}, {"name":
"sit0"}]}
Signed-off-by: Joao Martins <joao.m.martins@oracle.com>
Signed-off-by: Jim Fehlig <jfehlig@suse.com>
2016-09-26 18:33:16 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
libxlMakeChannelList(const char *channelDir,
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainDef *def,
|
libxl: channels support
And allow libxl to handle channel element which creates a Xen
console visible to the guest as a low-bandwitdh communication
channel. If type is PTY we also fetch the tty after boot using
libxl_channel_getinfo to fetch the tty path. On socket case,
we autogenerate a path if not specified in the XML. Path autogenerated
is slightly different from qemu driver: qemu stores also on
"channels/target" but it creates then a directory per domain with
each channel target name. libxl doesn't appear to have a clear
definition of private files associated with each domain, so for
simplicity we do it slightly different. On qemu each autogenerated
channel goes like:
channels/target/<domain-name>/<target name>
Whereas for libxl:
channels/target/<domain-name>-<target name>
Should note that if path is not specified it won't persist,
existing only on live XML, unless user had initially specified it.
Since support for libxl channels only came on Xen >= 4.5 we therefore
need to conditionally compile it with LIBXL_HAVE_DEVICE_CHANNEL.
After this patch and having a qemu guest agent:
$ cat domain.xml | grep -a1 channel | head -n 5 | tail -n 4
<channel type='unix'>
<source mode='bind' path='/tmp/channel'/>
<target type='xen' name='org.qemu.guest_agent.0'/>
</channel>
$ virsh create domain.xml
$ echo '{"execute":"guest-network-get-interfaces"}' | socat
stdio,ignoreeof unix-connect:/tmp/channel
{"execute":"guest-network-get-interfaces"}
{"return": [{"name": "lo", "ip-addresses": [{"ip-address-type": "ipv4",
"ip-address": "127.0.0.1", "prefix": 8}, {"ip-address-type": "ipv6",
"ip-address": "::1", "prefix": 128}], "hardware-address":
"00:00:00:00:00:00"}, {"name": "eth0", "ip-addresses":
[{"ip-address-type": "ipv4", "ip-address": "10.100.0.6", "prefix": 24},
{"ip-address-type": "ipv6", "ip-address": "fe80::216:3eff:fe40:88eb",
"prefix": 64}], "hardware-address": "00:16:3e:40:88:eb"}, {"name":
"sit0"}]}
Signed-off-by: Joao Martins <joao.m.martins@oracle.com>
Signed-off-by: Jim Fehlig <jfehlig@suse.com>
2016-09-26 18:33:16 +01:00
|
|
|
libxl_domain_config *d_config)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainChrDef **l_channels = def->channels;
|
libxl: channels support
And allow libxl to handle channel element which creates a Xen
console visible to the guest as a low-bandwitdh communication
channel. If type is PTY we also fetch the tty after boot using
libxl_channel_getinfo to fetch the tty path. On socket case,
we autogenerate a path if not specified in the XML. Path autogenerated
is slightly different from qemu driver: qemu stores also on
"channels/target" but it creates then a directory per domain with
each channel target name. libxl doesn't appear to have a clear
definition of private files associated with each domain, so for
simplicity we do it slightly different. On qemu each autogenerated
channel goes like:
channels/target/<domain-name>/<target name>
Whereas for libxl:
channels/target/<domain-name>-<target name>
Should note that if path is not specified it won't persist,
existing only on live XML, unless user had initially specified it.
Since support for libxl channels only came on Xen >= 4.5 we therefore
need to conditionally compile it with LIBXL_HAVE_DEVICE_CHANNEL.
After this patch and having a qemu guest agent:
$ cat domain.xml | grep -a1 channel | head -n 5 | tail -n 4
<channel type='unix'>
<source mode='bind' path='/tmp/channel'/>
<target type='xen' name='org.qemu.guest_agent.0'/>
</channel>
$ virsh create domain.xml
$ echo '{"execute":"guest-network-get-interfaces"}' | socat
stdio,ignoreeof unix-connect:/tmp/channel
{"execute":"guest-network-get-interfaces"}
{"return": [{"name": "lo", "ip-addresses": [{"ip-address-type": "ipv4",
"ip-address": "127.0.0.1", "prefix": 8}, {"ip-address-type": "ipv6",
"ip-address": "::1", "prefix": 128}], "hardware-address":
"00:00:00:00:00:00"}, {"name": "eth0", "ip-addresses":
[{"ip-address-type": "ipv4", "ip-address": "10.100.0.6", "prefix": 24},
{"ip-address-type": "ipv6", "ip-address": "fe80::216:3eff:fe40:88eb",
"prefix": 64}], "hardware-address": "00:16:3e:40:88:eb"}, {"name":
"sit0"}]}
Signed-off-by: Joao Martins <joao.m.martins@oracle.com>
Signed-off-by: Jim Fehlig <jfehlig@suse.com>
2016-09-26 18:33:16 +01:00
|
|
|
size_t nchannels = def->nchannels;
|
|
|
|
libxl_device_channel *x_channels;
|
|
|
|
size_t i, nvchannels = 0;
|
|
|
|
|
2020-09-23 20:43:09 +02:00
|
|
|
x_channels = g_new0(libxl_device_channel, nchannels);
|
libxl: channels support
And allow libxl to handle channel element which creates a Xen
console visible to the guest as a low-bandwitdh communication
channel. If type is PTY we also fetch the tty after boot using
libxl_channel_getinfo to fetch the tty path. On socket case,
we autogenerate a path if not specified in the XML. Path autogenerated
is slightly different from qemu driver: qemu stores also on
"channels/target" but it creates then a directory per domain with
each channel target name. libxl doesn't appear to have a clear
definition of private files associated with each domain, so for
simplicity we do it slightly different. On qemu each autogenerated
channel goes like:
channels/target/<domain-name>/<target name>
Whereas for libxl:
channels/target/<domain-name>-<target name>
Should note that if path is not specified it won't persist,
existing only on live XML, unless user had initially specified it.
Since support for libxl channels only came on Xen >= 4.5 we therefore
need to conditionally compile it with LIBXL_HAVE_DEVICE_CHANNEL.
After this patch and having a qemu guest agent:
$ cat domain.xml | grep -a1 channel | head -n 5 | tail -n 4
<channel type='unix'>
<source mode='bind' path='/tmp/channel'/>
<target type='xen' name='org.qemu.guest_agent.0'/>
</channel>
$ virsh create domain.xml
$ echo '{"execute":"guest-network-get-interfaces"}' | socat
stdio,ignoreeof unix-connect:/tmp/channel
{"execute":"guest-network-get-interfaces"}
{"return": [{"name": "lo", "ip-addresses": [{"ip-address-type": "ipv4",
"ip-address": "127.0.0.1", "prefix": 8}, {"ip-address-type": "ipv6",
"ip-address": "::1", "prefix": 128}], "hardware-address":
"00:00:00:00:00:00"}, {"name": "eth0", "ip-addresses":
[{"ip-address-type": "ipv4", "ip-address": "10.100.0.6", "prefix": 24},
{"ip-address-type": "ipv6", "ip-address": "fe80::216:3eff:fe40:88eb",
"prefix": 64}], "hardware-address": "00:16:3e:40:88:eb"}, {"name":
"sit0"}]}
Signed-off-by: Joao Martins <joao.m.martins@oracle.com>
Signed-off-by: Jim Fehlig <jfehlig@suse.com>
2016-09-26 18:33:16 +01:00
|
|
|
|
|
|
|
for (i = 0; i < nchannels; i++) {
|
|
|
|
if (l_channels[i]->deviceType != VIR_DOMAIN_CHR_DEVICE_TYPE_CHANNEL)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (libxlPrepareChannel(l_channels[i], channelDir, def->name) < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (libxlMakeChannel(l_channels[i], &x_channels[nvchannels]) < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
nvchannels++;
|
|
|
|
}
|
|
|
|
|
|
|
|
VIR_SHRINK_N(x_channels, nchannels, nchannels - nvchannels);
|
|
|
|
d_config->channels = x_channels;
|
|
|
|
d_config->num_channels = nvchannels;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
error:
|
|
|
|
for (i = 0; i < nchannels; i++)
|
|
|
|
libxl_device_channel_dispose(&x_channels[i]);
|
|
|
|
VIR_FREE(x_channels);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2016-06-15 14:00:09 +08:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlMakeUSBController(virDomainControllerDef *controller,
|
2016-06-15 14:00:09 +08:00
|
|
|
libxl_device_usbctrl *usbctrl)
|
|
|
|
{
|
|
|
|
usbctrl->devid = controller->idx;
|
|
|
|
|
|
|
|
if (controller->type != VIR_DOMAIN_CONTROLLER_TYPE_USB)
|
|
|
|
return -1;
|
|
|
|
|
2018-02-14 10:51:26 +00:00
|
|
|
if (controller->model == VIR_DOMAIN_CONTROLLER_MODEL_USB_DEFAULT) {
|
2016-06-15 14:00:09 +08:00
|
|
|
usbctrl->version = 2;
|
|
|
|
usbctrl->type = LIBXL_USBCTRL_TYPE_QUSB;
|
|
|
|
} else {
|
|
|
|
switch (controller->model) {
|
|
|
|
case VIR_DOMAIN_CONTROLLER_MODEL_USB_QUSB1:
|
|
|
|
usbctrl->version = 1;
|
|
|
|
usbctrl->type = LIBXL_USBCTRL_TYPE_QUSB;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_CONTROLLER_MODEL_USB_QUSB2:
|
|
|
|
usbctrl->version = 2;
|
|
|
|
usbctrl->type = LIBXL_USBCTRL_TYPE_QUSB;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("unsupported usb model"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (controller->opts.usbopts.ports == -1)
|
|
|
|
usbctrl->ports = 8;
|
|
|
|
else
|
|
|
|
usbctrl->ports = controller->opts.usbopts.ports;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-03-28 17:34:16 -06:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlMakeDefaultUSBControllers(virDomainDef *def,
|
2017-03-28 17:34:16 -06:00
|
|
|
libxl_domain_config *d_config)
|
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainControllerDef *l_controller = NULL;
|
2017-03-28 17:34:16 -06:00
|
|
|
libxl_device_usbctrl *x_controllers = NULL;
|
|
|
|
size_t nusbdevs = 0;
|
|
|
|
size_t ncontrollers;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < def->nhostdevs; i++) {
|
|
|
|
if (def->hostdevs[i]->mode == VIR_DOMAIN_HOSTDEV_MODE_SUBSYS &&
|
|
|
|
def->hostdevs[i]->source.subsys.type == VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_USB)
|
|
|
|
nusbdevs++;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* No controllers needed if there are no USB devs */
|
|
|
|
if (nusbdevs == 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
/* Create USB controllers with 8 ports */
|
|
|
|
ncontrollers = VIR_DIV_UP(nusbdevs, 8);
|
2020-09-23 20:43:09 +02:00
|
|
|
x_controllers = g_new0(libxl_device_usbctrl, ncontrollers);
|
2017-03-28 17:34:16 -06:00
|
|
|
|
|
|
|
for (i = 0; i < ncontrollers; i++) {
|
|
|
|
if (!(l_controller = virDomainControllerDefNew(VIR_DOMAIN_CONTROLLER_TYPE_USB)))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
l_controller->model = VIR_DOMAIN_CONTROLLER_MODEL_USB_QUSB2;
|
|
|
|
l_controller->idx = i;
|
|
|
|
l_controller->opts.usbopts.ports = 8;
|
|
|
|
|
|
|
|
libxl_device_usbctrl_init(&x_controllers[i]);
|
|
|
|
|
|
|
|
if (libxlMakeUSBController(l_controller, &x_controllers[i]) < 0)
|
|
|
|
goto error;
|
|
|
|
|
2020-11-05 22:32:58 -05:00
|
|
|
virDomainControllerInsert(def, l_controller);
|
2017-03-28 17:34:16 -06:00
|
|
|
|
|
|
|
l_controller = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
d_config->usbctrls = x_controllers;
|
|
|
|
d_config->num_usbctrls = ncontrollers;
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
error:
|
|
|
|
virDomainControllerDefFree(l_controller);
|
|
|
|
for (i = 0; i < ncontrollers; i++)
|
|
|
|
libxl_device_usbctrl_dispose(&x_controllers[i]);
|
|
|
|
VIR_FREE(x_controllers);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2016-06-15 14:00:09 +08:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlMakeUSBControllerList(virDomainDef *def, libxl_domain_config *d_config)
|
2016-06-15 14:00:09 +08:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainControllerDef **l_controllers = def->controllers;
|
2016-06-15 14:00:09 +08:00
|
|
|
size_t ncontrollers = def->ncontrollers;
|
|
|
|
size_t nusbctrls = 0;
|
|
|
|
libxl_device_usbctrl *x_usbctrls;
|
2017-03-28 17:34:16 -06:00
|
|
|
size_t i, j;
|
2016-06-15 14:00:09 +08:00
|
|
|
|
2017-03-28 17:34:16 -06:00
|
|
|
for (i = 0; i < ncontrollers; i++) {
|
|
|
|
if (l_controllers[i]->type == VIR_DOMAIN_CONTROLLER_TYPE_USB)
|
|
|
|
nusbctrls++;
|
|
|
|
}
|
2016-06-15 14:00:09 +08:00
|
|
|
|
2017-03-28 17:34:16 -06:00
|
|
|
if (nusbctrls == 0)
|
|
|
|
return libxlMakeDefaultUSBControllers(def, d_config);
|
|
|
|
|
2020-09-23 20:43:09 +02:00
|
|
|
x_usbctrls = g_new0(libxl_device_usbctrl, nusbctrls);
|
2016-06-15 14:00:09 +08:00
|
|
|
|
2017-03-28 17:34:16 -06:00
|
|
|
for (i = 0, j = 0; i < ncontrollers && j < nusbctrls; i++) {
|
2016-06-15 14:00:09 +08:00
|
|
|
if (l_controllers[i]->type != VIR_DOMAIN_CONTROLLER_TYPE_USB)
|
|
|
|
continue;
|
|
|
|
|
2017-03-28 17:34:16 -06:00
|
|
|
libxl_device_usbctrl_init(&x_usbctrls[j]);
|
2016-06-15 14:00:09 +08:00
|
|
|
|
|
|
|
if (libxlMakeUSBController(l_controllers[i],
|
2017-03-28 17:34:16 -06:00
|
|
|
&x_usbctrls[j]) < 0)
|
2016-06-15 14:00:09 +08:00
|
|
|
goto error;
|
|
|
|
|
2017-03-28 17:34:16 -06:00
|
|
|
j++;
|
2016-06-15 14:00:09 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
d_config->usbctrls = x_usbctrls;
|
|
|
|
d_config->num_usbctrls = nusbctrls;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
error:
|
|
|
|
for (i = 0; i < nusbctrls; i++)
|
|
|
|
libxl_device_usbctrl_dispose(&x_usbctrls[i]);
|
|
|
|
|
|
|
|
VIR_FREE(x_usbctrls);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2016-05-19 16:14:33 +08:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlMakeUSB(virDomainHostdevDef *hostdev, libxl_device_usbdev *usbdev)
|
2016-05-19 16:14:33 +08:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainHostdevSubsysUSB *usbsrc = &hostdev->source.subsys.u.usb;
|
|
|
|
virUSBDevice *usb = NULL;
|
2016-08-10 18:39:12 -06:00
|
|
|
int ret = -1;
|
2016-05-19 16:14:33 +08:00
|
|
|
|
|
|
|
if (hostdev->mode != VIR_DOMAIN_HOSTDEV_MODE_SUBSYS)
|
2016-08-10 18:39:12 -06:00
|
|
|
return ret;
|
2016-05-19 16:14:33 +08:00
|
|
|
if (hostdev->source.subsys.type != VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_USB)
|
2016-08-10 18:39:12 -06:00
|
|
|
return ret;
|
2016-05-19 16:14:33 +08:00
|
|
|
|
2016-08-10 18:39:12 -06:00
|
|
|
if (usbsrc->bus > 0 && usbsrc->device > 0) {
|
|
|
|
usbdev->u.hostdev.hostbus = usbsrc->bus;
|
|
|
|
usbdev->u.hostdev.hostaddr = usbsrc->device;
|
|
|
|
} else {
|
|
|
|
if (virHostdevFindUSBDevice(hostdev, true, &usb) < 0) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED,
|
|
|
|
_("failed to find USB device busnum:devnum "
|
|
|
|
"for %x:%x"),
|
|
|
|
usbsrc->vendor, usbsrc->product);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
usbdev->u.hostdev.hostbus = virUSBDeviceGetBus(usb);
|
|
|
|
usbdev->u.hostdev.hostaddr = virUSBDeviceGetDevno(usb);
|
2016-05-19 16:14:33 +08:00
|
|
|
}
|
|
|
|
|
2016-08-10 18:39:12 -06:00
|
|
|
ret = 0;
|
2016-05-19 16:14:33 +08:00
|
|
|
|
2016-08-10 18:39:12 -06:00
|
|
|
cleanup:
|
|
|
|
virUSBDeviceFree(usb);
|
|
|
|
|
|
|
|
return ret;
|
2016-05-19 16:14:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlMakeUSBList(virDomainDef *def, libxl_domain_config *d_config)
|
2016-05-19 16:14:33 +08:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainHostdevDef **l_hostdevs = def->hostdevs;
|
2016-05-19 16:14:33 +08:00
|
|
|
size_t nhostdevs = def->nhostdevs;
|
|
|
|
size_t nusbdevs = 0;
|
|
|
|
libxl_device_usbdev *x_usbdevs;
|
|
|
|
size_t i, j;
|
|
|
|
|
|
|
|
if (nhostdevs == 0)
|
|
|
|
return 0;
|
|
|
|
|
2020-09-23 20:43:09 +02:00
|
|
|
x_usbdevs = g_new0(libxl_device_usbdev, nhostdevs);
|
2016-05-19 16:14:33 +08:00
|
|
|
|
|
|
|
for (i = 0, j = 0; i < nhostdevs; i++) {
|
|
|
|
if (l_hostdevs[i]->mode != VIR_DOMAIN_HOSTDEV_MODE_SUBSYS)
|
|
|
|
continue;
|
|
|
|
if (l_hostdevs[i]->source.subsys.type != VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_USB)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
libxl_device_usbdev_init(&x_usbdevs[j]);
|
|
|
|
|
|
|
|
if (libxlMakeUSB(l_hostdevs[i], &x_usbdevs[j]) < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
nusbdevs++;
|
|
|
|
j++;
|
|
|
|
}
|
|
|
|
|
|
|
|
VIR_SHRINK_N(x_usbdevs, nhostdevs, nhostdevs - nusbdevs);
|
|
|
|
d_config->usbdevs = x_usbdevs;
|
|
|
|
d_config->num_usbdevs = nusbdevs;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
error:
|
|
|
|
for (i = 0; i < nusbdevs; i++)
|
|
|
|
libxl_device_usbdev_dispose(&x_usbdevs[i]);
|
|
|
|
|
|
|
|
VIR_FREE(x_usbdevs);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2014-03-06 17:46:11 +08:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlMakePCI(virDomainHostdevDef *hostdev, libxl_device_pci *pcidev)
|
2014-03-06 17:46:11 +08:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainHostdevSubsysPCI *pcisrc = &hostdev->source.subsys.u.pci;
|
2014-03-06 17:46:11 +08:00
|
|
|
if (hostdev->mode != VIR_DOMAIN_HOSTDEV_MODE_SUBSYS)
|
|
|
|
return -1;
|
|
|
|
if (hostdev->source.subsys.type != VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_PCI)
|
|
|
|
return -1;
|
|
|
|
|
2014-07-03 16:31:39 -04:00
|
|
|
pcidev->domain = pcisrc->addr.domain;
|
|
|
|
pcidev->bus = pcisrc->addr.bus;
|
|
|
|
pcidev->dev = pcisrc->addr.slot;
|
|
|
|
pcidev->func = pcisrc->addr.function;
|
2020-08-14 10:28:38 -06:00
|
|
|
pcidev->permissive = hostdev->writeFiltering == VIR_TRISTATE_BOOL_NO;
|
2014-03-06 17:46:11 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlMakePCIList(virDomainDef *def, libxl_domain_config *d_config)
|
2014-03-06 17:46:11 +08:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainHostdevDef **l_hostdevs = def->hostdevs;
|
2014-03-06 17:46:11 +08:00
|
|
|
size_t nhostdevs = def->nhostdevs;
|
|
|
|
size_t npcidevs = 0;
|
|
|
|
libxl_device_pci *x_pcidevs;
|
|
|
|
size_t i, j;
|
|
|
|
|
|
|
|
if (nhostdevs == 0)
|
|
|
|
return 0;
|
|
|
|
|
2020-09-23 20:43:09 +02:00
|
|
|
x_pcidevs = g_new0(libxl_device_pci, nhostdevs);
|
2014-03-06 17:46:11 +08:00
|
|
|
|
|
|
|
for (i = 0, j = 0; i < nhostdevs; i++) {
|
|
|
|
if (l_hostdevs[i]->mode != VIR_DOMAIN_HOSTDEV_MODE_SUBSYS)
|
|
|
|
continue;
|
|
|
|
if (l_hostdevs[i]->source.subsys.type != VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_PCI)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
libxl_device_pci_init(&x_pcidevs[j]);
|
|
|
|
|
2014-03-13 11:59:32 +00:00
|
|
|
if (libxlMakePCI(l_hostdevs[i], &x_pcidevs[j]) < 0)
|
2014-03-06 17:46:11 +08:00
|
|
|
goto error;
|
|
|
|
|
|
|
|
npcidevs++;
|
|
|
|
j++;
|
|
|
|
}
|
|
|
|
|
|
|
|
VIR_SHRINK_N(x_pcidevs, nhostdevs, nhostdevs - npcidevs);
|
|
|
|
d_config->pcidevs = x_pcidevs;
|
|
|
|
d_config->num_pcidevs = npcidevs;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
2014-03-25 07:51:17 +01:00
|
|
|
error:
|
2014-03-06 17:46:11 +08:00
|
|
|
for (i = 0; i < npcidevs; i++)
|
|
|
|
libxl_device_pci_dispose(&x_pcidevs[i]);
|
|
|
|
|
|
|
|
VIR_FREE(x_pcidevs);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2014-07-01 09:58:18 +02:00
|
|
|
static int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlMakeVideo(virDomainDef *def, libxl_domain_config *d_config)
|
2014-07-01 09:58:18 +02:00
|
|
|
|
|
|
|
{
|
|
|
|
libxl_domain_build_info *b_info = &d_config->b_info;
|
|
|
|
int dm_type = libxlDomainGetEmulatorType(def);
|
|
|
|
|
|
|
|
if (d_config->c_info.type != LIBXL_DOMAIN_TYPE_HVM)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Take the first defined video device (graphics card) to display
|
|
|
|
* on the first graphics device (display).
|
|
|
|
*/
|
|
|
|
if (def->nvideos) {
|
|
|
|
switch (def->videos[0]->type) {
|
|
|
|
case VIR_DOMAIN_VIDEO_TYPE_VGA:
|
|
|
|
case VIR_DOMAIN_VIDEO_TYPE_XEN:
|
|
|
|
b_info->u.hvm.vga.kind = LIBXL_VGA_INTERFACE_TYPE_STD;
|
|
|
|
if (dm_type == LIBXL_DEVICE_MODEL_VERSION_QEMU_XEN) {
|
|
|
|
if (def->videos[0]->vram < 16 * 1024) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("videoram must be at least 16MB for VGA"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (def->videos[0]->vram < 8 * 1024) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("videoram must be at least 8MB for VGA"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_VIDEO_TYPE_CIRRUS:
|
|
|
|
b_info->u.hvm.vga.kind = LIBXL_VGA_INTERFACE_TYPE_CIRRUS;
|
|
|
|
if (dm_type == LIBXL_DEVICE_MODEL_VERSION_QEMU_XEN) {
|
|
|
|
if (def->videos[0]->vram < 8 * 1024) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("videoram must be at least 8MB for CIRRUS"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (def->videos[0]->vram < 4 * 1024) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("videoram must be at least 4MB for CIRRUS"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2015-04-29 10:50:33 -06:00
|
|
|
case VIR_DOMAIN_VIDEO_TYPE_QXL:
|
|
|
|
b_info->u.hvm.vga.kind = LIBXL_VGA_INTERFACE_TYPE_QXL;
|
|
|
|
if (def->videos[0]->vram < 128 * 1024) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("videoram must be at least 128MB for QXL"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2014-07-01 09:58:18 +02:00
|
|
|
default:
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("video type %s is not supported by libxl"),
|
|
|
|
virDomainVideoTypeToString(def->videos[0]->type));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
/* vram validated for each video type, now set it */
|
|
|
|
b_info->video_memkb = def->videos[0]->vram;
|
|
|
|
} else {
|
|
|
|
libxl_defbool_set(&b_info->u.hvm.nographic, 1);
|
2020-05-07 13:24:35 +02:00
|
|
|
b_info->u.hvm.vga.kind = LIBXL_VGA_INTERFACE_TYPE_NONE;
|
2014-07-01 09:58:18 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-02-26 13:44:09 -07:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlDriverNodeGetInfo(libxlDriverPrivate *driver, virNodeInfoPtr info)
|
2014-02-26 13:44:09 -07:00
|
|
|
{
|
|
|
|
libxl_physinfo phy_info;
|
|
|
|
virArch hostarch = virArchFromHost();
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlDriverConfig *cfg = libxlDriverConfigGet(driver);
|
2014-02-26 13:44:09 -07:00
|
|
|
int ret = -1;
|
|
|
|
|
2017-01-31 20:07:30 -07:00
|
|
|
libxl_physinfo_init(&phy_info);
|
2014-02-26 13:44:09 -07:00
|
|
|
if (libxl_get_physinfo(cfg->ctx, &phy_info)) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("libxl_get_physinfo_info failed"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2018-07-20 09:50:37 +02:00
|
|
|
if (virStrcpyStatic(info->model, virArchToString(hostarch)) < 0) {
|
2014-02-26 13:44:09 -07:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
2019-11-26 15:59:13 +00:00
|
|
|
_("host arch %s is too big for destination"),
|
2014-02-26 13:44:09 -07:00
|
|
|
virArchToString(hostarch));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
info->memory = phy_info.total_pages * (cfg->verInfo->pagesize / 1024);
|
|
|
|
info->cpus = phy_info.nr_cpus;
|
|
|
|
info->nodes = phy_info.nr_nodes;
|
|
|
|
info->cores = phy_info.cores_per_socket;
|
|
|
|
info->threads = phy_info.threads_per_core;
|
|
|
|
info->sockets = 1;
|
|
|
|
info->mhz = phy_info.cpu_khz / 1000;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
2014-03-25 07:51:17 +01:00
|
|
|
cleanup:
|
2017-01-31 20:07:30 -07:00
|
|
|
libxl_physinfo_dispose(&phy_info);
|
2014-02-26 13:44:09 -07:00
|
|
|
virObjectUnref(cfg);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-02-10 15:42:34 -07:00
|
|
|
int
|
2021-03-11 08:16:13 +01:00
|
|
|
libxlBuildDomainConfig(virPortAllocatorRange *graphicsports,
|
|
|
|
virDomainDef *def,
|
|
|
|
libxlDriverConfig *cfg,
|
2014-05-30 14:46:35 +01:00
|
|
|
libxl_domain_config *d_config)
|
2011-02-10 15:42:34 -07:00
|
|
|
{
|
2021-03-11 08:16:13 +01:00
|
|
|
virCaps *caps = cfg->caps;
|
2018-04-12 03:03:20 +02:00
|
|
|
libxl_ctx *ctx = cfg->ctx;
|
2011-02-10 15:42:34 -07:00
|
|
|
|
2014-05-30 14:46:35 +01:00
|
|
|
if (libxlMakeDomCreateInfo(ctx, def, &d_config->c_info) < 0)
|
2011-02-10 15:42:34 -07:00
|
|
|
return -1;
|
|
|
|
|
2018-04-12 03:03:20 +02:00
|
|
|
if (libxlMakeDomBuildInfo(def, cfg, caps, d_config) < 0)
|
2013-04-29 17:04:55 -06:00
|
|
|
return -1;
|
2011-02-10 15:42:34 -07:00
|
|
|
|
2017-11-02 16:47:22 +01:00
|
|
|
if (libxlMakeVnumaList(def, ctx, d_config) < 0)
|
|
|
|
return -1;
|
|
|
|
|
2013-08-14 15:33:48 -06:00
|
|
|
if (libxlMakeDiskList(def, d_config) < 0)
|
2013-04-29 17:04:55 -06:00
|
|
|
return -1;
|
2011-02-10 15:42:34 -07:00
|
|
|
|
2013-08-14 15:33:48 -06:00
|
|
|
if (libxlMakeNicList(def, d_config) < 0)
|
2013-04-29 17:04:55 -06:00
|
|
|
return -1;
|
2011-02-10 15:42:34 -07:00
|
|
|
|
2014-05-30 14:48:56 +01:00
|
|
|
if (libxlMakeVfbList(graphicsports, def, d_config) < 0)
|
2013-04-29 17:04:55 -06:00
|
|
|
return -1;
|
2011-02-10 15:42:34 -07:00
|
|
|
|
2015-04-27 16:27:52 -06:00
|
|
|
if (libxlMakeBuildInfoVfb(graphicsports, def, d_config) < 0)
|
2015-04-27 12:32:55 -06:00
|
|
|
return -1;
|
|
|
|
|
2014-03-13 11:59:32 +00:00
|
|
|
if (libxlMakePCIList(def, d_config) < 0)
|
2014-03-06 17:46:11 +08:00
|
|
|
return -1;
|
|
|
|
|
2016-06-15 14:00:09 +08:00
|
|
|
if (libxlMakeUSBControllerList(def, d_config) < 0)
|
|
|
|
return -1;
|
|
|
|
|
2016-05-19 16:14:33 +08:00
|
|
|
if (libxlMakeUSBList(def, d_config) < 0)
|
|
|
|
return -1;
|
|
|
|
|
2018-04-12 03:03:20 +02:00
|
|
|
if (libxlMakeChannelList(cfg->channelDir, def, d_config) < 0)
|
libxl: channels support
And allow libxl to handle channel element which creates a Xen
console visible to the guest as a low-bandwitdh communication
channel. If type is PTY we also fetch the tty after boot using
libxl_channel_getinfo to fetch the tty path. On socket case,
we autogenerate a path if not specified in the XML. Path autogenerated
is slightly different from qemu driver: qemu stores also on
"channels/target" but it creates then a directory per domain with
each channel target name. libxl doesn't appear to have a clear
definition of private files associated with each domain, so for
simplicity we do it slightly different. On qemu each autogenerated
channel goes like:
channels/target/<domain-name>/<target name>
Whereas for libxl:
channels/target/<domain-name>-<target name>
Should note that if path is not specified it won't persist,
existing only on live XML, unless user had initially specified it.
Since support for libxl channels only came on Xen >= 4.5 we therefore
need to conditionally compile it with LIBXL_HAVE_DEVICE_CHANNEL.
After this patch and having a qemu guest agent:
$ cat domain.xml | grep -a1 channel | head -n 5 | tail -n 4
<channel type='unix'>
<source mode='bind' path='/tmp/channel'/>
<target type='xen' name='org.qemu.guest_agent.0'/>
</channel>
$ virsh create domain.xml
$ echo '{"execute":"guest-network-get-interfaces"}' | socat
stdio,ignoreeof unix-connect:/tmp/channel
{"execute":"guest-network-get-interfaces"}
{"return": [{"name": "lo", "ip-addresses": [{"ip-address-type": "ipv4",
"ip-address": "127.0.0.1", "prefix": 8}, {"ip-address-type": "ipv6",
"ip-address": "::1", "prefix": 128}], "hardware-address":
"00:00:00:00:00:00"}, {"name": "eth0", "ip-addresses":
[{"ip-address-type": "ipv4", "ip-address": "10.100.0.6", "prefix": 24},
{"ip-address-type": "ipv6", "ip-address": "fe80::216:3eff:fe40:88eb",
"prefix": 64}], "hardware-address": "00:16:3e:40:88:eb"}, {"name":
"sit0"}]}
Signed-off-by: Joao Martins <joao.m.martins@oracle.com>
Signed-off-by: Jim Fehlig <jfehlig@suse.com>
2016-09-26 18:33:16 +01:00
|
|
|
return -1;
|
|
|
|
|
2014-07-01 09:58:18 +02:00
|
|
|
/*
|
|
|
|
* Now that any potential VFBs are defined, update the build info with
|
2018-12-04 19:08:14 +02:00
|
|
|
* the data of the primary display. Some day libxl might implicitly do
|
2014-07-01 09:58:18 +02:00
|
|
|
* so but as it does not right now, better be explicit.
|
|
|
|
*/
|
|
|
|
if (libxlMakeVideo(def, d_config) < 0)
|
|
|
|
return -1;
|
|
|
|
|
2014-09-03 14:14:50 -06:00
|
|
|
d_config->on_reboot = libxlActionFromVirLifecycle(def->onReboot);
|
|
|
|
d_config->on_poweroff = libxlActionFromVirLifecycle(def->onPoweroff);
|
2017-10-10 14:51:38 +02:00
|
|
|
d_config->on_crash = libxlActionFromVirLifecycle(def->onCrash);
|
2011-02-10 15:42:34 -07:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2014-05-30 15:53:12 +01:00
|
|
|
|
2021-03-11 08:16:13 +01:00
|
|
|
virDomainXMLOption *
|
|
|
|
libxlCreateXMLConf(libxlDriverPrivate *driver)
|
2014-05-30 15:53:12 +01:00
|
|
|
{
|
2019-12-03 10:49:49 +00:00
|
|
|
libxlDomainDefParserConfig.priv = driver;
|
2014-05-30 15:53:12 +01:00
|
|
|
return virDomainXMLOptionNew(&libxlDomainDefParserConfig,
|
|
|
|
&libxlDomainXMLPrivateDataCallbacks,
|
2020-07-30 13:25:20 -06:00
|
|
|
&libxlDriverDomainXMLNamespace,
|
|
|
|
NULL, NULL);
|
2014-05-30 15:53:12 +01:00
|
|
|
}
|