643 lines
17 KiB
C
643 lines
17 KiB
C
/* -*- indent-tabs-mode: nil; tab-width: 4; c-basic-offset: 4; -*-
|
|
|
|
mainloop.c for the Openbox window manager
|
|
Copyright (c) 2003 Ben Jansens
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program 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 General Public License for more details.
|
|
|
|
See the COPYING file for a copy of the GNU General Public License.
|
|
*/
|
|
|
|
#include "mainloop.h"
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <sys/select.h>
|
|
#include <signal.h>
|
|
|
|
typedef struct _ObMainLoopTimer ObMainLoopTimer;
|
|
typedef struct _ObMainLoopSignal ObMainLoopSignal;
|
|
typedef struct _ObMainLoopSignalHandlerType ObMainLoopSignalHandlerType;
|
|
typedef struct _ObMainLoopXHandlerType ObMainLoopXHandlerType;
|
|
typedef struct _ObMainLoopFdHandlerType ObMainLoopFdHandlerType;
|
|
|
|
/* this should be more than the number of possible signals on any
|
|
architecture... */
|
|
#define NUM_SIGNALS 99
|
|
|
|
/* all created ObMainLoops. Used by the signal handler to pass along signals */
|
|
static GSList *all_loops;
|
|
|
|
/* signals are global to all loops */
|
|
struct {
|
|
guint installed; /* a ref count */
|
|
struct sigaction oldact;
|
|
} all_signals[NUM_SIGNALS];
|
|
|
|
/* a set of all possible signals */
|
|
sigset_t all_signals_set;
|
|
|
|
/* signals which cause a core dump, these can't be used for callbacks */
|
|
static gint core_signals[] =
|
|
{
|
|
SIGABRT,
|
|
SIGSEGV,
|
|
SIGFPE,
|
|
SIGILL,
|
|
SIGQUIT,
|
|
SIGTRAP,
|
|
SIGSYS,
|
|
SIGBUS,
|
|
SIGXCPU,
|
|
SIGXFSZ
|
|
};
|
|
#define NUM_CORE_SIGNALS (sizeof(core_signals) / sizeof(core_signals[0]))
|
|
|
|
static void sighandler(gint sig);
|
|
static void timer_dispatch(ObMainLoop *loop, GTimeVal **wait);
|
|
static void fd_handler_destroy(gpointer data);
|
|
|
|
struct _ObMainLoop
|
|
{
|
|
Display *display;
|
|
|
|
gboolean run; /* do keep running */
|
|
gboolean running; /* is still running */
|
|
|
|
GSList *x_handlers;
|
|
|
|
gint fd_x; /* The X fd is a special case! */
|
|
gint fd_max;
|
|
GHashTable *fd_handlers;
|
|
fd_set fd_set;
|
|
|
|
GSList *timers;
|
|
GTimeVal now;
|
|
GTimeVal ret_wait;
|
|
|
|
gboolean signal_fired;
|
|
guint signals_fired[NUM_SIGNALS];
|
|
GSList *signal_handlers[NUM_SIGNALS];
|
|
};
|
|
|
|
struct _ObMainLoopTimer
|
|
{
|
|
gulong delay;
|
|
GSourceFunc func;
|
|
gpointer data;
|
|
GDestroyNotify destroy;
|
|
|
|
/* The timer needs to be freed */
|
|
gboolean del_me;
|
|
/* The time the last fire should've been at */
|
|
GTimeVal last;
|
|
/* When this timer will next trigger */
|
|
GTimeVal timeout;
|
|
};
|
|
|
|
struct _ObMainLoopSignalHandlerType
|
|
{
|
|
ObMainLoop *loop;
|
|
gint signal;
|
|
gpointer data;
|
|
ObMainLoopSignalHandler func;
|
|
GDestroyNotify destroy;
|
|
};
|
|
|
|
struct _ObMainLoopXHandlerType
|
|
{
|
|
ObMainLoop *loop;
|
|
gpointer data;
|
|
ObMainLoopXHandler func;
|
|
GDestroyNotify destroy;
|
|
};
|
|
|
|
struct _ObMainLoopFdHandlerType
|
|
{
|
|
ObMainLoop *loop;
|
|
gint fd;
|
|
gpointer data;
|
|
ObMainLoopFdHandler func;
|
|
GDestroyNotify destroy;
|
|
};
|
|
|
|
ObMainLoop *ob_main_loop_new(Display *display)
|
|
{
|
|
ObMainLoop *loop;
|
|
|
|
loop = g_new0(ObMainLoop, 1);
|
|
loop->display = display;
|
|
loop->fd_x = ConnectionNumber(display);
|
|
FD_ZERO(&loop->fd_set);
|
|
FD_SET(loop->fd_x, &loop->fd_set);
|
|
loop->fd_max = loop->fd_x;
|
|
|
|
loop->fd_handlers = g_hash_table_new_full(g_int_hash, g_int_equal,
|
|
NULL, fd_handler_destroy);
|
|
|
|
g_get_current_time(&loop->now);
|
|
|
|
/* only do this if we're the first loop created */
|
|
if (!all_loops) {
|
|
guint i;
|
|
struct sigaction action;
|
|
sigset_t sigset;
|
|
|
|
/* initialize the all_signals_set */
|
|
sigfillset(&all_signals_set);
|
|
|
|
sigemptyset(&sigset);
|
|
action.sa_handler = sighandler;
|
|
action.sa_mask = sigset;
|
|
action.sa_flags = SA_NOCLDSTOP;
|
|
|
|
/* grab all the signals that cause core dumps */
|
|
for (i = 0; i < NUM_CORE_SIGNALS; ++i) {
|
|
/* SIGABRT is curiously not grabbed here!! that's because when we
|
|
get one of the core_signals, we use abort() to dump the core.
|
|
And having the abort() only go back to our signal handler again
|
|
is less than optimal */
|
|
if (core_signals[i] != SIGABRT) {
|
|
sigaction(core_signals[i], &action,
|
|
&all_signals[core_signals[i]].oldact);
|
|
all_signals[core_signals[i]].installed++;
|
|
}
|
|
}
|
|
}
|
|
|
|
all_loops = g_slist_prepend(all_loops, loop);
|
|
|
|
return loop;
|
|
}
|
|
|
|
void ob_main_loop_destroy(ObMainLoop *loop)
|
|
{
|
|
guint i;
|
|
GSList *it, *next;
|
|
|
|
if (loop) {
|
|
g_assert(loop->running == FALSE);
|
|
|
|
for (it = loop->x_handlers; it; it = next) {
|
|
ObMainLoopXHandlerType *h = it->data;
|
|
next = g_slist_next(it);
|
|
ob_main_loop_x_remove(loop, h->func);
|
|
}
|
|
|
|
g_hash_table_destroy(loop->fd_handlers);
|
|
|
|
for (it = loop->timers; it; it = g_slist_next(it)) {
|
|
ObMainLoopTimer *t = it->data;
|
|
if (t->destroy) t->destroy(t->data);
|
|
g_free(t);
|
|
}
|
|
g_slist_free(loop->timers);
|
|
loop->timers = NULL;
|
|
|
|
for (i = 0; i < NUM_SIGNALS; ++i)
|
|
for (it = loop->signal_handlers[i]; it; it = next) {
|
|
ObMainLoopSignalHandlerType *h = it->data;
|
|
next = g_slist_next(it);
|
|
ob_main_loop_signal_remove(loop, h->func);
|
|
}
|
|
|
|
all_loops = g_slist_remove(all_loops, loop);
|
|
|
|
/* only do this if we're the last loop destroyed */
|
|
if (!all_loops) {
|
|
guint i;
|
|
|
|
/* grab all the signals that cause core dumps */
|
|
for (i = 0; i < NUM_CORE_SIGNALS; ++i) {
|
|
if (all_signals[core_signals[i]].installed) {
|
|
sigaction(core_signals[i],
|
|
&all_signals[core_signals[i]].oldact, NULL);
|
|
all_signals[core_signals[i]].installed--;
|
|
}
|
|
}
|
|
}
|
|
|
|
g_free(loop);
|
|
}
|
|
}
|
|
|
|
static void fd_handle_foreach(gpointer key,
|
|
gpointer value,
|
|
gpointer data)
|
|
{
|
|
ObMainLoopFdHandlerType *h = value;
|
|
fd_set *set = data;
|
|
|
|
if (FD_ISSET(h->fd, set))
|
|
h->func(h->fd, h->data);
|
|
}
|
|
|
|
void ob_main_loop_run(ObMainLoop *loop)
|
|
{
|
|
XEvent e;
|
|
struct timeval *wait;
|
|
fd_set selset;
|
|
GSList *it;
|
|
|
|
loop->run = TRUE;
|
|
loop->running = TRUE;
|
|
|
|
while (loop->run) {
|
|
if (loop->signal_fired) {
|
|
guint i;
|
|
sigset_t oldset;
|
|
|
|
/* block signals so that we can do this without the data changing
|
|
on us */
|
|
sigprocmask(SIG_SETMASK, &all_signals_set, &oldset);
|
|
|
|
for (i = 0; i < NUM_SIGNALS; ++i) {
|
|
while (loop->signals_fired[i]) {
|
|
for (it = loop->signal_handlers[i];
|
|
it; it = g_slist_next(it)) {
|
|
ObMainLoopSignalHandlerType *h = it->data;
|
|
h->func(i, h->data);
|
|
}
|
|
loop->signals_fired[i]--;
|
|
}
|
|
}
|
|
loop->signal_fired = FALSE;
|
|
|
|
sigprocmask(SIG_SETMASK, &oldset, NULL);
|
|
} else if (XPending(loop->display)) {
|
|
do {
|
|
XNextEvent(loop->display, &e);
|
|
|
|
for (it = loop->x_handlers; it; it = g_slist_next(it)) {
|
|
ObMainLoopXHandlerType *h = it->data;
|
|
h->func(&e, h->data);
|
|
}
|
|
} while (XPending(loop->display));
|
|
} else {
|
|
/* this only runs if there were no x events received */
|
|
|
|
timer_dispatch(loop, (GTimeVal**)&wait);
|
|
selset = loop->fd_set;
|
|
/* there is a small race condition here. if a signal occurs
|
|
between this if() and the select() then we will not process
|
|
the signal until 'wait' expires. possible solutions include
|
|
using GStaticMutex, and having the signal handler set 'wait'
|
|
to 0 */
|
|
if (!loop->signal_fired)
|
|
select(loop->fd_max + 1, &selset, NULL, NULL, wait);
|
|
|
|
/* handle the X events with highest prioirity */
|
|
if (FD_ISSET(loop->fd_x, &selset))
|
|
continue;
|
|
|
|
g_hash_table_foreach(loop->fd_handlers,
|
|
fd_handle_foreach, &selset);
|
|
}
|
|
}
|
|
|
|
loop->running = FALSE;
|
|
}
|
|
|
|
void ob_main_loop_exit(ObMainLoop *loop)
|
|
{
|
|
loop->run = FALSE;
|
|
}
|
|
|
|
/*** XEVENT WATCHERS ***/
|
|
|
|
void ob_main_loop_x_add(ObMainLoop *loop,
|
|
ObMainLoopXHandler handler,
|
|
gpointer data,
|
|
GDestroyNotify notify)
|
|
{
|
|
ObMainLoopXHandlerType *h;
|
|
|
|
h = g_new(ObMainLoopXHandlerType, 1);
|
|
h->loop = loop;
|
|
h->func = handler;
|
|
h->data = data;
|
|
h->destroy = notify;
|
|
loop->x_handlers = g_slist_prepend(loop->x_handlers, h);
|
|
}
|
|
|
|
void ob_main_loop_x_remove(ObMainLoop *loop,
|
|
ObMainLoopXHandler handler)
|
|
{
|
|
GSList *it, *next;
|
|
|
|
for (it = loop->x_handlers; it; it = next) {
|
|
ObMainLoopXHandlerType *h = it->data;
|
|
next = g_slist_next(it);
|
|
if (h->func == handler) {
|
|
loop->x_handlers = g_slist_delete_link(loop->x_handlers, it);
|
|
if (h->destroy) h->destroy(h->data);
|
|
g_free(h);
|
|
}
|
|
}
|
|
}
|
|
|
|
/*** SIGNAL WATCHERS ***/
|
|
|
|
static void sighandler(gint sig)
|
|
{
|
|
GSList *it;
|
|
guint i;
|
|
|
|
g_return_if_fail(sig < NUM_SIGNALS);
|
|
|
|
for (i = 0; i < NUM_CORE_SIGNALS; ++i)
|
|
if (sig == core_signals[i]) {
|
|
/* XXX special case for signals that default to core dump.
|
|
but throw some helpful output here... */
|
|
|
|
fprintf(stderr, "Fuck yah. Core dump. (Signal=%d)\n", sig);
|
|
|
|
/* die with a core dump */
|
|
abort();
|
|
}
|
|
|
|
for (it = all_loops; it; it = g_slist_next(it)) {
|
|
ObMainLoop *loop = it->data;
|
|
loop->signal_fired = TRUE;
|
|
loop->signals_fired[sig]++;
|
|
}
|
|
}
|
|
|
|
void ob_main_loop_signal_add(ObMainLoop *loop,
|
|
gint signal,
|
|
ObMainLoopSignalHandler handler,
|
|
gpointer data,
|
|
GDestroyNotify notify)
|
|
{
|
|
ObMainLoopSignalHandlerType *h;
|
|
|
|
g_return_if_fail(signal < NUM_SIGNALS);
|
|
|
|
h = g_new(ObMainLoopSignalHandlerType, 1);
|
|
h->loop = loop;
|
|
h->signal = signal;
|
|
h->func = handler;
|
|
h->data = data;
|
|
h->destroy = notify;
|
|
loop->signal_handlers[h->signal] =
|
|
g_slist_prepend(loop->signal_handlers[h->signal], h);
|
|
|
|
if (!all_signals[signal].installed) {
|
|
struct sigaction action;
|
|
sigset_t sigset;
|
|
|
|
sigemptyset(&sigset);
|
|
action.sa_handler = sighandler;
|
|
action.sa_mask = sigset;
|
|
action.sa_flags = SA_NOCLDSTOP;
|
|
|
|
sigaction(signal, &action, &all_signals[signal].oldact);
|
|
}
|
|
|
|
all_signals[signal].installed++;
|
|
}
|
|
|
|
void ob_main_loop_signal_remove(ObMainLoop *loop,
|
|
ObMainLoopSignalHandler handler)
|
|
{
|
|
guint i;
|
|
GSList *it, *next;
|
|
|
|
for (i = 0; i < NUM_SIGNALS; ++i) {
|
|
for (it = loop->signal_handlers[i]; it; it = next) {
|
|
ObMainLoopSignalHandlerType *h = it->data;
|
|
|
|
next = g_slist_next(it);
|
|
|
|
if (h->func == handler) {
|
|
g_assert(all_signals[h->signal].installed > 0);
|
|
|
|
all_signals[h->signal].installed--;
|
|
if (!all_signals[h->signal].installed) {
|
|
sigaction(h->signal, &all_signals[h->signal].oldact, NULL);
|
|
}
|
|
|
|
loop->signal_handlers[i] =
|
|
g_slist_delete_link(loop->signal_handlers[i], it);
|
|
if (h->destroy) h->destroy(h->data);
|
|
|
|
g_free(h);
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
/*** FILE DESCRIPTOR WATCHERS ***/
|
|
|
|
static void max_fd_func(gpointer key, gpointer value, gpointer data)
|
|
{
|
|
ObMainLoop *loop = data;
|
|
|
|
/* key is the fd */
|
|
loop->fd_max = MAX(loop->fd_max, *(gint*)key);
|
|
}
|
|
|
|
static void calc_max_fd(ObMainLoop *loop)
|
|
{
|
|
loop->fd_max = loop->fd_x;
|
|
|
|
g_hash_table_foreach(loop->fd_handlers, max_fd_func, loop);
|
|
}
|
|
|
|
void ob_main_loop_fd_add(ObMainLoop *loop,
|
|
gint fd,
|
|
ObMainLoopFdHandler handler,
|
|
gpointer data,
|
|
GDestroyNotify notify)
|
|
{
|
|
ObMainLoopFdHandlerType *h;
|
|
|
|
h = g_new(ObMainLoopFdHandlerType, 1);
|
|
h->loop = loop;
|
|
h->fd = fd;
|
|
h->func = handler;
|
|
h->data = data;
|
|
h->destroy = notify;
|
|
|
|
g_hash_table_replace(loop->fd_handlers, &h->fd, h);
|
|
FD_SET(h->fd, &loop->fd_set);
|
|
calc_max_fd(loop);
|
|
}
|
|
|
|
static void fd_handler_destroy(gpointer data)
|
|
{
|
|
ObMainLoopFdHandlerType *h = data;
|
|
|
|
FD_CLR(h->fd, &h->loop->fd_set);
|
|
|
|
if (h->destroy)
|
|
h->destroy(h->data);
|
|
}
|
|
|
|
void ob_main_loop_fd_remove(ObMainLoop *loop,
|
|
gint fd)
|
|
{
|
|
g_hash_table_remove(loop->fd_handlers, &fd);
|
|
}
|
|
|
|
/*** TIMEOUTS ***/
|
|
|
|
#define NEAREST_TIMEOUT(loop) \
|
|
(((ObMainLoopTimer*)(loop)->timers->data)->timeout)
|
|
|
|
static long timecompare(GTimeVal *a, GTimeVal *b)
|
|
{
|
|
long r;
|
|
|
|
if ((r = b->tv_sec - a->tv_sec)) return r;
|
|
return b->tv_usec - a->tv_usec;
|
|
|
|
}
|
|
|
|
static void insert_timer(ObMainLoop *loop, ObMainLoopTimer *ins)
|
|
{
|
|
GSList *it;
|
|
for (it = loop->timers; it; it = g_slist_next(it)) {
|
|
ObMainLoopTimer *t = it->data;
|
|
if (timecompare(&ins->timeout, &t->timeout) >= 0) {
|
|
loop->timers = g_slist_insert_before(loop->timers, it, ins);
|
|
break;
|
|
}
|
|
}
|
|
if (it == NULL) /* didnt fit anywhere in the list */
|
|
loop->timers = g_slist_append(loop->timers, ins);
|
|
}
|
|
|
|
void ob_main_loop_timeout_add(ObMainLoop *loop,
|
|
gulong microseconds,
|
|
GSourceFunc handler,
|
|
gpointer data,
|
|
GDestroyNotify notify)
|
|
{
|
|
ObMainLoopTimer *t = g_new(ObMainLoopTimer, 1);
|
|
t->delay = microseconds;
|
|
t->func = handler;
|
|
t->data = data;
|
|
t->destroy = notify;
|
|
t->del_me = FALSE;
|
|
g_get_current_time(&loop->now);
|
|
t->last = t->timeout = loop->now;
|
|
g_time_val_add(&t->timeout, t->delay);
|
|
|
|
insert_timer(loop, t);
|
|
}
|
|
|
|
void ob_main_loop_timeout_remove(ObMainLoop *loop,
|
|
GSourceFunc handler)
|
|
{
|
|
GSList *it;
|
|
|
|
for (it = loop->timers; it; it = g_slist_next(it)) {
|
|
ObMainLoopTimer *t = it->data;
|
|
if (t->func == handler)
|
|
t->del_me = TRUE;
|
|
}
|
|
}
|
|
|
|
void ob_main_loop_timeout_remove_data(ObMainLoop *loop,
|
|
GSourceFunc handler,
|
|
gpointer data)
|
|
{
|
|
GSList *it;
|
|
|
|
for (it = loop->timers; it; it = g_slist_next(it)) {
|
|
ObMainLoopTimer *t = it->data;
|
|
if (t->func == handler && t->data == data)
|
|
t->del_me = TRUE;
|
|
}
|
|
}
|
|
|
|
/* find the time to wait for the nearest timeout */
|
|
static gboolean nearest_timeout_wait(ObMainLoop *loop, GTimeVal *tm)
|
|
{
|
|
if (loop->timers == NULL)
|
|
return FALSE;
|
|
|
|
tm->tv_sec = NEAREST_TIMEOUT(loop).tv_sec - loop->now.tv_sec;
|
|
tm->tv_usec = NEAREST_TIMEOUT(loop).tv_usec - loop->now.tv_usec;
|
|
|
|
while (tm->tv_usec < 0) {
|
|
tm->tv_usec += G_USEC_PER_SEC;
|
|
tm->tv_sec--;
|
|
}
|
|
tm->tv_sec += tm->tv_usec / G_USEC_PER_SEC;
|
|
tm->tv_usec %= G_USEC_PER_SEC;
|
|
if (tm->tv_sec < 0)
|
|
tm->tv_sec = 0;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void timer_dispatch(ObMainLoop *loop, GTimeVal **wait)
|
|
{
|
|
GSList *it, *next;
|
|
|
|
gboolean fired = FALSE;
|
|
|
|
g_get_current_time(&loop->now);
|
|
|
|
for (it = loop->timers; it; it = next) {
|
|
ObMainLoopTimer *curr;
|
|
|
|
next = g_slist_next(it);
|
|
|
|
curr = it->data;
|
|
|
|
/* since timer_stop doesn't actually free the timer, we have to do our
|
|
real freeing in here.
|
|
*/
|
|
if (curr->del_me) {
|
|
/* delete the top */
|
|
loop->timers = g_slist_delete_link(loop->timers, it);
|
|
g_free(curr);
|
|
continue;
|
|
}
|
|
|
|
/* the queue is sorted, so if this timer shouldn't fire, none are
|
|
ready */
|
|
if (timecompare(&NEAREST_TIMEOUT(loop), &loop->now) < 0)
|
|
break;
|
|
|
|
/* we set the last fired time to delay msec after the previous firing,
|
|
then re-insert. timers maintain their order and may trigger more
|
|
than once if they've waited more than one delay's worth of time.
|
|
*/
|
|
loop->timers = g_slist_delete_link(loop->timers, it);
|
|
g_time_val_add(&curr->last, curr->delay);
|
|
if (curr->func(curr->data)) {
|
|
g_time_val_add(&curr->timeout, curr->delay);
|
|
insert_timer(loop, curr);
|
|
} else {
|
|
if (curr->destroy)
|
|
curr->destroy(curr->data);
|
|
g_free(curr);
|
|
}
|
|
|
|
fired = TRUE;
|
|
}
|
|
|
|
if (fired) {
|
|
/* if at least one timer fires, then don't wait on X events, as there
|
|
may already be some in the queue from the timer callbacks.
|
|
*/
|
|
loop->ret_wait.tv_sec = loop->ret_wait.tv_usec = 0;
|
|
*wait = &loop->ret_wait;
|
|
} else if (nearest_timeout_wait(loop, &loop->ret_wait))
|
|
*wait = &loop->ret_wait;
|
|
else
|
|
*wait = NULL;
|
|
}
|