mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-04 12:05:19 +00:00
249fbea03d
Introduce use of a virDomainDefPtr in the domain property APIs to simplify introduction of ACL security checks. The virDomainPtr cannot be safely used, since the app may have supplied mis-matching name/uuid/id fields. eg the name points to domain X, while the uuid points to domain Y. Resolving the virDomainPtr to a virDomainDefPtr ensures a consistent name/uuid/id set. Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
1506 lines
40 KiB
C
1506 lines
40 KiB
C
/*
|
|
* xm_internal.h: helper routines for dealing with inactive domains
|
|
*
|
|
* Copyright (C) 2006-2007, 2009-2013 Red Hat, Inc.
|
|
* Copyright (C) 2006 Daniel P. Berrange
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library. If not, see
|
|
* <http://www.gnu.org/licenses/>.
|
|
*
|
|
* Author: Daniel P. Berrange <berrange@redhat.com>
|
|
*
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include <dirent.h>
|
|
#include <time.h>
|
|
#include <sys/stat.h>
|
|
#include <limits.h>
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
|
|
#include <unistd.h>
|
|
#include <stdint.h>
|
|
#include <xen/dom0_ops.h>
|
|
|
|
#include "virerror.h"
|
|
#include "virfile.h"
|
|
#include "datatypes.h"
|
|
#include "xm_internal.h"
|
|
#include "xen_driver.h"
|
|
#include "xend_internal.h"
|
|
#include "xen_sxpr.h"
|
|
#include "xen_xm.h"
|
|
#include "virhash.h"
|
|
#include "virbuffer.h"
|
|
#include "viruuid.h"
|
|
#include "viralloc.h"
|
|
#include "virlog.h"
|
|
#include "count-one-bits.h"
|
|
#include "virstring.h"
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_XENXM
|
|
|
|
#ifdef WITH_RHEL5_API
|
|
# define XEND_CONFIG_MAX_VERS_NET_TYPE_IOEMU 0
|
|
# define XEND_CONFIG_MIN_VERS_PVFB_NEWCONF 2
|
|
#else
|
|
# define XEND_CONFIG_MAX_VERS_NET_TYPE_IOEMU 3
|
|
# define XEND_CONFIG_MIN_VERS_PVFB_NEWCONF 3
|
|
#endif
|
|
|
|
/* The true Xen limit varies but so far is always way
|
|
less than 1024, which is the Linux kernel limit according
|
|
to sched.h, so we'll match that for now */
|
|
#define XEN_MAX_PHYSICAL_CPU 1024
|
|
|
|
char * xenXMAutoAssignMac(void);
|
|
|
|
#define XM_REFRESH_INTERVAL 10
|
|
|
|
#define XM_CONFIG_DIR "/etc/xen"
|
|
#define XM_EXAMPLE_PREFIX "xmexample"
|
|
#define XEND_CONFIG_FILE "xend-config.sxp"
|
|
#define XEND_PCI_CONFIG_PREFIX "xend-pci-"
|
|
#define QEMU_IF_SCRIPT "qemu-ifup"
|
|
#define XM_XML_ERROR "Invalid xml"
|
|
|
|
#ifndef WITH_XEN_INOTIFY
|
|
static int xenInotifyActive(virConnectPtr conn ATTRIBUTE_UNUSED)
|
|
{
|
|
return 0;
|
|
}
|
|
#else
|
|
static int xenInotifyActive(virConnectPtr conn)
|
|
{
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
|
return priv->inotifyWatch > 0;
|
|
}
|
|
#endif
|
|
|
|
|
|
/* Release memory associated with a cached config object */
|
|
static void xenXMConfigFree(void *payload, const void *key ATTRIBUTE_UNUSED) {
|
|
xenXMConfCachePtr entry = (xenXMConfCachePtr)payload;
|
|
virDomainDefFree(entry->def);
|
|
VIR_FREE(entry->filename);
|
|
VIR_FREE(entry);
|
|
}
|
|
|
|
struct xenXMConfigReaperData {
|
|
xenUnifiedPrivatePtr priv;
|
|
time_t now;
|
|
};
|
|
|
|
/* Remove any configs which were not refreshed recently */
|
|
static int
|
|
xenXMConfigReaper(const void *payload,
|
|
const void *key ATTRIBUTE_UNUSED,
|
|
const void *data)
|
|
{
|
|
const struct xenXMConfigReaperData *args = data;
|
|
xenXMConfCachePtr entry = (xenXMConfCachePtr)payload;
|
|
|
|
/* We're going to purge this config file, so check if it
|
|
is currently mapped as owner of a named domain. */
|
|
if (entry->refreshedAt != args->now) {
|
|
const char *olddomname = entry->def->name;
|
|
char *nameowner = (char *)virHashLookup(args->priv->nameConfigMap, olddomname);
|
|
if (nameowner && STREQ(nameowner, key)) {
|
|
virHashRemoveEntry(args->priv->nameConfigMap, olddomname);
|
|
}
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
static virDomainDefPtr
|
|
xenXMConfigReadFile(virConnectPtr conn, const char *filename)
|
|
{
|
|
virConfPtr conf;
|
|
virDomainDefPtr def;
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
|
|
|
if (!(conf = virConfReadFile(filename, 0)))
|
|
return NULL;
|
|
|
|
def = xenParseXM(conf, priv->xendConfigVersion, priv->caps);
|
|
virConfFree(conf);
|
|
|
|
return def;
|
|
}
|
|
|
|
static int
|
|
xenXMConfigSaveFile(virConnectPtr conn,
|
|
const char *filename,
|
|
virDomainDefPtr def)
|
|
{
|
|
virConfPtr conf;
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
|
int ret;
|
|
|
|
if (!(conf = xenFormatXM(conn, def, priv->xendConfigVersion)))
|
|
return -1;
|
|
|
|
ret = virConfWriteFile(filename, conf);
|
|
virConfFree(conf);
|
|
return ret;
|
|
}
|
|
|
|
|
|
/*
|
|
* Caller must hold the lock on 'conn->privateData' before
|
|
* calling this function
|
|
*/
|
|
int
|
|
xenXMConfigCacheRemoveFile(virConnectPtr conn, const char *filename)
|
|
{
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
|
xenXMConfCachePtr entry;
|
|
|
|
entry = virHashLookup(priv->configCache, filename);
|
|
if (!entry) {
|
|
VIR_DEBUG("No config entry for %s", filename);
|
|
return 0;
|
|
}
|
|
|
|
virHashRemoveEntry(priv->nameConfigMap, entry->def->name);
|
|
virHashRemoveEntry(priv->configCache, filename);
|
|
VIR_DEBUG("Removed %s %s", entry->def->name, filename);
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
* Caller must hold the lock on 'conn->privateData' before
|
|
* calling this function
|
|
*/
|
|
int
|
|
xenXMConfigCacheAddFile(virConnectPtr conn, const char *filename)
|
|
{
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
|
xenXMConfCachePtr entry;
|
|
struct stat st;
|
|
int newborn = 0;
|
|
time_t now = time(NULL);
|
|
|
|
VIR_DEBUG("Adding file %s", filename);
|
|
|
|
/* Get modified time */
|
|
if ((stat(filename, &st) < 0)) {
|
|
virReportSystemError(errno,
|
|
_("cannot stat: %s"),
|
|
filename);
|
|
return -1;
|
|
}
|
|
|
|
/* Ignore zero length files, because inotify fires before
|
|
any content has actually been created */
|
|
if (st.st_size == 0) {
|
|
VIR_DEBUG("Ignoring zero length file %s", filename);
|
|
return -1;
|
|
}
|
|
|
|
/* If we already have a matching entry and it is not
|
|
modified, then carry on to next one*/
|
|
if ((entry = virHashLookup(priv->configCache, filename))) {
|
|
char *nameowner;
|
|
|
|
if (entry->refreshedAt >= st.st_mtime) {
|
|
entry->refreshedAt = now;
|
|
/* return success if up-to-date */
|
|
return 0;
|
|
}
|
|
|
|
/* If we currently own the name, then release it and
|
|
re-acquire it later - just in case it was renamed */
|
|
nameowner = (char *)virHashLookup(priv->nameConfigMap, entry->def->name);
|
|
if (nameowner && STREQ(nameowner, filename)) {
|
|
virHashRemoveEntry(priv->nameConfigMap, entry->def->name);
|
|
}
|
|
|
|
/* Clear existing config entry which needs refresh */
|
|
virDomainDefFree(entry->def);
|
|
entry->def = NULL;
|
|
} else { /* Completely new entry */
|
|
newborn = 1;
|
|
if (VIR_ALLOC(entry) < 0) {
|
|
virReportOOMError();
|
|
return -1;
|
|
}
|
|
if ((entry->filename = strdup(filename)) == NULL) {
|
|
virReportOOMError();
|
|
VIR_FREE(entry);
|
|
return -1;
|
|
}
|
|
}
|
|
entry->refreshedAt = now;
|
|
|
|
if (!(entry->def = xenXMConfigReadFile(conn, entry->filename))) {
|
|
VIR_DEBUG("Failed to read %s", entry->filename);
|
|
if (!newborn)
|
|
virHashSteal(priv->configCache, filename);
|
|
VIR_FREE(entry->filename);
|
|
VIR_FREE(entry);
|
|
return -1;
|
|
}
|
|
|
|
/* If its a completely new entry, it must be stuck into
|
|
the cache (refresh'd entries are already registered) */
|
|
if (newborn) {
|
|
if (virHashAddEntry(priv->configCache, entry->filename, entry) < 0) {
|
|
virDomainDefFree(entry->def);
|
|
VIR_FREE(entry->filename);
|
|
VIR_FREE(entry);
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
"%s", _("xenXMConfigCacheRefresh: virHashAddEntry"));
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
/* See if we need to map this config file in as the primary owner
|
|
* of the domain in question
|
|
*/
|
|
if (!virHashLookup(priv->nameConfigMap, entry->def->name)) {
|
|
if (virHashAddEntry(priv->nameConfigMap, entry->def->name,
|
|
entry->filename) < 0) {
|
|
virHashSteal(priv->configCache, filename);
|
|
virDomainDefFree(entry->def);
|
|
VIR_FREE(entry->filename);
|
|
VIR_FREE(entry);
|
|
}
|
|
}
|
|
VIR_DEBUG("Added config %s %s", entry->def->name, filename);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* This method is called by various methods to scan /etc/xen
|
|
* (or whatever directory was set by LIBVIRT_XM_CONFIG_DIR
|
|
* environment variable) and process any domain configs. It
|
|
* has rate-limited so never rescans more frequently than
|
|
* once every X seconds
|
|
*
|
|
* Caller must hold the lock on 'conn->privateData' before
|
|
* calling this function
|
|
*/
|
|
int
|
|
xenXMConfigCacheRefresh(virConnectPtr conn)
|
|
{
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
|
DIR *dh;
|
|
struct dirent *ent;
|
|
time_t now = time(NULL);
|
|
int ret = -1;
|
|
struct xenXMConfigReaperData args;
|
|
|
|
if (now == ((time_t)-1)) {
|
|
virReportSystemError(errno,
|
|
"%s", _("cannot get time of day"));
|
|
return -1;
|
|
}
|
|
|
|
/* Rate limit re-scans */
|
|
if ((now - priv->lastRefresh) < XM_REFRESH_INTERVAL)
|
|
return 0;
|
|
|
|
priv->lastRefresh = now;
|
|
|
|
/* Process the files in the config dir */
|
|
if (!(dh = opendir(priv->configDir))) {
|
|
virReportSystemError(errno,
|
|
_("cannot read directory %s"),
|
|
priv->configDir);
|
|
return -1;
|
|
}
|
|
|
|
while ((ent = readdir(dh))) {
|
|
struct stat st;
|
|
char *path;
|
|
|
|
/*
|
|
* Skip a bunch of crufty files that clearly aren't config files
|
|
*/
|
|
|
|
/* Like 'dot' files... */
|
|
if (STRPREFIX(ent->d_name, "."))
|
|
continue;
|
|
/* ...and the XenD server config file */
|
|
if (STRPREFIX(ent->d_name, XEND_CONFIG_FILE))
|
|
continue;
|
|
/* ...and random PCI config cruft */
|
|
if (STRPREFIX(ent->d_name, XEND_PCI_CONFIG_PREFIX))
|
|
continue;
|
|
/* ...and the example domain configs */
|
|
if (STRPREFIX(ent->d_name, XM_EXAMPLE_PREFIX))
|
|
continue;
|
|
/* ...and the QEMU networking script */
|
|
if (STRPREFIX(ent->d_name, QEMU_IF_SCRIPT))
|
|
continue;
|
|
|
|
/* ...and editor backups */
|
|
if (ent->d_name[0] == '#')
|
|
continue;
|
|
if (ent->d_name[strlen(ent->d_name)-1] == '~')
|
|
continue;
|
|
|
|
/* Build the full file path */
|
|
if (!(path = virFileBuildPath(priv->configDir, ent->d_name, NULL))) {
|
|
closedir(dh);
|
|
return -1;
|
|
}
|
|
|
|
/* Skip anything which isn't a file (takes care of scripts/ subdir */
|
|
if ((stat(path, &st) < 0) ||
|
|
(!S_ISREG(st.st_mode))) {
|
|
VIR_FREE(path);
|
|
continue;
|
|
}
|
|
|
|
/* If we already have a matching entry and it is not
|
|
modified, then carry on to next one*/
|
|
if (xenXMConfigCacheAddFile(conn, path) < 0) {
|
|
/* Ignoring errors, since a lot of stuff goes wrong in /etc/xen */
|
|
}
|
|
|
|
VIR_FREE(path);
|
|
}
|
|
|
|
/* Reap all entries which were not changed, by comparing
|
|
their refresh timestamp - the timestamp should match
|
|
'now' if they were refreshed. If timestamp doesn't match
|
|
then the config is no longer on disk */
|
|
args.now = now;
|
|
args.priv = priv;
|
|
virHashRemoveSet(priv->configCache, xenXMConfigReaper, &args);
|
|
ret = 0;
|
|
|
|
closedir(dh);
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
/*
|
|
* The XM driver keeps a cache of config files as virDomainDefPtr
|
|
* objects in the xenUnifiedPrivatePtr. Optionally inotify lets
|
|
* us watch for changes (see separate driver), otherwise we poll
|
|
* every few seconds
|
|
*/
|
|
int
|
|
xenXMOpen(virConnectPtr conn,
|
|
virConnectAuthPtr auth ATTRIBUTE_UNUSED,
|
|
unsigned int flags)
|
|
{
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
|
|
|
virCheckFlags(VIR_CONNECT_RO, -1);
|
|
|
|
priv->configDir = XM_CONFIG_DIR;
|
|
|
|
priv->configCache = virHashCreate(50, xenXMConfigFree);
|
|
if (!priv->configCache)
|
|
return -1;
|
|
priv->nameConfigMap = virHashCreate(50, NULL);
|
|
if (!priv->nameConfigMap) {
|
|
virHashFree(priv->configCache);
|
|
priv->configCache = NULL;
|
|
return -1;
|
|
}
|
|
/* Force the cache to be reloaded next time that
|
|
* xenXMConfigCacheRefresh is called.
|
|
*/
|
|
priv->lastRefresh = 0;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Free the cached config files associated with this
|
|
* connection
|
|
*/
|
|
int
|
|
xenXMClose(virConnectPtr conn)
|
|
{
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
|
|
|
virHashFree(priv->nameConfigMap);
|
|
virHashFree(priv->configCache);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Since these are all offline domains, the state is always SHUTOFF.
|
|
*/
|
|
int
|
|
xenXMDomainGetState(virConnectPtr conn ATTRIBUTE_UNUSED,
|
|
virDomainDefPtr def ATTRIBUTE_UNUSED,
|
|
int *state,
|
|
int *reason)
|
|
{
|
|
*state = VIR_DOMAIN_SHUTOFF;
|
|
if (reason)
|
|
*reason = 0;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
* Since these are all offline domains, we only return info about
|
|
* VCPUs and memory.
|
|
*/
|
|
int
|
|
xenXMDomainGetInfo(virConnectPtr conn,
|
|
virDomainDefPtr def,
|
|
virDomainInfoPtr info)
|
|
{
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
|
const char *filename;
|
|
xenXMConfCachePtr entry;
|
|
|
|
xenUnifiedLock(priv);
|
|
|
|
if (!(filename = virHashLookup(priv->nameConfigMap, def->name)))
|
|
goto error;
|
|
|
|
if (!(entry = virHashLookup(priv->configCache, filename)))
|
|
goto error;
|
|
|
|
memset(info, 0, sizeof(virDomainInfo));
|
|
info->maxMem = entry->def->mem.max_balloon;
|
|
info->memory = entry->def->mem.cur_balloon;
|
|
info->nrVirtCpu = entry->def->vcpus;
|
|
info->state = VIR_DOMAIN_SHUTOFF;
|
|
info->cpuTime = 0;
|
|
|
|
xenUnifiedUnlock(priv);
|
|
return 0;
|
|
|
|
error:
|
|
xenUnifiedUnlock(priv);
|
|
return -1;
|
|
}
|
|
|
|
|
|
/*
|
|
* Turn a config record into a lump of XML describing the
|
|
* domain, suitable for later feeding for virDomainCreateXML
|
|
*/
|
|
char *
|
|
xenXMDomainGetXMLDesc(virDomainPtr domain, unsigned int flags)
|
|
{
|
|
xenUnifiedPrivatePtr priv = domain->conn->privateData;
|
|
const char *filename;
|
|
xenXMConfCachePtr entry;
|
|
char *ret = NULL;
|
|
|
|
/* Flags checked by virDomainDefFormat */
|
|
|
|
if (domain->id != -1)
|
|
return NULL;
|
|
|
|
xenUnifiedLock(priv);
|
|
|
|
if (!(filename = virHashLookup(priv->nameConfigMap, domain->name)))
|
|
goto cleanup;
|
|
|
|
if (!(entry = virHashLookup(priv->configCache, filename)))
|
|
goto cleanup;
|
|
|
|
ret = virDomainDefFormat(entry->def, flags);
|
|
|
|
cleanup:
|
|
xenUnifiedUnlock(priv);
|
|
return ret;
|
|
}
|
|
|
|
|
|
/*
|
|
* Update amount of memory in the config file
|
|
*/
|
|
int
|
|
xenXMDomainSetMemory(virConnectPtr conn,
|
|
virDomainDefPtr def,
|
|
unsigned long memory)
|
|
{
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
|
const char *filename;
|
|
xenXMConfCachePtr entry;
|
|
int ret = -1;
|
|
|
|
if (memory < 1024 * MIN_XEN_GUEST_SIZE) {
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
_("Memory %lu too small, min %lu"),
|
|
memory, (unsigned long)1024 * MIN_XEN_GUEST_SIZE);
|
|
return -1;
|
|
}
|
|
|
|
xenUnifiedLock(priv);
|
|
|
|
if (!(filename = virHashLookup(priv->nameConfigMap, def->name)))
|
|
goto cleanup;
|
|
|
|
if (!(entry = virHashLookup(priv->configCache, filename)))
|
|
goto cleanup;
|
|
|
|
entry->def->mem.cur_balloon = memory;
|
|
if (entry->def->mem.cur_balloon > entry->def->mem.max_balloon)
|
|
entry->def->mem.cur_balloon = entry->def->mem.max_balloon;
|
|
|
|
/* If this fails, should we try to undo our changes to the
|
|
* in-memory representation of the config file. I say not!
|
|
*/
|
|
if (xenXMConfigSaveFile(conn, entry->filename, entry->def) < 0)
|
|
goto cleanup;
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
xenUnifiedUnlock(priv);
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Update maximum memory limit in config
|
|
*/
|
|
int
|
|
xenXMDomainSetMaxMemory(virConnectPtr conn,
|
|
virDomainDefPtr def,
|
|
unsigned long memory)
|
|
{
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
|
const char *filename;
|
|
xenXMConfCachePtr entry;
|
|
int ret = -1;
|
|
|
|
if (memory < 1024 * MIN_XEN_GUEST_SIZE) {
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
_("Memory %lu too small, min %lu"),
|
|
memory, (unsigned long)1024 * MIN_XEN_GUEST_SIZE);
|
|
return -1;
|
|
}
|
|
|
|
xenUnifiedLock(priv);
|
|
|
|
if (!(filename = virHashLookup(priv->nameConfigMap, def->name)))
|
|
goto cleanup;
|
|
|
|
if (!(entry = virHashLookup(priv->configCache, filename)))
|
|
goto cleanup;
|
|
|
|
entry->def->mem.max_balloon = memory;
|
|
if (entry->def->mem.cur_balloon > entry->def->mem.max_balloon)
|
|
entry->def->mem.cur_balloon = entry->def->mem.max_balloon;
|
|
|
|
/* If this fails, should we try to undo our changes to the
|
|
* in-memory representation of the config file. I say not!
|
|
*/
|
|
if (xenXMConfigSaveFile(conn, entry->filename, entry->def) < 0)
|
|
goto cleanup;
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
xenUnifiedUnlock(priv);
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Get max memory limit from config
|
|
*/
|
|
unsigned long long
|
|
xenXMDomainGetMaxMemory(virConnectPtr conn,
|
|
virDomainDefPtr def)
|
|
{
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
|
const char *filename;
|
|
xenXMConfCachePtr entry;
|
|
unsigned long long ret = 0;
|
|
|
|
xenUnifiedLock(priv);
|
|
|
|
if (!(filename = virHashLookup(priv->nameConfigMap, def->name)))
|
|
goto cleanup;
|
|
|
|
if (!(entry = virHashLookup(priv->configCache, filename)))
|
|
goto cleanup;
|
|
|
|
ret = entry->def->mem.max_balloon;
|
|
|
|
cleanup:
|
|
xenUnifiedUnlock(priv);
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* xenXMDomainSetVcpusFlags:
|
|
* @domain: pointer to domain object
|
|
* @nvcpus: number of vcpus
|
|
* @flags: bitwise-ORd from virDomainVcpuFlags
|
|
*
|
|
* Change virtual CPUs allocation of domain according to flags.
|
|
*
|
|
* Returns 0 on success, -1 if an error message was issued
|
|
*/
|
|
int
|
|
xenXMDomainSetVcpusFlags(virDomainPtr domain,
|
|
unsigned int vcpus,
|
|
unsigned int flags)
|
|
{
|
|
xenUnifiedPrivatePtr priv = domain->conn->privateData;
|
|
const char *filename;
|
|
xenXMConfCachePtr entry;
|
|
int ret = -1;
|
|
int max;
|
|
|
|
virCheckFlags(VIR_DOMAIN_VCPU_LIVE |
|
|
VIR_DOMAIN_VCPU_CONFIG |
|
|
VIR_DOMAIN_VCPU_MAXIMUM, -1);
|
|
|
|
if (flags & VIR_DOMAIN_VCPU_LIVE) {
|
|
virReportError(VIR_ERR_OPERATION_INVALID, "%s",
|
|
_("domain is not running"));
|
|
return -1;
|
|
}
|
|
|
|
xenUnifiedLock(priv);
|
|
|
|
if (!(filename = virHashLookup(priv->nameConfigMap, domain->name)))
|
|
goto cleanup;
|
|
|
|
if (!(entry = virHashLookup(priv->configCache, filename)))
|
|
goto cleanup;
|
|
|
|
/* Hypervisor maximum. */
|
|
if ((max = xenUnifiedConnectGetMaxVcpus(domain->conn, NULL)) < 0) {
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
_("could not determine max vcpus for the domain"));
|
|
goto cleanup;
|
|
}
|
|
/* Can't specify a current larger than stored maximum; but
|
|
* reducing maximum can silently reduce current. */
|
|
if (!(flags & VIR_DOMAIN_VCPU_MAXIMUM))
|
|
max = entry->def->maxvcpus;
|
|
if (vcpus > max) {
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
_("requested vcpus is greater than max allowable"
|
|
" vcpus for the domain: %d > %d"), vcpus, max);
|
|
goto cleanup;
|
|
}
|
|
|
|
if (flags & VIR_DOMAIN_VCPU_MAXIMUM) {
|
|
entry->def->maxvcpus = vcpus;
|
|
if (entry->def->vcpus > vcpus)
|
|
entry->def->vcpus = vcpus;
|
|
} else {
|
|
entry->def->vcpus = vcpus;
|
|
}
|
|
|
|
/* If this fails, should we try to undo our changes to the
|
|
* in-memory representation of the config file. I say not!
|
|
*/
|
|
if (xenXMConfigSaveFile(domain->conn, entry->filename, entry->def) < 0)
|
|
goto cleanup;
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
xenUnifiedUnlock(priv);
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* xenXMDomainGetVcpusFlags:
|
|
* @domain: pointer to domain object
|
|
* @flags: bitwise-ORd from virDomainVcpuFlags
|
|
*
|
|
* Extract information about virtual CPUs of domain according to flags.
|
|
*
|
|
* Returns the number of vcpus on success, -1 if an error message was
|
|
* issued
|
|
*/
|
|
int
|
|
xenXMDomainGetVcpusFlags(virDomainPtr domain, unsigned int flags)
|
|
{
|
|
xenUnifiedPrivatePtr priv = domain->conn->privateData;
|
|
const char *filename;
|
|
xenXMConfCachePtr entry;
|
|
int ret = -2;
|
|
|
|
virCheckFlags(VIR_DOMAIN_VCPU_LIVE |
|
|
VIR_DOMAIN_VCPU_CONFIG |
|
|
VIR_DOMAIN_VCPU_MAXIMUM, -1);
|
|
|
|
if (flags & VIR_DOMAIN_VCPU_LIVE) {
|
|
virReportError(VIR_ERR_OPERATION_FAILED, "%s", _("domain not active"));
|
|
return -1;
|
|
}
|
|
|
|
xenUnifiedLock(priv);
|
|
|
|
if (!(filename = virHashLookup(priv->nameConfigMap, domain->name)))
|
|
goto cleanup;
|
|
|
|
if (!(entry = virHashLookup(priv->configCache, filename)))
|
|
goto cleanup;
|
|
|
|
ret = ((flags & VIR_DOMAIN_VCPU_MAXIMUM) ? entry->def->maxvcpus
|
|
: entry->def->vcpus);
|
|
|
|
cleanup:
|
|
xenUnifiedUnlock(priv);
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* xenXMDomainPinVcpu:
|
|
* @domain: pointer to domain object
|
|
* @vcpu: virtual CPU number (reserved)
|
|
* @cpumap: pointer to a bit map of real CPUs (in 8-bit bytes)
|
|
* @maplen: length of cpumap in bytes
|
|
*
|
|
* Set the vcpu affinity in config
|
|
*
|
|
* Returns 0 for success; -1 (with errno) on error
|
|
*/
|
|
int
|
|
xenXMDomainPinVcpu(virDomainPtr domain,
|
|
unsigned int vcpu ATTRIBUTE_UNUSED,
|
|
unsigned char *cpumap,
|
|
int maplen)
|
|
{
|
|
xenUnifiedPrivatePtr priv = domain->conn->privateData;
|
|
const char *filename;
|
|
xenXMConfCachePtr entry;
|
|
int ret = -1;
|
|
|
|
if (maplen > (int)sizeof(cpumap_t)) {
|
|
virReportError(VIR_ERR_INVALID_ARG, __FUNCTION__);
|
|
return -1;
|
|
}
|
|
|
|
xenUnifiedLock(priv);
|
|
|
|
if (!(filename = virHashLookup(priv->nameConfigMap, domain->name))) {
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("virHashLookup"));
|
|
goto cleanup;
|
|
}
|
|
if (!(entry = virHashLookup(priv->configCache, filename))) {
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
"%s", _("can't retrieve config file for domain"));
|
|
goto cleanup;
|
|
}
|
|
|
|
virBitmapFree(entry->def->cpumask);
|
|
entry->def->cpumask = virBitmapNewData(cpumap, maplen);
|
|
if (!entry->def->cpumask)
|
|
goto cleanup;
|
|
if (xenXMConfigSaveFile(domain->conn, entry->filename, entry->def) < 0)
|
|
goto cleanup;
|
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
xenUnifiedUnlock(priv);
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Find an inactive domain based on its name
|
|
*/
|
|
virDomainDefPtr
|
|
xenXMDomainLookupByName(virConnectPtr conn, const char *domname)
|
|
{
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
|
const char *filename;
|
|
xenXMConfCachePtr entry;
|
|
virDomainDefPtr ret = NULL;
|
|
|
|
xenUnifiedLock(priv);
|
|
|
|
if (!xenInotifyActive(conn) && xenXMConfigCacheRefresh(conn) < 0)
|
|
goto cleanup;
|
|
|
|
if (!(filename = virHashLookup(priv->nameConfigMap, domname)))
|
|
goto cleanup;
|
|
|
|
if (!(entry = virHashLookup(priv->configCache, filename)))
|
|
goto cleanup;
|
|
|
|
ret = virDomainDefNew(domname, entry->def->uuid, -1);
|
|
|
|
cleanup:
|
|
xenUnifiedUnlock(priv);
|
|
return ret;
|
|
}
|
|
|
|
|
|
/*
|
|
* Hash table iterator to search for a domain based on UUID
|
|
*/
|
|
static int
|
|
xenXMDomainSearchForUUID(const void *payload,
|
|
const void *name ATTRIBUTE_UNUSED,
|
|
const void *data)
|
|
{
|
|
const unsigned char *wantuuid = (const unsigned char *)data;
|
|
const xenXMConfCachePtr entry = (const xenXMConfCachePtr)payload;
|
|
|
|
if (!memcmp(entry->def->uuid, wantuuid, VIR_UUID_BUFLEN))
|
|
return 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Find an inactive domain based on its UUID
|
|
*/
|
|
virDomainDefPtr
|
|
xenXMDomainLookupByUUID(virConnectPtr conn, const unsigned char *uuid)
|
|
{
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
|
xenXMConfCachePtr entry;
|
|
virDomainDefPtr ret = NULL;
|
|
|
|
xenUnifiedLock(priv);
|
|
|
|
if (!xenInotifyActive(conn) && xenXMConfigCacheRefresh(conn) < 0)
|
|
goto cleanup;
|
|
|
|
if (!(entry = virHashSearch(priv->configCache, xenXMDomainSearchForUUID, (const void *)uuid)))
|
|
goto cleanup;
|
|
|
|
ret = virDomainDefNew(entry->def->name, uuid, -1);
|
|
|
|
cleanup:
|
|
xenUnifiedUnlock(priv);
|
|
return ret;
|
|
}
|
|
|
|
|
|
/*
|
|
* Start a domain from an existing defined config file
|
|
*/
|
|
int
|
|
xenXMDomainCreate(virDomainPtr domain)
|
|
{
|
|
char *sexpr;
|
|
int ret = -1;
|
|
xenUnifiedPrivatePtr priv= domain->conn->privateData;
|
|
const char *filename;
|
|
xenXMConfCachePtr entry = NULL;
|
|
|
|
xenUnifiedLock(priv);
|
|
|
|
if (!(filename = virHashLookup(priv->nameConfigMap, domain->name)))
|
|
goto error;
|
|
|
|
if (!(entry = virHashLookup(priv->configCache, filename)))
|
|
goto error;
|
|
|
|
if (!(sexpr = xenFormatSxpr(domain->conn, entry->def, priv->xendConfigVersion)))
|
|
goto error;
|
|
|
|
ret = xenDaemonDomainCreateXML(domain->conn, sexpr);
|
|
VIR_FREE(sexpr);
|
|
if (ret != 0)
|
|
goto error;
|
|
|
|
if ((ret = xenDaemonDomainLookupByName_ids(domain->conn, domain->name,
|
|
entry->def->uuid)) < 0)
|
|
goto error;
|
|
domain->id = ret;
|
|
|
|
if (xend_wait_for_devices(domain->conn, domain->name) < 0)
|
|
goto error;
|
|
|
|
if (xenDaemonDomainResume(domain->conn, entry->def) < 0)
|
|
goto error;
|
|
|
|
xenUnifiedUnlock(priv);
|
|
return 0;
|
|
|
|
error:
|
|
if (domain->id != -1 && entry) {
|
|
xenDaemonDomainDestroy(domain->conn, entry->def);
|
|
domain->id = -1;
|
|
}
|
|
xenUnifiedUnlock(priv);
|
|
return -1;
|
|
}
|
|
|
|
/*
|
|
* Create a config file for a domain, based on an XML
|
|
* document describing its config
|
|
*/
|
|
virDomainPtr
|
|
xenXMDomainDefineXML(virConnectPtr conn, const char *xml)
|
|
{
|
|
virDomainPtr ret;
|
|
char *filename = NULL;
|
|
const char *oldfilename;
|
|
virDomainDefPtr def = NULL;
|
|
virConfPtr conf = NULL;
|
|
xenXMConfCachePtr entry = NULL;
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
|
|
|
xenUnifiedLock(priv);
|
|
|
|
if (!xenInotifyActive(conn) && xenXMConfigCacheRefresh(conn) < 0) {
|
|
xenUnifiedUnlock(priv);
|
|
return NULL;
|
|
}
|
|
|
|
if (!(def = virDomainDefParseString(xml, priv->caps, priv->xmlopt,
|
|
1 << VIR_DOMAIN_VIRT_XEN,
|
|
VIR_DOMAIN_XML_INACTIVE))) {
|
|
xenUnifiedUnlock(priv);
|
|
return NULL;
|
|
}
|
|
|
|
if (!(conf = xenFormatXM(conn, def, priv->xendConfigVersion)))
|
|
goto error;
|
|
|
|
/*
|
|
* check that if there is another domain defined with the same uuid
|
|
* it has the same name
|
|
*/
|
|
if ((entry = virHashSearch(priv->configCache, xenXMDomainSearchForUUID,
|
|
(const void *)&(def->uuid))) != NULL) {
|
|
if ((entry->def != NULL) && (entry->def->name != NULL) &&
|
|
(STRNEQ(def->name, entry->def->name))) {
|
|
char uuidstr[VIR_UUID_STRING_BUFLEN];
|
|
|
|
virUUIDFormat(entry->def->uuid, uuidstr);
|
|
virReportError(VIR_ERR_OPERATION_FAILED,
|
|
_("domain '%s' is already defined with uuid %s"),
|
|
entry->def->name, uuidstr);
|
|
entry = NULL;
|
|
goto error;
|
|
}
|
|
entry = NULL;
|
|
}
|
|
|
|
if (virHashLookup(priv->nameConfigMap, def->name)) {
|
|
/* domain exists, we will overwrite it */
|
|
|
|
if (!(oldfilename = (char *)virHashLookup(priv->nameConfigMap, def->name))) {
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
"%s", _("can't retrieve config filename for domain to overwrite"));
|
|
goto error;
|
|
}
|
|
|
|
if (!(entry = virHashLookup(priv->configCache, oldfilename))) {
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
"%s", _("can't retrieve config entry for domain to overwrite"));
|
|
goto error;
|
|
}
|
|
|
|
/* Remove the name -> filename mapping */
|
|
if (virHashRemoveEntry(priv->nameConfigMap, def->name) < 0) {
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
"%s", _("failed to remove old domain from config map"));
|
|
goto error;
|
|
}
|
|
|
|
/* Remove the config record itself */
|
|
if (virHashRemoveEntry(priv->configCache, oldfilename) < 0) {
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
"%s", _("failed to remove old domain from config map"));
|
|
goto error;
|
|
}
|
|
|
|
entry = NULL;
|
|
}
|
|
|
|
if (!(filename = virFileBuildPath(priv->configDir, def->name, NULL)))
|
|
goto error;
|
|
|
|
if (virConfWriteFile(filename, conf) < 0)
|
|
goto error;
|
|
|
|
if (VIR_ALLOC(entry) < 0) {
|
|
virReportOOMError();
|
|
goto error;
|
|
}
|
|
|
|
if ((entry->refreshedAt = time(NULL)) == ((time_t)-1)) {
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
"%s", _("unable to get current time"));
|
|
goto error;
|
|
}
|
|
|
|
if ((entry->filename = strdup(filename)) == NULL) {
|
|
virReportOOMError();
|
|
goto error;
|
|
}
|
|
entry->def = def;
|
|
|
|
if (virHashAddEntry(priv->configCache, filename, entry) < 0) {
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
"%s", _("unable to store config file handle"));
|
|
goto error;
|
|
}
|
|
|
|
if (virHashAddEntry(priv->nameConfigMap, def->name, entry->filename) < 0) {
|
|
virHashSteal(priv->configCache, filename);
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
"%s", _("unable to store config file handle"));
|
|
goto error;
|
|
}
|
|
|
|
ret = virGetDomain(conn, def->name, def->uuid);
|
|
xenUnifiedUnlock(priv);
|
|
VIR_FREE(filename);
|
|
return ret;
|
|
|
|
error:
|
|
VIR_FREE(filename);
|
|
if (entry)
|
|
VIR_FREE(entry->filename);
|
|
VIR_FREE(entry);
|
|
virConfFree(conf);
|
|
virDomainDefFree(def);
|
|
xenUnifiedUnlock(priv);
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* Delete a domain from disk
|
|
*/
|
|
int
|
|
xenXMDomainUndefine(virDomainPtr domain)
|
|
{
|
|
xenUnifiedPrivatePtr priv = domain->conn->privateData;
|
|
const char *filename;
|
|
xenXMConfCachePtr entry;
|
|
int ret = -1;
|
|
|
|
xenUnifiedLock(priv);
|
|
|
|
if (!(filename = virHashLookup(priv->nameConfigMap, domain->name)))
|
|
goto cleanup;
|
|
|
|
if (!(entry = virHashLookup(priv->configCache, filename)))
|
|
goto cleanup;
|
|
|
|
if (unlink(entry->filename) < 0)
|
|
goto cleanup;
|
|
|
|
/* Remove the name -> filename mapping */
|
|
if (virHashRemoveEntry(priv->nameConfigMap, domain->name) < 0)
|
|
goto cleanup;
|
|
|
|
/* Remove the config record itself */
|
|
if (virHashRemoveEntry(priv->configCache, entry->filename) < 0)
|
|
goto cleanup;
|
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
xenUnifiedUnlock(priv);
|
|
return ret;
|
|
}
|
|
|
|
struct xenXMListIteratorContext {
|
|
virConnectPtr conn;
|
|
int oom;
|
|
int max;
|
|
int count;
|
|
char ** names;
|
|
};
|
|
|
|
static void
|
|
xenXMListIterator(void *payload ATTRIBUTE_UNUSED, const void *name, void *data) {
|
|
struct xenXMListIteratorContext *ctx = data;
|
|
virDomainDefPtr def = NULL;
|
|
|
|
if (ctx->oom)
|
|
return;
|
|
|
|
if (ctx->count == ctx->max)
|
|
return;
|
|
|
|
def = xenDaemonLookupByName(ctx->conn, name);
|
|
if (!def) {
|
|
if (!(ctx->names[ctx->count] = strdup(name)))
|
|
ctx->oom = 1;
|
|
else
|
|
ctx->count++;
|
|
} else {
|
|
virDomainDefFree(def);
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* List all defined domains, filtered to remove any which
|
|
* are currently running
|
|
*/
|
|
int
|
|
xenXMListDefinedDomains(virConnectPtr conn, char **const names, int maxnames)
|
|
{
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
|
struct xenXMListIteratorContext ctx;
|
|
int i, ret = -1;
|
|
|
|
xenUnifiedLock(priv);
|
|
|
|
if (!xenInotifyActive(conn) && xenXMConfigCacheRefresh(conn) < 0)
|
|
goto cleanup;
|
|
|
|
if (maxnames > virHashSize(priv->configCache))
|
|
maxnames = virHashSize(priv->configCache);
|
|
|
|
ctx.conn = conn;
|
|
ctx.oom = 0;
|
|
ctx.count = 0;
|
|
ctx.max = maxnames;
|
|
ctx.names = names;
|
|
|
|
virHashForEach(priv->nameConfigMap, xenXMListIterator, &ctx);
|
|
|
|
if (ctx.oom) {
|
|
for (i = 0; i < ctx.count; i++)
|
|
VIR_FREE(ctx.names[i]);
|
|
|
|
virReportOOMError();
|
|
goto cleanup;
|
|
}
|
|
|
|
ret = ctx.count;
|
|
|
|
cleanup:
|
|
xenUnifiedUnlock(priv);
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Return the maximum number of defined domains - not filtered
|
|
* based on number running
|
|
*/
|
|
int
|
|
xenXMNumOfDefinedDomains(virConnectPtr conn)
|
|
{
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
|
int ret = -1;
|
|
|
|
xenUnifiedLock(priv);
|
|
|
|
if (!xenInotifyActive(conn) && xenXMConfigCacheRefresh(conn) < 0)
|
|
goto cleanup;
|
|
|
|
ret = virHashSize(priv->nameConfigMap);
|
|
|
|
cleanup:
|
|
xenUnifiedUnlock(priv);
|
|
return ret;
|
|
}
|
|
|
|
|
|
/**
|
|
* xenXMDomainAttachDeviceFlags:
|
|
* @domain: pointer to domain object
|
|
* @xml: pointer to XML description of device
|
|
* @flags: an OR'ed set of virDomainDeviceModifyFlags
|
|
*
|
|
* Create a virtual device attachment to backend.
|
|
* XML description is translated into config file.
|
|
* This driver only supports device allocation to
|
|
* persisted config.
|
|
*
|
|
* Returns 0 in case of success, -1 in case of failure.
|
|
*/
|
|
int
|
|
xenXMDomainAttachDeviceFlags(virDomainPtr domain,
|
|
const char *xml,
|
|
unsigned int flags)
|
|
{
|
|
const char *filename = NULL;
|
|
xenXMConfCachePtr entry = NULL;
|
|
int ret = -1;
|
|
virDomainDeviceDefPtr dev = NULL;
|
|
virDomainDefPtr def;
|
|
xenUnifiedPrivatePtr priv = domain->conn->privateData;
|
|
|
|
virCheckFlags(VIR_DOMAIN_AFFECT_LIVE | VIR_DOMAIN_AFFECT_CONFIG, -1);
|
|
|
|
if ((flags & VIR_DOMAIN_DEVICE_MODIFY_LIVE) ||
|
|
(domain->id != -1 && flags == VIR_DOMAIN_DEVICE_MODIFY_CURRENT)) {
|
|
virReportError(VIR_ERR_OPERATION_INVALID, "%s",
|
|
_("Xm driver only supports modifying persistent config"));
|
|
return -1;
|
|
}
|
|
|
|
xenUnifiedLock(priv);
|
|
|
|
if (!(filename = virHashLookup(priv->nameConfigMap, domain->name)))
|
|
goto cleanup;
|
|
if (!(entry = virHashLookup(priv->configCache, filename)))
|
|
goto cleanup;
|
|
def = entry->def;
|
|
|
|
if (!(dev = virDomainDeviceDefParse(xml, entry->def,
|
|
priv->caps,
|
|
priv->xmlopt,
|
|
VIR_DOMAIN_XML_INACTIVE)))
|
|
goto cleanup;
|
|
|
|
switch (dev->type) {
|
|
case VIR_DOMAIN_DEVICE_DISK:
|
|
{
|
|
if (virDomainDiskInsert(def, dev->data.disk) < 0) {
|
|
virReportOOMError();
|
|
goto cleanup;
|
|
}
|
|
dev->data.disk = NULL;
|
|
}
|
|
break;
|
|
|
|
case VIR_DOMAIN_DEVICE_NET:
|
|
{
|
|
if (VIR_REALLOC_N(def->nets, def->nnets+1) < 0) {
|
|
virReportOOMError();
|
|
goto cleanup;
|
|
}
|
|
def->nets[def->nnets++] = dev->data.net;
|
|
dev->data.net = NULL;
|
|
break;
|
|
}
|
|
|
|
default:
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
_("Xm driver only supports adding disk or network devices"));
|
|
goto cleanup;
|
|
}
|
|
|
|
/* If this fails, should we try to undo our changes to the
|
|
* in-memory representation of the config file. I say not!
|
|
*/
|
|
if (xenXMConfigSaveFile(domain->conn, entry->filename, entry->def) < 0)
|
|
goto cleanup;
|
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
virDomainDeviceDefFree(dev);
|
|
xenUnifiedUnlock(priv);
|
|
return ret;
|
|
}
|
|
|
|
|
|
/**
|
|
* xenXMDomainDetachDeviceFlags:
|
|
* @domain: pointer to domain object
|
|
* @xml: pointer to XML description of device
|
|
* @flags: an OR'ed set of virDomainDeviceModifyFlags
|
|
*
|
|
* Destroy a virtual device attachment to backend.
|
|
* This driver only supports device deallocation from
|
|
* persisted config.
|
|
*
|
|
* Returns 0 in case of success, -1 in case of failure.
|
|
*/
|
|
int
|
|
xenXMDomainDetachDeviceFlags(virDomainPtr domain,
|
|
const char *xml,
|
|
unsigned int flags)
|
|
{
|
|
const char *filename = NULL;
|
|
xenXMConfCachePtr entry = NULL;
|
|
virDomainDeviceDefPtr dev = NULL;
|
|
virDomainDefPtr def;
|
|
int ret = -1;
|
|
int i;
|
|
xenUnifiedPrivatePtr priv = domain->conn->privateData;
|
|
|
|
virCheckFlags(VIR_DOMAIN_AFFECT_LIVE | VIR_DOMAIN_AFFECT_CONFIG, -1);
|
|
|
|
if ((flags & VIR_DOMAIN_DEVICE_MODIFY_LIVE) ||
|
|
(domain->id != -1 && flags == VIR_DOMAIN_DEVICE_MODIFY_CURRENT)) {
|
|
virReportError(VIR_ERR_OPERATION_INVALID, "%s",
|
|
_("Xm driver only supports modifying persistent config"));
|
|
return -1;
|
|
}
|
|
|
|
xenUnifiedLock(priv);
|
|
|
|
if (!(filename = virHashLookup(priv->nameConfigMap, domain->name)))
|
|
goto cleanup;
|
|
if (!(entry = virHashLookup(priv->configCache, filename)))
|
|
goto cleanup;
|
|
def = entry->def;
|
|
|
|
if (!(dev = virDomainDeviceDefParse(xml, entry->def,
|
|
priv->caps,
|
|
priv->xmlopt,
|
|
VIR_DOMAIN_XML_INACTIVE)))
|
|
goto cleanup;
|
|
|
|
switch (dev->type) {
|
|
case VIR_DOMAIN_DEVICE_DISK:
|
|
{
|
|
for (i = 0 ; i < def->ndisks ; i++) {
|
|
if (def->disks[i]->dst &&
|
|
dev->data.disk->dst &&
|
|
STREQ(def->disks[i]->dst, dev->data.disk->dst)) {
|
|
virDomainDiskDefFree(def->disks[i]);
|
|
if (i < (def->ndisks - 1))
|
|
memmove(def->disks + i,
|
|
def->disks + i + 1,
|
|
sizeof(*def->disks) *
|
|
(def->ndisks - (i + 1)));
|
|
def->ndisks--;
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
case VIR_DOMAIN_DEVICE_NET:
|
|
{
|
|
for (i = 0 ; i < def->nnets ; i++) {
|
|
if (!virMacAddrCmp(&def->nets[i]->mac, &dev->data.net->mac)) {
|
|
virDomainNetDefFree(def->nets[i]);
|
|
if (i < (def->nnets - 1))
|
|
memmove(def->nets + i,
|
|
def->nets + i + 1,
|
|
sizeof(*def->nets) *
|
|
(def->nnets - (i + 1)));
|
|
def->nnets--;
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
default:
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
_("device type '%s' cannot be detached"),
|
|
virDomainDeviceTypeToString(dev->type));
|
|
goto cleanup;
|
|
}
|
|
|
|
/* If this fails, should we try to undo our changes to the
|
|
* in-memory representation of the config file. I say not!
|
|
*/
|
|
if (xenXMConfigSaveFile(domain->conn, entry->filename, entry->def) < 0)
|
|
goto cleanup;
|
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
virDomainDeviceDefFree(dev);
|
|
xenUnifiedUnlock(priv);
|
|
return ret;
|
|
}
|
|
|
|
int
|
|
xenXMDomainBlockPeek(virDomainPtr dom ATTRIBUTE_UNUSED,
|
|
const char *path ATTRIBUTE_UNUSED,
|
|
unsigned long long offset ATTRIBUTE_UNUSED,
|
|
size_t size ATTRIBUTE_UNUSED,
|
|
void *buffer ATTRIBUTE_UNUSED)
|
|
{
|
|
virReportError(VIR_ERR_OPERATION_INVALID, "%s",
|
|
_("block peeking not implemented"));
|
|
return -1;
|
|
}
|
|
|
|
|
|
static char *
|
|
xenXMAutostartLinkName(virDomainPtr dom)
|
|
{
|
|
char *ret;
|
|
if (virAsprintf(&ret, "/etc/xen/auto/%s", dom->name) < 0)
|
|
return NULL;
|
|
return ret;
|
|
}
|
|
|
|
static char *
|
|
xenXMDomainConfigName(virDomainPtr dom)
|
|
{
|
|
char *ret;
|
|
if (virAsprintf(&ret, "/etc/xen/%s", dom->name) < 0)
|
|
return NULL;
|
|
return ret;
|
|
}
|
|
|
|
int
|
|
xenXMDomainGetAutostart(virDomainPtr dom, int *autostart)
|
|
{
|
|
char *linkname = xenXMAutostartLinkName(dom);
|
|
char *config = xenXMDomainConfigName(dom);
|
|
int ret = -1;
|
|
|
|
if (!linkname || !config) {
|
|
virReportOOMError();
|
|
goto cleanup;
|
|
}
|
|
|
|
*autostart = virFileLinkPointsTo(linkname, config);
|
|
if (*autostart < 0) {
|
|
virReportSystemError(errno,
|
|
_("cannot check link %s points to config %s"),
|
|
linkname, config);
|
|
goto cleanup;
|
|
}
|
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
VIR_FREE(linkname);
|
|
VIR_FREE(config);
|
|
return ret;
|
|
}
|
|
|
|
|
|
int
|
|
xenXMDomainSetAutostart(virDomainPtr dom, int autostart)
|
|
{
|
|
char *linkname = xenXMAutostartLinkName(dom);
|
|
char *config = xenXMDomainConfigName(dom);
|
|
int ret = -1;
|
|
|
|
if (!linkname || !config) {
|
|
virReportOOMError();
|
|
goto cleanup;
|
|
}
|
|
|
|
if (autostart) {
|
|
if (symlink(config, linkname) < 0 &&
|
|
errno != EEXIST) {
|
|
virReportSystemError(errno,
|
|
_("failed to create link %s to %s"),
|
|
config, linkname);
|
|
goto cleanup;
|
|
}
|
|
} else {
|
|
if (unlink(linkname) < 0 &&
|
|
errno != ENOENT) {
|
|
virReportSystemError(errno,
|
|
_("failed to remove link %s"),
|
|
linkname);
|
|
goto cleanup;
|
|
}
|
|
}
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
VIR_FREE(linkname);
|
|
VIR_FREE(config);
|
|
|
|
return ret;
|
|
}
|