mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-01 10:35:27 +00:00
1bf25ba249
http://www.uhv.edu/ac/newsletters/writing/grammartip2009.07.01.htm (and several other sites) give hints that 'onto' is best used if you can also add 'up' just before it and still make sense. In many cases in the code base, we really want the two-word form, or even a simplification to just 'on' or 'to'. * docs/hacking.html.in: Use correct 'on to'. * python/libvirt-override.c: Likewise. * src/lxc/lxc_controller.c: Likewise. * src/util/virpci.c: Likewise. * daemon/THREADS.txt: Use simpler 'on'. * docs/formatdomain.html.in: Better usage. * docs/internals/rpc.html.in: Likewise. * src/conf/domain_event.c: Likewise. * src/rpc/virnetclient.c: Likewise. * tests/qemumonitortestutils.c: Likewise. * HACKING: Regenerate. Signed-off-by: Eric Blake <eblake@redhat.com>
1676 lines
52 KiB
C
1676 lines
52 KiB
C
/*
|
|
* domain_event.c: domain event queue processing helpers
|
|
*
|
|
* Copyright (C) 2010-2013 Red Hat, Inc.
|
|
* Copyright (C) 2008 VirtualIron
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library. If not, see
|
|
* <http://www.gnu.org/licenses/>.
|
|
*
|
|
* Author: Ben Guthro
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include "domain_event.h"
|
|
#include "virlog.h"
|
|
#include "datatypes.h"
|
|
#include "viralloc.h"
|
|
#include "virerror.h"
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_NONE
|
|
|
|
struct _virDomainMeta {
|
|
int id;
|
|
char *name;
|
|
unsigned char uuid[VIR_UUID_BUFLEN];
|
|
};
|
|
typedef struct _virDomainMeta virDomainMeta;
|
|
typedef virDomainMeta *virDomainMetaPtr;
|
|
|
|
struct _virDomainEventCallbackList {
|
|
unsigned int nextID;
|
|
unsigned int count;
|
|
virDomainEventCallbackPtr *callbacks;
|
|
};
|
|
|
|
struct _virDomainEventQueue {
|
|
unsigned int count;
|
|
virDomainEventPtr *events;
|
|
};
|
|
|
|
struct _virDomainEventState {
|
|
/* The list of domain event callbacks */
|
|
virDomainEventCallbackListPtr callbacks;
|
|
/* The queue of domain events */
|
|
virDomainEventQueuePtr queue;
|
|
/* Timer for flushing events queue */
|
|
int timer;
|
|
/* Flag if we're in process of dispatching */
|
|
bool isDispatching;
|
|
virMutex lock;
|
|
};
|
|
|
|
struct _virDomainEventCallback {
|
|
int callbackID;
|
|
int eventID;
|
|
virConnectPtr conn;
|
|
virDomainMetaPtr dom;
|
|
virConnectDomainEventGenericCallback cb;
|
|
void *opaque;
|
|
virFreeCallback freecb;
|
|
int deleted;
|
|
};
|
|
|
|
struct _virDomainEvent {
|
|
int eventID;
|
|
|
|
virDomainMeta dom;
|
|
|
|
union {
|
|
struct {
|
|
int type;
|
|
int detail;
|
|
} lifecycle;
|
|
struct {
|
|
long long offset;
|
|
} rtcChange;
|
|
struct {
|
|
int action;
|
|
} watchdog;
|
|
struct {
|
|
char *srcPath;
|
|
char *devAlias;
|
|
int action;
|
|
char *reason;
|
|
} ioError;
|
|
struct {
|
|
int phase;
|
|
virDomainEventGraphicsAddressPtr local;
|
|
virDomainEventGraphicsAddressPtr remote;
|
|
char *authScheme;
|
|
virDomainEventGraphicsSubjectPtr subject;
|
|
} graphics;
|
|
struct {
|
|
char *path;
|
|
int type;
|
|
int status;
|
|
} blockJob;
|
|
struct {
|
|
char *oldSrcPath;
|
|
char *newSrcPath;
|
|
char *devAlias;
|
|
int reason;
|
|
} diskChange;
|
|
struct {
|
|
char *devAlias;
|
|
int reason;
|
|
} trayChange;
|
|
struct {
|
|
/* In unit of 1024 bytes */
|
|
unsigned long long actual;
|
|
} balloonChange;
|
|
} data;
|
|
};
|
|
|
|
/**
|
|
* virDomainEventCallbackListFree:
|
|
* @list: event callback list head
|
|
*
|
|
* Free the memory in the domain event callback list
|
|
*/
|
|
static void
|
|
virDomainEventCallbackListFree(virDomainEventCallbackListPtr list)
|
|
{
|
|
int i;
|
|
if (!list)
|
|
return;
|
|
|
|
for (i=0; i<list->count; i++) {
|
|
virFreeCallback freecb = list->callbacks[i]->freecb;
|
|
if (freecb)
|
|
(*freecb)(list->callbacks[i]->opaque);
|
|
VIR_FREE(list->callbacks[i]);
|
|
}
|
|
VIR_FREE(list);
|
|
}
|
|
|
|
|
|
/**
|
|
* virDomainEventCallbackListRemove:
|
|
* @conn: pointer to the connection
|
|
* @cbList: the list
|
|
* @callback: the callback to remove
|
|
*
|
|
* Internal function to remove a callback from a virDomainEventCallbackListPtr
|
|
*/
|
|
static int
|
|
virDomainEventCallbackListRemove(virConnectPtr conn,
|
|
virDomainEventCallbackListPtr cbList,
|
|
virConnectDomainEventCallback callback)
|
|
{
|
|
int ret = 0;
|
|
int i;
|
|
for (i = 0 ; i < cbList->count ; i++) {
|
|
if (cbList->callbacks[i]->cb == VIR_DOMAIN_EVENT_CALLBACK(callback) &&
|
|
cbList->callbacks[i]->eventID == VIR_DOMAIN_EVENT_ID_LIFECYCLE &&
|
|
cbList->callbacks[i]->conn == conn) {
|
|
virFreeCallback freecb = cbList->callbacks[i]->freecb;
|
|
if (freecb)
|
|
(*freecb)(cbList->callbacks[i]->opaque);
|
|
virObjectUnref(cbList->callbacks[i]->conn);
|
|
VIR_FREE(cbList->callbacks[i]);
|
|
|
|
if (i < (cbList->count - 1))
|
|
memmove(cbList->callbacks + i,
|
|
cbList->callbacks + i + 1,
|
|
sizeof(*(cbList->callbacks)) *
|
|
(cbList->count - (i + 1)));
|
|
|
|
if (VIR_REALLOC_N(cbList->callbacks,
|
|
cbList->count - 1) < 0) {
|
|
; /* Failure to reduce memory allocation isn't fatal */
|
|
}
|
|
cbList->count--;
|
|
|
|
for (i = 0 ; i < cbList->count ; i++) {
|
|
if (!cbList->callbacks[i]->deleted)
|
|
ret++;
|
|
}
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
_("could not find event callback for removal"));
|
|
return -1;
|
|
}
|
|
|
|
|
|
/**
|
|
* virDomainEventCallbackListRemoveID:
|
|
* @conn: pointer to the connection
|
|
* @cbList: the list
|
|
* @callback: the callback to remove
|
|
*
|
|
* Internal function to remove a callback from a virDomainEventCallbackListPtr
|
|
*/
|
|
static int
|
|
virDomainEventCallbackListRemoveID(virConnectPtr conn,
|
|
virDomainEventCallbackListPtr cbList,
|
|
int callbackID)
|
|
{
|
|
int ret = 0;
|
|
int i;
|
|
for (i = 0 ; i < cbList->count ; i++) {
|
|
if (cbList->callbacks[i]->callbackID == callbackID &&
|
|
cbList->callbacks[i]->conn == conn) {
|
|
virFreeCallback freecb = cbList->callbacks[i]->freecb;
|
|
if (freecb)
|
|
(*freecb)(cbList->callbacks[i]->opaque);
|
|
virObjectUnref(cbList->callbacks[i]->conn);
|
|
VIR_FREE(cbList->callbacks[i]);
|
|
|
|
if (i < (cbList->count - 1))
|
|
memmove(cbList->callbacks + i,
|
|
cbList->callbacks + i + 1,
|
|
sizeof(*(cbList->callbacks)) *
|
|
(cbList->count - (i + 1)));
|
|
|
|
if (VIR_REALLOC_N(cbList->callbacks,
|
|
cbList->count - 1) < 0) {
|
|
; /* Failure to reduce memory allocation isn't fatal */
|
|
}
|
|
cbList->count--;
|
|
|
|
for (i = 0 ; i < cbList->count ; i++) {
|
|
if (!cbList->callbacks[i]->deleted)
|
|
ret++;
|
|
}
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
_("could not find event callback for removal"));
|
|
return -1;
|
|
}
|
|
|
|
|
|
static int
|
|
virDomainEventCallbackListMarkDelete(virConnectPtr conn,
|
|
virDomainEventCallbackListPtr cbList,
|
|
virConnectDomainEventCallback callback)
|
|
{
|
|
int ret = 0;
|
|
int i;
|
|
for (i = 0 ; i < cbList->count ; i++) {
|
|
if (cbList->callbacks[i]->cb == VIR_DOMAIN_EVENT_CALLBACK(callback) &&
|
|
cbList->callbacks[i]->eventID == VIR_DOMAIN_EVENT_ID_LIFECYCLE &&
|
|
cbList->callbacks[i]->conn == conn) {
|
|
cbList->callbacks[i]->deleted = 1;
|
|
for (i = 0 ; i < cbList->count ; i++) {
|
|
if (!cbList->callbacks[i]->deleted)
|
|
ret++;
|
|
}
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
_("could not find event callback for deletion"));
|
|
return -1;
|
|
}
|
|
|
|
|
|
static int
|
|
virDomainEventCallbackListMarkDeleteID(virConnectPtr conn,
|
|
virDomainEventCallbackListPtr cbList,
|
|
int callbackID)
|
|
{
|
|
int ret = 0;
|
|
int i;
|
|
for (i = 0 ; i < cbList->count ; i++) {
|
|
if (cbList->callbacks[i]->callbackID == callbackID &&
|
|
cbList->callbacks[i]->conn == conn) {
|
|
cbList->callbacks[i]->deleted = 1;
|
|
for (i = 0 ; i < cbList->count ; i++) {
|
|
if (!cbList->callbacks[i]->deleted)
|
|
ret++;
|
|
}
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
_("could not find event callback for deletion"));
|
|
return -1;
|
|
}
|
|
|
|
|
|
static int
|
|
virDomainEventCallbackListPurgeMarked(virDomainEventCallbackListPtr cbList)
|
|
{
|
|
int old_count = cbList->count;
|
|
int i;
|
|
for (i = 0 ; i < cbList->count ; i++) {
|
|
if (cbList->callbacks[i]->deleted) {
|
|
virFreeCallback freecb = cbList->callbacks[i]->freecb;
|
|
if (freecb)
|
|
(*freecb)(cbList->callbacks[i]->opaque);
|
|
virObjectUnref(cbList->callbacks[i]->conn);
|
|
VIR_FREE(cbList->callbacks[i]);
|
|
|
|
if (i < (cbList->count - 1))
|
|
memmove(cbList->callbacks + i,
|
|
cbList->callbacks + i + 1,
|
|
sizeof(*(cbList->callbacks)) *
|
|
(cbList->count - (i + 1)));
|
|
cbList->count--;
|
|
i--;
|
|
}
|
|
}
|
|
if (cbList->count < old_count &&
|
|
VIR_REALLOC_N(cbList->callbacks, cbList->count) < 0) {
|
|
; /* Failure to reduce memory allocation isn't fatal */
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
/**
|
|
* virDomainEventCallbackListAddID:
|
|
* @conn: pointer to the connection
|
|
* @cbList: the list
|
|
* @eventID: the event ID
|
|
* @callback: the callback to add
|
|
* @opaque: opaque data tio pass to callback
|
|
* @callbackID: filled with callback ID
|
|
*
|
|
* Internal function to add a callback from a virDomainEventCallbackListPtr
|
|
*/
|
|
static int
|
|
virDomainEventCallbackListAddID(virConnectPtr conn,
|
|
virDomainEventCallbackListPtr cbList,
|
|
virDomainPtr dom,
|
|
int eventID,
|
|
virConnectDomainEventGenericCallback callback,
|
|
void *opaque,
|
|
virFreeCallback freecb,
|
|
int *callbackID)
|
|
{
|
|
virDomainEventCallbackPtr event;
|
|
int i;
|
|
int ret = 0;
|
|
|
|
/* Check incoming */
|
|
if (!cbList) {
|
|
return -1;
|
|
}
|
|
|
|
/* check if we already have this callback on our list */
|
|
for (i = 0 ; i < cbList->count ; i++) {
|
|
if (cbList->callbacks[i]->cb == VIR_DOMAIN_EVENT_CALLBACK(callback) &&
|
|
cbList->callbacks[i]->eventID == eventID &&
|
|
cbList->callbacks[i]->conn == conn &&
|
|
((dom && cbList->callbacks[i]->dom &&
|
|
memcmp(cbList->callbacks[i]->dom->uuid,
|
|
dom->uuid, VIR_UUID_BUFLEN) == 0) ||
|
|
(!dom && !cbList->callbacks[i]->dom))) {
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
_("event callback already tracked"));
|
|
return -1;
|
|
}
|
|
}
|
|
/* Allocate new event */
|
|
if (VIR_ALLOC(event) < 0)
|
|
goto no_memory;
|
|
event->conn = conn;
|
|
event->cb = callback;
|
|
event->eventID = eventID;
|
|
event->opaque = opaque;
|
|
event->freecb = freecb;
|
|
|
|
if (dom) {
|
|
if (VIR_ALLOC(event->dom) < 0)
|
|
goto no_memory;
|
|
if (!(event->dom->name = strdup(dom->name)))
|
|
goto no_memory;
|
|
memcpy(event->dom->uuid, dom->uuid, VIR_UUID_BUFLEN);
|
|
event->dom->id = dom->id;
|
|
}
|
|
|
|
/* Make space on list */
|
|
if (VIR_REALLOC_N(cbList->callbacks, cbList->count + 1) < 0)
|
|
goto no_memory;
|
|
|
|
virObjectRef(event->conn);
|
|
|
|
cbList->callbacks[cbList->count] = event;
|
|
cbList->count++;
|
|
|
|
event->callbackID = cbList->nextID++;
|
|
|
|
for (i = 0 ; i < cbList->count ; i++) {
|
|
if (cbList->callbacks[i]->eventID == eventID &&
|
|
cbList->callbacks[i]->conn == conn &&
|
|
!cbList->callbacks[i]->deleted)
|
|
ret++;
|
|
}
|
|
|
|
if (callbackID)
|
|
*callbackID = event->callbackID;
|
|
|
|
return ret;
|
|
|
|
no_memory:
|
|
virReportOOMError();
|
|
|
|
if (event) {
|
|
if (event->dom)
|
|
VIR_FREE(event->dom->name);
|
|
VIR_FREE(event->dom);
|
|
}
|
|
VIR_FREE(event);
|
|
return -1;
|
|
}
|
|
|
|
|
|
/**
|
|
* virDomainEventCallbackListAdd:
|
|
* @conn: pointer to the connection
|
|
* @cbList: the list
|
|
* @callback: the callback to add
|
|
* @opaque: opaque data tio pass to callback
|
|
*
|
|
* Internal function to add a callback from a virDomainEventCallbackListPtr
|
|
*/
|
|
static int
|
|
virDomainEventCallbackListAdd(virConnectPtr conn,
|
|
virDomainEventCallbackListPtr cbList,
|
|
virConnectDomainEventCallback callback,
|
|
void *opaque,
|
|
virFreeCallback freecb)
|
|
{
|
|
return virDomainEventCallbackListAddID(conn, cbList, NULL,
|
|
VIR_DOMAIN_EVENT_ID_LIFECYCLE,
|
|
VIR_DOMAIN_EVENT_CALLBACK(callback),
|
|
opaque, freecb, NULL);
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
virDomainEventCallbackListEventID(virConnectPtr conn,
|
|
virDomainEventCallbackListPtr cbList,
|
|
int callbackID)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0 ; i < cbList->count ; i++) {
|
|
if (cbList->callbacks[i]->deleted)
|
|
continue;
|
|
|
|
if (cbList->callbacks[i]->callbackID == callbackID &&
|
|
cbList->callbacks[i]->conn == conn)
|
|
return cbList->callbacks[i]->eventID;
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
|
|
void virDomainEventFree(virDomainEventPtr event)
|
|
{
|
|
if (!event)
|
|
return;
|
|
|
|
switch (event->eventID) {
|
|
case VIR_DOMAIN_EVENT_ID_IO_ERROR_REASON:
|
|
case VIR_DOMAIN_EVENT_ID_IO_ERROR:
|
|
VIR_FREE(event->data.ioError.srcPath);
|
|
VIR_FREE(event->data.ioError.devAlias);
|
|
VIR_FREE(event->data.ioError.reason);
|
|
break;
|
|
|
|
case VIR_DOMAIN_EVENT_ID_GRAPHICS:
|
|
if (event->data.graphics.local) {
|
|
VIR_FREE(event->data.graphics.local->node);
|
|
VIR_FREE(event->data.graphics.local->service);
|
|
VIR_FREE(event->data.graphics.local);
|
|
}
|
|
if (event->data.graphics.remote) {
|
|
VIR_FREE(event->data.graphics.remote->node);
|
|
VIR_FREE(event->data.graphics.remote->service);
|
|
VIR_FREE(event->data.graphics.remote);
|
|
}
|
|
VIR_FREE(event->data.graphics.authScheme);
|
|
if (event->data.graphics.subject) {
|
|
int i;
|
|
for (i = 0 ; i < event->data.graphics.subject->nidentity ; i++) {
|
|
VIR_FREE(event->data.graphics.subject->identities[i].type);
|
|
VIR_FREE(event->data.graphics.subject->identities[i].name);
|
|
}
|
|
VIR_FREE(event->data.graphics.subject);
|
|
}
|
|
break;
|
|
|
|
case VIR_DOMAIN_EVENT_ID_BLOCK_JOB:
|
|
VIR_FREE(event->data.blockJob.path);
|
|
break;
|
|
|
|
case VIR_DOMAIN_EVENT_ID_DISK_CHANGE:
|
|
VIR_FREE(event->data.diskChange.oldSrcPath);
|
|
VIR_FREE(event->data.diskChange.newSrcPath);
|
|
VIR_FREE(event->data.diskChange.devAlias);
|
|
break;
|
|
case VIR_DOMAIN_EVENT_ID_TRAY_CHANGE:
|
|
VIR_FREE(event->data.trayChange.devAlias);
|
|
break;
|
|
}
|
|
|
|
VIR_FREE(event->dom.name);
|
|
VIR_FREE(event);
|
|
}
|
|
|
|
/**
|
|
* virDomainEventQueueClear:
|
|
* @queue: pointer to the queue
|
|
*
|
|
* Removes all elements from the queue
|
|
*/
|
|
static void
|
|
virDomainEventQueueClear(virDomainEventQueuePtr queue)
|
|
{
|
|
int i;
|
|
if (!queue)
|
|
return;
|
|
|
|
for (i = 0; i < queue->count ; i++) {
|
|
virDomainEventFree(queue->events[i]);
|
|
}
|
|
VIR_FREE(queue->events);
|
|
queue->count = 0;
|
|
}
|
|
|
|
/**
|
|
* virDomainEventQueueFree:
|
|
* @queue: pointer to the queue
|
|
*
|
|
* Free the memory in the queue. We process this like a list here
|
|
*/
|
|
static void
|
|
virDomainEventQueueFree(virDomainEventQueuePtr queue)
|
|
{
|
|
if (!queue)
|
|
return;
|
|
|
|
virDomainEventQueueClear(queue);
|
|
VIR_FREE(queue);
|
|
}
|
|
|
|
static virDomainEventQueuePtr
|
|
virDomainEventQueueNew(void)
|
|
{
|
|
virDomainEventQueuePtr ret;
|
|
|
|
if (VIR_ALLOC(ret) < 0) {
|
|
virReportOOMError();
|
|
return NULL;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void
|
|
virDomainEventStateLock(virDomainEventStatePtr state)
|
|
{
|
|
virMutexLock(&state->lock);
|
|
}
|
|
|
|
static void
|
|
virDomainEventStateUnlock(virDomainEventStatePtr state)
|
|
{
|
|
virMutexUnlock(&state->lock);
|
|
}
|
|
|
|
/**
|
|
* virDomainEventStateFree:
|
|
* @list: virDomainEventStatePtr to free
|
|
*
|
|
* Free a virDomainEventStatePtr and its members, and unregister the timer.
|
|
*/
|
|
void
|
|
virDomainEventStateFree(virDomainEventStatePtr state)
|
|
{
|
|
if (!state)
|
|
return;
|
|
|
|
virDomainEventCallbackListFree(state->callbacks);
|
|
virDomainEventQueueFree(state->queue);
|
|
|
|
if (state->timer != -1)
|
|
virEventRemoveTimeout(state->timer);
|
|
|
|
virMutexDestroy(&state->lock);
|
|
VIR_FREE(state);
|
|
}
|
|
|
|
|
|
static void virDomainEventStateFlush(virDomainEventStatePtr state);
|
|
|
|
static void
|
|
virDomainEventTimer(int timer ATTRIBUTE_UNUSED, void *opaque)
|
|
{
|
|
virDomainEventStatePtr state = opaque;
|
|
|
|
virDomainEventStateFlush(state);
|
|
}
|
|
|
|
/**
|
|
* virDomainEventStateNew:
|
|
*/
|
|
virDomainEventStatePtr
|
|
virDomainEventStateNew(void)
|
|
{
|
|
virDomainEventStatePtr state = NULL;
|
|
|
|
if (VIR_ALLOC(state) < 0) {
|
|
virReportOOMError();
|
|
goto error;
|
|
}
|
|
|
|
if (virMutexInit(&state->lock) < 0) {
|
|
virReportSystemError(errno, "%s",
|
|
_("unable to initialize state mutex"));
|
|
VIR_FREE(state);
|
|
goto error;
|
|
}
|
|
|
|
if (VIR_ALLOC(state->callbacks) < 0) {
|
|
virReportOOMError();
|
|
goto error;
|
|
}
|
|
|
|
if (!(state->queue = virDomainEventQueueNew()))
|
|
goto error;
|
|
|
|
state->timer = -1;
|
|
|
|
return state;
|
|
|
|
error:
|
|
virDomainEventStateFree(state);
|
|
return NULL;
|
|
}
|
|
|
|
static virDomainEventPtr virDomainEventNewInternal(int eventID,
|
|
int id,
|
|
const char *name,
|
|
const unsigned char *uuid)
|
|
{
|
|
virDomainEventPtr event;
|
|
|
|
if (VIR_ALLOC(event) < 0) {
|
|
virReportOOMError();
|
|
return NULL;
|
|
}
|
|
|
|
event->eventID = eventID;
|
|
if (!(event->dom.name = strdup(name))) {
|
|
virReportOOMError();
|
|
VIR_FREE(event);
|
|
return NULL;
|
|
}
|
|
event->dom.id = id;
|
|
memcpy(event->dom.uuid, uuid, VIR_UUID_BUFLEN);
|
|
|
|
return event;
|
|
}
|
|
|
|
virDomainEventPtr virDomainEventNew(int id, const char *name,
|
|
const unsigned char *uuid,
|
|
int type, int detail)
|
|
{
|
|
virDomainEventPtr event = virDomainEventNewInternal(VIR_DOMAIN_EVENT_ID_LIFECYCLE,
|
|
id, name, uuid);
|
|
|
|
if (event) {
|
|
event->data.lifecycle.type = type;
|
|
event->data.lifecycle.detail = detail;
|
|
}
|
|
|
|
return event;
|
|
}
|
|
|
|
virDomainEventPtr virDomainEventNewFromDom(virDomainPtr dom, int type, int detail)
|
|
{
|
|
return virDomainEventNew(dom->id, dom->name, dom->uuid, type, detail);
|
|
}
|
|
|
|
virDomainEventPtr virDomainEventNewFromObj(virDomainObjPtr obj, int type, int detail)
|
|
{
|
|
return virDomainEventNewFromDef(obj->def, type, detail);
|
|
}
|
|
|
|
virDomainEventPtr virDomainEventNewFromDef(virDomainDefPtr def, int type, int detail)
|
|
{
|
|
return virDomainEventNew(def->id, def->name, def->uuid, type, detail);
|
|
}
|
|
|
|
virDomainEventPtr virDomainEventRebootNew(int id, const char *name,
|
|
const unsigned char *uuid)
|
|
{
|
|
return virDomainEventNewInternal(VIR_DOMAIN_EVENT_ID_REBOOT,
|
|
id, name, uuid);
|
|
}
|
|
|
|
virDomainEventPtr virDomainEventRebootNewFromDom(virDomainPtr dom)
|
|
{
|
|
return virDomainEventNewInternal(VIR_DOMAIN_EVENT_ID_REBOOT,
|
|
dom->id, dom->name, dom->uuid);
|
|
}
|
|
|
|
virDomainEventPtr virDomainEventRebootNewFromObj(virDomainObjPtr obj)
|
|
{
|
|
return virDomainEventNewInternal(VIR_DOMAIN_EVENT_ID_REBOOT,
|
|
obj->def->id, obj->def->name, obj->def->uuid);
|
|
}
|
|
|
|
virDomainEventPtr virDomainEventRTCChangeNewFromDom(virDomainPtr dom,
|
|
long long offset)
|
|
{
|
|
virDomainEventPtr ev =
|
|
virDomainEventNewInternal(VIR_DOMAIN_EVENT_ID_RTC_CHANGE,
|
|
dom->id, dom->name, dom->uuid);
|
|
|
|
if (ev)
|
|
ev->data.rtcChange.offset = offset;
|
|
|
|
return ev;
|
|
}
|
|
virDomainEventPtr virDomainEventRTCChangeNewFromObj(virDomainObjPtr obj,
|
|
long long offset)
|
|
{
|
|
virDomainEventPtr ev =
|
|
virDomainEventNewInternal(VIR_DOMAIN_EVENT_ID_RTC_CHANGE,
|
|
obj->def->id, obj->def->name, obj->def->uuid);
|
|
|
|
if (ev)
|
|
ev->data.rtcChange.offset = offset;
|
|
|
|
return ev;
|
|
}
|
|
|
|
virDomainEventPtr virDomainEventWatchdogNewFromDom(virDomainPtr dom,
|
|
int action)
|
|
{
|
|
virDomainEventPtr ev =
|
|
virDomainEventNewInternal(VIR_DOMAIN_EVENT_ID_WATCHDOG,
|
|
dom->id, dom->name, dom->uuid);
|
|
|
|
if (ev)
|
|
ev->data.watchdog.action = action;
|
|
|
|
return ev;
|
|
}
|
|
virDomainEventPtr virDomainEventWatchdogNewFromObj(virDomainObjPtr obj,
|
|
int action)
|
|
{
|
|
virDomainEventPtr ev =
|
|
virDomainEventNewInternal(VIR_DOMAIN_EVENT_ID_WATCHDOG,
|
|
obj->def->id, obj->def->name, obj->def->uuid);
|
|
|
|
if (ev)
|
|
ev->data.watchdog.action = action;
|
|
|
|
return ev;
|
|
}
|
|
|
|
static virDomainEventPtr virDomainEventIOErrorNewFromDomImpl(int event,
|
|
virDomainPtr dom,
|
|
const char *srcPath,
|
|
const char *devAlias,
|
|
int action,
|
|
const char *reason)
|
|
{
|
|
virDomainEventPtr ev =
|
|
virDomainEventNewInternal(event,
|
|
dom->id, dom->name, dom->uuid);
|
|
|
|
if (ev) {
|
|
ev->data.ioError.action = action;
|
|
if (!(ev->data.ioError.srcPath = strdup(srcPath)) ||
|
|
!(ev->data.ioError.devAlias = strdup(devAlias)) ||
|
|
(reason && !(ev->data.ioError.reason = strdup(reason)))) {
|
|
virDomainEventFree(ev);
|
|
ev = NULL;
|
|
}
|
|
}
|
|
|
|
return ev;
|
|
}
|
|
|
|
static virDomainEventPtr virDomainEventIOErrorNewFromObjImpl(int event,
|
|
virDomainObjPtr obj,
|
|
const char *srcPath,
|
|
const char *devAlias,
|
|
int action,
|
|
const char *reason)
|
|
{
|
|
virDomainEventPtr ev =
|
|
virDomainEventNewInternal(event,
|
|
obj->def->id, obj->def->name, obj->def->uuid);
|
|
|
|
if (ev) {
|
|
ev->data.ioError.action = action;
|
|
if (!(ev->data.ioError.srcPath = strdup(srcPath)) ||
|
|
!(ev->data.ioError.devAlias = strdup(devAlias)) ||
|
|
(reason && !(ev->data.ioError.reason = strdup(reason)))) {
|
|
virDomainEventFree(ev);
|
|
ev = NULL;
|
|
}
|
|
}
|
|
|
|
return ev;
|
|
}
|
|
|
|
virDomainEventPtr virDomainEventIOErrorNewFromDom(virDomainPtr dom,
|
|
const char *srcPath,
|
|
const char *devAlias,
|
|
int action)
|
|
{
|
|
return virDomainEventIOErrorNewFromDomImpl(VIR_DOMAIN_EVENT_ID_IO_ERROR,
|
|
dom, srcPath, devAlias,
|
|
action, NULL);
|
|
}
|
|
|
|
virDomainEventPtr virDomainEventIOErrorNewFromObj(virDomainObjPtr obj,
|
|
const char *srcPath,
|
|
const char *devAlias,
|
|
int action)
|
|
{
|
|
return virDomainEventIOErrorNewFromObjImpl(VIR_DOMAIN_EVENT_ID_IO_ERROR,
|
|
obj, srcPath, devAlias,
|
|
action, NULL);
|
|
}
|
|
|
|
virDomainEventPtr virDomainEventIOErrorReasonNewFromDom(virDomainPtr dom,
|
|
const char *srcPath,
|
|
const char *devAlias,
|
|
int action,
|
|
const char *reason)
|
|
{
|
|
return virDomainEventIOErrorNewFromDomImpl(VIR_DOMAIN_EVENT_ID_IO_ERROR_REASON,
|
|
dom, srcPath, devAlias,
|
|
action, reason);
|
|
}
|
|
|
|
virDomainEventPtr virDomainEventIOErrorReasonNewFromObj(virDomainObjPtr obj,
|
|
const char *srcPath,
|
|
const char *devAlias,
|
|
int action,
|
|
const char *reason)
|
|
{
|
|
return virDomainEventIOErrorNewFromObjImpl(VIR_DOMAIN_EVENT_ID_IO_ERROR_REASON,
|
|
obj, srcPath, devAlias,
|
|
action, reason);
|
|
}
|
|
|
|
|
|
virDomainEventPtr virDomainEventGraphicsNewFromDom(virDomainPtr dom,
|
|
int phase,
|
|
virDomainEventGraphicsAddressPtr local,
|
|
virDomainEventGraphicsAddressPtr remote,
|
|
const char *authScheme,
|
|
virDomainEventGraphicsSubjectPtr subject)
|
|
{
|
|
virDomainEventPtr ev =
|
|
virDomainEventNewInternal(VIR_DOMAIN_EVENT_ID_GRAPHICS,
|
|
dom->id, dom->name, dom->uuid);
|
|
|
|
if (ev) {
|
|
ev->data.graphics.phase = phase;
|
|
if (!(ev->data.graphics.authScheme = strdup(authScheme))) {
|
|
virDomainEventFree(ev);
|
|
return NULL;
|
|
}
|
|
ev->data.graphics.local = local;
|
|
ev->data.graphics.remote = remote;
|
|
ev->data.graphics.subject = subject;
|
|
}
|
|
|
|
return ev;
|
|
}
|
|
|
|
virDomainEventPtr virDomainEventGraphicsNewFromObj(virDomainObjPtr obj,
|
|
int phase,
|
|
virDomainEventGraphicsAddressPtr local,
|
|
virDomainEventGraphicsAddressPtr remote,
|
|
const char *authScheme,
|
|
virDomainEventGraphicsSubjectPtr subject)
|
|
{
|
|
virDomainEventPtr ev =
|
|
virDomainEventNewInternal(VIR_DOMAIN_EVENT_ID_GRAPHICS,
|
|
obj->def->id, obj->def->name, obj->def->uuid);
|
|
|
|
if (ev) {
|
|
ev->data.graphics.phase = phase;
|
|
if (!(ev->data.graphics.authScheme = strdup(authScheme))) {
|
|
virDomainEventFree(ev);
|
|
return NULL;
|
|
}
|
|
ev->data.graphics.local = local;
|
|
ev->data.graphics.remote = remote;
|
|
ev->data.graphics.subject = subject;
|
|
}
|
|
|
|
return ev;
|
|
}
|
|
|
|
static virDomainEventPtr
|
|
virDomainEventBlockJobNew(int id, const char *name, unsigned char *uuid,
|
|
const char *path, int type, int status)
|
|
{
|
|
virDomainEventPtr ev =
|
|
virDomainEventNewInternal(VIR_DOMAIN_EVENT_ID_BLOCK_JOB,
|
|
id, name, uuid);
|
|
|
|
if (ev) {
|
|
if (!(ev->data.blockJob.path = strdup(path))) {
|
|
virReportOOMError();
|
|
virDomainEventFree(ev);
|
|
return NULL;
|
|
}
|
|
ev->data.blockJob.type = type;
|
|
ev->data.blockJob.status = status;
|
|
}
|
|
|
|
return ev;
|
|
}
|
|
|
|
virDomainEventPtr virDomainEventBlockJobNewFromObj(virDomainObjPtr obj,
|
|
const char *path,
|
|
int type,
|
|
int status)
|
|
{
|
|
return virDomainEventBlockJobNew(obj->def->id, obj->def->name,
|
|
obj->def->uuid, path, type, status);
|
|
}
|
|
|
|
virDomainEventPtr virDomainEventBlockJobNewFromDom(virDomainPtr dom,
|
|
const char *path,
|
|
int type,
|
|
int status)
|
|
{
|
|
return virDomainEventBlockJobNew(dom->id, dom->name, dom->uuid,
|
|
path, type, status);
|
|
}
|
|
|
|
virDomainEventPtr virDomainEventControlErrorNewFromDom(virDomainPtr dom)
|
|
{
|
|
virDomainEventPtr ev =
|
|
virDomainEventNewInternal(VIR_DOMAIN_EVENT_ID_CONTROL_ERROR,
|
|
dom->id, dom->name, dom->uuid);
|
|
return ev;
|
|
}
|
|
|
|
|
|
virDomainEventPtr virDomainEventControlErrorNewFromObj(virDomainObjPtr obj)
|
|
{
|
|
virDomainEventPtr ev =
|
|
virDomainEventNewInternal(VIR_DOMAIN_EVENT_ID_CONTROL_ERROR,
|
|
obj->def->id, obj->def->name, obj->def->uuid);
|
|
return ev;
|
|
}
|
|
|
|
static virDomainEventPtr
|
|
virDomainEventDiskChangeNew(int id, const char *name,
|
|
unsigned char *uuid,
|
|
const char *oldSrcPath,
|
|
const char *newSrcPath,
|
|
const char *devAlias, int reason)
|
|
{
|
|
virDomainEventPtr ev =
|
|
virDomainEventNewInternal(VIR_DOMAIN_EVENT_ID_DISK_CHANGE,
|
|
id, name, uuid);
|
|
|
|
if (ev) {
|
|
if (!(ev->data.diskChange.devAlias = strdup(devAlias)))
|
|
goto error;
|
|
|
|
if (oldSrcPath &&
|
|
!(ev->data.diskChange.oldSrcPath = strdup(oldSrcPath)))
|
|
goto error;
|
|
|
|
if (newSrcPath &&
|
|
!(ev->data.diskChange.newSrcPath = strdup(newSrcPath)))
|
|
goto error;
|
|
|
|
ev->data.diskChange.reason = reason;
|
|
}
|
|
|
|
return ev;
|
|
|
|
error:
|
|
virReportOOMError();
|
|
virDomainEventFree(ev);
|
|
return NULL;
|
|
}
|
|
|
|
virDomainEventPtr virDomainEventDiskChangeNewFromObj(virDomainObjPtr obj,
|
|
const char *oldSrcPath,
|
|
const char *newSrcPath,
|
|
const char *devAlias,
|
|
int reason)
|
|
{
|
|
return virDomainEventDiskChangeNew(obj->def->id, obj->def->name,
|
|
obj->def->uuid, oldSrcPath,
|
|
newSrcPath, devAlias, reason);
|
|
}
|
|
|
|
virDomainEventPtr virDomainEventDiskChangeNewFromDom(virDomainPtr dom,
|
|
const char *oldSrcPath,
|
|
const char *newSrcPath,
|
|
const char *devAlias,
|
|
int reason)
|
|
{
|
|
return virDomainEventDiskChangeNew(dom->id, dom->name, dom->uuid,
|
|
oldSrcPath, newSrcPath,
|
|
devAlias, reason);
|
|
}
|
|
|
|
static virDomainEventPtr
|
|
virDomainEventTrayChangeNew(int id, const char *name,
|
|
unsigned char *uuid,
|
|
const char *devAlias,
|
|
int reason)
|
|
{
|
|
virDomainEventPtr ev =
|
|
virDomainEventNewInternal(VIR_DOMAIN_EVENT_ID_TRAY_CHANGE,
|
|
id, name, uuid);
|
|
|
|
if (ev) {
|
|
if (!(ev->data.trayChange.devAlias = strdup(devAlias)))
|
|
goto error;
|
|
|
|
ev->data.trayChange.reason = reason;
|
|
}
|
|
|
|
return ev;
|
|
|
|
error:
|
|
virReportOOMError();
|
|
virDomainEventFree(ev);
|
|
return NULL;
|
|
}
|
|
|
|
virDomainEventPtr virDomainEventTrayChangeNewFromObj(virDomainObjPtr obj,
|
|
const char *devAlias,
|
|
int reason)
|
|
{
|
|
return virDomainEventTrayChangeNew(obj->def->id,
|
|
obj->def->name,
|
|
obj->def->uuid,
|
|
devAlias,
|
|
reason);
|
|
}
|
|
|
|
virDomainEventPtr virDomainEventTrayChangeNewFromDom(virDomainPtr dom,
|
|
const char *devAlias,
|
|
int reason)
|
|
{
|
|
return virDomainEventTrayChangeNew(dom->id, dom->name, dom->uuid,
|
|
devAlias, reason);
|
|
}
|
|
|
|
static virDomainEventPtr
|
|
virDomainEventPMWakeupNew(int id, const char *name,
|
|
unsigned char *uuid)
|
|
{
|
|
virDomainEventPtr ev =
|
|
virDomainEventNewInternal(VIR_DOMAIN_EVENT_ID_PMWAKEUP,
|
|
id, name, uuid);
|
|
|
|
return ev;
|
|
}
|
|
|
|
virDomainEventPtr
|
|
virDomainEventPMWakeupNewFromObj(virDomainObjPtr obj)
|
|
{
|
|
return virDomainEventPMWakeupNew(obj->def->id,
|
|
obj->def->name,
|
|
obj->def->uuid);
|
|
}
|
|
|
|
virDomainEventPtr
|
|
virDomainEventPMWakeupNewFromDom(virDomainPtr dom)
|
|
{
|
|
return virDomainEventPMWakeupNew(dom->id, dom->name, dom->uuid);
|
|
}
|
|
|
|
static virDomainEventPtr
|
|
virDomainEventPMSuspendNew(int id, const char *name,
|
|
unsigned char *uuid)
|
|
{
|
|
virDomainEventPtr ev =
|
|
virDomainEventNewInternal(VIR_DOMAIN_EVENT_ID_PMSUSPEND,
|
|
id, name, uuid);
|
|
|
|
return ev;
|
|
}
|
|
|
|
virDomainEventPtr
|
|
virDomainEventPMSuspendNewFromObj(virDomainObjPtr obj)
|
|
{
|
|
return virDomainEventPMSuspendNew(obj->def->id,
|
|
obj->def->name,
|
|
obj->def->uuid);
|
|
}
|
|
|
|
virDomainEventPtr
|
|
virDomainEventPMSuspendNewFromDom(virDomainPtr dom)
|
|
{
|
|
return virDomainEventPMSuspendNew(dom->id, dom->name, dom->uuid);
|
|
}
|
|
|
|
static virDomainEventPtr
|
|
virDomainEventPMSuspendDiskNew(int id, const char *name,
|
|
unsigned char *uuid)
|
|
{
|
|
virDomainEventPtr ev =
|
|
virDomainEventNewInternal(VIR_DOMAIN_EVENT_ID_PMSUSPEND_DISK,
|
|
id, name, uuid);
|
|
return ev;
|
|
}
|
|
|
|
virDomainEventPtr
|
|
virDomainEventPMSuspendDiskNewFromObj(virDomainObjPtr obj)
|
|
{
|
|
return virDomainEventPMSuspendDiskNew(obj->def->id,
|
|
obj->def->name,
|
|
obj->def->uuid);
|
|
}
|
|
|
|
virDomainEventPtr
|
|
virDomainEventPMSuspendDiskNewFromDom(virDomainPtr dom)
|
|
{
|
|
return virDomainEventPMSuspendDiskNew(dom->id, dom->name, dom->uuid);
|
|
}
|
|
|
|
virDomainEventPtr virDomainEventBalloonChangeNewFromDom(virDomainPtr dom,
|
|
unsigned long long actual)
|
|
{
|
|
virDomainEventPtr ev =
|
|
virDomainEventNewInternal(VIR_DOMAIN_EVENT_ID_BALLOON_CHANGE,
|
|
dom->id, dom->name, dom->uuid);
|
|
|
|
if (ev)
|
|
ev->data.balloonChange.actual = actual;
|
|
|
|
return ev;
|
|
}
|
|
virDomainEventPtr virDomainEventBalloonChangeNewFromObj(virDomainObjPtr obj,
|
|
unsigned long long actual)
|
|
{
|
|
virDomainEventPtr ev =
|
|
virDomainEventNewInternal(VIR_DOMAIN_EVENT_ID_BALLOON_CHANGE,
|
|
obj->def->id, obj->def->name, obj->def->uuid);
|
|
|
|
if (ev)
|
|
ev->data.balloonChange.actual = actual;
|
|
|
|
return ev;
|
|
}
|
|
|
|
/**
|
|
* virDomainEventQueuePush:
|
|
* @evtQueue: the dom event queue
|
|
* @event: the event to add
|
|
*
|
|
* Internal function to push to the back of a virDomainEventQueue
|
|
*
|
|
* Returns: 0 on success, -1 on failure
|
|
*/
|
|
static int
|
|
virDomainEventQueuePush(virDomainEventQueuePtr evtQueue,
|
|
virDomainEventPtr event)
|
|
{
|
|
if (!evtQueue) {
|
|
return -1;
|
|
}
|
|
|
|
/* Make space on queue */
|
|
if (VIR_REALLOC_N(evtQueue->events,
|
|
evtQueue->count + 1) < 0) {
|
|
virReportOOMError();
|
|
return -1;
|
|
}
|
|
|
|
evtQueue->events[evtQueue->count] = event;
|
|
evtQueue->count++;
|
|
return 0;
|
|
}
|
|
|
|
|
|
typedef void (*virDomainEventDispatchFunc)(virConnectPtr conn,
|
|
virDomainEventPtr event,
|
|
virConnectDomainEventGenericCallback cb,
|
|
void *cbopaque,
|
|
void *opaque);
|
|
|
|
|
|
static void
|
|
virDomainEventDispatchDefaultFunc(virConnectPtr conn,
|
|
virDomainEventPtr event,
|
|
virConnectDomainEventGenericCallback cb,
|
|
void *cbopaque,
|
|
void *opaque ATTRIBUTE_UNUSED)
|
|
{
|
|
virDomainPtr dom = virGetDomain(conn, event->dom.name, event->dom.uuid);
|
|
if (!dom)
|
|
return;
|
|
dom->id = event->dom.id;
|
|
|
|
switch (event->eventID) {
|
|
case VIR_DOMAIN_EVENT_ID_LIFECYCLE:
|
|
((virConnectDomainEventCallback)cb)(conn, dom,
|
|
event->data.lifecycle.type,
|
|
event->data.lifecycle.detail,
|
|
cbopaque);
|
|
break;
|
|
|
|
case VIR_DOMAIN_EVENT_ID_REBOOT:
|
|
(cb)(conn, dom,
|
|
cbopaque);
|
|
break;
|
|
|
|
case VIR_DOMAIN_EVENT_ID_RTC_CHANGE:
|
|
((virConnectDomainEventRTCChangeCallback)cb)(conn, dom,
|
|
event->data.rtcChange.offset,
|
|
cbopaque);
|
|
break;
|
|
|
|
case VIR_DOMAIN_EVENT_ID_WATCHDOG:
|
|
((virConnectDomainEventWatchdogCallback)cb)(conn, dom,
|
|
event->data.watchdog.action,
|
|
cbopaque);
|
|
break;
|
|
|
|
case VIR_DOMAIN_EVENT_ID_IO_ERROR:
|
|
((virConnectDomainEventIOErrorCallback)cb)(conn, dom,
|
|
event->data.ioError.srcPath,
|
|
event->data.ioError.devAlias,
|
|
event->data.ioError.action,
|
|
cbopaque);
|
|
break;
|
|
|
|
case VIR_DOMAIN_EVENT_ID_IO_ERROR_REASON:
|
|
((virConnectDomainEventIOErrorReasonCallback)cb)(conn, dom,
|
|
event->data.ioError.srcPath,
|
|
event->data.ioError.devAlias,
|
|
event->data.ioError.action,
|
|
event->data.ioError.reason,
|
|
cbopaque);
|
|
break;
|
|
|
|
case VIR_DOMAIN_EVENT_ID_GRAPHICS:
|
|
((virConnectDomainEventGraphicsCallback)cb)(conn, dom,
|
|
event->data.graphics.phase,
|
|
event->data.graphics.local,
|
|
event->data.graphics.remote,
|
|
event->data.graphics.authScheme,
|
|
event->data.graphics.subject,
|
|
cbopaque);
|
|
break;
|
|
|
|
case VIR_DOMAIN_EVENT_ID_CONTROL_ERROR:
|
|
(cb)(conn, dom,
|
|
cbopaque);
|
|
break;
|
|
|
|
case VIR_DOMAIN_EVENT_ID_BLOCK_JOB:
|
|
((virConnectDomainEventBlockJobCallback)cb)(conn, dom,
|
|
event->data.blockJob.path,
|
|
event->data.blockJob.type,
|
|
event->data.blockJob.status,
|
|
cbopaque);
|
|
break;
|
|
|
|
case VIR_DOMAIN_EVENT_ID_DISK_CHANGE:
|
|
((virConnectDomainEventDiskChangeCallback)cb)(conn, dom,
|
|
event->data.diskChange.oldSrcPath,
|
|
event->data.diskChange.newSrcPath,
|
|
event->data.diskChange.devAlias,
|
|
event->data.diskChange.reason,
|
|
cbopaque);
|
|
break;
|
|
|
|
case VIR_DOMAIN_EVENT_ID_TRAY_CHANGE:
|
|
((virConnectDomainEventTrayChangeCallback)cb)(conn, dom,
|
|
event->data.trayChange.devAlias,
|
|
event->data.trayChange.reason,
|
|
cbopaque);
|
|
break;
|
|
|
|
case VIR_DOMAIN_EVENT_ID_PMWAKEUP:
|
|
((virConnectDomainEventPMWakeupCallback)cb)(conn, dom, 0, cbopaque);
|
|
break;
|
|
|
|
case VIR_DOMAIN_EVENT_ID_PMSUSPEND:
|
|
((virConnectDomainEventPMSuspendCallback)cb)(conn, dom, 0, cbopaque);
|
|
break;
|
|
|
|
case VIR_DOMAIN_EVENT_ID_BALLOON_CHANGE:
|
|
((virConnectDomainEventBalloonChangeCallback)cb)(conn, dom,
|
|
event->data.balloonChange.actual,
|
|
cbopaque);
|
|
break;
|
|
|
|
case VIR_DOMAIN_EVENT_ID_PMSUSPEND_DISK:
|
|
((virConnectDomainEventPMSuspendDiskCallback)cb)(conn, dom, 0, cbopaque);
|
|
break;
|
|
|
|
default:
|
|
VIR_WARN("Unexpected event ID %d", event->eventID);
|
|
break;
|
|
}
|
|
|
|
virDomainFree(dom);
|
|
}
|
|
|
|
|
|
static int virDomainEventDispatchMatchCallback(virDomainEventPtr event,
|
|
virDomainEventCallbackPtr cb)
|
|
{
|
|
if (!cb)
|
|
return 0;
|
|
if (cb->deleted)
|
|
return 0;
|
|
if (cb->eventID != event->eventID)
|
|
return 0;
|
|
|
|
if (cb->dom) {
|
|
/* Deliberately ignoring 'id' for matching, since that
|
|
* will cause problems when a domain switches between
|
|
* running & shutoff states & ignoring 'name' since
|
|
* Xen sometimes renames guests during migration, thus
|
|
* leaving 'uuid' as the only truly reliable ID we can use*/
|
|
|
|
if (memcmp(event->dom.uuid, cb->dom->uuid, VIR_UUID_BUFLEN) == 0)
|
|
return 1;
|
|
|
|
return 0;
|
|
} else {
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
virDomainEventDispatch(virDomainEventPtr event,
|
|
virDomainEventCallbackListPtr callbacks,
|
|
virDomainEventDispatchFunc dispatch,
|
|
void *opaque)
|
|
{
|
|
int i;
|
|
/* Cache this now, since we may be dropping the lock,
|
|
and have more callbacks added. We're guaranteed not
|
|
to have any removed */
|
|
int cbCount = callbacks->count;
|
|
|
|
for (i = 0 ; i < cbCount ; i++) {
|
|
if (!virDomainEventDispatchMatchCallback(event, callbacks->callbacks[i]))
|
|
continue;
|
|
|
|
(*dispatch)(callbacks->callbacks[i]->conn,
|
|
event,
|
|
callbacks->callbacks[i]->cb,
|
|
callbacks->callbacks[i]->opaque,
|
|
opaque);
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
virDomainEventQueueDispatch(virDomainEventQueuePtr queue,
|
|
virDomainEventCallbackListPtr callbacks,
|
|
virDomainEventDispatchFunc dispatch,
|
|
void *opaque)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0 ; i < queue->count ; i++) {
|
|
virDomainEventDispatch(queue->events[i], callbacks, dispatch, opaque);
|
|
virDomainEventFree(queue->events[i]);
|
|
}
|
|
VIR_FREE(queue->events);
|
|
queue->count = 0;
|
|
}
|
|
|
|
void
|
|
virDomainEventStateQueue(virDomainEventStatePtr state,
|
|
virDomainEventPtr event)
|
|
{
|
|
if (state->timer < 0) {
|
|
virDomainEventFree(event);
|
|
return;
|
|
}
|
|
|
|
virDomainEventStateLock(state);
|
|
|
|
if (virDomainEventQueuePush(state->queue, event) < 0) {
|
|
VIR_DEBUG("Error adding event to queue");
|
|
virDomainEventFree(event);
|
|
}
|
|
|
|
if (state->queue->count == 1)
|
|
virEventUpdateTimeout(state->timer, 0);
|
|
virDomainEventStateUnlock(state);
|
|
}
|
|
|
|
|
|
static void
|
|
virDomainEventStateDispatchFunc(virConnectPtr conn,
|
|
virDomainEventPtr event,
|
|
virConnectDomainEventGenericCallback cb,
|
|
void *cbopaque,
|
|
void *opaque)
|
|
{
|
|
virDomainEventStatePtr state = opaque;
|
|
|
|
/* Drop the lock whle dispatching, for sake of re-entrancy */
|
|
virDomainEventStateUnlock(state);
|
|
virDomainEventDispatchDefaultFunc(conn, event, cb, cbopaque, NULL);
|
|
virDomainEventStateLock(state);
|
|
}
|
|
|
|
|
|
static void
|
|
virDomainEventStateFlush(virDomainEventStatePtr state)
|
|
{
|
|
virDomainEventQueue tempQueue;
|
|
|
|
virDomainEventStateLock(state);
|
|
state->isDispatching = true;
|
|
|
|
/* Copy the queue, so we're reentrant safe when dispatchFunc drops the
|
|
* driver lock */
|
|
tempQueue.count = state->queue->count;
|
|
tempQueue.events = state->queue->events;
|
|
state->queue->count = 0;
|
|
state->queue->events = NULL;
|
|
virEventUpdateTimeout(state->timer, -1);
|
|
|
|
virDomainEventQueueDispatch(&tempQueue,
|
|
state->callbacks,
|
|
virDomainEventStateDispatchFunc,
|
|
state);
|
|
|
|
/* Purge any deleted callbacks */
|
|
virDomainEventCallbackListPurgeMarked(state->callbacks);
|
|
|
|
state->isDispatching = false;
|
|
virDomainEventStateUnlock(state);
|
|
}
|
|
|
|
|
|
/**
|
|
* virDomainEventStateRegister:
|
|
* @conn: connection to associate with callback
|
|
* @state: domain event state
|
|
* @callback: function to remove from event
|
|
* @opaque: data blob to pass to callback
|
|
* @freecb: callback to free @opaque
|
|
*
|
|
* Register the function @callback with connection @conn,
|
|
* from @state, for lifecycle events.
|
|
*
|
|
* Returns: the number of lifecycle callbacks now registered, or -1 on error
|
|
*/
|
|
int
|
|
virDomainEventStateRegister(virConnectPtr conn,
|
|
virDomainEventStatePtr state,
|
|
virConnectDomainEventCallback callback,
|
|
void *opaque,
|
|
virFreeCallback freecb)
|
|
{
|
|
int ret = -1;
|
|
|
|
virDomainEventStateLock(state);
|
|
|
|
if ((state->callbacks->count == 0) &&
|
|
(state->timer == -1) &&
|
|
(state->timer = virEventAddTimeout(-1,
|
|
virDomainEventTimer,
|
|
state,
|
|
NULL)) < 0) {
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
_("could not initialize domain event timer"));
|
|
goto cleanup;
|
|
}
|
|
|
|
ret = virDomainEventCallbackListAdd(conn, state->callbacks,
|
|
callback, opaque, freecb);
|
|
|
|
if (ret == -1 &&
|
|
state->callbacks->count == 0 &&
|
|
state->timer != -1) {
|
|
virEventRemoveTimeout(state->timer);
|
|
state->timer = -1;
|
|
}
|
|
|
|
cleanup:
|
|
virDomainEventStateUnlock(state);
|
|
return ret;
|
|
}
|
|
|
|
|
|
/**
|
|
* virDomainEventStateRegisterID:
|
|
* @conn: connection to associate with callback
|
|
* @state: domain event state
|
|
* @eventID: ID of the event type to register for
|
|
* @cb: function to remove from event
|
|
* @opaque: data blob to pass to callback
|
|
* @freecb: callback to free @opaque
|
|
* @callbackID: filled with callback ID
|
|
*
|
|
* Register the function @callbackID with connection @conn,
|
|
* from @state, for events of type @eventID.
|
|
*
|
|
* Returns: the number of callbacks now registered, or -1 on error
|
|
*/
|
|
int
|
|
virDomainEventStateRegisterID(virConnectPtr conn,
|
|
virDomainEventStatePtr state,
|
|
virDomainPtr dom,
|
|
int eventID,
|
|
virConnectDomainEventGenericCallback cb,
|
|
void *opaque,
|
|
virFreeCallback freecb,
|
|
int *callbackID)
|
|
{
|
|
int ret = -1;
|
|
|
|
virDomainEventStateLock(state);
|
|
|
|
if ((state->callbacks->count == 0) &&
|
|
(state->timer == -1) &&
|
|
(state->timer = virEventAddTimeout(-1,
|
|
virDomainEventTimer,
|
|
state,
|
|
NULL)) < 0) {
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
_("could not initialize domain event timer"));
|
|
goto cleanup;
|
|
}
|
|
|
|
ret = virDomainEventCallbackListAddID(conn, state->callbacks,
|
|
dom, eventID, cb, opaque, freecb,
|
|
callbackID);
|
|
|
|
if (ret == -1 &&
|
|
state->callbacks->count == 0 &&
|
|
state->timer != -1) {
|
|
virEventRemoveTimeout(state->timer);
|
|
state->timer = -1;
|
|
}
|
|
|
|
cleanup:
|
|
virDomainEventStateUnlock(state);
|
|
return ret;
|
|
}
|
|
|
|
|
|
/**
|
|
* virDomainEventStateDeregister:
|
|
* @conn: connection to associate with callback
|
|
* @state: domain event state
|
|
* @callback: function to remove from event
|
|
*
|
|
* Unregister the function @callback with connection @conn,
|
|
* from @state, for lifecycle events.
|
|
*
|
|
* Returns: the number of lifecycle callbacks still registered, or -1 on error
|
|
*/
|
|
int
|
|
virDomainEventStateDeregister(virConnectPtr conn,
|
|
virDomainEventStatePtr state,
|
|
virConnectDomainEventCallback callback)
|
|
{
|
|
int ret;
|
|
|
|
virDomainEventStateLock(state);
|
|
if (state->isDispatching)
|
|
ret = virDomainEventCallbackListMarkDelete(conn,
|
|
state->callbacks, callback);
|
|
else
|
|
ret = virDomainEventCallbackListRemove(conn, state->callbacks, callback);
|
|
|
|
if (state->callbacks->count == 0 &&
|
|
state->timer != -1) {
|
|
virEventRemoveTimeout(state->timer);
|
|
state->timer = -1;
|
|
virDomainEventQueueClear(state->queue);
|
|
}
|
|
|
|
virDomainEventStateUnlock(state);
|
|
return ret;
|
|
}
|
|
|
|
|
|
/**
|
|
* virDomainEventStateDeregisterID:
|
|
* @conn: connection to associate with callback
|
|
* @state: domain event state
|
|
* @callbackID: ID of the function to remove from event
|
|
*
|
|
* Unregister the function @callbackID with connection @conn,
|
|
* from @state, for events.
|
|
*
|
|
* Returns: the number of callbacks still registered, or -1 on error
|
|
*/
|
|
int
|
|
virDomainEventStateDeregisterID(virConnectPtr conn,
|
|
virDomainEventStatePtr state,
|
|
int callbackID)
|
|
{
|
|
int ret;
|
|
|
|
virDomainEventStateLock(state);
|
|
if (state->isDispatching)
|
|
ret = virDomainEventCallbackListMarkDeleteID(conn,
|
|
state->callbacks, callbackID);
|
|
else
|
|
ret = virDomainEventCallbackListRemoveID(conn,
|
|
state->callbacks, callbackID);
|
|
|
|
if (state->callbacks->count == 0 &&
|
|
state->timer != -1) {
|
|
virEventRemoveTimeout(state->timer);
|
|
state->timer = -1;
|
|
virDomainEventQueueClear(state->queue);
|
|
}
|
|
|
|
virDomainEventStateUnlock(state);
|
|
return ret;
|
|
}
|
|
|
|
|
|
/**
|
|
* virDomainEventStateEventID:
|
|
* @conn: connection associated with the callback
|
|
* @state: domain event state
|
|
* @callbackID: the callback to query
|
|
*
|
|
* Query what event ID type is associated with the
|
|
* callback @callbackID for connection @conn
|
|
*
|
|
* Returns 0 on success, -1 on error
|
|
*/
|
|
int
|
|
virDomainEventStateEventID(virConnectPtr conn,
|
|
virDomainEventStatePtr state,
|
|
int callbackID)
|
|
{
|
|
int ret;
|
|
|
|
virDomainEventStateLock(state);
|
|
ret = virDomainEventCallbackListEventID(conn,
|
|
state->callbacks, callbackID);
|
|
virDomainEventStateUnlock(state);
|
|
return ret;
|
|
}
|