mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2024-11-02 03:11:12 +00:00
b65f37a4a1
These VIR_XXXX0 APIs make us confused, use the non-0-suffix APIs instead. How do these coversions works? The magic is using the gcc extension of ##. When __VA_ARGS__ is empty, "##" will swallow the "," in "fmt," to avoid compile error. example: origin after CPP high_level_api("%d", a_int) low_level_api("%d", a_int) high_level_api("a string") low_level_api("a string") About 400 conversions. 8 special conversions: VIR_XXXX0("") -> VIR_XXXX("msg") (avoid empty format) 2 conversions VIR_XXXX0(string_literal_with_%) -> VIR_XXXX(%->%%) 0 conversions VIR_XXXX0(non_string_literal) -> VIR_XXXX("%s", non_string_literal) (for security) 6 conversions Signed-off-by: Lai Jiangshan <laijs@cn.fujitsu.com>
198 lines
5.6 KiB
C
198 lines
5.6 KiB
C
/*
|
|
* event.c: event loop for monitoring file handles
|
|
*
|
|
* Copyright (C) 2007, 2011 Red Hat, Inc.
|
|
* 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
|
|
* 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: Daniel P. Berrange <berrange@redhat.com>
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include "event.h"
|
|
#include "event_poll.h"
|
|
#include "logging.h"
|
|
#include "virterror_internal.h"
|
|
|
|
#include <stdlib.h>
|
|
|
|
static virEventAddHandleFunc addHandleImpl = NULL;
|
|
static virEventUpdateHandleFunc updateHandleImpl = NULL;
|
|
static virEventRemoveHandleFunc removeHandleImpl = NULL;
|
|
static virEventAddTimeoutFunc addTimeoutImpl = NULL;
|
|
static virEventUpdateTimeoutFunc updateTimeoutImpl = NULL;
|
|
static virEventRemoveTimeoutFunc removeTimeoutImpl = NULL;
|
|
|
|
int virEventAddHandle(int fd,
|
|
int events,
|
|
virEventHandleCallback cb,
|
|
void *opaque,
|
|
virFreeCallback ff) {
|
|
if (!addHandleImpl)
|
|
return -1;
|
|
|
|
return addHandleImpl(fd, events, cb, opaque, ff);
|
|
}
|
|
|
|
void virEventUpdateHandle(int watch, int events) {
|
|
updateHandleImpl(watch, events);
|
|
}
|
|
|
|
int virEventRemoveHandle(int watch) {
|
|
if (!removeHandleImpl)
|
|
return -1;
|
|
|
|
return removeHandleImpl(watch);
|
|
}
|
|
|
|
int virEventAddTimeout(int timeout,
|
|
virEventTimeoutCallback cb,
|
|
void *opaque,
|
|
virFreeCallback ff) {
|
|
if (!addTimeoutImpl)
|
|
return -1;
|
|
|
|
return addTimeoutImpl(timeout, cb, opaque, ff);
|
|
}
|
|
|
|
void virEventUpdateTimeout(int timer, int timeout) {
|
|
updateTimeoutImpl(timer, timeout);
|
|
}
|
|
|
|
int virEventRemoveTimeout(int timer) {
|
|
if (!removeTimeoutImpl)
|
|
return -1;
|
|
|
|
return removeTimeoutImpl(timer);
|
|
}
|
|
|
|
|
|
/*****************************************************
|
|
*
|
|
* Below this point are 3 *PUBLIC* APIs for event
|
|
* loop integration with applications using libvirt.
|
|
* These API contracts cannot be changed.
|
|
*
|
|
*****************************************************/
|
|
|
|
/**
|
|
* 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
|
|
*
|
|
* 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.
|
|
*/
|
|
void virEventRegisterImpl(virEventAddHandleFunc addHandle,
|
|
virEventUpdateHandleFunc updateHandle,
|
|
virEventRemoveHandleFunc removeHandle,
|
|
virEventAddTimeoutFunc addTimeout,
|
|
virEventUpdateTimeoutFunc updateTimeout,
|
|
virEventRemoveTimeoutFunc removeTimeout)
|
|
{
|
|
VIR_DEBUG("addHandle=%p updateHandle=%p removeHandle=%p "
|
|
"addTimeout=%p updateTimeout=%p removeTimeout=%p",
|
|
addHandle, updateHandle, removeHandle,
|
|
addTimeout, updateTimeout, removeTimeout);
|
|
|
|
addHandleImpl = addHandle;
|
|
updateHandleImpl = updateHandle;
|
|
removeHandleImpl = removeHandle;
|
|
addTimeoutImpl = addTimeout;
|
|
updateTimeoutImpl = updateTimeout;
|
|
removeTimeoutImpl = removeTimeout;
|
|
}
|
|
|
|
/**
|
|
* 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.
|
|
*
|
|
* Once registered, the application can invoke
|
|
* virEventRunDefaultImpl in a loop to process
|
|
* events
|
|
*
|
|
* Returns 0 on success, -1 on failure.
|
|
*/
|
|
int virEventRegisterDefaultImpl(void)
|
|
{
|
|
VIR_DEBUG("registering default event implementation");
|
|
|
|
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...
|
|
* }
|
|
*
|
|
* Returns 0 on success, -1 on failure.
|
|
*/
|
|
int virEventRunDefaultImpl(void)
|
|
{
|
|
VIR_DEBUG("running default event implementation");
|
|
virResetLastError();
|
|
|
|
if (virEventPollRunOnce() < 0) {
|
|
virDispatchError(NULL);
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|