2007-06-26 22:51:01 +00:00
|
|
|
/*
|
2008-05-15 06:12:32 +00:00
|
|
|
* event.c: event loop for monitoring file handles
|
2007-06-26 22:51:01 +00:00
|
|
|
*
|
2011-03-08 20:42:19 -07:00
|
|
|
* Copyright (C) 2007, 2011 Red Hat, Inc.
|
2007-06-26 22:51:01 +00:00
|
|
|
* Copyright (C) 2007 Daniel P. Berrange
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
2012-09-20 16:30:55 -06:00
|
|
|
* License along with this library. If not, see
|
2012-07-21 18:06:23 +08:00
|
|
|
* <http://www.gnu.org/licenses/>.
|
2007-06-26 22:51:01 +00:00
|
|
|
*
|
|
|
|
* Author: Daniel P. Berrange <berrange@redhat.com>
|
|
|
|
*/
|
|
|
|
|
2008-01-29 18:15:54 +00:00
|
|
|
#include <config.h>
|
2007-06-26 22:51:01 +00:00
|
|
|
|
|
|
|
#include "event.h"
|
2011-03-02 16:59:54 +00:00
|
|
|
#include "event_poll.h"
|
|
|
|
#include "logging.h"
|
|
|
|
#include "virterror_internal.h"
|
2007-06-26 22:51:01 +00:00
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
static virEventAddHandleFunc addHandleImpl = NULL;
|
2007-09-19 01:27:32 +00:00
|
|
|
static virEventUpdateHandleFunc updateHandleImpl = NULL;
|
2007-06-26 22:51:01 +00:00
|
|
|
static virEventRemoveHandleFunc removeHandleImpl = NULL;
|
|
|
|
static virEventAddTimeoutFunc addTimeoutImpl = NULL;
|
2007-09-19 01:27:32 +00:00
|
|
|
static virEventUpdateTimeoutFunc updateTimeoutImpl = NULL;
|
2007-06-26 22:51:01 +00:00
|
|
|
static virEventRemoveTimeoutFunc removeTimeoutImpl = NULL;
|
|
|
|
|
2011-06-15 17:54:30 -04:00
|
|
|
/**
|
|
|
|
* virEventAddHandle: register a callback for monitoring file handle events
|
|
|
|
*
|
|
|
|
* @fd: file handle to monitor for events
|
|
|
|
* @events: bitset of events to watch from virEventHandleType constants
|
|
|
|
* @cb: callback to invoke when an event occurs
|
|
|
|
* @opaque: user data to pass to callback
|
|
|
|
*
|
|
|
|
* returns -1 if the file handle cannot be registered, 0 upon success
|
|
|
|
*/
|
2008-11-19 16:24:01 +00:00
|
|
|
int virEventAddHandle(int fd,
|
|
|
|
int events,
|
|
|
|
virEventHandleCallback cb,
|
|
|
|
void *opaque,
|
|
|
|
virFreeCallback ff) {
|
2007-06-26 22:51:01 +00:00
|
|
|
if (!addHandleImpl)
|
|
|
|
return -1;
|
|
|
|
|
2008-11-19 16:24:01 +00:00
|
|
|
return addHandleImpl(fd, events, cb, opaque, ff);
|
2007-06-26 22:51:01 +00:00
|
|
|
}
|
|
|
|
|
2011-06-15 17:54:30 -04:00
|
|
|
/**
|
|
|
|
* virEventUpdateHandle: change event set for a monitored file handle
|
|
|
|
*
|
|
|
|
* @watch: watch whose file handle to update
|
|
|
|
* @events: bitset of events to watch from virEventHandleType constants
|
|
|
|
*
|
|
|
|
* Will not fail if fd exists
|
|
|
|
*/
|
2008-11-19 16:19:36 +00:00
|
|
|
void virEventUpdateHandle(int watch, int events) {
|
|
|
|
updateHandleImpl(watch, events);
|
2007-09-19 01:27:32 +00:00
|
|
|
}
|
|
|
|
|
2011-06-15 17:54:30 -04:00
|
|
|
/**
|
|
|
|
* virEventRemoveHandle: unregister a callback from a file handle
|
|
|
|
*
|
|
|
|
* @watch: watch whose file handle to remove
|
|
|
|
*
|
|
|
|
* returns -1 if the file handle was not registered, 0 upon success
|
|
|
|
*/
|
2008-11-19 16:19:36 +00:00
|
|
|
int virEventRemoveHandle(int watch) {
|
2007-06-26 22:51:01 +00:00
|
|
|
if (!removeHandleImpl)
|
|
|
|
return -1;
|
|
|
|
|
2008-11-19 16:19:36 +00:00
|
|
|
return removeHandleImpl(watch);
|
2007-06-26 22:51:01 +00:00
|
|
|
}
|
|
|
|
|
2011-06-15 17:54:30 -04:00
|
|
|
/**
|
|
|
|
* virEventAddTimeout: register a callback for a timer event
|
|
|
|
*
|
2011-06-15 19:35:19 -04:00
|
|
|
* @timeout: time between events in milliseconds
|
2011-06-15 17:54:30 -04:00
|
|
|
* @cb: callback to invoke when an event occurs
|
|
|
|
* @opaque: user data to pass to callback
|
|
|
|
*
|
2011-06-15 19:35:19 -04:00
|
|
|
* Setting timeout to -1 will disable the timer. Setting the timeout
|
2011-06-15 17:54:30 -04:00
|
|
|
* to zero will cause it to fire on every event loop iteration.
|
|
|
|
*
|
|
|
|
* returns -1 if the timer cannot be registered, a positive
|
|
|
|
* integer timer id upon success
|
|
|
|
*/
|
2008-11-19 16:24:01 +00:00
|
|
|
int virEventAddTimeout(int timeout,
|
|
|
|
virEventTimeoutCallback cb,
|
|
|
|
void *opaque,
|
|
|
|
virFreeCallback ff) {
|
2007-06-26 22:51:01 +00:00
|
|
|
if (!addTimeoutImpl)
|
|
|
|
return -1;
|
|
|
|
|
2008-11-19 16:24:01 +00:00
|
|
|
return addTimeoutImpl(timeout, cb, opaque, ff);
|
2007-06-26 22:51:01 +00:00
|
|
|
}
|
|
|
|
|
2011-06-15 17:54:30 -04:00
|
|
|
/**
|
|
|
|
* virEventUpdateTimeoutImpl: change frequency for a timer
|
|
|
|
*
|
|
|
|
* @timer: timer id to change
|
|
|
|
* @frequency: time between events in milliseconds
|
|
|
|
*
|
|
|
|
* Setting frequency to -1 will disable the timer. Setting the frequency
|
|
|
|
* to zero will cause it to fire on every event loop iteration.
|
|
|
|
*
|
|
|
|
* Will not fail if timer exists
|
|
|
|
*/
|
2007-09-19 01:27:32 +00:00
|
|
|
void virEventUpdateTimeout(int timer, int timeout) {
|
|
|
|
updateTimeoutImpl(timer, timeout);
|
|
|
|
}
|
|
|
|
|
2011-06-15 17:54:30 -04:00
|
|
|
/**
|
|
|
|
* virEventRemoveTimeout: unregister a callback for a timer
|
|
|
|
*
|
|
|
|
* @timer: the timer id to remove
|
|
|
|
*
|
|
|
|
* returns -1 if the timer was not registered, 0 upon success
|
|
|
|
*/
|
2007-06-26 22:51:01 +00:00
|
|
|
int virEventRemoveTimeout(int timer) {
|
|
|
|
if (!removeTimeoutImpl)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return removeTimeoutImpl(timer);
|
|
|
|
}
|
|
|
|
|
2011-03-02 16:59:54 +00:00
|
|
|
|
|
|
|
/*****************************************************
|
|
|
|
*
|
|
|
|
* Below this point are 3 *PUBLIC* APIs for event
|
|
|
|
* loop integration with applications using libvirt.
|
|
|
|
* These API contracts cannot be changed.
|
|
|
|
*
|
|
|
|
*****************************************************/
|
|
|
|
|
2008-10-23 13:18:18 +00:00
|
|
|
/**
|
|
|
|
* virEventRegisterImpl:
|
|
|
|
* @addHandle: the callback to add fd handles
|
|
|
|
* @updateHandle: the callback to update fd handles
|
|
|
|
* @removeHandle: the callback to remove fd handles
|
|
|
|
* @addTimeout: the callback to add a timeout
|
|
|
|
* @updateTimeout: the callback to update a timeout
|
|
|
|
* @removeTimeout: the callback to remove a timeout
|
2010-10-13 12:19:02 +02:00
|
|
|
*
|
2011-03-02 16:59:54 +00:00
|
|
|
* Registers an event implementation, to allow integration
|
|
|
|
* with an external event loop. Applications would use this
|
|
|
|
* to integrate with the libglib2 event loop, or libevent
|
|
|
|
* or the QT event loop.
|
|
|
|
*
|
|
|
|
* If an application does not need to integrate with an
|
|
|
|
* existing event loop implementation, then the
|
|
|
|
* virEventRegisterDefaultImpl method can be used to setup
|
|
|
|
* the generic libvirt implementation.
|
2008-10-23 13:18:18 +00:00
|
|
|
*/
|
|
|
|
void virEventRegisterImpl(virEventAddHandleFunc addHandle,
|
|
|
|
virEventUpdateHandleFunc updateHandle,
|
|
|
|
virEventRemoveHandleFunc removeHandle,
|
|
|
|
virEventAddTimeoutFunc addTimeout,
|
|
|
|
virEventUpdateTimeoutFunc updateTimeout,
|
2011-03-02 16:59:54 +00:00
|
|
|
virEventRemoveTimeoutFunc removeTimeout)
|
|
|
|
{
|
|
|
|
VIR_DEBUG("addHandle=%p updateHandle=%p removeHandle=%p "
|
|
|
|
"addTimeout=%p updateTimeout=%p removeTimeout=%p",
|
|
|
|
addHandle, updateHandle, removeHandle,
|
|
|
|
addTimeout, updateTimeout, removeTimeout);
|
|
|
|
|
2007-06-26 22:51:01 +00:00
|
|
|
addHandleImpl = addHandle;
|
2007-09-19 01:27:32 +00:00
|
|
|
updateHandleImpl = updateHandle;
|
2007-06-26 22:51:01 +00:00
|
|
|
removeHandleImpl = removeHandle;
|
|
|
|
addTimeoutImpl = addTimeout;
|
2007-09-19 01:27:32 +00:00
|
|
|
updateTimeoutImpl = updateTimeout;
|
2007-06-26 22:51:01 +00:00
|
|
|
removeTimeoutImpl = removeTimeout;
|
|
|
|
}
|
2011-03-02 16:59:54 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* virEventRegisterDefaultImpl:
|
|
|
|
*
|
|
|
|
* Registers a default event implementation based on the
|
|
|
|
* poll() system call. This is a generic implementation
|
|
|
|
* that can be used by any client application which does
|
|
|
|
* not have a need to integrate with an external event
|
|
|
|
* loop impl.
|
|
|
|
*
|
2011-09-22 13:47:07 +02:00
|
|
|
* Once registered, the application has to invoke virEventRunDefaultImpl in
|
|
|
|
* a loop to process events. Failure to do so may result in connections being
|
|
|
|
* closed unexpectedly as a result of keepalive timeout.
|
2011-03-08 20:42:19 -07:00
|
|
|
*
|
|
|
|
* Returns 0 on success, -1 on failure.
|
2011-03-02 16:59:54 +00:00
|
|
|
*/
|
|
|
|
int virEventRegisterDefaultImpl(void)
|
|
|
|
{
|
2011-05-09 17:24:09 +08:00
|
|
|
VIR_DEBUG("registering default event implementation");
|
2011-03-02 16:59:54 +00:00
|
|
|
|
|
|
|
virResetLastError();
|
|
|
|
|
|
|
|
if (virEventPollInit() < 0) {
|
|
|
|
virDispatchError(NULL);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
virEventRegisterImpl(
|
|
|
|
virEventPollAddHandle,
|
|
|
|
virEventPollUpdateHandle,
|
|
|
|
virEventPollRemoveHandle,
|
|
|
|
virEventPollAddTimeout,
|
|
|
|
virEventPollUpdateTimeout,
|
|
|
|
virEventPollRemoveTimeout
|
|
|
|
);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* virEventRunDefaultImpl:
|
|
|
|
*
|
|
|
|
* Run one iteration of the event loop. Applications
|
|
|
|
* will generally want to have a thread which invokes
|
|
|
|
* this method in an infinite loop
|
|
|
|
*
|
|
|
|
* static bool quit = false;
|
|
|
|
*
|
|
|
|
* while (!quit) {
|
|
|
|
* if (virEventRunDefaultImpl() < 0)
|
|
|
|
* ...print error...
|
|
|
|
* }
|
2011-03-08 20:42:19 -07:00
|
|
|
*
|
|
|
|
* Returns 0 on success, -1 on failure.
|
2011-03-02 16:59:54 +00:00
|
|
|
*/
|
|
|
|
int virEventRunDefaultImpl(void)
|
|
|
|
{
|
2011-05-09 17:24:09 +08:00
|
|
|
VIR_DEBUG("running default event implementation");
|
2011-03-02 16:59:54 +00:00
|
|
|
virResetLastError();
|
|
|
|
|
|
|
|
if (virEventPollRunOnce() < 0) {
|
|
|
|
virDispatchError(NULL);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|