The Main Event Loop

The Main Event Loop — manages all available sources of events

Synopsis


#include <glib.h>

                    GMainLoop;
GMainLoop*          g_main_loop_new                     (GMainContext *context,
                                                         gboolean is_running);
GMainLoop*          g_main_loop_ref                     (GMainLoop *loop);
void                g_main_loop_unref                   (GMainLoop *loop);
void                g_main_loop_run                     (GMainLoop *loop);
void                g_main_loop_quit                    (GMainLoop *loop);
gboolean            g_main_loop_is_running              (GMainLoop *loop);
GMainContext*       g_main_loop_get_context             (GMainLoop *loop);
#define             g_main_new                          (is_running)
#define             g_main_destroy                      (loop)
#define             g_main_run                          (loop)
#define             g_main_quit                         (loop)
#define             g_main_is_running                   (loop)

#define             G_PRIORITY_HIGH
#define             G_PRIORITY_DEFAULT
#define             G_PRIORITY_HIGH_IDLE
#define             G_PRIORITY_DEFAULT_IDLE
#define             G_PRIORITY_LOW

                    GMainContext;
GMainContext*       g_main_context_new                  (void);
GMainContext*       g_main_context_ref                  (GMainContext *context);
void                g_main_context_unref                (GMainContext *context);
GMainContext*       g_main_context_default              (void);
gboolean            g_main_context_iteration            (GMainContext *context,
                                                         gboolean may_block);
#define             g_main_iteration                    (may_block)
gboolean            g_main_context_pending              (GMainContext *context);
#define             g_main_pending                      ()
GSource*            g_main_context_find_source_by_id    (GMainContext *context,
                                                         guint source_id);
GSource*            g_main_context_find_source_by_user_data
                                                        (GMainContext *context,
                                                         gpointer user_data);
GSource*            g_main_context_find_source_by_funcs_user_data
                                                        (GMainContext *context,
                                                         GSourceFuncs *funcs,
                                                         gpointer user_data);
void                g_main_context_wakeup               (GMainContext *context);
gboolean            g_main_context_acquire              (GMainContext *context);
void                g_main_context_release              (GMainContext *context);
gboolean            g_main_context_is_owner             (GMainContext *context);
gboolean            g_main_context_wait                 (GMainContext *context,
                                                         GCond *cond,
                                                         GMutex *mutex);
gboolean            g_main_context_prepare              (GMainContext *context,
                                                         gint *priority);
gint                g_main_context_query                (GMainContext *context,
                                                         gint max_priority,
                                                         gint *timeout_,
                                                         GPollFD *fds,
                                                         gint n_fds);
gint                g_main_context_check                (GMainContext *context,
                                                         gint max_priority,
                                                         GPollFD *fds,
                                                         gint n_fds);
void                g_main_context_dispatch             (GMainContext *context);
void                g_main_context_set_poll_func        (GMainContext *context,
                                                         GPollFunc func);
GPollFunc           g_main_context_get_poll_func        (GMainContext *context);
gint                (*GPollFunc)                        (GPollFD *ufds,
                                                         guint nfsd,
                                                         gint timeout_);
void                g_main_context_add_poll             (GMainContext *context,
                                                         GPollFD *fd,
                                                         gint priority);
void                g_main_context_remove_poll          (GMainContext *context,
                                                         GPollFD *fd);
gint                g_main_depth                        (void);
GSource*            g_main_current_source               (void);
#define             g_main_set_poll_func                (func)

GSource*            g_timeout_source_new                (guint interval);
GSource*            g_timeout_source_new_seconds        (guint interval);
guint               g_timeout_add                       (guint interval,
                                                         GSourceFunc function,
                                                         gpointer data);
guint               g_timeout_add_full                  (gint priority,
                                                         guint interval,
                                                         GSourceFunc function,
                                                         gpointer data,
                                                         GDestroyNotify notify);
guint               g_timeout_add_seconds               (guint interval,
                                                         GSourceFunc function,
                                                         gpointer data);
guint               g_timeout_add_seconds_full          (gint priority,
                                                         guint interval,
                                                         GSourceFunc function,
                                                         gpointer data,
                                                         GDestroyNotify notify);

