mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-23 04:55:18 +00:00
487c063381
This patch introduces a new event type for the QMP event SUSPEND: VIR_DOMAIN_EVENT_ID_PMSUSPEND The event doesn't take any data, but considering there might be reason for wakeup in future, the callback definition is: typedef void (*virConnectDomainEventSuspendCallback)(virConnectPtr conn, virDomainPtr dom, int reason, void *opaque); "reason" is unused currently, always passes "0".
1656 lines
51 KiB
C
1656 lines
51 KiB
C
/*
|
|
* domain_event.c: domain event queue processing helpers
|
|
*
|
|
* Copyright (C) 2010-2011 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, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*
|
|
* Author: Ben Guthro
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include "domain_event.h"
|
|
#include "logging.h"
|
|
#include "datatypes.h"
|
|
#include "memory.h"
|
|
#include "virterror_internal.h"
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_NONE
|
|
|
|
#define eventReportError(code, ...) \
|
|
virReportErrorHelper(VIR_FROM_THIS, code, __FILE__, \
|
|
__FUNCTION__, __LINE__, __VA_ARGS__)
|
|
|
|
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;
|
|
} 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);
|
|
virUnrefConnect(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;
|
|
}
|
|
}
|
|
|
|
eventReportError(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);
|
|
virUnrefConnect(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;
|
|
}
|
|
}
|
|
|
|
eventReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
_("could not find event callback for removal"));
|
|
return -1;
|
|
}
|
|
|
|
|
|
/**
|
|
* virDomainEventCallbackListRemoveConn:
|
|
* @conn: pointer to the connection
|
|
* @cbList: the list
|
|
*
|
|
* Internal function to remove all of a given connection's callback
|
|
* from a virDomainEventCallbackListPtr
|
|
*/
|
|
static int
|
|
virDomainEventCallbackListRemoveConn(virConnectPtr conn,
|
|
virDomainEventCallbackListPtr cbList)
|
|
{
|
|
int old_count = cbList->count;
|
|
int i;
|
|
for (i = 0 ; i < cbList->count ; i++) {
|
|
if (cbList->callbacks[i]->conn == conn) {
|
|
virFreeCallback freecb = cbList->callbacks[i]->freecb;
|
|
if (freecb)
|
|
(*freecb)(cbList->callbacks[i]->opaque);
|
|
virUnrefConnect(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;
|
|
}
|
|
|
|
|
|
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;
|
|
}
|
|
}
|
|
|
|
eventReportError(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;
|
|
}
|
|
}
|
|
|
|
eventReportError(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);
|
|
virUnrefConnect(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) {
|
|
eventReportError(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;
|
|
|
|
event->conn->refs++;
|
|
|
|
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);
|
|
}
|
|
|
|
/**
|
|
* virDomainEventQueueFree:
|
|
* @queue: pointer to the queue
|
|
*
|
|
* Free the memory in the queue. We process this like a list here
|
|
*/
|
|
static void
|
|
virDomainEventQueueFree(virDomainEventQueuePtr queue)
|
|
{
|
|
int i;
|
|
if (!queue)
|
|
return;
|
|
|
|
for (i = 0; i < queue->count ; i++) {
|
|
virDomainEventFree(queue->events[i]);
|
|
}
|
|
VIR_FREE(queue->events);
|
|
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);
|
|
}
|
|
|
|
/**
|
|
* virDomainEventQueuePush:
|
|
* @evtQueue: the dom event queue
|
|
* @event: the event to add
|
|
*
|
|
* Internal function to push onto 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;
|
|
|
|
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) {
|
|
eventReportError(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) {
|
|
eventReportError(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;
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
virDomainEventStateUnlock(state);
|
|
return ret;
|
|
}
|
|
|
|
|
|
/**
|
|
* virDomainEventStateDeregisterConn:
|
|
* @conn: connection to associate with callbacks
|
|
* @state: domain event state
|
|
*
|
|
* Remove all callbacks from @state associated with the
|
|
* connection @conn
|
|
*
|
|
* Returns 0 on success, -1 on error
|
|
*/
|
|
int
|
|
virDomainEventStateDeregisterConn(virConnectPtr conn,
|
|
virDomainEventStatePtr state)
|
|
{
|
|
int ret;
|
|
virDomainEventStateLock(state);
|
|
ret = virDomainEventCallbackListRemoveConn(conn, state->callbacks);
|
|
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;
|
|
}
|