2006-01-12 15:38:07 +00:00
|
|
|
/*
|
|
|
|
* xend_internal.c: access to Xen though the Xen Daemon interface
|
|
|
|
*
|
2014-03-18 17:12:43 -06:00
|
|
|
* Copyright (C) 2010-2014 Red Hat, Inc.
|
2010-03-01 16:38:28 -07:00
|
|
|
* Copyright (C) 2005 Anthony Liguori <aliguori@us.ibm.com>
|
2006-01-12 15:38:07 +00:00
|
|
|
*
|
2013-05-14 17:42:12 -06: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
|
|
|
|
* License along with this library. If not, see
|
|
|
|
* <http://www.gnu.org/licenses/>.
|
2006-01-12 15:38:07 +00:00
|
|
|
*/
|
|
|
|
|
2008-01-29 18:15:54 +00:00
|
|
|
#include <config.h>
|
2007-08-21 09:31:12 +00:00
|
|
|
|
2006-01-12 15:38:07 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <sys/un.h>
|
|
|
|
#include <sys/errno.h>
|
virDomainBlockPeek call
* configure.in: Document AC_SYS_LARGEFILE.
* docs/hvsupport.html.in: Document HV support for virDomainBlockPeek.
* include/libvirt/libvirt.h.in, src/driver.h, src/libvirt.c,
src/libvirt_sym.version: Add virDomainBlockPeek infrastructure.
* src/qemu_driver.c, src/test.c: Null versions of this call.
* src/xen_unified.c, src/xend_internal.c, src/xend_internal.h,
src/xm_internal.c, src/xm_internal.h: Xen implementation.
* tests/sexpr2xmldata/sexpr2xml-curmem.xml,
tests/sexpr2xmldata/sexpr2xml-no-source-cdrom.xml: XML output
has been reordered slightly in the Xen driver, but should be
functionally the same.
2008-06-05 13:17:45 +00:00
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <fcntl.h>
|
2006-01-12 15:38:07 +00:00
|
|
|
#include <unistd.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <math.h>
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include <netinet/in.h>
|
|
|
|
#include <netinet/tcp.h>
|
|
|
|
#include <arpa/inet.h>
|
|
|
|
#include <netdb.h>
|
2007-06-15 15:24:20 +00:00
|
|
|
#include <errno.h>
|
2006-01-12 15:38:07 +00:00
|
|
|
|
2012-12-13 18:21:53 +00:00
|
|
|
#include "virerror.h"
|
2012-12-12 17:59:27 +00:00
|
|
|
#include "virlog.h"
|
2008-11-04 23:22:06 +00:00
|
|
|
#include "datatypes.h"
|
2008-07-25 13:17:27 +00:00
|
|
|
#include "xend_internal.h"
|
2006-03-22 13:44:01 +00:00
|
|
|
#include "driver.h"
|
2012-12-13 15:07:43 +00:00
|
|
|
#include "virsexpr.h"
|
2011-02-21 14:40:08 +01:00
|
|
|
#include "xen_sxpr.h"
|
2012-12-04 12:04:07 +00:00
|
|
|
#include "virbuffer.h"
|
2012-12-13 18:01:25 +00:00
|
|
|
#include "viruuid.h"
|
Move xen driver code into src/xen/ directory
* src/Makefile.am, src/proxy_internal.c, src/proxy_internal.h
src/sexpr.c, src/sexpr.h, src/xen_unified.c, src/xen_unified.h,
src/xen_internal.c, src/xen_internal.h, src/xen_inotify.c,
src/xen_inotify.h, src/xend_internal.c, src/xend_internal.h,
src/xm_internal.c, src/xm_internal.h, src/xs_internal.c,
src/xs_internal.h: Move to src/xen/ directory
* proxy/Makefile.am, proxy/libvirt_proxy.c, src/Makefile.am,
src/libvirt.c, tests/sexpr2xmltest.c, tests/statstest.c,
tests/xencapstest.c, tests/xmconfigtest.c, tests/xml2sexprtest.c:
Adapt to changed xen location
* src/stats_linux.h, src/stats_linux.c: Remove xen specific block
stats APIs
* src/qemu_driver.c, src/uml_driver.c: Add missing sys/un.h include
uncovered after change to stats_linux.h
* src/xen/block_stats.h, src/xen/block_stats.c: Add xen specific
block stats APIs
2009-09-15 16:38:33 +01:00
|
|
|
#include "xen_driver.h"
|
|
|
|
#include "xen_hypervisor.h"
|
2006-08-08 20:14:40 +00:00
|
|
|
#include "xs_internal.h" /* To extract VNC port & Serial console TTY */
|
2012-12-12 18:06:53 +00:00
|
|
|
#include "viralloc.h"
|
2010-10-05 08:18:52 -06:00
|
|
|
#include "count-one-bits.h"
|
2011-07-19 12:32:58 -06:00
|
|
|
#include "virfile.h"
|
2012-02-24 19:48:55 +01:00
|
|
|
#include "viruri.h"
|
2012-08-16 16:41:06 +01:00
|
|
|
#include "device_conf.h"
|
2013-04-03 12:36:23 +02:00
|
|
|
#include "virstring.h"
|
2006-01-12 15:38:07 +00:00
|
|
|
|
2007-03-15 07:43:16 +00:00
|
|
|
/* required for cpumap_t */
|
|
|
|
#include <xen/dom0_ops.h>
|
|
|
|
|
2009-01-20 17:13:33 +00:00
|
|
|
#define VIR_FROM_THIS VIR_FROM_XEND
|
|
|
|
|
2014-02-28 12:16:17 +00:00
|
|
|
VIR_LOG_INIT("xen.xend_internal");
|
|
|
|
|
2008-03-24 09:23:32 +00:00
|
|
|
/*
|
|
|
|
* The number of Xen scheduler parameters
|
|
|
|
*/
|
2006-03-29 12:46:03 +00:00
|
|
|
|
2011-09-01 16:40:46 -06:00
|
|
|
#define XEND_RCV_BUF_MAX_LEN (256 * 1024)
|
2006-02-27 16:27:18 +00:00
|
|
|
|
2009-04-03 12:38:52 +00:00
|
|
|
static int
|
2013-05-02 10:56:31 +01:00
|
|
|
virDomainXMLDevID(virConnectPtr conn, virDomainDefPtr domain,
|
|
|
|
virDomainDeviceDefPtr dev, char *class,
|
2013-01-29 09:53:13 -05:00
|
|
|
char *ref, int ref_len);
|
2008-07-25 13:17:27 +00:00
|
|
|
|
2006-01-12 15:38:07 +00:00
|
|
|
/**
|
|
|
|
* do_connect:
|
|
|
|
* @xend: pointer to the Xen Daemon structure
|
|
|
|
*
|
|
|
|
* Internal routine to (re)connect to the daemon
|
|
|
|
*
|
|
|
|
* Returns the socket file descriptor or -1 in case of error
|
|
|
|
*/
|
|
|
|
static int
|
2006-01-13 16:41:01 +00:00
|
|
|
do_connect(virConnectPtr xend)
|
2006-01-12 15:38:07 +00:00
|
|
|
{
|
|
|
|
int s;
|
2006-07-11 16:57:03 +00:00
|
|
|
int no_slow_start = 1;
|
2013-01-29 09:53:13 -05:00
|
|
|
xenUnifiedPrivatePtr priv = xend->privateData;
|
2006-01-12 15:38:07 +00:00
|
|
|
|
2009-01-22 19:41:48 +00:00
|
|
|
s = socket(priv->addrfamily, SOCK_STREAM, priv->addrprotocol);
|
2006-02-27 16:27:18 +00:00
|
|
|
if (s == -1) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
"%s", _("failed to create a socket"));
|
2006-01-12 15:38:07 +00:00
|
|
|
return -1;
|
2006-02-27 16:27:18 +00:00
|
|
|
}
|
2006-01-12 15:38:07 +00:00
|
|
|
|
2006-07-11 16:57:03 +00:00
|
|
|
/*
|
2013-01-08 12:40:21 -05:00
|
|
|
* try to deactivate slow-start
|
2006-07-11 16:57:03 +00:00
|
|
|
*/
|
2013-01-08 12:40:21 -05:00
|
|
|
ignore_value(setsockopt(s, IPPROTO_TCP, TCP_NODELAY, (void *)&no_slow_start,
|
|
|
|
sizeof(no_slow_start)));
|
2006-07-11 16:57:03 +00:00
|
|
|
|
2009-01-22 19:41:48 +00:00
|
|
|
if (connect(s, (struct sockaddr *)&priv->addr, priv->addrlen) == -1) {
|
2010-11-09 15:48:48 -05:00
|
|
|
VIR_FORCE_CLOSE(s); /* preserves errno */
|
2008-03-17 17:30:48 +00:00
|
|
|
|
|
|
|
/*
|
2009-01-22 17:49:41 +00:00
|
|
|
* Connecting to XenD when privileged is mandatory, so log this
|
|
|
|
* error
|
2008-03-17 17:30:48 +00:00
|
|
|
*/
|
2009-01-22 17:49:41 +00:00
|
|
|
if (xenHavePrivilege()) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
"%s", _("failed to connect to xend"));
|
2007-11-27 14:39:42 +00:00
|
|
|
}
|
2006-01-12 15:38:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* wr_sync:
|
2006-11-07 21:52:44 +00:00
|
|
|
* @xend: the xend connection object
|
2006-01-12 15:38:07 +00:00
|
|
|
* @fd: the file descriptor
|
|
|
|
* @buffer: the I/O buffer
|
|
|
|
* @size: the size of the I/O
|
|
|
|
* @do_read: write operation if 0, read operation otherwise
|
|
|
|
*
|
|
|
|
* Do a synchronous read or write on the file descriptor
|
|
|
|
*
|
|
|
|
* Returns the number of bytes exchanged, or -1 in case of error
|
|
|
|
*/
|
|
|
|
static size_t
|
2010-04-03 01:10:49 +02:00
|
|
|
wr_sync(int fd, void *buffer, size_t size, int do_read)
|
2006-01-12 15:38:07 +00:00
|
|
|
{
|
|
|
|
size_t offset = 0;
|
|
|
|
|
|
|
|
while (offset < size) {
|
|
|
|
ssize_t len;
|
|
|
|
|
|
|
|
if (do_read) {
|
2006-03-15 12:13:25 +00:00
|
|
|
len = read(fd, ((char *) buffer) + offset, size - offset);
|
2006-01-12 15:38:07 +00:00
|
|
|
} else {
|
2006-03-15 12:13:25 +00:00
|
|
|
len = write(fd, ((char *) buffer) + offset, size - offset);
|
2006-01-12 15:38:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* recoverable error, retry */
|
2014-11-13 15:29:45 +01:00
|
|
|
if ((len == -1) && ((errno == EAGAIN) || (errno == EINTR)))
|
2006-01-12 15:38:07 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
/* eof */
|
2014-11-13 15:29:45 +01:00
|
|
|
if (len == 0)
|
2006-01-12 15:38:07 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
/* unrecoverable error */
|
|
|
|
if (len == -1) {
|
2006-03-15 12:13:25 +00:00
|
|
|
if (do_read)
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
"%s", _("failed to read from Xen Daemon"));
|
2006-03-15 12:13:25 +00:00
|
|
|
else
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
"%s", _("failed to write to Xen Daemon"));
|
2006-03-15 12:13:25 +00:00
|
|
|
|
2012-03-22 12:33:35 +01:00
|
|
|
return -1;
|
2006-01-12 15:38:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
offset += len;
|
|
|
|
}
|
|
|
|
|
|
|
|
return offset;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* sread:
|
|
|
|
* @fd: the file descriptor
|
|
|
|
* @buffer: the I/O buffer
|
|
|
|
* @size: the size of the I/O
|
|
|
|
*
|
|
|
|
* Internal routine to do a synchronous read
|
|
|
|
*
|
|
|
|
* Returns the number of bytes read, or -1 in case of error
|
|
|
|
*/
|
|
|
|
static ssize_t
|
2010-04-03 01:10:49 +02:00
|
|
|
sread(int fd, void *buffer, size_t size)
|
2006-01-12 15:38:07 +00:00
|
|
|
{
|
2010-04-03 01:10:49 +02:00
|
|
|
return wr_sync(fd, buffer, size, 1);
|
2006-01-12 15:38:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* swrite:
|
|
|
|
* @fd: the file descriptor
|
|
|
|
* @buffer: the I/O buffer
|
|
|
|
* @size: the size of the I/O
|
|
|
|
*
|
|
|
|
* Internal routine to do a synchronous write
|
|
|
|
*
|
|
|
|
* Returns the number of bytes written, or -1 in case of error
|
|
|
|
*/
|
|
|
|
static ssize_t
|
2010-04-03 01:10:49 +02:00
|
|
|
swrite(int fd, const void *buffer, size_t size)
|
2006-01-12 15:38:07 +00:00
|
|
|
{
|
2010-04-03 01:10:49 +02:00
|
|
|
return wr_sync(fd, (void *) buffer, size, 0);
|
2006-01-12 15:38:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* swrites:
|
|
|
|
* @fd: the file descriptor
|
|
|
|
* @string: the string to write
|
|
|
|
*
|
|
|
|
* Internal routine to do a synchronous write of a string
|
|
|
|
*
|
|
|
|
* Returns the number of bytes written, or -1 in case of error
|
|
|
|
*/
|
|
|
|
static ssize_t
|
2010-04-03 01:10:49 +02:00
|
|
|
swrites(int fd, const char *string)
|
2006-01-12 15:38:07 +00:00
|
|
|
{
|
2010-04-03 01:10:49 +02:00
|
|
|
return swrite(fd, string, strlen(string));
|
2006-01-12 15:38:07 +00:00
|
|
|
}
|
|
|
|
|
2006-07-06 09:29:34 +00:00
|
|
|
/**
|
|
|
|
* sreads:
|
|
|
|
* @fd: the file descriptor
|
|
|
|
* @buffer: the I/O buffer
|
|
|
|
* @n_buffer: the size of the I/O buffer
|
|
|
|
*
|
|
|
|
* Internal routine to do a synchronous read of a line
|
|
|
|
*
|
|
|
|
* Returns the number of bytes read, or -1 in case of error
|
|
|
|
*/
|
|
|
|
static ssize_t
|
2010-04-03 01:10:49 +02:00
|
|
|
sreads(int fd, char *buffer, size_t n_buffer)
|
2006-07-06 09:29:34 +00:00
|
|
|
{
|
|
|
|
size_t offset;
|
|
|
|
|
|
|
|
if (n_buffer < 1)
|
2012-03-22 12:33:35 +01:00
|
|
|
return -1;
|
2006-07-06 09:29:34 +00:00
|
|
|
|
|
|
|
for (offset = 0; offset < (n_buffer - 1); offset++) {
|
|
|
|
ssize_t ret;
|
|
|
|
|
2010-04-03 01:10:49 +02:00
|
|
|
ret = sread(fd, buffer + offset, 1);
|
2006-07-06 09:29:34 +00:00
|
|
|
if (ret == 0)
|
|
|
|
break;
|
|
|
|
else if (ret == -1)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
if (buffer[offset] == '\n') {
|
|
|
|
offset++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
buffer[offset] = 0;
|
|
|
|
|
|
|
|
return offset;
|
|
|
|
}
|
2006-01-12 15:38:07 +00:00
|
|
|
|
|
|
|
static int
|
|
|
|
istartswith(const char *haystack, const char *needle)
|
|
|
|
{
|
2008-05-14 19:51:24 +00:00
|
|
|
return STRCASEEQLEN(haystack, needle, strlen(needle));
|
2006-01-12 15:38:07 +00:00
|
|
|
}
|
|
|
|
|
2006-03-30 16:37:15 +00:00
|
|
|
|
2006-01-12 15:38:07 +00:00
|
|
|
/**
|
|
|
|
* xend_req:
|
|
|
|
* @fd: the file descriptor
|
|
|
|
* @content: the buffer to store the content
|
|
|
|
*
|
|
|
|
* Read the HTTP response from a Xen Daemon request.
|
2010-06-02 18:07:17 -06:00
|
|
|
* If the response contains content, memory is allocated to
|
|
|
|
* hold the content.
|
2006-01-12 15:38:07 +00:00
|
|
|
*
|
2010-06-02 18:07:17 -06:00
|
|
|
* Returns the HTTP return code and @content is set to the
|
|
|
|
* allocated memory containing HTTP content.
|
2006-01-12 15:38:07 +00:00
|
|
|
*/
|
2012-10-17 10:23:12 +01:00
|
|
|
static int ATTRIBUTE_NONNULL(2)
|
2010-06-02 18:07:17 -06:00
|
|
|
xend_req(int fd, char **content)
|
2006-01-12 15:38:07 +00:00
|
|
|
{
|
2011-04-03 11:21:32 +02:00
|
|
|
char *buffer;
|
|
|
|
size_t buffer_size = 4096;
|
2010-06-02 18:07:17 -06:00
|
|
|
int content_length = 0;
|
2006-01-12 15:38:07 +00:00
|
|
|
int retcode = 0;
|
2014-01-28 18:15:48 -07:00
|
|
|
char *end_ptr;
|
2006-01-12 15:38:07 +00:00
|
|
|
|
2013-07-04 12:18:49 +02:00
|
|
|
if (VIR_ALLOC_N(buffer, buffer_size) < 0)
|
2011-04-03 11:21:32 +02:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
while (sreads(fd, buffer, buffer_size) > 0) {
|
2008-05-14 19:51:24 +00:00
|
|
|
if (STREQ(buffer, "\r\n"))
|
2006-01-12 15:38:07 +00:00
|
|
|
break;
|
2006-07-06 09:29:34 +00:00
|
|
|
|
2013-11-14 17:14:26 +01:00
|
|
|
if (istartswith(buffer, "Content-Length: ")) {
|
2014-01-28 18:15:48 -07:00
|
|
|
if (virStrToLong_i(buffer + 16, &end_ptr, 10, &content_length) < 0) {
|
2013-11-14 17:14:26 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("failed to parse Xend response content length"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
} else if (istartswith(buffer, "HTTP/1.1 ")) {
|
2014-01-28 18:15:48 -07:00
|
|
|
if (virStrToLong_i(buffer + 9, &end_ptr, 10, &retcode) < 0) {
|
2013-11-14 17:14:26 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("failed to parse Xend response return code"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
2006-01-12 15:38:07 +00:00
|
|
|
}
|
|
|
|
|
2011-04-03 11:21:32 +02:00
|
|
|
VIR_FREE(buffer);
|
|
|
|
|
2010-06-02 18:07:17 -06:00
|
|
|
if (content_length > 0) {
|
2006-07-06 09:29:34 +00:00
|
|
|
ssize_t ret;
|
|
|
|
|
2010-06-04 10:04:03 -06:00
|
|
|
if (content_length > XEND_RCV_BUF_MAX_LEN) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Xend returned HTTP Content-Length of %d, "
|
|
|
|
"which exceeds maximum of %d"),
|
|
|
|
content_length,
|
|
|
|
XEND_RCV_BUF_MAX_LEN);
|
2010-06-04 10:04:03 -06:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Allocate one byte beyond the end of the largest buffer we will read.
|
|
|
|
Combined with the fact that VIR_ALLOC_N zeros the returned buffer,
|
|
|
|
this guarantees that "content" will always be NUL-terminated. */
|
2013-07-04 12:18:49 +02:00
|
|
|
if (VIR_ALLOC_N(*content, content_length + 1) < 0)
|
2010-06-02 18:07:17 -06:00
|
|
|
return -1;
|
2006-01-12 15:38:07 +00:00
|
|
|
|
2010-06-02 18:07:17 -06:00
|
|
|
ret = sread(fd, *content, content_length);
|
2006-07-06 09:29:34 +00:00
|
|
|
if (ret < 0)
|
|
|
|
return -1;
|
2006-01-12 15:38:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return retcode;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* xend_get:
|
|
|
|
* @xend: pointer to the Xen Daemon structure
|
|
|
|
* @path: the path used for the HTTP request
|
|
|
|
* @content: the buffer to store the content
|
|
|
|
*
|
|
|
|
* Do an HTTP GET RPC with the Xen Daemon
|
|
|
|
*
|
|
|
|
* Returns the HTTP return code or -1 in case or error.
|
|
|
|
*/
|
2010-06-04 10:04:03 -06:00
|
|
|
static int ATTRIBUTE_NONNULL(3)
|
2013-01-29 09:53:13 -05:00
|
|
|
xend_get(virConnectPtr xend, const char *path, char **content)
|
2006-01-12 15:38:07 +00:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
int s = do_connect(xend);
|
|
|
|
|
|
|
|
if (s == -1)
|
|
|
|
return s;
|
|
|
|
|
2010-04-03 01:10:49 +02:00
|
|
|
swrites(s, "GET ");
|
|
|
|
swrites(s, path);
|
|
|
|
swrites(s, " HTTP/1.1\r\n");
|
2006-01-12 15:38:07 +00:00
|
|
|
|
2010-04-03 01:10:49 +02:00
|
|
|
swrites(s,
|
2006-01-12 15:38:07 +00:00
|
|
|
"Host: localhost:8000\r\n"
|
|
|
|
"Accept-Encoding: identity\r\n"
|
|
|
|
"Content-Type: application/x-www-form-urlencoded\r\n" "\r\n");
|
|
|
|
|
2010-06-02 18:07:17 -06:00
|
|
|
ret = xend_req(s, content);
|
2010-11-09 15:48:48 -05:00
|
|
|
VIR_FORCE_CLOSE(s);
|
2006-01-12 15:38:07 +00:00
|
|
|
|
2011-09-01 16:22:38 -06:00
|
|
|
if (ret < 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
if ((ret >= 300) && ((ret != 404) || (!STRPREFIX(path, "/xend/domain/")))) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_GET_FAILED,
|
|
|
|
_("%d status from xen daemon: %s:%s"),
|
|
|
|
ret, path, NULLSTR(*content));
|
2006-02-27 16:27:18 +00:00
|
|
|
}
|
|
|
|
|
2006-01-12 15:38:07 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* xend_post:
|
|
|
|
* @xend: pointer to the Xen Daemon structure
|
|
|
|
* @path: the path used for the HTTP request
|
2006-07-19 22:24:37 +00:00
|
|
|
* @ops: the information sent for the POST
|
2006-01-12 15:38:07 +00:00
|
|
|
*
|
|
|
|
* Do an HTTP POST RPC with the Xen Daemon, this usually makes changes at the
|
|
|
|
* Xen level.
|
|
|
|
*
|
|
|
|
* Returns the HTTP return code or -1 in case or error.
|
|
|
|
*/
|
|
|
|
static int
|
2010-06-02 18:07:17 -06:00
|
|
|
xend_post(virConnectPtr xend, const char *path, const char *ops)
|
2006-01-12 15:38:07 +00:00
|
|
|
{
|
|
|
|
char buffer[100];
|
2010-06-02 18:07:17 -06:00
|
|
|
char *err_buf = NULL;
|
2006-01-12 15:38:07 +00:00
|
|
|
int ret;
|
|
|
|
int s = do_connect(xend);
|
|
|
|
|
|
|
|
if (s == -1)
|
|
|
|
return s;
|
|
|
|
|
2010-04-03 01:10:49 +02:00
|
|
|
swrites(s, "POST ");
|
|
|
|
swrites(s, path);
|
|
|
|
swrites(s, " HTTP/1.1\r\n");
|
2006-01-12 15:38:07 +00:00
|
|
|
|
2010-04-03 01:10:49 +02:00
|
|
|
swrites(s,
|
2006-01-12 15:38:07 +00:00
|
|
|
"Host: localhost:8000\r\n"
|
|
|
|
"Accept-Encoding: identity\r\n"
|
|
|
|
"Content-Type: application/x-www-form-urlencoded\r\n"
|
|
|
|
"Content-Length: ");
|
2006-08-08 22:22:55 +00:00
|
|
|
snprintf(buffer, sizeof(buffer), "%d", (int) strlen(ops));
|
2010-04-03 01:10:49 +02:00
|
|
|
swrites(s, buffer);
|
|
|
|
swrites(s, "\r\n\r\n");
|
|
|
|
swrites(s, ops);
|
2006-01-12 15:38:07 +00:00
|
|
|
|
2010-06-02 18:07:17 -06:00
|
|
|
ret = xend_req(s, &err_buf);
|
2010-11-09 15:48:48 -05:00
|
|
|
VIR_FORCE_CLOSE(s);
|
2006-01-12 15:38:07 +00:00
|
|
|
|
2006-02-27 16:27:18 +00:00
|
|
|
if ((ret < 0) || (ret >= 300)) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_POST_FAILED,
|
|
|
|
_("xend_post: error from xen daemon: %s"), err_buf);
|
2010-06-02 18:07:17 -06:00
|
|
|
} else if ((ret == 202) && err_buf && (strstr(err_buf, "failed") != NULL)) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_POST_FAILED,
|
|
|
|
_("xend_post: error from xen daemon: %s"), err_buf);
|
2007-02-27 15:50:03 +00:00
|
|
|
ret = -1;
|
2010-06-02 18:07:17 -06:00
|
|
|
} else if (((ret >= 200) && (ret <= 202)) && err_buf &&
|
|
|
|
(strstr(err_buf, "xend.err") != NULL)) {
|
2007-03-01 23:24:09 +00:00
|
|
|
/* This is to catch case of things like 'virsh dump Domain-0 foo'
|
|
|
|
* which returns a success code, but the word 'xend.err'
|
|
|
|
* in body to indicate error :-(
|
2007-02-27 15:50:03 +00:00
|
|
|
*/
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_POST_FAILED,
|
|
|
|
_("xend_post: error from xen daemon: %s"), err_buf);
|
2006-04-20 14:28:01 +00:00
|
|
|
ret = -1;
|
2006-02-27 16:27:18 +00:00
|
|
|
}
|
|
|
|
|
2010-06-02 18:07:17 -06:00
|
|
|
VIR_FREE(err_buf);
|
2006-01-12 15:38:07 +00:00
|
|
|
return ret;
|
|
|
|
}
|
2006-08-09 15:21:16 +00:00
|
|
|
|
2006-01-12 15:38:07 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* http2unix:
|
|
|
|
* @ret: the http return code
|
|
|
|
*
|
|
|
|
* Convert the HTTP return code to 0/-1 and set errno if needed
|
|
|
|
*
|
|
|
|
* Return -1 in case of error code 0 otherwise
|
|
|
|
*/
|
|
|
|
static int
|
2010-04-03 01:10:49 +02:00
|
|
|
http2unix(int ret)
|
2006-01-12 15:38:07 +00:00
|
|
|
{
|
|
|
|
switch (ret) {
|
|
|
|
case -1:
|
|
|
|
break;
|
|
|
|
case 200:
|
|
|
|
case 201:
|
|
|
|
case 202:
|
|
|
|
return 0;
|
|
|
|
case 404:
|
|
|
|
errno = ESRCH;
|
|
|
|
break;
|
2006-08-29 18:12:22 +00:00
|
|
|
case 500:
|
|
|
|
errno = EIO;
|
|
|
|
break;
|
2006-01-12 15:38:07 +00:00
|
|
|
default:
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_HTTP_ERROR,
|
|
|
|
_("Unexpected HTTP error code %d"), ret);
|
2006-01-12 15:38:07 +00:00
|
|
|
errno = EINVAL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2008-07-25 09:51:23 +00:00
|
|
|
* xend_op_ext:
|
2006-01-12 15:38:07 +00:00
|
|
|
* @xend: pointer to the Xen Daemon structure
|
|
|
|
* @path: path for the object
|
|
|
|
* @key: the key for the operation
|
|
|
|
* @ap: input values to pass to the operation
|
|
|
|
*
|
|
|
|
* internal routine to run a POST RPC operation to the Xen Daemon
|
|
|
|
*
|
|
|
|
* Returns 0 in case of success, -1 in case of failure.
|
|
|
|
*/
|
|
|
|
static int
|
2010-06-02 18:07:17 -06:00
|
|
|
xend_op_ext(virConnectPtr xend, const char *path, const char *key, va_list ap)
|
2006-01-12 15:38:07 +00:00
|
|
|
{
|
|
|
|
const char *k = key, *v;
|
2008-04-28 15:14:59 +00:00
|
|
|
virBuffer buf = VIR_BUFFER_INITIALIZER;
|
2007-04-24 13:44:16 +00:00
|
|
|
int ret;
|
2008-04-28 15:14:59 +00:00
|
|
|
char *content;
|
2006-01-12 15:38:07 +00:00
|
|
|
|
|
|
|
while (k) {
|
|
|
|
v = va_arg(ap, const char *);
|
|
|
|
|
2011-09-08 16:34:10 +02:00
|
|
|
virBufferURIEncodeString(&buf, k);
|
|
|
|
virBufferAddChar(&buf, '=');
|
|
|
|
virBufferURIEncodeString(&buf, v);
|
2006-01-12 15:38:07 +00:00
|
|
|
k = va_arg(ap, const char *);
|
|
|
|
|
|
|
|
if (k)
|
2011-04-14 13:27:47 -06:00
|
|
|
virBufferAddChar(&buf, '&');
|
2006-01-12 15:38:07 +00:00
|
|
|
}
|
|
|
|
|
2014-06-27 10:40:15 +02:00
|
|
|
if (virBufferCheckError(&buf) < 0)
|
2008-04-28 15:14:59 +00:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
content = virBufferContentAndReset(&buf);
|
2015-10-27 19:14:01 +01:00
|
|
|
VIR_DEBUG("xend op: %s", content);
|
2010-06-02 18:07:17 -06:00
|
|
|
ret = http2unix(xend_post(xend, path, content));
|
2008-05-29 19:20:22 +00:00
|
|
|
VIR_FREE(content);
|
2007-04-24 13:44:16 +00:00
|
|
|
|
|
|
|
return ret;
|
2006-01-12 15:38:07 +00:00
|
|
|
}
|
|
|
|
|
2006-08-09 15:21:16 +00:00
|
|
|
|
2006-01-12 15:38:07 +00:00
|
|
|
/**
|
2008-07-25 09:51:23 +00:00
|
|
|
* xend_op:
|
2006-01-12 15:38:07 +00:00
|
|
|
* @xend: pointer to the Xen Daemon structure
|
|
|
|
* @name: the domain name target of this operation
|
|
|
|
* @key: the key for the operation
|
|
|
|
* @ap: input values to pass to the operation
|
|
|
|
* @...: input values to pass to the operation
|
|
|
|
*
|
2015-03-19 16:53:00 +01:00
|
|
|
* internal routine to run a POST RPC operation to the Xen Daemon targeting
|
2006-01-12 15:38:07 +00:00
|
|
|
* a given domain.
|
|
|
|
*
|
|
|
|
* Returns 0 in case of success, -1 in case of failure.
|
|
|
|
*/
|
2009-11-30 10:49:05 +01:00
|
|
|
static int ATTRIBUTE_SENTINEL
|
2007-09-29 18:16:26 +00:00
|
|
|
xend_op(virConnectPtr xend, const char *name, const char *key, ...)
|
2006-01-12 15:38:07 +00:00
|
|
|
{
|
|
|
|
char buffer[1024];
|
|
|
|
va_list ap;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
snprintf(buffer, sizeof(buffer), "/xend/domain/%s", name);
|
|
|
|
|
|
|
|
va_start(ap, key);
|
2010-06-02 18:07:17 -06:00
|
|
|
ret = xend_op_ext(xend, buffer, key, ap);
|
2006-01-12 15:38:07 +00:00
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* sexpr_get:
|
|
|
|
* @xend: pointer to the Xen Daemon structure
|
|
|
|
* @fmt: format string for the path of the operation
|
|
|
|
* @...: extra data to build the path of the operation
|
|
|
|
*
|
|
|
|
* Internal routine to run a simple GET RPC operation to the Xen Daemon
|
|
|
|
*
|
|
|
|
* Returns a parsed S-Expression in case of success, NULL in case of failure
|
|
|
|
*/
|
2007-10-17 10:33:16 +00:00
|
|
|
static struct sexpr *sexpr_get(virConnectPtr xend, const char *fmt, ...)
|
2013-11-19 15:32:34 -07:00
|
|
|
ATTRIBUTE_FMT_PRINTF(2, 3);
|
2007-10-17 10:33:16 +00:00
|
|
|
|
2006-01-12 15:38:07 +00:00
|
|
|
static struct sexpr *
|
2006-01-13 16:41:01 +00:00
|
|
|
sexpr_get(virConnectPtr xend, const char *fmt, ...)
|
2006-01-12 15:38:07 +00:00
|
|
|
{
|
2010-06-02 18:07:17 -06:00
|
|
|
char *buffer = NULL;
|
2006-01-12 15:38:07 +00:00
|
|
|
char path[1024];
|
|
|
|
va_list ap;
|
|
|
|
int ret;
|
2010-06-02 18:07:17 -06:00
|
|
|
struct sexpr *res = NULL;
|
2006-01-12 15:38:07 +00:00
|
|
|
|
|
|
|
va_start(ap, fmt);
|
|
|
|
vsnprintf(path, sizeof(path), fmt, ap);
|
|
|
|
va_end(ap);
|
|
|
|
|
2010-06-02 18:07:17 -06:00
|
|
|
ret = xend_get(xend, path, &buffer);
|
2010-04-03 01:10:49 +02:00
|
|
|
ret = http2unix(ret);
|
2006-01-12 15:38:07 +00:00
|
|
|
if (ret == -1)
|
2010-06-02 18:07:17 -06:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (buffer == NULL)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
res = string2sexpr(buffer);
|
2006-01-12 15:38:07 +00:00
|
|
|
|
2014-03-25 07:54:51 +01:00
|
|
|
cleanup:
|
2010-06-02 18:07:17 -06:00
|
|
|
VIR_FREE(buffer);
|
|
|
|
return res;
|
2006-01-12 15:38:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* sexpr_uuid:
|
|
|
|
* @ptr: where to store the UUID, incremented
|
|
|
|
* @sexpr: an S-Expression
|
|
|
|
* @name: the name for the value
|
|
|
|
*
|
2014-04-21 01:37:46 +05:30
|
|
|
* convenience function to lookup a UUID value from the S-Expression
|
2006-01-12 15:38:07 +00:00
|
|
|
*
|
2007-08-09 20:19:12 +00:00
|
|
|
* Returns a -1 on error, 0 on success
|
2006-01-12 15:38:07 +00:00
|
|
|
*/
|
2007-08-09 20:19:12 +00:00
|
|
|
static int
|
Adjust sexpr-related interfaces to be const-correct.
* src/sexpr.c (sexpr_cons, append, sexpr_append, sexpr2string)
(sexpr_lookup_key, sexpr_lookup, sexpr_node, sexpr_fmt_node):
Add "const" attribute where appropriate.
* src/xend_internal.c (sexpr_int, sexpr_float, sexpr_u64)
(sexpr_uuid, sexpr_to_xend_domain_info, sexpr_to_xend_node_info)
(sexpr_to_xend_topology_xml, sexpr_to_domain): Likewise.
* src/sexpr.h: Adjust prototypes.
2008-01-21 14:22:15 +00:00
|
|
|
sexpr_uuid(unsigned char *ptr, const struct sexpr *node, const char *path)
|
2006-01-12 15:38:07 +00:00
|
|
|
{
|
|
|
|
const char *r = sexpr_node(node, path);
|
2007-08-09 20:19:12 +00:00
|
|
|
if (!r)
|
|
|
|
return -1;
|
|
|
|
return virUUIDParse(r, ptr);
|
2006-01-12 15:38:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* PUBLIC FUNCTIONS */
|
|
|
|
|
|
|
|
/**
|
2006-03-22 13:44:01 +00:00
|
|
|
* xenDaemonOpen_unix:
|
2006-01-13 16:41:01 +00:00
|
|
|
* @conn: an existing virtual connection block
|
2006-01-12 15:38:07 +00:00
|
|
|
* @path: the path for the Xen Daemon socket
|
|
|
|
*
|
|
|
|
* Creates a localhost Xen Daemon connection
|
|
|
|
* Note: this doesn't try to check if the connection actually works
|
|
|
|
*
|
2006-01-13 16:41:01 +00:00
|
|
|
* Returns 0 in case of success, -1 in case of error.
|
2006-01-12 15:38:07 +00:00
|
|
|
*/
|
2006-01-13 16:41:01 +00:00
|
|
|
int
|
2006-06-29 14:44:37 +00:00
|
|
|
xenDaemonOpen_unix(virConnectPtr conn, const char *path)
|
2006-01-12 15:38:07 +00:00
|
|
|
{
|
|
|
|
struct sockaddr_un *addr;
|
2013-01-29 09:53:13 -05:00
|
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
2006-01-12 15:38:07 +00:00
|
|
|
|
2009-01-22 19:41:48 +00:00
|
|
|
memset(&priv->addr, 0, sizeof(priv->addr));
|
|
|
|
priv->addrfamily = AF_UNIX;
|
2009-01-28 14:36:23 +00:00
|
|
|
/*
|
|
|
|
* This must be zero on Solaris at least for AF_UNIX (which should
|
|
|
|
* really be PF_UNIX, but doesn't matter).
|
|
|
|
*/
|
|
|
|
priv->addrprotocol = 0;
|
2009-01-22 19:41:48 +00:00
|
|
|
priv->addrlen = sizeof(struct sockaddr_un);
|
|
|
|
|
|
|
|
addr = (struct sockaddr_un *)&priv->addr;
|
2006-01-12 15:38:07 +00:00
|
|
|
addr->sun_family = AF_UNIX;
|
|
|
|
memset(addr->sun_path, 0, sizeof(addr->sun_path));
|
2009-08-03 14:37:44 +02:00
|
|
|
if (virStrcpyStatic(addr->sun_path, path) == NULL)
|
|
|
|
return -1;
|
2006-01-12 15:38:07 +00:00
|
|
|
|
2012-03-22 12:33:35 +01:00
|
|
|
return 0;
|
2006-01-12 15:38:07 +00:00
|
|
|
}
|
|
|
|
|
2010-11-08 17:32:02 +01:00
|
|
|
|
2006-01-12 15:38:07 +00:00
|
|
|
/**
|
2006-03-22 13:44:01 +00:00
|
|
|
* xenDaemonOpen_tcp:
|
2006-01-13 16:41:01 +00:00
|
|
|
* @conn: an existing virtual connection block
|
2006-01-12 15:38:07 +00:00
|
|
|
* @host: the host name for the Xen Daemon
|
2008-02-05 19:27:37 +00:00
|
|
|
* @port: the port
|
2006-01-12 15:38:07 +00:00
|
|
|
*
|
|
|
|
* Creates a possibly remote Xen Daemon connection
|
|
|
|
* Note: this doesn't try to check if the connection actually works
|
|
|
|
*
|
2006-01-13 16:41:01 +00:00
|
|
|
* Returns 0 in case of success, -1 in case of error.
|
2006-01-12 15:38:07 +00:00
|
|
|
*/
|
2008-07-25 09:51:23 +00:00
|
|
|
static int
|
2009-01-22 19:41:48 +00:00
|
|
|
xenDaemonOpen_tcp(virConnectPtr conn, const char *host, const char *port)
|
2006-01-12 15:38:07 +00:00
|
|
|
{
|
2013-01-29 09:53:13 -05:00
|
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
2009-01-22 19:41:48 +00:00
|
|
|
struct addrinfo *res, *r;
|
|
|
|
struct addrinfo hints;
|
|
|
|
int saved_errno = EINVAL;
|
|
|
|
int ret;
|
2006-01-12 15:38:07 +00:00
|
|
|
|
2009-01-22 19:41:48 +00:00
|
|
|
priv->addrlen = 0;
|
|
|
|
memset(&priv->addr, 0, sizeof(priv->addr));
|
|
|
|
|
2011-01-28 22:38:06 +01:00
|
|
|
/* http://people.redhat.com/drepper/userapi-ipv6.html */
|
2012-03-29 10:52:04 +01:00
|
|
|
memset (&hints, 0, sizeof(hints));
|
2009-01-22 19:41:48 +00:00
|
|
|
hints.ai_socktype = SOCK_STREAM;
|
|
|
|
hints.ai_flags = AI_ADDRCONFIG;
|
|
|
|
|
|
|
|
ret = getaddrinfo (host, port, &hints, &res);
|
|
|
|
if (ret != 0) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_UNKNOWN_HOST,
|
|
|
|
_("unable to resolve hostname '%s': %s"),
|
|
|
|
host, gai_strerror (ret));
|
2009-01-22 19:41:48 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Try to connect to each returned address in turn. */
|
|
|
|
for (r = res; r; r = r->ai_next) {
|
|
|
|
int sock;
|
|
|
|
|
2012-10-17 10:23:12 +01:00
|
|
|
sock = socket(r->ai_family, SOCK_STREAM, r->ai_protocol);
|
2009-01-22 19:41:48 +00:00
|
|
|
if (sock == -1) {
|
|
|
|
saved_errno = errno;
|
|
|
|
continue;
|
2006-01-12 15:38:07 +00:00
|
|
|
}
|
2009-01-22 19:41:48 +00:00
|
|
|
|
2012-10-17 10:23:12 +01:00
|
|
|
if (connect(sock, r->ai_addr, r->ai_addrlen) == -1) {
|
2009-01-22 19:41:48 +00:00
|
|
|
saved_errno = errno;
|
2010-11-09 15:48:48 -05:00
|
|
|
VIR_FORCE_CLOSE(sock);
|
2009-01-22 19:41:48 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
priv->addrlen = r->ai_addrlen;
|
|
|
|
priv->addrfamily = r->ai_family;
|
|
|
|
priv->addrprotocol = r->ai_protocol;
|
|
|
|
memcpy(&priv->addr,
|
|
|
|
r->ai_addr,
|
|
|
|
r->ai_addrlen);
|
2010-11-09 15:48:48 -05:00
|
|
|
VIR_FORCE_CLOSE(sock);
|
2009-01-22 19:41:48 +00:00
|
|
|
break;
|
2006-01-12 15:38:07 +00:00
|
|
|
}
|
|
|
|
|
2012-10-17 10:23:12 +01:00
|
|
|
freeaddrinfo(res);
|
2006-01-12 15:38:07 +00:00
|
|
|
|
2009-01-22 19:41:48 +00:00
|
|
|
if (!priv->addrlen) {
|
2009-01-29 23:01:37 +00:00
|
|
|
/* Don't raise error when unprivileged, since proxy takes over */
|
|
|
|
if (xenHavePrivilege())
|
2010-02-04 21:02:58 +01:00
|
|
|
virReportSystemError(saved_errno,
|
2009-01-29 23:01:37 +00:00
|
|
|
_("unable to connect to '%s:%s'"),
|
|
|
|
host, port);
|
2009-01-22 19:41:48 +00:00
|
|
|
return -1;
|
|
|
|
}
|
2006-01-12 15:38:07 +00:00
|
|
|
|
2009-01-22 19:41:48 +00:00
|
|
|
return 0;
|
2006-01-12 15:38:07 +00:00
|
|
|
}
|
|
|
|
|
2006-08-09 15:21:16 +00:00
|
|
|
|
2006-01-12 15:38:07 +00:00
|
|
|
/**
|
|
|
|
* xend_wait_for_devices:
|
2010-09-23 17:22:03 +02:00
|
|
|
* @xend: pointer to the Xen Daemon block
|
2006-01-12 15:38:07 +00:00
|
|
|
* @name: name for the domain
|
|
|
|
*
|
|
|
|
* Block the domain until all the virtual devices are ready. This operation
|
|
|
|
* is needed when creating a domain before resuming it.
|
|
|
|
*
|
|
|
|
* Returns 0 in case of success, -1 (with errno) in case of error.
|
|
|
|
*/
|
|
|
|
int
|
2006-01-13 16:41:01 +00:00
|
|
|
xend_wait_for_devices(virConnectPtr xend, const char *name)
|
2006-01-12 15:38:07 +00:00
|
|
|
{
|
|
|
|
return xend_op(xend, name, "op", "wait_for_devices", NULL);
|
|
|
|
}
|
|
|
|
|
2006-08-09 15:21:16 +00:00
|
|
|
|
2006-01-12 15:38:07 +00:00
|
|
|
/**
|
2006-06-14 15:44:14 +00:00
|
|
|
* xenDaemonListDomainsOld:
|
2010-09-23 17:22:03 +02:00
|
|
|
* @xend: pointer to the Xen Daemon block
|
2006-01-12 15:38:07 +00:00
|
|
|
*
|
|
|
|
* This method will return an array of names of currently running
|
|
|
|
* domains. The memory should be released will a call to free().
|
|
|
|
*
|
|
|
|
* Returns a list of names or NULL in case of error.
|
|
|
|
*/
|
2006-06-29 23:53:31 +00:00
|
|
|
char **
|
2006-06-14 15:44:14 +00:00
|
|
|
xenDaemonListDomainsOld(virConnectPtr xend)
|
2006-01-12 15:38:07 +00:00
|
|
|
{
|
|
|
|
struct sexpr *root = NULL;
|
|
|
|
char **ret = NULL;
|
|
|
|
int count = 0;
|
Convert 'int i' to 'size_t i' in src/{xen,xenapi,xenxs} 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;
|
2006-01-12 15:38:07 +00:00
|
|
|
struct sexpr *_for_i, *node;
|
|
|
|
|
|
|
|
root = sexpr_get(xend, "/xend/domain");
|
|
|
|
if (root == NULL)
|
|
|
|
goto error;
|
|
|
|
|
2007-09-29 18:16:26 +00:00
|
|
|
for (_for_i = root, node = root->u.s.car; _for_i->kind == SEXPR_CONS;
|
|
|
|
_for_i = _for_i->u.s.cdr, node = _for_i->u.s.car) {
|
2006-01-12 15:38:07 +00:00
|
|
|
if (node->kind != SEXPR_VALUE)
|
|
|
|
continue;
|
|
|
|
count++;
|
|
|
|
}
|
|
|
|
|
2013-07-04 12:18:49 +02:00
|
|
|
if (VIR_ALLOC_N(ret, count + 1) < 0)
|
2006-01-12 15:38:07 +00:00
|
|
|
goto error;
|
|
|
|
|
|
|
|
i = 0;
|
2007-09-29 18:16:26 +00:00
|
|
|
for (_for_i = root, node = root->u.s.car; _for_i->kind == SEXPR_CONS;
|
|
|
|
_for_i = _for_i->u.s.cdr, node = _for_i->u.s.car) {
|
2006-01-12 15:38:07 +00:00
|
|
|
if (node->kind != SEXPR_VALUE)
|
|
|
|
continue;
|
2013-05-03 14:51:25 +02:00
|
|
|
if (VIR_STRDUP(ret[i], node->u.value) < 0)
|
2011-09-20 12:11:31 -06:00
|
|
|
goto no_memory;
|
2006-01-12 15:38:07 +00:00
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret[i] = NULL;
|
|
|
|
|
2014-03-25 07:54:51 +01:00
|
|
|
error:
|
2006-01-12 15:38:07 +00:00
|
|
|
sexpr_free(root);
|
|
|
|
return ret;
|
2011-09-20 12:11:31 -06:00
|
|
|
|
2014-03-25 07:54:51 +01:00
|
|
|
no_memory:
|
2011-09-20 12:11:31 -06:00
|
|
|
for (i = 0; i < count; i++)
|
|
|
|
VIR_FREE(ret[i]);
|
|
|
|
VIR_FREE(ret);
|
|
|
|
goto error;
|
2006-01-12 15:38:07 +00:00
|
|
|
}
|
|
|
|
|
2010-11-08 17:32:02 +01:00
|
|
|
|
2006-01-12 15:38:07 +00:00
|
|
|
/**
|
2008-10-10 09:32:27 +00:00
|
|
|
* xenDaemonDomainCreateXML:
|
2006-02-16 22:50:52 +00:00
|
|
|
* @xend: A xend instance
|
|
|
|
* @sexpr: An S-Expr description of the domain.
|
|
|
|
*
|
2010-10-27 15:25:39 +02:00
|
|
|
* This method will create a domain based on the passed in description. The
|
2006-02-16 22:50:52 +00:00
|
|
|
* domain will be paused after creation and must be unpaused with
|
2006-03-22 13:44:01 +00:00
|
|
|
* xenDaemonResumeDomain() to begin execution.
|
2006-02-16 22:50:52 +00:00
|
|
|
* This method may be deprecated once switching to XML-RPC based communcations
|
|
|
|
* with xend.
|
|
|
|
*
|
|
|
|
* Returns 0 for success, -1 (with errno) on error
|
|
|
|
*/
|
|
|
|
|
|
|
|
int
|
2008-10-10 09:32:27 +00:00
|
|
|
xenDaemonDomainCreateXML(virConnectPtr xend, const char *sexpr)
|
2006-02-16 22:50:52 +00:00
|
|
|
{
|
2011-09-08 16:34:10 +02:00
|
|
|
int ret;
|
2006-02-16 22:50:52 +00:00
|
|
|
|
2011-09-08 16:34:10 +02:00
|
|
|
ret = xend_op(xend, "", "op", "create", "config", sexpr, NULL);
|
2006-02-16 22:50:52 +00:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
2010-11-08 17:32:02 +01:00
|
|
|
|
2006-02-16 22:50:52 +00:00
|
|
|
|
2006-01-19 10:23:15 +00:00
|
|
|
/**
|
2006-03-22 13:44:01 +00:00
|
|
|
* xenDaemonDomainLookupByName_ids:
|
2006-01-19 10:23:15 +00:00
|
|
|
* @xend: A xend instance
|
2006-02-23 10:13:55 +00:00
|
|
|
* @domname: The name of the domain
|
|
|
|
* @uuid: return value for the UUID if not NULL
|
2006-01-19 10:23:15 +00:00
|
|
|
*
|
|
|
|
* This method looks up the id of a domain
|
|
|
|
*
|
|
|
|
* Returns the id on success; -1 (with errno) on error
|
|
|
|
*/
|
|
|
|
int
|
2013-01-29 09:53:13 -05:00
|
|
|
xenDaemonDomainLookupByName_ids(virConnectPtr xend,
|
|
|
|
const char *domname,
|
2008-04-10 16:54:54 +00:00
|
|
|
unsigned char *uuid)
|
2006-01-19 10:23:15 +00:00
|
|
|
{
|
|
|
|
struct sexpr *root;
|
|
|
|
const char *value;
|
|
|
|
int ret = -1;
|
|
|
|
|
2006-03-15 12:13:25 +00:00
|
|
|
if (uuid != NULL)
|
Mon Jan 23 14:36:18 IST 2007 Mark McLoughlin <markmc@redhat.com>
* include/libvirt/libvirt.h.in: add VIR_UUID_BUFLEN and
VIR_UUID_STRING_BUFLEN
* libvirt/proxy/libvirt_proxy.c, libvirt/src/hash.c,
libvirt/src/internal.h, libvirt/src/libvirt.c,
libvirt/src/proxy_internal.c, libvirt/src/test.c,
libvirt/src/virsh.c, libvirt/src/xend_internal.c,
libvirt/src/xm_internal.c, libvirt/src/xml.c,
libvirt/python/libvir.c: use them
2007-01-23 14:39:45 +00:00
|
|
|
memset(uuid, 0, VIR_UUID_BUFLEN);
|
2006-01-19 10:23:15 +00:00
|
|
|
root = sexpr_get(xend, "/xend/domain/%s?detail=1", domname);
|
|
|
|
if (root == NULL)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
value = sexpr_node(root, "domain/domid");
|
2006-02-27 19:56:23 +00:00
|
|
|
if (value == NULL) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
"%s", _("domain information incomplete, missing domid"));
|
2006-01-19 10:23:15 +00:00
|
|
|
goto error;
|
2014-07-18 09:46:39 +02:00
|
|
|
} else if (virStrToLong_i(value, NULL, 0, &ret) < 0) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
"%s", _("domain information incorrect domid not numeric"));
|
2006-01-19 10:23:15 +00:00
|
|
|
ret = -1;
|
2006-02-23 10:13:55 +00:00
|
|
|
} else if (uuid != NULL) {
|
2007-08-09 20:19:12 +00:00
|
|
|
if (sexpr_uuid(uuid, root, "domain/uuid") < 0) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
"%s", _("domain information incomplete, missing uuid"));
|
2006-03-15 12:13:25 +00:00
|
|
|
}
|
2006-02-23 10:13:55 +00:00
|
|
|
}
|
2006-01-19 10:23:15 +00:00
|
|
|
|
2014-03-25 07:54:51 +01:00
|
|
|
error:
|
2006-01-19 10:23:15 +00:00
|
|
|
sexpr_free(root);
|
2012-03-22 12:33:35 +01:00
|
|
|
return ret;
|
2006-01-19 10:23:15 +00:00
|
|
|
}
|
|
|
|
|
2006-07-07 18:58:35 +00:00
|
|
|
|
2006-12-14 01:56:14 +00:00
|
|
|
static int
|
2013-01-29 09:53:13 -05:00
|
|
|
xend_detect_config_version(virConnectPtr conn)
|
|
|
|
{
|
2006-09-12 01:16:22 +00:00
|
|
|
struct sexpr *root;
|
|
|
|
const char *value;
|
2013-01-29 09:53:13 -05:00
|
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
2014-07-18 09:46:39 +02:00
|
|
|
int ret = -1;
|
2007-04-04 14:19:49 +00:00
|
|
|
|
2006-09-12 01:16:22 +00:00
|
|
|
root = sexpr_get(conn, "/xend/node/");
|
|
|
|
if (root == NULL)
|
2014-07-18 09:46:39 +02:00
|
|
|
return ret;
|
2008-02-05 19:27:37 +00:00
|
|
|
|
2006-09-12 01:16:22 +00:00
|
|
|
value = sexpr_node(root, "node/xend_config_format");
|
2008-02-05 19:27:37 +00:00
|
|
|
|
2006-09-12 01:16:22 +00:00
|
|
|
if (value) {
|
2014-07-18 09:46:39 +02:00
|
|
|
if (virStrToLong_i(value, NULL, 10, &priv->xendConfigVersion) < 0)
|
|
|
|
goto cleanup;
|
2014-09-03 13:04:59 -06:00
|
|
|
} else {
|
2006-12-14 01:56:14 +00:00
|
|
|
/* Xen prior to 3.0.3 did not have the xend_config_format
|
|
|
|
field, and is implicitly version 1. */
|
2012-01-31 18:08:09 +01:00
|
|
|
priv->xendConfigVersion = XEND_CONFIG_VERSION_3_0_2;
|
2006-12-14 01:56:14 +00:00
|
|
|
}
|
2014-07-18 09:46:39 +02:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
2006-09-12 01:16:22 +00:00
|
|
|
sexpr_free(root);
|
2014-07-18 09:46:39 +02:00
|
|
|
return ret;
|
2006-09-12 01:16:22 +00:00
|
|
|
}
|
|
|
|
|
2006-02-20 17:22:16 +00:00
|
|
|
|
2011-05-02 11:35:29 +02:00
|
|
|
/**
|
|
|
|
* sexpr_to_xend_domain_state:
|
|
|
|
* @root: an S-Expression describing a domain
|
|
|
|
*
|
|
|
|
* Internal routine getting the domain's state from the domain root provided.
|
|
|
|
*
|
|
|
|
* Returns domain's state.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
ATTRIBUTE_NONNULL(1) ATTRIBUTE_NONNULL(2)
|
2013-05-01 11:29:08 +01:00
|
|
|
sexpr_to_xend_domain_state(virDomainDefPtr def, const struct sexpr *root)
|
2011-05-02 11:35:29 +02:00
|
|
|
{
|
|
|
|
const char *flags;
|
|
|
|
int state = VIR_DOMAIN_NOSTATE;
|
|
|
|
|
|
|
|
if ((flags = sexpr_node(root, "domain/state"))) {
|
|
|
|
if (strchr(flags, 'c'))
|
|
|
|
state = VIR_DOMAIN_CRASHED;
|
|
|
|
else if (strchr(flags, 's'))
|
|
|
|
state = VIR_DOMAIN_SHUTOFF;
|
|
|
|
else if (strchr(flags, 'd'))
|
|
|
|
state = VIR_DOMAIN_SHUTDOWN;
|
|
|
|
else if (strchr(flags, 'p'))
|
|
|
|
state = VIR_DOMAIN_PAUSED;
|
|
|
|
else if (strchr(flags, 'b'))
|
|
|
|
state = VIR_DOMAIN_BLOCKED;
|
|
|
|
else if (strchr(flags, 'r'))
|
|
|
|
state = VIR_DOMAIN_RUNNING;
|
2013-05-01 11:29:08 +01:00
|
|
|
} else if (def->id < 0 || sexpr_int(root, "domain/status") == 0) {
|
2012-04-12 15:42:37 +02:00
|
|
|
/* As far as I can see the domain->id is a bad sign for checking
|
|
|
|
* inactive domains as this is inaccurate after the domain has
|
|
|
|
* been running once. However domain/status from xend seems to
|
|
|
|
* be always present and 0 for inactive domains.
|
|
|
|
* (keeping the check for id < 0 to be extra safe about backward
|
|
|
|
* compatibility)
|
|
|
|
*/
|
2011-05-02 11:35:29 +02:00
|
|
|
state = VIR_DOMAIN_SHUTOFF;
|
|
|
|
}
|
|
|
|
|
|
|
|
return state;
|
|
|
|
}
|
|
|
|
|
2006-02-20 17:22:16 +00:00
|
|
|
/**
|
2006-03-22 13:44:01 +00:00
|
|
|
* sexpr_to_xend_domain_info:
|
|
|
|
* @root: an S-Expression describing a domain
|
|
|
|
* @info: a info data structure to fill=up
|
|
|
|
*
|
|
|
|
* Internal routine filling up the info structure with the values from
|
|
|
|
* the domain root provided.
|
|
|
|
*
|
|
|
|
* Returns 0 in case of success, -1 in case of error
|
|
|
|
*/
|
|
|
|
static int
|
2013-05-01 11:29:08 +01:00
|
|
|
sexpr_to_xend_domain_info(virDomainDefPtr def,
|
2013-01-29 09:53:13 -05:00
|
|
|
const struct sexpr *root,
|
Adjust sexpr-related interfaces to be const-correct.
* src/sexpr.c (sexpr_cons, append, sexpr_append, sexpr2string)
(sexpr_lookup_key, sexpr_lookup, sexpr_node, sexpr_fmt_node):
Add "const" attribute where appropriate.
* src/xend_internal.c (sexpr_int, sexpr_float, sexpr_u64)
(sexpr_uuid, sexpr_to_xend_domain_info, sexpr_to_xend_node_info)
(sexpr_to_xend_topology_xml, sexpr_to_domain): Likewise.
* src/sexpr.h: Adjust prototypes.
2008-01-21 14:22:15 +00:00
|
|
|
virDomainInfoPtr info)
|
2006-03-22 13:44:01 +00:00
|
|
|
{
|
2010-10-05 08:18:52 -06:00
|
|
|
int vcpus;
|
2006-03-22 13:44:01 +00:00
|
|
|
|
2013-05-01 11:29:08 +01:00
|
|
|
info->state = sexpr_to_xend_domain_state(def, root);
|
2006-03-22 13:44:01 +00:00
|
|
|
info->memory = sexpr_u64(root, "domain/memory") << 10;
|
|
|
|
info->maxMem = sexpr_u64(root, "domain/maxmem") << 10;
|
|
|
|
info->cpuTime = sexpr_float(root, "domain/cpu_time") * 1000000000;
|
2011-05-02 11:35:29 +02:00
|
|
|
|
2010-10-05 08:18:52 -06:00
|
|
|
vcpus = sexpr_int(root, "domain/vcpus");
|
2010-10-29 10:51:01 -06:00
|
|
|
info->nrVirtCpu = count_one_bits_l(sexpr_u64(root, "domain/vcpu_avail"));
|
2010-10-05 08:18:52 -06:00
|
|
|
if (!info->nrVirtCpu || vcpus < info->nrVirtCpu)
|
|
|
|
info->nrVirtCpu = vcpus;
|
|
|
|
|
2012-03-22 12:33:35 +01:00
|
|
|
return 0;
|
2006-03-22 13:44:01 +00:00
|
|
|
}
|
|
|
|
|
2006-03-29 12:46:03 +00:00
|
|
|
/**
|
|
|
|
* sexpr_to_xend_node_info:
|
|
|
|
* @root: an S-Expression describing a domain
|
|
|
|
* @info: a info data structure to fill up
|
|
|
|
*
|
|
|
|
* Internal routine filling up the info structure with the values from
|
|
|
|
* the node root provided.
|
|
|
|
*
|
|
|
|
* Returns 0 in case of success, -1 in case of error
|
|
|
|
*/
|
|
|
|
static int
|
Adjust sexpr-related interfaces to be const-correct.
* src/sexpr.c (sexpr_cons, append, sexpr_append, sexpr2string)
(sexpr_lookup_key, sexpr_lookup, sexpr_node, sexpr_fmt_node):
Add "const" attribute where appropriate.
* src/xend_internal.c (sexpr_int, sexpr_float, sexpr_u64)
(sexpr_uuid, sexpr_to_xend_domain_info, sexpr_to_xend_node_info)
(sexpr_to_xend_topology_xml, sexpr_to_domain): Likewise.
* src/sexpr.h: Adjust prototypes.
2008-01-21 14:22:15 +00:00
|
|
|
sexpr_to_xend_node_info(const struct sexpr *root, virNodeInfoPtr info)
|
2006-03-29 12:46:03 +00:00
|
|
|
{
|
|
|
|
const char *machine;
|
|
|
|
|
|
|
|
machine = sexpr_node(root, "node/machine");
|
2007-12-17 23:04:33 +00:00
|
|
|
if (machine == NULL) {
|
2006-03-29 12:46:03 +00:00
|
|
|
info->model[0] = 0;
|
2007-12-17 23:04:33 +00:00
|
|
|
} else {
|
2006-03-29 12:46:03 +00:00
|
|
|
snprintf(&info->model[0], sizeof(info->model) - 1, "%s", machine);
|
2007-12-17 23:04:33 +00:00
|
|
|
info->model[sizeof(info->model) - 1] = 0;
|
2006-03-29 12:46:03 +00:00
|
|
|
}
|
|
|
|
info->memory = (unsigned long) sexpr_u64(root, "node/total_memory") << 10;
|
|
|
|
|
|
|
|
info->cpus = sexpr_int(root, "node/nr_cpus");
|
|
|
|
info->mhz = sexpr_int(root, "node/cpu_mhz");
|
|
|
|
info->nodes = sexpr_int(root, "node/nr_nodes");
|
|
|
|
info->sockets = sexpr_int(root, "node/sockets_per_node");
|
2008-01-20 15:56:49 +00:00
|
|
|
info->cores = sexpr_int(root, "node/cores_per_socket");
|
|
|
|
info->threads = sexpr_int(root, "node/threads_per_core");
|
|
|
|
|
2007-12-17 23:04:33 +00:00
|
|
|
/* Xen 3.2.0 replaces sockets_per_node with 'nr_cpus'.
|
|
|
|
* Old Xen calculated sockets_per_node using its internal
|
|
|
|
* nr_cpus / (nodes*cores*threads), so fake it ourselves
|
|
|
|
* in the same way
|
|
|
|
*/
|
|
|
|
if (info->sockets == 0) {
|
|
|
|
int nr_cpus = sexpr_int(root, "node/nr_cpus");
|
2008-01-20 15:56:49 +00:00
|
|
|
int procs = info->nodes * info->cores * info->threads;
|
|
|
|
if (procs == 0) /* Sanity check in case of Xen bugs in futures..*/
|
2012-03-22 12:33:35 +01:00
|
|
|
return -1;
|
2008-01-20 15:56:49 +00:00
|
|
|
info->sockets = nr_cpus / procs;
|
2007-12-17 23:04:33 +00:00
|
|
|
}
|
2010-11-24 11:25:19 +01:00
|
|
|
|
|
|
|
/* On systems where NUMA nodes are not composed of whole sockets either Xen
|
|
|
|
* provided us wrong number of sockets per node or we computed the wrong
|
|
|
|
* number in the compatibility code above. In such case, we compute the
|
|
|
|
* correct number of sockets on the host, lie about the number of NUMA
|
|
|
|
* nodes, and force apps to check capabilities XML for the actual NUMA
|
|
|
|
* topology.
|
|
|
|
*/
|
|
|
|
if (info->nodes * info->sockets * info->cores * info->threads
|
|
|
|
!= info->cpus) {
|
|
|
|
info->nodes = 1;
|
|
|
|
info->sockets = info->cpus / (info->cores * info->threads);
|
|
|
|
}
|
|
|
|
|
2012-03-22 12:33:35 +01:00
|
|
|
return 0;
|
2006-03-29 12:46:03 +00:00
|
|
|
}
|
|
|
|
|
2008-02-27 04:35:08 +00:00
|
|
|
|
2007-09-28 14:28:12 +00:00
|
|
|
/**
|
2008-02-27 04:35:08 +00:00
|
|
|
* sexpr_to_xend_topology
|
2007-09-28 14:28:12 +00:00
|
|
|
* @root: an S-Expression describing a node
|
2008-02-27 04:35:08 +00:00
|
|
|
* @caps: capability info
|
2007-09-28 14:28:12 +00:00
|
|
|
*
|
2008-02-27 04:35:08 +00:00
|
|
|
* Internal routine populating capability info with
|
|
|
|
* NUMA node mapping details
|
2007-09-28 14:28:12 +00:00
|
|
|
*
|
2009-01-19 20:16:10 +00:00
|
|
|
* Does nothing when the system doesn't support NUMA (not an error).
|
|
|
|
*
|
2007-09-28 14:28:12 +00:00
|
|
|
* Returns 0 in case of success, -1 in case of error
|
|
|
|
*/
|
2008-02-05 19:27:37 +00:00
|
|
|
static int
|
2013-01-29 09:53:13 -05:00
|
|
|
sexpr_to_xend_topology(const struct sexpr *root, virCapsPtr caps)
|
2007-09-28 14:28:12 +00:00
|
|
|
{
|
|
|
|
const char *nodeToCpu;
|
2008-02-27 04:35:08 +00:00
|
|
|
const char *cur;
|
2013-01-22 18:42:08 +01:00
|
|
|
virCapsHostNUMACellCPUPtr cpuInfo = NULL;
|
2013-01-24 14:02:49 +01:00
|
|
|
int cell, cpu, nb_cpus = 0;
|
2008-02-27 04:35:08 +00:00
|
|
|
int n = 0;
|
2013-01-24 14:02:49 +01:00
|
|
|
int numCpus;
|
2007-09-28 14:28:12 +00:00
|
|
|
|
|
|
|
nodeToCpu = sexpr_node(root, "node/node_to_cpu");
|
2009-01-19 20:16:10 +00:00
|
|
|
if (nodeToCpu == NULL)
|
|
|
|
return 0; /* no NUMA support */
|
2007-09-28 14:28:12 +00:00
|
|
|
|
|
|
|
numCpus = sexpr_int(root, "node/nr_cpus");
|
|
|
|
|
2008-02-27 04:35:08 +00:00
|
|
|
|
|
|
|
cur = nodeToCpu;
|
|
|
|
while (*cur != 0) {
|
2013-01-10 12:46:52 -05:00
|
|
|
virBitmapPtr cpuset = NULL;
|
2008-02-27 04:35:08 +00:00
|
|
|
/*
|
|
|
|
* Find the next NUMA cell described in the xend output
|
|
|
|
*/
|
|
|
|
cur = strstr(cur, "node");
|
|
|
|
if (cur == NULL)
|
|
|
|
break;
|
|
|
|
cur += 4;
|
|
|
|
cell = virParseNumber(&cur);
|
|
|
|
if (cell < 0)
|
|
|
|
goto parse_error;
|
2011-06-29 11:30:43 -06:00
|
|
|
virSkipSpacesAndBackslash(&cur);
|
2008-02-27 04:35:08 +00:00
|
|
|
if (*cur != ':')
|
|
|
|
goto parse_error;
|
|
|
|
cur++;
|
2011-06-29 11:30:43 -06:00
|
|
|
virSkipSpacesAndBackslash(&cur);
|
2008-05-15 14:21:34 +00:00
|
|
|
if (STRPREFIX(cur, "no cpus")) {
|
2008-02-27 04:35:08 +00:00
|
|
|
nb_cpus = 0;
|
2012-09-17 14:38:20 -04:00
|
|
|
if (!(cpuset = virBitmapNew(numCpus)))
|
2013-07-04 12:18:49 +02:00
|
|
|
goto error;
|
2008-02-27 04:35:08 +00:00
|
|
|
} else {
|
2012-09-17 14:38:20 -04:00
|
|
|
nb_cpus = virBitmapParse(cur, 'n', &cpuset, numCpus);
|
2008-02-27 04:35:08 +00:00
|
|
|
if (nb_cpus < 0)
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2013-01-29 09:54:16 -05:00
|
|
|
if (VIR_ALLOC_N(cpuInfo, numCpus) < 0) {
|
|
|
|
virBitmapFree(cpuset);
|
2013-07-04 12:18:49 +02:00
|
|
|
goto error;
|
2013-01-29 09:54:16 -05:00
|
|
|
}
|
2013-01-22 18:42:08 +01:00
|
|
|
|
2012-09-17 14:38:20 -04:00
|
|
|
for (n = 0, cpu = 0; cpu < numCpus; cpu++) {
|
2015-03-11 16:41:57 +01:00
|
|
|
if (virBitmapIsBitSet(cpuset, cpu))
|
2013-01-22 18:42:08 +01:00
|
|
|
cpuInfo[n++].id = cpu;
|
2012-09-17 14:38:20 -04:00
|
|
|
}
|
2013-01-10 12:46:52 -05:00
|
|
|
virBitmapFree(cpuset);
|
2008-02-27 04:35:08 +00:00
|
|
|
|
2014-06-03 15:18:27 +02:00
|
|
|
if (virCapabilitiesAddHostNUMACell(caps, cell, 0,
|
|
|
|
nb_cpus, cpuInfo,
|
2014-06-06 18:12:51 +02:00
|
|
|
0, NULL,
|
2014-06-03 15:18:27 +02:00
|
|
|
0, NULL) < 0)
|
2013-07-04 12:18:49 +02:00
|
|
|
goto error;
|
2013-01-22 18:42:08 +01:00
|
|
|
cpuInfo = NULL;
|
2008-02-27 04:35:08 +00:00
|
|
|
}
|
2013-01-22 18:42:08 +01:00
|
|
|
|
2012-03-22 12:33:35 +01:00
|
|
|
return 0;
|
2008-02-05 19:27:37 +00:00
|
|
|
|
2014-03-25 07:54:51 +01:00
|
|
|
parse_error:
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_XEN_CALL, "%s", _("topology syntax error"));
|
2014-03-25 07:54:51 +01:00
|
|
|
error:
|
2014-09-04 17:11:57 -04:00
|
|
|
if (nb_cpus > 0)
|
|
|
|
virCapabilitiesClearHostNUMACellCPUTopology(cpuInfo, nb_cpus);
|
2013-01-22 18:42:08 +01:00
|
|
|
VIR_FREE(cpuInfo);
|
2012-03-22 12:33:35 +01:00
|
|
|
return -1;
|
2007-09-28 14:28:12 +00:00
|
|
|
}
|
|
|
|
|
2008-02-27 04:35:08 +00:00
|
|
|
|
2006-03-23 15:42:10 +00:00
|
|
|
/**
|
|
|
|
* sexpr_to_domain:
|
|
|
|
* @conn: an existing virtual connection block
|
|
|
|
* @root: an S-Expression describing a domain
|
|
|
|
*
|
|
|
|
* Internal routine returning the associated virDomainPtr for this domain
|
|
|
|
*
|
2013-05-01 10:31:23 +01:00
|
|
|
* Returns the domain def pointer or NULL in case of error.
|
2006-03-23 15:42:10 +00:00
|
|
|
*/
|
2013-05-01 10:31:23 +01:00
|
|
|
static virDomainDefPtr
|
Adjust sexpr-related interfaces to be const-correct.
* src/sexpr.c (sexpr_cons, append, sexpr_append, sexpr2string)
(sexpr_lookup_key, sexpr_lookup, sexpr_node, sexpr_fmt_node):
Add "const" attribute where appropriate.
* src/xend_internal.c (sexpr_int, sexpr_float, sexpr_u64)
(sexpr_uuid, sexpr_to_xend_domain_info, sexpr_to_xend_node_info)
(sexpr_to_xend_topology_xml, sexpr_to_domain): Likewise.
* src/sexpr.h: Adjust prototypes.
2008-01-21 14:22:15 +00:00
|
|
|
sexpr_to_domain(virConnectPtr conn, const struct sexpr *root)
|
2006-03-23 15:42:10 +00:00
|
|
|
{
|
2013-05-01 10:31:23 +01:00
|
|
|
virDomainDefPtr ret = NULL;
|
2007-08-09 20:19:12 +00:00
|
|
|
unsigned char uuid[VIR_UUID_BUFLEN];
|
2006-03-23 15:42:10 +00:00
|
|
|
const char *name;
|
2006-12-14 01:56:14 +00:00
|
|
|
const char *tmp;
|
2013-05-01 10:31:23 +01:00
|
|
|
int id = -1;
|
2013-01-29 09:53:13 -05:00
|
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
2007-04-04 14:19:49 +00:00
|
|
|
|
2007-08-09 20:19:12 +00:00
|
|
|
if (sexpr_uuid(uuid, root, "domain/uuid") < 0)
|
2006-04-09 13:11:22 +00:00
|
|
|
goto error;
|
|
|
|
name = sexpr_node(root, "domain/name");
|
|
|
|
if (name == NULL)
|
|
|
|
goto error;
|
|
|
|
|
2006-12-14 01:56:14 +00:00
|
|
|
tmp = sexpr_node(root, "domain/domid");
|
|
|
|
/* New 3.0.4 XenD will not report a domid for inactive domains,
|
|
|
|
* so only error out for old XenD
|
|
|
|
*/
|
2012-01-31 18:08:09 +01:00
|
|
|
if (!tmp && priv->xendConfigVersion < XEND_CONFIG_VERSION_3_0_4)
|
2006-03-23 15:42:10 +00:00
|
|
|
goto error;
|
|
|
|
|
2006-12-14 01:56:14 +00:00
|
|
|
if (tmp)
|
2013-05-01 10:31:23 +01:00
|
|
|
id = sexpr_int(root, "domain/domid");
|
2006-12-14 01:56:14 +00:00
|
|
|
|
2015-02-16 15:58:13 +01:00
|
|
|
return virDomainDefNewFull(name, uuid, id);
|
2006-04-09 13:11:22 +00:00
|
|
|
|
2014-03-25 07:54:51 +01:00
|
|
|
error:
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
"%s", _("failed to parse Xend domain information"));
|
Convert public datatypes to inherit from virObject
This converts the following public API datatypes to use the
virObject infrastructure:
virConnectPtr
virDomainPtr
virDomainSnapshotPtr
virInterfacePtr
virNetworkPtr
virNodeDevicePtr
virNWFilterPtr
virSecretPtr
virStreamPtr
virStorageVolPtr
virStoragePoolPtr
The code is significantly simplified, since the mutex in the
virConnectPtr object now only needs to be held when accessing
the per-connection virError object instance. All other operations
are completely lock free.
* src/datatypes.c, src/datatypes.h, src/libvirt.c: Convert
public datatypes to use virObject
* src/conf/domain_event.c, src/phyp/phyp_driver.c,
src/qemu/qemu_command.c, src/qemu/qemu_migration.c,
src/qemu/qemu_process.c, src/storage/storage_driver.c,
src/vbox/vbox_tmpl.c, src/xen/xend_internal.c,
tests/qemuxml2argvtest.c, tests/qemuxmlnstest.c,
tests/sexpr2xmltest.c, tests/xmconfigtest.c: Convert
to use virObjectUnref/virObjectRef
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2012-07-31 17:55:36 +01:00
|
|
|
virObjectUnref(ret);
|
2012-03-22 12:33:35 +01:00
|
|
|
return NULL;
|
2006-03-23 15:42:10 +00:00
|
|
|
}
|
2010-11-08 17:32:02 +01:00
|
|
|
|
2006-03-22 13:44:01 +00:00
|
|
|
|
|
|
|
/*****************************************************************
|
|
|
|
******
|
|
|
|
******
|
|
|
|
******
|
|
|
|
******
|
|
|
|
Refactored
|
|
|
|
******
|
|
|
|
******
|
|
|
|
******
|
|
|
|
******
|
|
|
|
*****************************************************************/
|
|
|
|
/**
|
|
|
|
* xenDaemonOpen:
|
|
|
|
* @conn: an existing virtual connection block
|
|
|
|
* @name: optional argument to select a connection type
|
|
|
|
* @flags: combination of virDrvOpenFlag(s)
|
|
|
|
*
|
|
|
|
* Creates a localhost Xen Daemon connection
|
|
|
|
*
|
|
|
|
* Returns 0 in case of success, -1 in case of error.
|
|
|
|
*/
|
2013-04-30 14:41:48 +01:00
|
|
|
int
|
2007-12-05 18:28:05 +00:00
|
|
|
xenDaemonOpen(virConnectPtr conn,
|
|
|
|
virConnectAuthPtr auth ATTRIBUTE_UNUSED,
|
xen: reject unknown flags
Also fix a logic bug in xenXMDomain{Attach,Detach}DeviceFlags,
where (flags & VIR_DOMAIN_DEVICE_MODIFY_CURRENT) is always false.
* src/xen/xen_driver.c (xenUnifiedDomainXMLFromNative)
(xenUnifiedDomainXMLToNative, xenUnifiedDomainBlockPeek): Reject
unknown flags.
* src/xen/xen_hypervisor.c (xenHypervisorOpen)
(xenHypervisorGetDomainState): Likewise.
* src/xen/xen_inotify.c (xenInotifyOpen): Likewise.
* src/xen/xs_internal.c (xenStoreOpen, xenStoreDomainGetState)
(xenStoreDomainReboot): Likewise.
* src/xen/xend_internal.c (xenDaemonOpen, xenDaemonDomainReboot)
(xenDaemonDomainCoreDump, xenDaemonDomainGetState)
(xenDaemonDomainMigratePrepare, xenDaemonDomainSetVcpusFlags,
xenDaemonDomainGetVcpusFlags, xenDaemonAttachDeviceFlags,
xenDaemonDetachDeviceFlags): Likewise.
(xenDaemonDomainGetXMLDesc): Prefer unsigned flags.
* src/xen/xend_internal.h (xenDaemonDomainGetXMLDesc): Likewise.
* src/xen/xm_internal.h (xenXMDomainGetXMLDesc): Likewise.
* src/xen/xm_internal.c (xenXMDomainGetXMLDesc): Likewise.
(xenXMOpen, xenXMDomainGetState, xenXMDomainSetVcpusFlags)
(xenXMDomainGetVcpusFlags): Reject unknown flags.
(xenXMDomainAttachDeviceFlags, xenXMDomainDetachDeviceFlags):
Likewise, and avoid always-false conditional.
* src/xen/xen_driver.h (XEN_MIGRATION_FLAGS): New define.
2011-07-06 17:15:33 -06:00
|
|
|
unsigned int flags)
|
2006-03-22 13:44:01 +00:00
|
|
|
{
|
2009-01-29 23:01:37 +00:00
|
|
|
char *port = NULL;
|
2013-04-30 14:41:48 +01:00
|
|
|
int ret = -1;
|
2007-11-14 11:40:57 +00:00
|
|
|
|
2013-04-30 14:41:48 +01:00
|
|
|
virCheckFlags(VIR_CONNECT_RO, -1);
|
xen: reject unknown flags
Also fix a logic bug in xenXMDomain{Attach,Detach}DeviceFlags,
where (flags & VIR_DOMAIN_DEVICE_MODIFY_CURRENT) is always false.
* src/xen/xen_driver.c (xenUnifiedDomainXMLFromNative)
(xenUnifiedDomainXMLToNative, xenUnifiedDomainBlockPeek): Reject
unknown flags.
* src/xen/xen_hypervisor.c (xenHypervisorOpen)
(xenHypervisorGetDomainState): Likewise.
* src/xen/xen_inotify.c (xenInotifyOpen): Likewise.
* src/xen/xs_internal.c (xenStoreOpen, xenStoreDomainGetState)
(xenStoreDomainReboot): Likewise.
* src/xen/xend_internal.c (xenDaemonOpen, xenDaemonDomainReboot)
(xenDaemonDomainCoreDump, xenDaemonDomainGetState)
(xenDaemonDomainMigratePrepare, xenDaemonDomainSetVcpusFlags,
xenDaemonDomainGetVcpusFlags, xenDaemonAttachDeviceFlags,
xenDaemonDetachDeviceFlags): Likewise.
(xenDaemonDomainGetXMLDesc): Prefer unsigned flags.
* src/xen/xend_internal.h (xenDaemonDomainGetXMLDesc): Likewise.
* src/xen/xm_internal.h (xenXMDomainGetXMLDesc): Likewise.
* src/xen/xm_internal.c (xenXMDomainGetXMLDesc): Likewise.
(xenXMOpen, xenXMDomainGetState, xenXMDomainSetVcpusFlags)
(xenXMDomainGetVcpusFlags): Reject unknown flags.
(xenXMDomainAttachDeviceFlags, xenXMDomainDetachDeviceFlags):
Likewise, and avoid always-false conditional.
* src/xen/xen_driver.h (XEN_MIGRATION_FLAGS): New define.
2011-07-06 17:15:33 -06:00
|
|
|
|
2007-11-14 11:40:57 +00:00
|
|
|
/* Switch on the scheme, which we expect to be NULL (file),
|
|
|
|
* "http" or "xen".
|
2007-04-04 14:19:49 +00:00
|
|
|
*/
|
2008-11-17 11:44:51 +00:00
|
|
|
if (conn->uri->scheme == NULL) {
|
2007-11-14 11:40:57 +00:00
|
|
|
/* It should be a file access */
|
2008-11-17 11:44:51 +00:00
|
|
|
if (conn->uri->path == NULL) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_NO_CONNECT, __FUNCTION__);
|
2007-11-14 11:40:57 +00:00
|
|
|
goto failed;
|
|
|
|
}
|
2009-01-29 23:01:37 +00:00
|
|
|
if (xenDaemonOpen_unix(conn, conn->uri->path) < 0 ||
|
|
|
|
xend_detect_config_version(conn) == -1)
|
2007-11-14 11:40:57 +00:00
|
|
|
goto failed;
|
2014-09-03 13:04:59 -06:00
|
|
|
} else if (STRCASEEQ(conn->uri->scheme, "xen")) {
|
2006-07-06 09:29:34 +00:00
|
|
|
/*
|
2007-04-04 14:19:49 +00:00
|
|
|
* try first to open the unix socket
|
|
|
|
*/
|
2009-01-29 23:01:37 +00:00
|
|
|
if (xenDaemonOpen_unix(conn, "/var/lib/xend/xend-socket") == 0 &&
|
|
|
|
xend_detect_config_version(conn) != -1)
|
2007-04-04 14:19:49 +00:00
|
|
|
goto done;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* try though http on port 8000
|
|
|
|
*/
|
2009-01-29 23:01:37 +00:00
|
|
|
if (xenDaemonOpen_tcp(conn, "localhost", "8000") < 0 ||
|
|
|
|
xend_detect_config_version(conn) == -1)
|
2007-04-04 14:19:49 +00:00
|
|
|
goto failed;
|
2012-10-17 10:23:12 +01:00
|
|
|
} else if (STRCASEEQ(conn->uri->scheme, "http")) {
|
2009-06-12 12:06:15 +00:00
|
|
|
if (conn->uri->port &&
|
2013-07-04 12:18:49 +02:00
|
|
|
virAsprintf(&port, "%d", conn->uri->port) == -1)
|
2009-01-22 19:41:48 +00:00
|
|
|
goto failed;
|
2009-01-29 23:01:37 +00:00
|
|
|
|
2009-06-12 12:06:15 +00:00
|
|
|
if (xenDaemonOpen_tcp(conn,
|
|
|
|
conn->uri->server ? conn->uri->server : "localhost",
|
|
|
|
port ? port : "8000") < 0 ||
|
2009-01-29 23:01:37 +00:00
|
|
|
xend_detect_config_version(conn) == -1)
|
2007-04-04 14:19:49 +00:00
|
|
|
goto failed;
|
2007-11-14 11:40:57 +00:00
|
|
|
} else {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_NO_CONNECT, __FUNCTION__);
|
2007-11-14 11:40:57 +00:00
|
|
|
goto failed;
|
2006-06-12 22:21:04 +00:00
|
|
|
}
|
2006-03-22 13:44:01 +00:00
|
|
|
|
2007-04-13 00:43:57 +00:00
|
|
|
done:
|
2013-04-30 14:41:48 +01:00
|
|
|
ret = 0;
|
2007-04-04 14:19:49 +00:00
|
|
|
|
2014-03-25 07:54:51 +01:00
|
|
|
failed:
|
2009-01-29 23:01:37 +00:00
|
|
|
VIR_FREE(port);
|
|
|
|
return ret;
|
2006-03-22 13:44:01 +00:00
|
|
|
}
|
2006-08-09 15:21:16 +00:00
|
|
|
|
2006-03-22 13:44:01 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* xenDaemonClose:
|
|
|
|
* @conn: an existing virtual connection block
|
|
|
|
*
|
|
|
|
* This method should be called when a connection to xend instance
|
|
|
|
* initialized with xenDaemonOpen is no longer needed
|
|
|
|
* to free the associated resources.
|
|
|
|
*
|
2007-04-04 14:19:49 +00:00
|
|
|
* Returns 0 in case of success, -1 in case of error
|
2006-03-22 13:44:01 +00:00
|
|
|
*/
|
|
|
|
int
|
|
|
|
xenDaemonClose(virConnectPtr conn ATTRIBUTE_UNUSED)
|
|
|
|
{
|
2007-04-04 14:19:49 +00:00
|
|
|
return 0;
|
2006-03-22 13:44:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* xenDaemonDomainSuspend:
|
2013-05-01 10:54:30 +01:00
|
|
|
* @conn: the connection object
|
|
|
|
* @def: the domain to suspend
|
2006-03-22 13:44:01 +00:00
|
|
|
*
|
|
|
|
* Pause the domain, the domain is not scheduled anymore though its resources
|
|
|
|
* are preserved. Use xenDaemonDomainResume() to resume execution.
|
|
|
|
*
|
|
|
|
* Returns 0 in case of success, -1 (with errno) in case of error.
|
|
|
|
*/
|
|
|
|
int
|
2013-05-01 10:54:30 +01:00
|
|
|
xenDaemonDomainSuspend(virConnectPtr conn, virDomainDefPtr def)
|
2006-03-22 13:44:01 +00:00
|
|
|
{
|
2013-05-01 10:54:30 +01:00
|
|
|
if (def->id < 0) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID,
|
2013-05-01 10:54:30 +01:00
|
|
|
_("Domain %s isn't running."), def->name);
|
2012-03-22 12:33:35 +01:00
|
|
|
return -1;
|
2009-02-05 17:24:51 +00:00
|
|
|
}
|
|
|
|
|
2013-05-01 10:54:30 +01:00
|
|
|
return xend_op(conn, def->name, "op", "pause", NULL);
|
2006-03-22 13:44:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* xenDaemonDomainResume:
|
2013-05-01 10:54:30 +01:00
|
|
|
* @conn: the connection object
|
|
|
|
* @def: the domain to resume
|
2006-03-22 13:44:01 +00:00
|
|
|
*
|
|
|
|
* Resume the domain after xenDaemonDomainSuspend() has been called
|
|
|
|
*
|
|
|
|
* Returns 0 in case of success, -1 (with errno) in case of error.
|
|
|
|
*/
|
|
|
|
int
|
2013-05-01 10:54:30 +01:00
|
|
|
xenDaemonDomainResume(virConnectPtr conn, virDomainDefPtr def)
|
2006-03-22 13:44:01 +00:00
|
|
|
{
|
2013-05-01 10:54:30 +01:00
|
|
|
if (def->id < 0) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID,
|
2013-05-01 10:54:30 +01:00
|
|
|
_("Domain %s isn't running."), def->name);
|
2012-03-22 12:33:35 +01:00
|
|
|
return -1;
|
2009-02-05 17:24:51 +00:00
|
|
|
}
|
|
|
|
|
2013-05-01 10:54:30 +01:00
|
|
|
return xend_op(conn, def->name, "op", "unpause", NULL);
|
2006-03-22 13:44:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* xenDaemonDomainShutdown:
|
2013-05-01 10:54:30 +01:00
|
|
|
* @conn: the connection object
|
|
|
|
* @def: the domain to shutdown
|
2006-03-22 13:44:01 +00:00
|
|
|
*
|
|
|
|
* Shutdown the domain, the OS is requested to properly shutdown
|
|
|
|
* and the domain may ignore it. It will return immediately
|
|
|
|
* after queuing the request.
|
|
|
|
*
|
|
|
|
* Returns 0 in case of success, -1 (with errno) in case of error.
|
|
|
|
*/
|
|
|
|
int
|
2013-05-01 10:54:30 +01:00
|
|
|
xenDaemonDomainShutdown(virConnectPtr conn, virDomainDefPtr def)
|
2006-03-22 13:44:01 +00:00
|
|
|
{
|
2013-05-01 10:54:30 +01:00
|
|
|
if (def->id < 0) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID,
|
2013-05-01 10:54:30 +01:00
|
|
|
_("Domain %s isn't running."), def->name);
|
2012-03-22 12:33:35 +01:00
|
|
|
return -1;
|
2009-02-05 17:24:51 +00:00
|
|
|
}
|
|
|
|
|
2013-05-01 10:54:30 +01:00
|
|
|
return xend_op(conn, def->name, "op", "shutdown", "reason", "poweroff", NULL);
|
2006-03-22 13:44:01 +00:00
|
|
|
}
|
|
|
|
|
2006-04-03 13:46:43 +00:00
|
|
|
/**
|
|
|
|
* xenDaemonDomainReboot:
|
2013-05-01 10:54:30 +01:00
|
|
|
* @conn: the connection object
|
|
|
|
* @def: the domain to reboot
|
2006-04-03 13:46:43 +00:00
|
|
|
*
|
|
|
|
* Reboot the domain, the OS is requested to properly shutdown
|
|
|
|
* and restart but the domain may ignore it. It will return immediately
|
|
|
|
* after queuing the request.
|
|
|
|
*
|
|
|
|
* Returns 0 in case of success, -1 (with errno) in case of error.
|
|
|
|
*/
|
|
|
|
int
|
2013-05-01 10:54:30 +01:00
|
|
|
xenDaemonDomainReboot(virConnectPtr conn, virDomainDefPtr def)
|
2006-04-03 13:46:43 +00:00
|
|
|
{
|
2013-05-01 10:54:30 +01:00
|
|
|
if (def->id < 0) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID,
|
2013-05-01 10:54:30 +01:00
|
|
|
_("Domain %s isn't running."), def->name);
|
2012-03-22 12:33:35 +01:00
|
|
|
return -1;
|
2009-02-05 17:24:51 +00:00
|
|
|
}
|
|
|
|
|
2013-05-01 10:54:30 +01:00
|
|
|
return xend_op(conn, def->name, "op", "shutdown", "reason", "reboot", NULL);
|
2006-04-03 13:46:43 +00:00
|
|
|
}
|
|
|
|
|
2006-03-22 13:44:01 +00:00
|
|
|
/**
|
2013-04-30 15:41:52 +01:00
|
|
|
* xenDaemonDomainDestroy:
|
2013-05-01 10:54:30 +01:00
|
|
|
* @conn: the connection object
|
|
|
|
* @def: the domain to destroy
|
2006-03-22 13:44:01 +00:00
|
|
|
*
|
|
|
|
* Abruptly halt the domain, the OS is not properly shutdown and the
|
|
|
|
* resources allocated for the domain are immediately freed, mounted
|
|
|
|
* filesystems will be marked as uncleanly shutdown.
|
|
|
|
* After calling this function, the domain's status will change to
|
|
|
|
* dying and will go away completely once all of the resources have been
|
|
|
|
* unmapped (usually from the backend devices).
|
|
|
|
*
|
|
|
|
* Returns 0 in case of success, -1 (with errno) in case of error.
|
|
|
|
*/
|
|
|
|
int
|
2013-05-01 10:54:30 +01:00
|
|
|
xenDaemonDomainDestroy(virConnectPtr conn, virDomainDefPtr def)
|
2006-03-22 13:44:01 +00:00
|
|
|
{
|
2013-05-01 10:54:30 +01:00
|
|
|
if (def->id < 0) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID,
|
2013-05-01 10:54:30 +01:00
|
|
|
_("Domain %s isn't running."), def->name);
|
2012-03-22 12:33:35 +01:00
|
|
|
return -1;
|
2009-02-05 17:24:51 +00:00
|
|
|
}
|
|
|
|
|
2013-05-01 10:54:30 +01:00
|
|
|
return xend_op(conn, def->name, "op", "destroy", NULL);
|
2006-03-22 13:44:01 +00:00
|
|
|
}
|
|
|
|
|
2007-04-13 00:43:57 +00:00
|
|
|
/**
|
|
|
|
* xenDaemonDomainGetOSType:
|
|
|
|
* @domain: a domain object
|
|
|
|
*
|
|
|
|
* Get the type of domain operation system.
|
|
|
|
*
|
|
|
|
* Returns the new string or NULL in case of error, the string must be
|
|
|
|
* freed by the caller.
|
|
|
|
*/
|
2013-04-30 16:04:36 +01:00
|
|
|
char *
|
2013-05-01 11:29:08 +01:00
|
|
|
xenDaemonDomainGetOSType(virConnectPtr conn,
|
|
|
|
virDomainDefPtr def)
|
2007-04-13 00:43:57 +00:00
|
|
|
{
|
|
|
|
char *type;
|
|
|
|
struct sexpr *root;
|
|
|
|
|
|
|
|
/* can we ask for a subset ? worth it ? */
|
2013-05-01 11:29:08 +01:00
|
|
|
root = sexpr_get(conn, "/xend/domain/%s?detail=1", def->name);
|
2007-04-13 00:43:57 +00:00
|
|
|
if (root == NULL)
|
2012-03-22 12:33:35 +01:00
|
|
|
return NULL;
|
2007-04-13 00:43:57 +00:00
|
|
|
|
2013-05-03 14:51:25 +02:00
|
|
|
ignore_value(VIR_STRDUP(type,
|
|
|
|
sexpr_lookup(root, "domain/image/hvm") ? "hvm" : "linux"));
|
2009-11-08 22:08:54 +01:00
|
|
|
|
2007-04-13 00:43:57 +00:00
|
|
|
sexpr_free(root);
|
|
|
|
|
2012-03-22 12:33:35 +01:00
|
|
|
return type;
|
2007-04-13 00:43:57 +00:00
|
|
|
}
|
|
|
|
|
2006-03-22 13:44:01 +00:00
|
|
|
/**
|
|
|
|
* xenDaemonDomainSave:
|
|
|
|
* @domain: pointer to the Domain block
|
|
|
|
* @filename: path for the output file
|
|
|
|
*
|
|
|
|
* This method will suspend a domain and save its memory contents to
|
|
|
|
* a file on disk. Use xenDaemonDomainRestore() to restore a domain after
|
|
|
|
* saving.
|
|
|
|
* Note that for remote Xen Daemon the file path will be interpreted in
|
|
|
|
* the remote host.
|
|
|
|
*
|
|
|
|
* Returns 0 in case of success, -1 (with errno) in case of error.
|
|
|
|
*/
|
|
|
|
int
|
2013-05-01 14:03:26 +01:00
|
|
|
xenDaemonDomainSave(virConnectPtr conn,
|
|
|
|
virDomainDefPtr def,
|
|
|
|
const char *filename)
|
2006-03-22 13:44:01 +00:00
|
|
|
{
|
2013-05-01 14:03:26 +01:00
|
|
|
if (def->id < 0) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID,
|
2013-05-01 14:03:26 +01:00
|
|
|
_("Domain %s isn't running."), def->name);
|
2012-03-22 12:33:35 +01:00
|
|
|
return -1;
|
2009-02-05 17:24:51 +00:00
|
|
|
}
|
2007-04-24 13:44:16 +00:00
|
|
|
|
|
|
|
/* We can't save the state of Domain-0, that would mean stopping it too */
|
2013-05-01 14:03:26 +01:00
|
|
|
if (def->id == 0) {
|
2013-04-30 16:44:21 +01:00
|
|
|
virReportError(VIR_ERR_INVALID_ARG, "%s",
|
|
|
|
_("Cannot save host domain"));
|
2012-03-22 12:33:35 +01:00
|
|
|
return -1;
|
2007-04-24 13:44:16 +00:00
|
|
|
}
|
|
|
|
|
2013-05-01 14:03:26 +01:00
|
|
|
return xend_op(conn, def->name, "op", "save", "file", filename, NULL);
|
2006-03-22 13:44:01 +00:00
|
|
|
}
|
|
|
|
|
2006-11-22 17:48:29 +00:00
|
|
|
/**
|
|
|
|
* xenDaemonDomainCoreDump:
|
2013-05-02 11:26:40 +01:00
|
|
|
* @conn: the connection object
|
|
|
|
* @def: domain configuration
|
2006-11-22 17:48:29 +00:00
|
|
|
* @filename: path for the output file
|
|
|
|
* @flags: extra flags, currently unused
|
|
|
|
*
|
|
|
|
* This method will dump the core of a domain on a given file for analysis.
|
|
|
|
* Note that for remote Xen Daemon the file path will be interpreted in
|
|
|
|
* the remote host.
|
|
|
|
*
|
|
|
|
* Returns 0 in case of success, -1 in case of error.
|
|
|
|
*/
|
2011-07-21 15:11:32 -06:00
|
|
|
int
|
2013-05-02 11:26:40 +01:00
|
|
|
xenDaemonDomainCoreDump(virConnectPtr conn,
|
|
|
|
virDomainDefPtr def,
|
2013-01-29 09:53:13 -05:00
|
|
|
const char *filename,
|
xen: reject unknown flags
Also fix a logic bug in xenXMDomain{Attach,Detach}DeviceFlags,
where (flags & VIR_DOMAIN_DEVICE_MODIFY_CURRENT) is always false.
* src/xen/xen_driver.c (xenUnifiedDomainXMLFromNative)
(xenUnifiedDomainXMLToNative, xenUnifiedDomainBlockPeek): Reject
unknown flags.
* src/xen/xen_hypervisor.c (xenHypervisorOpen)
(xenHypervisorGetDomainState): Likewise.
* src/xen/xen_inotify.c (xenInotifyOpen): Likewise.
* src/xen/xs_internal.c (xenStoreOpen, xenStoreDomainGetState)
(xenStoreDomainReboot): Likewise.
* src/xen/xend_internal.c (xenDaemonOpen, xenDaemonDomainReboot)
(xenDaemonDomainCoreDump, xenDaemonDomainGetState)
(xenDaemonDomainMigratePrepare, xenDaemonDomainSetVcpusFlags,
xenDaemonDomainGetVcpusFlags, xenDaemonAttachDeviceFlags,
xenDaemonDetachDeviceFlags): Likewise.
(xenDaemonDomainGetXMLDesc): Prefer unsigned flags.
* src/xen/xend_internal.h (xenDaemonDomainGetXMLDesc): Likewise.
* src/xen/xm_internal.h (xenXMDomainGetXMLDesc): Likewise.
* src/xen/xm_internal.c (xenXMDomainGetXMLDesc): Likewise.
(xenXMOpen, xenXMDomainGetState, xenXMDomainSetVcpusFlags)
(xenXMDomainGetVcpusFlags): Reject unknown flags.
(xenXMDomainAttachDeviceFlags, xenXMDomainDetachDeviceFlags):
Likewise, and avoid always-false conditional.
* src/xen/xen_driver.h (XEN_MIGRATION_FLAGS): New define.
2011-07-06 17:15:33 -06:00
|
|
|
unsigned int flags)
|
2006-11-22 17:48:29 +00:00
|
|
|
{
|
xen: reject unknown flags
Also fix a logic bug in xenXMDomain{Attach,Detach}DeviceFlags,
where (flags & VIR_DOMAIN_DEVICE_MODIFY_CURRENT) is always false.
* src/xen/xen_driver.c (xenUnifiedDomainXMLFromNative)
(xenUnifiedDomainXMLToNative, xenUnifiedDomainBlockPeek): Reject
unknown flags.
* src/xen/xen_hypervisor.c (xenHypervisorOpen)
(xenHypervisorGetDomainState): Likewise.
* src/xen/xen_inotify.c (xenInotifyOpen): Likewise.
* src/xen/xs_internal.c (xenStoreOpen, xenStoreDomainGetState)
(xenStoreDomainReboot): Likewise.
* src/xen/xend_internal.c (xenDaemonOpen, xenDaemonDomainReboot)
(xenDaemonDomainCoreDump, xenDaemonDomainGetState)
(xenDaemonDomainMigratePrepare, xenDaemonDomainSetVcpusFlags,
xenDaemonDomainGetVcpusFlags, xenDaemonAttachDeviceFlags,
xenDaemonDetachDeviceFlags): Likewise.
(xenDaemonDomainGetXMLDesc): Prefer unsigned flags.
* src/xen/xend_internal.h (xenDaemonDomainGetXMLDesc): Likewise.
* src/xen/xm_internal.h (xenXMDomainGetXMLDesc): Likewise.
* src/xen/xm_internal.c (xenXMDomainGetXMLDesc): Likewise.
(xenXMOpen, xenXMDomainGetState, xenXMDomainSetVcpusFlags)
(xenXMDomainGetVcpusFlags): Reject unknown flags.
(xenXMDomainAttachDeviceFlags, xenXMDomainDetachDeviceFlags):
Likewise, and avoid always-false conditional.
* src/xen/xen_driver.h (XEN_MIGRATION_FLAGS): New define.
2011-07-06 17:15:33 -06:00
|
|
|
virCheckFlags(VIR_DUMP_LIVE | VIR_DUMP_CRASH, -1);
|
|
|
|
|
2013-05-02 11:26:40 +01:00
|
|
|
if (def->id < 0) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID,
|
2013-05-02 11:26:40 +01:00
|
|
|
_("Domain %s isn't running."), def->name);
|
2012-03-22 12:33:35 +01:00
|
|
|
return -1;
|
2009-02-05 17:24:51 +00:00
|
|
|
}
|
|
|
|
|
2013-05-02 11:26:40 +01:00
|
|
|
return xend_op(conn, def->name,
|
2010-10-26 11:54:38 +02:00
|
|
|
"op", "dump", "file", filename,
|
2009-12-14 12:10:38 +01:00
|
|
|
"live", (flags & VIR_DUMP_LIVE ? "1" : "0"),
|
2009-12-14 11:59:27 +01:00
|
|
|
"crash", (flags & VIR_DUMP_CRASH ? "1" : "0"),
|
|
|
|
NULL);
|
2006-11-22 17:48:29 +00:00
|
|
|
}
|
|
|
|
|
2006-03-22 13:44:01 +00:00
|
|
|
/**
|
|
|
|
* xenDaemonDomainRestore:
|
2010-09-23 17:22:03 +02:00
|
|
|
* @conn: pointer to the Xen Daemon block
|
2006-03-22 13:44:01 +00:00
|
|
|
* @filename: path for the output file
|
|
|
|
*
|
|
|
|
* This method will restore a domain saved to disk by xenDaemonDomainSave().
|
|
|
|
* Note that for remote Xen Daemon the file path will be interpreted in
|
|
|
|
* the remote host.
|
|
|
|
*
|
|
|
|
* Returns 0 in case of success, -1 (with errno) in case of error.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
xenDaemonDomainRestore(virConnectPtr conn, const char *filename)
|
|
|
|
{
|
|
|
|
return xend_op(conn, "", "op", "restore", "file", filename, NULL);
|
|
|
|
}
|
2010-11-08 17:32:02 +01:00
|
|
|
|
2006-03-22 13:44:01 +00:00
|
|
|
|
2006-03-23 15:42:10 +00:00
|
|
|
/**
|
|
|
|
* xenDaemonDomainGetMaxMemory:
|
|
|
|
* @domain: pointer to the domain block
|
|
|
|
*
|
|
|
|
* Ask the Xen Daemon for the maximum memory allowed for a domain
|
|
|
|
*
|
|
|
|
* Returns the memory size in kilobytes or 0 in case of error.
|
|
|
|
*/
|
xml: use long long internally, to centralize overflow checks
On 64-bit platforms, unsigned long and unsigned long long are
identical, so we don't have to worry about overflow checks.
On 32-bit platforms, anywhere we narrow unsigned long long back
to unsigned long, we have to worry about overflow; it's easier
to do this in one place by having most of the code use the same
or wider types, and only doing the narrowing at the last minute.
Therefore, the memory set commands remain unsigned long, and
the memory get command now centralizes the overflow check into
libvirt.c, so that drivers don't have to repeat the work.
This also fixes a bug where xen returned the wrong value on
failure (most APIs return -1 on failure, but getMaxMemory
must return 0 on failure).
* src/driver.h (virDrvDomainGetMaxMemory): Use long long.
* src/libvirt.c (virDomainGetMaxMemory): Raise overflow.
* src/test/test_driver.c (testGetMaxMemory): Fix driver.
* src/rpc/gendispatch.pl (name_to_ProcName): Likewise.
* src/xen/xen_hypervisor.c (xenHypervisorGetMaxMemory): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainGetMaxMemory): Likewise.
* src/xen/xend_internal.c (xenDaemonDomainGetMaxMemory):
Likewise.
* src/xen/xend_internal.h (xenDaemonDomainGetMaxMemory):
Likewise.
* src/xen/xm_internal.c (xenXMDomainGetMaxMemory): Likewise.
* src/xen/xm_internal.h (xenXMDomainGetMaxMemory): Likewise.
* src/xen/xs_internal.c (xenStoreDomainGetMaxMemory): Likewise.
* src/xen/xs_internal.h (xenStoreDomainGetMaxMemory): Likewise.
* src/xenapi/xenapi_driver.c (xenapiDomainGetMaxMemory):
Likewise.
* src/esx/esx_driver.c (esxDomainGetMaxMemory): Likewise.
* src/libxl/libxl_driver.c (libxlDomainGetMaxMemory): Likewise.
* src/qemu/qemu_driver.c (qemudDomainGetMaxMemory): Likewise.
* src/lxc/lxc_driver.c (lxcDomainGetMaxMemory): Likewise.
* src/uml/uml_driver.c (umlDomainGetMaxMemory): Likewise.
2012-03-02 17:47:16 -07:00
|
|
|
unsigned long long
|
2013-05-01 11:29:08 +01:00
|
|
|
xenDaemonDomainGetMaxMemory(virConnectPtr conn, virDomainDefPtr def)
|
2006-03-23 15:42:10 +00:00
|
|
|
{
|
xml: use long long internally, to centralize overflow checks
On 64-bit platforms, unsigned long and unsigned long long are
identical, so we don't have to worry about overflow checks.
On 32-bit platforms, anywhere we narrow unsigned long long back
to unsigned long, we have to worry about overflow; it's easier
to do this in one place by having most of the code use the same
or wider types, and only doing the narrowing at the last minute.
Therefore, the memory set commands remain unsigned long, and
the memory get command now centralizes the overflow check into
libvirt.c, so that drivers don't have to repeat the work.
This also fixes a bug where xen returned the wrong value on
failure (most APIs return -1 on failure, but getMaxMemory
must return 0 on failure).
* src/driver.h (virDrvDomainGetMaxMemory): Use long long.
* src/libvirt.c (virDomainGetMaxMemory): Raise overflow.
* src/test/test_driver.c (testGetMaxMemory): Fix driver.
* src/rpc/gendispatch.pl (name_to_ProcName): Likewise.
* src/xen/xen_hypervisor.c (xenHypervisorGetMaxMemory): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainGetMaxMemory): Likewise.
* src/xen/xend_internal.c (xenDaemonDomainGetMaxMemory):
Likewise.
* src/xen/xend_internal.h (xenDaemonDomainGetMaxMemory):
Likewise.
* src/xen/xm_internal.c (xenXMDomainGetMaxMemory): Likewise.
* src/xen/xm_internal.h (xenXMDomainGetMaxMemory): Likewise.
* src/xen/xs_internal.c (xenStoreDomainGetMaxMemory): Likewise.
* src/xen/xs_internal.h (xenStoreDomainGetMaxMemory): Likewise.
* src/xenapi/xenapi_driver.c (xenapiDomainGetMaxMemory):
Likewise.
* src/esx/esx_driver.c (esxDomainGetMaxMemory): Likewise.
* src/libxl/libxl_driver.c (libxlDomainGetMaxMemory): Likewise.
* src/qemu/qemu_driver.c (qemudDomainGetMaxMemory): Likewise.
* src/lxc/lxc_driver.c (lxcDomainGetMaxMemory): Likewise.
* src/uml/uml_driver.c (umlDomainGetMaxMemory): Likewise.
2012-03-02 17:47:16 -07:00
|
|
|
unsigned long long ret = 0;
|
2006-03-23 15:42:10 +00:00
|
|
|
struct sexpr *root;
|
|
|
|
|
|
|
|
/* can we ask for a subset ? worth it ? */
|
2013-05-01 11:29:08 +01:00
|
|
|
root = sexpr_get(conn, "/xend/domain/%s?detail=1", def->name);
|
2006-03-23 15:42:10 +00:00
|
|
|
if (root == NULL)
|
xml: use long long internally, to centralize overflow checks
On 64-bit platforms, unsigned long and unsigned long long are
identical, so we don't have to worry about overflow checks.
On 32-bit platforms, anywhere we narrow unsigned long long back
to unsigned long, we have to worry about overflow; it's easier
to do this in one place by having most of the code use the same
or wider types, and only doing the narrowing at the last minute.
Therefore, the memory set commands remain unsigned long, and
the memory get command now centralizes the overflow check into
libvirt.c, so that drivers don't have to repeat the work.
This also fixes a bug where xen returned the wrong value on
failure (most APIs return -1 on failure, but getMaxMemory
must return 0 on failure).
* src/driver.h (virDrvDomainGetMaxMemory): Use long long.
* src/libvirt.c (virDomainGetMaxMemory): Raise overflow.
* src/test/test_driver.c (testGetMaxMemory): Fix driver.
* src/rpc/gendispatch.pl (name_to_ProcName): Likewise.
* src/xen/xen_hypervisor.c (xenHypervisorGetMaxMemory): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainGetMaxMemory): Likewise.
* src/xen/xend_internal.c (xenDaemonDomainGetMaxMemory):
Likewise.
* src/xen/xend_internal.h (xenDaemonDomainGetMaxMemory):
Likewise.
* src/xen/xm_internal.c (xenXMDomainGetMaxMemory): Likewise.
* src/xen/xm_internal.h (xenXMDomainGetMaxMemory): Likewise.
* src/xen/xs_internal.c (xenStoreDomainGetMaxMemory): Likewise.
* src/xen/xs_internal.h (xenStoreDomainGetMaxMemory): Likewise.
* src/xenapi/xenapi_driver.c (xenapiDomainGetMaxMemory):
Likewise.
* src/esx/esx_driver.c (esxDomainGetMaxMemory): Likewise.
* src/libxl/libxl_driver.c (libxlDomainGetMaxMemory): Likewise.
* src/qemu/qemu_driver.c (qemudDomainGetMaxMemory): Likewise.
* src/lxc/lxc_driver.c (lxcDomainGetMaxMemory): Likewise.
* src/uml/uml_driver.c (umlDomainGetMaxMemory): Likewise.
2012-03-02 17:47:16 -07:00
|
|
|
return 0;
|
2006-03-23 15:42:10 +00:00
|
|
|
|
xml: use long long internally, to centralize overflow checks
On 64-bit platforms, unsigned long and unsigned long long are
identical, so we don't have to worry about overflow checks.
On 32-bit platforms, anywhere we narrow unsigned long long back
to unsigned long, we have to worry about overflow; it's easier
to do this in one place by having most of the code use the same
or wider types, and only doing the narrowing at the last minute.
Therefore, the memory set commands remain unsigned long, and
the memory get command now centralizes the overflow check into
libvirt.c, so that drivers don't have to repeat the work.
This also fixes a bug where xen returned the wrong value on
failure (most APIs return -1 on failure, but getMaxMemory
must return 0 on failure).
* src/driver.h (virDrvDomainGetMaxMemory): Use long long.
* src/libvirt.c (virDomainGetMaxMemory): Raise overflow.
* src/test/test_driver.c (testGetMaxMemory): Fix driver.
* src/rpc/gendispatch.pl (name_to_ProcName): Likewise.
* src/xen/xen_hypervisor.c (xenHypervisorGetMaxMemory): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainGetMaxMemory): Likewise.
* src/xen/xend_internal.c (xenDaemonDomainGetMaxMemory):
Likewise.
* src/xen/xend_internal.h (xenDaemonDomainGetMaxMemory):
Likewise.
* src/xen/xm_internal.c (xenXMDomainGetMaxMemory): Likewise.
* src/xen/xm_internal.h (xenXMDomainGetMaxMemory): Likewise.
* src/xen/xs_internal.c (xenStoreDomainGetMaxMemory): Likewise.
* src/xen/xs_internal.h (xenStoreDomainGetMaxMemory): Likewise.
* src/xenapi/xenapi_driver.c (xenapiDomainGetMaxMemory):
Likewise.
* src/esx/esx_driver.c (esxDomainGetMaxMemory): Likewise.
* src/libxl/libxl_driver.c (libxlDomainGetMaxMemory): Likewise.
* src/qemu/qemu_driver.c (qemudDomainGetMaxMemory): Likewise.
* src/lxc/lxc_driver.c (lxcDomainGetMaxMemory): Likewise.
* src/uml/uml_driver.c (umlDomainGetMaxMemory): Likewise.
2012-03-02 17:47:16 -07:00
|
|
|
ret = sexpr_u64(root, "domain/memory") << 10;
|
2006-03-23 15:42:10 +00:00
|
|
|
sexpr_free(root);
|
|
|
|
|
xml: use long long internally, to centralize overflow checks
On 64-bit platforms, unsigned long and unsigned long long are
identical, so we don't have to worry about overflow checks.
On 32-bit platforms, anywhere we narrow unsigned long long back
to unsigned long, we have to worry about overflow; it's easier
to do this in one place by having most of the code use the same
or wider types, and only doing the narrowing at the last minute.
Therefore, the memory set commands remain unsigned long, and
the memory get command now centralizes the overflow check into
libvirt.c, so that drivers don't have to repeat the work.
This also fixes a bug where xen returned the wrong value on
failure (most APIs return -1 on failure, but getMaxMemory
must return 0 on failure).
* src/driver.h (virDrvDomainGetMaxMemory): Use long long.
* src/libvirt.c (virDomainGetMaxMemory): Raise overflow.
* src/test/test_driver.c (testGetMaxMemory): Fix driver.
* src/rpc/gendispatch.pl (name_to_ProcName): Likewise.
* src/xen/xen_hypervisor.c (xenHypervisorGetMaxMemory): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainGetMaxMemory): Likewise.
* src/xen/xend_internal.c (xenDaemonDomainGetMaxMemory):
Likewise.
* src/xen/xend_internal.h (xenDaemonDomainGetMaxMemory):
Likewise.
* src/xen/xm_internal.c (xenXMDomainGetMaxMemory): Likewise.
* src/xen/xm_internal.h (xenXMDomainGetMaxMemory): Likewise.
* src/xen/xs_internal.c (xenStoreDomainGetMaxMemory): Likewise.
* src/xen/xs_internal.h (xenStoreDomainGetMaxMemory): Likewise.
* src/xenapi/xenapi_driver.c (xenapiDomainGetMaxMemory):
Likewise.
* src/esx/esx_driver.c (esxDomainGetMaxMemory): Likewise.
* src/libxl/libxl_driver.c (libxlDomainGetMaxMemory): Likewise.
* src/qemu/qemu_driver.c (qemudDomainGetMaxMemory): Likewise.
* src/lxc/lxc_driver.c (lxcDomainGetMaxMemory): Likewise.
* src/uml/uml_driver.c (umlDomainGetMaxMemory): Likewise.
2012-03-02 17:47:16 -07:00
|
|
|
return ret;
|
2006-03-23 15:42:10 +00:00
|
|
|
}
|
|
|
|
|
2010-11-08 17:32:02 +01:00
|
|
|
|
2006-03-22 13:44:01 +00:00
|
|
|
/**
|
|
|
|
* xenDaemonDomainSetMaxMemory:
|
|
|
|
* @domain: pointer to the Domain block
|
|
|
|
* @memory: The maximum memory in kilobytes
|
|
|
|
*
|
|
|
|
* This method will set the maximum amount of memory that can be allocated to
|
|
|
|
* a domain. Please note that a domain is able to allocate up to this amount
|
2006-04-13 17:18:49 +00:00
|
|
|
* on its own.
|
2006-03-22 13:44:01 +00:00
|
|
|
*
|
|
|
|
* Returns 0 for success; -1 (with errno) on error
|
|
|
|
*/
|
|
|
|
int
|
2013-05-01 11:29:08 +01:00
|
|
|
xenDaemonDomainSetMaxMemory(virConnectPtr conn,
|
|
|
|
virDomainDefPtr def,
|
|
|
|
unsigned long memory)
|
2006-03-22 13:44:01 +00:00
|
|
|
{
|
|
|
|
char buf[1024];
|
2006-11-15 21:03:34 +00:00
|
|
|
|
2011-01-28 22:03:24 +01:00
|
|
|
snprintf(buf, sizeof(buf), "%lu", VIR_DIV_UP(memory, 1024));
|
2013-05-01 11:29:08 +01:00
|
|
|
return xend_op(conn, def->name, "op", "maxmem_set", "memory",
|
2006-03-22 13:44:01 +00:00
|
|
|
buf, NULL);
|
|
|
|
}
|
|
|
|
|
2006-04-13 17:18:49 +00:00
|
|
|
/**
|
|
|
|
* xenDaemonDomainSetMemory:
|
|
|
|
* @domain: pointer to the Domain block
|
|
|
|
* @memory: The target memory in kilobytes
|
|
|
|
*
|
|
|
|
* This method will set a target memory allocation for a given domain and
|
|
|
|
* request that the guest meet this target. The guest may or may not actually
|
|
|
|
* achieve this target. When this function returns, it does not signify that
|
|
|
|
* the domain has actually reached that target.
|
|
|
|
*
|
|
|
|
* Memory for a domain can only be allocated up to the maximum memory setting.
|
|
|
|
* There is no safe guard for allocations that are too small so be careful
|
|
|
|
* when using this function to reduce a domain's memory usage.
|
|
|
|
*
|
|
|
|
* Returns 0 for success; -1 (with errno) on error
|
|
|
|
*/
|
|
|
|
int
|
2013-05-01 11:29:08 +01:00
|
|
|
xenDaemonDomainSetMemory(virConnectPtr conn,
|
|
|
|
virDomainDefPtr def,
|
|
|
|
unsigned long memory)
|
2006-04-13 17:18:49 +00:00
|
|
|
{
|
|
|
|
char buf[1024];
|
2006-11-15 21:03:34 +00:00
|
|
|
|
2011-01-28 22:03:24 +01:00
|
|
|
snprintf(buf, sizeof(buf), "%lu", VIR_DIV_UP(memory, 1024));
|
2013-05-01 11:29:08 +01:00
|
|
|
return xend_op(conn, def->name, "op", "mem_target_set",
|
2006-04-13 17:18:49 +00:00
|
|
|
"target", buf, NULL);
|
|
|
|
}
|
|
|
|
|
2006-08-09 15:21:16 +00:00
|
|
|
|
2008-07-25 13:17:27 +00:00
|
|
|
virDomainDefPtr
|
2013-01-29 09:53:13 -05:00
|
|
|
xenDaemonDomainFetch(virConnectPtr conn, int domid, const char *name,
|
2008-07-25 13:17:27 +00:00
|
|
|
const char *cpus)
|
2006-12-14 01:56:14 +00:00
|
|
|
{
|
|
|
|
struct sexpr *root;
|
2013-01-29 09:53:13 -05:00
|
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
2013-09-25 09:31:30 +01:00
|
|
|
virDomainDefPtr def = NULL;
|
2011-02-21 14:40:08 +01:00
|
|
|
int id;
|
|
|
|
char * tty;
|
|
|
|
int vncport;
|
2006-12-14 01:56:14 +00:00
|
|
|
|
2008-07-25 13:17:27 +00:00
|
|
|
if (name)
|
|
|
|
root = sexpr_get(conn, "/xend/domain/%s?detail=1", name);
|
|
|
|
else
|
|
|
|
root = sexpr_get(conn, "/xend/domain/%d?detail=1", domid);
|
2011-09-01 16:22:38 -06:00
|
|
|
if (root == NULL)
|
2012-03-22 12:33:35 +01:00
|
|
|
return NULL;
|
2006-09-12 01:16:22 +00:00
|
|
|
|
2013-09-25 09:31:30 +01:00
|
|
|
if (xenGetDomIdFromSxpr(root, priv->xendConfigVersion, &id) < 0)
|
|
|
|
goto cleanup;
|
2011-02-21 14:40:08 +01:00
|
|
|
xenUnifiedLock(priv);
|
2012-01-03 15:39:59 -07:00
|
|
|
if (sexpr_lookup(root, "domain/image/hvm"))
|
|
|
|
tty = xenStoreDomainGetSerialConsolePath(conn, id);
|
|
|
|
else
|
|
|
|
tty = xenStoreDomainGetConsolePath(conn, id);
|
2011-02-21 14:40:08 +01:00
|
|
|
vncport = xenStoreDomainGetVNCPort(conn, id);
|
|
|
|
xenUnifiedUnlock(priv);
|
2011-02-21 14:40:12 +01:00
|
|
|
if (!(def = xenParseSxpr(root,
|
|
|
|
priv->xendConfigVersion,
|
|
|
|
cpus,
|
|
|
|
tty,
|
|
|
|
vncport)))
|
2008-07-25 10:49:33 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2014-03-25 07:54:51 +01:00
|
|
|
cleanup:
|
2006-08-09 15:21:16 +00:00
|
|
|
sexpr_free(root);
|
|
|
|
|
2012-03-22 12:33:35 +01:00
|
|
|
return def;
|
2006-08-09 15:21:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-03-22 13:44:01 +00:00
|
|
|
/**
|
2011-05-06 21:53:10 +02:00
|
|
|
* xenDaemonDomainGetXMLDesc:
|
2006-02-20 17:22:16 +00:00
|
|
|
* @domain: a domain object
|
2007-10-31 09:39:13 +00:00
|
|
|
* @cpus: list of cpu the domain is pinned to.
|
2006-02-20 17:22:16 +00:00
|
|
|
*
|
2013-05-01 14:15:10 +01:00
|
|
|
* Get the XML description of the domain as a structure.
|
2006-02-20 17:22:16 +00:00
|
|
|
*
|
2013-05-01 14:15:10 +01:00
|
|
|
* Returns a virDomainDefPtr instance, or NULL in case of error.
|
2006-02-20 17:22:16 +00:00
|
|
|
*/
|
2013-05-01 14:15:10 +01:00
|
|
|
virDomainDefPtr
|
|
|
|
xenDaemonDomainGetXMLDesc(virConnectPtr conn,
|
|
|
|
virDomainDefPtr minidef,
|
xen: reject unknown flags
Also fix a logic bug in xenXMDomain{Attach,Detach}DeviceFlags,
where (flags & VIR_DOMAIN_DEVICE_MODIFY_CURRENT) is always false.
* src/xen/xen_driver.c (xenUnifiedDomainXMLFromNative)
(xenUnifiedDomainXMLToNative, xenUnifiedDomainBlockPeek): Reject
unknown flags.
* src/xen/xen_hypervisor.c (xenHypervisorOpen)
(xenHypervisorGetDomainState): Likewise.
* src/xen/xen_inotify.c (xenInotifyOpen): Likewise.
* src/xen/xs_internal.c (xenStoreOpen, xenStoreDomainGetState)
(xenStoreDomainReboot): Likewise.
* src/xen/xend_internal.c (xenDaemonOpen, xenDaemonDomainReboot)
(xenDaemonDomainCoreDump, xenDaemonDomainGetState)
(xenDaemonDomainMigratePrepare, xenDaemonDomainSetVcpusFlags,
xenDaemonDomainGetVcpusFlags, xenDaemonAttachDeviceFlags,
xenDaemonDetachDeviceFlags): Likewise.
(xenDaemonDomainGetXMLDesc): Prefer unsigned flags.
* src/xen/xend_internal.h (xenDaemonDomainGetXMLDesc): Likewise.
* src/xen/xm_internal.h (xenXMDomainGetXMLDesc): Likewise.
* src/xen/xm_internal.c (xenXMDomainGetXMLDesc): Likewise.
(xenXMOpen, xenXMDomainGetState, xenXMDomainSetVcpusFlags)
(xenXMDomainGetVcpusFlags): Reject unknown flags.
(xenXMDomainAttachDeviceFlags, xenXMDomainDetachDeviceFlags):
Likewise, and avoid always-false conditional.
* src/xen/xen_driver.h (XEN_MIGRATION_FLAGS): New define.
2011-07-06 17:15:33 -06:00
|
|
|
const char *cpus)
|
2006-03-15 12:13:25 +00:00
|
|
|
{
|
2013-05-01 14:15:10 +01:00
|
|
|
return xenDaemonDomainFetch(conn,
|
|
|
|
minidef->id,
|
|
|
|
minidef->name,
|
|
|
|
cpus);
|
2006-02-20 17:22:16 +00:00
|
|
|
}
|
2010-11-08 17:32:02 +01:00
|
|
|
|
2006-03-22 13:44:01 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* xenDaemonDomainGetInfo:
|
|
|
|
* @domain: a domain object
|
|
|
|
* @info: pointer to a virDomainInfo structure allocated by the user
|
|
|
|
*
|
|
|
|
* This method looks up information about a domain and update the
|
|
|
|
* information block provided.
|
|
|
|
*
|
|
|
|
* Returns 0 in case of success, -1 in case of error
|
|
|
|
*/
|
|
|
|
int
|
2013-05-01 11:29:08 +01:00
|
|
|
xenDaemonDomainGetInfo(virConnectPtr conn,
|
|
|
|
virDomainDefPtr def,
|
|
|
|
virDomainInfoPtr info)
|
2006-03-22 13:44:01 +00:00
|
|
|
{
|
|
|
|
struct sexpr *root;
|
|
|
|
int ret;
|
|
|
|
|
2013-05-01 11:29:08 +01:00
|
|
|
root = sexpr_get(conn, "/xend/domain/%s?detail=1", def->name);
|
2006-03-22 13:44:01 +00:00
|
|
|
if (root == NULL)
|
2012-03-22 12:33:35 +01:00
|
|
|
return -1;
|
2006-03-22 13:44:01 +00:00
|
|
|
|
2013-05-01 11:29:08 +01:00
|
|
|
ret = sexpr_to_xend_domain_info(def, root, info);
|
2006-03-22 13:44:01 +00:00
|
|
|
sexpr_free(root);
|
2012-03-22 12:33:35 +01:00
|
|
|
return ret;
|
2006-03-22 13:44:01 +00:00
|
|
|
}
|
2011-05-02 11:35:29 +02:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* xenDaemonDomainGetState:
|
|
|
|
* @domain: a domain object
|
|
|
|
* @state: returned domain's state
|
|
|
|
* @reason: returned reason for the state
|
|
|
|
*
|
|
|
|
* This method looks up domain state and reason.
|
|
|
|
*
|
|
|
|
* Returns 0 in case of success, -1 in case of error
|
|
|
|
*/
|
|
|
|
int
|
2013-05-01 11:29:08 +01:00
|
|
|
xenDaemonDomainGetState(virConnectPtr conn,
|
|
|
|
virDomainDefPtr def,
|
2011-05-02 11:35:29 +02:00
|
|
|
int *state,
|
2013-04-30 16:42:29 +01:00
|
|
|
int *reason)
|
2011-05-02 11:35:29 +02:00
|
|
|
{
|
|
|
|
struct sexpr *root;
|
|
|
|
|
2013-05-01 11:29:08 +01:00
|
|
|
root = sexpr_get(conn, "/xend/domain/%s?detail=1", def->name);
|
2011-05-02 11:35:29 +02:00
|
|
|
if (!root)
|
|
|
|
return -1;
|
|
|
|
|
2013-05-01 11:29:08 +01:00
|
|
|
*state = sexpr_to_xend_domain_state(def, root);
|
2011-05-02 11:35:29 +02:00
|
|
|
if (reason)
|
|
|
|
*reason = 0;
|
|
|
|
|
|
|
|
sexpr_free(root);
|
|
|
|
return 0;
|
|
|
|
}
|
2006-03-22 13:44:01 +00:00
|
|
|
|
2010-11-08 17:32:02 +01:00
|
|
|
|
2006-03-23 15:42:10 +00:00
|
|
|
/**
|
Fri Jul 6 16:08:00 BST 2007 Richard W.M. Jones <rjones@redhat.com>
* src/proxy_internal.c, src/proxy_internal.h,
src.xen_internal.c, src/xen_internal.h,
src/xen_unified.c, src/xen_unified.h,
src/xend_internal.c, src/xend_internal.h,
src/xm_internal.c, src/xm_internal.h,
src/xs_internal.c, src/xs_internal.h: The interface
between xen_unified.c and its underlying driver now uses
a custom structure (struct xenUnifiedDriver) instead
of reusing virDriver.
* src/xen_unified.c: virDomainLookup* functions in Xen
now throw VIR_ERR_NO_DOMAIN if the domain does not exist.
* src/xs_internal.c: Fix indentation.
2007-07-06 15:11:22 +00:00
|
|
|
* xenDaemonLookupByName:
|
2006-03-23 15:42:10 +00:00
|
|
|
* @conn: A xend instance
|
|
|
|
* @name: The name of the domain
|
|
|
|
*
|
|
|
|
* This method looks up information about a domain and returns
|
|
|
|
* it in the form of a struct xend_domain. This should be
|
|
|
|
* free()'d when no longer needed.
|
|
|
|
*
|
2013-05-01 10:31:23 +01:00
|
|
|
* Returns domain def pointer on success; NULL on error
|
2006-03-23 15:42:10 +00:00
|
|
|
*/
|
2013-05-01 10:31:23 +01:00
|
|
|
virDomainDefPtr
|
Fri Jul 6 16:08:00 BST 2007 Richard W.M. Jones <rjones@redhat.com>
* src/proxy_internal.c, src/proxy_internal.h,
src.xen_internal.c, src/xen_internal.h,
src/xen_unified.c, src/xen_unified.h,
src/xend_internal.c, src/xend_internal.h,
src/xm_internal.c, src/xm_internal.h,
src/xs_internal.c, src/xs_internal.h: The interface
between xen_unified.c and its underlying driver now uses
a custom structure (struct xenUnifiedDriver) instead
of reusing virDriver.
* src/xen_unified.c: virDomainLookup* functions in Xen
now throw VIR_ERR_NO_DOMAIN if the domain does not exist.
* src/xs_internal.c: Fix indentation.
2007-07-06 15:11:22 +00:00
|
|
|
xenDaemonLookupByName(virConnectPtr conn, const char *domname)
|
2006-03-23 15:42:10 +00:00
|
|
|
{
|
|
|
|
struct sexpr *root;
|
2013-05-01 10:31:23 +01:00
|
|
|
virDomainDefPtr ret = NULL;
|
2006-03-23 15:42:10 +00:00
|
|
|
|
|
|
|
root = sexpr_get(conn, "/xend/domain/%s?detail=1", domname);
|
|
|
|
if (root == NULL)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
ret = sexpr_to_domain(conn, root);
|
|
|
|
|
2014-03-25 07:54:51 +01:00
|
|
|
error:
|
2006-03-23 15:42:10 +00:00
|
|
|
sexpr_free(root);
|
2012-03-22 12:33:35 +01:00
|
|
|
return ret;
|
2006-03-23 15:42:10 +00:00
|
|
|
}
|
2010-11-08 17:32:02 +01:00
|
|
|
|
2006-03-23 15:42:10 +00:00
|
|
|
|
2006-03-29 12:46:03 +00:00
|
|
|
/**
|
|
|
|
* xenDaemonNodeGetInfo:
|
|
|
|
* @conn: pointer to the Xen Daemon block
|
|
|
|
* @info: pointer to a virNodeInfo structure allocated by the user
|
2008-02-05 19:27:37 +00:00
|
|
|
*
|
2006-03-29 12:46:03 +00:00
|
|
|
* Extract hardware information about the node.
|
|
|
|
*
|
|
|
|
* Returns 0 in case of success and -1 in case of failure.
|
|
|
|
*/
|
2006-07-03 11:12:12 +00:00
|
|
|
int
|
2013-01-29 09:53:13 -05:00
|
|
|
xenDaemonNodeGetInfo(virConnectPtr conn, virNodeInfoPtr info)
|
|
|
|
{
|
2006-03-29 12:46:03 +00:00
|
|
|
int ret = -1;
|
|
|
|
struct sexpr *root;
|
|
|
|
|
|
|
|
root = sexpr_get(conn, "/xend/node/");
|
|
|
|
if (root == NULL)
|
2012-03-22 12:33:35 +01:00
|
|
|
return -1;
|
2006-03-29 12:46:03 +00:00
|
|
|
|
|
|
|
ret = sexpr_to_xend_node_info(root, info);
|
|
|
|
sexpr_free(root);
|
2012-03-22 12:33:35 +01:00
|
|
|
return ret;
|
2006-03-29 12:46:03 +00:00
|
|
|
}
|
|
|
|
|
2007-09-28 14:28:12 +00:00
|
|
|
/**
|
|
|
|
* xenDaemonNodeGetTopology:
|
|
|
|
* @conn: pointer to the Xen Daemon block
|
2008-02-27 04:35:08 +00:00
|
|
|
* @caps: capabilities info
|
2007-09-28 14:28:12 +00:00
|
|
|
*
|
|
|
|
* This method retrieves a node's topology information.
|
|
|
|
*
|
|
|
|
* Returns -1 in case of error, 0 otherwise.
|
|
|
|
*/
|
|
|
|
int
|
2013-01-29 09:53:13 -05:00
|
|
|
xenDaemonNodeGetTopology(virConnectPtr conn, virCapsPtr caps)
|
|
|
|
{
|
2007-09-28 14:28:12 +00:00
|
|
|
int ret = -1;
|
|
|
|
struct sexpr *root;
|
|
|
|
|
|
|
|
root = sexpr_get(conn, "/xend/node/");
|
2014-11-13 15:29:45 +01:00
|
|
|
if (root == NULL)
|
2012-03-22 12:33:35 +01:00
|
|
|
return -1;
|
2007-09-28 14:28:12 +00:00
|
|
|
|
2010-04-03 01:10:49 +02:00
|
|
|
ret = sexpr_to_xend_topology(root, caps);
|
2007-09-28 14:28:12 +00:00
|
|
|
sexpr_free(root);
|
2012-03-22 12:33:35 +01:00
|
|
|
return ret;
|
2007-09-28 14:28:12 +00:00
|
|
|
}
|
|
|
|
|
2010-11-08 17:32:02 +01:00
|
|
|
|
2010-10-14 16:17:18 -06:00
|
|
|
/**
|
|
|
|
* xenDaemonDomainSetVcpusFlags:
|
2013-05-01 18:37:29 +01:00
|
|
|
* @conn: the connection object
|
|
|
|
* @def: domain configuration
|
2010-10-14 16:17:18 -06:00
|
|
|
* @nvcpus: the new number of virtual CPUs for this domain
|
|
|
|
* @flags: bitwise-ORd from virDomainVcpuFlags
|
|
|
|
*
|
|
|
|
* Change virtual CPUs allocation of domain according to flags.
|
|
|
|
*
|
2013-04-30 17:25:27 +01:00
|
|
|
* Returns 0 on success, -1 if an error message was issued
|
2010-10-14 16:17:18 -06:00
|
|
|
*/
|
|
|
|
int
|
2013-05-01 18:37:29 +01:00
|
|
|
xenDaemonDomainSetVcpusFlags(virConnectPtr conn,
|
|
|
|
virDomainDefPtr def,
|
2013-01-29 09:53:13 -05:00
|
|
|
unsigned int vcpus,
|
2010-10-14 16:17:18 -06:00
|
|
|
unsigned int flags)
|
|
|
|
{
|
|
|
|
char buf[VIR_UUID_BUFLEN];
|
|
|
|
int max;
|
|
|
|
|
xen: reject unknown flags
Also fix a logic bug in xenXMDomain{Attach,Detach}DeviceFlags,
where (flags & VIR_DOMAIN_DEVICE_MODIFY_CURRENT) is always false.
* src/xen/xen_driver.c (xenUnifiedDomainXMLFromNative)
(xenUnifiedDomainXMLToNative, xenUnifiedDomainBlockPeek): Reject
unknown flags.
* src/xen/xen_hypervisor.c (xenHypervisorOpen)
(xenHypervisorGetDomainState): Likewise.
* src/xen/xen_inotify.c (xenInotifyOpen): Likewise.
* src/xen/xs_internal.c (xenStoreOpen, xenStoreDomainGetState)
(xenStoreDomainReboot): Likewise.
* src/xen/xend_internal.c (xenDaemonOpen, xenDaemonDomainReboot)
(xenDaemonDomainCoreDump, xenDaemonDomainGetState)
(xenDaemonDomainMigratePrepare, xenDaemonDomainSetVcpusFlags,
xenDaemonDomainGetVcpusFlags, xenDaemonAttachDeviceFlags,
xenDaemonDetachDeviceFlags): Likewise.
(xenDaemonDomainGetXMLDesc): Prefer unsigned flags.
* src/xen/xend_internal.h (xenDaemonDomainGetXMLDesc): Likewise.
* src/xen/xm_internal.h (xenXMDomainGetXMLDesc): Likewise.
* src/xen/xm_internal.c (xenXMDomainGetXMLDesc): Likewise.
(xenXMOpen, xenXMDomainGetState, xenXMDomainSetVcpusFlags)
(xenXMDomainGetVcpusFlags): Reject unknown flags.
(xenXMDomainAttachDeviceFlags, xenXMDomainDetachDeviceFlags):
Likewise, and avoid always-false conditional.
* src/xen/xen_driver.h (XEN_MIGRATION_FLAGS): New define.
2011-07-06 17:15:33 -06:00
|
|
|
virCheckFlags(VIR_DOMAIN_VCPU_LIVE |
|
|
|
|
VIR_DOMAIN_VCPU_CONFIG |
|
|
|
|
VIR_DOMAIN_VCPU_MAXIMUM, -1);
|
|
|
|
|
2013-01-29 09:53:13 -05:00
|
|
|
if (vcpus < 1) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INVALID_ARG, __FUNCTION__);
|
2012-03-22 12:33:35 +01:00
|
|
|
return -1;
|
2010-10-14 16:17:18 -06:00
|
|
|
}
|
|
|
|
|
2013-05-01 18:37:29 +01:00
|
|
|
if (def->id < 0) {
|
2010-10-14 16:17:18 -06:00
|
|
|
if (flags & VIR_DOMAIN_VCPU_LIVE) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID, "%s",
|
|
|
|
_("domain not running"));
|
2010-10-14 16:17:18 -06:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if ((flags & (VIR_DOMAIN_VCPU_LIVE | VIR_DOMAIN_VCPU_CONFIG)) !=
|
|
|
|
(VIR_DOMAIN_VCPU_LIVE | VIR_DOMAIN_VCPU_CONFIG)) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID, "%s",
|
|
|
|
_("Xend only supports modifying both live and "
|
|
|
|
"persistent config"));
|
2010-10-14 16:17:18 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Unfortunately, xend_op does not validate whether this exceeds
|
|
|
|
* the maximum. */
|
|
|
|
flags |= VIR_DOMAIN_VCPU_MAXIMUM;
|
2013-05-01 18:37:29 +01:00
|
|
|
if ((max = xenDaemonDomainGetVcpusFlags(conn, def, flags)) < 0) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID, "%s",
|
|
|
|
_("could not determine max vcpus for the domain"));
|
2010-10-14 16:17:18 -06:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (vcpus > max) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
|
|
_("requested vcpus is greater than max allowable"
|
|
|
|
" vcpus for the domain: %d > %d"), vcpus, max);
|
2010-10-14 16:17:18 -06:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
snprintf(buf, sizeof(buf), "%d", vcpus);
|
2013-05-01 18:37:29 +01:00
|
|
|
return xend_op(conn, def->name, "op", "set_vcpus", "vcpus",
|
2010-10-14 16:17:18 -06:00
|
|
|
buf, NULL);
|
|
|
|
}
|
|
|
|
|
2006-08-04 10:41:05 +00:00
|
|
|
/**
|
|
|
|
* xenDaemonDomainPinCpu:
|
2013-05-01 18:37:29 +01:00
|
|
|
* @conn: the connection object
|
|
|
|
* @minidef: minimal domain configuration
|
2006-08-04 10:41:05 +00:00
|
|
|
* @vcpu: virtual CPU number
|
|
|
|
* @cpumap: pointer to a bit map of real CPUs (in 8-bit bytes)
|
|
|
|
* @maplen: length of cpumap in bytes
|
2008-02-05 19:27:37 +00:00
|
|
|
*
|
2006-08-04 10:41:05 +00:00
|
|
|
* Dynamically change the real CPUs which can be allocated to a virtual CPU.
|
2009-05-14 10:31:11 +00:00
|
|
|
* NOTE: The XenD cpu affinity map format changed from "[0,1,2]" to
|
|
|
|
* "0,1,2"
|
|
|
|
* the XenD cpu affinity works only after cset 19579.
|
|
|
|
* there is no fine grained xend version detection possible, so we
|
|
|
|
* use the old format for anything before version 3
|
2006-08-04 10:41:05 +00:00
|
|
|
*
|
|
|
|
* Returns 0 for success; -1 (with errno) on error
|
|
|
|
*/
|
|
|
|
int
|
2013-05-01 18:37:29 +01:00
|
|
|
xenDaemonDomainPinVcpu(virConnectPtr conn,
|
|
|
|
virDomainDefPtr minidef,
|
2013-01-29 09:53:13 -05:00
|
|
|
unsigned int vcpu,
|
|
|
|
unsigned char *cpumap,
|
|
|
|
int maplen)
|
2006-08-04 10:41:05 +00:00
|
|
|
{
|
2009-05-14 10:31:11 +00:00
|
|
|
char buf[VIR_UUID_BUFLEN], mapstr[sizeof(cpumap_t) * 64];
|
Convert 'int i' to 'size_t i' in src/{xen,xenapi,xenxs} 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, j;
|
2013-05-01 18:37:29 +01:00
|
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
2006-08-04 10:41:05 +00:00
|
|
|
|
2013-01-29 09:53:13 -05:00
|
|
|
if (maplen > (int)sizeof(cpumap_t)) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INVALID_ARG, __FUNCTION__);
|
2012-03-22 12:33:35 +01:00
|
|
|
return -1;
|
2006-08-04 10:41:05 +00:00
|
|
|
}
|
2009-02-05 17:24:51 +00:00
|
|
|
|
2012-01-31 18:08:09 +01:00
|
|
|
if (priv->xendConfigVersion < XEND_CONFIG_VERSION_3_0_4) {
|
2009-07-30 13:29:46 +02:00
|
|
|
mapstr[0] = '[';
|
|
|
|
mapstr[1] = 0;
|
2009-05-14 10:31:11 +00:00
|
|
|
} else {
|
2009-07-30 13:29:46 +02:00
|
|
|
mapstr[0] = 0;
|
2009-05-14 10:31:11 +00:00
|
|
|
}
|
|
|
|
|
2006-08-04 10:41:05 +00:00
|
|
|
/* from bit map, build character string of mapped CPU numbers */
|
|
|
|
for (i = 0; i < maplen; i++) for (j = 0; j < 8; j++)
|
|
|
|
if (cpumap[i] & (1 << j)) {
|
Convert 'int i' to 'size_t i' in src/{xen,xenapi,xenxs} 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
|
|
|
snprintf(buf, sizeof(buf), "%zu,", (8 * i) + j);
|
2006-08-04 10:41:05 +00:00
|
|
|
strcat(mapstr, buf);
|
|
|
|
}
|
2012-01-31 18:08:09 +01:00
|
|
|
if (priv->xendConfigVersion < XEND_CONFIG_VERSION_3_0_4)
|
2009-05-14 10:31:11 +00:00
|
|
|
mapstr[strlen(mapstr) - 1] = ']';
|
|
|
|
else
|
|
|
|
mapstr[strlen(mapstr) - 1] = 0;
|
|
|
|
|
2006-08-04 10:41:05 +00:00
|
|
|
snprintf(buf, sizeof(buf), "%d", vcpu);
|
2011-03-29 21:43:41 +08:00
|
|
|
|
2015-05-28 14:40:30 +02:00
|
|
|
return xend_op(conn, minidef->name, "op", "pincpu", "vcpu", buf,
|
|
|
|
"cpumap", mapstr, NULL);
|
2011-03-29 21:43:41 +08:00
|
|
|
|
2006-08-04 10:41:05 +00:00
|
|
|
}
|
|
|
|
|
2010-10-06 17:54:41 -06:00
|
|
|
/**
|
|
|
|
* xenDaemonDomainGetVcpusFlags:
|
2013-05-01 18:37:29 +01:00
|
|
|
* @conn: the connection object
|
|
|
|
* @def: domain configuration
|
2010-10-06 17:54:41 -06:00
|
|
|
* @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
|
2013-04-30 17:25:27 +01:00
|
|
|
* issued
|
2010-10-06 17:54:41 -06:00
|
|
|
|
|
|
|
*/
|
|
|
|
int
|
2013-05-01 18:37:29 +01:00
|
|
|
xenDaemonDomainGetVcpusFlags(virConnectPtr conn,
|
|
|
|
virDomainDefPtr def,
|
|
|
|
unsigned int flags)
|
2010-10-06 17:54:41 -06:00
|
|
|
{
|
|
|
|
struct sexpr *root;
|
|
|
|
int ret;
|
|
|
|
|
xen: reject unknown flags
Also fix a logic bug in xenXMDomain{Attach,Detach}DeviceFlags,
where (flags & VIR_DOMAIN_DEVICE_MODIFY_CURRENT) is always false.
* src/xen/xen_driver.c (xenUnifiedDomainXMLFromNative)
(xenUnifiedDomainXMLToNative, xenUnifiedDomainBlockPeek): Reject
unknown flags.
* src/xen/xen_hypervisor.c (xenHypervisorOpen)
(xenHypervisorGetDomainState): Likewise.
* src/xen/xen_inotify.c (xenInotifyOpen): Likewise.
* src/xen/xs_internal.c (xenStoreOpen, xenStoreDomainGetState)
(xenStoreDomainReboot): Likewise.
* src/xen/xend_internal.c (xenDaemonOpen, xenDaemonDomainReboot)
(xenDaemonDomainCoreDump, xenDaemonDomainGetState)
(xenDaemonDomainMigratePrepare, xenDaemonDomainSetVcpusFlags,
xenDaemonDomainGetVcpusFlags, xenDaemonAttachDeviceFlags,
xenDaemonDetachDeviceFlags): Likewise.
(xenDaemonDomainGetXMLDesc): Prefer unsigned flags.
* src/xen/xend_internal.h (xenDaemonDomainGetXMLDesc): Likewise.
* src/xen/xm_internal.h (xenXMDomainGetXMLDesc): Likewise.
* src/xen/xm_internal.c (xenXMDomainGetXMLDesc): Likewise.
(xenXMOpen, xenXMDomainGetState, xenXMDomainSetVcpusFlags)
(xenXMDomainGetVcpusFlags): Reject unknown flags.
(xenXMDomainAttachDeviceFlags, xenXMDomainDetachDeviceFlags):
Likewise, and avoid always-false conditional.
* src/xen/xen_driver.h (XEN_MIGRATION_FLAGS): New define.
2011-07-06 17:15:33 -06:00
|
|
|
virCheckFlags(VIR_DOMAIN_VCPU_LIVE |
|
|
|
|
VIR_DOMAIN_VCPU_CONFIG |
|
|
|
|
VIR_DOMAIN_VCPU_MAXIMUM, -1);
|
|
|
|
|
2013-05-01 18:37:29 +01:00
|
|
|
if (def->id < 0 && (flags & VIR_DOMAIN_VCPU_LIVE)) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID, "%s",
|
|
|
|
_("domain not active"));
|
2010-10-06 17:54:41 -06:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2013-05-01 18:37:29 +01:00
|
|
|
root = sexpr_get(conn, "/xend/domain/%s?detail=1", def->name);
|
2010-10-06 17:54:41 -06:00
|
|
|
if (root == NULL)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
ret = sexpr_int(root, "domain/vcpus");
|
|
|
|
if (!(flags & VIR_DOMAIN_VCPU_MAXIMUM)) {
|
2010-10-29 10:51:01 -06:00
|
|
|
int vcpus = count_one_bits_l(sexpr_u64(root, "domain/vcpu_avail"));
|
2010-10-06 17:54:41 -06:00
|
|
|
if (vcpus)
|
|
|
|
ret = MIN(vcpus, ret);
|
|
|
|
}
|
|
|
|
if (!ret)
|
2013-04-30 17:25:27 +01:00
|
|
|
ret = -1;
|
2010-10-06 17:54:41 -06:00
|
|
|
sexpr_free(root);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2006-08-04 10:41:05 +00:00
|
|
|
/**
|
|
|
|
* virDomainGetVcpus:
|
2013-05-01 18:37:29 +01:00
|
|
|
* @conn: the connection object
|
|
|
|
* @def: domain configuration
|
2006-08-04 10:41:05 +00:00
|
|
|
* @info: pointer to an array of virVcpuInfo structures (OUT)
|
|
|
|
* @maxinfo: number of structures in info array
|
2011-12-03 17:06:07 -07:00
|
|
|
* @cpumaps: pointer to a bit map of real CPUs for all vcpus of this domain (in 8-bit bytes) (OUT)
|
2008-04-04 07:58:29 +00:00
|
|
|
* If cpumaps is NULL, then no cpumap information is returned by the API.
|
2006-08-04 10:41:05 +00:00
|
|
|
* It's assumed there is <maxinfo> cpumap in cpumaps array.
|
|
|
|
* The memory allocated to cpumaps must be (maxinfo * maplen) bytes
|
|
|
|
* (ie: calloc(maxinfo, maplen)).
|
|
|
|
* One cpumap inside cpumaps has the format described in virDomainPinVcpu() API.
|
|
|
|
* @maplen: number of bytes in one cpumap, from 1 up to size of CPU map in
|
|
|
|
* underlying virtualization system (Xen...).
|
2008-02-05 19:27:37 +00:00
|
|
|
*
|
2006-08-04 10:41:05 +00:00
|
|
|
* Extract information about virtual CPUs of domain, store it in info array
|
2008-04-04 07:58:29 +00:00
|
|
|
* and also in cpumaps if this pointer isn't NULL.
|
2006-08-04 10:41:05 +00:00
|
|
|
*
|
|
|
|
* Returns the number of info filled in case of success, -1 in case of failure.
|
|
|
|
*/
|
|
|
|
int
|
2013-05-01 18:37:29 +01:00
|
|
|
xenDaemonDomainGetVcpus(virConnectPtr conn,
|
|
|
|
virDomainDefPtr def,
|
2013-01-29 09:53:13 -05:00
|
|
|
virVcpuInfoPtr info,
|
|
|
|
int maxinfo,
|
|
|
|
unsigned char *cpumaps,
|
|
|
|
int maplen)
|
2006-08-04 10:41:05 +00:00
|
|
|
{
|
|
|
|
struct sexpr *root, *s, *t;
|
|
|
|
virVcpuInfoPtr ipt = info;
|
|
|
|
int nbinfo = 0, oln;
|
|
|
|
unsigned char *cpumap;
|
|
|
|
int vcpu, cpu;
|
|
|
|
|
2013-05-01 18:37:29 +01:00
|
|
|
root = sexpr_get(conn, "/xend/domain/%s?op=vcpuinfo", def->name);
|
2006-08-04 10:41:05 +00:00
|
|
|
if (root == NULL)
|
2012-03-22 12:33:35 +01:00
|
|
|
return -1;
|
2006-08-04 10:41:05 +00:00
|
|
|
|
|
|
|
if (cpumaps != NULL)
|
2006-09-28 19:20:52 +00:00
|
|
|
memset(cpumaps, 0, maxinfo * maplen);
|
2006-08-04 10:41:05 +00:00
|
|
|
|
|
|
|
/* scan the sexprs from "(vcpu (number x)...)" and get parameter values */
|
2007-09-29 18:16:26 +00:00
|
|
|
for (s = root; s->kind == SEXPR_CONS; s = s->u.s.cdr) {
|
|
|
|
if ((s->u.s.car->kind == SEXPR_CONS) &&
|
|
|
|
(s->u.s.car->u.s.car->kind == SEXPR_VALUE) &&
|
2008-05-14 19:51:24 +00:00
|
|
|
STREQ(s->u.s.car->u.s.car->u.value, "vcpu")) {
|
2007-09-29 18:16:26 +00:00
|
|
|
t = s->u.s.car;
|
2006-09-28 19:20:52 +00:00
|
|
|
vcpu = ipt->number = sexpr_int(t, "vcpu/number");
|
|
|
|
if ((oln = sexpr_int(t, "vcpu/online")) != 0) {
|
|
|
|
if (sexpr_int(t, "vcpu/running")) ipt->state = VIR_VCPU_RUNNING;
|
|
|
|
if (sexpr_int(t, "vcpu/blocked")) ipt->state = VIR_VCPU_BLOCKED;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
ipt->state = VIR_VCPU_OFFLINE;
|
|
|
|
ipt->cpuTime = sexpr_float(t, "vcpu/cpu_time") * 1000000000;
|
|
|
|
ipt->cpu = oln ? sexpr_int(t, "vcpu/cpu") : -1;
|
|
|
|
|
|
|
|
if (cpumaps != NULL && vcpu >= 0 && vcpu < maxinfo) {
|
|
|
|
cpumap = (unsigned char *) VIR_GET_CPUMAP(cpumaps, maplen, vcpu);
|
|
|
|
/*
|
|
|
|
* get sexpr from "(cpumap (x y z...))" and convert values
|
|
|
|
* to bitmap
|
|
|
|
*/
|
2007-09-29 18:16:26 +00:00
|
|
|
for (t = t->u.s.cdr; t->kind == SEXPR_CONS; t = t->u.s.cdr)
|
|
|
|
if ((t->u.s.car->kind == SEXPR_CONS) &&
|
|
|
|
(t->u.s.car->u.s.car->kind == SEXPR_VALUE) &&
|
2008-05-14 19:51:24 +00:00
|
|
|
STREQ(t->u.s.car->u.s.car->u.value, "cpumap") &&
|
2007-09-29 18:16:26 +00:00
|
|
|
(t->u.s.car->u.s.cdr->kind == SEXPR_CONS)) {
|
|
|
|
for (t = t->u.s.car->u.s.cdr->u.s.car; t->kind == SEXPR_CONS; t = t->u.s.cdr)
|
2007-11-12 14:00:32 +00:00
|
|
|
if (t->u.s.car->kind == SEXPR_VALUE
|
2008-02-08 09:15:16 +00:00
|
|
|
&& virStrToLong_i(t->u.s.car->u.value, NULL, 10, &cpu) == 0
|
2007-11-12 14:00:32 +00:00
|
|
|
&& cpu >= 0
|
|
|
|
&& (VIR_CPU_MAPLEN(cpu+1) <= maplen)) {
|
|
|
|
VIR_USE_CPU(cpumap, cpu);
|
2006-09-28 19:20:52 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2006-08-04 10:41:05 +00:00
|
|
|
}
|
|
|
|
|
2006-09-28 19:20:52 +00:00
|
|
|
if (++nbinfo == maxinfo) break;
|
|
|
|
ipt++;
|
|
|
|
}
|
2006-08-04 10:41:05 +00:00
|
|
|
}
|
|
|
|
sexpr_free(root);
|
2012-03-22 12:33:35 +01:00
|
|
|
return nbinfo;
|
2006-08-04 10:41:05 +00:00
|
|
|
}
|
|
|
|
|
2006-06-15 14:50:48 +00:00
|
|
|
/**
|
|
|
|
* xenDaemonLookupByUUID:
|
|
|
|
* @conn: pointer to the hypervisor connection
|
|
|
|
* @uuid: the raw UUID for the domain
|
|
|
|
*
|
|
|
|
* Try to lookup a domain on xend based on its UUID.
|
|
|
|
*
|
2013-05-01 10:31:23 +01:00
|
|
|
* Returns domain def pointer on success; NULL on error
|
2006-06-15 14:50:48 +00:00
|
|
|
*/
|
2013-05-01 10:31:23 +01:00
|
|
|
virDomainDefPtr
|
2006-06-15 14:50:48 +00:00
|
|
|
xenDaemonLookupByUUID(virConnectPtr conn, const unsigned char *uuid)
|
|
|
|
{
|
2013-05-01 10:31:23 +01:00
|
|
|
virDomainDefPtr ret;
|
2006-06-15 14:50:48 +00:00
|
|
|
char *name = NULL;
|
|
|
|
int id = -1;
|
2013-01-29 09:53:13 -05:00
|
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
2007-04-04 14:19:49 +00:00
|
|
|
|
2007-03-02 20:19:08 +00:00
|
|
|
/* Old approach for xen <= 3.0.3 */
|
2012-01-31 18:08:09 +01:00
|
|
|
if (priv->xendConfigVersion < XEND_CONFIG_VERSION_3_0_4) {
|
2007-03-02 20:19:08 +00:00
|
|
|
char **names, **tmp;
|
|
|
|
unsigned char ident[VIR_UUID_BUFLEN];
|
|
|
|
names = xenDaemonListDomainsOld(conn);
|
|
|
|
tmp = names;
|
|
|
|
|
2014-11-13 15:29:45 +01:00
|
|
|
if (names == NULL)
|
2012-03-22 12:33:35 +01:00
|
|
|
return NULL;
|
2007-03-02 20:19:08 +00:00
|
|
|
while (*tmp != NULL) {
|
|
|
|
id = xenDaemonDomainLookupByName_ids(conn, *tmp, &ident[0]);
|
|
|
|
if (id >= 0) {
|
|
|
|
if (!memcmp(uuid, ident, VIR_UUID_BUFLEN)) {
|
2011-09-20 12:11:31 -06:00
|
|
|
name = *tmp;
|
2007-03-02 20:19:08 +00:00
|
|
|
break;
|
|
|
|
}
|
2006-06-15 14:50:48 +00:00
|
|
|
}
|
2007-03-02 20:19:08 +00:00
|
|
|
tmp++;
|
2006-06-15 14:50:48 +00:00
|
|
|
}
|
2011-09-20 12:11:31 -06:00
|
|
|
tmp = names;
|
|
|
|
while (*tmp) {
|
|
|
|
if (*tmp != name)
|
|
|
|
VIR_FREE(*tmp);
|
|
|
|
tmp++;
|
|
|
|
}
|
2008-05-29 19:20:22 +00:00
|
|
|
VIR_FREE(names);
|
2007-03-02 20:19:08 +00:00
|
|
|
} else { /* New approach for xen >= 3.0.4 */
|
|
|
|
char *domname = NULL;
|
|
|
|
char uuidstr[VIR_UUID_STRING_BUFLEN];
|
|
|
|
struct sexpr *root = NULL;
|
|
|
|
|
2007-08-09 20:19:12 +00:00
|
|
|
virUUIDFormat(uuid, uuidstr);
|
2007-03-02 20:19:08 +00:00
|
|
|
root = sexpr_get(conn, "/xend/domain/%s?detail=1", uuidstr);
|
|
|
|
if (root == NULL)
|
2012-03-22 12:33:35 +01:00
|
|
|
return NULL;
|
2007-03-02 20:19:08 +00:00
|
|
|
domname = (char*)sexpr_node(root, "domain/name");
|
|
|
|
if (sexpr_node(root, "domain/domid")) /* only active domains have domid */
|
|
|
|
id = sexpr_int(root, "domain/domid");
|
|
|
|
else
|
|
|
|
id = -1;
|
2009-11-08 22:08:54 +01:00
|
|
|
|
2013-05-03 14:51:25 +02:00
|
|
|
ignore_value(VIR_STRDUP(name, domname));
|
2009-11-08 22:08:54 +01:00
|
|
|
|
2007-03-02 20:19:08 +00:00
|
|
|
sexpr_free(root);
|
2006-06-15 14:50:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (name == NULL)
|
2012-03-22 12:33:35 +01:00
|
|
|
return NULL;
|
2006-06-15 14:50:48 +00:00
|
|
|
|
2015-02-16 15:58:13 +01:00
|
|
|
ret = virDomainDefNewFull(name, uuid, id);
|
2009-09-08 16:48:20 +02:00
|
|
|
|
2008-05-29 19:20:22 +00:00
|
|
|
VIR_FREE(name);
|
2012-03-22 12:33:35 +01:00
|
|
|
return ret;
|
2006-06-15 14:50:48 +00:00
|
|
|
}
|
2006-06-16 12:36:40 +00:00
|
|
|
|
|
|
|
/**
|
2008-10-10 09:32:27 +00:00
|
|
|
* xenDaemonCreateXML:
|
2006-06-16 12:36:40 +00:00
|
|
|
* @conn: pointer to the hypervisor connection
|
2013-05-01 14:15:10 +01:00
|
|
|
* @def: domain configuration
|
2006-06-16 12:36:40 +00:00
|
|
|
* @flags: an optional set of virDomainFlags
|
|
|
|
*
|
|
|
|
* Launch a new Linux guest domain, based on an XML description similar
|
|
|
|
* to the one returned by virDomainGetXMLDesc()
|
2008-03-17 10:27:31 +00:00
|
|
|
* This function may requires privileged access to the hypervisor.
|
2008-02-05 19:27:37 +00:00
|
|
|
*
|
2006-06-16 12:36:40 +00:00
|
|
|
* Returns a new domain object or NULL in case of failure
|
|
|
|
*/
|
2013-05-01 14:15:10 +01:00
|
|
|
int
|
|
|
|
xenDaemonCreateXML(virConnectPtr conn, virDomainDefPtr def)
|
2006-06-16 12:36:40 +00:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
char *sexpr;
|
2013-05-01 14:15:10 +01:00
|
|
|
const char *tmp;
|
|
|
|
struct sexpr *root;
|
2013-01-29 09:53:13 -05:00
|
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
2006-06-16 12:36:40 +00:00
|
|
|
|
2013-05-01 14:15:10 +01:00
|
|
|
if (def->id != -1) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID,
|
|
|
|
_("Domain %s is already running"),
|
|
|
|
def->name);
|
|
|
|
return -1;
|
2006-06-16 12:36:40 +00:00
|
|
|
}
|
|
|
|
|
2013-05-01 14:15:10 +01:00
|
|
|
if (!(sexpr = xenFormatSxpr(conn, def, priv->xendConfigVersion)))
|
|
|
|
return -1;
|
|
|
|
|
2008-10-10 09:32:27 +00:00
|
|
|
ret = xenDaemonDomainCreateXML(conn, sexpr);
|
2008-05-29 19:20:22 +00:00
|
|
|
VIR_FREE(sexpr);
|
2014-11-13 15:29:45 +01:00
|
|
|
if (ret != 0)
|
2006-06-16 12:36:40 +00:00
|
|
|
goto error;
|
|
|
|
|
2007-02-22 19:09:29 +00:00
|
|
|
/* This comes before wait_for_devices, to ensure that latter
|
|
|
|
cleanup will destroy the domain upon failure */
|
2013-05-01 14:15:10 +01:00
|
|
|
root = sexpr_get(conn, "/xend/domain/%s?detail=1", def->name);
|
|
|
|
if (root == NULL)
|
2006-06-16 12:36:40 +00:00
|
|
|
goto error;
|
|
|
|
|
2013-05-01 14:15:10 +01:00
|
|
|
tmp = sexpr_node(root, "domain/domid");
|
|
|
|
if (!tmp) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Domain %s did not start"),
|
|
|
|
def->name);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
if (tmp)
|
|
|
|
def->id = sexpr_int(root, "domain/domid");
|
|
|
|
|
2009-09-04 19:14:09 +02:00
|
|
|
if (xend_wait_for_devices(conn, def->name) < 0)
|
2006-06-16 12:36:40 +00:00
|
|
|
goto error;
|
|
|
|
|
2013-05-01 10:54:30 +01:00
|
|
|
if (xenDaemonDomainResume(conn, def) < 0)
|
2006-06-16 12:36:40 +00:00
|
|
|
goto error;
|
|
|
|
|
2013-05-01 14:15:10 +01:00
|
|
|
return 0;
|
2007-02-22 19:09:29 +00:00
|
|
|
|
2014-03-25 07:54:51 +01:00
|
|
|
error:
|
2007-02-22 19:09:29 +00:00
|
|
|
/* Make sure we don't leave a still-born domain around */
|
2013-05-01 14:15:10 +01:00
|
|
|
if (def->id != -1)
|
2013-05-01 10:54:30 +01:00
|
|
|
xenDaemonDomainDestroy(conn, def);
|
2013-05-01 14:15:10 +01:00
|
|
|
return -1;
|
2006-06-16 12:36:40 +00:00
|
|
|
}
|
2006-11-16 18:11:28 +00:00
|
|
|
|
|
|
|
/**
|
2010-01-13 18:44:26 -07:00
|
|
|
* xenDaemonAttachDeviceFlags:
|
2013-05-02 10:56:31 +01:00
|
|
|
* @conn: the connection object
|
|
|
|
* @minidef: domain configuration
|
2006-11-16 18:11:28 +00:00
|
|
|
* @xml: pointer to XML description of device
|
2010-01-13 18:44:26 -07:00
|
|
|
* @flags: an OR'ed set of virDomainDeviceModifyFlags
|
2008-02-05 19:27:37 +00:00
|
|
|
*
|
2006-11-16 18:11:28 +00:00
|
|
|
* Create a virtual device attachment to backend.
|
|
|
|
* XML description is translated into S-expression.
|
|
|
|
*
|
|
|
|
* Returns 0 in case of success, -1 in case of failure.
|
|
|
|
*/
|
2013-04-30 17:43:07 +01:00
|
|
|
int
|
2013-05-02 10:56:31 +01:00
|
|
|
xenDaemonAttachDeviceFlags(virConnectPtr conn,
|
|
|
|
virDomainDefPtr minidef,
|
2013-01-29 09:53:13 -05:00
|
|
|
const char *xml,
|
2010-01-13 18:44:26 -07:00
|
|
|
unsigned int flags)
|
2006-11-16 18:11:28 +00:00
|
|
|
{
|
2013-05-02 10:56:31 +01:00
|
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
2008-07-25 13:17:27 +00:00
|
|
|
char *sexpr = NULL;
|
|
|
|
int ret = -1;
|
|
|
|
virDomainDeviceDefPtr dev = NULL;
|
|
|
|
virDomainDefPtr def = NULL;
|
|
|
|
virBuffer buf = VIR_BUFFER_INITIALIZER;
|
2007-09-10 11:47:17 +00:00
|
|
|
char class[8], ref[80];
|
2011-02-12 14:29:09 +08:00
|
|
|
char *target = NULL;
|
2006-11-16 18:11:28 +00:00
|
|
|
|
xen: reject unknown flags
Also fix a logic bug in xenXMDomain{Attach,Detach}DeviceFlags,
where (flags & VIR_DOMAIN_DEVICE_MODIFY_CURRENT) is always false.
* src/xen/xen_driver.c (xenUnifiedDomainXMLFromNative)
(xenUnifiedDomainXMLToNative, xenUnifiedDomainBlockPeek): Reject
unknown flags.
* src/xen/xen_hypervisor.c (xenHypervisorOpen)
(xenHypervisorGetDomainState): Likewise.
* src/xen/xen_inotify.c (xenInotifyOpen): Likewise.
* src/xen/xs_internal.c (xenStoreOpen, xenStoreDomainGetState)
(xenStoreDomainReboot): Likewise.
* src/xen/xend_internal.c (xenDaemonOpen, xenDaemonDomainReboot)
(xenDaemonDomainCoreDump, xenDaemonDomainGetState)
(xenDaemonDomainMigratePrepare, xenDaemonDomainSetVcpusFlags,
xenDaemonDomainGetVcpusFlags, xenDaemonAttachDeviceFlags,
xenDaemonDetachDeviceFlags): Likewise.
(xenDaemonDomainGetXMLDesc): Prefer unsigned flags.
* src/xen/xend_internal.h (xenDaemonDomainGetXMLDesc): Likewise.
* src/xen/xm_internal.h (xenXMDomainGetXMLDesc): Likewise.
* src/xen/xm_internal.c (xenXMDomainGetXMLDesc): Likewise.
(xenXMOpen, xenXMDomainGetState, xenXMDomainSetVcpusFlags)
(xenXMDomainGetVcpusFlags): Reject unknown flags.
(xenXMDomainAttachDeviceFlags, xenXMDomainDetachDeviceFlags):
Likewise, and avoid always-false conditional.
* src/xen/xen_driver.h (XEN_MIGRATION_FLAGS): New define.
2011-07-06 17:15:33 -06:00
|
|
|
virCheckFlags(VIR_DOMAIN_AFFECT_LIVE | VIR_DOMAIN_AFFECT_CONFIG, -1);
|
|
|
|
|
2013-05-02 10:56:31 +01:00
|
|
|
if (minidef->id < 0) {
|
2010-10-01 16:22:29 +02:00
|
|
|
/* Cannot modify live config if domain is inactive */
|
|
|
|
if (flags & VIR_DOMAIN_DEVICE_MODIFY_LIVE) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID, "%s",
|
|
|
|
_("Cannot modify live config if domain is inactive"));
|
2010-10-01 16:22:29 +02:00
|
|
|
return -1;
|
|
|
|
}
|
2010-01-13 18:44:26 -07:00
|
|
|
} else {
|
|
|
|
/* Only live config can be changed if xendConfigVersion < 3 */
|
2012-01-31 18:08:09 +01:00
|
|
|
if (priv->xendConfigVersion < XEND_CONFIG_VERSION_3_0_4 &&
|
2010-07-28 17:12:16 -06:00
|
|
|
(flags != VIR_DOMAIN_DEVICE_MODIFY_CURRENT &&
|
2010-01-13 18:44:26 -07:00
|
|
|
flags != VIR_DOMAIN_DEVICE_MODIFY_LIVE)) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID, "%s",
|
|
|
|
_("Xend version does not support modifying "
|
|
|
|
"persistent config"));
|
2010-01-13 18:44:26 -07:00
|
|
|
return -1;
|
|
|
|
}
|
2010-03-22 13:13:53 +00:00
|
|
|
/* Xen only supports modifying both live and persistent config if
|
2010-01-13 18:44:26 -07:00
|
|
|
* xendConfigVersion >= 3
|
|
|
|
*/
|
2012-01-31 18:08:09 +01:00
|
|
|
if (priv->xendConfigVersion >= XEND_CONFIG_VERSION_3_0_4 &&
|
2010-10-01 16:31:04 +02:00
|
|
|
(flags != (VIR_DOMAIN_DEVICE_MODIFY_LIVE |
|
|
|
|
VIR_DOMAIN_DEVICE_MODIFY_CONFIG))) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID, "%s",
|
|
|
|
_("Xend only supports modifying both live and "
|
|
|
|
"persistent config"));
|
2010-01-13 18:44:26 -07:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
2007-08-16 15:38:38 +00:00
|
|
|
|
2013-05-02 10:56:31 +01:00
|
|
|
if (!(def = xenDaemonDomainFetch(conn,
|
|
|
|
minidef->id,
|
|
|
|
minidef->name,
|
2008-07-25 13:17:27 +00:00
|
|
|
NULL)))
|
|
|
|
goto cleanup;
|
|
|
|
|
2013-03-28 14:55:55 +01:00
|
|
|
if (!(dev = virDomainDeviceDefParse(xml, def, priv->caps, priv->xmlopt,
|
2014-11-18 16:44:00 +00:00
|
|
|
VIR_DOMAIN_DEF_PARSE_INACTIVE)))
|
2008-07-25 13:17:27 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
|
|
|
|
switch (dev->type) {
|
|
|
|
case VIR_DOMAIN_DEVICE_DISK:
|
2011-07-08 15:58:28 -06:00
|
|
|
if (xenFormatSxprDisk(dev->data.disk,
|
|
|
|
&buf,
|
2015-04-16 20:11:06 -04:00
|
|
|
def->os.type == VIR_DOMAIN_OSTYPE_HVM ? 1 : 0,
|
2011-07-08 15:58:28 -06:00
|
|
|
priv->xendConfigVersion, 1) < 0)
|
2008-07-25 13:17:27 +00:00
|
|
|
goto cleanup;
|
2011-02-12 14:29:09 +08:00
|
|
|
|
2013-05-03 14:51:25 +02:00
|
|
|
if (dev->data.disk->device != VIR_DOMAIN_DISK_DEVICE_CDROM &&
|
|
|
|
VIR_STRDUP(target, dev->data.disk->dst) < 0)
|
|
|
|
goto cleanup;
|
With the recent refactoring of the domain code, plus the changes with the Xend
code, a couple of bugs were introduced into the attach-disk and attach-interface
functionality. This patch fixes 3 bugs:
1) In xenDaemonAttachDevice(), there is a switch statement to determine which
of the xenDaemonFormatSxpr{Disk,Net} functions to call. Unfortunately, the case
statements are all missing the corresponding "break", so we always fall-through
to the default error case. This patch just adds the appropriate break statements.
2) (minor) In xenDaemonDomainDefineXML (that's a mouthful!), there is a stray
"fprintf". This is now converted to a proper virXendError().
3) xenDaemonFormatSxpr{Disk,Net} were adding an extra (device to the front of
the sexpr expressions that xend did not expect (this is Xend on RHEL 5.2).
Because of this, the attaches would fail. The patch fixes this by removing the
(device from the front, which makes attach-disk and attach-interface work again.
Signed-off-by: Chris Lalancette <clalance@redhat.com>
2008-08-05 16:45:07 +00:00
|
|
|
break;
|
2008-07-25 13:17:27 +00:00
|
|
|
|
|
|
|
case VIR_DOMAIN_DEVICE_NET:
|
2013-05-02 10:56:31 +01:00
|
|
|
if (xenFormatSxprNet(conn,
|
2011-02-21 14:40:12 +01:00
|
|
|
dev->data.net,
|
|
|
|
&buf,
|
2015-04-16 20:11:06 -04:00
|
|
|
def->os.type == VIR_DOMAIN_OSTYPE_HVM ? 1 : 0,
|
2011-02-21 14:40:12 +01:00
|
|
|
priv->xendConfigVersion, 1) < 0)
|
2008-07-25 13:17:27 +00:00
|
|
|
goto cleanup;
|
2011-02-12 14:29:09 +08:00
|
|
|
|
|
|
|
char macStr[VIR_MAC_STRING_BUFLEN];
|
2012-07-17 08:07:59 -04:00
|
|
|
virMacAddrFormat(&dev->data.net->mac, macStr);
|
2011-02-12 14:29:09 +08:00
|
|
|
|
2013-05-03 14:51:25 +02:00
|
|
|
if (VIR_STRDUP(target, macStr) < 0)
|
2011-02-12 14:29:09 +08:00
|
|
|
goto cleanup;
|
With the recent refactoring of the domain code, plus the changes with the Xend
code, a couple of bugs were introduced into the attach-disk and attach-interface
functionality. This patch fixes 3 bugs:
1) In xenDaemonAttachDevice(), there is a switch statement to determine which
of the xenDaemonFormatSxpr{Disk,Net} functions to call. Unfortunately, the case
statements are all missing the corresponding "break", so we always fall-through
to the default error case. This patch just adds the appropriate break statements.
2) (minor) In xenDaemonDomainDefineXML (that's a mouthful!), there is a stray
"fprintf". This is now converted to a proper virXendError().
3) xenDaemonFormatSxpr{Disk,Net} were adding an extra (device to the front of
the sexpr expressions that xend did not expect (this is Xend on RHEL 5.2).
Because of this, the attaches would fail. The patch fixes this by removing the
(device from the front, which makes attach-disk and attach-interface work again.
Signed-off-by: Chris Lalancette <clalance@redhat.com>
2008-08-05 16:45:07 +00:00
|
|
|
break;
|
2008-07-25 13:17:27 +00:00
|
|
|
|
2009-04-03 12:38:52 +00:00
|
|
|
case VIR_DOMAIN_DEVICE_HOSTDEV:
|
|
|
|
if (dev->data.hostdev->mode == VIR_DOMAIN_HOSTDEV_MODE_SUBSYS &&
|
|
|
|
dev->data.hostdev->source.subsys.type == VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_PCI) {
|
2011-02-21 14:40:12 +01:00
|
|
|
if (xenFormatSxprOnePCI(dev->data.hostdev, &buf, 0) < 0)
|
2009-04-03 12:38:52 +00:00
|
|
|
goto cleanup;
|
2011-02-12 14:29:09 +08:00
|
|
|
|
2012-08-16 16:41:06 +01:00
|
|
|
virDevicePCIAddress PCIAddr;
|
2011-02-12 14:29:09 +08:00
|
|
|
|
2013-03-18 15:56:12 -04:00
|
|
|
PCIAddr = dev->data.hostdev->source.subsys.u.pci.addr;
|
2013-01-30 15:43:39 -05:00
|
|
|
if (virAsprintf(&target, "PCI device: %.4x:%.2x:%.2x",
|
2013-07-04 12:18:49 +02:00
|
|
|
PCIAddr.domain, PCIAddr.bus, PCIAddr.slot) < 0)
|
2011-02-12 14:29:09 +08:00
|
|
|
goto cleanup;
|
2009-04-03 12:38:52 +00:00
|
|
|
} else {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("unsupported device type"));
|
2009-04-03 12:38:52 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2008-07-25 13:17:27 +00:00
|
|
|
default:
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("unsupported device type"));
|
2008-07-25 13:17:27 +00:00
|
|
|
goto cleanup;
|
2006-11-16 18:11:28 +00:00
|
|
|
}
|
2008-07-25 13:17:27 +00:00
|
|
|
|
|
|
|
sexpr = virBufferContentAndReset(&buf);
|
|
|
|
|
2013-05-02 10:56:31 +01:00
|
|
|
if (virDomainXMLDevID(conn, minidef, dev, class, ref, sizeof(ref))) {
|
2007-09-10 11:47:17 +00:00
|
|
|
/* device doesn't exist, define it */
|
2013-05-02 10:56:31 +01:00
|
|
|
ret = xend_op(conn, def->name, "op", "device_create",
|
2008-07-25 13:17:27 +00:00
|
|
|
"config", sexpr, NULL);
|
2011-02-12 14:29:09 +08:00
|
|
|
} else {
|
|
|
|
if (dev->data.disk->device != VIR_DOMAIN_DISK_DEVICE_CDROM) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID,
|
|
|
|
_("target '%s' already exists"), target);
|
2011-02-12 14:29:09 +08:00
|
|
|
} else {
|
|
|
|
/* device exists, attempt to modify it */
|
2013-05-02 10:56:31 +01:00
|
|
|
ret = xend_op(conn, minidef->name, "op", "device_configure",
|
2011-02-12 14:29:09 +08:00
|
|
|
"config", sexpr, "dev", ref, NULL);
|
|
|
|
}
|
2007-09-10 11:47:17 +00:00
|
|
|
}
|
2008-07-25 13:17:27 +00:00
|
|
|
|
2014-03-25 07:54:51 +01:00
|
|
|
cleanup:
|
2008-05-29 19:20:22 +00:00
|
|
|
VIR_FREE(sexpr);
|
2008-07-25 13:17:27 +00:00
|
|
|
virDomainDefFree(def);
|
|
|
|
virDomainDeviceDefFree(dev);
|
2011-02-12 14:29:09 +08:00
|
|
|
VIR_FREE(target);
|
2006-11-16 18:11:28 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-03-22 13:13:53 +00:00
|
|
|
/**
|
|
|
|
* xenDaemonUpdateDeviceFlags:
|
2013-05-02 10:56:31 +01:00
|
|
|
* @conn: the connection object
|
|
|
|
* @minidef: domain configuration
|
2010-03-22 13:13:53 +00:00
|
|
|
* @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 S-expression.
|
|
|
|
*
|
|
|
|
* Returns 0 in case of success, -1 in case of failure.
|
|
|
|
*/
|
2011-07-21 15:11:32 -06:00
|
|
|
int
|
2013-05-02 10:56:31 +01:00
|
|
|
xenDaemonUpdateDeviceFlags(virConnectPtr conn,
|
|
|
|
virDomainDefPtr minidef,
|
2013-01-29 09:53:13 -05:00
|
|
|
const char *xml,
|
2010-03-22 13:13:53 +00:00
|
|
|
unsigned int flags)
|
|
|
|
{
|
2013-05-02 10:56:31 +01:00
|
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
2010-03-22 13:13:53 +00:00
|
|
|
char *sexpr = NULL;
|
|
|
|
int ret = -1;
|
|
|
|
virDomainDeviceDefPtr dev = NULL;
|
|
|
|
virDomainDefPtr def = NULL;
|
|
|
|
virBuffer buf = VIR_BUFFER_INITIALIZER;
|
|
|
|
char class[8], ref[80];
|
|
|
|
|
xen: reject unknown flags
Also fix a logic bug in xenXMDomain{Attach,Detach}DeviceFlags,
where (flags & VIR_DOMAIN_DEVICE_MODIFY_CURRENT) is always false.
* src/xen/xen_driver.c (xenUnifiedDomainXMLFromNative)
(xenUnifiedDomainXMLToNative, xenUnifiedDomainBlockPeek): Reject
unknown flags.
* src/xen/xen_hypervisor.c (xenHypervisorOpen)
(xenHypervisorGetDomainState): Likewise.
* src/xen/xen_inotify.c (xenInotifyOpen): Likewise.
* src/xen/xs_internal.c (xenStoreOpen, xenStoreDomainGetState)
(xenStoreDomainReboot): Likewise.
* src/xen/xend_internal.c (xenDaemonOpen, xenDaemonDomainReboot)
(xenDaemonDomainCoreDump, xenDaemonDomainGetState)
(xenDaemonDomainMigratePrepare, xenDaemonDomainSetVcpusFlags,
xenDaemonDomainGetVcpusFlags, xenDaemonAttachDeviceFlags,
xenDaemonDetachDeviceFlags): Likewise.
(xenDaemonDomainGetXMLDesc): Prefer unsigned flags.
* src/xen/xend_internal.h (xenDaemonDomainGetXMLDesc): Likewise.
* src/xen/xm_internal.h (xenXMDomainGetXMLDesc): Likewise.
* src/xen/xm_internal.c (xenXMDomainGetXMLDesc): Likewise.
(xenXMOpen, xenXMDomainGetState, xenXMDomainSetVcpusFlags)
(xenXMDomainGetVcpusFlags): Reject unknown flags.
(xenXMDomainAttachDeviceFlags, xenXMDomainDetachDeviceFlags):
Likewise, and avoid always-false conditional.
* src/xen/xen_driver.h (XEN_MIGRATION_FLAGS): New define.
2011-07-06 17:15:33 -06:00
|
|
|
virCheckFlags(VIR_DOMAIN_DEVICE_MODIFY_LIVE |
|
2010-04-16 14:04:31 +02:00
|
|
|
VIR_DOMAIN_DEVICE_MODIFY_CONFIG, -1);
|
|
|
|
|
2013-05-02 10:56:31 +01:00
|
|
|
if (minidef->id < 0) {
|
2010-10-01 16:22:29 +02:00
|
|
|
/* Cannot modify live config if domain is inactive */
|
|
|
|
if (flags & VIR_DOMAIN_DEVICE_MODIFY_LIVE) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID, "%s",
|
|
|
|
_("Cannot modify live config if domain is inactive"));
|
2010-10-01 16:22:29 +02:00
|
|
|
return -1;
|
|
|
|
}
|
2010-03-22 13:13:53 +00:00
|
|
|
} else {
|
|
|
|
/* Only live config can be changed if xendConfigVersion < 3 */
|
2012-01-31 18:08:09 +01:00
|
|
|
if (priv->xendConfigVersion < XEND_CONFIG_VERSION_3_0_4 &&
|
2010-07-28 17:12:16 -06:00
|
|
|
(flags != VIR_DOMAIN_DEVICE_MODIFY_CURRENT &&
|
2010-03-22 13:13:53 +00:00
|
|
|
flags != VIR_DOMAIN_DEVICE_MODIFY_LIVE)) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID, "%s",
|
|
|
|
_("Xend version does not support modifying "
|
|
|
|
"persistent config"));
|
2010-03-22 13:13:53 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
/* Xen only supports modifying both live and persistent config if
|
|
|
|
* xendConfigVersion >= 3
|
|
|
|
*/
|
2012-01-31 18:08:09 +01:00
|
|
|
if (priv->xendConfigVersion >= XEND_CONFIG_VERSION_3_0_4 &&
|
2010-10-01 16:31:04 +02:00
|
|
|
(flags != (VIR_DOMAIN_DEVICE_MODIFY_LIVE |
|
|
|
|
VIR_DOMAIN_DEVICE_MODIFY_CONFIG))) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID, "%s",
|
|
|
|
_("Xend only supports modifying both live and "
|
|
|
|
"persistent config"));
|
2010-03-22 13:13:53 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-02 10:56:31 +01:00
|
|
|
if (!(def = xenDaemonDomainFetch(conn,
|
|
|
|
minidef->id,
|
|
|
|
minidef->name,
|
2010-03-22 13:13:53 +00:00
|
|
|
NULL)))
|
|
|
|
goto cleanup;
|
|
|
|
|
2013-03-28 14:55:55 +01:00
|
|
|
if (!(dev = virDomainDeviceDefParse(xml, def, priv->caps, priv->xmlopt,
|
2014-11-18 16:44:00 +00:00
|
|
|
VIR_DOMAIN_DEF_PARSE_INACTIVE)))
|
2010-03-22 13:13:53 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
|
|
|
|
switch (dev->type) {
|
|
|
|
case VIR_DOMAIN_DEVICE_DISK:
|
2011-07-08 15:58:28 -06:00
|
|
|
if (xenFormatSxprDisk(dev->data.disk,
|
2011-02-21 14:40:12 +01:00
|
|
|
&buf,
|
2015-04-16 20:11:06 -04:00
|
|
|
def->os.type == VIR_DOMAIN_OSTYPE_HVM ? 1 : 0,
|
2011-02-21 14:40:12 +01:00
|
|
|
priv->xendConfigVersion, 1) < 0)
|
2010-03-22 13:13:53 +00:00
|
|
|
goto cleanup;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("unsupported device type"));
|
2010-03-22 13:13:53 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
sexpr = virBufferContentAndReset(&buf);
|
|
|
|
|
2013-05-02 10:56:31 +01:00
|
|
|
if (virDomainXMLDevID(conn, minidef, dev, class, ref, sizeof(ref))) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID, "%s",
|
|
|
|
_("requested device does not exist"));
|
2010-03-22 13:13:53 +00:00
|
|
|
goto cleanup;
|
|
|
|
} else {
|
|
|
|
/* device exists, attempt to modify it */
|
2013-05-02 10:56:31 +01:00
|
|
|
ret = xend_op(conn, minidef->name, "op", "device_configure",
|
2010-03-22 13:13:53 +00:00
|
|
|
"config", sexpr, "dev", ref, NULL);
|
|
|
|
}
|
|
|
|
|
2014-03-25 07:54:51 +01:00
|
|
|
cleanup:
|
2010-03-22 13:13:53 +00:00
|
|
|
VIR_FREE(sexpr);
|
|
|
|
virDomainDefFree(def);
|
|
|
|
virDomainDeviceDefFree(dev);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2006-11-16 18:11:28 +00:00
|
|
|
/**
|
2010-01-13 18:44:26 -07:00
|
|
|
* xenDaemonDetachDeviceFlags:
|
2013-05-02 10:56:31 +01:00
|
|
|
* @conn: the connection object
|
|
|
|
* @minidef: domain configuration
|
2006-11-16 18:11:28 +00:00
|
|
|
* @xml: pointer to XML description of device
|
2010-01-13 18:44:26 -07:00
|
|
|
* @flags: an OR'ed set of virDomainDeviceModifyFlags
|
2008-02-05 19:27:37 +00:00
|
|
|
*
|
2006-11-16 18:11:28 +00:00
|
|
|
* Destroy a virtual device attachment to backend.
|
|
|
|
*
|
|
|
|
* Returns 0 in case of success, -1 in case of failure.
|
|
|
|
*/
|
2013-04-30 17:43:07 +01:00
|
|
|
int
|
2013-05-02 10:56:31 +01:00
|
|
|
xenDaemonDetachDeviceFlags(virConnectPtr conn,
|
|
|
|
virDomainDefPtr minidef,
|
2013-01-29 09:53:13 -05:00
|
|
|
const char *xml,
|
2010-01-13 18:44:26 -07:00
|
|
|
unsigned int flags)
|
2006-11-16 18:11:28 +00:00
|
|
|
{
|
2013-05-02 10:56:31 +01:00
|
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
2006-11-16 18:11:28 +00:00
|
|
|
char class[8], ref[80];
|
2008-07-25 13:17:27 +00:00
|
|
|
virDomainDeviceDefPtr dev = NULL;
|
|
|
|
virDomainDefPtr def = NULL;
|
|
|
|
int ret = -1;
|
2009-12-22 10:53:51 +01:00
|
|
|
char *xendev = NULL;
|
|
|
|
virBuffer buf = VIR_BUFFER_INITIALIZER;
|
2006-11-16 18:11:28 +00:00
|
|
|
|
xen: reject unknown flags
Also fix a logic bug in xenXMDomain{Attach,Detach}DeviceFlags,
where (flags & VIR_DOMAIN_DEVICE_MODIFY_CURRENT) is always false.
* src/xen/xen_driver.c (xenUnifiedDomainXMLFromNative)
(xenUnifiedDomainXMLToNative, xenUnifiedDomainBlockPeek): Reject
unknown flags.
* src/xen/xen_hypervisor.c (xenHypervisorOpen)
(xenHypervisorGetDomainState): Likewise.
* src/xen/xen_inotify.c (xenInotifyOpen): Likewise.
* src/xen/xs_internal.c (xenStoreOpen, xenStoreDomainGetState)
(xenStoreDomainReboot): Likewise.
* src/xen/xend_internal.c (xenDaemonOpen, xenDaemonDomainReboot)
(xenDaemonDomainCoreDump, xenDaemonDomainGetState)
(xenDaemonDomainMigratePrepare, xenDaemonDomainSetVcpusFlags,
xenDaemonDomainGetVcpusFlags, xenDaemonAttachDeviceFlags,
xenDaemonDetachDeviceFlags): Likewise.
(xenDaemonDomainGetXMLDesc): Prefer unsigned flags.
* src/xen/xend_internal.h (xenDaemonDomainGetXMLDesc): Likewise.
* src/xen/xm_internal.h (xenXMDomainGetXMLDesc): Likewise.
* src/xen/xm_internal.c (xenXMDomainGetXMLDesc): Likewise.
(xenXMOpen, xenXMDomainGetState, xenXMDomainSetVcpusFlags)
(xenXMDomainGetVcpusFlags): Reject unknown flags.
(xenXMDomainAttachDeviceFlags, xenXMDomainDetachDeviceFlags):
Likewise, and avoid always-false conditional.
* src/xen/xen_driver.h (XEN_MIGRATION_FLAGS): New define.
2011-07-06 17:15:33 -06:00
|
|
|
virCheckFlags(VIR_DOMAIN_AFFECT_LIVE | VIR_DOMAIN_AFFECT_CONFIG, -1);
|
|
|
|
|
2013-05-02 10:56:31 +01:00
|
|
|
if (minidef->id < 0) {
|
2010-10-01 16:22:29 +02:00
|
|
|
/* Cannot modify live config if domain is inactive */
|
|
|
|
if (flags & VIR_DOMAIN_DEVICE_MODIFY_LIVE) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID, "%s",
|
|
|
|
_("Cannot modify live config if domain is inactive"));
|
2010-10-01 16:22:29 +02:00
|
|
|
return -1;
|
|
|
|
}
|
2010-01-13 18:44:26 -07:00
|
|
|
} else {
|
|
|
|
/* Only live config can be changed if xendConfigVersion < 3 */
|
2012-01-31 18:08:09 +01:00
|
|
|
if (priv->xendConfigVersion < XEND_CONFIG_VERSION_3_0_4 &&
|
2010-07-28 17:12:16 -06:00
|
|
|
(flags != VIR_DOMAIN_DEVICE_MODIFY_CURRENT &&
|
2010-01-13 18:44:26 -07:00
|
|
|
flags != VIR_DOMAIN_DEVICE_MODIFY_LIVE)) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID, "%s",
|
|
|
|
_("Xend version does not support modifying "
|
|
|
|
"persistent config"));
|
2010-01-13 18:44:26 -07:00
|
|
|
return -1;
|
|
|
|
}
|
2010-03-22 13:13:53 +00:00
|
|
|
/* Xen only supports modifying both live and persistent config if
|
2010-01-13 18:44:26 -07:00
|
|
|
* xendConfigVersion >= 3
|
|
|
|
*/
|
2012-01-31 18:08:09 +01:00
|
|
|
if (priv->xendConfigVersion >= XEND_CONFIG_VERSION_3_0_4 &&
|
2010-10-01 16:31:04 +02:00
|
|
|
(flags != (VIR_DOMAIN_DEVICE_MODIFY_LIVE |
|
|
|
|
VIR_DOMAIN_DEVICE_MODIFY_CONFIG))) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID, "%s",
|
|
|
|
_("Xend only supports modifying both live and "
|
|
|
|
"persistent config"));
|
2010-01-13 18:44:26 -07:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
2008-07-25 13:17:27 +00:00
|
|
|
|
2013-05-02 10:56:31 +01:00
|
|
|
if (!(def = xenDaemonDomainFetch(conn,
|
|
|
|
minidef->id,
|
|
|
|
minidef->name,
|
2008-07-25 13:17:27 +00:00
|
|
|
NULL)))
|
|
|
|
goto cleanup;
|
|
|
|
|
2013-03-28 14:55:55 +01:00
|
|
|
if (!(dev = virDomainDeviceDefParse(xml, def, priv->caps, priv->xmlopt,
|
2014-11-18 16:44:00 +00:00
|
|
|
VIR_DOMAIN_DEF_PARSE_INACTIVE)))
|
2008-07-25 13:17:27 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2013-05-02 10:56:31 +01:00
|
|
|
if (virDomainXMLDevID(conn, minidef, dev, class, ref, sizeof(ref)))
|
2008-07-25 13:17:27 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2009-12-22 10:53:51 +01:00
|
|
|
if (dev->type == VIR_DOMAIN_DEVICE_HOSTDEV) {
|
|
|
|
if (dev->data.hostdev->mode == VIR_DOMAIN_HOSTDEV_MODE_SUBSYS &&
|
|
|
|
dev->data.hostdev->source.subsys.type == VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_PCI) {
|
2011-02-21 14:40:12 +01:00
|
|
|
if (xenFormatSxprOnePCI(dev->data.hostdev, &buf, 1) < 0)
|
2009-12-22 10:53:51 +01:00
|
|
|
goto cleanup;
|
|
|
|
} else {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("unsupported device type"));
|
2009-12-22 10:53:51 +01:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
xendev = virBufferContentAndReset(&buf);
|
2013-05-02 10:56:31 +01:00
|
|
|
ret = xend_op(conn, minidef->name, "op", "device_configure",
|
2009-12-22 10:53:51 +01:00
|
|
|
"config", xendev, "dev", ref, NULL);
|
|
|
|
VIR_FREE(xendev);
|
2014-09-03 13:04:59 -06:00
|
|
|
} else {
|
2013-05-02 10:56:31 +01:00
|
|
|
ret = xend_op(conn, minidef->name, "op", "device_destroy",
|
2009-12-22 10:53:51 +01:00
|
|
|
"type", class, "dev", ref, "force", "0", "rm_cfg", "1",
|
|
|
|
NULL);
|
|
|
|
}
|
2008-07-25 13:17:27 +00:00
|
|
|
|
2014-03-25 07:54:51 +01:00
|
|
|
cleanup:
|
2008-07-25 13:17:27 +00:00
|
|
|
virDomainDefFree(def);
|
|
|
|
virDomainDeviceDefFree(dev);
|
|
|
|
|
|
|
|
return ret;
|
2006-11-16 18:11:28 +00:00
|
|
|
}
|
2006-12-14 01:56:14 +00:00
|
|
|
|
2008-05-09 08:17:18 +00:00
|
|
|
int
|
2013-05-02 11:05:55 +01:00
|
|
|
xenDaemonDomainGetAutostart(virConnectPtr conn,
|
|
|
|
virDomainDefPtr def,
|
|
|
|
int *autostart)
|
2008-05-09 08:17:18 +00:00
|
|
|
{
|
|
|
|
struct sexpr *root;
|
|
|
|
const char *tmp;
|
|
|
|
|
2013-05-02 11:05:55 +01:00
|
|
|
root = sexpr_get(conn, "/xend/domain/%s?detail=1", def->name);
|
2008-05-09 08:17:18 +00:00
|
|
|
if (root == NULL) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_XEN_CALL,
|
|
|
|
"%s", _("xenDaemonGetAutostart failed to find this domain"));
|
2012-03-22 12:33:35 +01:00
|
|
|
return -1;
|
2008-05-09 08:17:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
*autostart = 0;
|
|
|
|
|
|
|
|
tmp = sexpr_node(root, "domain/on_xend_start");
|
2014-11-13 15:29:45 +01:00
|
|
|
if (tmp && STREQ(tmp, "start"))
|
2008-05-09 08:17:18 +00:00
|
|
|
*autostart = 1;
|
|
|
|
|
|
|
|
sexpr_free(root);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2013-05-02 11:05:55 +01:00
|
|
|
xenDaemonDomainSetAutostart(virConnectPtr conn,
|
|
|
|
virDomainDefPtr def,
|
|
|
|
int autostart)
|
2008-05-09 08:17:18 +00:00
|
|
|
{
|
|
|
|
struct sexpr *root, *autonode;
|
2011-04-03 11:21:33 +02:00
|
|
|
virBuffer buffer = VIR_BUFFER_INITIALIZER;
|
|
|
|
char *content = NULL;
|
2008-05-09 08:17:18 +00:00
|
|
|
int ret = -1;
|
|
|
|
|
2013-05-02 11:05:55 +01:00
|
|
|
root = sexpr_get(conn, "/xend/domain/%s?detail=1", def->name);
|
2008-05-09 08:17:18 +00:00
|
|
|
if (root == NULL) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_XEN_CALL,
|
|
|
|
"%s", _("xenDaemonSetAutostart failed to find this domain"));
|
2012-03-22 12:33:35 +01:00
|
|
|
return -1;
|
2008-05-09 08:17:18 +00:00
|
|
|
}
|
|
|
|
|
2010-02-17 22:14:25 +01:00
|
|
|
autonode = sexpr_lookup(root, "domain/on_xend_start");
|
|
|
|
if (autonode) {
|
|
|
|
const char *val = (autonode->u.s.car->kind == SEXPR_VALUE
|
|
|
|
? autonode->u.s.car->u.value : NULL);
|
2015-10-20 21:45:12 +05:30
|
|
|
if (!val || (STRNEQ(val, "ignore") && STRNEQ(val, "start"))) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
"%s", _("unexpected value from on_xend_start"));
|
2008-05-09 08:17:18 +00:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2011-01-28 22:38:06 +01:00
|
|
|
/* Change the autostart value in place, then define the new sexpr */
|
2008-05-29 19:20:22 +00:00
|
|
|
VIR_FREE(autonode->u.s.car->u.value);
|
2013-05-03 14:51:25 +02:00
|
|
|
if (VIR_STRDUP(autonode->u.s.car->u.value,
|
|
|
|
autostart ? "start" : "ignore") < 0)
|
2008-05-09 08:17:18 +00:00
|
|
|
goto error;
|
|
|
|
|
2011-04-03 11:21:33 +02:00
|
|
|
if (sexpr2string(root, &buffer) < 0) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
"%s", _("sexpr2string failed"));
|
2008-05-09 08:17:18 +00:00
|
|
|
goto error;
|
|
|
|
}
|
2011-04-03 11:21:33 +02:00
|
|
|
|
2014-06-27 10:40:15 +02:00
|
|
|
if (virBufferCheckError(&buffer) < 0)
|
2011-04-03 11:21:33 +02:00
|
|
|
goto error;
|
|
|
|
|
|
|
|
content = virBufferContentAndReset(&buffer);
|
|
|
|
|
2013-05-02 11:05:55 +01:00
|
|
|
if (xend_op(conn, "", "op", "new", "config", content, NULL) != 0) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_XEN_CALL,
|
|
|
|
"%s", _("Failed to redefine sexpr"));
|
2008-05-09 08:17:18 +00:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
} else {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
"%s", _("on_xend_start not present in sexpr"));
|
2008-05-09 08:17:18 +00:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
2014-03-25 07:54:51 +01:00
|
|
|
error:
|
2011-04-03 11:21:33 +02:00
|
|
|
virBufferFreeAndReset(&buffer);
|
|
|
|
VIR_FREE(content);
|
2008-05-09 08:17:18 +00:00
|
|
|
sexpr_free(root);
|
|
|
|
return ret;
|
|
|
|
}
|
2006-12-14 01:56:14 +00:00
|
|
|
|
2007-08-21 09:31:12 +00:00
|
|
|
int
|
2013-04-26 17:39:11 +01:00
|
|
|
xenDaemonDomainMigratePrepare(virConnectPtr dconn ATTRIBUTE_UNUSED,
|
2012-10-17 10:23:12 +01:00
|
|
|
char **cookie ATTRIBUTE_UNUSED,
|
|
|
|
int *cookielen ATTRIBUTE_UNUSED,
|
|
|
|
const char *uri_in,
|
|
|
|
char **uri_out,
|
|
|
|
unsigned long flags,
|
|
|
|
const char *dname ATTRIBUTE_UNUSED,
|
|
|
|
unsigned long resource ATTRIBUTE_UNUSED)
|
2007-08-21 09:31:12 +00:00
|
|
|
{
|
xen: reject unknown flags
Also fix a logic bug in xenXMDomain{Attach,Detach}DeviceFlags,
where (flags & VIR_DOMAIN_DEVICE_MODIFY_CURRENT) is always false.
* src/xen/xen_driver.c (xenUnifiedDomainXMLFromNative)
(xenUnifiedDomainXMLToNative, xenUnifiedDomainBlockPeek): Reject
unknown flags.
* src/xen/xen_hypervisor.c (xenHypervisorOpen)
(xenHypervisorGetDomainState): Likewise.
* src/xen/xen_inotify.c (xenInotifyOpen): Likewise.
* src/xen/xs_internal.c (xenStoreOpen, xenStoreDomainGetState)
(xenStoreDomainReboot): Likewise.
* src/xen/xend_internal.c (xenDaemonOpen, xenDaemonDomainReboot)
(xenDaemonDomainCoreDump, xenDaemonDomainGetState)
(xenDaemonDomainMigratePrepare, xenDaemonDomainSetVcpusFlags,
xenDaemonDomainGetVcpusFlags, xenDaemonAttachDeviceFlags,
xenDaemonDetachDeviceFlags): Likewise.
(xenDaemonDomainGetXMLDesc): Prefer unsigned flags.
* src/xen/xend_internal.h (xenDaemonDomainGetXMLDesc): Likewise.
* src/xen/xm_internal.h (xenXMDomainGetXMLDesc): Likewise.
* src/xen/xm_internal.c (xenXMDomainGetXMLDesc): Likewise.
(xenXMOpen, xenXMDomainGetState, xenXMDomainSetVcpusFlags)
(xenXMDomainGetVcpusFlags): Reject unknown flags.
(xenXMDomainAttachDeviceFlags, xenXMDomainDetachDeviceFlags):
Likewise, and avoid always-false conditional.
* src/xen/xen_driver.h (XEN_MIGRATION_FLAGS): New define.
2011-07-06 17:15:33 -06:00
|
|
|
virCheckFlags(XEN_MIGRATION_FLAGS, -1);
|
|
|
|
|
2007-08-21 09:31:12 +00:00
|
|
|
/* If uri_in is NULL, get the current hostname as a best guess
|
|
|
|
* of how the source host should connect to us. Note that caller
|
|
|
|
* deallocates this string.
|
|
|
|
*/
|
|
|
|
if (uri_in == NULL) {
|
2013-04-26 17:39:11 +01:00
|
|
|
*uri_out = virGetHostname();
|
2009-10-23 13:01:22 -04:00
|
|
|
if (*uri_out == NULL)
|
2007-08-21 09:31:12 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2013-05-01 14:15:10 +01:00
|
|
|
xenDaemonDomainMigratePerform(virConnectPtr conn,
|
|
|
|
virDomainDefPtr def,
|
2012-10-17 10:23:12 +01:00
|
|
|
const char *cookie ATTRIBUTE_UNUSED,
|
|
|
|
int cookielen ATTRIBUTE_UNUSED,
|
|
|
|
const char *uri,
|
|
|
|
unsigned long flags,
|
|
|
|
const char *dname,
|
|
|
|
unsigned long bandwidth)
|
2007-08-21 09:31:12 +00:00
|
|
|
{
|
|
|
|
/* Upper layers have already checked domain. */
|
|
|
|
/* NB: Passing port=0 to xend means it ignores
|
|
|
|
* the port. However this is somewhat specific to
|
|
|
|
* the internals of the xend Python code. (XXX).
|
|
|
|
*/
|
|
|
|
char port[16] = "0";
|
|
|
|
char live[2] = "0";
|
|
|
|
int ret;
|
|
|
|
char *p, *hostname = NULL;
|
|
|
|
|
2009-11-12 16:04:43 +01:00
|
|
|
int undefined_source = 0;
|
|
|
|
|
xen: reject unknown flags
Also fix a logic bug in xenXMDomain{Attach,Detach}DeviceFlags,
where (flags & VIR_DOMAIN_DEVICE_MODIFY_CURRENT) is always false.
* src/xen/xen_driver.c (xenUnifiedDomainXMLFromNative)
(xenUnifiedDomainXMLToNative, xenUnifiedDomainBlockPeek): Reject
unknown flags.
* src/xen/xen_hypervisor.c (xenHypervisorOpen)
(xenHypervisorGetDomainState): Likewise.
* src/xen/xen_inotify.c (xenInotifyOpen): Likewise.
* src/xen/xs_internal.c (xenStoreOpen, xenStoreDomainGetState)
(xenStoreDomainReboot): Likewise.
* src/xen/xend_internal.c (xenDaemonOpen, xenDaemonDomainReboot)
(xenDaemonDomainCoreDump, xenDaemonDomainGetState)
(xenDaemonDomainMigratePrepare, xenDaemonDomainSetVcpusFlags,
xenDaemonDomainGetVcpusFlags, xenDaemonAttachDeviceFlags,
xenDaemonDetachDeviceFlags): Likewise.
(xenDaemonDomainGetXMLDesc): Prefer unsigned flags.
* src/xen/xend_internal.h (xenDaemonDomainGetXMLDesc): Likewise.
* src/xen/xm_internal.h (xenXMDomainGetXMLDesc): Likewise.
* src/xen/xm_internal.c (xenXMDomainGetXMLDesc): Likewise.
(xenXMOpen, xenXMDomainGetState, xenXMDomainSetVcpusFlags)
(xenXMDomainGetVcpusFlags): Reject unknown flags.
(xenXMDomainAttachDeviceFlags, xenXMDomainDetachDeviceFlags):
Likewise, and avoid always-false conditional.
* src/xen/xen_driver.h (XEN_MIGRATION_FLAGS): New define.
2011-07-06 17:15:33 -06:00
|
|
|
virCheckFlags(XEN_MIGRATION_FLAGS, -1);
|
|
|
|
|
2007-08-21 09:31:12 +00:00
|
|
|
/* Xen doesn't support renaming domains during migration. */
|
|
|
|
if (dname) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID,
|
|
|
|
"%s", _("xenDaemonDomainMigrate: Xen does not support"
|
|
|
|
" renaming domains during migration"));
|
2007-08-21 09:31:12 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Xen (at least up to 3.1.0) takes a resource parameter but
|
|
|
|
* ignores it.
|
|
|
|
*/
|
|
|
|
if (bandwidth) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID,
|
|
|
|
"%s", _("xenDaemonDomainMigrate: Xen does not support"
|
|
|
|
" bandwidth limits during migration"));
|
2007-08-21 09:31:12 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2009-11-12 16:04:43 +01:00
|
|
|
/*
|
|
|
|
* Check the flags.
|
|
|
|
*/
|
2007-08-21 09:31:12 +00:00
|
|
|
if ((flags & VIR_MIGRATE_LIVE)) {
|
2012-10-17 10:23:12 +01:00
|
|
|
strcpy(live, "1");
|
2007-08-21 09:31:12 +00:00
|
|
|
flags &= ~VIR_MIGRATE_LIVE;
|
|
|
|
}
|
2009-11-12 16:04:43 +01:00
|
|
|
|
|
|
|
/* Undefine the VM on the source host after migration? */
|
|
|
|
if (flags & VIR_MIGRATE_UNDEFINE_SOURCE) {
|
|
|
|
undefined_source = 1;
|
|
|
|
flags &= ~VIR_MIGRATE_UNDEFINE_SOURCE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Ignore the persist_dest flag here */
|
|
|
|
if (flags & VIR_MIGRATE_PERSIST_DEST)
|
|
|
|
flags &= ~VIR_MIGRATE_PERSIST_DEST;
|
|
|
|
|
2009-12-11 10:06:05 +01:00
|
|
|
/* This is buggy in Xend, but could be supported in principle. Give
|
|
|
|
* a nice error message.
|
|
|
|
*/
|
|
|
|
if (flags & VIR_MIGRATE_PAUSED) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID,
|
|
|
|
"%s", _("xenDaemonDomainMigrate: xend cannot migrate paused domains"));
|
2009-12-11 10:06:05 +01:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2009-09-17 18:10:04 +01:00
|
|
|
/* XXX we could easily do tunnelled & peer2peer migration too
|
|
|
|
if we want to. support these... */
|
2007-08-21 09:31:12 +00:00
|
|
|
if (flags != 0) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID,
|
|
|
|
"%s", _("xenDaemonDomainMigrate: unsupported flag"));
|
2007-08-21 09:31:12 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set hostname and port.
|
|
|
|
*
|
|
|
|
* URI is non-NULL (guaranteed by caller). We expect either
|
|
|
|
* "hostname", "hostname:port" or "xenmigr://hostname[:port]/".
|
|
|
|
*/
|
2012-10-17 10:23:12 +01:00
|
|
|
if (strstr(uri, "//")) { /* Full URI. */
|
Centralize error reporting for URI parsing/formatting problems
Move error reporting out of the callers, into virURIParse
and virURIFormat, to get consistency.
* include/libvirt/virterror.h, src/util/virterror.c: Add VIR_FROM_URI
* src/util/viruri.c, src/util/viruri.h: Add error reporting
* src/esx/esx_driver.c, src/libvirt.c, src/libxl/libxl_driver.c,
src/lxc/lxc_driver.c, src/openvz/openvz_driver.c,
src/qemu/qemu_driver.c, src/qemu/qemu_migration.c,
src/remote/remote_driver.c, src/uml/uml_driver.c,
src/vbox/vbox_tmpl.c, src/vmx/vmx.c, src/xen/xen_driver.c,
src/xen/xend_internal.c, tests/viruritest.c: Remove error
reporting
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2012-03-20 12:16:54 +00:00
|
|
|
virURIPtr uriptr;
|
2012-10-17 10:23:12 +01:00
|
|
|
if (!(uriptr = virURIParse(uri)))
|
2007-08-21 09:31:12 +00:00
|
|
|
return -1;
|
Centralize error reporting for URI parsing/formatting problems
Move error reporting out of the callers, into virURIParse
and virURIFormat, to get consistency.
* include/libvirt/virterror.h, src/util/virterror.c: Add VIR_FROM_URI
* src/util/viruri.c, src/util/viruri.h: Add error reporting
* src/esx/esx_driver.c, src/libvirt.c, src/libxl/libxl_driver.c,
src/lxc/lxc_driver.c, src/openvz/openvz_driver.c,
src/qemu/qemu_driver.c, src/qemu/qemu_migration.c,
src/remote/remote_driver.c, src/uml/uml_driver.c,
src/vbox/vbox_tmpl.c, src/vmx/vmx.c, src/xen/xen_driver.c,
src/xen/xend_internal.c, tests/viruritest.c: Remove error
reporting
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2012-03-20 12:16:54 +00:00
|
|
|
|
2012-10-17 10:23:12 +01:00
|
|
|
if (uriptr->scheme && STRCASENEQ(uriptr->scheme, "xenmigr")) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
|
|
"%s", _("xenDaemonDomainMigrate: only xenmigr://"
|
|
|
|
" migrations are supported by Xen"));
|
2012-10-17 10:23:12 +01:00
|
|
|
virURIFree(uriptr);
|
2007-08-21 09:31:12 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (!uriptr->server) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
|
|
"%s", _("xenDaemonDomainMigrate: a hostname must be"
|
|
|
|
" specified in the URI"));
|
2012-10-17 10:23:12 +01:00
|
|
|
virURIFree(uriptr);
|
2007-08-21 09:31:12 +00:00
|
|
|
return -1;
|
|
|
|
}
|
2013-05-03 14:51:25 +02:00
|
|
|
if (VIR_STRDUP(hostname, uriptr->server) < 0) {
|
2012-10-17 10:23:12 +01:00
|
|
|
virURIFree(uriptr);
|
2007-08-21 09:31:12 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (uriptr->port)
|
2012-10-17 10:23:12 +01:00
|
|
|
snprintf(port, sizeof(port), "%d", uriptr->port);
|
|
|
|
virURIFree(uriptr);
|
2014-09-03 13:04:59 -06:00
|
|
|
} else if ((p = strrchr(uri, ':')) != NULL) { /* "hostname:port" */
|
2007-08-21 09:31:12 +00:00
|
|
|
int port_nr, n;
|
|
|
|
|
2010-03-30 16:52:42 +02:00
|
|
|
if (virStrToLong_i(p+1, NULL, 10, &port_nr) < 0) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
|
|
"%s", _("xenDaemonDomainMigrate: invalid port number"));
|
2007-08-21 09:31:12 +00:00
|
|
|
return -1;
|
|
|
|
}
|
2012-10-17 10:23:12 +01:00
|
|
|
snprintf(port, sizeof(port), "%d", port_nr);
|
2007-08-21 09:31:12 +00:00
|
|
|
|
|
|
|
/* Get the hostname. */
|
|
|
|
n = p - uri; /* n = Length of hostname in bytes. */
|
2013-05-03 14:51:25 +02:00
|
|
|
if (VIR_STRDUP(hostname, uri) < 0)
|
2007-08-21 09:31:12 +00:00
|
|
|
return -1;
|
|
|
|
hostname[n] = '\0';
|
2014-09-03 13:04:59 -06:00
|
|
|
} else { /* "hostname" (or IP address) */
|
2013-05-03 14:51:25 +02:00
|
|
|
if (VIR_STRDUP(hostname, uri) < 0)
|
2007-08-21 09:31:12 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2011-02-16 16:37:57 -07:00
|
|
|
VIR_DEBUG("hostname = %s, port = %s", hostname, port);
|
2007-08-21 09:31:12 +00:00
|
|
|
|
2010-01-12 11:18:34 -07:00
|
|
|
/* Make the call.
|
|
|
|
* NB: xend will fail the operation if any parameters are
|
|
|
|
* missing but happily accept unknown parameters. This works
|
|
|
|
* to our advantage since all parameters supported and required
|
|
|
|
* by current xend can be included without breaking older xend.
|
|
|
|
*/
|
2013-05-01 14:15:10 +01:00
|
|
|
ret = xend_op(conn, def->name,
|
2012-10-17 10:23:12 +01:00
|
|
|
"op", "migrate",
|
|
|
|
"destination", hostname,
|
|
|
|
"live", live,
|
|
|
|
"port", port,
|
|
|
|
"node", "-1", /* xen-unstable c/s 17753 */
|
|
|
|
"ssl", "0", /* xen-unstable c/s 17709 */
|
|
|
|
"change_home_server", "0", /* xen-unstable c/s 20326 */
|
|
|
|
"resource", "0", /* removed by xen-unstable c/s 17553 */
|
|
|
|
NULL);
|
|
|
|
VIR_FREE(hostname);
|
2007-08-21 09:31:12 +00:00
|
|
|
|
2009-11-12 16:04:43 +01:00
|
|
|
if (ret == 0 && undefined_source)
|
2013-05-01 14:15:10 +01:00
|
|
|
xenDaemonDomainUndefine(conn, def);
|
2009-11-12 16:04:43 +01:00
|
|
|
|
2011-05-09 17:24:09 +08:00
|
|
|
VIR_DEBUG("migration done");
|
2007-08-21 09:31:12 +00:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2013-05-01 14:15:10 +01:00
|
|
|
int
|
|
|
|
xenDaemonDomainDefineXML(virConnectPtr conn, virDomainDefPtr def)
|
2013-01-29 09:53:13 -05:00
|
|
|
{
|
2013-05-01 14:15:10 +01:00
|
|
|
int ret = -1;
|
2006-12-14 01:56:14 +00:00
|
|
|
char *sexpr;
|
2013-01-29 09:53:13 -05:00
|
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
2006-12-14 01:56:14 +00:00
|
|
|
|
2011-02-21 14:40:12 +01:00
|
|
|
if (!(sexpr = xenFormatSxpr(conn, def, priv->xendConfigVersion))) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_XML_ERROR,
|
|
|
|
"%s", _("failed to build sexpr"));
|
2013-05-01 14:15:10 +01:00
|
|
|
goto cleanup;
|
2008-07-25 13:17:27 +00:00
|
|
|
}
|
|
|
|
|
2006-12-14 01:56:14 +00:00
|
|
|
ret = xend_op(conn, "", "op", "new", "config", sexpr, NULL);
|
2008-05-29 19:20:22 +00:00
|
|
|
VIR_FREE(sexpr);
|
2006-12-14 01:56:14 +00:00
|
|
|
if (ret != 0) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_XEN_CALL,
|
|
|
|
_("Failed to create inactive domain %s"), def->name);
|
2013-05-01 14:15:10 +01:00
|
|
|
goto cleanup;
|
2006-12-14 01:56:14 +00:00
|
|
|
}
|
|
|
|
|
2013-05-01 14:15:10 +01:00
|
|
|
ret = 0;
|
2008-07-25 13:17:27 +00:00
|
|
|
|
2014-03-25 07:54:51 +01:00
|
|
|
cleanup:
|
2013-05-01 14:15:10 +01:00
|
|
|
return ret;
|
2006-12-14 01:56:14 +00:00
|
|
|
}
|
2013-05-01 14:15:10 +01:00
|
|
|
|
2013-01-29 09:53:13 -05:00
|
|
|
int
|
2013-05-01 14:03:26 +01:00
|
|
|
xenDaemonDomainCreate(virConnectPtr conn,
|
|
|
|
virDomainDefPtr def)
|
2007-04-04 14:19:49 +00:00
|
|
|
{
|
2009-09-04 14:10:44 +02:00
|
|
|
int ret;
|
2006-12-14 01:56:14 +00:00
|
|
|
|
2013-05-01 14:03:26 +01:00
|
|
|
ret = xend_op(conn, def->name, "op", "start", NULL);
|
2009-09-04 14:10:44 +02:00
|
|
|
|
2013-04-30 17:33:48 +01:00
|
|
|
if (ret == 0) {
|
2013-05-01 14:03:26 +01:00
|
|
|
int id = xenDaemonDomainLookupByName_ids(conn, def->name,
|
|
|
|
def->uuid);
|
2013-04-30 17:33:48 +01:00
|
|
|
if (id > 0)
|
2013-05-01 14:03:26 +01:00
|
|
|
def->id = id;
|
2009-09-04 14:10:44 +02:00
|
|
|
}
|
2013-04-30 17:33:48 +01:00
|
|
|
|
2009-09-04 14:10:44 +02:00
|
|
|
return ret;
|
2006-12-14 01:56:14 +00:00
|
|
|
}
|
|
|
|
|
2013-01-29 09:53:13 -05:00
|
|
|
int
|
2013-05-01 14:15:10 +01:00
|
|
|
xenDaemonDomainUndefine(virConnectPtr conn, virDomainDefPtr def)
|
2007-04-04 14:19:49 +00:00
|
|
|
{
|
2013-05-01 14:15:10 +01:00
|
|
|
return xend_op(conn, def->name, "op", "delete", NULL);
|
2006-12-14 01:56:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* xenDaemonNumOfDomains:
|
|
|
|
* @conn: pointer to the hypervisor connection
|
|
|
|
*
|
|
|
|
* Provides the number of active domains.
|
|
|
|
*
|
|
|
|
* Returns the number of domain found or -1 in case of error
|
|
|
|
*/
|
2013-04-30 17:31:40 +01:00
|
|
|
int
|
2006-12-14 01:56:14 +00:00
|
|
|
xenDaemonNumOfDefinedDomains(virConnectPtr conn)
|
|
|
|
{
|
|
|
|
struct sexpr *root = NULL;
|
|
|
|
int ret = -1;
|
|
|
|
struct sexpr *_for_i, *node;
|
2007-01-18 18:38:09 +00:00
|
|
|
|
2006-12-14 01:56:14 +00:00
|
|
|
root = sexpr_get(conn, "/xend/domain?state=halted");
|
|
|
|
if (root == NULL)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
2013-01-22 09:40:56 -05:00
|
|
|
/* coverity[copy_paste_error] */
|
2007-09-29 18:16:26 +00:00
|
|
|
for (_for_i = root, node = root->u.s.car; _for_i->kind == SEXPR_CONS;
|
|
|
|
_for_i = _for_i->u.s.cdr, node = _for_i->u.s.car) {
|
2006-12-14 01:56:14 +00:00
|
|
|
if (node->kind != SEXPR_VALUE)
|
|
|
|
continue;
|
|
|
|
ret++;
|
|
|
|
}
|
|
|
|
|
2014-03-25 07:54:51 +01:00
|
|
|
error:
|
2008-01-29 17:41:07 +00:00
|
|
|
sexpr_free(root);
|
2012-03-22 12:33:35 +01:00
|
|
|
return ret;
|
2006-12-14 01:56:14 +00:00
|
|
|
}
|
|
|
|
|
2013-04-30 17:31:40 +01:00
|
|
|
int
|
2013-01-29 09:53:13 -05:00
|
|
|
xenDaemonListDefinedDomains(virConnectPtr conn,
|
|
|
|
char **const names,
|
|
|
|
int maxnames)
|
|
|
|
{
|
2006-12-14 01:56:14 +00:00
|
|
|
struct sexpr *root = NULL;
|
Convert 'int i' to 'size_t i' in src/{xen,xenapi,xenxs} 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;
|
2013-08-05 10:27:23 -06:00
|
|
|
int ret = 0;
|
2006-12-14 01:56:14 +00:00
|
|
|
struct sexpr *_for_i, *node;
|
2007-01-18 18:38:09 +00:00
|
|
|
|
2008-06-19 14:39:49 +00:00
|
|
|
if (maxnames == 0)
|
2012-03-22 12:33:35 +01:00
|
|
|
return 0;
|
2008-06-19 14:39:49 +00:00
|
|
|
|
2006-12-14 01:56:14 +00:00
|
|
|
root = sexpr_get(conn, "/xend/domain?state=halted");
|
|
|
|
if (root == NULL)
|
|
|
|
goto error;
|
|
|
|
|
2013-01-22 09:40:56 -05:00
|
|
|
/* coverity[copy_paste_error] */
|
2007-09-29 18:16:26 +00:00
|
|
|
for (_for_i = root, node = root->u.s.car; _for_i->kind == SEXPR_CONS;
|
|
|
|
_for_i = _for_i->u.s.cdr, node = _for_i->u.s.car) {
|
2006-12-14 01:56:14 +00:00
|
|
|
if (node->kind != SEXPR_VALUE)
|
|
|
|
continue;
|
|
|
|
|
2013-05-03 14:51:25 +02:00
|
|
|
if (VIR_STRDUP(names[ret++], node->u.value) < 0)
|
2009-11-08 22:08:54 +01:00
|
|
|
goto error;
|
|
|
|
|
2006-12-14 01:56:14 +00:00
|
|
|
if (ret >= maxnames)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2014-03-25 07:54:51 +01:00
|
|
|
cleanup:
|
2009-11-26 00:09:40 +01:00
|
|
|
sexpr_free(root);
|
2012-03-22 12:33:35 +01:00
|
|
|
return ret;
|
2009-11-26 00:09:40 +01:00
|
|
|
|
2014-03-25 07:54:51 +01:00
|
|
|
error:
|
2013-08-05 10:27:23 -06:00
|
|
|
for (i = 0; i < ret; ++i)
|
2009-11-08 22:08:54 +01:00
|
|
|
VIR_FREE(names[i]);
|
|
|
|
|
2013-08-05 10:27:23 -06:00
|
|
|
ret = -1;
|
2009-11-26 00:09:40 +01:00
|
|
|
goto cleanup;
|
2006-12-14 01:56:14 +00:00
|
|
|
}
|
|
|
|
|
2008-03-24 09:23:32 +00:00
|
|
|
/**
|
|
|
|
* xenDaemonGetSchedulerType:
|
2013-05-02 11:13:39 +01:00
|
|
|
* @conn: the hypervisor connection
|
2008-03-24 09:23:32 +00:00
|
|
|
* @nparams: give a number of scheduler parameters
|
|
|
|
*
|
|
|
|
* Get the scheduler type of Xen
|
|
|
|
*
|
|
|
|
* Returns a scheduler name (credit or sedf) which must be freed by the
|
|
|
|
* caller or NULL in case of failure
|
|
|
|
*/
|
2013-04-30 17:52:19 +01:00
|
|
|
char *
|
2013-05-02 11:13:39 +01:00
|
|
|
xenDaemonGetSchedulerType(virConnectPtr conn,
|
|
|
|
int *nparams)
|
2008-03-24 09:23:32 +00:00
|
|
|
{
|
2013-05-02 11:13:39 +01:00
|
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
2008-03-24 09:23:32 +00:00
|
|
|
struct sexpr *root;
|
|
|
|
const char *ret = NULL;
|
|
|
|
char *schedulertype = NULL;
|
|
|
|
|
|
|
|
/* Support only xendConfigVersion >=4 */
|
2012-01-31 18:08:09 +01:00
|
|
|
if (priv->xendConfigVersion < XEND_CONFIG_VERSION_3_1_0) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID,
|
|
|
|
"%s", _("unsupported in xendConfigVersion < 4"));
|
2008-03-24 09:23:32 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2013-05-02 11:13:39 +01:00
|
|
|
root = sexpr_get(conn, "/xend/node/");
|
2008-03-24 09:23:32 +00:00
|
|
|
if (root == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
/* get xen_scheduler from xend/node */
|
|
|
|
ret = sexpr_node(root, "node/xen_scheduler");
|
2014-08-20 13:00:30 +02:00
|
|
|
if (ret == NULL) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
"%s", _("node information incomplete, missing scheduler name"));
|
2008-03-24 09:23:32 +00:00
|
|
|
goto error;
|
|
|
|
}
|
2012-10-17 10:23:12 +01:00
|
|
|
if (STREQ(ret, "credit")) {
|
2013-05-03 14:51:25 +02:00
|
|
|
if (VIR_STRDUP(schedulertype, "credit") < 0)
|
2008-03-24 09:23:32 +00:00
|
|
|
goto error;
|
2011-05-18 09:55:32 +02:00
|
|
|
if (nparams)
|
|
|
|
*nparams = XEN_SCHED_CRED_NPARAM;
|
2012-10-17 10:23:12 +01:00
|
|
|
} else if (STREQ(ret, "sedf")) {
|
2013-05-03 14:51:25 +02:00
|
|
|
if (VIR_STRDUP(schedulertype, "sedf") < 0)
|
2008-03-24 09:23:32 +00:00
|
|
|
goto error;
|
2011-05-18 09:55:32 +02:00
|
|
|
if (nparams)
|
|
|
|
*nparams = XEN_SCHED_SEDF_NPARAM;
|
2008-03-24 09:23:32 +00:00
|
|
|
} else {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Unknown scheduler"));
|
2008-03-24 09:23:32 +00:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2014-03-25 07:54:51 +01:00
|
|
|
error:
|
2008-03-24 09:23:32 +00:00
|
|
|
sexpr_free(root);
|
|
|
|
return schedulertype;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* xenDaemonGetSchedulerParameters:
|
2013-05-02 11:13:39 +01:00
|
|
|
* @conn: the hypervisor connection
|
|
|
|
* @def: domain configuration
|
2008-03-24 09:23:32 +00:00
|
|
|
* @params: pointer to scheduler parameters
|
|
|
|
* This memory area must be allocated by the caller
|
|
|
|
* @nparams: a number of scheduler parameters which should be same as a
|
|
|
|
* given number from xenDaemonGetSchedulerType()
|
|
|
|
*
|
|
|
|
* Get the scheduler parameters
|
|
|
|
*
|
|
|
|
* Returns 0 or -1 in case of failure
|
|
|
|
*/
|
2013-04-30 17:52:19 +01:00
|
|
|
int
|
2013-05-02 11:13:39 +01:00
|
|
|
xenDaemonGetSchedulerParameters(virConnectPtr conn,
|
|
|
|
virDomainDefPtr def,
|
2013-01-29 09:53:13 -05:00
|
|
|
virTypedParameterPtr params,
|
|
|
|
int *nparams)
|
2008-03-24 09:23:32 +00:00
|
|
|
{
|
2013-05-02 11:13:39 +01:00
|
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
2008-03-24 09:23:32 +00:00
|
|
|
struct sexpr *root;
|
|
|
|
char *sched_type = NULL;
|
|
|
|
int sched_nparam = 0;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
/* Support only xendConfigVersion >=4 */
|
2012-01-31 18:08:09 +01:00
|
|
|
if (priv->xendConfigVersion < XEND_CONFIG_VERSION_3_1_0) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID,
|
|
|
|
"%s", _("unsupported in xendConfigVersion < 4"));
|
2012-03-22 12:33:35 +01:00
|
|
|
return -1;
|
2008-03-24 09:23:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* look up the information by domain name */
|
2013-05-02 11:13:39 +01:00
|
|
|
root = sexpr_get(conn, "/xend/domain/%s?detail=1", def->name);
|
2008-03-24 09:23:32 +00:00
|
|
|
if (root == NULL)
|
2012-03-22 12:33:35 +01:00
|
|
|
return -1;
|
2008-03-24 09:23:32 +00:00
|
|
|
|
|
|
|
/* get the scheduler type */
|
2013-05-02 11:13:39 +01:00
|
|
|
sched_type = xenDaemonGetSchedulerType(conn, &sched_nparam);
|
2008-03-24 09:23:32 +00:00
|
|
|
if (sched_type == NULL) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
"%s", _("Failed to get a scheduler name"));
|
2008-03-24 09:23:32 +00:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2014-08-20 13:00:30 +02:00
|
|
|
switch (sched_nparam) {
|
2008-03-24 09:23:32 +00:00
|
|
|
case XEN_SCHED_SEDF_NPARAM:
|
2011-05-18 10:52:57 +02:00
|
|
|
if (*nparams < XEN_SCHED_SEDF_NPARAM) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
|
|
"%s", _("Invalid parameter count"));
|
2011-05-18 10:52:57 +02:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2008-03-24 09:23:32 +00:00
|
|
|
/* TODO: Implement for Xen/SEDF */
|
|
|
|
TODO
|
|
|
|
goto error;
|
|
|
|
case XEN_SCHED_CRED_NPARAM:
|
|
|
|
/* get cpu_weight/cpu_cap from xend/domain */
|
|
|
|
if (sexpr_node(root, "domain/cpu_weight") == NULL) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
"%s", _("domain information incomplete, missing cpu_weight"));
|
2008-03-24 09:23:32 +00:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
if (sexpr_node(root, "domain/cpu_cap") == NULL) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
"%s", _("domain information incomplete, missing cpu_cap"));
|
2008-03-24 09:23:32 +00:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2011-11-01 14:38:42 -06:00
|
|
|
if (virStrcpyStatic(params[0].field,
|
|
|
|
VIR_DOMAIN_SCHEDULER_WEIGHT) == NULL) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Weight %s too big for destination"),
|
|
|
|
VIR_DOMAIN_SCHEDULER_WEIGHT);
|
2009-08-03 14:37:44 +02:00
|
|
|
goto error;
|
|
|
|
}
|
2011-05-26 11:39:04 -06:00
|
|
|
params[0].type = VIR_TYPED_PARAM_UINT;
|
2008-03-24 09:23:32 +00:00
|
|
|
params[0].value.ui = sexpr_int(root, "domain/cpu_weight");
|
|
|
|
|
2011-11-02 11:57:29 -06:00
|
|
|
if (*nparams > 1) {
|
|
|
|
if (virStrcpyStatic(params[1].field,
|
|
|
|
VIR_DOMAIN_SCHEDULER_CAP) == NULL) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Cap %s too big for destination"),
|
|
|
|
VIR_DOMAIN_SCHEDULER_CAP);
|
2011-11-02 11:57:29 -06:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
params[1].type = VIR_TYPED_PARAM_UINT;
|
|
|
|
params[1].value.ui = sexpr_int(root, "domain/cpu_cap");
|
2009-08-03 14:37:44 +02:00
|
|
|
}
|
2011-11-02 11:57:29 -06:00
|
|
|
|
|
|
|
if (*nparams > XEN_SCHED_CRED_NPARAM)
|
|
|
|
*nparams = XEN_SCHED_CRED_NPARAM;
|
2008-03-24 09:23:32 +00:00
|
|
|
ret = 0;
|
|
|
|
break;
|
|
|
|
default:
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Unknown scheduler"));
|
2008-03-24 09:23:32 +00:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2014-03-25 07:54:51 +01:00
|
|
|
error:
|
2008-03-24 09:23:32 +00:00
|
|
|
sexpr_free(root);
|
2008-05-29 19:20:22 +00:00
|
|
|
VIR_FREE(sched_type);
|
2012-03-22 12:33:35 +01:00
|
|
|
return ret;
|
2008-03-24 09:23:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* xenDaemonSetSchedulerParameters:
|
2013-05-02 11:13:39 +01:00
|
|
|
* @conn: the hypervisor connection
|
|
|
|
* @def: domain configuration
|
2008-03-24 09:23:32 +00:00
|
|
|
* @params: pointer to scheduler parameters
|
|
|
|
* @nparams: a number of scheduler setting parameters
|
|
|
|
*
|
|
|
|
* Set the scheduler parameters
|
|
|
|
*
|
|
|
|
* Returns 0 or -1 in case of failure
|
|
|
|
*/
|
2013-04-30 17:52:19 +01:00
|
|
|
int
|
2013-05-02 11:13:39 +01:00
|
|
|
xenDaemonSetSchedulerParameters(virConnectPtr conn,
|
|
|
|
virDomainDefPtr def,
|
2013-01-29 09:53:13 -05:00
|
|
|
virTypedParameterPtr params,
|
|
|
|
int nparams)
|
2008-03-24 09:23:32 +00:00
|
|
|
{
|
2013-05-02 11:13:39 +01:00
|
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
2008-03-24 09:23:32 +00:00
|
|
|
struct sexpr *root;
|
|
|
|
char *sched_type = NULL;
|
Convert 'int i' to 'size_t i' in src/{xen,xenapi,xenxs} 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;
|
2008-03-24 09:23:32 +00:00
|
|
|
int sched_nparam = 0;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
/* Support only xendConfigVersion >=4 and active domains */
|
2012-01-31 18:08:09 +01:00
|
|
|
if (priv->xendConfigVersion < XEND_CONFIG_VERSION_3_1_0) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID,
|
|
|
|
"%s", _("unsupported in xendConfigVersion < 4"));
|
2012-03-22 12:33:35 +01:00
|
|
|
return -1;
|
2008-03-24 09:23:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* look up the information by domain name */
|
2013-05-02 11:13:39 +01:00
|
|
|
root = sexpr_get(conn, "/xend/domain/%s?detail=1", def->name);
|
2008-03-24 09:23:32 +00:00
|
|
|
if (root == NULL)
|
2012-03-22 12:33:35 +01:00
|
|
|
return -1;
|
2008-03-24 09:23:32 +00:00
|
|
|
|
|
|
|
/* get the scheduler type */
|
2013-05-02 11:13:39 +01:00
|
|
|
sched_type = xenDaemonGetSchedulerType(conn, &sched_nparam);
|
2008-03-24 09:23:32 +00:00
|
|
|
if (sched_type == NULL) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
"%s", _("Failed to get a scheduler name"));
|
2008-03-24 09:23:32 +00:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2014-08-20 13:00:30 +02:00
|
|
|
switch (sched_nparam) {
|
2008-03-24 09:23:32 +00:00
|
|
|
case XEN_SCHED_SEDF_NPARAM:
|
|
|
|
/* TODO: Implement for Xen/SEDF */
|
|
|
|
TODO
|
|
|
|
goto error;
|
|
|
|
case XEN_SCHED_CRED_NPARAM: {
|
|
|
|
char buf_weight[VIR_UUID_BUFLEN];
|
|
|
|
char buf_cap[VIR_UUID_BUFLEN];
|
|
|
|
const char *weight = NULL;
|
|
|
|
const char *cap = NULL;
|
|
|
|
|
|
|
|
/* get the scheduler parameters */
|
|
|
|
memset(&buf_weight, 0, VIR_UUID_BUFLEN);
|
|
|
|
memset(&buf_cap, 0, VIR_UUID_BUFLEN);
|
|
|
|
for (i = 0; i < nparams; i++) {
|
2012-10-17 10:23:12 +01:00
|
|
|
if (STREQ(params[i].field, VIR_DOMAIN_SCHEDULER_WEIGHT) &&
|
2011-05-26 11:39:04 -06:00
|
|
|
params[i].type == VIR_TYPED_PARAM_UINT) {
|
2008-03-24 09:23:32 +00:00
|
|
|
snprintf(buf_weight, sizeof(buf_weight), "%u", params[i].value.ui);
|
2012-10-17 10:23:12 +01:00
|
|
|
} else if (STREQ(params[i].field, VIR_DOMAIN_SCHEDULER_CAP) &&
|
2011-05-26 11:39:04 -06:00
|
|
|
params[i].type == VIR_TYPED_PARAM_UINT) {
|
2008-03-24 09:23:32 +00:00
|
|
|
snprintf(buf_cap, sizeof(buf_cap), "%u", params[i].value.ui);
|
|
|
|
} else {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INVALID_ARG, __FUNCTION__);
|
2008-03-24 09:23:32 +00:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* if not get the scheduler parameter, set the current setting */
|
|
|
|
if (strlen(buf_weight) == 0) {
|
|
|
|
weight = sexpr_node(root, "domain/cpu_weight");
|
|
|
|
if (weight == NULL) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
"%s", _("domain information incomplete, missing cpu_weight"));
|
2008-03-24 09:23:32 +00:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
snprintf(buf_weight, sizeof(buf_weight), "%s", weight);
|
|
|
|
}
|
|
|
|
if (strlen(buf_cap) == 0) {
|
|
|
|
cap = sexpr_node(root, "domain/cpu_cap");
|
|
|
|
if (cap == NULL) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
"%s", _("domain information incomplete, missing cpu_cap"));
|
2008-03-24 09:23:32 +00:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
snprintf(buf_cap, sizeof(buf_cap), "%s", cap);
|
|
|
|
}
|
|
|
|
|
2013-05-02 11:13:39 +01:00
|
|
|
ret = xend_op(conn, def->name, "op",
|
2008-03-24 09:23:32 +00:00
|
|
|
"domain_sched_credit_set", "weight", buf_weight,
|
|
|
|
"cap", buf_cap, NULL);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Unknown scheduler"));
|
2008-03-24 09:23:32 +00:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2014-03-25 07:54:51 +01:00
|
|
|
error:
|
2008-03-24 09:23:32 +00:00
|
|
|
sexpr_free(root);
|
2008-05-29 19:20:22 +00:00
|
|
|
VIR_FREE(sched_type);
|
2012-03-22 12:33:35 +01:00
|
|
|
return ret;
|
2008-03-24 09:23:32 +00:00
|
|
|
}
|
|
|
|
|
virDomainBlockPeek call
* configure.in: Document AC_SYS_LARGEFILE.
* docs/hvsupport.html.in: Document HV support for virDomainBlockPeek.
* include/libvirt/libvirt.h.in, src/driver.h, src/libvirt.c,
src/libvirt_sym.version: Add virDomainBlockPeek infrastructure.
* src/qemu_driver.c, src/test.c: Null versions of this call.
* src/xen_unified.c, src/xend_internal.c, src/xend_internal.h,
src/xm_internal.c, src/xm_internal.h: Xen implementation.
* tests/sexpr2xmldata/sexpr2xml-curmem.xml,
tests/sexpr2xmldata/sexpr2xml-no-source-cdrom.xml: XML output
has been reordered slightly in the Xen driver, but should be
functionally the same.
2008-06-05 13:17:45 +00:00
|
|
|
/**
|
|
|
|
* xenDaemonDomainBlockPeek:
|
2013-05-02 11:24:49 +01:00
|
|
|
* @conn: the hypervisor connection
|
|
|
|
* @minidef: minimal domain configuration
|
virDomainBlockPeek call
* configure.in: Document AC_SYS_LARGEFILE.
* docs/hvsupport.html.in: Document HV support for virDomainBlockPeek.
* include/libvirt/libvirt.h.in, src/driver.h, src/libvirt.c,
src/libvirt_sym.version: Add virDomainBlockPeek infrastructure.
* src/qemu_driver.c, src/test.c: Null versions of this call.
* src/xen_unified.c, src/xend_internal.c, src/xend_internal.h,
src/xm_internal.c, src/xm_internal.h: Xen implementation.
* tests/sexpr2xmldata/sexpr2xml-curmem.xml,
tests/sexpr2xmldata/sexpr2xml-no-source-cdrom.xml: XML output
has been reordered slightly in the Xen driver, but should be
functionally the same.
2008-06-05 13:17:45 +00:00
|
|
|
* @path: path to the file or device
|
|
|
|
* @offset: offset
|
|
|
|
* @size: size
|
|
|
|
* @buffer: return buffer
|
|
|
|
*
|
2013-04-30 17:55:18 +01:00
|
|
|
* Returns 0 if successful, -1 if error
|
virDomainBlockPeek call
* configure.in: Document AC_SYS_LARGEFILE.
* docs/hvsupport.html.in: Document HV support for virDomainBlockPeek.
* include/libvirt/libvirt.h.in, src/driver.h, src/libvirt.c,
src/libvirt_sym.version: Add virDomainBlockPeek infrastructure.
* src/qemu_driver.c, src/test.c: Null versions of this call.
* src/xen_unified.c, src/xend_internal.c, src/xend_internal.h,
src/xm_internal.c, src/xm_internal.h: Xen implementation.
* tests/sexpr2xmldata/sexpr2xml-curmem.xml,
tests/sexpr2xmldata/sexpr2xml-no-source-cdrom.xml: XML output
has been reordered slightly in the Xen driver, but should be
functionally the same.
2008-06-05 13:17:45 +00:00
|
|
|
*/
|
|
|
|
int
|
2013-05-02 11:24:49 +01:00
|
|
|
xenDaemonDomainBlockPeek(virConnectPtr conn,
|
|
|
|
virDomainDefPtr minidef,
|
2013-01-29 09:53:13 -05:00
|
|
|
const char *path,
|
|
|
|
unsigned long long offset,
|
|
|
|
size_t size,
|
2012-10-17 10:23:12 +01:00
|
|
|
void *buffer)
|
virDomainBlockPeek call
* configure.in: Document AC_SYS_LARGEFILE.
* docs/hvsupport.html.in: Document HV support for virDomainBlockPeek.
* include/libvirt/libvirt.h.in, src/driver.h, src/libvirt.c,
src/libvirt_sym.version: Add virDomainBlockPeek infrastructure.
* src/qemu_driver.c, src/test.c: Null versions of this call.
* src/xen_unified.c, src/xend_internal.c, src/xend_internal.h,
src/xm_internal.c, src/xm_internal.h: Xen implementation.
* tests/sexpr2xmldata/sexpr2xml-curmem.xml,
tests/sexpr2xmldata/sexpr2xml-no-source-cdrom.xml: XML output
has been reordered slightly in the Xen driver, but should be
functionally the same.
2008-06-05 13:17:45 +00:00
|
|
|
{
|
2013-05-02 11:24:49 +01:00
|
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
2008-07-25 10:49:33 +00:00
|
|
|
struct sexpr *root = NULL;
|
|
|
|
int fd = -1, ret = -1;
|
2013-09-25 09:31:30 +01:00
|
|
|
virDomainDefPtr def = NULL;
|
2011-02-21 14:40:08 +01:00
|
|
|
int id;
|
|
|
|
char * tty;
|
|
|
|
int vncport;
|
snapshot: also support disks by path
I got confused when 'virsh domblkinfo dom disk' required the
path to a disk (which can be ambiguous, since a single file
can back multiple disks), rather than the unambiguous target
device name that I was using in disk snapshots. So, in true
developer fashion, I went for the best of both worlds - all
interfaces that operate on a disk (aka block) now accept
either the target name or the unambiguous path to the backing
file used by the disk.
* src/conf/domain_conf.h (virDomainDiskIndexByName): Add
parameter.
(virDomainDiskPathByName): New prototype.
* src/libvirt_private.syms (domain_conf.h): Export it.
* src/conf/domain_conf.c (virDomainDiskIndexByName): Also allow
searching by path, and decide whether ambiguity is okay.
(virDomainDiskPathByName): New function.
(virDomainDiskRemoveByName, virDomainSnapshotAlignDisks): Update
callers.
* src/qemu/qemu_driver.c (qemudDomainBlockPeek)
(qemuDomainAttachDeviceConfig, qemuDomainUpdateDeviceConfig)
(qemuDomainGetBlockInfo, qemuDiskPathToAlias): Likewise.
* src/qemu/qemu_process.c (qemuProcessFindDomainDiskByPath):
Likewise.
* src/libxl/libxl_driver.c (libxlDomainAttachDeviceDiskLive)
(libxlDomainDetachDeviceDiskLive, libxlDomainAttachDeviceConfig)
(libxlDomainUpdateDeviceConfig): Likewise.
* src/uml/uml_driver.c (umlDomainBlockPeek): Likewise.
* src/xen/xend_internal.c (xenDaemonDomainBlockPeek): Likewise.
* docs/formatsnapshot.html.in: Update documentation.
* tools/virsh.pod (domblkstat, domblkinfo): Likewise.
* docs/schemas/domaincommon.rng (diskTarget): Tighten pattern on
disk targets.
* docs/schemas/domainsnapshot.rng (disksnapshot): Update to match.
* tests/domainsnapshotxml2xmlin/disk_snapshot.xml: Update test.
2011-08-19 20:38:36 -06:00
|
|
|
const char *actual;
|
virDomainBlockPeek call
* configure.in: Document AC_SYS_LARGEFILE.
* docs/hvsupport.html.in: Document HV support for virDomainBlockPeek.
* include/libvirt/libvirt.h.in, src/driver.h, src/libvirt.c,
src/libvirt_sym.version: Add virDomainBlockPeek infrastructure.
* src/qemu_driver.c, src/test.c: Null versions of this call.
* src/xen_unified.c, src/xend_internal.c, src/xend_internal.h,
src/xm_internal.c, src/xm_internal.h: Xen implementation.
* tests/sexpr2xmldata/sexpr2xml-curmem.xml,
tests/sexpr2xmldata/sexpr2xml-no-source-cdrom.xml: XML output
has been reordered slightly in the Xen driver, but should be
functionally the same.
2008-06-05 13:17:45 +00:00
|
|
|
|
|
|
|
/* Security check: The path must correspond to a block device. */
|
2014-09-03 13:04:59 -06:00
|
|
|
if (minidef->id > 0) {
|
2013-05-02 11:24:49 +01:00
|
|
|
root = sexpr_get(conn, "/xend/domain/%d?detail=1",
|
|
|
|
minidef->id);
|
2014-09-03 13:04:59 -06:00
|
|
|
} else if (minidef->id < 0) {
|
2013-05-02 11:24:49 +01:00
|
|
|
root = sexpr_get(conn, "/xend/domain/%s?detail=1",
|
|
|
|
minidef->name);
|
2014-09-03 13:04:59 -06:00
|
|
|
} else {
|
virDomainBlockPeek call
* configure.in: Document AC_SYS_LARGEFILE.
* docs/hvsupport.html.in: Document HV support for virDomainBlockPeek.
* include/libvirt/libvirt.h.in, src/driver.h, src/libvirt.c,
src/libvirt_sym.version: Add virDomainBlockPeek infrastructure.
* src/qemu_driver.c, src/test.c: Null versions of this call.
* src/xen_unified.c, src/xend_internal.c, src/xend_internal.h,
src/xm_internal.c, src/xm_internal.h: Xen implementation.
* tests/sexpr2xmldata/sexpr2xml-curmem.xml,
tests/sexpr2xmldata/sexpr2xml-no-source-cdrom.xml: XML output
has been reordered slightly in the Xen driver, but should be
functionally the same.
2008-06-05 13:17:45 +00:00
|
|
|
/* This call always fails for dom0. */
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID,
|
|
|
|
"%s", _("domainBlockPeek is not supported for dom0"));
|
virDomainBlockPeek call
* configure.in: Document AC_SYS_LARGEFILE.
* docs/hvsupport.html.in: Document HV support for virDomainBlockPeek.
* include/libvirt/libvirt.h.in, src/driver.h, src/libvirt.c,
src/libvirt_sym.version: Add virDomainBlockPeek infrastructure.
* src/qemu_driver.c, src/test.c: Null versions of this call.
* src/xen_unified.c, src/xend_internal.c, src/xend_internal.h,
src/xm_internal.c, src/xm_internal.h: Xen implementation.
* tests/sexpr2xmldata/sexpr2xml-curmem.xml,
tests/sexpr2xmldata/sexpr2xml-no-source-cdrom.xml: XML output
has been reordered slightly in the Xen driver, but should be
functionally the same.
2008-06-05 13:17:45 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!root) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_XEN_CALL, __FUNCTION__);
|
virDomainBlockPeek call
* configure.in: Document AC_SYS_LARGEFILE.
* docs/hvsupport.html.in: Document HV support for virDomainBlockPeek.
* include/libvirt/libvirt.h.in, src/driver.h, src/libvirt.c,
src/libvirt_sym.version: Add virDomainBlockPeek infrastructure.
* src/qemu_driver.c, src/test.c: Null versions of this call.
* src/xen_unified.c, src/xend_internal.c, src/xend_internal.h,
src/xm_internal.c, src/xm_internal.h: Xen implementation.
* tests/sexpr2xmldata/sexpr2xml-curmem.xml,
tests/sexpr2xmldata/sexpr2xml-no-source-cdrom.xml: XML output
has been reordered slightly in the Xen driver, but should be
functionally the same.
2008-06-05 13:17:45 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2013-09-25 09:31:30 +01:00
|
|
|
if (xenGetDomIdFromSxpr(root, priv->xendConfigVersion, &id) < 0)
|
|
|
|
goto cleanup;
|
2011-02-21 14:40:08 +01:00
|
|
|
xenUnifiedLock(priv);
|
2013-05-02 11:24:49 +01:00
|
|
|
tty = xenStoreDomainGetConsolePath(conn, id);
|
|
|
|
vncport = xenStoreDomainGetVNCPort(conn, id);
|
2011-02-21 14:40:08 +01:00
|
|
|
xenUnifiedUnlock(priv);
|
|
|
|
|
2011-02-21 14:40:12 +01:00
|
|
|
if (!(def = xenParseSxpr(root, priv->xendConfigVersion, NULL, tty,
|
|
|
|
vncport)))
|
2008-07-25 10:49:33 +00:00
|
|
|
goto cleanup;
|
virDomainBlockPeek call
* configure.in: Document AC_SYS_LARGEFILE.
* docs/hvsupport.html.in: Document HV support for virDomainBlockPeek.
* include/libvirt/libvirt.h.in, src/driver.h, src/libvirt.c,
src/libvirt_sym.version: Add virDomainBlockPeek infrastructure.
* src/qemu_driver.c, src/test.c: Null versions of this call.
* src/xen_unified.c, src/xend_internal.c, src/xend_internal.h,
src/xm_internal.c, src/xm_internal.h: Xen implementation.
* tests/sexpr2xmldata/sexpr2xml-curmem.xml,
tests/sexpr2xmldata/sexpr2xml-no-source-cdrom.xml: XML output
has been reordered slightly in the Xen driver, but should be
functionally the same.
2008-06-05 13:17:45 +00:00
|
|
|
|
snapshot: also support disks by path
I got confused when 'virsh domblkinfo dom disk' required the
path to a disk (which can be ambiguous, since a single file
can back multiple disks), rather than the unambiguous target
device name that I was using in disk snapshots. So, in true
developer fashion, I went for the best of both worlds - all
interfaces that operate on a disk (aka block) now accept
either the target name or the unambiguous path to the backing
file used by the disk.
* src/conf/domain_conf.h (virDomainDiskIndexByName): Add
parameter.
(virDomainDiskPathByName): New prototype.
* src/libvirt_private.syms (domain_conf.h): Export it.
* src/conf/domain_conf.c (virDomainDiskIndexByName): Also allow
searching by path, and decide whether ambiguity is okay.
(virDomainDiskPathByName): New function.
(virDomainDiskRemoveByName, virDomainSnapshotAlignDisks): Update
callers.
* src/qemu/qemu_driver.c (qemudDomainBlockPeek)
(qemuDomainAttachDeviceConfig, qemuDomainUpdateDeviceConfig)
(qemuDomainGetBlockInfo, qemuDiskPathToAlias): Likewise.
* src/qemu/qemu_process.c (qemuProcessFindDomainDiskByPath):
Likewise.
* src/libxl/libxl_driver.c (libxlDomainAttachDeviceDiskLive)
(libxlDomainDetachDeviceDiskLive, libxlDomainAttachDeviceConfig)
(libxlDomainUpdateDeviceConfig): Likewise.
* src/uml/uml_driver.c (umlDomainBlockPeek): Likewise.
* src/xen/xend_internal.c (xenDaemonDomainBlockPeek): Likewise.
* docs/formatsnapshot.html.in: Update documentation.
* tools/virsh.pod (domblkstat, domblkinfo): Likewise.
* docs/schemas/domaincommon.rng (diskTarget): Tighten pattern on
disk targets.
* docs/schemas/domainsnapshot.rng (disksnapshot): Update to match.
* tests/domainsnapshotxml2xmlin/disk_snapshot.xml: Update test.
2011-08-19 20:38:36 -06:00
|
|
|
if (!(actual = virDomainDiskPathByName(def, path))) {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
|
|
_("%s: invalid path"), path);
|
2008-07-25 10:49:33 +00:00
|
|
|
goto cleanup;
|
virDomainBlockPeek call
* configure.in: Document AC_SYS_LARGEFILE.
* docs/hvsupport.html.in: Document HV support for virDomainBlockPeek.
* include/libvirt/libvirt.h.in, src/driver.h, src/libvirt.c,
src/libvirt_sym.version: Add virDomainBlockPeek infrastructure.
* src/qemu_driver.c, src/test.c: Null versions of this call.
* src/xen_unified.c, src/xend_internal.c, src/xend_internal.h,
src/xm_internal.c, src/xm_internal.h: Xen implementation.
* tests/sexpr2xmldata/sexpr2xml-curmem.xml,
tests/sexpr2xmldata/sexpr2xml-no-source-cdrom.xml: XML output
has been reordered slightly in the Xen driver, but should be
functionally the same.
2008-06-05 13:17:45 +00:00
|
|
|
}
|
snapshot: also support disks by path
I got confused when 'virsh domblkinfo dom disk' required the
path to a disk (which can be ambiguous, since a single file
can back multiple disks), rather than the unambiguous target
device name that I was using in disk snapshots. So, in true
developer fashion, I went for the best of both worlds - all
interfaces that operate on a disk (aka block) now accept
either the target name or the unambiguous path to the backing
file used by the disk.
* src/conf/domain_conf.h (virDomainDiskIndexByName): Add
parameter.
(virDomainDiskPathByName): New prototype.
* src/libvirt_private.syms (domain_conf.h): Export it.
* src/conf/domain_conf.c (virDomainDiskIndexByName): Also allow
searching by path, and decide whether ambiguity is okay.
(virDomainDiskPathByName): New function.
(virDomainDiskRemoveByName, virDomainSnapshotAlignDisks): Update
callers.
* src/qemu/qemu_driver.c (qemudDomainBlockPeek)
(qemuDomainAttachDeviceConfig, qemuDomainUpdateDeviceConfig)
(qemuDomainGetBlockInfo, qemuDiskPathToAlias): Likewise.
* src/qemu/qemu_process.c (qemuProcessFindDomainDiskByPath):
Likewise.
* src/libxl/libxl_driver.c (libxlDomainAttachDeviceDiskLive)
(libxlDomainDetachDeviceDiskLive, libxlDomainAttachDeviceConfig)
(libxlDomainUpdateDeviceConfig): Likewise.
* src/uml/uml_driver.c (umlDomainBlockPeek): Likewise.
* src/xen/xend_internal.c (xenDaemonDomainBlockPeek): Likewise.
* docs/formatsnapshot.html.in: Update documentation.
* tools/virsh.pod (domblkstat, domblkinfo): Likewise.
* docs/schemas/domaincommon.rng (diskTarget): Tighten pattern on
disk targets.
* docs/schemas/domainsnapshot.rng (disksnapshot): Update to match.
* tests/domainsnapshotxml2xmlin/disk_snapshot.xml: Update test.
2011-08-19 20:38:36 -06:00
|
|
|
path = actual;
|
virDomainBlockPeek call
* configure.in: Document AC_SYS_LARGEFILE.
* docs/hvsupport.html.in: Document HV support for virDomainBlockPeek.
* include/libvirt/libvirt.h.in, src/driver.h, src/libvirt.c,
src/libvirt_sym.version: Add virDomainBlockPeek infrastructure.
* src/qemu_driver.c, src/test.c: Null versions of this call.
* src/xen_unified.c, src/xend_internal.c, src/xend_internal.h,
src/xm_internal.c, src/xm_internal.h: Xen implementation.
* tests/sexpr2xmldata/sexpr2xml-curmem.xml,
tests/sexpr2xmldata/sexpr2xml-no-source-cdrom.xml: XML output
has been reordered slightly in the Xen driver, but should be
functionally the same.
2008-06-05 13:17:45 +00:00
|
|
|
|
|
|
|
/* The path is correct, now try to open it and get its size. */
|
2012-10-17 10:23:12 +01:00
|
|
|
fd = open(path, O_RDONLY);
|
2008-06-05 21:12:26 +00:00
|
|
|
if (fd == -1) {
|
2010-02-04 21:02:58 +01:00
|
|
|
virReportSystemError(errno,
|
2009-01-20 17:13:33 +00:00
|
|
|
_("failed to open for reading: %s"),
|
|
|
|
path);
|
2008-07-25 10:49:33 +00:00
|
|
|
goto cleanup;
|
virDomainBlockPeek call
* configure.in: Document AC_SYS_LARGEFILE.
* docs/hvsupport.html.in: Document HV support for virDomainBlockPeek.
* include/libvirt/libvirt.h.in, src/driver.h, src/libvirt.c,
src/libvirt_sym.version: Add virDomainBlockPeek infrastructure.
* src/qemu_driver.c, src/test.c: Null versions of this call.
* src/xen_unified.c, src/xend_internal.c, src/xend_internal.h,
src/xm_internal.c, src/xm_internal.h: Xen implementation.
* tests/sexpr2xmldata/sexpr2xml-curmem.xml,
tests/sexpr2xmldata/sexpr2xml-no-source-cdrom.xml: XML output
has been reordered slightly in the Xen driver, but should be
functionally the same.
2008-06-05 13:17:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Seek and read. */
|
|
|
|
/* NB. Because we configure with AC_SYS_LARGEFILE, off_t should
|
|
|
|
* be 64 bits on all platforms.
|
|
|
|
*/
|
2012-10-17 10:23:12 +01:00
|
|
|
if (lseek(fd, offset, SEEK_SET) == (off_t) -1 ||
|
|
|
|
saferead(fd, buffer, size) == (ssize_t) -1) {
|
2010-02-04 21:02:58 +01:00
|
|
|
virReportSystemError(errno,
|
2009-01-20 17:13:33 +00:00
|
|
|
_("failed to lseek or read from file: %s"),
|
|
|
|
path);
|
2008-07-25 10:49:33 +00:00
|
|
|
goto cleanup;
|
virDomainBlockPeek call
* configure.in: Document AC_SYS_LARGEFILE.
* docs/hvsupport.html.in: Document HV support for virDomainBlockPeek.
* include/libvirt/libvirt.h.in, src/driver.h, src/libvirt.c,
src/libvirt_sym.version: Add virDomainBlockPeek infrastructure.
* src/qemu_driver.c, src/test.c: Null versions of this call.
* src/xen_unified.c, src/xend_internal.c, src/xend_internal.h,
src/xm_internal.c, src/xm_internal.h: Xen implementation.
* tests/sexpr2xmldata/sexpr2xml-curmem.xml,
tests/sexpr2xmldata/sexpr2xml-no-source-cdrom.xml: XML output
has been reordered slightly in the Xen driver, but should be
functionally the same.
2008-06-05 13:17:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
2008-07-25 10:49:33 +00:00
|
|
|
cleanup:
|
2010-11-09 15:48:48 -05:00
|
|
|
VIR_FORCE_CLOSE(fd);
|
2008-07-25 10:49:33 +00:00
|
|
|
sexpr_free(root);
|
|
|
|
virDomainDefFree(def);
|
virDomainBlockPeek call
* configure.in: Document AC_SYS_LARGEFILE.
* docs/hvsupport.html.in: Document HV support for virDomainBlockPeek.
* include/libvirt/libvirt.h.in, src/driver.h, src/libvirt.c,
src/libvirt_sym.version: Add virDomainBlockPeek infrastructure.
* src/qemu_driver.c, src/test.c: Null versions of this call.
* src/xen_unified.c, src/xend_internal.c, src/xend_internal.h,
src/xm_internal.c, src/xm_internal.h: Xen implementation.
* tests/sexpr2xmldata/sexpr2xml-curmem.xml,
tests/sexpr2xmldata/sexpr2xml-no-source-cdrom.xml: XML output
has been reordered slightly in the Xen driver, but should be
functionally the same.
2008-06-05 13:17:45 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2008-07-25 13:17:27 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* virDomainXMLDevID:
|
2013-05-02 11:24:49 +01:00
|
|
|
* @conn: the hypervisor connection
|
|
|
|
* @minidef: minimal domain configuration
|
2008-07-25 13:17:27 +00:00
|
|
|
* @dev: pointer to device config object
|
|
|
|
* @class: Xen device class "vbd" or "vif" (OUT)
|
|
|
|
* @ref: Xen device reference (OUT)
|
|
|
|
*
|
|
|
|
* Set class according to XML root, and:
|
|
|
|
* - if disk, copy in ref the target name from description
|
|
|
|
* - if network, get MAC address from description, scan XenStore and
|
|
|
|
* copy in ref the corresponding vif number.
|
2009-12-22 10:53:51 +01:00
|
|
|
* - if pci, get BDF from description, scan XenStore and
|
|
|
|
* copy in ref the corresponding dev number.
|
2008-07-25 13:17:27 +00:00
|
|
|
*
|
|
|
|
* Returns 0 in case of success, -1 in case of failure.
|
|
|
|
*/
|
|
|
|
static int
|
2013-05-02 10:56:31 +01:00
|
|
|
virDomainXMLDevID(virConnectPtr conn,
|
|
|
|
virDomainDefPtr def,
|
2008-07-25 13:17:27 +00:00
|
|
|
virDomainDeviceDefPtr dev,
|
|
|
|
char *class,
|
|
|
|
char *ref,
|
|
|
|
int ref_len)
|
|
|
|
{
|
2013-05-02 10:56:31 +01:00
|
|
|
xenUnifiedPrivatePtr priv = conn->privateData;
|
2008-07-25 13:17:27 +00:00
|
|
|
char *xref;
|
2009-08-03 14:37:44 +02:00
|
|
|
char *tmp;
|
2008-07-25 13:17:27 +00:00
|
|
|
|
|
|
|
if (dev->type == VIR_DOMAIN_DEVICE_DISK) {
|
2015-01-30 10:37:10 +01:00
|
|
|
const char *driver = virDomainDiskGetDriver(dev->data.disk);
|
|
|
|
|
2014-03-18 17:12:43 -06:00
|
|
|
if (STREQ_NULLABLE(driver, "tap") || STREQ_NULLABLE(driver, "tap2"))
|
|
|
|
strcpy(class, driver);
|
2009-03-10 10:32:24 +00:00
|
|
|
else
|
|
|
|
strcpy(class, "vbd");
|
|
|
|
|
2008-07-25 13:17:27 +00:00
|
|
|
if (dev->data.disk->dst == NULL)
|
|
|
|
return -1;
|
2009-01-21 18:11:14 +00:00
|
|
|
xenUnifiedLock(priv);
|
2013-05-02 10:56:31 +01:00
|
|
|
xref = xenStoreDomainGetDiskID(conn, def->id,
|
2008-07-25 13:17:27 +00:00
|
|
|
dev->data.disk->dst);
|
2009-01-21 18:11:14 +00:00
|
|
|
xenUnifiedUnlock(priv);
|
2008-07-25 13:17:27 +00:00
|
|
|
if (xref == NULL)
|
|
|
|
return -1;
|
|
|
|
|
2009-08-03 14:37:44 +02:00
|
|
|
tmp = virStrcpy(ref, xref, ref_len);
|
2009-12-10 00:00:50 +01:00
|
|
|
VIR_FREE(xref);
|
2009-08-03 14:37:44 +02:00
|
|
|
if (tmp == NULL)
|
|
|
|
return -1;
|
2008-07-25 13:17:27 +00:00
|
|
|
} else if (dev->type == VIR_DOMAIN_DEVICE_NET) {
|
2013-03-26 12:21:33 +01:00
|
|
|
char mac[VIR_MAC_STRING_BUFLEN];
|
2013-05-02 10:56:31 +01:00
|
|
|
virDomainNetDefPtr netdef = dev->data.net;
|
|
|
|
virMacAddrFormat(&netdef->mac, mac);
|
2008-07-25 13:17:27 +00:00
|
|
|
|
|
|
|
strcpy(class, "vif");
|
|
|
|
|
2009-01-21 18:11:14 +00:00
|
|
|
xenUnifiedLock(priv);
|
2013-05-02 10:56:31 +01:00
|
|
|
xref = xenStoreDomainGetNetworkID(conn, def->id, mac);
|
2009-01-21 18:11:14 +00:00
|
|
|
xenUnifiedUnlock(priv);
|
2008-07-25 13:17:27 +00:00
|
|
|
if (xref == NULL)
|
|
|
|
return -1;
|
|
|
|
|
2009-08-03 14:37:44 +02:00
|
|
|
tmp = virStrcpy(ref, xref, ref_len);
|
2009-12-10 00:00:50 +01:00
|
|
|
VIR_FREE(xref);
|
2009-08-03 14:37:44 +02:00
|
|
|
if (tmp == NULL)
|
|
|
|
return -1;
|
2009-04-03 12:38:52 +00:00
|
|
|
} else if (dev->type == VIR_DOMAIN_DEVICE_HOSTDEV &&
|
|
|
|
dev->data.hostdev->mode == VIR_DOMAIN_HOSTDEV_MODE_SUBSYS &&
|
|
|
|
dev->data.hostdev->source.subsys.type == VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_PCI) {
|
2009-12-22 10:53:51 +01:00
|
|
|
char *bdf;
|
2013-05-02 10:56:31 +01:00
|
|
|
virDomainHostdevDefPtr hostdef = dev->data.hostdev;
|
2009-12-22 10:53:51 +01:00
|
|
|
|
|
|
|
if (virAsprintf(&bdf, "%04x:%02x:%02x.%0x",
|
2013-05-02 10:56:31 +01:00
|
|
|
hostdef->source.subsys.u.pci.addr.domain,
|
|
|
|
hostdef->source.subsys.u.pci.addr.bus,
|
|
|
|
hostdef->source.subsys.u.pci.addr.slot,
|
2013-07-04 12:18:49 +02:00
|
|
|
hostdef->source.subsys.u.pci.addr.function) < 0)
|
2009-12-22 10:53:51 +01:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
strcpy(class, "pci");
|
|
|
|
|
|
|
|
xenUnifiedLock(priv);
|
2013-05-02 10:56:31 +01:00
|
|
|
xref = xenStoreDomainGetPCIID(conn, def->id, bdf);
|
2009-12-22 10:53:51 +01:00
|
|
|
xenUnifiedUnlock(priv);
|
|
|
|
VIR_FREE(bdf);
|
|
|
|
if (xref == NULL)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
tmp = virStrcpy(ref, xref, ref_len);
|
|
|
|
VIR_FREE(xref);
|
|
|
|
if (tmp == NULL)
|
|
|
|
return -1;
|
2008-07-25 13:17:27 +00:00
|
|
|
} else {
|
2012-07-18 14:48:05 +01:00
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID,
|
|
|
|
"%s", _("hotplug of device type not supported"));
|
2008-07-25 13:17:27 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|