GSource*            g_idle_source_new                   (void);
guint               g_idle_add                          (GSourceFunc function,
                                                         gpointer data);
guint               g_idle_add_full                     (gint priority,
                                                         GSourceFunc function,
                                                         gpointer data,
                                                         GDestroyNotify notify);
gboolean            g_idle_remove_by_data               (gpointer data);

void                (*GChildWatchFunc)                  (GPid pid,
                                                         gint status,
                                                         gpointer data);
GSource*            g_child_watch_source_new            (GPid pid);
guint               g_child_watch_add                   (GPid pid,
                                                         GChildWatchFunc function,
                                                         gpointer data);
guint               g_child_watch_add_full              (gint priority,
                                                         GPid pid,
                                                         GChildWatchFunc function,
                                                         gpointer data,
                                                         GDestroyNotify notify);

                    GPollFD;

                    GSource;
void                (*GSourceDummyMarshal)              (void);
                    GSourceFuncs;
                    GSourceCallbackFuncs;
GSource*            g_source_new                        (GSourceFuncs *source_funcs,
                                                         guint struct_size);
GSource*            g_source_ref                        (GSource *source);
void                g_source_unref                      (GSource *source);
void                g_source_set_funcs                  (GSource *source,
                                                         GSourceFuncs *funcs);
guint               g_source_attach                     (GSource *source,
                                                         GMainContext *context);
void                g_source_destroy                    (GSource *source);
gboolean            g_source_is_destroyed               (GSource *source);
void                g_source_set_priority               (GSource *source,
                                                         gint priority);
gint                g_source_get_priority               (GSource *source);
void                g_source_set_can_recurse            (GSource *source,
                                                         gboolean can_recurse);
gboolean            g_source_get_can_recurse            (GSource *source);
guint               g_source_get_id                     (GSource *source);
GMainContext*       g_source_get_context                (GSource *source);
void                g_source_set_callback               (GSource *source,
                                                         GSourceFunc func,
                                                         gpointer data,
                                                         GDestroyNotify notify);
gboolean            (*GSourceFunc)                      (gpointer data);
void                g_source_set_callback_indirect      (GSource *source,
                                                         gpointer callback_data,
                                                         GSourceCallbackFuncs *callback_funcs);
void                g_source_add_poll                   (GSource *source,
                                                         GPollFD *fd);
void                g_source_remove_poll                (GSource *source,
                                                         GPollFD *fd);
void                g_source_get_current_time           (GSource *source,
                                                         GTimeVal *timeval);
gboolean            g_source_remove                     (guint tag);
gboolean            g_source_remove_by_funcs_user_data  (GSourceFuncs *funcs,
                                                         gpointer user_data);
gboolean            g_source_remove_by_user_data        (gpointer user_data);

Description

The main event loop manages all the available sources of events for GLib and GTK+ applications. These events can come from any number of different types of sources such as file descriptors (plain files, pipes or sockets) and timeouts. New types of event sources can also be added using g_source_attach().

To allow multiple independent sets of sources to be handled in different threads, each source is associated with a GMainContext. A GMainContext can only be running in a single thread, but sources can be added to it and removed from it from other threads.

Each event source is assigned a priority. The default priority, G_PRIORITY_DEFAULT, is 0. Values less than 0 denote higher priorities. Values greater than 0 denote lower priorities. Events from high priority sources are always processed before events from lower priority sources.

Idle functions can also be added, and assigned a priority. These will be run whenever no events with a higher priority are ready to be processed.

The GMainLoop data type represents a main event loop. A GMainLoop is created with g_main_loop_new(). After adding the initial event sources, g_main_loop_run() is called. This continuously checks for new events from each of the event sources and dispatches them. Finally, the processing of an event from one of the sources leads to a call to g_main_loop_quit() to exit the main loop, and g_main_loop_run() returns.

It is possible to create new instances of GMainLoop recursively. This is often used in GTK+ applications when showing modal dialog boxes. Note that event sources are associated with a particular GMainContext, and will be checked and dispatched for all main loops associated with that GMainContext.

