summaryrefslogtreecommitdiff
path: root/gmain.c
diff options
context:
space:
mode:
authorOwen Taylor <otaylor@redhat.com>1998-12-02 14:55:27 +0000
committerOwen Taylor <otaylor@src.gnome.org>1998-12-02 14:55:27 +0000
commitbeab982e3b8547c1f7d95e6d8f51d4ad430a7694 (patch)
tree320b704f249ec6bab3cb176847422a4482e73e15 /gmain.c
parentc8477277fec2943a203242cf6bf5c43c9a141693 (diff)
Merge main loop into head. This probably breaks Win32, until
someone does the necessary updates. Sat Nov 28 12:53:47 1998 Owen Taylor <otaylor@redhat.com> * Makefile.am configure.in acconfig.h giochannel.c glib.h glist.c gmain.c gutils.c: - Revised GIOChannel to provide a generic virtual-function based interface. - Added unix fd-based GIOChannel's - Added generic main-loop abstraction - Added timeouts and idle functions using main-loop abstraction.
Diffstat (limited to 'gmain.c')
-rw-r--r--gmain.c731
1 files changed, 731 insertions, 0 deletions
diff --git a/gmain.c b/gmain.c
new file mode 100644
index 000000000..79c1191f1
--- /dev/null
+++ b/gmain.c
@@ -0,0 +1,731 @@
+/* GLIB - Library of useful routines for C programming
+ * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
+ *
+ * gmain.c: Main loop abstraction, timeouts, and idle functions
+ * Copyright 1998 Owen Taylor
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library 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.
+ */
+
+#include "glib.h"
+#include <sys/time.h>
+#include <unistd.h>
+#include "config.h"
+
+/* Types */
+
+typedef struct _GIdleData GIdleData;
+typedef struct _GTimeoutData GTimeoutData;
+typedef struct _GSource GSource;
+typedef struct _GPollRec GPollRec;
+
+typedef enum {
+ G_SOURCE_READY = 1 << G_HOOK_FLAG_USER_SHIFT,
+ G_SOURCE_CAN_RECURSE = 1 << (G_HOOK_FLAG_USER_SHIFT + 1)
+} GSourceFlags;
+
+struct _GSource {
+ GHook hook;
+ gint priority;
+ gpointer source_data;
+};
+
+struct _GMainLoop {
+ gboolean flag;
+};
+
+struct _GIdleData {
+ GSourceFunc callback;
+};
+
+struct _GTimeoutData {
+ GTimeVal expiration;
+ gint interval;
+ GSourceFunc callback;
+};
+
+struct _GPollRec {
+ gint priority;
+ GPollFD *fd;
+ GPollRec *next;
+};
+
+/* Forward declarations */
+
+static void g_main_poll (gint timeout,
+ gboolean use_priority,
+ gint priority);
+static gboolean g_timeout_prepare (gpointer source_data,
+ GTimeVal *current_time,
+ gint *timeout);
+static gboolean g_timeout_check (gpointer source_data,
+ GTimeVal *current_time);
+static gboolean g_timeout_dispatch (gpointer source_data,
+ GTimeVal *current_time,
+ gpointer user_data);
+static gboolean g_idle_prepare (gpointer source_data,
+ GTimeVal *current_time,
+ gint *timeout);
+static gboolean g_idle_check (gpointer source_data,
+ GTimeVal *current_time);
+static gboolean g_idle_dispatch (gpointer source_data,
+ GTimeVal *current_time,
+ gpointer user_data);
+
+/* Data */
+
+static GSList *pending_dispatches = NULL;
+static GHookList source_list = { 0 };
+
+static GSourceFuncs timeout_funcs = {
+ g_timeout_prepare,
+ g_timeout_check,
+ g_timeout_dispatch,
+ (GDestroyNotify)g_free
+};
+
+static GSourceFuncs idle_funcs = {
+ g_idle_prepare,
+ g_idle_check,
+ g_idle_dispatch,
+ (GDestroyNotify)g_free
+};
+
+#ifdef HAVE_POLL
+static GPollFunc poll_func = (GPollFunc)poll;
+#else
+
+/* The following implementation of poll() comes from the GNU C Library.
+ * Copyright (C) 1994, 1996, 1997 Free Software Foundation, Inc.
+ */
+
+#ifdef HAVE_SYS_SELECT_H
+#include <sys/select.h>
+#endif /* HAVE_SYS_SELECT_H_ */
+
+#ifndef NO_FD_SET
+# define SELECT_MASK fd_set
+#else
+# ifndef _AIX
+typedef long fd_mask;
+# endif
+# if defined(_IBMR2)
+# define SELECT_MASK void
+# else
+# define SELECT_MASK int
+# endif
+#endif
+
+static gint
+g_poll (GPollFD *fds, guint nfds, gint timeout)
+{
+ struct timeval tv;
+ SELECT_MASK rset, wset, xset;
+ GPollFD *f;
+ int ready;
+ int maxfd = 0;
+
+ FD_ZERO (&rset);
+ FD_ZERO (&wset);
+ FD_ZERO (&xset);
+
+ for (f = fds; f < &fds[nfds]; ++f)
+ if (f->fd >= 0)
+ {
+ if (f->events & G_IO_IN)
+ FD_SET (f->fd, &rset);
+ if (f->events & G_IO_OUT)
+ FD_SET (f->fd, &wset);
+ if (f->events & G_IO_PRI)
+ FD_SET (f->fd, &xset);
+ if (f->fd > maxfd && (f->events & (POLLIN|POLLOUT|POLLPRI)))
+ maxfd = f->fd;
+ }
+
+ tv.tv_sec = timeout / 1000;
+ tv.tv_usec = (timeout % 1000) * 1000;
+
+ ready = select (maxfd + 1, &rset, &wset, &xset,
+ timeout == -1 ? NULL : &tv);
+ if (ready > 0)
+ for (f = fds; f < &fds[nfds]; ++f)
+ {
+ f->revents = 0;
+ if (f->fd >= 0)
+ {
+ if (FD_ISSET (f->fd, &rset))
+ f->revents |= G_IO_IN;
+ if (FD_ISSET (f->fd, &wset))
+ f->revents |= G_IO_OUT;
+ if (FD_ISSET (f->fd, &xset))
+ f->revents |= G_IO_PRI;
+ }
+ }
+
+ return ready;
+}
+
+static GPollFunc poll_func = g_poll;
+#endif
+
+/* Hooks for adding to the main loop */
+
+/* Use knowledge of insert_sorted algorithm here to make
+ * sure we insert at the end of equal priority items
+ */
+static gint
+g_source_compare (GHook *a, GHook *b)
+{
+ GSource *source_a = (GSource *)a;
+ GSource *source_b = (GSource *)b;
+
+ return (source_a->priority < source_b->priority) ? -1 : 1;
+}
+
+guint
+g_source_add (gint priority,
+ gboolean can_recurse,
+ GSourceFuncs *funcs,
+ gpointer source_data,
+ gpointer user_data,
+ GDestroyNotify notify)
+{
+ GSource *source;
+
+ if (!source_list.is_setup)
+ g_hook_list_init (&source_list, sizeof(GSource));
+
+ source = (GSource *)g_hook_alloc (&source_list);
+ source->priority = priority;
+ source->source_data = source_data;
+ source->hook.func = funcs;
+ source->hook.data = user_data;
+ source->hook.destroy = notify;
+
+ g_hook_insert_sorted (&source_list,
+ (GHook *)source,
+ g_source_compare);
+
+ if (can_recurse)
+ source->hook.flags |= G_SOURCE_CAN_RECURSE;
+
+ return source->hook.hook_id;
+}
+
+void
+g_source_remove (guint tag)
+{
+ GHook *hook = g_hook_get (&source_list, tag);
+ if (hook)
+ {
+ GSource *source = (GSource *)hook;
+ ((GSourceFuncs *)source->hook.func)->destroy (source->source_data);
+ g_hook_destroy_link (&source_list, hook);
+ }
+}
+
+void
+g_source_remove_by_user_data (gpointer user_data)
+{
+ GHook *hook = g_hook_find_data (&source_list, TRUE, user_data);
+ if (hook)
+ {
+ GSource *source = (GSource *)hook;
+ ((GSourceFuncs *)source->hook.func)->destroy (source->source_data);
+ g_hook_destroy_link (&source_list, hook);
+ }
+}
+
+static gboolean
+g_source_find_source_data (GHook *hook,
+ gpointer data)
+{
+ GSource *source = (GSource *)hook;
+ return (source->source_data == data);
+}
+
+void
+g_source_remove_by_source_data (gpointer source_data)
+{
+ GHook *hook = g_hook_find (&source_list, TRUE,
+ g_source_find_source_data, source_data);
+ if (hook)
+ {
+ GSource *source = (GSource *)hook;
+ ((GSourceFuncs *)source->hook.func)->destroy (source->source_data);
+ g_hook_destroy_link (&source_list, hook);
+ }
+}
+
+void g_get_current_time (GTimeVal *result)
+{
+ gettimeofday ((struct timeval *)result, NULL);
+}
+
+/* Running the main loop */
+
+static void
+g_main_dispatch (GTimeVal *current_time)
+{
+ while (pending_dispatches != NULL)
+ {
+ gboolean need_destroy;
+ GSource *source = pending_dispatches->data;
+ GSList *tmp_list;
+
+ tmp_list = pending_dispatches;
+ pending_dispatches = g_slist_remove_link (pending_dispatches, pending_dispatches);
+ g_slist_free_1 (tmp_list);
+
+ if (G_HOOK_IS_VALID (source))
+ {
+ source->hook.flags |= G_HOOK_FLAG_IN_CALL;
+ need_destroy = !((GSourceFuncs *)source->hook.func)->dispatch (source->source_data,
+ current_time,
+ source->hook.data);
+ source->hook.flags &= ~G_HOOK_FLAG_IN_CALL;
+
+ if (need_destroy)
+ g_hook_destroy_link (&source_list, (GHook *)source);
+ }
+
+ g_hook_unref (&source_list, (GHook *)source);
+ }
+}
+
+/* Run a single iteration of the mainloop, or, if !dispatch
+ * check to see if any events need dispatching, but don't
+ * run the loop.
+ */
+static gboolean
+g_main_iterate (gboolean block, gboolean dispatch)
+{
+ GHook *hook;
+ GTimeVal current_time;
+ gint nready = 0;
+ gint current_priority = 0;
+ gint timeout;
+
+ g_return_val_if_fail (!block || dispatch, FALSE);
+
+ g_get_current_time (&current_time);
+
+ /* If recursing, finish up current dispatch, before starting over */
+ if (pending_dispatches)
+ {
+ if (dispatch)
+ g_main_dispatch (&current_time);
+
+ return TRUE;
+ }
+
+ /* Prepare all sources */
+
+ timeout = block ? -1 : 0;
+
+ hook = g_hook_first_valid (&source_list, TRUE);
+ while (hook)
+ {
+ GSource *source = (GSource *)hook;
+ GHook *tmp;
+ gint source_timeout;
+
+ if ((nready > 0) && (source->priority > current_priority))
+ break;
+ if (!(hook->flags & G_SOURCE_CAN_RECURSE) && G_HOOK_IN_CALL (hook))
+ {
+ hook = g_hook_next_valid (hook, TRUE);
+ continue;
+ }
+
+ g_hook_ref (&source_list, hook);
+
+ if (((GSourceFuncs *)hook->func)->prepare (source->source_data,
+ &current_time,
+ &source_timeout))
+ {
+ if (!dispatch)
+ {
+ g_hook_unref (&source_list, hook);
+ return TRUE;
+ }
+ else
+ {
+ hook->flags |= G_SOURCE_READY;
+ nready++;
+ current_priority = source->priority;
+ timeout = 0;
+ }
+ }
+
+ if (source_timeout >= 0)
+ {
+ if (timeout < 0)
+ timeout = source_timeout;
+ else
+ timeout = MIN (timeout, source_timeout);
+ }
+
+ tmp = g_hook_next_valid (hook, TRUE);
+
+ g_hook_unref (&source_list, hook);
+ hook = tmp;
+ }
+
+ /* poll(), if necessary */
+
+ g_main_poll (timeout, nready > 0, current_priority);
+
+ /* Check to see what sources need to be dispatched */
+
+ nready = 0;
+
+ hook = g_hook_first_valid (&source_list, TRUE);
+ while (hook)
+ {
+ GSource *source = (GSource *)hook;
+ GHook *tmp;
+
+ if ((nready > 0) && (source->priority > current_priority))
+ break;
+ if (!(hook->flags & G_SOURCE_CAN_RECURSE) && G_HOOK_IN_CALL (hook))
+ {
+ hook = g_hook_next_valid (hook, TRUE);
+ continue;
+ }
+
+ g_hook_ref (&source_list, hook);
+
+ if ((hook->flags & G_SOURCE_READY) ||
+ ((GSourceFuncs *)hook->func)->check (source->source_data,
+ &current_time))
+ {
+ if (dispatch)
+ {
+ hook->flags &= ~G_SOURCE_READY;
+ g_hook_ref (&source_list, hook);
+ pending_dispatches = g_slist_prepend (pending_dispatches, source);
+ current_priority = source->priority;
+ nready++;
+ }
+ else
+ {
+ g_hook_unref (&source_list, hook);
+ return TRUE;
+ }
+ }
+
+ tmp = g_hook_next_valid (hook, TRUE);
+
+ g_hook_unref (&source_list, hook);
+ hook = tmp;
+ }
+
+ /* Now invoke the callbacks */
+
+ if (pending_dispatches)
+ {
+ pending_dispatches = g_slist_reverse (pending_dispatches);
+ g_main_dispatch (&current_time);
+ return TRUE;
+ }
+ else
+ return FALSE;
+}
+
+/* See if any events are pending
+ */
+gboolean
+g_main_pending ()
+{
+ return g_main_iterate (FALSE, FALSE);
+}
+
+/* Run a single iteration of the mainloop. If block is FALSE,
+ * will never block
+ */
+gboolean
+g_main_iteration (gboolean block)
+{
+ return g_main_iterate (block, TRUE);
+}
+
+GMainLoop *
+g_main_new ()
+{
+ GMainLoop *result = g_new (GMainLoop, 1);
+ result->flag = FALSE;
+
+ return result;
+}
+
+void
+g_main_run (GMainLoop *loop)
+{
+ loop->flag = FALSE;
+ while (!loop->flag)
+ g_main_iterate (TRUE, TRUE);
+}
+
+void
+g_main_quit (GMainLoop *loop)
+{
+ loop->flag = TRUE;
+}
+
+void
+g_main_destroy (GMainLoop *loop)
+{
+ g_free (loop);
+}
+
+static GPollRec *poll_records = NULL;
+static GPollRec *poll_free_list = NULL;
+static GMemChunk *poll_chunk;
+static guint n_poll_records = 0;
+
+static void
+g_main_poll (gint timeout, gboolean use_priority, gint priority)
+{
+ GPollFD *fd_array = g_new (GPollFD, n_poll_records);
+ GPollRec *pollrec;
+
+ gint i;
+ gint npoll;
+
+ pollrec = poll_records;
+ i = 0;
+ while (pollrec && (!use_priority || priority >= pollrec->priority))
+ {
+ fd_array[i].fd = pollrec->fd->fd;
+ fd_array[i].events = pollrec->fd->events;
+ fd_array[i].revents = 0;
+
+ pollrec = pollrec->next;
+ i++;
+ }
+
+ npoll = i;
+ (*poll_func) (fd_array, npoll, timeout);
+
+ pollrec = poll_records;
+ i = 0;
+ while (i < npoll)
+ {
+ pollrec->fd->revents = fd_array[i].revents;
+ pollrec = pollrec->next;
+ i++;
+ }
+
+ g_free (fd_array);
+}
+
+void
+g_main_poll_add (gint priority,
+ GPollFD *fd)
+{
+ GPollRec *lastrec, *pollrec, *newrec;
+
+ if (!poll_chunk)
+ poll_chunk = g_mem_chunk_create (GPollRec, 32, G_ALLOC_ONLY);
+
+ newrec = g_chunk_new (GPollRec, poll_chunk);
+ newrec->fd = fd;
+ newrec->priority = priority;
+
+ lastrec = NULL;
+ pollrec = poll_records;
+ while (pollrec && priority >= pollrec->priority)
+ {
+ lastrec = pollrec;
+ pollrec = pollrec->next;
+ }
+
+ if (lastrec)
+ lastrec->next = newrec;
+ else
+ poll_records = newrec;
+
+ newrec->next = pollrec;
+
+ n_poll_records++;
+}
+
+void
+g_main_poll_remove (GPollFD *fd)
+{
+ GPollRec *pollrec, *lastrec;
+
+ lastrec = NULL;
+ pollrec = poll_records;
+
+ while (pollrec)
+ {
+ if (pollrec->fd == fd)
+ {
+ if (lastrec != NULL)
+ lastrec->next = pollrec->next;
+ else
+ poll_records = pollrec->next;
+
+ pollrec->next = poll_free_list;
+ poll_free_list = pollrec;
+ }
+ lastrec = pollrec;
+ pollrec = pollrec->next;
+ }
+
+ n_poll_records--;
+}
+
+void
+g_main_set_poll_func (GPollFunc func)
+{
+ if (func)
+ poll_func = func;
+ else
+ poll_func = (GPollFunc)poll;
+}
+
+/* Timeouts */
+
+static gboolean
+g_timeout_prepare (gpointer source_data,
+ GTimeVal *current_time,
+ gint *timeout)
+{
+ glong msec;
+ GTimeoutData *data = source_data;
+
+ msec = (data->expiration.tv_sec - current_time->tv_sec) * 1000 +
+ (data->expiration.tv_usec - current_time->tv_usec) / 1000;
+
+ *timeout = (msec <= 0) ? 0 : msec;
+
+ return (msec <= 0);
+}
+
+static gboolean
+g_timeout_check (gpointer source_data,
+ GTimeVal *current_time)
+{
+ GTimeoutData *data = source_data;
+
+ return (data->expiration.tv_sec < current_time->tv_sec) ||
+ ((data->expiration.tv_sec == current_time->tv_sec) &&
+ (data->expiration.tv_usec <= current_time->tv_usec));
+}
+
+static gboolean
+g_timeout_dispatch (gpointer source_data,
+ GTimeVal *current_time,
+ gpointer user_data)
+{
+ GTimeoutData *data = source_data;
+
+ if (data->callback(user_data))
+ {
+ data->expiration.tv_sec = current_time->tv_sec;
+ data->expiration.tv_usec = current_time->tv_usec + data->interval * 1000;
+ if (data->expiration.tv_usec >= 1000000)
+ {
+ data->expiration.tv_usec -= 1000000;
+ data->expiration.tv_sec++;
+ }
+ return TRUE;
+ }
+ else
+ return FALSE;
+}
+
+guint
+g_timeout_add_full (gint priority,
+ guint interval,
+ GSourceFunc function,
+ gpointer data,
+ GDestroyNotify notify)
+{
+ GTimeoutData *timeout_data = g_new (GTimeoutData, 1);
+
+ timeout_data->interval = interval;
+ timeout_data->callback = function;
+ g_get_current_time (&timeout_data->expiration);
+
+ timeout_data->expiration.tv_usec += timeout_data->interval * 1000;
+ if (timeout_data->expiration.tv_usec >= 1000000)
+ {
+ timeout_data->expiration.tv_usec -= 1000000;
+ timeout_data->expiration.tv_sec++;
+ }
+
+ return g_source_add (priority, FALSE, &timeout_funcs, timeout_data, data, notify);
+}
+
+guint
+g_timeout_add (guint32 interval,
+ GSourceFunc function,
+ gpointer data)
+{
+ return g_timeout_add_full (0, interval, function, data, NULL);
+}
+
+/* Idle functions */
+
+static gboolean
+g_idle_prepare (gpointer source_data,
+ GTimeVal *current_time,
+ gint *timeout)
+{
+ timeout = 0;
+ return TRUE;
+}
+
+static gboolean
+g_idle_check (gpointer source_data,
+ GTimeVal *current_time)
+{
+ return TRUE;
+}
+
+static gboolean
+g_idle_dispatch (gpointer source_data,
+ GTimeVal *current_time,
+ gpointer user_data)
+{
+ GIdleData *data = source_data;
+
+ return (*data->callback)(user_data);
+}
+
+guint
+g_idle_add_full (gint priority,
+ GSourceFunc function,
+ gpointer data,
+ GDestroyNotify notify)
+{
+ GIdleData *idle_data = g_new (GIdleData, 1);
+
+ idle_data->callback = function;
+
+ return g_source_add (priority, FALSE, &idle_funcs, idle_data, data, notify);
+}
+
+guint
+g_idle_add (GSourceFunc function,
+ gpointer data)
+{
+ return g_idle_add_full (0, function, data, NULL);
+}