2013-08-30 17:18:31 +00:00
|
|
|
/*
|
|
|
|
* libxl_domain.c: libxl domain object private state
|
|
|
|
*
|
2014-01-31 23:57:09 +00:00
|
|
|
* Copyright (C) 2011-2014 SUSE LINUX Products GmbH, Nuernberg, Germany.
|
2013-08-30 17:18:31 +00: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/>.
|
|
|
|
*
|
|
|
|
* Authors:
|
|
|
|
* Jim Fehlig <jfehlig@suse.com>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include "libxl_domain.h"
|
|
|
|
|
|
|
|
#include "viralloc.h"
|
|
|
|
#include "virfile.h"
|
|
|
|
#include "virerror.h"
|
|
|
|
#include "virlog.h"
|
|
|
|
#include "virstring.h"
|
|
|
|
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_LIBXL
|
|
|
|
|
|
|
|
|
|
|
|
/* Object used to store info related to libxl event registrations */
|
2014-02-01 01:09:53 +00:00
|
|
|
typedef struct _libxlEventHookInfo libxlEventHookInfo;
|
|
|
|
typedef libxlEventHookInfo *libxlEventHookInfoPtr;
|
2013-08-30 17:18:31 +00:00
|
|
|
struct _libxlEventHookInfo {
|
|
|
|
libxlEventHookInfoPtr next;
|
|
|
|
libxlDomainObjPrivatePtr priv;
|
|
|
|
void *xl_priv;
|
|
|
|
int id;
|
|
|
|
};
|
|
|
|
|
|
|
|
static virClassPtr libxlDomainObjPrivateClass;
|
|
|
|
|
|
|
|
static void
|
|
|
|
libxlDomainObjPrivateDispose(void *obj);
|
|
|
|
|
|
|
|
static int
|
|
|
|
libxlDomainObjPrivateOnceInit(void)
|
|
|
|
{
|
|
|
|
if (!(libxlDomainObjPrivateClass = virClassNew(virClassForObjectLockable(),
|
|
|
|
"libxlDomainObjPrivate",
|
|
|
|
sizeof(libxlDomainObjPrivate),
|
|
|
|
libxlDomainObjPrivateDispose)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
VIR_ONCE_GLOBAL_INIT(libxlDomainObjPrivate)
|
|
|
|
|
|
|
|
static void
|
2014-01-31 23:57:09 +00:00
|
|
|
libxlDomainObjFDEventHookInfoFree(void *obj)
|
|
|
|
{
|
|
|
|
VIR_FREE(obj);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
libxlDomainObjTimerEventHookInfoFree(void *obj)
|
2013-08-30 17:18:31 +00:00
|
|
|
{
|
|
|
|
libxlEventHookInfoPtr info = obj;
|
|
|
|
|
|
|
|
/* Drop reference on libxlDomainObjPrivate */
|
|
|
|
virObjectUnref(info->priv);
|
|
|
|
VIR_FREE(info);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
libxlDomainObjFDEventCallback(int watch ATTRIBUTE_UNUSED,
|
|
|
|
int fd,
|
|
|
|
int vir_events,
|
|
|
|
void *fd_info)
|
|
|
|
{
|
|
|
|
libxlEventHookInfoPtr info = fd_info;
|
|
|
|
int events = 0;
|
|
|
|
|
|
|
|
virObjectLock(info->priv);
|
|
|
|
if (vir_events & VIR_EVENT_HANDLE_READABLE)
|
|
|
|
events |= POLLIN;
|
|
|
|
if (vir_events & VIR_EVENT_HANDLE_WRITABLE)
|
|
|
|
events |= POLLOUT;
|
|
|
|
if (vir_events & VIR_EVENT_HANDLE_ERROR)
|
|
|
|
events |= POLLERR;
|
|
|
|
if (vir_events & VIR_EVENT_HANDLE_HANGUP)
|
|
|
|
events |= POLLHUP;
|
|
|
|
|
|
|
|
virObjectUnlock(info->priv);
|
|
|
|
libxl_osevent_occurred_fd(info->priv->ctx, info->xl_priv, fd, 0, events);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
libxlDomainObjFDRegisterEventHook(void *priv,
|
|
|
|
int fd,
|
|
|
|
void **hndp,
|
|
|
|
short events,
|
|
|
|
void *xl_priv)
|
|
|
|
{
|
|
|
|
int vir_events = VIR_EVENT_HANDLE_ERROR;
|
|
|
|
libxlEventHookInfoPtr info;
|
|
|
|
|
|
|
|
if (VIR_ALLOC(info) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
info->priv = priv;
|
|
|
|
info->xl_priv = xl_priv;
|
|
|
|
|
|
|
|
if (events & POLLIN)
|
|
|
|
vir_events |= VIR_EVENT_HANDLE_READABLE;
|
|
|
|
if (events & POLLOUT)
|
|
|
|
vir_events |= VIR_EVENT_HANDLE_WRITABLE;
|
|
|
|
|
|
|
|
info->id = virEventAddHandle(fd, vir_events, libxlDomainObjFDEventCallback,
|
2014-01-31 23:57:09 +00:00
|
|
|
info, libxlDomainObjFDEventHookInfoFree);
|
2013-08-30 17:18:31 +00:00
|
|
|
if (info->id < 0) {
|
|
|
|
VIR_FREE(info);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
*hndp = info;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
libxlDomainObjFDModifyEventHook(void *priv ATTRIBUTE_UNUSED,
|
|
|
|
int fd ATTRIBUTE_UNUSED,
|
|
|
|
void **hndp,
|
|
|
|
short events)
|
|
|
|
{
|
|
|
|
libxlEventHookInfoPtr info = *hndp;
|
|
|
|
int vir_events = VIR_EVENT_HANDLE_ERROR;
|
|
|
|
|
|
|
|
virObjectLock(info->priv);
|
|
|
|
if (events & POLLIN)
|
|
|
|
vir_events |= VIR_EVENT_HANDLE_READABLE;
|
|
|
|
if (events & POLLOUT)
|
|
|
|
vir_events |= VIR_EVENT_HANDLE_WRITABLE;
|
|
|
|
|
|
|
|
virEventUpdateHandle(info->id, vir_events);
|
|
|
|
virObjectUnlock(info->priv);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
libxlDomainObjFDDeregisterEventHook(void *priv ATTRIBUTE_UNUSED,
|
|
|
|
int fd ATTRIBUTE_UNUSED,
|
|
|
|
void *hnd)
|
|
|
|
{
|
|
|
|
libxlEventHookInfoPtr info = hnd;
|
|
|
|
libxlDomainObjPrivatePtr p = info->priv;
|
|
|
|
|
|
|
|
virObjectLock(p);
|
|
|
|
virEventRemoveHandle(info->id);
|
|
|
|
virObjectUnlock(p);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
libxlDomainObjTimerCallback(int timer ATTRIBUTE_UNUSED, void *timer_info)
|
|
|
|
{
|
|
|
|
libxlEventHookInfoPtr info = timer_info;
|
|
|
|
libxlDomainObjPrivatePtr p = info->priv;
|
|
|
|
|
|
|
|
virObjectLock(p);
|
|
|
|
/*
|
|
|
|
* libxl expects the event to be deregistered when calling
|
|
|
|
* libxl_osevent_occurred_timeout, but we dont want the event info
|
|
|
|
* destroyed. Disable the timeout and only remove it after returning
|
|
|
|
* from libxl.
|
|
|
|
*/
|
|
|
|
virEventUpdateTimeout(info->id, -1);
|
|
|
|
virObjectUnlock(p);
|
|
|
|
libxl_osevent_occurred_timeout(p->ctx, info->xl_priv);
|
|
|
|
virObjectLock(p);
|
2014-02-01 01:09:53 +00:00
|
|
|
virEventRemoveTimeout(info->id);
|
2013-08-30 17:18:31 +00:00
|
|
|
virObjectUnlock(p);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
libxlDomainObjTimeoutRegisterEventHook(void *priv,
|
|
|
|
void **hndp,
|
|
|
|
struct timeval abs_t,
|
|
|
|
void *xl_priv)
|
|
|
|
{
|
|
|
|
libxlEventHookInfoPtr info;
|
|
|
|
struct timeval now;
|
|
|
|
struct timeval res;
|
|
|
|
static struct timeval zero;
|
|
|
|
int timeout;
|
|
|
|
|
|
|
|
if (VIR_ALLOC(info) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
info->priv = priv;
|
|
|
|
/*
|
|
|
|
* Also take a reference on the domain object. Reference is dropped in
|
|
|
|
* libxlDomainObjEventHookInfoFree, ensuring the domain object outlives the
|
|
|
|
* timeout event objects.
|
|
|
|
*/
|
|
|
|
virObjectRef(info->priv);
|
|
|
|
info->xl_priv = xl_priv;
|
|
|
|
|
|
|
|
gettimeofday(&now, NULL);
|
|
|
|
timersub(&abs_t, &now, &res);
|
|
|
|
/* Ensure timeout is not overflowed */
|
|
|
|
if (timercmp(&res, &zero, <)) {
|
|
|
|
timeout = 0;
|
|
|
|
} else if (res.tv_sec > INT_MAX / 1000) {
|
|
|
|
timeout = INT_MAX;
|
|
|
|
} else {
|
|
|
|
timeout = res.tv_sec * 1000 + (res.tv_usec + 999) / 1000;
|
|
|
|
}
|
|
|
|
info->id = virEventAddTimeout(timeout, libxlDomainObjTimerCallback,
|
2014-01-31 23:57:09 +00:00
|
|
|
info, libxlDomainObjTimerEventHookInfoFree);
|
2013-08-30 17:18:31 +00:00
|
|
|
if (info->id < 0) {
|
|
|
|
virObjectUnref(info->priv);
|
|
|
|
VIR_FREE(info);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
*hndp = info;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Note: There are two changes wrt timeouts starting with xen-unstable
|
|
|
|
* changeset 26469:
|
|
|
|
*
|
|
|
|
* 1. Timeout modify callbacks will only be invoked with an abs_t of {0,0},
|
|
|
|
* i.e. make the timeout fire immediately. Prior to this commit, timeout
|
|
|
|
* modify callbacks were never invoked.
|
|
|
|
*
|
|
|
|
* 2. Timeout deregister hooks will no longer be called.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
libxlDomainObjTimeoutModifyEventHook(void *priv ATTRIBUTE_UNUSED,
|
|
|
|
void **hndp,
|
|
|
|
struct timeval abs_t ATTRIBUTE_UNUSED)
|
|
|
|
{
|
|
|
|
libxlEventHookInfoPtr info = *hndp;
|
|
|
|
|
|
|
|
virObjectLock(info->priv);
|
|
|
|
/* Make the timeout fire */
|
|
|
|
virEventUpdateTimeout(info->id, 0);
|
|
|
|
virObjectUnlock(info->priv);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
libxlDomainObjTimeoutDeregisterEventHook(void *priv ATTRIBUTE_UNUSED,
|
|
|
|
void *hnd)
|
|
|
|
{
|
|
|
|
libxlEventHookInfoPtr info = hnd;
|
|
|
|
libxlDomainObjPrivatePtr p = info->priv;
|
|
|
|
|
|
|
|
virObjectLock(p);
|
2014-02-01 01:09:53 +00:00
|
|
|
virEventRemoveTimeout(info->id);
|
2013-08-30 17:18:31 +00:00
|
|
|
virObjectUnlock(p);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static const libxl_osevent_hooks libxl_event_callbacks = {
|
|
|
|
.fd_register = libxlDomainObjFDRegisterEventHook,
|
|
|
|
.fd_modify = libxlDomainObjFDModifyEventHook,
|
|
|
|
.fd_deregister = libxlDomainObjFDDeregisterEventHook,
|
|
|
|
.timeout_register = libxlDomainObjTimeoutRegisterEventHook,
|
|
|
|
.timeout_modify = libxlDomainObjTimeoutModifyEventHook,
|
|
|
|
.timeout_deregister = libxlDomainObjTimeoutDeregisterEventHook,
|
|
|
|
};
|
|
|
|
|
|
|
|
static void *
|
|
|
|
libxlDomainObjPrivateAlloc(void)
|
|
|
|
{
|
|
|
|
libxlDomainObjPrivatePtr priv;
|
|
|
|
|
|
|
|
if (libxlDomainObjPrivateInitialize() < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (!(priv = virObjectLockableNew(libxlDomainObjPrivateClass)))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (!(priv->devs = virChrdevAlloc())) {
|
|
|
|
virObjectUnref(priv);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return priv;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
libxlDomainObjPrivateDispose(void *obj)
|
|
|
|
{
|
|
|
|
libxlDomainObjPrivatePtr priv = obj;
|
|
|
|
|
|
|
|
if (priv->deathW)
|
|
|
|
libxl_evdisable_domain_death(priv->ctx, priv->deathW);
|
|
|
|
|
|
|
|
virChrdevFree(priv->devs);
|
2013-12-20 07:14:42 +00:00
|
|
|
libxl_ctx_free(priv->ctx);
|
2013-08-30 17:18:31 +00:00
|
|
|
if (priv->logger_file)
|
|
|
|
VIR_FORCE_FCLOSE(priv->logger_file);
|
|
|
|
|
2013-12-20 07:14:42 +00:00
|
|
|
xtl_logger_destroy(priv->logger);
|
2013-08-30 17:18:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
libxlDomainObjPrivateFree(void *data)
|
|
|
|
{
|
|
|
|
libxlDomainObjPrivatePtr priv = data;
|
|
|
|
|
|
|
|
if (priv->deathW) {
|
|
|
|
libxl_evdisable_domain_death(priv->ctx, priv->deathW);
|
|
|
|
priv->deathW = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
virObjectUnref(priv);
|
|
|
|
}
|
|
|
|
|
|
|
|
virDomainXMLPrivateDataCallbacks libxlDomainXMLPrivateDataCallbacks = {
|
|
|
|
.alloc = libxlDomainObjPrivateAlloc,
|
|
|
|
.free = libxlDomainObjPrivateFree,
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
libxlDomainDeviceDefPostParse(virDomainDeviceDefPtr dev,
|
maint: avoid 'const fooPtr' in domain_conf
'const fooPtr' is the same as 'foo * const' (the pointer won't
change, but it's contents can). But in general, if an interface
is trying to be const-correct, it should be using 'const foo *'
(the pointer is to data that can't be changed).
Fix up offenders in src/conf/domain_conf, and their fallout.
Several things to note: virObjectLock() requires a non-const
argument; if this were C++, we could treat the locking field
as 'mutable' and allow locking an otherwise 'const' object, but
that is a more invasive change, so I instead dropped attempts
to be const-correct on domain lookup. virXMLPropString and
friends require a non-const xmlNodePtr - this is because libxml2
is not a const-correct library. We could make the src/util/virxml
wrappers cast away const, but I figured it was easier to not
try to mark xmlNodePtr as const. Finally, virDomainDeviceDefCopy
was a rather hard conversion - it calls virDomainDeviceDefPostParse,
which in turn in the xen driver was actually modifying the domain
outside of the current device being visited. We should not be
adding a device on the first per-device callback, but waiting until
after all per-device callbacks are complete.
* src/conf/domain_conf.h (virDomainObjListFindByID)
(virDomainObjListFindByUUID, virDomainObjListFindByName)
(virDomainObjAssignDef, virDomainObjListAdd): Drop attempt at
const.
(virDomainDeviceDefCopy): Use intended type.
(virDomainDeviceDefParse, virDomainDeviceDefPostParseCallback)
(virDomainVideoDefaultType, virDomainVideoDefaultRAM)
(virDomainChrGetDomainPtrs): Make const-correct.
* src/conf/domain_conf.c (virDomainObjListFindByID)
(virDomainObjListFindByUUID, virDomainObjListFindByName)
(virDomainDeviceDefCopy, virDomainObjListAdd)
(virDomainObjAssignDef, virDomainHostdevSubsysUsbDefParseXML)
(virDomainHostdevSubsysPciOrigStatesDefParseXML)
(virDomainHostdevSubsysPciDefParseXML)
(virDomainHostdevSubsysScsiDefParseXML)
(virDomainControllerModelTypeFromString)
(virDomainTPMDefParseXML, virDomainTimerDefParseXML)
(virDomainSoundCodecDefParseXML, virDomainSoundDefParseXML)
(virDomainWatchdogDefParseXML, virDomainRNGDefParseXML)
(virDomainMemballoonDefParseXML, virDomainNVRAMDefParseXML)
(virSysinfoParseXML, virDomainVideoAccelDefParseXML)
(virDomainVideoDefParseXML, virDomainHostdevDefParseXML)
(virDomainRedirdevDefParseXML)
(virDomainRedirFilterUsbDevDefParseXML)
(virDomainRedirFilterDefParseXML, virDomainIdMapEntrySort)
(virDomainIdmapDefParseXML, virDomainVcpuPinDefParseXML)
(virDiskNameToBusDeviceIndex, virDomainDeviceDefCopy)
(virDomainVideoDefaultType, virDomainHostdevAssignAddress)
(virDomainDeviceDefPostParseInternal, virDomainDeviceDefPostParse)
(virDomainChrGetDomainPtrs, virDomainControllerSCSINextUnit)
(virDomainSCSIDriveAddressIsUsed)
(virDomainDriveAddressIsUsedByDisk)
(virDomainDriveAddressIsUsedByHostdev): Fix fallout.
* src/openvz/openvz_driver.c (openvzDomainDeviceDefPostParse):
Likewise.
* src/libxl/libxl_domain.c (libxlDomainDeviceDefPostParse):
Likewise.
* src/qemu/qemu_domain.c (qemuDomainDeviceDefPostParse)
(qemuDomainDefaultNetModel): Likewise.
* src/lxc/lxc_domain.c (virLXCDomainDeviceDefPostParse):
Likewise.
* src/uml/uml_driver.c (umlDomainDeviceDefPostParse): Likewise.
* src/xen/xen_driver.c (xenDomainDeviceDefPostParse): Split...
(xenDomainDefPostParse): ...since per-device callback is not the
time to be adding a device.
Signed-off-by: Eric Blake <eblake@redhat.com>
2013-10-08 15:08:25 +00:00
|
|
|
const virDomainDef *def,
|
2013-08-30 17:18:31 +00:00
|
|
|
virCapsPtr caps ATTRIBUTE_UNUSED,
|
|
|
|
void *opaque ATTRIBUTE_UNUSED)
|
|
|
|
{
|
|
|
|
if (dev->type == VIR_DOMAIN_DEVICE_CHR &&
|
|
|
|
dev->data.chr->deviceType == VIR_DOMAIN_CHR_DEVICE_TYPE_CONSOLE &&
|
|
|
|
dev->data.chr->targetType == VIR_DOMAIN_CHR_CONSOLE_TARGET_TYPE_NONE &&
|
|
|
|
STRNEQ(def->os.type, "hvm"))
|
|
|
|
dev->data.chr->targetType = VIR_DOMAIN_CHR_CONSOLE_TARGET_TYPE_XEN;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
virDomainDefParserConfig libxlDomainDefParserConfig = {
|
|
|
|
.macPrefix = { 0x00, 0x16, 0x3e },
|
|
|
|
.devicesPostParseCallback = libxlDomainDeviceDefPostParse,
|
|
|
|
};
|
|
|
|
|
2014-01-16 23:16:37 +00:00
|
|
|
static const libxl_childproc_hooks libxl_child_hooks = {
|
|
|
|
#ifdef LIBXL_HAVE_SIGCHLD_OWNER_SELECTIVE_REAP
|
|
|
|
.chldowner = libxl_sigchld_owner_libxl_always_selective_reap,
|
|
|
|
#else
|
|
|
|
.chldowner = libxl_sigchld_owner_libxl,
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
2013-08-30 17:18:31 +00:00
|
|
|
int
|
|
|
|
libxlDomainObjPrivateInitCtx(virDomainObjPtr vm)
|
|
|
|
{
|
|
|
|
libxlDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
char *log_file;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
if (priv->ctx)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (virAsprintf(&log_file, "%s/%s.log", LIBXL_LOG_DIR, vm->def->name) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if ((priv->logger_file = fopen(log_file, "a")) == NULL) {
|
|
|
|
virReportSystemError(errno,
|
|
|
|
_("failed to open logfile %s"),
|
|
|
|
log_file);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
priv->logger =
|
|
|
|
(xentoollog_logger *)xtl_createlogger_stdiostream(priv->logger_file,
|
|
|
|
XTL_DEBUG, 0);
|
|
|
|
if (!priv->logger) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("cannot create libxenlight logger for domain %s"),
|
|
|
|
vm->def->name);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (libxl_ctx_alloc(&priv->ctx, LIBXL_VERSION, 0, priv->logger)) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Failed libxl context initialization"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
libxl_osevent_register_hooks(priv->ctx, &libxl_event_callbacks, priv);
|
2014-01-16 23:16:37 +00:00
|
|
|
libxl_childproc_setmode(priv->ctx, &libxl_child_hooks, priv);
|
2013-08-30 17:18:31 +00:00
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(log_file);
|
|
|
|
return ret;
|
|
|
|
}
|