GTK+ contains wrappers of some of these functions, e.g. gtk_main(), gtk_main_quit() and gtk_events_pending().

Creating new sources types

One of the unusual features of the GTK+ main loop functionality is that new types of event source can be created and used in addition to the builtin type of event source. A new event source type is used for handling GDK events. A new source type is created by deriving from the GSource structure. The derived type of source is represented by a structure that has the GSource structure as a first element, and other elements specific to the new source type. To create an instance of the new source type, call g_source_new() passing in the size of the derived structure and a table of functions. These GSourceFuncs determine the behavior of the new source types.

New source types basically interact with the main context in two ways. Their prepare function in GSourceFuncs can set a timeout to determine the maximum amount of time that the main loop will sleep before checking the source again. In addition, or as well, the source can add file descriptors to the set that the main context checks using g_source_add_poll().


Customizing the main loop iteration

Single iterations of a GMainContext can be run with g_main_context_iteration(). In some cases, more detailed control of exactly how the details of the main loop work is desired, for instance, when integrating the GMainLoop with an external main loop. In such cases, you can call the component functions of g_main_context_iteration() directly. These functions are g_main_context_prepare(), g_main_context_query(), g_main_context_check() and g_main_context_dispatch().

The operation of these functions can best be seen in terms of a state diagram, as shown in Figure 1, “States of a Main Context”.

Figure 1. States of a Main Context

States of a Main Context

Details

GMainLoop

typedef struct _GMainLoop GMainLoop;

The GMainLoop struct is an opaque data type representing the main event loop of a GLib or GTK+ application.


g_main_loop_new ()

GMainLoop*          g_main_loop_new                     (GMainContext *context,
                                                         gboolean is_running);

Creates a new GMainLoop structure.

context : a GMainContext (if NULL, the default context will be used).
is_running : set to TRUE to indicate that the loop is running. This is not very important since calling g_main_loop_run() will set this to TRUE anyway.
Returns : a new GMainLoop.

g_main_loop_ref ()

GMainLoop*          g_main_loop_ref                     (GMainLoop *loop);

Increases the reference count on a GMainLoop object by one.

loop : a GMainLoop
Returns : loop

g_main_loop_unref ()

void                g_main_loop_unref                   (GMainLoop *loop);

Decreases the reference count on a GMainLoop object by one. If the result is zero, free the loop and free all associated memory.

loop : a GMainLoop

g_main_loop_run ()

void                g_main_loop_run                     (GMainLoop *loop);

Runs a main loop until g_main_loop_quit() is called on the loop. If this is called for the thread of the loop's GMainContext, it will process events from the loop, otherwise it will simply wait.

loop : a GMainLoop

g_main_loop_quit ()

void                g_main_loop_quit                    (GMainLoop *loop);

Stops a GMainLoop from running. Any calls to g_main_loop_run() for the loop will return.

Note that sources that have already been dispatched when g_main_loop_quit() is called will still be executed.

loop : a GMainLoop

g_main_loop_is_running ()

gboolean            g_main_loop_is_running              (GMainLoop *loop);

Checks to see if the main loop is currently being run via g_main_loop_run().

loop : a GMainLoop.
Returns : TRUE if the mainloop is currently being run.

g_main_loop_get_context ()

GMainContext*       g_main_loop_get_context             (GMainLoop *loop);

Returns the GMainContext of loop.

loop : a GMainLoop.
Returns : the GMainContext of loop

g_main_new()

#define             g_main_new(is_running)

Warning

g_main_new has been deprecated since version 2.2 and should not be used in newly-written code. Use g_main_loop_new() instead.

Creates a new GMainLoop for the default main loop.

is_running : set to TRUE to indicate that the loop is running. This is not very important since calling g_main_run() will set this to TRUE anyway.
Returns : a new GMainLoop.

g_main_destroy()

#define             g_main_destroy(loop)

Warning

g_main_destroy has been deprecated since version 2.2 and should not be used in newly-written code. Use g_main_loop_unref() instead.

Frees the memory allocated for the GMainLoop.

loop : a GMainLoop.

g_main_run()

#define             g_main_run(loop)

Warning

