mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-01 02:25:24 +00:00
0db4743645
There is a fairly long standing race condition bug in glib which can hit if you call g_source_destroy or g_source_unref from a non-main thread: https://gitlab.gnome.org/GNOME/glib/-/merge_requests/1358 Unfortunately it is really common for libvirt to call g_source_destroy from a non-main thread. This glib bug is the cause of non-determinstic crashes in eventtest, and probably in libvirtd too. To work around the problem we need to ensure that we never release the last reference on a GSource from a non-main thread. The previous patch replaced our use of g_source_destroy with a pair of g_source_remove and g_source_unref. We can now delay the g_source_unref call by using a idle callback to invoke it from the main thread which avoids the race condition. Reviewed-by: Michal Privoznik <mprivozn@redhat.com> Signed-off-by: Daniel P. Berrangé <berrange@redhat.com>
537 lines
13 KiB
C
537 lines
13 KiB
C
/*
|
|
* vireventglib.c: GMainContext based event loop
|
|
*
|
|
* Copyright (C) 2008 Daniel P. Berrange
|
|
* Copyright (C) 2010-2019 Red Hat, Inc.
|
|
*
|
|
* 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/>.
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
|
|
#include "vireventglib.h"
|
|
#include "vireventglibwatch.h"
|
|
#include "virerror.h"
|
|
#include "virlog.h"
|
|
#include "virprobe.h"
|
|
|
|
#ifdef G_OS_WIN32
|
|
# include <io.h>
|
|
#endif
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_EVENT
|
|
|
|
VIR_LOG_INIT("util.eventglib");
|
|
|
|
struct virEventGLibHandle
|
|
{
|
|
int watch;
|
|
int fd;
|
|
int events;
|
|
int removed;
|
|
GSource *source;
|
|
virEventHandleCallback cb;
|
|
void *opaque;
|
|
virFreeCallback ff;
|
|
};
|
|
|
|
struct virEventGLibTimeout
|
|
{
|
|
int timer;
|
|
int interval;
|
|
int removed;
|
|
GSource *source;
|
|
virEventTimeoutCallback cb;
|
|
void *opaque;
|
|
virFreeCallback ff;
|
|
};
|
|
|
|
static GMutex *eventlock;
|
|
|
|
static int nextwatch = 1;
|
|
static GPtrArray *handles;
|
|
|
|
static int nexttimer = 1;
|
|
static GPtrArray *timeouts;
|
|
|
|
static GIOCondition
|
|
virEventGLibEventsToCondition(int events)
|
|
{
|
|
GIOCondition cond = 0;
|
|
if (events & VIR_EVENT_HANDLE_READABLE)
|
|
cond |= G_IO_IN;
|
|
if (events & VIR_EVENT_HANDLE_WRITABLE)
|
|
cond |= G_IO_OUT;
|
|
if (events & VIR_EVENT_HANDLE_ERROR)
|
|
cond |= G_IO_ERR;
|
|
if (events & VIR_EVENT_HANDLE_HANGUP)
|
|
cond |= G_IO_HUP;
|
|
return cond;
|
|
}
|
|
|
|
static int
|
|
virEventGLibConditionToEvents(GIOCondition cond)
|
|
{
|
|
int events = 0;
|
|
if (cond & G_IO_IN)
|
|
events |= VIR_EVENT_HANDLE_READABLE;
|
|
if (cond & G_IO_OUT)
|
|
events |= VIR_EVENT_HANDLE_WRITABLE;
|
|
if (cond & G_IO_ERR)
|
|
events |= VIR_EVENT_HANDLE_ERROR;
|
|
if (cond & G_IO_NVAL) /* Treat NVAL as error, since libvirt doesn't distinguish */
|
|
events |= VIR_EVENT_HANDLE_ERROR;
|
|
if (cond & G_IO_HUP)
|
|
events |= VIR_EVENT_HANDLE_HANGUP;
|
|
return events;
|
|
}
|
|
|
|
static gboolean
|
|
virEventGLibHandleDispatch(int fd G_GNUC_UNUSED,
|
|
GIOCondition condition,
|
|
gpointer opaque)
|
|
{
|
|
struct virEventGLibHandle *data = opaque;
|
|
int events = virEventGLibConditionToEvents(condition);
|
|
|
|
VIR_DEBUG("Dispatch handler data=%p watch=%d fd=%d events=%d opaque=%p",
|
|
data, data->watch, data->fd, events, data->opaque);
|
|
|
|
PROBE(EVENT_GLIB_DISPATCH_HANDLE,
|
|
"watch=%d events=%d cb=%p opaque=%p",
|
|
data->watch, events, data->cb, data->opaque);
|
|
|
|
(data->cb)(data->watch, data->fd, events, data->opaque);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static int
|
|
virEventGLibHandleAdd(int fd,
|
|
int events,
|
|
virEventHandleCallback cb,
|
|
void *opaque,
|
|
virFreeCallback ff)
|
|
{
|
|
struct virEventGLibHandle *data;
|
|
GIOCondition cond = virEventGLibEventsToCondition(events);
|
|
int ret;
|
|
|
|
g_mutex_lock(eventlock);
|
|
|
|
data = g_new0(struct virEventGLibHandle, 1);
|
|
|
|
data->watch = nextwatch++;
|
|
data->fd = fd;
|
|
data->events = events;
|
|
data->cb = cb;
|
|
data->opaque = opaque;
|
|
data->ff = ff;
|
|
|
|
VIR_DEBUG("Add handle data=%p watch=%d fd=%d events=%d opaque=%p",
|
|
data, data->watch, data->fd, events, data->opaque);
|
|
|
|
if (events != 0) {
|
|
data->source = virEventGLibAddSocketWatch(
|
|
fd, cond, NULL, virEventGLibHandleDispatch, data, NULL);
|
|
}
|
|
|
|
g_ptr_array_add(handles, data);
|
|
|
|
ret = data->watch;
|
|
|
|
PROBE(EVENT_GLIB_ADD_HANDLE,
|
|
"watch=%d fd=%d events=%d cb=%p opaque=%p ff=%p",
|
|
ret, fd, events, cb, opaque, ff);
|
|
g_mutex_unlock(eventlock);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static struct virEventGLibHandle *
|
|
virEventGLibHandleFind(int watch)
|
|
{
|
|
guint i;
|
|
|
|
for (i = 0; i < handles->len; i++) {
|
|
struct virEventGLibHandle *h = g_ptr_array_index(handles, i);
|
|
|
|
if (h == NULL) {
|
|
g_warn_if_reached();
|
|
continue;
|
|
}
|
|
|
|
if ((h->watch == watch) && !h->removed)
|
|
return h;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* If the last refernce to a GSource is released in a non-main
|
|
* thread we're exposed to a race condition that causes a
|
|
* crash:
|
|
* https://gitlab.gnome.org/GNOME/glib/-/merge_requests/1358
|
|
* Thus we're using an idle func to release our ref
|
|
*/
|
|
static gboolean
|
|
virEventGLibSourceUnrefIdle(gpointer data)
|
|
{
|
|
GSource *src = data;
|
|
|
|
g_source_unref(src);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
static void
|
|
virEventGLibHandleUpdate(int watch,
|
|
int events)
|
|
{
|
|
struct virEventGLibHandle *data;
|
|
|
|
PROBE(EVENT_GLIB_UPDATE_HANDLE,
|
|
"watch=%d events=%d",
|
|
watch, events);
|
|
g_mutex_lock(eventlock);
|
|
|
|
data = virEventGLibHandleFind(watch);
|
|
if (!data) {
|
|
VIR_DEBUG("Update for missing handle watch=%d", watch);
|
|
goto cleanup;
|
|
}
|
|
|
|
VIR_DEBUG("Update handle data=%p watch=%d fd=%d events=%d",
|
|
data, watch, data->fd, events);
|
|
|
|
if (events != 0) {
|
|
GIOCondition cond = virEventGLibEventsToCondition(events);
|
|
if (events == data->events)
|
|
goto cleanup;
|
|
|
|
if (data->source != NULL) {
|
|
VIR_DEBUG("Removed old handle source=%p", data->source);
|
|
g_source_destroy(data->source);
|
|
g_idle_add(virEventGLibSourceUnrefIdle, data->source);
|
|
}
|
|
|
|
data->source = virEventGLibAddSocketWatch(
|
|
data->fd, cond, NULL, virEventGLibHandleDispatch, data, NULL);
|
|
|
|
data->events = events;
|
|
VIR_DEBUG("Added new handle source=%p", data->source);
|
|
} else {
|
|
if (data->source == NULL)
|
|
goto cleanup;
|
|
|
|
VIR_DEBUG("Removed old handle source=%p", data->source);
|
|
g_source_destroy(data->source);
|
|
g_idle_add(virEventGLibSourceUnrefIdle, data->source);
|
|
data->source = NULL;
|
|
data->events = 0;
|
|
}
|
|
|
|
cleanup:
|
|
g_mutex_unlock(eventlock);
|
|
}
|
|
|
|
static gboolean
|
|
virEventGLibHandleRemoveIdle(gpointer data)
|
|
{
|
|
struct virEventGLibHandle *h = data;
|
|
|
|
PROBE(EVENT_GLIB_REMOVE_HANDLE_IDLE,
|
|
"watch=%d ff=%p opaque=%p",
|
|
h->watch, h->ff, h->opaque);
|
|
if (h->ff)
|
|
(h->ff)(h->opaque);
|
|
|
|
g_mutex_lock(eventlock);
|
|
g_ptr_array_remove_fast(handles, h);
|
|
g_mutex_unlock(eventlock);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static int
|
|
virEventGLibHandleRemove(int watch)
|
|
{
|
|
struct virEventGLibHandle *data;
|
|
int ret = -1;
|
|
|
|
PROBE(EVENT_GLIB_REMOVE_HANDLE,
|
|
"watch=%d",
|
|
watch);
|
|
g_mutex_lock(eventlock);
|
|
|
|
data = virEventGLibHandleFind(watch);
|
|
if (!data) {
|
|
VIR_DEBUG("Remove of missing handle watch=%d", watch);
|
|
goto cleanup;
|
|
}
|
|
|
|
VIR_DEBUG("Remove handle data=%p watch=%d fd=%d",
|
|
data, watch, data->fd);
|
|
|
|
if (data->source != NULL) {
|
|
g_source_destroy(data->source);
|
|
g_idle_add(virEventGLibSourceUnrefIdle, data->source);
|
|
data->source = NULL;
|
|
data->events = 0;
|
|
}
|
|
|
|
/* since the actual watch deletion is done asynchronously, a handleUpdate call may
|
|
* reschedule the watch before it's fully deleted, that's why we need to mark it as
|
|
* 'removed' to prevent reuse
|
|
*/
|
|
data->removed = TRUE;
|
|
g_idle_add(virEventGLibHandleRemoveIdle, data);
|
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
g_mutex_unlock(eventlock);
|
|
return ret;
|
|
}
|
|
|
|
|
|
static gboolean
|
|
virEventGLibTimeoutDispatch(void *opaque)
|
|
{
|
|
struct virEventGLibTimeout *data = opaque;
|
|
|
|
VIR_DEBUG("Dispatch timeout data=%p cb=%p timer=%d opaque=%p",
|
|
data, data->cb, data->timer, data->opaque);
|
|
|
|
PROBE(EVENT_GLIB_DISPATCH_TIMEOUT,
|
|
"timer=%d cb=%p opaque=%p",
|
|
data->timer, data->cb, data->opaque);
|
|
(data->cb)(data->timer, data->opaque);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static GSource *
|
|
virEventGLibTimeoutCreate(int interval,
|
|
struct virEventGLibTimeout *data)
|
|
{
|
|
GSource *source = g_timeout_source_new(interval);
|
|
|
|
g_source_set_callback(source,
|
|
virEventGLibTimeoutDispatch,
|
|
data, NULL);
|
|
g_source_attach(source, NULL);
|
|
|
|
return source;
|
|
}
|
|
|
|
|
|
static int
|
|
virEventGLibTimeoutAdd(int interval,
|
|
virEventTimeoutCallback cb,
|
|
void *opaque,
|
|
virFreeCallback ff)
|
|
{
|
|
struct virEventGLibTimeout *data;
|
|
int ret;
|
|
|
|
g_mutex_lock(eventlock);
|
|
|
|
data = g_new0(struct virEventGLibTimeout, 1);
|
|
data->timer = nexttimer++;
|
|
data->interval = interval;
|
|
data->cb = cb;
|
|
data->opaque = opaque;
|
|
data->ff = ff;
|
|
if (interval >= 0)
|
|
data->source = virEventGLibTimeoutCreate(interval, data);
|
|
|
|
g_ptr_array_add(timeouts, data);
|
|
|
|
VIR_DEBUG("Add timeout data=%p interval=%d ms cb=%p opaque=%p timer=%d",
|
|
data, interval, cb, opaque, data->timer);
|
|
|
|
ret = data->timer;
|
|
|
|
PROBE(EVENT_GLIB_ADD_TIMEOUT,
|
|
"timer=%d interval=%d cb=%p opaque=%p ff=%p",
|
|
ret, interval, cb, opaque, ff);
|
|
g_mutex_unlock(eventlock);
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
static struct virEventGLibTimeout *
|
|
virEventGLibTimeoutFind(int timer)
|
|
{
|
|
guint i;
|
|
|
|
g_return_val_if_fail(timeouts != NULL, NULL);
|
|
|
|
for (i = 0; i < timeouts->len; i++) {
|
|
struct virEventGLibTimeout *t = g_ptr_array_index(timeouts, i);
|
|
|
|
if (t == NULL) {
|
|
g_warn_if_reached();
|
|
continue;
|
|
}
|
|
|
|
if ((t->timer == timer) && !t->removed)
|
|
return t;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
|
|
static void
|
|
virEventGLibTimeoutUpdate(int timer,
|
|
int interval)
|
|
{
|
|
struct virEventGLibTimeout *data;
|
|
|
|
PROBE(EVENT_GLIB_UPDATE_TIMEOUT,
|
|
"timer=%d interval=%d",
|
|
timer, interval);
|
|
g_mutex_lock(eventlock);
|
|
|
|
data = virEventGLibTimeoutFind(timer);
|
|
if (!data) {
|
|
VIR_DEBUG("Update of missing timeout timer=%d", timer);
|
|
goto cleanup;
|
|
}
|
|
|
|
VIR_DEBUG("Update timeout data=%p timer=%d interval=%d ms", data, timer, interval);
|
|
|
|
if (interval >= 0) {
|
|
if (data->source != NULL) {
|
|
g_source_destroy(data->source);
|
|
g_idle_add(virEventGLibSourceUnrefIdle, data->source);
|
|
}
|
|
|
|
data->interval = interval;
|
|
data->source = virEventGLibTimeoutCreate(interval, data);
|
|
} else {
|
|
if (data->source == NULL)
|
|
goto cleanup;
|
|
|
|
g_source_destroy(data->source);
|
|
g_idle_add(virEventGLibSourceUnrefIdle, data->source);
|
|
data->source = NULL;
|
|
}
|
|
|
|
cleanup:
|
|
g_mutex_unlock(eventlock);
|
|
}
|
|
|
|
static gboolean
|
|
virEventGLibTimeoutRemoveIdle(gpointer data)
|
|
{
|
|
struct virEventGLibTimeout *t = data;
|
|
|
|
PROBE(EVENT_GLIB_REMOVE_TIMEOUT_IDLE,
|
|
"timer=%d ff=%p opaque=%p",
|
|
t->timer, t->ff, t->opaque);
|
|
|
|
if (t->ff)
|
|
(t->ff)(t->opaque);
|
|
|
|
g_mutex_lock(eventlock);
|
|
g_ptr_array_remove_fast(timeouts, t);
|
|
g_mutex_unlock(eventlock);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static int
|
|
virEventGLibTimeoutRemove(int timer)
|
|
{
|
|
struct virEventGLibTimeout *data;
|
|
int ret = -1;
|
|
|
|
PROBE(EVENT_GLIB_REMOVE_TIMEOUT,
|
|
"timer=%d",
|
|
timer);
|
|
g_mutex_lock(eventlock);
|
|
|
|
data = virEventGLibTimeoutFind(timer);
|
|
if (!data) {
|
|
VIR_DEBUG("Remove of missing timeout timer=%d", timer);
|
|
goto cleanup;
|
|
}
|
|
|
|
VIR_DEBUG("Remove timeout data=%p timer=%d",
|
|
data, timer);
|
|
|
|
if (data->source != NULL) {
|
|
g_source_destroy(data->source);
|
|
g_idle_add(virEventGLibSourceUnrefIdle, data->source);
|
|
data->source = NULL;
|
|
}
|
|
|
|
/* since the actual timeout deletion is done asynchronously, a timeoutUpdate call may
|
|
* reschedule the timeout before it's fully deleted, that's why we need to mark it as
|
|
* 'removed' to prevent reuse
|
|
*/
|
|
data->removed = TRUE;
|
|
g_idle_add(virEventGLibTimeoutRemoveIdle, data);
|
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
g_mutex_unlock(eventlock);
|
|
return ret;
|
|
}
|
|
|
|
|
|
static gpointer virEventGLibRegisterOnce(gpointer data G_GNUC_UNUSED)
|
|
{
|
|
eventlock = g_new0(GMutex, 1);
|
|
timeouts = g_ptr_array_new_with_free_func(g_free);
|
|
handles = g_ptr_array_new_with_free_func(g_free);
|
|
virEventRegisterImpl(virEventGLibHandleAdd,
|
|
virEventGLibHandleUpdate,
|
|
virEventGLibHandleRemove,
|
|
virEventGLibTimeoutAdd,
|
|
virEventGLibTimeoutUpdate,
|
|
virEventGLibTimeoutRemove);
|
|
return NULL;
|
|
}
|
|
|
|
|
|
void virEventGLibRegister(void)
|
|
{
|
|
static GOnce once = G_ONCE_INIT;
|
|
|
|
g_once(&once, virEventGLibRegisterOnce, NULL);
|
|
}
|
|
|
|
|
|
int virEventGLibRunOnce(void)
|
|
{
|
|
g_main_context_iteration(NULL, TRUE);
|
|
|
|
return 0;
|
|
}
|