g_main_run has been deprecated since version 2.2 and should not be used in newly-written code. Use g_main_loop_run() instead.

Runs a main loop until it stops running.

loop : a GMainLoop.

g_main_quit()

#define             g_main_quit(loop)

Warning

g_main_quit has been deprecated since version 2.2 and should not be used in newly-written code. Use g_main_loop_quit() instead.

Stops the GMainLoop. If g_main_run() was called to run the GMainLoop, it will now return.

loop : a GMainLoop.

g_main_is_running()

#define             g_main_is_running(loop)

Warning

g_main_is_running has been deprecated since version 2.2 and should not be used in newly-written code. USe g_main_loop_is_running() instead.

Checks if the main loop is running.

loop : a GMainLoop.
Returns : TRUE if the main loop is running.

G_PRIORITY_HIGH

#define G_PRIORITY_HIGH            -100

Use this for high priority event sources. It is not used within GLib or GTK+.


G_PRIORITY_DEFAULT

#define G_PRIORITY_DEFAULT          0

Use this for default priority event sources. In GLib this priority is used when adding timeout functions with g_timeout_add(). In GDK this priority is used for events from the X server.


G_PRIORITY_HIGH_IDLE

#define G_PRIORITY_HIGH_IDLE        100

Use this for high priority idle functions. GTK+ uses G_PRIORITY_HIGH_IDLE + 10 for resizing operations, and G_PRIORITY_HIGH_IDLE + 20 for redrawing operations. (This is done to ensure that any pending resizes are processed before any pending redraws, so that widgets are not redrawn twice unnecessarily.)


G_PRIORITY_DEFAULT_IDLE

#define G_PRIORITY_DEFAULT_IDLE     200

Use this for default priority idle functions. In GLib this priority is used when adding idle functions with g_idle_add().


G_PRIORITY_LOW

#define G_PRIORITY_LOW	            300

Use this for very low priority background tasks. It is not used within GLib or GTK+.


GMainContext

typedef struct _GMainContext GMainContext;

The GMainContext struct is an opaque data type representing a set of sources to be handled in a main loop.


g_main_context_new ()

GMainContext*       g_main_context_new                  (void);

Creates a new GMainContext structure.

Returns : the new GMainContext

g_main_context_ref ()

GMainContext*       g_main_context_ref                  (GMainContext *context);

Increases the reference count on a GMainContext object by one.

context : a GMainContext
Returns : the context that was passed in (since 2.6)

g_main_context_unref ()

void                g_main_context_unref                (GMainContext *context);

Decreases the reference count on a GMainContext object by one. If the result is zero, free the context and free all associated memory.

context : a GMainContext

g_main_context_default ()

GMainContext*       g_main_context_default              (void);

Returns the default main context. This is the main context used for main loop functions when a main loop is not explicitly specified.

Returns : the default main context.

g_main_context_iteration ()

gboolean            g_main_context_iteration            (GMainContext *context,
                                                         gboolean may_block);

Runs a single iteration for the given main loop. This involves checking to see if any event sources are ready to be processed, then if no events sources are ready and may_block is TRUE, waiting for a source to become ready, then dispatching the highest priority events sources that are ready. Otherwise, if may_block is FALSE sources are not waited to become ready, only those highest priority events sources will be dispatched (if any), that are ready at this given moment without further waiting.

Note that even when may_block is TRUE, it is still possible for g_main_context_iteration() to return FALSE, since the the wait may be interrupted for other reasons than an event source becoming ready.

context : a GMainContext (if NULL, the default context will be used)
may_block : whether the call may block.
Returns : TRUE if events were dispatched.

g_main_iteration()

#define             g_main_iteration(may_block)

Warning

g_main_iteration has been deprecated since version 2.2 and should not be used in newly-written code. Use g_main_context_iteration() instead.

Runs a single iteration for the default GMainContext.

may_block : set to TRUE if it should block (i.e. wait) until an event source becomes ready. It will return after an event source has been processed. If set to FALSE it will return immediately if no event source is ready to be processed.
Returns : TRUE if more events are pending.

g_main_context_pending ()

gboolean            g_main_context_pending              (GMainContext *context);

Checks if any sources have pending events for the given context.

context : a GMainContext (if NULL, the default context will be used)
Returns : TRUE if events are pending.

g_main_pending()

#define             g_main_pending()

Warning

g_main_pending has been deprecated since version 2.2 and should not be used in newly-written code. Use g_main_context_pending() instead.

Checks if any events are pending for the default GMainContext (i.e. ready to be processed).

Returns : TRUE if any events are pending.

g_main_context_find_source_by_id ()

GSource*            g_main_context_find_source_by_id    (GMainContext *context,
                                                         guint source_id);

Finds a GSource given a pair of context and ID.

context : a GMainContext (if NULL, the default context will be used)
source_id : the source ID, as returned by g_source_get_id().
Returns : the GSource if found, otherwise, NULL

g_main_context_find_source_by_user_data ()

GSource*            g_main_context_find_source_by_user_data
                                                        (GMainContext *context,
                                                         gpointer user_data);

Finds a source with the given user data for the callback. If multiple sources exist with the same user data, the first one found will be returned.

context : a GMainContext
user_data : the user_data for the callback.
Returns : the source, if one was found, otherwise NULL

g_main_context_find_source_by_funcs_user_data ()

GSource*            g_main_context_find_source_by_funcs_user_data
                                                        (GMainContext *context,
                                                         GSourceFuncs *funcs,
                                                         gpointer user_data);

Finds a source with the given source functions and user data. If multiple sources exist with the same source function and user data, the first one found will be returned.

context : a GMainContext (if NULL, the default context will be used).
funcs : the source_funcs passed to g_source_new().
user_data : the user data from the callback.
Returns : the source, if one was found, otherwise NULL

g_main_context_wakeup ()

void                g_main_context_wakeup               (GMainContext *context);

If context is currently waiting in a poll(), interrupt the poll(), and continue the iteration process.

context : a GMainContext

g_main_context_acquire ()

gboolean            g_main_context_acquire              (GMainContext *context);

Tries to become the owner of the specified context. If some other thread is the owner of the context, returns FALSE immediately. Ownership is properly recursive: the owner can require ownership again and will release ownership when g_main_context_release() is called as many times as g_main_context_acquire().

You must be the owner of a context before you can call g_main_context_prepare(), g_main_context_query(), g_main_context_check(), g_main_context_dispatch().

context : a GMainContext
Returns : TRUE if the operation succeeded, and this thread is now the owner of context.

g_main_context_release ()

void                g_main_context_release              (GMainContext *context);

Releases ownership of a context previously acquired by this thread with g_main_context_acquire(). If the context was acquired multiple times, the ownership will be released only when g_main_context_release() is called as many times as it was acquired.

context : a GMainContext

g_main_context_is_owner ()

gboolean            g_main_context_is_owner             (GMainContext *context);

Determines whether this thread holds the (recursive) ownership of this GMaincontext. This is useful to know before waiting on another thread that may be blocking to get ownership of context.

context : a GMainContext
Returns : TRUE if current thread is owner of context.

Since 2.10


g_main_context_wait ()

gboolean            g_main_context_wait                 (GMainContext *context,
                                                         GCond *cond,
                                                         GMutex *mutex);

Tries to become the owner of the specified context, as with g_main_context_acquire(). But if another thread is the owner, atomically drop mutex and wait on cond until that owner releases ownership or until cond is signaled, then try again (once) to become the owner.

context : a GMainContext
cond : a condition variable
mutex : a mutex, currently held
Returns : TRUE if the operation succeeded, and this thread is now the owner of context.

g_main_context_prepare ()

gboolean            g_main_context_prepare              (GMainContext *context,
                                                         gint *priority);

Prepares to poll sources within a main loop. The resulting information for polling is determined by calling g_main_context_query().

context : a GMainContext
priority : location to store priority of highest priority source already ready.
Returns : TRUE if some source is ready to be dispatched prior to polling.

g_main_context_query ()

gint                g_main_context_query                (GMainContext *context,
                                                         gint max_priority,
                                                         gint *timeout_,
                                                         GPollFD *fds,
                                                         gint n_fds);

Determines information necessary to poll this main loop.

context : a GMainContext
max_priority : maximum priority source to check
timeout_ : location to store timeout to be used in polling
fds : location to store GPollFD records that need to be polled.
n_fds : length of fds.
Returns : the number of records actually stored in fds, or, if more than n_fds records need to be stored, the number of records that need to be stored.

g_main_context_check ()

gint                g_main_context_check                (GMainContext *context,
                                                         gint max_priority,
                                                         GPollFD *fds,
                                                         gint n_fds);

Passes the results of polling back to the main loop.

context : a GMainContext
max_priority : the maximum numerical priority of sources to check
fds : array of GPollFD's that was passed to the last call to g_main_context_query()
n_fds : return value of g_main_context_query()
Returns : TRUE if some sources are ready to be dispatched.

g_main_context_dispatch ()

void                g_main_context_dispatch             (GMainContext *context);

Dispatches all pending sources.

context : a GMainContext

g_main_context_set_poll_func ()

void                g_main_context_set_poll_func        (GMainContext *context,
                                                         GPollFunc func);

Sets the function to use to handle polling of file descriptors. It will be used instead of the poll() system call (or GLib's replacement function, which is used where poll() isn't available).

This function could possibly be used to integrate the GLib event loop with an external event loop.

context : a GMainContext
func : the function to call to poll all file descriptors

g_main_context_get_poll_func ()

GPollFunc           g_main_context_get_poll_func        (GMainContext *context);

Gets the poll function set by g_main_context_set_poll_func().

context : a GMainContext
Returns : the poll function

GPollFunc ()

gint                (*GPollFunc)                        (GPollFD *ufds,
                                                         guint nfsd,
                                                         gint timeout_);

Specifies the type of function passed to g_main_context_set_poll_func(). The semantics of the function should match those of the poll() system call.

ufds : an array of GPollFD elements.
nfsd : the number of elements in ufds.
timeout_ : the maximum time to wait for an event of the file descriptors. A negative value indicates an infinite timeout.
Returns : the number of GPollFD elements which have events or errors reported, or -1 if an error occurred.

g_main_context_add_poll ()

void                g_main_context_add_poll             (GMainContext *context,
                                                         GPollFD *fd,
                                                         gint priority);

Adds a file descriptor to the set of file descriptors polled for this context. This will very seldomly be used directly. Instead a typical event source will use g_source_add_poll() instead.

context : a GMainContext (or NULL for the default context)
fd : a GPollFD structure holding information about a file descriptor to watch.
priority : the priority for this file descriptor which should be the same as the priority used for g_source_attach() to ensure that the file descriptor is polled whenever the results may be needed.

g_main_context_remove_poll ()

void                g_main_context_remove_poll          (GMainContext *context,
                                                         GPollFD *fd);

Removes file descriptor from the set of file descriptors to be polled for a particular context.

context : a GMainContext
fd : a GPollFD descriptor previously added with g_main_context_add_poll()

g_main_depth ()

gint                g_main_depth                        (void);

Returns the depth of the stack of calls to g_main_context_dispatch() on any GMainContext in the current thread. That is, when called from the toplevel, it gives 0. When called from within a callback from g_main_context_iteration() (or g_main_loop_run(), etc.) it returns 1. When called from within a callback to a recursive call to g_main_context_iterate(), it returns 2. And so forth.

This function is useful in a situation like the following: Imagine an extremely simple "garbage collected" system.

static GList *free_list;

gpointer
allocate_memory (gsize size)
{ 
  gpointer result = g_malloc (size);
  free_list = g_list_prepend (free_list, result);
  return result;
}

void
free_allocated_memory (void)
{
  GList *l;
  for (l = free_list; l; l = l->next);
    g_free (l->data);
  g_list_free (free_list);
  free_list = NULL;
 }

[...]

while (TRUE); 
 {
   g_main_context_iteration (NULL, TRUE);
   free_allocated_memory();
  }

This works from an application, however, if you want to do the same thing from a library, it gets more difficult, since you no longer control the main loop. You might think you can simply use an idle function to make the call to free_allocated_memory(), but that doesn't work, since the idle function could be called from a recursive callback. This can be fixed by using g_main_depth()

gpointer
allocate_memory (gsize size)
{ 
  FreeListBlock *block = g_new (FreeListBlock, 1);
  block->mem = g_malloc (size);
  block->depth = g_main_depth ();   
  free_list = g_list_prepend (free_list, block);
  return block->mem;
}

void
free_allocated_memory (void)
{
  GList *l;
  
  int depth = g_main_depth ();
  for (l = free_list; l; );
    {
      GList *next = l->next;
      FreeListBlock *block = l->data;
      if (block->depth > depth)
        {
          g_free (block->mem);
          g_free (block);
          free_list = g_list_delete_link (free_list, l);
        }
              
      l = next;
    }
  }

There is a temptation to use g_main_depth() to solve problems with reentrancy. For instance, while waiting for data to be received from the network in response to a menu item, the menu item might be selected again. It might seem that one could make the menu item's callback return immediately and do nothing if g_main_depth() returns a value greater than 1. However, this should be avoided since the user then sees selecting the menu item do nothing. Furthermore, you'll find yourself adding these checks all over your code, since there are doubtless many, many things that the user could do. Instead, you can use the following techniques:

  1. Use gtk_widget_set_sensitive() or modal dialogs to prevent the user from interacting with elements while the main loop is recursing.

  2. Avoid main loop recursion in situations where you can't handle arbitrary callbacks. Instead, structure your code so that you simply return to the main loop and then get called again when there is more work to do.

Returns : The main loop recursion level in the current thread

g_main_current_source ()

GSource*            g_main_current_source               (void);

Returns the currently firing source for this thread.

Returns : The currently firing source or NULL.

Since 2.12


g_main_set_poll_func()

#define             g_main_set_poll_func(func)

Warning

g_main_set_poll_func has been deprecated since version 2.2 and should not be used in newly-written code. Use g_main_context_set_poll_func() instead.

Sets the function to use for the handle polling of file descriptors for the default main context.

func : the function to call to poll all file descriptors.

g_timeout_source_new ()

GSource*            g_timeout_source_new                (guint interval);

Creates a new timeout source.

The source will not initially be associated with any GMainContext and must be added to one with g_source_attach() before it will be executed.

interval : the timeout interval in milliseconds.
Returns : the newly-created timeout source

g_timeout_source_new_seconds ()

GSource*            g_timeout_source_new_seconds        (guint interval);

Creates a new timeout source.

The source will not initially be associated with any GMainContext and must be added to one with g_source_attach() before it will be executed.

The scheduling granularity/accuracy of this timeout source will be in seconds.

interval : the timeout interval in seconds
Returns : the newly-created timeout source

Since 2.14


g_timeout_add ()

guint               g_timeout_add                       (guint interval,
                                                         GSourceFunc function,
                                                         gpointer data);

Sets a function to be called at regular intervals, with the default priority, G_PRIORITY_DEFAULT. The function is called repeatedly until it returns FALSE, at which point the timeout is automatically destroyed and the function will not be called again. The first call to the function will be at the end of the first interval.

Note that timeout functions may be delayed, due to the processing of other event sources. Thus they should not be relied on for precise timing. After each call to the timeout function, the time of the next timeout is recalculated based on the current time and the given interval (it does not try to 'catch up' time lost in delays).

If you want to have a timer in the "seconds" range and do not care about the exact time of the first call of the timer, use the g_timeout_add_seconds() function; this function allows for more optimizations and more efficient system power usage.

interval : the time between calls to the function, in milliseconds (1/1000ths of a second)
function : function to call
data : data to pass to function
Returns : the ID (greater than 0) of the event source.

g_timeout_add_full ()

guint               g_timeout_add_full                  (gint priority,
                                                         guint interval,
                                                         GSourceFunc function,
                                                         gpointer data,
                                                         GDestroyNotify notify);

Sets a function to be called at regular intervals, with the given priority. The function is called repeatedly until it returns FALSE, at which point the timeout is automatically destroyed and the function will not be called again. The notify function is called when the timeout is destroyed. The first call to the function will be at the end of the first interval.

Note that timeout functions may be delayed, due to the processing of other event sources. Thus they should not be relied on for precise timing. After each call to the timeout function, the time of the next timeout is recalculated based on the current time and the given interval (it does not try to 'catch up' time lost in delays).

priority : the priority of the timeout source. Typically this will be in the range between G_PRIORITY_DEFAULT and G_PRIORITY_HIGH.
interval : the time between calls to the function, in milliseconds (1/1000ths of a second)
function : function to call
data : data to pass to function
notify : function to call when the timeout is removed, or NULL
Returns : the ID (greater than 0) of the event source.

g_timeout_add_seconds ()

guint               g_timeout_add_seconds               (guint interval,
                                                         GSourceFunc function,
                                                         gpointer data);

Sets a function to be called at regular intervals with the default priority, G_PRIORITY_DEFAULT. The function is called repeatedly until it returns FALSE, at which point the timeout is automatically destroyed and the function will not be called again.

See g_timeout_add_seconds_full() for the differences between g_timeout_add() and g_timeout_add_seconds().

interval : the time between calls to the function, in seconds
function : function to call
data : data to pass to function
Returns : the ID (greater than 0) of the event source.

Since 2.14


g_timeout_add_seconds_full ()

guint               g_timeout_add_seconds_full          (gint priority,
                                                         guint interval,
                                                         GSourceFunc function,
                                                         gpointer data,
                                                         GDestroyNotify notify);

Sets a function to be called at regular intervals, with priority. The function is called repeatedly until it returns FALSE, at which point the timeout is automatically destroyed and the function will not be called again.

Unlike g_timeout_add(), this function operates at whole second granularity. The initial starting point of the timer is determined by the implementation and the implementation is expected to group multiple timers together so that they fire all at the same time. To allow this grouping, the interval to the first timer is rounded and can deviate up to one second from the specified interval. Subsequent timer iterations will generally run at the specified interval.

Note that timeout functions may be delayed, due to the processing of other event sources. Thus they should not be relied on for precise timing. After each call to the timeout function, the time of the next timeout is recalculated based on the current time and the given interval

If you want timing more precise than whole seconds, use g_timeout_add() instead.

The grouping of timers to fire at the same time results in a more power and CPU efficient behavior so if your timer is in multiples of seconds and you don't require the first timer exactly one second from now, the use of g_timeout_add_seconds() is preferred over g_timeout_add().

priority : the priority of the timeout source. Typically this will be in the range between G_PRIORITY_DEFAULT and G_PRIORITY_HIGH.
interval : the time between calls to the function, in seconds
function : function to call
data : data to pass to function
notify : function to call when the timeout is removed, or NULL
Returns : the ID (greater than 0) of the event source.

Since 2.14


g_idle_source_new ()

GSource*            g_idle_source_new                   (void);

Creates a new idle source.

The source will not initially be associated with any GMainContext and must be added to one with g_source_attach() before it will be executed. Note that the default priority for idle sources is G_PRIORITY_DEFAULT_IDLE, as compared to other sources which have a default priority of G_PRIORITY_DEFAULT.

Returns : the newly-created idle source

g_idle_add ()

guint               g_idle_add                          (GSourceFunc function,
                                                         gpointer data);

Adds a function to be called whenever there are no higher priority events pending to the default main loop. The function is given the default idle priority, G_PRIORITY_DEFAULT_IDLE. If the function returns FALSE it is automatically removed from the list of event sources and will not be called again.

function : function to call
data : data to pass to function.
Returns : the ID (greater than 0) of the event source.

g_idle_add_full ()

guint               g_idle_add_full                     (gint priority,
                                                         GSourceFunc function,
                                                         gpointer data,
                                                         GDestroyNotify notify);

Adds a function to be called whenever there are no higher priority events pending. If the function returns FALSE it is automatically removed from the list of event sources and will not be called again.

priority : the priority of the idle source. Typically this will be in the range btweeen G_PRIORITY_DEFAULT_IDLE and G_PRIORITY_HIGH_IDLE.
function : function to call
data : data to pass to function
notify : function to call when the idle is removed, or NULL
Returns : the ID (greater than 0) of the event source.