1998-12-02 15:55:27 +01:00
|
|
|
/* 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
|
2000-07-26 13:02:02 +02:00
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
1998-12-02 15:55:27 +01:00
|
|
|
* 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
|
2000-07-26 13:02:02 +02:00
|
|
|
* Lesser General Public License for more details.
|
1998-12-02 15:55:27 +01:00
|
|
|
*
|
2000-07-26 13:02:02 +02:00
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
1998-12-02 15:55:27 +01:00
|
|
|
* License along with this library; if not, write to the
|
|
|
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
|
|
* Boston, MA 02111-1307, USA.
|
|
|
|
*/
|
|
|
|
|
1999-02-24 07:14:27 +01:00
|
|
|
/*
|
2000-07-26 13:02:02 +02:00
|
|
|
* Modified by the GLib Team and others 1997-2000. See the AUTHORS
|
1999-02-24 07:14:27 +01:00
|
|
|
* file for a list of people on the GLib Team. See the ChangeLog
|
|
|
|
* files for a list of changes. These files are distributed with
|
2010-09-04 05:03:14 +02:00
|
|
|
* GLib at ftp://ftp.gtk.org/pub/gtk/.
|
1999-02-24 07:14:27 +01:00
|
|
|
*/
|
|
|
|
|
2010-09-04 05:03:14 +02:00
|
|
|
/*
|
1998-12-15 06:28:02 +01:00
|
|
|
* MT safe
|
|
|
|
*/
|
|
|
|
|
2010-09-06 14:56:16 +02:00
|
|
|
#ifndef _WIN32
|
2010-09-04 19:15:15 +02:00
|
|
|
/* for pipe2; need to define it first to avoid
|
|
|
|
* other headers pulling in unistd.h
|
|
|
|
*/
|
2010-09-06 14:56:16 +02:00
|
|
|
/* The meaning of_GNU_SOURCE that is intended here is present only on
|
|
|
|
* Linux; avoid the possibility that some misguided header in MinGW
|
|
|
|
* looks at it. Ideally we should define _GNU_SOURCE only on platforms
|
|
|
|
* where we know what it means and that is what we want here
|
|
|
|
* (i.e. Linux with glibc). After all, there might be some other POSIX
|
|
|
|
* platform even where _GNU_SOURCE is used for some unrelated change
|
|
|
|
* in semantics that isn't wanted. Sigh.
|
|
|
|
*/
|
2010-09-04 19:15:15 +02:00
|
|
|
#define _GNU_SOURCE
|
2010-09-06 14:56:16 +02:00
|
|
|
#endif
|
2010-09-04 19:15:15 +02:00
|
|
|
|
Merge in current Win32 version. Almost no Unix code touched.
* README.win32: More text.
* config.h.win32 glibconfig.h.win32: Update to match the
corresponding generated files on Unix.
* makefile.msc: Update with new source files, and gthread
library. Use the compiler flag -MD instead of using -D_DLL and
"/nodefaultlib:libc msvcrt.lib" in the link phase.
* glib.def: Include new functions, drop removed ones.
* glib.h: Add comments about main loop and polling on Win32. (In
general, it's only for the GIMP's use.) Add Win32 IO Channel
functions. Remove the obsoleted old IO Channel stuff (which was
in #if 0 already).
* giowin32.c: New file.
* gmain.c: Include config.h, conditionalize <sys/time.h>
inclusion. Add g_poll implementation for Win32 (only for the
GIMP's needs for now, it's hard or even impossible to be as clean
and generic as on Unix). Implement g_get_current_time on Win32. If
threads aren't supported, don't try to wake up main thread's
loop. On Win32, use a semaphore and not a pipe to wake up the main
loop.
* gmessages.c: On Win32, allocate a console window if the standard
output handle is invalid before writing to stdout, and reopen stdout
to that console window.
* giochannel.c: Conditionalize unistd.h inclusion. Some indentation
cleanup.
* gstrfuncs.c: Include <signal.h>.
* gutils.c: On Win32, also check the HOMEDRIVE and HOMEPATH
environment variables.
* gmodule-dl.c gmodule-dld.c: In
_g_module_build_path, don't add the "lib" prefix and
".so" or ".sl" suffix if already there.
* gmodule-win32.c: Likewise for the ".dll" suffix.
* gthread-posix.c: Conditionalize <sys/time.h> inclusion.
1999-01-17 00:46:42 +01:00
|
|
|
#include "config.h"
|
2010-09-04 05:03:14 +02:00
|
|
|
#include "glibconfig.h"
|
Merge in current Win32 version. Almost no Unix code touched.
* README.win32: More text.
* config.h.win32 glibconfig.h.win32: Update to match the
corresponding generated files on Unix.
* makefile.msc: Update with new source files, and gthread
library. Use the compiler flag -MD instead of using -D_DLL and
"/nodefaultlib:libc msvcrt.lib" in the link phase.
* glib.def: Include new functions, drop removed ones.
* glib.h: Add comments about main loop and polling on Win32. (In
general, it's only for the GIMP's use.) Add Win32 IO Channel
functions. Remove the obsoleted old IO Channel stuff (which was
in #if 0 already).
* giowin32.c: New file.
* gmain.c: Include config.h, conditionalize <sys/time.h>
inclusion. Add g_poll implementation for Win32 (only for the
GIMP's needs for now, it's hard or even impossible to be as clean
and generic as on Unix). Implement g_get_current_time on Win32. If
threads aren't supported, don't try to wake up main thread's
loop. On Win32, use a semaphore and not a pipe to wake up the main
loop.
* gmessages.c: On Win32, allocate a console window if the standard
output handle is invalid before writing to stdout, and reopen stdout
to that console window.
* giochannel.c: Conditionalize unistd.h inclusion. Some indentation
cleanup.
* gstrfuncs.c: Include <signal.h>.
* gutils.c: On Win32, also check the HOMEDRIVE and HOMEPATH
environment variables.
* gmodule-dl.c gmodule-dld.c: In
_g_module_build_path, don't add the "lib" prefix and
".so" or ".sl" suffix if already there.
* gmodule-win32.c: Likewise for the ".dll" suffix.
* gthread-posix.c: Conditionalize <sys/time.h> inclusion.
1999-01-17 00:46:42 +01:00
|
|
|
|
2008-09-23 18:32:30 +02:00
|
|
|
/* Uncomment the next line (and the corresponding line in gpoll.c) to
|
|
|
|
* enable debugging printouts if the environment variable
|
|
|
|
* G_MAIN_POLL_DEBUG is set to some value.
|
2008-05-17 04:25:59 +02:00
|
|
|
*/
|
1999-07-24 20:50:58 +02:00
|
|
|
/* #define G_MAIN_POLL_DEBUG */
|
|
|
|
|
2008-05-17 04:25:59 +02:00
|
|
|
#ifdef _WIN32
|
|
|
|
/* Always enable debugging printout on Windows, as it is more often
|
|
|
|
* needed there...
|
|
|
|
*/
|
|
|
|
#define G_MAIN_POLL_DEBUG
|
|
|
|
#endif
|
|
|
|
|
2004-02-18 23:55:15 +01:00
|
|
|
#include <signal.h>
|
1998-12-19 04:44:30 +01:00
|
|
|
#include <sys/types.h>
|
Merge in current Win32 version. Almost no Unix code touched.
* README.win32: More text.
* config.h.win32 glibconfig.h.win32: Update to match the
corresponding generated files on Unix.
* makefile.msc: Update with new source files, and gthread
library. Use the compiler flag -MD instead of using -D_DLL and
"/nodefaultlib:libc msvcrt.lib" in the link phase.
* glib.def: Include new functions, drop removed ones.
* glib.h: Add comments about main loop and polling on Win32. (In
general, it's only for the GIMP's use.) Add Win32 IO Channel
functions. Remove the obsoleted old IO Channel stuff (which was
in #if 0 already).
* giowin32.c: New file.
* gmain.c: Include config.h, conditionalize <sys/time.h>
inclusion. Add g_poll implementation for Win32 (only for the
GIMP's needs for now, it's hard or even impossible to be as clean
and generic as on Unix). Implement g_get_current_time on Win32. If
threads aren't supported, don't try to wake up main thread's
loop. On Win32, use a semaphore and not a pipe to wake up the main
loop.
* gmessages.c: On Win32, allocate a console window if the standard
output handle is invalid before writing to stdout, and reopen stdout
to that console window.
* giochannel.c: Conditionalize unistd.h inclusion. Some indentation
cleanup.
* gstrfuncs.c: Include <signal.h>.
* gutils.c: On Win32, also check the HOMEDRIVE and HOMEPATH
environment variables.
* gmodule-dl.c gmodule-dld.c: In
_g_module_build_path, don't add the "lib" prefix and
".so" or ".sl" suffix if already there.
* gmodule-win32.c: Likewise for the ".dll" suffix.
* gthread-posix.c: Conditionalize <sys/time.h> inclusion.
1999-01-17 00:46:42 +01:00
|
|
|
#include <time.h>
|
2006-09-10 07:44:46 +02:00
|
|
|
#include <stdlib.h>
|
Merge in current Win32 version. Almost no Unix code touched.
* README.win32: More text.
* config.h.win32 glibconfig.h.win32: Update to match the
corresponding generated files on Unix.
* makefile.msc: Update with new source files, and gthread
library. Use the compiler flag -MD instead of using -D_DLL and
"/nodefaultlib:libc msvcrt.lib" in the link phase.
* glib.def: Include new functions, drop removed ones.
* glib.h: Add comments about main loop and polling on Win32. (In
general, it's only for the GIMP's use.) Add Win32 IO Channel
functions. Remove the obsoleted old IO Channel stuff (which was
in #if 0 already).
* giowin32.c: New file.
* gmain.c: Include config.h, conditionalize <sys/time.h>
inclusion. Add g_poll implementation for Win32 (only for the
GIMP's needs for now, it's hard or even impossible to be as clean
and generic as on Unix). Implement g_get_current_time on Win32. If
threads aren't supported, don't try to wake up main thread's
loop. On Win32, use a semaphore and not a pipe to wake up the main
loop.
* gmessages.c: On Win32, allocate a console window if the standard
output handle is invalid before writing to stdout, and reopen stdout
to that console window.
* giochannel.c: Conditionalize unistd.h inclusion. Some indentation
cleanup.
* gstrfuncs.c: Include <signal.h>.
* gutils.c: On Win32, also check the HOMEDRIVE and HOMEPATH
environment variables.
* gmodule-dl.c gmodule-dld.c: In
_g_module_build_path, don't add the "lib" prefix and
".so" or ".sl" suffix if already there.
* gmodule-win32.c: Likewise for the ".dll" suffix.
* gthread-posix.c: Conditionalize <sys/time.h> inclusion.
1999-01-17 00:46:42 +01:00
|
|
|
#ifdef HAVE_SYS_TIME_H
|
1998-12-02 15:55:27 +01:00
|
|
|
#include <sys/time.h>
|
1999-01-17 05:49:43 +01:00
|
|
|
#endif /* HAVE_SYS_TIME_H */
|
Merge in current Win32 version. Almost no Unix code touched.
* README.win32: More text.
* config.h.win32 glibconfig.h.win32: Update to match the
corresponding generated files on Unix.
* makefile.msc: Update with new source files, and gthread
library. Use the compiler flag -MD instead of using -D_DLL and
"/nodefaultlib:libc msvcrt.lib" in the link phase.
* glib.def: Include new functions, drop removed ones.
* glib.h: Add comments about main loop and polling on Win32. (In
general, it's only for the GIMP's use.) Add Win32 IO Channel
functions. Remove the obsoleted old IO Channel stuff (which was
in #if 0 already).
* giowin32.c: New file.
* gmain.c: Include config.h, conditionalize <sys/time.h>
inclusion. Add g_poll implementation for Win32 (only for the
GIMP's needs for now, it's hard or even impossible to be as clean
and generic as on Unix). Implement g_get_current_time on Win32. If
threads aren't supported, don't try to wake up main thread's
loop. On Win32, use a semaphore and not a pipe to wake up the main
loop.
* gmessages.c: On Win32, allocate a console window if the standard
output handle is invalid before writing to stdout, and reopen stdout
to that console window.
* giochannel.c: Conditionalize unistd.h inclusion. Some indentation
cleanup.
* gstrfuncs.c: Include <signal.h>.
* gutils.c: On Win32, also check the HOMEDRIVE and HOMEPATH
environment variables.
* gmodule-dl.c gmodule-dld.c: In
_g_module_build_path, don't add the "lib" prefix and
".so" or ".sl" suffix if already there.
* gmodule-win32.c: Likewise for the ".dll" suffix.
* gthread-posix.c: Conditionalize <sys/time.h> inclusion.
1999-01-17 00:46:42 +01:00
|
|
|
#ifdef HAVE_UNISTD_H
|
1998-12-02 15:55:27 +01:00
|
|
|
#include <unistd.h>
|
1999-01-17 05:49:43 +01:00
|
|
|
#endif /* HAVE_UNISTD_H */
|
1998-12-15 06:28:02 +01:00
|
|
|
#include <errno.h>
|
Merge in current Win32 version. Almost no Unix code touched.
* README.win32: More text.
* config.h.win32 glibconfig.h.win32: Update to match the
corresponding generated files on Unix.
* makefile.msc: Update with new source files, and gthread
library. Use the compiler flag -MD instead of using -D_DLL and
"/nodefaultlib:libc msvcrt.lib" in the link phase.
* glib.def: Include new functions, drop removed ones.
* glib.h: Add comments about main loop and polling on Win32. (In
general, it's only for the GIMP's use.) Add Win32 IO Channel
functions. Remove the obsoleted old IO Channel stuff (which was
in #if 0 already).
* giowin32.c: New file.
* gmain.c: Include config.h, conditionalize <sys/time.h>
inclusion. Add g_poll implementation for Win32 (only for the
GIMP's needs for now, it's hard or even impossible to be as clean
and generic as on Unix). Implement g_get_current_time on Win32. If
threads aren't supported, don't try to wake up main thread's
loop. On Win32, use a semaphore and not a pipe to wake up the main
loop.
* gmessages.c: On Win32, allocate a console window if the standard
output handle is invalid before writing to stdout, and reopen stdout
to that console window.
* giochannel.c: Conditionalize unistd.h inclusion. Some indentation
cleanup.
* gstrfuncs.c: Include <signal.h>.
* gutils.c: On Win32, also check the HOMEDRIVE and HOMEPATH
environment variables.
* gmodule-dl.c gmodule-dld.c: In
_g_module_build_path, don't add the "lib" prefix and
".so" or ".sl" suffix if already there.
* gmodule-win32.c: Likewise for the ".dll" suffix.
* gthread-posix.c: Conditionalize <sys/time.h> inclusion.
1999-01-17 00:46:42 +01:00
|
|
|
|
1999-10-04 04:32:50 +02:00
|
|
|
#ifdef G_OS_WIN32
|
Merge in current Win32 version. Almost no Unix code touched.
* README.win32: More text.
* config.h.win32 glibconfig.h.win32: Update to match the
corresponding generated files on Unix.
* makefile.msc: Update with new source files, and gthread
library. Use the compiler flag -MD instead of using -D_DLL and
"/nodefaultlib:libc msvcrt.lib" in the link phase.
* glib.def: Include new functions, drop removed ones.
* glib.h: Add comments about main loop and polling on Win32. (In
general, it's only for the GIMP's use.) Add Win32 IO Channel
functions. Remove the obsoleted old IO Channel stuff (which was
in #if 0 already).
* giowin32.c: New file.
* gmain.c: Include config.h, conditionalize <sys/time.h>
inclusion. Add g_poll implementation for Win32 (only for the
GIMP's needs for now, it's hard or even impossible to be as clean
and generic as on Unix). Implement g_get_current_time on Win32. If
threads aren't supported, don't try to wake up main thread's
loop. On Win32, use a semaphore and not a pipe to wake up the main
loop.
* gmessages.c: On Win32, allocate a console window if the standard
output handle is invalid before writing to stdout, and reopen stdout
to that console window.
* giochannel.c: Conditionalize unistd.h inclusion. Some indentation
cleanup.
* gstrfuncs.c: Include <signal.h>.
* gutils.c: On Win32, also check the HOMEDRIVE and HOMEPATH
environment variables.
* gmodule-dl.c gmodule-dld.c: In
_g_module_build_path, don't add the "lib" prefix and
".so" or ".sl" suffix if already there.
* gmodule-win32.c: Likewise for the ".dll" suffix.
* gthread-posix.c: Conditionalize <sys/time.h> inclusion.
1999-01-17 00:46:42 +01:00
|
|
|
#define STRICT
|
|
|
|
#include <windows.h>
|
1999-10-04 04:32:50 +02:00
|
|
|
#endif /* G_OS_WIN32 */
|
Merge in current Win32 version. Almost no Unix code touched.
* README.win32: More text.
* config.h.win32 glibconfig.h.win32: Update to match the
corresponding generated files on Unix.
* makefile.msc: Update with new source files, and gthread
library. Use the compiler flag -MD instead of using -D_DLL and
"/nodefaultlib:libc msvcrt.lib" in the link phase.
* glib.def: Include new functions, drop removed ones.
* glib.h: Add comments about main loop and polling on Win32. (In
general, it's only for the GIMP's use.) Add Win32 IO Channel
functions. Remove the obsoleted old IO Channel stuff (which was
in #if 0 already).
* giowin32.c: New file.
* gmain.c: Include config.h, conditionalize <sys/time.h>
inclusion. Add g_poll implementation for Win32 (only for the
GIMP's needs for now, it's hard or even impossible to be as clean
and generic as on Unix). Implement g_get_current_time on Win32. If
threads aren't supported, don't try to wake up main thread's
loop. On Win32, use a semaphore and not a pipe to wake up the main
loop.
* gmessages.c: On Win32, allocate a console window if the standard
output handle is invalid before writing to stdout, and reopen stdout
to that console window.
* giochannel.c: Conditionalize unistd.h inclusion. Some indentation
cleanup.
* gstrfuncs.c: Include <signal.h>.
* gutils.c: On Win32, also check the HOMEDRIVE and HOMEPATH
environment variables.
* gmodule-dl.c gmodule-dld.c: In
_g_module_build_path, don't add the "lib" prefix and
".so" or ".sl" suffix if already there.
* gmodule-win32.c: Likewise for the ".dll" suffix.
* gthread-posix.c: Conditionalize <sys/time.h> inclusion.
1999-01-17 00:46:42 +01:00
|
|
|
|
1999-10-04 04:32:50 +02:00
|
|
|
#ifdef G_OS_BEOS
|
2005-06-30 21:43:48 +02:00
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <sys/wait.h>
|
1999-10-04 04:32:50 +02:00
|
|
|
#endif /* G_OS_BEOS */
|
1999-05-08 09:40:44 +02:00
|
|
|
|
2004-02-14 01:23:36 +01:00
|
|
|
#ifdef G_OS_UNIX
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <sys/wait.h>
|
|
|
|
#endif
|
2005-03-14 05:26:57 +01:00
|
|
|
|
2010-09-04 05:03:14 +02:00
|
|
|
#include "gmain.h"
|
|
|
|
|
|
|
|
#include "garray.h"
|
|
|
|
#include "giochannel.h"
|
|
|
|
#include "ghash.h"
|
|
|
|
#include "ghook.h"
|
|
|
|
#include "gqueue.h"
|
|
|
|
#include "gstrfuncs.h"
|
|
|
|
#include "gtestutils.h"
|
|
|
|
#include "gthreadprivate.h"
|
2005-03-14 05:26:57 +01:00
|
|
|
|
2010-09-06 14:56:16 +02:00
|
|
|
#ifdef G_OS_WIN32
|
|
|
|
#include "gwin32.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef G_MAIN_POLL_DEBUG
|
|
|
|
#include "gtimer.h"
|
|
|
|
#endif
|
|
|
|
|
2010-07-11 05:53:36 +02:00
|
|
|
/**
|
|
|
|
* SECTION:main
|
|
|
|
* @title: The Main Event Loop
|
|
|
|
* @short_description: manages all available sources of events
|
|
|
|
*
|
|
|
|
* 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().
|
|
|
|
*
|
|
|
|
* <refsect2><title>Creating new source types</title>
|
|
|
|
* <para>One of the unusual features of the #GMainLoop 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 <firstterm>deriving</firstterm> 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 type.</para>
|
|
|
|
* <para>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().</para>
|
|
|
|
* </refsect2>
|
|
|
|
* <refsect2><title>Customizing the main loop iteration</title>
|
|
|
|
* <para>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().</para>
|
|
|
|
* <para>The operation of these functions can best be seen in terms
|
|
|
|
* of a state diagram, as shown in <xref linkend="mainloop-states"/>.</para>
|
|
|
|
* <figure id="mainloop-states"><title>States of a Main Context</title>
|
|
|
|
* <graphic fileref="mainloop-states.gif" format="GIF"></graphic>
|
|
|
|
* </figure>
|
|
|
|
* </refsect2>
|
|
|
|
*/
|
|
|
|
|
1998-12-02 15:55:27 +01:00
|
|
|
/* Types */
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
typedef struct _GTimeoutSource GTimeoutSource;
|
2004-02-14 01:23:36 +01:00
|
|
|
typedef struct _GChildWatchSource GChildWatchSource;
|
1998-12-02 15:55:27 +01:00
|
|
|
typedef struct _GPollRec GPollRec;
|
2000-12-05 21:45:33 +01:00
|
|
|
typedef struct _GSourceCallback GSourceCallback;
|
1998-12-02 15:55:27 +01:00
|
|
|
|
1998-12-17 05:06:27 +01:00
|
|
|
typedef enum
|
|
|
|
{
|
1998-12-02 15:55:27 +01:00
|
|
|
G_SOURCE_READY = 1 << G_HOOK_FLAG_USER_SHIFT,
|
|
|
|
G_SOURCE_CAN_RECURSE = 1 << (G_HOOK_FLAG_USER_SHIFT + 1)
|
|
|
|
} GSourceFlags;
|
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
#ifdef G_THREADS_ENABLED
|
|
|
|
typedef struct _GMainWaiter GMainWaiter;
|
|
|
|
|
|
|
|
struct _GMainWaiter
|
|
|
|
{
|
|
|
|
GCond *cond;
|
|
|
|
GMutex *mutex;
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
2006-06-02 04:36:30 +02:00
|
|
|
typedef struct _GMainDispatch GMainDispatch;
|
|
|
|
|
|
|
|
struct _GMainDispatch
|
|
|
|
{
|
|
|
|
gint depth;
|
2008-05-02 13:22:10 +02:00
|
|
|
GSList *dispatching_sources; /* stack of current sources */
|
2006-06-02 04:36:30 +02:00
|
|
|
};
|
|
|
|
|
2008-05-17 04:25:59 +02:00
|
|
|
#ifdef G_MAIN_POLL_DEBUG
|
2008-09-25 21:59:49 +02:00
|
|
|
gboolean _g_main_poll_debug = FALSE;
|
2008-05-17 04:25:59 +02:00
|
|
|
#endif
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
struct _GMainContext
|
1998-12-17 05:06:27 +01:00
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
#ifdef G_THREADS_ENABLED
|
|
|
|
/* The following lock is used for both the list of sources
|
|
|
|
* and the list of poll records
|
|
|
|
*/
|
2001-06-30 21:56:47 +02:00
|
|
|
GStaticMutex mutex;
|
|
|
|
GCond *cond;
|
|
|
|
GThread *owner;
|
|
|
|
guint owner_count;
|
|
|
|
GSList *waiters;
|
2000-12-05 21:45:33 +01:00
|
|
|
#endif
|
|
|
|
|
2004-03-04 11:12:55 +01:00
|
|
|
gint ref_count;
|
2001-06-30 21:56:47 +02:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
GPtrArray *pending_dispatches;
|
|
|
|
gint timeout; /* Timeout for current iteration */
|
|
|
|
|
|
|
|
guint next_id;
|
|
|
|
GSource *source_list;
|
|
|
|
gint in_check_or_prepare;
|
|
|
|
|
|
|
|
GPollRec *poll_records;
|
|
|
|
guint n_poll_records;
|
|
|
|
GPollFD *cached_poll_array;
|
Changes for 64-bit cleanliness, loosely based on patch from Mark Murnane.
Wed Jun 20 12:00:54 2001 Owen Taylor <otaylor@redhat.com>
Changes for 64-bit cleanliness, loosely based on patch
from Mark Murnane.
* gconvert.c (g_convert/g_convert_with_fallback): Remove
workarounds for since-fixed GNU libc bugs. Minor
doc fix.
* gconvert.[ch]: Change gint to gsize/gssize as
appropriate.
* gconvert.c (g_locale/filename_to/from_utf8): Fix incorrect
computation of bytes_read / bytes_written.
* gfileutils.[ch] (g_file_get_contents): Make length
out parameter 'gsize *len'.
* ghook.c (g_hook_compare_ids): Don't compare a
and b as 'a - b'.
* gmacros.h (GSIZE_TO_POINTER): Add GPOINTER_TO_SIZE,
GSIZE_TO_POINTER.
* gmain.c (g_timeout_prepare): Rewrite to avoid
overflows. (Fixes bug when system clock skews
backwards more than 24 days.)
* gmarkup.[ch]: Make lengths passed to callbacks
gsize, length for g_markup_parse-context_parse(),
g_markup_escape_text() gssize.
* gmessages.[ch] (g_printf_string_upper_bound): Change
return value to gsize.
* gmessages.c (printf_string_upper_bound): Remove
a ridiculous use of 'inline' on a 300 line function.
* gstring.[ch]: Represent size of string as a gsize,
not gint. Make parameters to functions take gsize,
or gssize where -1 is allowed.
* gstring.c (g_string_erase): Make
g_string_erase (string, pos, -1) a synonym for
g_string_truncate for consistency with other G*
APIs.
* gstrfuncs.[ch]: Make all functions taking a string
length, take a gsize, or gssize if -1 is allowed.
(g_strstr_len, g_strrstr_len). Also fix some boundary
conditions in g_str[r]str[_len].
* gutf8.c tests/unicode-encoding.c: Make parameters that
are byte lengths gsize, gssize as appropriate. Make
character offsets, other counts, glong.
* gasyncqueue.c gcompletion.c
timeloop.c timeloop-basic.c gutils.c gspawn.c.
Small 64 bit cleanliness fixups.
* glist.c (g_list_sort2, g_list_sort_real): Fix functions
that should have been static.
* gdate.c (g_date_fill_parse_tokens): Fix extra
declaration that was shadowing another.
* tests/module-test.c: Include string.h
Mon Jun 18 15:43:29 2001 Owen Taylor <otaylor@redhat.com>
* gutf8.c (g_get_charset): Make argument
G_CONST_RETURN char **.
2001-06-23 15:55:09 +02:00
|
|
|
guint cached_poll_array_size;
|
2000-12-05 21:45:33 +01:00
|
|
|
|
|
|
|
#ifdef G_THREADS_ENABLED
|
|
|
|
#ifndef G_OS_WIN32
|
|
|
|
/* this pipe is used to wake up the main loop when a source is added.
|
|
|
|
*/
|
|
|
|
gint wake_up_pipe[2];
|
|
|
|
#else /* G_OS_WIN32 */
|
|
|
|
HANDLE wake_up_semaphore;
|
|
|
|
#endif /* G_OS_WIN32 */
|
|
|
|
|
|
|
|
GPollFD wake_up_rec;
|
|
|
|
gboolean poll_waiting;
|
|
|
|
|
|
|
|
/* Flag indicating whether the set of fd's changed during a poll */
|
|
|
|
gboolean poll_changed;
|
|
|
|
#endif /* G_THREADS_ENABLED */
|
|
|
|
|
|
|
|
GPollFunc poll_func;
|
|
|
|
|
2010-11-01 20:46:35 +01:00
|
|
|
gint64 time;
|
2010-10-22 18:40:08 +02:00
|
|
|
gboolean time_is_fresh;
|
2010-11-03 03:03:08 +01:00
|
|
|
gint64 real_time;
|
|
|
|
gboolean real_time_is_fresh;
|
2000-12-05 21:45:33 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
struct _GSourceCallback
|
|
|
|
{
|
|
|
|
guint ref_count;
|
|
|
|
GSourceFunc func;
|
|
|
|
gpointer data;
|
|
|
|
GDestroyNotify notify;
|
1998-12-02 15:55:27 +01:00
|
|
|
};
|
|
|
|
|
1998-12-17 05:06:27 +01:00
|
|
|
struct _GMainLoop
|
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
GMainContext *context;
|
1998-12-18 03:23:33 +01:00
|
|
|
gboolean is_running;
|
2004-03-04 11:12:55 +01:00
|
|
|
gint ref_count;
|
1998-12-02 15:55:27 +01:00
|
|
|
};
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
struct _GTimeoutSource
|
1998-12-17 05:06:27 +01:00
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
GSource source;
|
2010-11-01 20:46:35 +01:00
|
|
|
gint64 expiration;
|
2002-06-13 22:14:43 +02:00
|
|
|
guint interval;
|
2010-11-01 18:31:37 +01:00
|
|
|
gboolean seconds;
|
1998-12-02 15:55:27 +01:00
|
|
|
};
|
|
|
|
|
2004-02-14 01:23:36 +01:00
|
|
|
struct _GChildWatchSource
|
|
|
|
{
|
|
|
|
GSource source;
|
|
|
|
GPid pid;
|
|
|
|
gint child_status;
|
2004-02-28 14:03:55 +01:00
|
|
|
#ifdef G_OS_WIN32
|
|
|
|
GPollFD poll;
|
|
|
|
#else /* G_OS_WIN32 */
|
2004-03-04 11:12:55 +01:00
|
|
|
gint count;
|
|
|
|
gboolean child_exited;
|
2004-02-28 14:03:55 +01:00
|
|
|
#endif /* G_OS_WIN32 */
|
2004-02-14 01:23:36 +01:00
|
|
|
};
|
|
|
|
|
1998-12-17 05:06:27 +01:00
|
|
|
struct _GPollRec
|
|
|
|
{
|
1998-12-02 15:55:27 +01:00
|
|
|
GPollFD *fd;
|
2005-12-05 16:01:27 +01:00
|
|
|
GPollRec *next;
|
2005-11-01 19:10:31 +01:00
|
|
|
gint priority;
|
1998-12-02 15:55:27 +01:00
|
|
|
};
|
|
|
|
|
2010-11-06 15:11:15 +01:00
|
|
|
struct _GSourcePrivate
|
|
|
|
{
|
|
|
|
GSList *child_sources;
|
|
|
|
GSource *parent_source;
|
|
|
|
};
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
#ifdef G_THREADS_ENABLED
|
2001-06-30 21:56:47 +02:00
|
|
|
#define LOCK_CONTEXT(context) g_static_mutex_lock (&context->mutex)
|
|
|
|
#define UNLOCK_CONTEXT(context) g_static_mutex_unlock (&context->mutex)
|
|
|
|
#define G_THREAD_SELF g_thread_self ()
|
2000-12-05 21:45:33 +01:00
|
|
|
#else
|
|
|
|
#define LOCK_CONTEXT(context) (void)0
|
|
|
|
#define UNLOCK_CONTEXT(context) (void)0
|
2001-06-30 21:56:47 +02:00
|
|
|
#define G_THREAD_SELF NULL
|
2000-12-05 21:45:33 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#define SOURCE_DESTROYED(source) (((source)->flags & G_HOOK_FLAG_ACTIVE) == 0)
|
2004-03-19 21:25:03 +01:00
|
|
|
#define SOURCE_BLOCKED(source) (((source)->flags & G_HOOK_FLAG_IN_CALL) != 0 && \
|
|
|
|
((source)->flags & G_SOURCE_CAN_RECURSE) == 0)
|
2000-12-05 21:45:33 +01:00
|
|
|
|
|
|
|
#define SOURCE_UNREF(source, context) \
|
|
|
|
G_STMT_START { \
|
|
|
|
if ((source)->ref_count > 1) \
|
|
|
|
(source)->ref_count--; \
|
|
|
|
else \
|
|
|
|
g_source_unref_internal ((source), (context), TRUE); \
|
|
|
|
} G_STMT_END
|
|
|
|
|
|
|
|
|
1998-12-02 15:55:27 +01:00
|
|
|
/* Forward declarations */
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
static void g_source_unref_internal (GSource *source,
|
|
|
|
GMainContext *context,
|
|
|
|
gboolean have_lock);
|
|
|
|
static void g_source_destroy_internal (GSource *source,
|
|
|
|
GMainContext *context,
|
|
|
|
gboolean have_lock);
|
2010-11-06 15:11:15 +01:00
|
|
|
static void g_source_set_priority_unlocked (GSource *source,
|
|
|
|
GMainContext *context,
|
|
|
|
gint priority);
|
2000-12-05 21:45:33 +01:00
|
|
|
static void g_main_context_poll (GMainContext *context,
|
|
|
|
gint timeout,
|
|
|
|
gint priority,
|
|
|
|
GPollFD *fds,
|
|
|
|
gint n_fds);
|
|
|
|
static void g_main_context_add_poll_unlocked (GMainContext *context,
|
|
|
|
gint priority,
|
|
|
|
GPollFD *fd);
|
|
|
|
static void g_main_context_remove_poll_unlocked (GMainContext *context,
|
|
|
|
GPollFD *fd);
|
2001-06-30 21:56:47 +02:00
|
|
|
static void g_main_context_wakeup_unlocked (GMainContext *context);
|
2000-12-05 21:45:33 +01:00
|
|
|
|
|
|
|
static gboolean g_timeout_prepare (GSource *source,
|
|
|
|
gint *timeout);
|
|
|
|
static gboolean g_timeout_check (GSource *source);
|
|
|
|
static gboolean g_timeout_dispatch (GSource *source,
|
|
|
|
GSourceFunc callback,
|
|
|
|
gpointer user_data);
|
2004-02-14 01:23:36 +01:00
|
|
|
static gboolean g_child_watch_prepare (GSource *source,
|
|
|
|
gint *timeout);
|
|
|
|
static gboolean g_child_watch_check (GSource *source);
|
|
|
|
static gboolean g_child_watch_dispatch (GSource *source,
|
|
|
|
GSourceFunc callback,
|
|
|
|
gpointer user_data);
|
2000-12-05 21:45:33 +01:00
|
|
|
static gboolean g_idle_prepare (GSource *source,
|
|
|
|
gint *timeout);
|
|
|
|
static gboolean g_idle_check (GSource *source);
|
|
|
|
static gboolean g_idle_dispatch (GSource *source,
|
|
|
|
GSourceFunc callback,
|
|
|
|
gpointer user_data);
|
|
|
|
|
1999-02-10 10:40:46 +01:00
|
|
|
G_LOCK_DEFINE_STATIC (main_loop);
|
2000-12-05 21:45:33 +01:00
|
|
|
static GMainContext *default_main_context;
|
2002-08-06 16:50:52 +02:00
|
|
|
static GSList *main_contexts_without_pipe = NULL;
|
1998-12-15 06:28:02 +01:00
|
|
|
|
2004-02-28 14:03:55 +01:00
|
|
|
#ifndef G_OS_WIN32
|
2004-02-14 01:23:36 +01:00
|
|
|
/* Child status monitoring code */
|
|
|
|
enum {
|
|
|
|
CHILD_WATCH_UNINITIALIZED,
|
|
|
|
CHILD_WATCH_INITIALIZED_SINGLE,
|
|
|
|
CHILD_WATCH_INITIALIZED_THREADED
|
|
|
|
};
|
|
|
|
static gint child_watch_init_state = CHILD_WATCH_UNINITIALIZED;
|
2004-11-08 16:34:26 +01:00
|
|
|
static gint child_watch_count = 1;
|
2004-02-14 01:23:36 +01:00
|
|
|
static gint child_watch_wake_up_pipe[2] = {0, 0};
|
2004-02-28 14:03:55 +01:00
|
|
|
#endif /* !G_OS_WIN32 */
|
2004-02-14 01:23:36 +01:00
|
|
|
G_LOCK_DEFINE_STATIC (main_context_list);
|
|
|
|
static GSList *main_context_list = NULL;
|
|
|
|
|
2001-09-04 00:12:51 +02:00
|
|
|
GSourceFuncs g_timeout_funcs =
|
1999-01-17 05:49:43 +01:00
|
|
|
{
|
1998-12-02 15:55:27 +01:00
|
|
|
g_timeout_prepare,
|
|
|
|
g_timeout_check,
|
|
|
|
g_timeout_dispatch,
|
2000-12-05 21:45:33 +01:00
|
|
|
NULL
|
1998-12-02 15:55:27 +01:00
|
|
|
};
|
|
|
|
|
2004-02-14 01:23:36 +01:00
|
|
|
GSourceFuncs g_child_watch_funcs =
|
|
|
|
{
|
|
|
|
g_child_watch_prepare,
|
|
|
|
g_child_watch_check,
|
|
|
|
g_child_watch_dispatch,
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
2001-09-04 00:12:51 +02:00
|
|
|
GSourceFuncs g_idle_funcs =
|
1999-01-17 05:49:43 +01:00
|
|
|
{
|
1998-12-02 15:55:27 +01:00
|
|
|
g_idle_prepare,
|
|
|
|
g_idle_check,
|
|
|
|
g_idle_dispatch,
|
2000-12-05 21:45:33 +01:00
|
|
|
NULL
|
1998-12-02 15:55:27 +01:00
|
|
|
};
|
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
/**
|
|
|
|
* g_main_context_ref:
|
2001-09-10 17:50:26 +02:00
|
|
|
* @context: a #GMainContext
|
2001-06-30 21:56:47 +02:00
|
|
|
*
|
|
|
|
* Increases the reference count on a #GMainContext object by one.
|
2004-11-08 19:26:56 +01:00
|
|
|
*
|
|
|
|
* Returns: the @context that was passed in (since 2.6)
|
2001-06-30 21:56:47 +02:00
|
|
|
**/
|
2004-11-08 19:26:56 +01:00
|
|
|
GMainContext *
|
2001-06-30 21:56:47 +02:00
|
|
|
g_main_context_ref (GMainContext *context)
|
1999-01-07 21:12:19 +01:00
|
|
|
{
|
2004-11-08 19:49:35 +01:00
|
|
|
g_return_val_if_fail (context != NULL, NULL);
|
|
|
|
g_return_val_if_fail (g_atomic_int_get (&context->ref_count) > 0, NULL);
|
2001-06-30 21:56:47 +02:00
|
|
|
|
2004-03-04 11:12:55 +01:00
|
|
|
g_atomic_int_inc (&context->ref_count);
|
2004-11-08 19:49:35 +01:00
|
|
|
|
|
|
|
return context;
|
2001-06-30 21:56:47 +02:00
|
|
|
}
|
|
|
|
|
2005-11-01 19:10:31 +01:00
|
|
|
static inline void
|
2003-08-25 18:20:41 +02:00
|
|
|
poll_rec_list_free (GMainContext *context,
|
|
|
|
GPollRec *list)
|
|
|
|
{
|
2005-12-05 16:01:27 +01:00
|
|
|
g_slice_free_chain (GPollRec, list, next);
|
2003-08-25 18:20:41 +02:00
|
|
|
}
|
|
|
|
|
2004-03-04 11:12:55 +01:00
|
|
|
/**
|
|
|
|
* g_main_context_unref:
|
|
|
|
* @context: a #GMainContext
|
|
|
|
*
|
|
|
|
* Decreases the reference count on a #GMainContext object by one. If
|
|
|
|
* the result is zero, free the context and free all associated memory.
|
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_main_context_unref (GMainContext *context)
|
2001-06-30 21:56:47 +02:00
|
|
|
{
|
|
|
|
GSource *source;
|
2004-03-04 11:12:55 +01:00
|
|
|
g_return_if_fail (context != NULL);
|
|
|
|
g_return_if_fail (g_atomic_int_get (&context->ref_count) > 0);
|
2001-06-30 21:56:47 +02:00
|
|
|
|
2004-03-04 11:12:55 +01:00
|
|
|
if (!g_atomic_int_dec_and_test (&context->ref_count))
|
|
|
|
return;
|
2001-06-30 21:56:47 +02:00
|
|
|
|
2004-02-14 01:23:36 +01:00
|
|
|
G_LOCK (main_context_list);
|
|
|
|
main_context_list = g_slist_remove (main_context_list, context);
|
|
|
|
G_UNLOCK (main_context_list);
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
source = context->source_list;
|
|
|
|
while (source)
|
|
|
|
{
|
|
|
|
GSource *next = source->next;
|
2006-03-20 19:43:32 +01:00
|
|
|
g_source_destroy_internal (source, context, FALSE);
|
2000-12-05 21:45:33 +01:00
|
|
|
source = next;
|
|
|
|
}
|
1999-01-17 05:49:43 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
#ifdef G_THREADS_ENABLED
|
2001-06-30 21:56:47 +02:00
|
|
|
g_static_mutex_free (&context->mutex);
|
2000-12-05 21:45:33 +01:00
|
|
|
#endif
|
1999-01-17 05:49:43 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
g_ptr_array_free (context->pending_dispatches, TRUE);
|
|
|
|
g_free (context->cached_poll_array);
|
2003-08-25 18:20:41 +02:00
|
|
|
|
|
|
|
poll_rec_list_free (context, context->poll_records);
|
|
|
|
|
2000-12-13 05:23:45 +01:00
|
|
|
#ifdef G_THREADS_ENABLED
|
2000-12-05 21:45:33 +01:00
|
|
|
if (g_thread_supported())
|
|
|
|
{
|
|
|
|
#ifndef G_OS_WIN32
|
|
|
|
close (context->wake_up_pipe[0]);
|
|
|
|
close (context->wake_up_pipe[1]);
|
|
|
|
#else
|
|
|
|
CloseHandle (context->wake_up_semaphore);
|
|
|
|
#endif
|
2002-08-06 16:50:52 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
main_contexts_without_pipe = g_slist_remove (main_contexts_without_pipe,
|
|
|
|
context);
|
2007-11-08 04:53:41 +01:00
|
|
|
|
|
|
|
if (context->cond != NULL)
|
|
|
|
g_cond_free (context->cond);
|
2000-12-13 05:23:45 +01:00
|
|
|
#endif
|
2000-12-05 21:45:33 +01:00
|
|
|
|
|
|
|
g_free (context);
|
1999-01-07 21:12:19 +01:00
|
|
|
}
|
|
|
|
|
2002-08-06 16:50:52 +02:00
|
|
|
#ifdef G_THREADS_ENABLED
|
|
|
|
static void
|
|
|
|
g_main_context_init_pipe (GMainContext *context)
|
|
|
|
{
|
2004-02-26 00:48:22 +01:00
|
|
|
# ifndef G_OS_WIN32
|
2004-02-14 01:23:36 +01:00
|
|
|
if (context->wake_up_pipe[0] != -1)
|
|
|
|
return;
|
2009-06-20 05:44:29 +02:00
|
|
|
|
|
|
|
#ifdef HAVE_PIPE2
|
|
|
|
/* if this fails, we fall through and try pipe */
|
|
|
|
pipe2 (context->wake_up_pipe, O_CLOEXEC);
|
|
|
|
#endif
|
|
|
|
if (context->wake_up_pipe[0] == -1)
|
|
|
|
{
|
|
|
|
if (pipe (context->wake_up_pipe) < 0)
|
|
|
|
g_error ("Cannot create pipe main loop wake-up: %s\n",
|
|
|
|
g_strerror (errno));
|
2008-06-13 16:11:03 +02:00
|
|
|
|
2009-06-20 05:44:29 +02:00
|
|
|
fcntl (context->wake_up_pipe[0], F_SETFD, FD_CLOEXEC);
|
|
|
|
fcntl (context->wake_up_pipe[1], F_SETFD, FD_CLOEXEC);
|
|
|
|
}
|
2008-06-13 16:11:03 +02:00
|
|
|
|
2002-08-06 16:50:52 +02:00
|
|
|
context->wake_up_rec.fd = context->wake_up_pipe[0];
|
|
|
|
context->wake_up_rec.events = G_IO_IN;
|
|
|
|
# else
|
2004-02-26 00:48:22 +01:00
|
|
|
if (context->wake_up_semaphore != NULL)
|
|
|
|
return;
|
2002-08-06 16:50:52 +02:00
|
|
|
context->wake_up_semaphore = CreateSemaphore (NULL, 0, 100, NULL);
|
|
|
|
if (context->wake_up_semaphore == NULL)
|
|
|
|
g_error ("Cannot create wake-up semaphore: %s",
|
|
|
|
g_win32_error_message (GetLastError ()));
|
2008-08-04 20:46:59 +02:00
|
|
|
context->wake_up_rec.fd = (gintptr) context->wake_up_semaphore;
|
2002-08-06 16:50:52 +02:00
|
|
|
context->wake_up_rec.events = G_IO_IN;
|
2008-08-21 04:27:13 +02:00
|
|
|
|
2008-09-25 21:59:49 +02:00
|
|
|
if (_g_main_poll_debug)
|
Fix problems on 64-bit Windows. Avoid warnings, some of which indicated
2008-07-28 Tor Lillqvist <tml@novell.com>
Fix problems on 64-bit Windows. Avoid warnings, some of which
indicated actual problems, some which were just annoyances.
Where casts to an integer type are needed for pointers, use
gssize. Technically intptr_t would be the more proper type, but we
still want to be compilable with MSVS6 and 7 which don't have
intptr_t. MSVS8 and 9 do have intptr_t, but in <crtdefs.h>, not
<stdint.h>.
Use %p to print out handles. Use gssize casts when assigning
GPollFD::fd fields.
Use G_GSIZE_FORMAT when printing size_t values.
* configure.in: Define automake conditional G_OS_WIN32_X64 which
is true on Win64.
* glib/giochannel.h: Use slightly different prototype for
g_io_channel_win32_new_messages() on Win64 with gsize instead of
guint.
* glib/giowin32.c
* glib/gmain.c
* glib/gspawn-win32.c
* tests/testglib.c: Generic changes as described above.
* glib/gmain.h: Don't bother mentioning GIMP in comment.
* glib/grel.c (tuple_hash_2): Use all bits of pointer.
* glib/gspawn-win32.c
* glib/gspawn-win32-helper.c: Use gssize types in the
communication between parent and helper process, so that we can
pass process handles, which are pointers, also on Win64.
* glib/gtimer.c (g_time_val_to_iso8601): time_t is 64 bits on
Win64 so we can't pass the address of a GTimeVal::tv_sec which is
a long directly to gmtime(). On the other hand, changing
GTimeVal::tv_sec to be a gint64 on Win64 is not really feasible
either, as that would then require changes in much code that uses
GTimeVals.
* glib/gspawn-win32.c
* glib/Makefile.am: Call the helper programs
gspawn-win64-helper.exe and gspawn-win64-helper-console.exe on
Win64, to avoid potential risk of running a 32-bit version of the
helper.
svn path=/trunk/; revision=7260
2008-07-28 02:24:14 +02:00
|
|
|
g_print ("wake-up semaphore: %p\n", context->wake_up_semaphore);
|
2002-08-06 16:50:52 +02:00
|
|
|
# endif
|
|
|
|
g_main_context_add_poll_unlocked (context, 0, &context->wake_up_rec);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2004-10-24 03:37:42 +02:00
|
|
|
_g_main_thread_init (void)
|
2002-08-06 16:50:52 +02:00
|
|
|
{
|
|
|
|
GSList *curr = main_contexts_without_pipe;
|
|
|
|
while (curr)
|
|
|
|
{
|
|
|
|
g_main_context_init_pipe ((GMainContext *)curr->data);
|
|
|
|
curr = curr->next;
|
|
|
|
}
|
|
|
|
g_slist_free (main_contexts_without_pipe);
|
|
|
|
main_contexts_without_pipe = NULL;
|
|
|
|
}
|
|
|
|
#endif /* G_THREADS_ENABLED */
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
2001-06-30 21:56:47 +02:00
|
|
|
* g_main_context_new:
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
2007-11-09 04:29:51 +01:00
|
|
|
* Creates a new #GMainContext structure.
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
2001-06-30 21:56:47 +02:00
|
|
|
* Return value: the new #GMainContext
|
2000-12-05 21:45:33 +01:00
|
|
|
**/
|
|
|
|
GMainContext *
|
2004-10-24 03:37:42 +02:00
|
|
|
g_main_context_new (void)
|
2000-12-05 21:45:33 +01:00
|
|
|
{
|
2001-06-30 21:56:47 +02:00
|
|
|
GMainContext *context = g_new0 (GMainContext, 1);
|
2000-12-12 21:23:37 +01:00
|
|
|
|
2008-08-21 04:27:13 +02:00
|
|
|
#ifdef G_MAIN_POLL_DEBUG
|
|
|
|
{
|
|
|
|
static gboolean beenhere = FALSE;
|
|
|
|
|
|
|
|
if (!beenhere)
|
|
|
|
{
|
|
|
|
if (getenv ("G_MAIN_POLL_DEBUG") != NULL)
|
2008-09-25 21:59:49 +02:00
|
|
|
_g_main_poll_debug = TRUE;
|
2008-08-21 04:27:13 +02:00
|
|
|
beenhere = TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2000-12-12 21:23:37 +01:00
|
|
|
#ifdef G_THREADS_ENABLED
|
2001-06-30 21:56:47 +02:00
|
|
|
g_static_mutex_init (&context->mutex);
|
2000-12-12 21:23:37 +01:00
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
context->owner = NULL;
|
|
|
|
context->waiters = NULL;
|
2004-02-14 01:23:36 +01:00
|
|
|
|
2004-02-26 00:48:22 +01:00
|
|
|
# ifndef G_OS_WIN32
|
2004-02-14 01:23:36 +01:00
|
|
|
context->wake_up_pipe[0] = -1;
|
|
|
|
context->wake_up_pipe[1] = -1;
|
2004-02-26 00:48:22 +01:00
|
|
|
# else
|
|
|
|
context->wake_up_semaphore = NULL;
|
|
|
|
# endif
|
2004-02-18 10:14:17 +01:00
|
|
|
#endif
|
2004-02-14 01:23:36 +01:00
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
context->ref_count = 1;
|
|
|
|
|
2002-08-06 16:50:52 +02:00
|
|
|
context->next_id = 1;
|
|
|
|
|
|
|
|
context->source_list = NULL;
|
|
|
|
|
|
|
|
context->poll_func = g_poll;
|
|
|
|
|
|
|
|
context->cached_poll_array = NULL;
|
|
|
|
context->cached_poll_array_size = 0;
|
|
|
|
|
|
|
|
context->pending_dispatches = g_ptr_array_new ();
|
|
|
|
|
2010-10-22 18:40:08 +02:00
|
|
|
context->time_is_fresh = FALSE;
|
2010-11-03 03:03:08 +01:00
|
|
|
context->real_time_is_fresh = FALSE;
|
2002-08-06 16:50:52 +02:00
|
|
|
|
Merge in current Win32 version. Almost no Unix code touched.
* README.win32: More text.
* config.h.win32 glibconfig.h.win32: Update to match the
corresponding generated files on Unix.
* makefile.msc: Update with new source files, and gthread
library. Use the compiler flag -MD instead of using -D_DLL and
"/nodefaultlib:libc msvcrt.lib" in the link phase.
* glib.def: Include new functions, drop removed ones.
* glib.h: Add comments about main loop and polling on Win32. (In
general, it's only for the GIMP's use.) Add Win32 IO Channel
functions. Remove the obsoleted old IO Channel stuff (which was
in #if 0 already).
* giowin32.c: New file.
* gmain.c: Include config.h, conditionalize <sys/time.h>
inclusion. Add g_poll implementation for Win32 (only for the
GIMP's needs for now, it's hard or even impossible to be as clean
and generic as on Unix). Implement g_get_current_time on Win32. If
threads aren't supported, don't try to wake up main thread's
loop. On Win32, use a semaphore and not a pipe to wake up the main
loop.
* gmessages.c: On Win32, allocate a console window if the standard
output handle is invalid before writing to stdout, and reopen stdout
to that console window.
* giochannel.c: Conditionalize unistd.h inclusion. Some indentation
cleanup.
* gstrfuncs.c: Include <signal.h>.
* gutils.c: On Win32, also check the HOMEDRIVE and HOMEPATH
environment variables.
* gmodule-dl.c gmodule-dld.c: In
_g_module_build_path, don't add the "lib" prefix and
".so" or ".sl" suffix if already there.
* gmodule-win32.c: Likewise for the ".dll" suffix.
* gthread-posix.c: Conditionalize <sys/time.h> inclusion.
1999-01-17 00:46:42 +01:00
|
|
|
#ifdef G_THREADS_ENABLED
|
2002-08-06 16:50:52 +02:00
|
|
|
if (g_thread_supported ())
|
|
|
|
g_main_context_init_pipe (context);
|
|
|
|
else
|
|
|
|
main_contexts_without_pipe = g_slist_prepend (main_contexts_without_pipe,
|
|
|
|
context);
|
2000-12-05 21:45:33 +01:00
|
|
|
#endif
|
|
|
|
|
2004-02-14 01:23:36 +01:00
|
|
|
G_LOCK (main_context_list);
|
|
|
|
main_context_list = g_slist_append (main_context_list, context);
|
2008-05-17 04:25:59 +02:00
|
|
|
|
|
|
|
#ifdef G_MAIN_POLL_DEBUG
|
2008-09-25 21:59:49 +02:00
|
|
|
if (_g_main_poll_debug)
|
2008-05-17 04:25:59 +02:00
|
|
|
g_print ("created context=%p\n", context);
|
|
|
|
#endif
|
|
|
|
|
2004-02-14 01:23:36 +01:00
|
|
|
G_UNLOCK (main_context_list);
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
return context;
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_main_context_default:
|
|
|
|
*
|
2009-06-05 03:59:15 +02:00
|
|
|
* Returns the global default main context. This is the main context
|
|
|
|
* used for main loop functions when a main loop is not explicitly
|
|
|
|
* specified, and corresponds to the "main" main loop. See also
|
|
|
|
* g_main_context_get_thread_default().
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
2009-06-05 03:59:15 +02:00
|
|
|
* Return value: the global default main context.
|
2000-12-05 21:45:33 +01:00
|
|
|
**/
|
|
|
|
GMainContext *
|
|
|
|
g_main_context_default (void)
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
/* Slow, but safe */
|
|
|
|
|
1998-12-16 06:38:35 +01:00
|
|
|
G_LOCK (main_loop);
|
1998-12-15 06:28:02 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
if (!default_main_context)
|
2008-05-17 04:25:59 +02:00
|
|
|
{
|
|
|
|
default_main_context = g_main_context_new ();
|
|
|
|
#ifdef G_MAIN_POLL_DEBUG
|
2008-09-25 21:59:49 +02:00
|
|
|
if (_g_main_poll_debug)
|
2008-05-17 04:25:59 +02:00
|
|
|
g_print ("default context=%p\n", default_main_context);
|
|
|
|
#endif
|
|
|
|
}
|
1998-12-15 06:28:02 +01:00
|
|
|
|
1998-12-16 06:38:35 +01:00
|
|
|
G_UNLOCK (main_loop);
|
1999-01-17 05:49:43 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
return default_main_context;
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
2009-06-05 03:59:15 +02:00
|
|
|
static GStaticPrivate thread_context_stack = G_STATIC_PRIVATE_INIT;
|
|
|
|
|
|
|
|
static void
|
|
|
|
free_context_stack (gpointer data)
|
|
|
|
{
|
|
|
|
GQueue *stack = data;
|
|
|
|
GMainContext *context;
|
|
|
|
|
|
|
|
while (!g_queue_is_empty (stack))
|
|
|
|
{
|
|
|
|
context = g_queue_pop_head (stack);
|
|
|
|
g_main_context_release (context);
|
|
|
|
if (context)
|
|
|
|
g_main_context_unref (context);
|
|
|
|
}
|
|
|
|
g_queue_free (stack);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_main_context_push_thread_default:
|
|
|
|
* @context: a #GMainContext, or %NULL for the global default context
|
|
|
|
*
|
|
|
|
* Acquires @context and sets it as the thread-default context for the
|
|
|
|
* current thread. This will cause certain asynchronous operations
|
|
|
|
* (such as most <link linkend="gio">gio</link>-based I/O) which are
|
|
|
|
* started in this thread to run under @context and deliver their
|
|
|
|
* results to its main loop, rather than running under the global
|
|
|
|
* default context in the main thread. Note that calling this function
|
|
|
|
* changes the context returned by
|
|
|
|
* g_main_context_get_thread_default(), <emphasis>not</emphasis> the
|
|
|
|
* one returned by g_main_context_default(), so it does not affect the
|
|
|
|
* context used by functions like g_idle_add().
|
|
|
|
*
|
|
|
|
* Normally you would call this function shortly after creating a new
|
|
|
|
* thread, passing it a #GMainContext which will be run by a
|
|
|
|
* #GMainLoop in that thread, to set a new default context for all
|
|
|
|
* async operations in that thread. (In this case, you don't need to
|
|
|
|
* ever call g_main_context_pop_thread_default().) In some cases
|
|
|
|
* however, you may want to schedule a single operation in a
|
|
|
|
* non-default context, or temporarily use a non-default context in
|
|
|
|
* the main thread. In that case, you can wrap the call to the
|
|
|
|
* asynchronous operation inside a
|
|
|
|
* g_main_context_push_thread_default() /
|
|
|
|
* g_main_context_pop_thread_default() pair, but it is up to you to
|
|
|
|
* ensure that no other asynchronous operations accidentally get
|
|
|
|
* started while the non-default context is active.
|
|
|
|
*
|
|
|
|
* Beware that libraries that predate this function may not correctly
|
|
|
|
* handle being used from a thread with a thread-default context. Eg,
|
|
|
|
* see g_file_supports_thread_contexts().
|
|
|
|
*
|
|
|
|
* Since: 2.22
|
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_main_context_push_thread_default (GMainContext *context)
|
|
|
|
{
|
|
|
|
GQueue *stack;
|
|
|
|
gboolean acquired_context;
|
|
|
|
|
|
|
|
acquired_context = g_main_context_acquire (context);
|
|
|
|
g_return_if_fail (acquired_context);
|
|
|
|
|
|
|
|
if (context == g_main_context_default ())
|
|
|
|
context = NULL;
|
|
|
|
else if (context)
|
|
|
|
g_main_context_ref (context);
|
|
|
|
|
|
|
|
stack = g_static_private_get (&thread_context_stack);
|
|
|
|
if (!stack)
|
|
|
|
{
|
|
|
|
stack = g_queue_new ();
|
|
|
|
g_static_private_set (&thread_context_stack, stack,
|
|
|
|
free_context_stack);
|
|
|
|
}
|
|
|
|
|
|
|
|
g_queue_push_head (stack, context);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_main_context_pop_thread_default:
|
|
|
|
* @context: a #GMainContext object, or %NULL
|
|
|
|
*
|
|
|
|
* Pops @context off the thread-default context stack (verifying that
|
|
|
|
* it was on the top of the stack).
|
|
|
|
*
|
|
|
|
* Since: 2.22
|
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_main_context_pop_thread_default (GMainContext *context)
|
|
|
|
{
|
|
|
|
GQueue *stack;
|
|
|
|
|
|
|
|
if (context == g_main_context_default ())
|
|
|
|
context = NULL;
|
|
|
|
|
|
|
|
stack = g_static_private_get (&thread_context_stack);
|
|
|
|
|
|
|
|
g_return_if_fail (stack != NULL);
|
|
|
|
g_return_if_fail (g_queue_peek_head (stack) == context);
|
|
|
|
|
|
|
|
g_queue_pop_head (stack);
|
|
|
|
|
|
|
|
g_main_context_release (context);
|
|
|
|
if (context)
|
|
|
|
g_main_context_unref (context);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_main_context_get_thread_default:
|
|
|
|
*
|
|
|
|
* Gets the thread-default #GMainContext for this thread. Asynchronous
|
|
|
|
* operations that want to be able to be run in contexts other than
|
|
|
|
* the default one should call this method to get a #GMainContext to
|
|
|
|
* add their #GSource<!-- -->s to. (Note that even in single-threaded
|
|
|
|
* programs applications may sometimes want to temporarily push a
|
|
|
|
* non-default context, so it is not safe to assume that this will
|
|
|
|
* always return %NULL if threads are not initialized.)
|
|
|
|
*
|
|
|
|
* Returns: the thread-default #GMainContext, or %NULL if the
|
|
|
|
* thread-default context is the global default context.
|
|
|
|
*
|
|
|
|
* Since: 2.22
|
|
|
|
**/
|
|
|
|
GMainContext *
|
|
|
|
g_main_context_get_thread_default (void)
|
|
|
|
{
|
|
|
|
GQueue *stack;
|
|
|
|
|
|
|
|
stack = g_static_private_get (&thread_context_stack);
|
|
|
|
if (stack)
|
|
|
|
return g_queue_peek_head (stack);
|
|
|
|
else
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/* Hooks for adding to the main loop */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_source_new:
|
|
|
|
* @source_funcs: structure containing functions that implement
|
|
|
|
* the sources behavior.
|
2001-12-16 20:31:36 +01:00
|
|
|
* @struct_size: size of the #GSource structure to create.
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
2001-12-16 20:31:36 +01:00
|
|
|
* Creates a new #GSource structure. The size is specified to
|
|
|
|
* allow creating structures derived from #GSource that contain
|
2000-12-05 21:45:33 +01:00
|
|
|
* additional data. The size passed in must be at least
|
2001-12-16 20:31:36 +01:00
|
|
|
* <literal>sizeof (GSource)</literal>.
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
|
|
|
* The source will not initially be associated with any #GMainContext
|
Remove references to nonexisting functions
* glib/gmain.c: Remove references to nonexisting functions
g_source_set_callback_closure(), g_source_poll(), g_source_add()
from docs.
* glib/gdir.c (g_dir_open): Typo fix in docs.
* glib/gasyncqueue.c (g_async_queue_lock):
(g_async_queue_unref_and_unlock): Fix markup to avoid erroneous
<link>s in docs.
* glib/gwin32.c: Escape #'s leading to erroneous <link>s in docs.
* glib/gtree.c: Replace some occurances of Gtree by GTree in docs.
* glib/gstring.c (g_string_insert_unichar): Typo fix in docs.
* glib/tmpl/conversions.sgml: Add GIConv.
* glib/tmpl/main.sgml: Fix references to nonexisting functions
g_main_loop_destroy(), g_source_add(), g_source_connect().
* glib/glib-sections.txt: Add GIConv, g_str_has_prefix, g_str_has_suffix.
* glib/tmpl/linked_lists_single.sgml:
* glib/tmpl/linked_lists_double.sgml: GListAllocator doesn't exist.
* glib/glib-docs.sgml: Declare hash entity.
* glib/tmpl/macros.sgml: Escape # in #ifdef to suppress erroneous links.
* gobject/Makefile.am, gobject/gobject-docs.sgml, gobject/tmpl/*:
* glib/Makefile.am, glib/glib-docs.sgml, glib/tmpl/*: Produce XML,
not SGML.
2002-05-27 00:46:28 +02:00
|
|
|
* and must be added to one with g_source_attach() before it will be
|
2000-12-05 21:45:33 +01:00
|
|
|
* executed.
|
|
|
|
*
|
2001-12-16 20:31:36 +01:00
|
|
|
* Return value: the newly-created #GSource.
|
2000-12-05 21:45:33 +01:00
|
|
|
**/
|
|
|
|
GSource *
|
|
|
|
g_source_new (GSourceFuncs *source_funcs,
|
|
|
|
guint struct_size)
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
GSource *source;
|
|
|
|
|
|
|
|
g_return_val_if_fail (source_funcs != NULL, NULL);
|
|
|
|
g_return_val_if_fail (struct_size >= sizeof (GSource), NULL);
|
1998-12-15 06:28:02 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
source = (GSource*) g_malloc0 (struct_size);
|
|
|
|
|
|
|
|
source->source_funcs = source_funcs;
|
|
|
|
source->ref_count = 1;
|
1998-12-15 06:28:02 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
source->priority = G_PRIORITY_DEFAULT;
|
1998-12-15 06:28:02 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
source->flags = G_HOOK_FLAG_ACTIVE;
|
1999-01-17 05:49:43 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/* NULL/0 initialization for all other fields */
|
|
|
|
|
|
|
|
return source;
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/* Holds context's lock
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
g_source_list_add (GSource *source,
|
|
|
|
GMainContext *context)
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
GSource *tmp_source, *last_source;
|
|
|
|
|
2010-11-06 15:11:15 +01:00
|
|
|
if (source->priv && source->priv->parent_source)
|
|
|
|
{
|
|
|
|
/* Put the source immediately before its parent */
|
|
|
|
tmp_source = source->priv->parent_source;
|
|
|
|
last_source = source->priv->parent_source->prev;
|
|
|
|
}
|
|
|
|
else
|
2000-12-05 21:45:33 +01:00
|
|
|
{
|
2010-11-06 15:11:15 +01:00
|
|
|
last_source = NULL;
|
|
|
|
tmp_source = context->source_list;
|
|
|
|
while (tmp_source && tmp_source->priority <= source->priority)
|
|
|
|
{
|
|
|
|
last_source = tmp_source;
|
|
|
|
tmp_source = tmp_source->next;
|
|
|
|
}
|
2000-12-05 21:45:33 +01:00
|
|
|
}
|
1998-12-17 05:06:27 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
source->next = tmp_source;
|
|
|
|
if (tmp_source)
|
|
|
|
tmp_source->prev = source;
|
|
|
|
|
|
|
|
source->prev = last_source;
|
|
|
|
if (last_source)
|
|
|
|
last_source->next = source;
|
|
|
|
else
|
|
|
|
context->source_list = source;
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/* Holds context's lock
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
g_source_list_remove (GSource *source,
|
|
|
|
GMainContext *context)
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
if (source->prev)
|
|
|
|
source->prev->next = source->next;
|
|
|
|
else
|
|
|
|
context->source_list = source->next;
|
1998-12-15 06:28:02 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
if (source->next)
|
|
|
|
source->next->prev = source->prev;
|
1999-01-17 05:49:43 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
source->prev = NULL;
|
|
|
|
source->next = NULL;
|
1999-01-17 05:49:43 +01:00
|
|
|
}
|
|
|
|
|
2010-11-06 15:11:15 +01:00
|
|
|
static guint
|
|
|
|
g_source_attach_unlocked (GSource *source,
|
|
|
|
GMainContext *context)
|
|
|
|
{
|
|
|
|
guint result = 0;
|
|
|
|
GSList *tmp_list;
|
|
|
|
|
|
|
|
source->context = context;
|
|
|
|
result = source->source_id = context->next_id++;
|
|
|
|
|
|
|
|
source->ref_count++;
|
|
|
|
g_source_list_add (source, context);
|
|
|
|
|
|
|
|
tmp_list = source->poll_fds;
|
|
|
|
while (tmp_list)
|
|
|
|
{
|
|
|
|
g_main_context_add_poll_unlocked (context, source->priority, tmp_list->data);
|
|
|
|
tmp_list = tmp_list->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (source->priv)
|
|
|
|
{
|
|
|
|
tmp_list = source->priv->child_sources;
|
|
|
|
while (tmp_list)
|
|
|
|
{
|
|
|
|
g_source_attach_unlocked (tmp_list->data, context);
|
|
|
|
tmp_list = tmp_list->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_source_attach:
|
|
|
|
* @source: a #GSource
|
|
|
|
* @context: a #GMainContext (if %NULL, the default context will be used)
|
|
|
|
*
|
|
|
|
* Adds a #GSource to a @context so that it will be executed within
|
2008-09-24 15:44:27 +02:00
|
|
|
* that context. Remove it by calling g_source_destroy().
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
2006-01-05 22:07:55 +01:00
|
|
|
* Return value: the ID (greater than 0) for the source within the
|
|
|
|
* #GMainContext.
|
2000-12-05 21:45:33 +01:00
|
|
|
**/
|
|
|
|
guint
|
|
|
|
g_source_attach (GSource *source,
|
|
|
|
GMainContext *context)
|
1999-01-17 05:49:43 +01:00
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
guint result = 0;
|
1999-01-17 05:49:43 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
g_return_val_if_fail (source->context == NULL, 0);
|
|
|
|
g_return_val_if_fail (!SOURCE_DESTROYED (source), 0);
|
|
|
|
|
|
|
|
if (!context)
|
|
|
|
context = g_main_context_default ();
|
1999-01-17 05:49:43 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
LOCK_CONTEXT (context);
|
1999-01-17 05:49:43 +01:00
|
|
|
|
2010-11-06 15:11:15 +01:00
|
|
|
result = g_source_attach_unlocked (source, context);
|
1999-01-17 05:49:43 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
#ifdef G_THREADS_ENABLED
|
|
|
|
/* Now wake up the main loop if it is waiting in the poll() */
|
2001-06-30 21:56:47 +02:00
|
|
|
g_main_context_wakeup_unlocked (context);
|
2000-12-05 21:45:33 +01:00
|
|
|
#endif
|
1999-01-17 05:49:43 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
UNLOCK_CONTEXT (context);
|
1999-01-17 05:49:43 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
return result;
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
static void
|
|
|
|
g_source_destroy_internal (GSource *source,
|
|
|
|
GMainContext *context,
|
|
|
|
gboolean have_lock)
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
if (!have_lock)
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
if (!SOURCE_DESTROYED (source))
|
|
|
|
{
|
2000-12-10 17:02:48 +01:00
|
|
|
GSList *tmp_list;
|
2000-12-05 21:45:33 +01:00
|
|
|
gpointer old_cb_data;
|
|
|
|
GSourceCallbackFuncs *old_cb_funcs;
|
|
|
|
|
|
|
|
source->flags &= ~G_HOOK_FLAG_ACTIVE;
|
1998-12-17 05:06:27 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
old_cb_data = source->callback_data;
|
|
|
|
old_cb_funcs = source->callback_funcs;
|
Merge in current Win32 version. Almost no Unix code touched.
* README.win32: More text.
* config.h.win32 glibconfig.h.win32: Update to match the
corresponding generated files on Unix.
* makefile.msc: Update with new source files, and gthread
library. Use the compiler flag -MD instead of using -D_DLL and
"/nodefaultlib:libc msvcrt.lib" in the link phase.
* glib.def: Include new functions, drop removed ones.
* glib.h: Add comments about main loop and polling on Win32. (In
general, it's only for the GIMP's use.) Add Win32 IO Channel
functions. Remove the obsoleted old IO Channel stuff (which was
in #if 0 already).
* giowin32.c: New file.
* gmain.c: Include config.h, conditionalize <sys/time.h>
inclusion. Add g_poll implementation for Win32 (only for the
GIMP's needs for now, it's hard or even impossible to be as clean
and generic as on Unix). Implement g_get_current_time on Win32. If
threads aren't supported, don't try to wake up main thread's
loop. On Win32, use a semaphore and not a pipe to wake up the main
loop.
* gmessages.c: On Win32, allocate a console window if the standard
output handle is invalid before writing to stdout, and reopen stdout
to that console window.
* giochannel.c: Conditionalize unistd.h inclusion. Some indentation
cleanup.
* gstrfuncs.c: Include <signal.h>.
* gutils.c: On Win32, also check the HOMEDRIVE and HOMEPATH
environment variables.
* gmodule-dl.c gmodule-dld.c: In
_g_module_build_path, don't add the "lib" prefix and
".so" or ".sl" suffix if already there.
* gmodule-win32.c: Likewise for the ".dll" suffix.
* gthread-posix.c: Conditionalize <sys/time.h> inclusion.
1999-01-17 00:46:42 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
source->callback_data = NULL;
|
|
|
|
source->callback_funcs = NULL;
|
2000-08-31 22:56:42 +02:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
if (old_cb_funcs)
|
|
|
|
{
|
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
old_cb_funcs->unref (old_cb_data);
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
}
|
2004-03-19 21:25:03 +01:00
|
|
|
|
|
|
|
if (!SOURCE_BLOCKED (source))
|
2000-12-10 17:02:48 +01:00
|
|
|
{
|
2004-03-19 21:25:03 +01:00
|
|
|
tmp_list = source->poll_fds;
|
|
|
|
while (tmp_list)
|
|
|
|
{
|
|
|
|
g_main_context_remove_poll_unlocked (context, tmp_list->data);
|
|
|
|
tmp_list = tmp_list->next;
|
|
|
|
}
|
2000-12-10 17:02:48 +01:00
|
|
|
}
|
2010-11-06 15:11:15 +01:00
|
|
|
|
|
|
|
if (source->priv && source->priv->child_sources)
|
|
|
|
{
|
|
|
|
/* This is safe because even if a child_source finalizer or
|
|
|
|
* closure notify tried to modify source->priv->child_sources
|
|
|
|
* from outside the lock, it would fail since
|
|
|
|
* SOURCE_DESTROYED(source) is now TRUE.
|
|
|
|
*/
|
|
|
|
tmp_list = source->priv->child_sources;
|
|
|
|
while (tmp_list)
|
|
|
|
{
|
|
|
|
g_source_destroy_internal (tmp_list->data, context, TRUE);
|
|
|
|
g_source_unref_internal (tmp_list->data, context, TRUE);
|
|
|
|
tmp_list = tmp_list->next;
|
|
|
|
}
|
|
|
|
g_slist_free (source->priv->child_sources);
|
|
|
|
source->priv->child_sources = NULL;
|
|
|
|
}
|
2004-03-19 21:25:03 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
g_source_unref_internal (source, context, TRUE);
|
|
|
|
}
|
Merge in current Win32 version. Almost no Unix code touched.
* README.win32: More text.
* config.h.win32 glibconfig.h.win32: Update to match the
corresponding generated files on Unix.
* makefile.msc: Update with new source files, and gthread
library. Use the compiler flag -MD instead of using -D_DLL and
"/nodefaultlib:libc msvcrt.lib" in the link phase.
* glib.def: Include new functions, drop removed ones.
* glib.h: Add comments about main loop and polling on Win32. (In
general, it's only for the GIMP's use.) Add Win32 IO Channel
functions. Remove the obsoleted old IO Channel stuff (which was
in #if 0 already).
* giowin32.c: New file.
* gmain.c: Include config.h, conditionalize <sys/time.h>
inclusion. Add g_poll implementation for Win32 (only for the
GIMP's needs for now, it's hard or even impossible to be as clean
and generic as on Unix). Implement g_get_current_time on Win32. If
threads aren't supported, don't try to wake up main thread's
loop. On Win32, use a semaphore and not a pipe to wake up the main
loop.
* gmessages.c: On Win32, allocate a console window if the standard
output handle is invalid before writing to stdout, and reopen stdout
to that console window.
* giochannel.c: Conditionalize unistd.h inclusion. Some indentation
cleanup.
* gstrfuncs.c: Include <signal.h>.
* gutils.c: On Win32, also check the HOMEDRIVE and HOMEPATH
environment variables.
* gmodule-dl.c gmodule-dld.c: In
_g_module_build_path, don't add the "lib" prefix and
".so" or ".sl" suffix if already there.
* gmodule-win32.c: Likewise for the ".dll" suffix.
* gthread-posix.c: Conditionalize <sys/time.h> inclusion.
1999-01-17 00:46:42 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
if (!have_lock)
|
|
|
|
UNLOCK_CONTEXT (context);
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_source_destroy:
|
|
|
|
* @source: a #GSource
|
|
|
|
*
|
2001-12-16 20:31:36 +01:00
|
|
|
* Removes a source from its #GMainContext, if any, and mark it as
|
2000-12-05 21:45:33 +01:00
|
|
|
* destroyed. The source cannot be subsequently added to another
|
|
|
|
* context.
|
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_source_destroy (GSource *source)
|
|
|
|
{
|
|
|
|
GMainContext *context;
|
|
|
|
|
|
|
|
g_return_if_fail (source != NULL);
|
|
|
|
|
|
|
|
context = source->context;
|
|
|
|
|
|
|
|
if (context)
|
|
|
|
g_source_destroy_internal (source, context, FALSE);
|
|
|
|
else
|
|
|
|
source->flags &= ~G_HOOK_FLAG_ACTIVE;
|
|
|
|
}
|
1998-12-02 15:55:27 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_source_get_id:
|
|
|
|
* @source: a #GSource
|
|
|
|
*
|
2001-12-16 20:31:36 +01:00
|
|
|
* Returns the numeric ID for a particular source. The ID of a source
|
2006-01-05 22:07:55 +01:00
|
|
|
* is a positive integer which is unique within a particular main loop
|
|
|
|
* context. The reverse
|
2000-12-05 21:45:33 +01:00
|
|
|
* mapping from ID to source is done by g_main_context_find_source_by_id().
|
|
|
|
*
|
2006-01-05 22:07:55 +01:00
|
|
|
* Return value: the ID (greater than 0) for the source
|
2000-12-05 21:45:33 +01:00
|
|
|
**/
|
|
|
|
guint
|
|
|
|
g_source_get_id (GSource *source)
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
guint result;
|
|
|
|
|
|
|
|
g_return_val_if_fail (source != NULL, 0);
|
|
|
|
g_return_val_if_fail (source->context != NULL, 0);
|
|
|
|
|
|
|
|
LOCK_CONTEXT (source->context);
|
2001-11-29 00:51:51 +01:00
|
|
|
result = source->source_id;
|
2000-12-05 21:45:33 +01:00
|
|
|
UNLOCK_CONTEXT (source->context);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
1998-12-02 15:55:27 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_source_get_context:
|
|
|
|
* @source: a #GSource
|
|
|
|
*
|
2001-12-16 20:31:36 +01:00
|
|
|
* Gets the #GMainContext with which the source is associated.
|
2000-12-05 21:45:33 +01:00
|
|
|
* Calling this function on a destroyed source is an error.
|
|
|
|
*
|
|
|
|
* Return value: the #GMainContext with which the source is associated,
|
|
|
|
* or %NULL if the context has not yet been added
|
|
|
|
* to a source.
|
|
|
|
**/
|
|
|
|
GMainContext *
|
|
|
|
g_source_get_context (GSource *source)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (!SOURCE_DESTROYED (source), NULL);
|
1998-12-02 15:55:27 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
return source->context;
|
|
|
|
}
|
1998-12-15 06:28:02 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
2001-02-22 16:39:57 +01:00
|
|
|
* g_source_add_poll:
|
2000-12-05 21:45:33 +01:00
|
|
|
* @source:a #GSource
|
|
|
|
* @fd: a #GPollFD structure holding information about a file
|
|
|
|
* descriptor to watch.
|
|
|
|
*
|
2001-12-16 20:31:36 +01:00
|
|
|
* Adds a file descriptor to the set of file descriptors polled for
|
2000-12-05 21:45:33 +01:00
|
|
|
* this source. This is usually combined with g_source_new() to add an
|
|
|
|
* event source. The event source's check function will typically test
|
2001-12-16 20:31:36 +01:00
|
|
|
* the @revents field in the #GPollFD struct and return %TRUE if events need
|
2000-12-05 21:45:33 +01:00
|
|
|
* to be processed.
|
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_source_add_poll (GSource *source,
|
|
|
|
GPollFD *fd)
|
|
|
|
{
|
|
|
|
GMainContext *context;
|
|
|
|
|
|
|
|
g_return_if_fail (source != NULL);
|
|
|
|
g_return_if_fail (fd != NULL);
|
Warn if no callback. Call callback correctly. (g_io_win32_create_watch):
2000-12-14 Tor Lillqvist <tml@iki.fi>
* giowin32.c (g_io_win32_dispatch): Warn if no callback. Call
callback correctly.
(g_io_win32_create_watch): Fix typo.
(g_io_win32_fd_create_watch): Ditto.
(g_io_channel_unix_new): If it is a file descriptor (i.e., a Unix
fd lookalike provided by the C library), call
g_io_channel_win32_new_fd(). If it is a socket (from WinSock),
call g_io_cahnnel_win32_new_stream_socket(). Hopefully sockets and
fds don't overlap. TODO: Implement also datagram sockets.
(g_io_channel_win32_poll): Call g_main_context_get_poll_func().
* gcompletion.h: Include <unistd.h> only on Unix. Is this
inclusion really needed here? OTOH, do include <stddef.h>, for
size_t.
* gmessages.c: (Win32) Don't define a function called "write" that
might clash with the prototype from <io.h>, use a #define.
* glib.def: Update.
* gmain.c (g_source_add_poll): Don't return a value from void
function.
(g_main_context_get_poll_func): Compile also for non-Win32, as
presumably was intended. The result var is a GPollFunc, not a
GPollFunc*. Return the result!
gobject:
2000-12-14 Tor Lillqvist <tml@iki.fi>
* makefile.mingw.in: Update, include parts from Makefile.am to
build gmarshal.[ch]. Some day, we won't need these separate
makefiles for Win32 compilation. I hope.
* makefile.msc.in: Update. No use trying to build gmarshal.[ch]
here, it would require Unixish tools. MSVC users building from CVS
sources are out of luck.
* gobject.def: Update.
2000-12-14 22:02:20 +01:00
|
|
|
g_return_if_fail (!SOURCE_DESTROYED (source));
|
2000-12-10 17:02:48 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
context = source->context;
|
1998-12-15 06:28:02 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
if (context)
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
source->poll_fds = g_slist_prepend (source->poll_fds, fd);
|
1998-12-02 15:55:27 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
if (context)
|
|
|
|
{
|
2004-03-19 21:25:03 +01:00
|
|
|
if (!SOURCE_BLOCKED (source))
|
|
|
|
g_main_context_add_poll_unlocked (context, source->priority, fd);
|
2000-12-05 21:45:33 +01:00
|
|
|
UNLOCK_CONTEXT (context);
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-02-22 16:39:57 +01:00
|
|
|
/**
|
|
|
|
* g_source_remove_poll:
|
|
|
|
* @source:a #GSource
|
Remove references to nonexisting functions
* glib/gmain.c: Remove references to nonexisting functions
g_source_set_callback_closure(), g_source_poll(), g_source_add()
from docs.
* glib/gdir.c (g_dir_open): Typo fix in docs.
* glib/gasyncqueue.c (g_async_queue_lock):
(g_async_queue_unref_and_unlock): Fix markup to avoid erroneous
<link>s in docs.
* glib/gwin32.c: Escape #'s leading to erroneous <link>s in docs.
* glib/gtree.c: Replace some occurances of Gtree by GTree in docs.
* glib/gstring.c (g_string_insert_unichar): Typo fix in docs.
* glib/tmpl/conversions.sgml: Add GIConv.
* glib/tmpl/main.sgml: Fix references to nonexisting functions
g_main_loop_destroy(), g_source_add(), g_source_connect().
* glib/glib-sections.txt: Add GIConv, g_str_has_prefix, g_str_has_suffix.
* glib/tmpl/linked_lists_single.sgml:
* glib/tmpl/linked_lists_double.sgml: GListAllocator doesn't exist.
* glib/glib-docs.sgml: Declare hash entity.
* glib/tmpl/macros.sgml: Escape # in #ifdef to suppress erroneous links.
* gobject/Makefile.am, gobject/gobject-docs.sgml, gobject/tmpl/*:
* glib/Makefile.am, glib/glib-docs.sgml, glib/tmpl/*: Produce XML,
not SGML.
2002-05-27 00:46:28 +02:00
|
|
|
* @fd: a #GPollFD structure previously passed to g_source_add_poll().
|
2001-02-22 16:39:57 +01:00
|
|
|
*
|
2001-12-16 20:31:36 +01:00
|
|
|
* Removes a file descriptor from the set of file descriptors polled for
|
2001-02-22 16:39:57 +01:00
|
|
|
* this source.
|
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_source_remove_poll (GSource *source,
|
|
|
|
GPollFD *fd)
|
|
|
|
{
|
|
|
|
GMainContext *context;
|
|
|
|
|
|
|
|
g_return_if_fail (source != NULL);
|
|
|
|
g_return_if_fail (fd != NULL);
|
|
|
|
g_return_if_fail (!SOURCE_DESTROYED (source));
|
|
|
|
|
|
|
|
context = source->context;
|
|
|
|
|
|
|
|
if (context)
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
source->poll_fds = g_slist_remove (source->poll_fds, fd);
|
|
|
|
|
|
|
|
if (context)
|
|
|
|
{
|
2004-03-19 21:25:03 +01:00
|
|
|
if (!SOURCE_BLOCKED (source))
|
|
|
|
g_main_context_remove_poll_unlocked (context, fd);
|
2001-02-22 16:39:57 +01:00
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-11-06 15:11:15 +01:00
|
|
|
/**
|
|
|
|
* g_source_add_child_source:
|
|
|
|
* @source:a #GSource
|
|
|
|
* @child_source: a second #GSource that @source should "poll"
|
|
|
|
*
|
|
|
|
* Adds @child_source to @source as a "polled" source; when @source is
|
|
|
|
* added to a #GMainContext, @child_source will be automatically added
|
|
|
|
* with the same priority, when @child_source is triggered, it will
|
2011-01-28 16:17:54 +01:00
|
|
|
* cause @source to dispatch (in addition to calling its own
|
2010-11-06 15:11:15 +01:00
|
|
|
* callback), and when @source is destroyed, it will destroy
|
|
|
|
* @child_source as well. (@source will also still be dispatched if
|
|
|
|
* its own prepare/check functions indicate that it is ready.)
|
|
|
|
*
|
2011-01-28 16:17:54 +01:00
|
|
|
* If you don't need @child_source to do anything on its own when it
|
2010-11-06 15:11:15 +01:00
|
|
|
* triggers, you can call g_source_set_dummy_callback() on it to set a
|
|
|
|
* callback that does nothing (except return %TRUE if appropriate).
|
|
|
|
*
|
|
|
|
* @source will hold a reference on @child_source while @child_source
|
|
|
|
* is attached to it.
|
2010-12-01 00:04:17 +01:00
|
|
|
*
|
|
|
|
* Since: 2.28
|
2010-11-06 15:11:15 +01:00
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_source_add_child_source (GSource *source,
|
|
|
|
GSource *child_source)
|
|
|
|
{
|
|
|
|
GMainContext *context;
|
|
|
|
|
|
|
|
g_return_if_fail (source != NULL);
|
|
|
|
g_return_if_fail (child_source != NULL);
|
|
|
|
g_return_if_fail (!SOURCE_DESTROYED (source));
|
|
|
|
g_return_if_fail (!SOURCE_DESTROYED (child_source));
|
|
|
|
g_return_if_fail (child_source->context == NULL);
|
|
|
|
g_return_if_fail (child_source->priv == NULL || child_source->priv->parent_source == NULL);
|
|
|
|
|
|
|
|
context = source->context;
|
|
|
|
|
|
|
|
if (context)
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
if (!source->priv)
|
|
|
|
source->priv = g_slice_new0 (GSourcePrivate);
|
|
|
|
if (!child_source->priv)
|
|
|
|
child_source->priv = g_slice_new0 (GSourcePrivate);
|
|
|
|
|
|
|
|
source->priv->child_sources = g_slist_prepend (source->priv->child_sources,
|
|
|
|
g_source_ref (child_source));
|
|
|
|
child_source->priv->parent_source = source;
|
|
|
|
g_source_set_priority_unlocked (child_source, context, source->priority);
|
|
|
|
|
|
|
|
if (context)
|
|
|
|
{
|
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
g_source_attach (child_source, context);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_source_remove_child_source:
|
|
|
|
* @source:a #GSource
|
|
|
|
* @child_source: a #GSource previously passed to
|
|
|
|
* g_source_add_child_source().
|
|
|
|
*
|
|
|
|
* Detaches @child_source from @source and destroys it.
|
2010-12-01 00:04:17 +01:00
|
|
|
*
|
|
|
|
* Since: 2.28
|
2010-11-06 15:11:15 +01:00
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_source_remove_child_source (GSource *source,
|
|
|
|
GSource *child_source)
|
|
|
|
{
|
|
|
|
GMainContext *context;
|
|
|
|
|
|
|
|
g_return_if_fail (source != NULL);
|
|
|
|
g_return_if_fail (child_source != NULL);
|
|
|
|
g_return_if_fail (child_source->priv != NULL && child_source->priv->parent_source == source);
|
|
|
|
g_return_if_fail (!SOURCE_DESTROYED (source));
|
|
|
|
g_return_if_fail (!SOURCE_DESTROYED (child_source));
|
|
|
|
|
|
|
|
context = source->context;
|
|
|
|
|
|
|
|
if (context)
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
source->priv->child_sources = g_slist_remove (source->priv->child_sources, child_source);
|
|
|
|
g_source_destroy_internal (child_source, context, TRUE);
|
|
|
|
g_source_unref_internal (child_source, context, TRUE);
|
|
|
|
|
|
|
|
if (context)
|
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_source_set_callback_indirect:
|
|
|
|
* @source: the source
|
|
|
|
* @callback_data: pointer to callback data "object"
|
2001-12-16 20:31:36 +01:00
|
|
|
* @callback_funcs: functions for reference counting @callback_data
|
2000-12-05 21:45:33 +01:00
|
|
|
* and getting the callback and data
|
|
|
|
*
|
2001-12-16 20:31:36 +01:00
|
|
|
* Sets the callback function storing the data as a refcounted callback
|
Remove references to nonexisting functions
* glib/gmain.c: Remove references to nonexisting functions
g_source_set_callback_closure(), g_source_poll(), g_source_add()
from docs.
* glib/gdir.c (g_dir_open): Typo fix in docs.
* glib/gasyncqueue.c (g_async_queue_lock):
(g_async_queue_unref_and_unlock): Fix markup to avoid erroneous
<link>s in docs.
* glib/gwin32.c: Escape #'s leading to erroneous <link>s in docs.
* glib/gtree.c: Replace some occurances of Gtree by GTree in docs.
* glib/gstring.c (g_string_insert_unichar): Typo fix in docs.
* glib/tmpl/conversions.sgml: Add GIConv.
* glib/tmpl/main.sgml: Fix references to nonexisting functions
g_main_loop_destroy(), g_source_add(), g_source_connect().
* glib/glib-sections.txt: Add GIConv, g_str_has_prefix, g_str_has_suffix.
* glib/tmpl/linked_lists_single.sgml:
* glib/tmpl/linked_lists_double.sgml: GListAllocator doesn't exist.
* glib/glib-docs.sgml: Declare hash entity.
* glib/tmpl/macros.sgml: Escape # in #ifdef to suppress erroneous links.
* gobject/Makefile.am, gobject/gobject-docs.sgml, gobject/tmpl/*:
* glib/Makefile.am, glib/glib-docs.sgml, glib/tmpl/*: Produce XML,
not SGML.
2002-05-27 00:46:28 +02:00
|
|
|
* "object". This is used internally. Note that calling
|
|
|
|
* g_source_set_callback_indirect() assumes
|
2000-12-05 21:45:33 +01:00
|
|
|
* an initial reference count on @callback_data, and thus
|
|
|
|
* @callback_funcs->unref will eventually be called once more
|
|
|
|
* than @callback_funcs->ref.
|
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_source_set_callback_indirect (GSource *source,
|
|
|
|
gpointer callback_data,
|
|
|
|
GSourceCallbackFuncs *callback_funcs)
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
GMainContext *context;
|
|
|
|
gpointer old_cb_data;
|
|
|
|
GSourceCallbackFuncs *old_cb_funcs;
|
|
|
|
|
|
|
|
g_return_if_fail (source != NULL);
|
|
|
|
g_return_if_fail (callback_funcs != NULL || callback_data == NULL);
|
1998-12-02 15:55:27 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
context = source->context;
|
1998-12-02 15:55:27 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
if (context)
|
|
|
|
LOCK_CONTEXT (context);
|
1998-12-15 06:28:02 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
old_cb_data = source->callback_data;
|
|
|
|
old_cb_funcs = source->callback_funcs;
|
1999-07-24 20:50:58 +02:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
source->callback_data = callback_data;
|
|
|
|
source->callback_funcs = callback_funcs;
|
1998-12-02 15:55:27 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
if (context)
|
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
if (old_cb_funcs)
|
|
|
|
old_cb_funcs->unref (old_cb_data);
|
|
|
|
}
|
1998-12-18 03:23:33 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
static void
|
|
|
|
g_source_callback_ref (gpointer cb_data)
|
|
|
|
{
|
|
|
|
GSourceCallback *callback = cb_data;
|
1998-12-02 15:55:27 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
callback->ref_count++;
|
|
|
|
}
|
1998-12-02 15:55:27 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
|
|
|
|
static void
|
|
|
|
g_source_callback_unref (gpointer cb_data)
|
|
|
|
{
|
|
|
|
GSourceCallback *callback = cb_data;
|
|
|
|
|
|
|
|
callback->ref_count--;
|
|
|
|
if (callback->ref_count == 0)
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
if (callback->notify)
|
|
|
|
callback->notify (callback->data);
|
2001-01-23 08:35:52 +01:00
|
|
|
g_free (callback);
|
2000-12-05 21:45:33 +01:00
|
|
|
}
|
|
|
|
}
|
1998-12-02 15:55:27 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
static void
|
|
|
|
g_source_callback_get (gpointer cb_data,
|
2001-09-04 00:12:51 +02:00
|
|
|
GSource *source,
|
2000-12-05 21:45:33 +01:00
|
|
|
GSourceFunc *func,
|
|
|
|
gpointer *data)
|
|
|
|
{
|
|
|
|
GSourceCallback *callback = cb_data;
|
1998-12-02 15:55:27 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
*func = callback->func;
|
|
|
|
*data = callback->data;
|
|
|
|
}
|
1999-02-02 02:04:41 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
static GSourceCallbackFuncs g_source_callback_funcs = {
|
|
|
|
g_source_callback_ref,
|
|
|
|
g_source_callback_unref,
|
|
|
|
g_source_callback_get,
|
|
|
|
};
|
1999-02-02 02:04:41 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_source_set_callback:
|
|
|
|
* @source: the source
|
|
|
|
* @func: a callback function
|
|
|
|
* @data: the data to pass to callback function
|
|
|
|
* @notify: a function to call when @data is no longer in use, or %NULL.
|
|
|
|
*
|
2002-09-01 15:04:02 +02:00
|
|
|
* Sets the callback function for a source. The callback for a source is
|
|
|
|
* called from the source's dispatch function.
|
|
|
|
*
|
|
|
|
* The exact type of @func depends on the type of source; ie. you
|
|
|
|
* should not count on @func being called with @data as its first
|
|
|
|
* parameter.
|
|
|
|
*
|
|
|
|
* Typically, you won't use this function. Instead use functions specific
|
|
|
|
* to the type of source you are using.
|
2000-12-05 21:45:33 +01:00
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_source_set_callback (GSource *source,
|
|
|
|
GSourceFunc func,
|
|
|
|
gpointer data,
|
|
|
|
GDestroyNotify notify)
|
|
|
|
{
|
|
|
|
GSourceCallback *new_callback;
|
1999-02-02 02:04:41 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
g_return_if_fail (source != NULL);
|
1998-12-18 03:23:33 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
new_callback = g_new (GSourceCallback, 1);
|
2001-01-23 08:35:52 +01:00
|
|
|
|
|
|
|
new_callback->ref_count = 1;
|
2000-12-05 21:45:33 +01:00
|
|
|
new_callback->func = func;
|
|
|
|
new_callback->data = data;
|
|
|
|
new_callback->notify = notify;
|
|
|
|
|
|
|
|
g_source_set_callback_indirect (source, new_callback, &g_source_callback_funcs);
|
|
|
|
}
|
|
|
|
|
2006-06-02 04:36:30 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* g_source_set_funcs:
|
|
|
|
* @source: a #GSource
|
|
|
|
* @funcs: the new #GSourceFuncs
|
|
|
|
*
|
|
|
|
* Sets the source functions (can be used to override
|
|
|
|
* default implementations) of an unattached source.
|
|
|
|
*
|
|
|
|
* Since: 2.12
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
g_source_set_funcs (GSource *source,
|
|
|
|
GSourceFuncs *funcs)
|
|
|
|
{
|
|
|
|
g_return_if_fail (source != NULL);
|
|
|
|
g_return_if_fail (source->context == NULL);
|
|
|
|
g_return_if_fail (source->ref_count > 0);
|
|
|
|
g_return_if_fail (funcs != NULL);
|
|
|
|
|
|
|
|
source->source_funcs = funcs;
|
|
|
|
}
|
|
|
|
|
2010-11-06 15:11:15 +01:00
|
|
|
static void
|
|
|
|
g_source_set_priority_unlocked (GSource *source,
|
|
|
|
GMainContext *context,
|
|
|
|
gint priority)
|
2000-12-05 21:45:33 +01:00
|
|
|
{
|
|
|
|
GSList *tmp_list;
|
|
|
|
|
|
|
|
source->priority = priority;
|
|
|
|
|
|
|
|
if (context)
|
|
|
|
{
|
2004-03-19 21:25:03 +01:00
|
|
|
/* Remove the source from the context's source and then
|
2010-11-06 15:11:15 +01:00
|
|
|
* add it back so it is sorted in the correct place
|
2004-03-19 21:25:03 +01:00
|
|
|
*/
|
|
|
|
g_source_list_remove (source, source->context);
|
|
|
|
g_source_list_add (source, source->context);
|
|
|
|
|
|
|
|
if (!SOURCE_BLOCKED (source))
|
2000-12-05 21:45:33 +01:00
|
|
|
{
|
2004-03-19 21:25:03 +01:00
|
|
|
tmp_list = source->poll_fds;
|
|
|
|
while (tmp_list)
|
|
|
|
{
|
|
|
|
g_main_context_remove_poll_unlocked (context, tmp_list->data);
|
|
|
|
g_main_context_add_poll_unlocked (context, priority, tmp_list->data);
|
|
|
|
|
|
|
|
tmp_list = tmp_list->next;
|
|
|
|
}
|
2000-12-05 21:45:33 +01:00
|
|
|
}
|
2010-11-06 15:11:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (source->priv && source->priv->child_sources)
|
|
|
|
{
|
|
|
|
tmp_list = source->priv->child_sources;
|
|
|
|
while (tmp_list)
|
|
|
|
{
|
|
|
|
g_source_set_priority_unlocked (tmp_list->data, context, priority);
|
|
|
|
tmp_list = tmp_list->next;
|
|
|
|
}
|
2000-12-05 21:45:33 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-11-06 15:11:15 +01:00
|
|
|
/**
|
|
|
|
* g_source_set_priority:
|
|
|
|
* @source: a #GSource
|
|
|
|
* @priority: the new priority.
|
|
|
|
*
|
|
|
|
* Sets the priority of a source. While the main loop is being run, a
|
|
|
|
* source will be dispatched if it is ready to be dispatched and no
|
|
|
|
* sources at a higher (numerically smaller) priority are ready to be
|
|
|
|
* dispatched.
|
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_source_set_priority (GSource *source,
|
|
|
|
gint priority)
|
|
|
|
{
|
|
|
|
GMainContext *context;
|
|
|
|
|
|
|
|
g_return_if_fail (source != NULL);
|
|
|
|
|
|
|
|
context = source->context;
|
|
|
|
|
|
|
|
if (context)
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
g_source_set_priority_unlocked (source, context, priority);
|
|
|
|
if (context)
|
|
|
|
UNLOCK_CONTEXT (source->context);
|
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_source_get_priority:
|
|
|
|
* @source: a #GSource
|
|
|
|
*
|
2001-12-16 20:31:36 +01:00
|
|
|
* Gets the priority of a source.
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
|
|
|
* Return value: the priority of the source
|
|
|
|
**/
|
|
|
|
gint
|
|
|
|
g_source_get_priority (GSource *source)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (source != NULL, 0);
|
|
|
|
|
|
|
|
return source->priority;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_source_set_can_recurse:
|
|
|
|
* @source: a #GSource
|
|
|
|
* @can_recurse: whether recursion is allowed for this source
|
|
|
|
*
|
|
|
|
* Sets whether a source can be called recursively. If @can_recurse is
|
|
|
|
* %TRUE, then while the source is being dispatched then this source
|
|
|
|
* will be processed normally. Otherwise, all processing of this
|
|
|
|
* source is blocked until the dispatch function returns.
|
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_source_set_can_recurse (GSource *source,
|
|
|
|
gboolean can_recurse)
|
|
|
|
{
|
|
|
|
GMainContext *context;
|
|
|
|
|
|
|
|
g_return_if_fail (source != NULL);
|
|
|
|
|
|
|
|
context = source->context;
|
|
|
|
|
|
|
|
if (context)
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
if (can_recurse)
|
|
|
|
source->flags |= G_SOURCE_CAN_RECURSE;
|
|
|
|
else
|
|
|
|
source->flags &= ~G_SOURCE_CAN_RECURSE;
|
|
|
|
|
|
|
|
if (context)
|
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_source_get_can_recurse:
|
|
|
|
* @source: a #GSource
|
|
|
|
*
|
|
|
|
* Checks whether a source is allowed to be called recursively.
|
2001-12-16 20:31:36 +01:00
|
|
|
* see g_source_set_can_recurse().
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
|
|
|
* Return value: whether recursion is allowed.
|
|
|
|
**/
|
|
|
|
gboolean
|
|
|
|
g_source_get_can_recurse (GSource *source)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (source != NULL, FALSE);
|
|
|
|
|
|
|
|
return (source->flags & G_SOURCE_CAN_RECURSE) != 0;
|
|
|
|
}
|
|
|
|
|
2010-04-20 23:47:44 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* g_source_set_name:
|
|
|
|
* @source: a #GSource
|
|
|
|
* @name: debug name for the source
|
|
|
|
*
|
|
|
|
* Sets a name for the source, used in debugging and profiling.
|
|
|
|
* The name defaults to #NULL.
|
|
|
|
*
|
|
|
|
* The source name should describe in a human-readable way
|
|
|
|
* what the source does. For example, "X11 event queue"
|
|
|
|
* or "GTK+ repaint idle handler" or whatever it is.
|
|
|
|
*
|
|
|
|
* It is permitted to call this function multiple times, but is not
|
|
|
|
* recommended due to the potential performance impact. For example,
|
|
|
|
* one could change the name in the "check" function of a #GSourceFuncs
|
|
|
|
* to include details like the event type in the source name.
|
|
|
|
*
|
|
|
|
* Since: 2.26
|
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_source_set_name (GSource *source,
|
|
|
|
const char *name)
|
|
|
|
{
|
|
|
|
g_return_if_fail (source != NULL);
|
|
|
|
|
|
|
|
/* setting back to NULL is allowed, just because it's
|
|
|
|
* weird if get_name can return NULL but you can't
|
|
|
|
* set that.
|
|
|
|
*/
|
|
|
|
|
|
|
|
g_free (source->name);
|
|
|
|
source->name = g_strdup (name);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_source_get_name:
|
|
|
|
* @source: a #GSource
|
|
|
|
*
|
|
|
|
* Gets a name for the source, used in debugging and profiling.
|
|
|
|
* The name may be #NULL if it has never been set with
|
|
|
|
* g_source_set_name().
|
|
|
|
*
|
|
|
|
* Return value: the name of the source
|
|
|
|
* Since: 2.26
|
|
|
|
**/
|
|
|
|
G_CONST_RETURN char*
|
|
|
|
g_source_get_name (GSource *source)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (source != NULL, NULL);
|
|
|
|
|
|
|
|
return source->name;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_source_set_name_by_id:
|
|
|
|
* @tag: a #GSource ID
|
|
|
|
* @name: debug name for the source
|
|
|
|
*
|
|
|
|
* Sets the name of a source using its ID.
|
|
|
|
*
|
|
|
|
* This is a convenience utility to set source names from the return
|
|
|
|
* value of g_idle_add(), g_timeout_add(), etc.
|
|
|
|
*
|
|
|
|
* Since: 2.26
|
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_source_set_name_by_id (guint tag,
|
|
|
|
const char *name)
|
|
|
|
{
|
|
|
|
GSource *source;
|
|
|
|
|
|
|
|
g_return_if_fail (tag > 0);
|
|
|
|
|
|
|
|
source = g_main_context_find_source_by_id (NULL, tag);
|
|
|
|
if (source == NULL)
|
|
|
|
return;
|
|
|
|
|
|
|
|
g_source_set_name (source, name);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_source_ref:
|
|
|
|
* @source: a #GSource
|
|
|
|
*
|
|
|
|
* Increases the reference count on a source by one.
|
|
|
|
*
|
|
|
|
* Return value: @source
|
|
|
|
**/
|
|
|
|
GSource *
|
|
|
|
g_source_ref (GSource *source)
|
|
|
|
{
|
|
|
|
GMainContext *context;
|
|
|
|
|
|
|
|
g_return_val_if_fail (source != NULL, NULL);
|
|
|
|
|
|
|
|
context = source->context;
|
|
|
|
|
|
|
|
if (context)
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
source->ref_count++;
|
|
|
|
|
|
|
|
if (context)
|
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
return source;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* g_source_unref() but possible to call within context lock
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
g_source_unref_internal (GSource *source,
|
|
|
|
GMainContext *context,
|
|
|
|
gboolean have_lock)
|
|
|
|
{
|
2001-01-03 17:05:39 +01:00
|
|
|
gpointer old_cb_data = NULL;
|
|
|
|
GSourceCallbackFuncs *old_cb_funcs = NULL;
|
2000-12-05 21:45:33 +01:00
|
|
|
|
|
|
|
g_return_if_fail (source != NULL);
|
|
|
|
|
|
|
|
if (!have_lock && context)
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
source->ref_count--;
|
|
|
|
if (source->ref_count == 0)
|
|
|
|
{
|
2001-01-03 17:05:39 +01:00
|
|
|
old_cb_data = source->callback_data;
|
|
|
|
old_cb_funcs = source->callback_funcs;
|
|
|
|
|
|
|
|
source->callback_data = NULL;
|
|
|
|
source->callback_funcs = NULL;
|
|
|
|
|
2010-11-06 14:45:20 +01:00
|
|
|
if (context)
|
2000-12-05 21:45:33 +01:00
|
|
|
{
|
2010-11-06 14:45:20 +01:00
|
|
|
if (!SOURCE_DESTROYED (source))
|
|
|
|
g_warning (G_STRLOC ": ref_count == 0, but source was still attached to a context!");
|
|
|
|
g_source_list_remove (source, context);
|
2000-12-05 21:45:33 +01:00
|
|
|
}
|
2000-12-10 17:02:48 +01:00
|
|
|
|
2001-06-30 22:06:16 +02:00
|
|
|
if (source->source_funcs->finalize)
|
2010-11-06 14:35:25 +01:00
|
|
|
{
|
|
|
|
if (context)
|
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
source->source_funcs->finalize (source);
|
|
|
|
if (context)
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
}
|
2010-04-20 23:47:44 +02:00
|
|
|
|
|
|
|
g_free (source->name);
|
|
|
|
source->name = NULL;
|
|
|
|
|
2000-12-10 17:02:48 +01:00
|
|
|
g_slist_free (source->poll_fds);
|
|
|
|
source->poll_fds = NULL;
|
|
|
|
g_free (source);
|
2000-12-05 21:45:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!have_lock && context)
|
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
|
2001-01-03 17:05:39 +01:00
|
|
|
if (old_cb_funcs)
|
2000-12-05 21:45:33 +01:00
|
|
|
{
|
|
|
|
if (have_lock)
|
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
|
2001-01-03 17:05:39 +01:00
|
|
|
old_cb_funcs->unref (old_cb_data);
|
2000-12-05 21:45:33 +01:00
|
|
|
|
|
|
|
if (have_lock)
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_source_unref:
|
|
|
|
* @source: a #GSource
|
|
|
|
*
|
|
|
|
* Decreases the reference count of a source by one. If the
|
|
|
|
* resulting reference count is zero the source and associated
|
|
|
|
* memory will be destroyed.
|
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_source_unref (GSource *source)
|
|
|
|
{
|
|
|
|
g_return_if_fail (source != NULL);
|
|
|
|
|
|
|
|
g_source_unref_internal (source, source->context, FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_main_context_find_source_by_id:
|
|
|
|
* @context: a #GMainContext (if %NULL, the default context will be used)
|
2006-01-05 22:07:55 +01:00
|
|
|
* @source_id: the source ID, as returned by g_source_get_id().
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
2006-01-05 22:07:55 +01:00
|
|
|
* Finds a #GSource given a pair of context and ID.
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
|
|
|
* Return value: the #GSource if found, otherwise, %NULL
|
|
|
|
**/
|
|
|
|
GSource *
|
|
|
|
g_main_context_find_source_by_id (GMainContext *context,
|
2001-11-29 00:51:51 +01:00
|
|
|
guint source_id)
|
2000-12-05 21:45:33 +01:00
|
|
|
{
|
|
|
|
GSource *source;
|
|
|
|
|
2003-03-06 23:41:03 +01:00
|
|
|
g_return_val_if_fail (source_id > 0, NULL);
|
2000-12-05 21:45:33 +01:00
|
|
|
|
|
|
|
if (context == NULL)
|
|
|
|
context = g_main_context_default ();
|
|
|
|
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
source = context->source_list;
|
|
|
|
while (source)
|
|
|
|
{
|
|
|
|
if (!SOURCE_DESTROYED (source) &&
|
2001-11-29 00:51:51 +01:00
|
|
|
source->source_id == source_id)
|
2000-12-05 21:45:33 +01:00
|
|
|
break;
|
|
|
|
source = source->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
return source;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_main_context_find_source_by_funcs_user_data:
|
|
|
|
* @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.
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* Return value: the source, if one was found, otherwise %NULL
|
|
|
|
**/
|
|
|
|
GSource *
|
|
|
|
g_main_context_find_source_by_funcs_user_data (GMainContext *context,
|
|
|
|
GSourceFuncs *funcs,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
GSource *source;
|
|
|
|
|
2003-03-06 23:41:03 +01:00
|
|
|
g_return_val_if_fail (funcs != NULL, NULL);
|
2000-12-05 21:45:33 +01:00
|
|
|
|
|
|
|
if (context == NULL)
|
|
|
|
context = g_main_context_default ();
|
|
|
|
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
source = context->source_list;
|
|
|
|
while (source)
|
|
|
|
{
|
|
|
|
if (!SOURCE_DESTROYED (source) &&
|
|
|
|
source->source_funcs == funcs &&
|
2001-09-19 22:39:21 +02:00
|
|
|
source->callback_funcs)
|
|
|
|
{
|
|
|
|
GSourceFunc callback;
|
|
|
|
gpointer callback_data;
|
|
|
|
|
|
|
|
source->callback_funcs->get (source->callback_data, source, &callback, &callback_data);
|
|
|
|
|
|
|
|
if (callback_data == user_data)
|
|
|
|
break;
|
|
|
|
}
|
2000-12-05 21:45:33 +01:00
|
|
|
source = source->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
return source;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_main_context_find_source_by_user_data:
|
|
|
|
* @context: a #GMainContext
|
|
|
|
* @user_data: the user_data for the callback.
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* Return value: the source, if one was found, otherwise %NULL
|
|
|
|
**/
|
|
|
|
GSource *
|
|
|
|
g_main_context_find_source_by_user_data (GMainContext *context,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
GSource *source;
|
|
|
|
|
|
|
|
if (context == NULL)
|
|
|
|
context = g_main_context_default ();
|
|
|
|
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
source = context->source_list;
|
|
|
|
while (source)
|
|
|
|
{
|
|
|
|
if (!SOURCE_DESTROYED (source) &&
|
2001-09-19 22:39:21 +02:00
|
|
|
source->callback_funcs)
|
|
|
|
{
|
|
|
|
GSourceFunc callback;
|
|
|
|
gpointer callback_data = NULL;
|
|
|
|
|
|
|
|
source->callback_funcs->get (source->callback_data, source, &callback, &callback_data);
|
|
|
|
|
|
|
|
if (callback_data == user_data)
|
|
|
|
break;
|
|
|
|
}
|
2000-12-05 21:45:33 +01:00
|
|
|
source = source->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
return source;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_source_remove:
|
2006-01-05 22:07:55 +01:00
|
|
|
* @tag: the ID of the source to remove.
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
2006-01-05 22:07:55 +01:00
|
|
|
* Removes the source with the given id from the default main context.
|
|
|
|
* The id of
|
2004-02-14 01:23:36 +01:00
|
|
|
* a #GSource is given by g_source_get_id(), or will be returned by the
|
|
|
|
* functions g_source_attach(), g_idle_add(), g_idle_add_full(),
|
|
|
|
* g_timeout_add(), g_timeout_add_full(), g_child_watch_add(),
|
|
|
|
* g_child_watch_add_full(), g_io_add_watch(), and g_io_add_watch_full().
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
2008-09-24 15:44:27 +02:00
|
|
|
* See also g_source_destroy(). You must use g_source_destroy() for sources
|
|
|
|
* added to a non-default main context.
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
|
|
|
* Return value: %TRUE if the source was found and removed.
|
|
|
|
**/
|
|
|
|
gboolean
|
|
|
|
g_source_remove (guint tag)
|
|
|
|
{
|
|
|
|
GSource *source;
|
|
|
|
|
|
|
|
g_return_val_if_fail (tag > 0, FALSE);
|
|
|
|
|
|
|
|
source = g_main_context_find_source_by_id (NULL, tag);
|
|
|
|
if (source)
|
|
|
|
g_source_destroy (source);
|
|
|
|
|
|
|
|
return source != NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_source_remove_by_user_data:
|
|
|
|
* @user_data: the user_data for the callback.
|
|
|
|
*
|
|
|
|
* Removes a source from the default main loop context given the user
|
|
|
|
* data for the callback. If multiple sources exist with the same user
|
|
|
|
* data, only one will be destroyed.
|
|
|
|
*
|
|
|
|
* Return value: %TRUE if a source was found and removed.
|
|
|
|
**/
|
|
|
|
gboolean
|
|
|
|
g_source_remove_by_user_data (gpointer user_data)
|
|
|
|
{
|
|
|
|
GSource *source;
|
|
|
|
|
|
|
|
source = g_main_context_find_source_by_user_data (NULL, user_data);
|
|
|
|
if (source)
|
|
|
|
{
|
|
|
|
g_source_destroy (source);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_source_remove_by_funcs_user_data:
|
|
|
|
* @funcs: The @source_funcs passed to g_source_new()
|
|
|
|
* @user_data: the user data for the callback
|
|
|
|
*
|
|
|
|
* Removes a source from the default main loop context given the
|
|
|
|
* source functions and user data. If multiple sources exist with the
|
|
|
|
* same source functions and user data, only one will be destroyed.
|
|
|
|
*
|
|
|
|
* Return value: %TRUE if a source was found and removed.
|
|
|
|
**/
|
|
|
|
gboolean
|
|
|
|
g_source_remove_by_funcs_user_data (GSourceFuncs *funcs,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
GSource *source;
|
|
|
|
|
|
|
|
g_return_val_if_fail (funcs != NULL, FALSE);
|
|
|
|
|
|
|
|
source = g_main_context_find_source_by_funcs_user_data (NULL, funcs, user_data);
|
|
|
|
if (source)
|
|
|
|
{
|
|
|
|
g_source_destroy (source);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_get_current_time:
|
|
|
|
* @result: #GTimeVal structure in which to store current time.
|
2010-11-01 21:40:36 +01:00
|
|
|
*
|
2003-07-25 23:32:47 +02:00
|
|
|
* Equivalent to the UNIX gettimeofday() function, but portable.
|
2010-11-01 21:40:36 +01:00
|
|
|
*
|
|
|
|
* You may find g_get_real_time() to be more convenient.
|
2000-12-05 21:45:33 +01:00
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_get_current_time (GTimeVal *result)
|
|
|
|
{
|
|
|
|
#ifndef G_OS_WIN32
|
|
|
|
struct timeval r;
|
|
|
|
|
|
|
|
g_return_if_fail (result != NULL);
|
|
|
|
|
|
|
|
/*this is required on alpha, there the timeval structs are int's
|
|
|
|
not longs and a cast only would fail horribly*/
|
|
|
|
gettimeofday (&r, NULL);
|
|
|
|
result->tv_sec = r.tv_sec;
|
|
|
|
result->tv_usec = r.tv_usec;
|
|
|
|
#else
|
2005-03-29 10:24:48 +02:00
|
|
|
FILETIME ft;
|
2008-08-04 21:22:05 +02:00
|
|
|
guint64 time64;
|
2000-12-05 21:45:33 +01:00
|
|
|
|
2008-07-02 18:16:27 +02:00
|
|
|
g_return_if_fail (result != NULL);
|
|
|
|
|
2005-03-29 10:24:48 +02:00
|
|
|
GetSystemTimeAsFileTime (&ft);
|
2008-08-04 21:22:05 +02:00
|
|
|
memmove (&time64, &ft, sizeof (FILETIME));
|
2000-12-05 21:45:33 +01:00
|
|
|
|
2005-03-29 10:24:48 +02:00
|
|
|
/* Convert from 100s of nanoseconds since 1601-01-01
|
|
|
|
* to Unix epoch. Yes, this is Y2038 unsafe.
|
|
|
|
*/
|
2008-08-04 21:22:05 +02:00
|
|
|
time64 -= G_GINT64_CONSTANT (116444736000000000);
|
|
|
|
time64 /= 10;
|
2000-12-05 21:45:33 +01:00
|
|
|
|
2008-08-04 21:22:05 +02:00
|
|
|
result->tv_sec = time64 / 1000000;
|
|
|
|
result->tv_usec = time64 % 1000000;
|
2000-12-05 21:45:33 +01:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2010-11-01 21:40:36 +01:00
|
|
|
/**
|
|
|
|
* g_get_real_time:
|
|
|
|
*
|
|
|
|
* Queries the system wall-clock time.
|
|
|
|
*
|
|
|
|
* This call is functionally equivalent to g_get_current_time() except
|
|
|
|
* that the return value is often more convenient than dealing with a
|
|
|
|
* #GTimeVal.
|
|
|
|
*
|
|
|
|
* You should only use this call if you are actually interested in the real
|
|
|
|
* wall-clock time. g_get_monotonic_time() is probably more useful for
|
|
|
|
* measuring intervals.
|
|
|
|
*
|
|
|
|
* Returns: the number of microseconds since January 1, 1970 UTC.
|
|
|
|
*
|
|
|
|
* Since: 2.28
|
|
|
|
**/
|
|
|
|
gint64
|
|
|
|
g_get_real_time (void)
|
|
|
|
{
|
|
|
|
GTimeVal tv;
|
|
|
|
|
|
|
|
g_get_current_time (&tv);
|
|
|
|
|
|
|
|
return (((gint64) tv.tv_sec) * 1000000) + tv.tv_usec;
|
|
|
|
}
|
|
|
|
|
2010-10-22 18:42:32 +02:00
|
|
|
/**
|
|
|
|
* g_get_monotonic_time:
|
|
|
|
*
|
|
|
|
* Queries the system monotonic time, if available.
|
|
|
|
*
|
|
|
|
* On POSIX systems with clock_gettime() and %CLOCK_MONOTONIC this call
|
|
|
|
* is a very shallow wrapper for that. Otherwise, we make a best effort
|
|
|
|
* that probably involves returning the wall clock time (with at least
|
|
|
|
* microsecond accuracy, subject to the limitations of the OS kernel).
|
|
|
|
*
|
|
|
|
* Note that, on Windows, "limitations of the OS kernel" is a rather
|
|
|
|
* substantial statement. Depending on the configuration of the system,
|
|
|
|
* the wall clock time is updated as infrequently as 64 times a second
|
|
|
|
* (which is approximately every 16ms).
|
|
|
|
*
|
2010-11-01 20:46:35 +01:00
|
|
|
* Returns: the monotonic time, in microseconds
|
|
|
|
*
|
2010-10-22 18:42:32 +02:00
|
|
|
* Since: 2.28
|
|
|
|
**/
|
2010-11-01 20:46:35 +01:00
|
|
|
gint64
|
|
|
|
g_get_monotonic_time (void)
|
2010-10-22 18:42:32 +02:00
|
|
|
{
|
|
|
|
#ifdef HAVE_CLOCK_GETTIME
|
|
|
|
/* librt clock_gettime() is our first choice */
|
|
|
|
{
|
|
|
|
static int clockid = CLOCK_REALTIME;
|
|
|
|
struct timespec ts;
|
|
|
|
|
|
|
|
#ifdef HAVE_MONOTONIC_CLOCK
|
|
|
|
/* We have to check if we actually have monotonic clock support.
|
|
|
|
*
|
|
|
|
* There is no thread safety issue here since there is no harm if we
|
|
|
|
* check twice.
|
|
|
|
*/
|
|
|
|
{
|
|
|
|
static gboolean checked;
|
|
|
|
|
|
|
|
if G_UNLIKELY (!checked)
|
|
|
|
{
|
|
|
|
if (sysconf (_SC_MONOTONIC_CLOCK) >= 0)
|
|
|
|
clockid = CLOCK_MONOTONIC;
|
|
|
|
checked = TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
clock_gettime (clockid, &ts);
|
2010-11-01 20:46:35 +01:00
|
|
|
|
|
|
|
/* In theory monotonic time can have any epoch.
|
|
|
|
*
|
|
|
|
* glib presently assumes the following:
|
|
|
|
*
|
|
|
|
* 1) The epoch comes some time after the birth of Jesus of Nazareth, but
|
|
|
|
* not more than 10000 years later.
|
|
|
|
*
|
|
|
|
* 2) The current time also falls sometime within this range.
|
|
|
|
*
|
|
|
|
* These two reasonable assumptions leave us with a maximum deviation from
|
|
|
|
* the epoch of 10000 years, or 315569520000000000 seconds.
|
|
|
|
*
|
|
|
|
* If we restrict ourselves to this range then the number of microseconds
|
|
|
|
* will always fit well inside the constraints of a int64 (by a factor of
|
|
|
|
* about 29).
|
|
|
|
*
|
|
|
|
* If you actually hit the following assertion, probably you should file a
|
|
|
|
* bug against your operating system for being excessively silly.
|
|
|
|
**/
|
|
|
|
g_assert (G_GINT64_CONSTANT (-315569520000000000) < ts.tv_sec &&
|
|
|
|
ts.tv_sec < G_GINT64_CONSTANT (315569520000000000));
|
|
|
|
|
|
|
|
return (((gint64) ts.tv_sec) * 1000000) + (ts.tv_nsec / 1000);
|
2010-10-22 18:42:32 +02:00
|
|
|
}
|
|
|
|
#else
|
|
|
|
/* It may look like we are discarding accuracy on Windows (since its
|
|
|
|
* current time is expressed in 100s of nanoseconds) but according to
|
|
|
|
* many sources, the time is only updated 64 times per second, so
|
|
|
|
* microsecond accuracy is more than enough.
|
|
|
|
*/
|
|
|
|
{
|
|
|
|
GTimeVal tv;
|
|
|
|
|
|
|
|
g_get_current_time (&tv);
|
2010-11-01 20:46:35 +01:00
|
|
|
|
|
|
|
return (((gint64) tv.tv_sec) * 1000000) + tv.tv_usec;
|
2010-10-22 18:42:32 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2006-06-05 17:02:51 +02:00
|
|
|
static void
|
|
|
|
g_main_dispatch_free (gpointer dispatch)
|
|
|
|
{
|
|
|
|
g_slice_free (GMainDispatch, dispatch);
|
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/* Running the main loop */
|
|
|
|
|
2006-06-02 04:36:30 +02:00
|
|
|
static GMainDispatch *
|
|
|
|
get_dispatch (void)
|
2004-03-01 03:41:09 +01:00
|
|
|
{
|
|
|
|
static GStaticPrivate depth_private = G_STATIC_PRIVATE_INIT;
|
2006-06-02 04:36:30 +02:00
|
|
|
GMainDispatch *dispatch = g_static_private_get (&depth_private);
|
|
|
|
if (!dispatch)
|
2004-03-01 03:41:09 +01:00
|
|
|
{
|
2006-06-02 04:36:30 +02:00
|
|
|
dispatch = g_slice_new0 (GMainDispatch);
|
2006-06-05 17:02:51 +02:00
|
|
|
g_static_private_set (&depth_private, dispatch, g_main_dispatch_free);
|
2004-03-01 03:41:09 +01:00
|
|
|
}
|
|
|
|
|
2006-06-02 04:36:30 +02:00
|
|
|
return dispatch;
|
2004-03-01 03:41:09 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_main_depth:
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
2007-09-16 19:30:57 +02:00
|
|
|
* |[
|
2004-03-01 03:41:09 +01:00
|
|
|
* static GList *free_list;
|
2007-09-16 19:30:57 +02:00
|
|
|
*
|
2004-03-01 03:41:09 +01:00
|
|
|
* gpointer
|
|
|
|
* allocate_memory (gsize size)
|
|
|
|
* {
|
|
|
|
* gpointer result = g_malloc (size);
|
|
|
|
* free_list = g_list_prepend (free_list, result);
|
|
|
|
* return result;
|
|
|
|
* }
|
2007-09-16 19:30:57 +02:00
|
|
|
*
|
2004-03-01 03:41:09 +01:00
|
|
|
* 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;
|
|
|
|
* }
|
2007-09-16 19:30:57 +02:00
|
|
|
*
|
2004-03-01 03:41:09 +01:00
|
|
|
* [...]
|
2007-09-16 19:30:57 +02:00
|
|
|
*
|
2004-03-01 03:41:09 +01:00
|
|
|
* while (TRUE);
|
|
|
|
* {
|
|
|
|
* g_main_context_iteration (NULL, TRUE);
|
|
|
|
* free_allocated_memory();
|
|
|
|
* }
|
2007-09-16 19:30:57 +02:00
|
|
|
* ]|
|
2004-03-01 03:41:09 +01:00
|
|
|
*
|
|
|
|
* 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
|
2004-03-01 03:45:15 +01:00
|
|
|
* recursive callback. This can be fixed by using g_main_depth()
|
2004-03-01 03:41:09 +01:00
|
|
|
*
|
2007-09-16 19:30:57 +02:00
|
|
|
* |[
|
2004-03-01 03:41:09 +01:00
|
|
|
* gpointer
|
|
|
|
* allocate_memory (gsize size)
|
|
|
|
* {
|
2007-09-16 19:30:57 +02:00
|
|
|
* FreeListBlock *block = g_new (FreeListBlock, 1);
|
2004-03-01 03:41:09 +01:00
|
|
|
* block->mem = g_malloc (size);
|
2004-03-01 03:45:15 +01:00
|
|
|
* block->depth = g_main_depth ();
|
2004-03-01 03:41:09 +01:00
|
|
|
* free_list = g_list_prepend (free_list, block);
|
|
|
|
* return block->mem;
|
|
|
|
* }
|
2007-09-16 19:30:57 +02:00
|
|
|
*
|
2004-03-01 03:41:09 +01:00
|
|
|
* void
|
|
|
|
* free_allocated_memory (void)
|
|
|
|
* {
|
|
|
|
* GList *l;
|
2007-09-16 19:30:57 +02:00
|
|
|
*
|
2004-03-01 03:45:15 +01:00
|
|
|
* int depth = g_main_depth ();
|
2004-03-01 03:41:09 +01:00
|
|
|
* for (l = free_list; l; );
|
|
|
|
* {
|
|
|
|
* GList *next = l->next;
|
|
|
|
* FreeListBlock *block = l->data;
|
2004-03-11 02:03:12 +01:00
|
|
|
* if (block->depth > depth)
|
2004-03-01 03:41:09 +01:00
|
|
|
* {
|
|
|
|
* g_free (block->mem);
|
|
|
|
* g_free (block);
|
|
|
|
* free_list = g_list_delete_link (free_list, l);
|
|
|
|
* }
|
2007-09-16 19:30:57 +02:00
|
|
|
*
|
2004-03-01 03:41:09 +01:00
|
|
|
* l = next;
|
|
|
|
* }
|
|
|
|
* }
|
2007-09-16 19:30:57 +02:00
|
|
|
* ]|
|
2004-03-01 03:41:09 +01:00
|
|
|
*
|
2004-03-01 03:45:15 +01:00
|
|
|
* There is a temptation to use g_main_depth() to solve
|
2004-03-01 03:41:09 +01:00
|
|
|
* 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
|
2004-03-01 15:20:30 +01:00
|
|
|
* 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
|
2004-03-01 03:41:09 +01:00
|
|
|
* 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:
|
|
|
|
*
|
|
|
|
* <orderedlist>
|
|
|
|
* <listitem>
|
|
|
|
* <para>
|
|
|
|
* Use gtk_widget_set_sensitive() or modal dialogs to prevent
|
|
|
|
* the user from interacting with elements while the main
|
|
|
|
* loop is recursing.
|
|
|
|
* </para>
|
|
|
|
* </listitem>
|
|
|
|
* <listitem>
|
|
|
|
* <para>
|
|
|
|
* 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.
|
|
|
|
* </para>
|
|
|
|
* </listitem>
|
2004-03-02 01:05:36 +01:00
|
|
|
* </orderedlist>
|
2007-09-16 19:30:57 +02:00
|
|
|
*
|
|
|
|
* Return value: The main loop recursion level in the current thread
|
2004-03-01 03:41:09 +01:00
|
|
|
**/
|
|
|
|
int
|
|
|
|
g_main_depth (void)
|
|
|
|
{
|
2006-06-02 04:36:30 +02:00
|
|
|
GMainDispatch *dispatch = get_dispatch ();
|
|
|
|
return dispatch->depth;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_main_current_source:
|
|
|
|
*
|
2006-06-02 04:55:50 +02:00
|
|
|
* Returns the currently firing source for this thread.
|
|
|
|
*
|
|
|
|
* Return value: The currently firing source or %NULL.
|
|
|
|
*
|
|
|
|
* Since: 2.12
|
2006-06-02 04:36:30 +02:00
|
|
|
*/
|
|
|
|
GSource *
|
|
|
|
g_main_current_source (void)
|
|
|
|
{
|
|
|
|
GMainDispatch *dispatch = get_dispatch ();
|
2008-05-02 13:22:10 +02:00
|
|
|
return dispatch->dispatching_sources ? dispatch->dispatching_sources->data : NULL;
|
2004-03-01 03:41:09 +01:00
|
|
|
}
|
|
|
|
|
2006-06-02 04:36:30 +02:00
|
|
|
/**
|
|
|
|
* g_source_is_destroyed:
|
|
|
|
* @source: a #GSource
|
|
|
|
*
|
|
|
|
* Returns whether @source has been destroyed.
|
|
|
|
*
|
|
|
|
* This is important when you operate upon your objects
|
|
|
|
* from within idle handlers, but may have freed the object
|
|
|
|
* before the dispatch of your idle handler.
|
|
|
|
*
|
2007-11-25 07:05:06 +01:00
|
|
|
* |[
|
2006-06-02 04:36:30 +02:00
|
|
|
* static gboolean
|
|
|
|
* idle_callback (gpointer data)
|
|
|
|
* {
|
|
|
|
* SomeWidget *self = data;
|
|
|
|
*
|
2006-06-05 19:34:53 +02:00
|
|
|
* GDK_THREADS_ENTER (<!-- -->);
|
2006-06-02 04:36:30 +02:00
|
|
|
* /<!-- -->* do stuff with self *<!-- -->/
|
|
|
|
* self->idle_id = 0;
|
2006-06-05 19:34:53 +02:00
|
|
|
* GDK_THREADS_LEAVE (<!-- -->);
|
2006-06-02 04:36:30 +02:00
|
|
|
*
|
|
|
|
* return FALSE;
|
|
|
|
* }
|
2006-06-05 19:34:53 +02:00
|
|
|
*
|
2006-06-02 04:36:30 +02:00
|
|
|
* static void
|
|
|
|
* some_widget_do_stuff_later (SomeWidget *self)
|
|
|
|
* {
|
|
|
|
* self->idle_id = g_idle_add (idle_callback, self);
|
|
|
|
* }
|
2006-06-05 19:34:53 +02:00
|
|
|
*
|
2006-06-02 04:36:30 +02:00
|
|
|
* static void
|
|
|
|
* some_widget_finalize (GObject *object)
|
|
|
|
* {
|
|
|
|
* SomeWidget *self = SOME_WIDGET (object);
|
2006-06-05 19:34:53 +02:00
|
|
|
*
|
2006-06-02 04:36:30 +02:00
|
|
|
* if (self->idle_id)
|
|
|
|
* g_source_remove (self->idle_id);
|
2006-06-05 19:34:53 +02:00
|
|
|
*
|
2006-06-02 04:36:30 +02:00
|
|
|
* G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
|
|
* }
|
2007-11-25 07:05:06 +01:00
|
|
|
* ]|
|
2006-06-02 04:36:30 +02:00
|
|
|
*
|
|
|
|
* This will fail in a multi-threaded application if the
|
|
|
|
* widget is destroyed before the idle handler fires due
|
|
|
|
* to the use after free in the callback. A solution, to
|
|
|
|
* this particular problem, is to check to if the source
|
|
|
|
* has already been destroy within the callback.
|
|
|
|
*
|
2007-11-25 07:05:06 +01:00
|
|
|
* |[
|
2006-06-02 04:36:30 +02:00
|
|
|
* static gboolean
|
|
|
|
* idle_callback (gpointer data)
|
|
|
|
* {
|
|
|
|
* SomeWidget *self = data;
|
|
|
|
*
|
|
|
|
* GDK_THREADS_ENTER ();
|
2006-06-02 04:50:38 +02:00
|
|
|
* if (!g_source_is_destroyed (g_main_current_source ()))
|
2006-06-02 04:36:30 +02:00
|
|
|
* {
|
|
|
|
* /<!-- -->* do stuff with self *<!-- -->/
|
|
|
|
* }
|
|
|
|
* GDK_THREADS_LEAVE ();
|
|
|
|
*
|
|
|
|
* return FALSE;
|
|
|
|
* }
|
2007-11-25 07:05:06 +01:00
|
|
|
* ]|
|
2006-06-02 04:36:30 +02:00
|
|
|
*
|
|
|
|
* Return value: %TRUE if the source has been destroyed
|
2006-08-16 05:33:57 +02:00
|
|
|
*
|
|
|
|
* Since: 2.12
|
2006-06-02 04:36:30 +02:00
|
|
|
*/
|
|
|
|
gboolean
|
|
|
|
g_source_is_destroyed (GSource *source)
|
|
|
|
{
|
|
|
|
return SOURCE_DESTROYED (source);
|
|
|
|
}
|
|
|
|
|
2004-03-19 21:25:03 +01:00
|
|
|
/* Temporarily remove all this source's file descriptors from the
|
|
|
|
* poll(), so that if data comes available for one of the file descriptors
|
|
|
|
* we don't continually spin in the poll()
|
|
|
|
*/
|
|
|
|
/* HOLDS: source->context's lock */
|
2004-05-10 21:21:28 +02:00
|
|
|
static void
|
2004-03-19 21:25:03 +01:00
|
|
|
block_source (GSource *source)
|
|
|
|
{
|
|
|
|
GSList *tmp_list;
|
|
|
|
|
|
|
|
g_return_if_fail (!SOURCE_BLOCKED (source));
|
|
|
|
|
|
|
|
tmp_list = source->poll_fds;
|
|
|
|
while (tmp_list)
|
|
|
|
{
|
|
|
|
g_main_context_remove_poll_unlocked (source->context, tmp_list->data);
|
|
|
|
tmp_list = tmp_list->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* HOLDS: source->context's lock */
|
2004-05-10 21:21:28 +02:00
|
|
|
static void
|
2004-03-19 21:25:03 +01:00
|
|
|
unblock_source (GSource *source)
|
|
|
|
{
|
|
|
|
GSList *tmp_list;
|
|
|
|
|
|
|
|
g_return_if_fail (!SOURCE_BLOCKED (source)); /* Source already unblocked */
|
|
|
|
g_return_if_fail (!SOURCE_DESTROYED (source));
|
|
|
|
|
|
|
|
tmp_list = source->poll_fds;
|
|
|
|
while (tmp_list)
|
|
|
|
{
|
|
|
|
g_main_context_add_poll_unlocked (source->context, source->priority, tmp_list->data);
|
|
|
|
tmp_list = tmp_list->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/* HOLDS: context's lock */
|
|
|
|
static void
|
|
|
|
g_main_dispatch (GMainContext *context)
|
|
|
|
{
|
2006-06-02 04:36:30 +02:00
|
|
|
GMainDispatch *current = get_dispatch ();
|
Changes for 64-bit cleanliness, loosely based on patch from Mark Murnane.
Wed Jun 20 12:00:54 2001 Owen Taylor <otaylor@redhat.com>
Changes for 64-bit cleanliness, loosely based on patch
from Mark Murnane.
* gconvert.c (g_convert/g_convert_with_fallback): Remove
workarounds for since-fixed GNU libc bugs. Minor
doc fix.
* gconvert.[ch]: Change gint to gsize/gssize as
appropriate.
* gconvert.c (g_locale/filename_to/from_utf8): Fix incorrect
computation of bytes_read / bytes_written.
* gfileutils.[ch] (g_file_get_contents): Make length
out parameter 'gsize *len'.
* ghook.c (g_hook_compare_ids): Don't compare a
and b as 'a - b'.
* gmacros.h (GSIZE_TO_POINTER): Add GPOINTER_TO_SIZE,
GSIZE_TO_POINTER.
* gmain.c (g_timeout_prepare): Rewrite to avoid
overflows. (Fixes bug when system clock skews
backwards more than 24 days.)
* gmarkup.[ch]: Make lengths passed to callbacks
gsize, length for g_markup_parse-context_parse(),
g_markup_escape_text() gssize.
* gmessages.[ch] (g_printf_string_upper_bound): Change
return value to gsize.
* gmessages.c (printf_string_upper_bound): Remove
a ridiculous use of 'inline' on a 300 line function.
* gstring.[ch]: Represent size of string as a gsize,
not gint. Make parameters to functions take gsize,
or gssize where -1 is allowed.
* gstring.c (g_string_erase): Make
g_string_erase (string, pos, -1) a synonym for
g_string_truncate for consistency with other G*
APIs.
* gstrfuncs.[ch]: Make all functions taking a string
length, take a gsize, or gssize if -1 is allowed.
(g_strstr_len, g_strrstr_len). Also fix some boundary
conditions in g_str[r]str[_len].
* gutf8.c tests/unicode-encoding.c: Make parameters that
are byte lengths gsize, gssize as appropriate. Make
character offsets, other counts, glong.
* gasyncqueue.c gcompletion.c
timeloop.c timeloop-basic.c gutils.c gspawn.c.
Small 64 bit cleanliness fixups.
* glist.c (g_list_sort2, g_list_sort_real): Fix functions
that should have been static.
* gdate.c (g_date_fill_parse_tokens): Fix extra
declaration that was shadowing another.
* tests/module-test.c: Include string.h
Mon Jun 18 15:43:29 2001 Owen Taylor <otaylor@redhat.com>
* gutf8.c (g_get_charset): Make argument
G_CONST_RETURN char **.
2001-06-23 15:55:09 +02:00
|
|
|
guint i;
|
2000-12-05 21:45:33 +01:00
|
|
|
|
|
|
|
for (i = 0; i < context->pending_dispatches->len; i++)
|
|
|
|
{
|
|
|
|
GSource *source = context->pending_dispatches->pdata[i];
|
|
|
|
|
|
|
|
context->pending_dispatches->pdata[i] = NULL;
|
|
|
|
g_assert (source);
|
|
|
|
|
|
|
|
source->flags &= ~G_SOURCE_READY;
|
|
|
|
|
|
|
|
if (!SOURCE_DESTROYED (source))
|
|
|
|
{
|
|
|
|
gboolean was_in_call;
|
|
|
|
gpointer user_data = NULL;
|
|
|
|
GSourceFunc callback = NULL;
|
|
|
|
GSourceCallbackFuncs *cb_funcs;
|
|
|
|
gpointer cb_data;
|
|
|
|
gboolean need_destroy;
|
|
|
|
|
|
|
|
gboolean (*dispatch) (GSource *,
|
|
|
|
GSourceFunc,
|
|
|
|
gpointer);
|
2007-03-15 09:47:28 +01:00
|
|
|
GSList current_source_link;
|
2000-12-05 21:45:33 +01:00
|
|
|
|
|
|
|
dispatch = source->source_funcs->dispatch;
|
|
|
|
cb_funcs = source->callback_funcs;
|
|
|
|
cb_data = source->callback_data;
|
|
|
|
|
|
|
|
if (cb_funcs)
|
|
|
|
cb_funcs->ref (cb_data);
|
|
|
|
|
2004-03-19 21:25:03 +01:00
|
|
|
if ((source->flags & G_SOURCE_CAN_RECURSE) == 0)
|
|
|
|
block_source (source);
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
was_in_call = source->flags & G_HOOK_FLAG_IN_CALL;
|
|
|
|
source->flags |= G_HOOK_FLAG_IN_CALL;
|
|
|
|
|
|
|
|
if (cb_funcs)
|
2001-09-04 00:12:51 +02:00
|
|
|
cb_funcs->get (cb_data, source, &callback, &user_data);
|
2000-12-05 21:45:33 +01:00
|
|
|
|
2001-09-19 22:39:21 +02:00
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
|
2006-06-02 04:36:30 +02:00
|
|
|
current->depth++;
|
2007-03-15 09:47:28 +01:00
|
|
|
/* The on-stack allocation of the GSList is unconventional, but
|
|
|
|
* we know that the lifetime of the link is bounded to this
|
|
|
|
* function as the link is kept in a thread specific list and
|
|
|
|
* not manipulated outside of this function and its descendants.
|
|
|
|
* Avoiding the overhead of a g_slist_alloc() is useful as many
|
|
|
|
* applications do little more than dispatch events.
|
|
|
|
*
|
|
|
|
* This is a performance hack - do not revert to g_slist_prepend()!
|
|
|
|
*/
|
|
|
|
current_source_link.data = source;
|
2008-05-02 13:22:10 +02:00
|
|
|
current_source_link.next = current->dispatching_sources;
|
|
|
|
current->dispatching_sources = ¤t_source_link;
|
2000-12-05 21:45:33 +01:00
|
|
|
need_destroy = ! dispatch (source,
|
|
|
|
callback,
|
|
|
|
user_data);
|
2008-05-02 13:22:10 +02:00
|
|
|
g_assert (current->dispatching_sources == ¤t_source_link);
|
|
|
|
current->dispatching_sources = current_source_link.next;
|
2006-06-02 04:36:30 +02:00
|
|
|
current->depth--;
|
2004-03-01 03:41:09 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
if (cb_funcs)
|
|
|
|
cb_funcs->unref (cb_data);
|
|
|
|
|
2005-11-07 21:15:48 +01:00
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
if (!was_in_call)
|
2000-12-05 21:45:33 +01:00
|
|
|
source->flags &= ~G_HOOK_FLAG_IN_CALL;
|
|
|
|
|
2004-03-19 21:25:03 +01:00
|
|
|
if ((source->flags & G_SOURCE_CAN_RECURSE) == 0 &&
|
|
|
|
!SOURCE_DESTROYED (source))
|
|
|
|
unblock_source (source);
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/* Note: this depends on the fact that we can't switch
|
|
|
|
* sources from one main context to another
|
|
|
|
*/
|
|
|
|
if (need_destroy && !SOURCE_DESTROYED (source))
|
|
|
|
{
|
|
|
|
g_assert (source->context == context);
|
|
|
|
g_source_destroy_internal (source, context, TRUE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
SOURCE_UNREF (source, context);
|
|
|
|
}
|
|
|
|
|
|
|
|
g_ptr_array_set_size (context->pending_dispatches, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Holds context's lock */
|
|
|
|
static inline GSource *
|
|
|
|
next_valid_source (GMainContext *context,
|
|
|
|
GSource *source)
|
|
|
|
{
|
|
|
|
GSource *new_source = source ? source->next : context->source_list;
|
|
|
|
|
|
|
|
while (new_source)
|
|
|
|
{
|
|
|
|
if (!SOURCE_DESTROYED (new_source))
|
|
|
|
{
|
|
|
|
new_source->ref_count++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
new_source = new_source->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (source)
|
|
|
|
SOURCE_UNREF (source, context);
|
|
|
|
|
|
|
|
return new_source;
|
|
|
|
}
|
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
/**
|
|
|
|
* g_main_context_acquire:
|
|
|
|
* @context: a #GMainContext
|
|
|
|
*
|
|
|
|
* Tries to become the owner of the specified context.
|
2007-11-09 04:33:35 +01:00
|
|
|
* If some other thread is the owner of the context,
|
2001-06-30 21:56:47 +02:00
|
|
|
* 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().
|
|
|
|
*
|
|
|
|
* Return value: %TRUE if the operation succeeded, and
|
|
|
|
* this thread is now the owner of @context.
|
|
|
|
**/
|
|
|
|
gboolean
|
|
|
|
g_main_context_acquire (GMainContext *context)
|
|
|
|
{
|
2001-10-31 14:49:53 +01:00
|
|
|
#ifdef G_THREADS_ENABLED
|
2001-06-30 21:56:47 +02:00
|
|
|
gboolean result = FALSE;
|
|
|
|
GThread *self = G_THREAD_SELF;
|
|
|
|
|
|
|
|
if (context == NULL)
|
|
|
|
context = g_main_context_default ();
|
|
|
|
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
if (!context->owner)
|
2001-07-17 10:49:23 +02:00
|
|
|
{
|
|
|
|
context->owner = self;
|
|
|
|
g_assert (context->owner_count == 0);
|
|
|
|
}
|
2001-06-30 21:56:47 +02:00
|
|
|
|
|
|
|
if (context->owner == self)
|
|
|
|
{
|
|
|
|
context->owner_count++;
|
|
|
|
result = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
return result;
|
2001-10-31 14:49:53 +01:00
|
|
|
#else /* !G_THREADS_ENABLED */
|
2001-06-30 21:56:47 +02:00
|
|
|
return TRUE;
|
2001-10-31 14:49:53 +01:00
|
|
|
#endif /* G_THREADS_ENABLED */
|
2001-06-30 21:56:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_main_context_release:
|
|
|
|
* @context: a #GMainContext
|
|
|
|
*
|
2001-12-16 20:31:36 +01:00
|
|
|
* Releases ownership of a context previously acquired by this thread
|
2001-06-30 21:56:47 +02:00
|
|
|
* with g_main_context_acquire(). If the context was acquired multiple
|
2007-11-09 04:29:51 +01:00
|
|
|
* times, the ownership will be released only when g_main_context_release()
|
2001-06-30 21:56:47 +02:00
|
|
|
* is called as many times as it was acquired.
|
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_main_context_release (GMainContext *context)
|
|
|
|
{
|
2001-10-31 14:49:53 +01:00
|
|
|
#ifdef G_THREADS_ENABLED
|
2001-06-30 21:56:47 +02:00
|
|
|
if (context == NULL)
|
|
|
|
context = g_main_context_default ();
|
|
|
|
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
context->owner_count--;
|
|
|
|
if (context->owner_count == 0)
|
|
|
|
{
|
|
|
|
context->owner = NULL;
|
|
|
|
|
|
|
|
if (context->waiters)
|
|
|
|
{
|
2001-07-17 10:49:23 +02:00
|
|
|
GMainWaiter *waiter = context->waiters->data;
|
|
|
|
gboolean loop_internal_waiter =
|
|
|
|
(waiter->mutex == g_static_mutex_get_mutex (&context->mutex));
|
2001-06-30 21:56:47 +02:00
|
|
|
context->waiters = g_slist_delete_link (context->waiters,
|
|
|
|
context->waiters);
|
2001-07-17 10:49:23 +02:00
|
|
|
if (!loop_internal_waiter)
|
|
|
|
g_mutex_lock (waiter->mutex);
|
|
|
|
|
|
|
|
g_cond_signal (waiter->cond);
|
|
|
|
|
|
|
|
if (!loop_internal_waiter)
|
|
|
|
g_mutex_unlock (waiter->mutex);
|
2001-06-30 21:56:47 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-07-17 10:49:23 +02:00
|
|
|
UNLOCK_CONTEXT (context);
|
2001-10-31 14:49:53 +01:00
|
|
|
#endif /* G_THREADS_ENABLED */
|
2001-06-30 21:56:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_main_context_wait:
|
|
|
|
* @context: a #GMainContext
|
|
|
|
* @cond: a condition variable
|
|
|
|
* @mutex: a mutex, currently held
|
|
|
|
*
|
|
|
|
* Tries to become the owner of the specified context,
|
2001-12-16 20:31:36 +01:00
|
|
|
* 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
|
2001-06-30 21:56:47 +02:00
|
|
|
* try again (once) to become the owner.
|
|
|
|
*
|
|
|
|
* Return value: %TRUE if the operation succeeded, and
|
|
|
|
* this thread is now the owner of @context.
|
|
|
|
**/
|
|
|
|
gboolean
|
|
|
|
g_main_context_wait (GMainContext *context,
|
|
|
|
GCond *cond,
|
|
|
|
GMutex *mutex)
|
|
|
|
{
|
2001-10-31 14:49:53 +01:00
|
|
|
#ifdef G_THREADS_ENABLED
|
2001-06-30 21:56:47 +02:00
|
|
|
gboolean result = FALSE;
|
|
|
|
GThread *self = G_THREAD_SELF;
|
|
|
|
gboolean loop_internal_waiter;
|
|
|
|
|
|
|
|
if (context == NULL)
|
|
|
|
context = g_main_context_default ();
|
|
|
|
|
|
|
|
loop_internal_waiter = (mutex == g_static_mutex_get_mutex (&context->mutex));
|
|
|
|
|
|
|
|
if (!loop_internal_waiter)
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
if (context->owner && context->owner != self)
|
|
|
|
{
|
|
|
|
GMainWaiter waiter;
|
|
|
|
|
|
|
|
waiter.cond = cond;
|
|
|
|
waiter.mutex = mutex;
|
|
|
|
|
|
|
|
context->waiters = g_slist_append (context->waiters, &waiter);
|
|
|
|
|
|
|
|
if (!loop_internal_waiter)
|
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
g_cond_wait (cond, mutex);
|
|
|
|
if (!loop_internal_waiter)
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
context->waiters = g_slist_remove (context->waiters, &waiter);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!context->owner)
|
2001-07-17 10:49:23 +02:00
|
|
|
{
|
|
|
|
context->owner = self;
|
|
|
|
g_assert (context->owner_count == 0);
|
|
|
|
}
|
2001-06-30 21:56:47 +02:00
|
|
|
|
|
|
|
if (context->owner == self)
|
|
|
|
{
|
|
|
|
context->owner_count++;
|
|
|
|
result = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!loop_internal_waiter)
|
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
return result;
|
2001-10-31 14:49:53 +01:00
|
|
|
#else /* !G_THREADS_ENABLED */
|
2001-06-30 21:56:47 +02:00
|
|
|
return TRUE;
|
2001-10-31 14:49:53 +01:00
|
|
|
#endif /* G_THREADS_ENABLED */
|
2001-06-30 21:56:47 +02:00
|
|
|
}
|
2000-12-05 21:45:33 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* g_main_context_prepare:
|
|
|
|
* @context: a #GMainContext
|
|
|
|
* @priority: location to store priority of highest priority
|
|
|
|
* source already ready.
|
|
|
|
*
|
|
|
|
* Prepares to poll sources within a main loop. The resulting information
|
|
|
|
* for polling is determined by calling g_main_context_query ().
|
|
|
|
*
|
|
|
|
* Return value: %TRUE if some source is ready to be dispatched
|
|
|
|
* prior to polling.
|
|
|
|
**/
|
|
|
|
gboolean
|
|
|
|
g_main_context_prepare (GMainContext *context,
|
|
|
|
gint *priority)
|
|
|
|
{
|
2002-02-06 01:37:38 +01:00
|
|
|
gint i;
|
2000-12-05 21:45:33 +01:00
|
|
|
gint n_ready = 0;
|
|
|
|
gint current_priority = G_MAXINT;
|
|
|
|
GSource *source;
|
|
|
|
|
|
|
|
if (context == NULL)
|
|
|
|
context = g_main_context_default ();
|
|
|
|
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
|
2010-10-22 18:40:08 +02:00
|
|
|
context->time_is_fresh = FALSE;
|
2010-11-03 03:03:08 +01:00
|
|
|
context->real_time_is_fresh = FALSE;
|
2000-12-05 21:45:33 +01:00
|
|
|
|
|
|
|
if (context->in_check_or_prepare)
|
|
|
|
{
|
|
|
|
g_warning ("g_main_context_prepare() called recursively from within a source's check() or "
|
|
|
|
"prepare() member.");
|
2001-05-27 20:28:58 +02:00
|
|
|
UNLOCK_CONTEXT (context);
|
2000-12-05 21:45:33 +01:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef G_THREADS_ENABLED
|
|
|
|
if (context->poll_waiting)
|
|
|
|
{
|
|
|
|
g_warning("g_main_context_prepare(): main loop already active in another thread");
|
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
context->poll_waiting = TRUE;
|
2000-12-08 04:39:47 +01:00
|
|
|
#endif /* G_THREADS_ENABLED */
|
2000-12-05 21:45:33 +01:00
|
|
|
|
|
|
|
#if 0
|
|
|
|
/* If recursing, finish up current dispatch, before starting over */
|
|
|
|
if (context->pending_dispatches)
|
|
|
|
{
|
|
|
|
if (dispatch)
|
|
|
|
g_main_dispatch (context, ¤t_time);
|
|
|
|
|
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* If recursing, clear list of pending dispatches */
|
2002-02-06 01:37:38 +01:00
|
|
|
|
|
|
|
for (i = 0; i < context->pending_dispatches->len; i++)
|
2002-02-06 20:45:56 +01:00
|
|
|
{
|
|
|
|
if (context->pending_dispatches->pdata[i])
|
|
|
|
SOURCE_UNREF ((GSource *)context->pending_dispatches->pdata[i], context);
|
|
|
|
}
|
2000-12-05 21:45:33 +01:00
|
|
|
g_ptr_array_set_size (context->pending_dispatches, 0);
|
|
|
|
|
|
|
|
/* Prepare all sources */
|
|
|
|
|
|
|
|
context->timeout = -1;
|
|
|
|
|
|
|
|
source = next_valid_source (context, NULL);
|
|
|
|
while (source)
|
|
|
|
{
|
|
|
|
gint source_timeout = -1;
|
|
|
|
|
|
|
|
if ((n_ready > 0) && (source->priority > current_priority))
|
|
|
|
{
|
|
|
|
SOURCE_UNREF (source, context);
|
|
|
|
break;
|
|
|
|
}
|
2004-03-19 21:25:03 +01:00
|
|
|
if (SOURCE_BLOCKED (source))
|
2000-12-05 21:45:33 +01:00
|
|
|
goto next;
|
|
|
|
|
|
|
|
if (!(source->flags & G_SOURCE_READY))
|
|
|
|
{
|
|
|
|
gboolean result;
|
|
|
|
gboolean (*prepare) (GSource *source,
|
|
|
|
gint *timeout);
|
|
|
|
|
|
|
|
prepare = source->source_funcs->prepare;
|
|
|
|
context->in_check_or_prepare++;
|
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
result = (*prepare) (source, &source_timeout);
|
|
|
|
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
context->in_check_or_prepare--;
|
|
|
|
|
|
|
|
if (result)
|
2010-11-06 15:11:15 +01:00
|
|
|
{
|
|
|
|
GSource *ready_source = source;
|
|
|
|
|
|
|
|
while (ready_source)
|
|
|
|
{
|
|
|
|
ready_source->flags |= G_SOURCE_READY;
|
|
|
|
ready_source = ready_source->priv ? ready_source->priv->parent_source : NULL;
|
|
|
|
}
|
|
|
|
}
|
2000-12-05 21:45:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (source->flags & G_SOURCE_READY)
|
|
|
|
{
|
|
|
|
n_ready++;
|
|
|
|
current_priority = source->priority;
|
|
|
|
context->timeout = 0;
|
|
|
|
}
|
1998-12-02 15:55:27 +01:00
|
|
|
|
|
|
|
if (source_timeout >= 0)
|
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
if (context->timeout < 0)
|
|
|
|
context->timeout = source_timeout;
|
1998-12-02 15:55:27 +01:00
|
|
|
else
|
2000-12-05 21:45:33 +01:00
|
|
|
context->timeout = MIN (context->timeout, source_timeout);
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
next:
|
|
|
|
source = next_valid_source (context, source);
|
|
|
|
}
|
|
|
|
|
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
if (priority)
|
|
|
|
*priority = current_priority;
|
|
|
|
|
|
|
|
return (n_ready > 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_main_context_query:
|
|
|
|
* @context: a #GMainContext
|
|
|
|
* @max_priority: maximum priority source to check
|
2002-11-08 19:47:56 +01:00
|
|
|
* @timeout_: location to store timeout to be used in polling
|
2000-12-05 21:45:33 +01:00
|
|
|
* @fds: location to store #GPollFD records that need to be polled.
|
|
|
|
* @n_fds: length of @fds.
|
|
|
|
*
|
|
|
|
* Determines information necessary to poll this main loop.
|
|
|
|
*
|
2001-08-20 03:37:50 +02:00
|
|
|
* Return value: 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.
|
2000-12-05 21:45:33 +01:00
|
|
|
**/
|
|
|
|
gint
|
|
|
|
g_main_context_query (GMainContext *context,
|
|
|
|
gint max_priority,
|
|
|
|
gint *timeout,
|
|
|
|
GPollFD *fds,
|
|
|
|
gint n_fds)
|
|
|
|
{
|
|
|
|
gint n_poll;
|
|
|
|
GPollRec *pollrec;
|
|
|
|
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
pollrec = context->poll_records;
|
|
|
|
n_poll = 0;
|
|
|
|
while (pollrec && max_priority >= pollrec->priority)
|
|
|
|
{
|
2008-09-09 08:04:21 +02:00
|
|
|
/* We need to include entries with fd->events == 0 in the array because
|
|
|
|
* otherwise if the application changes fd->events behind our back and
|
|
|
|
* makes it non-zero, we'll be out of sync when we check the fds[] array.
|
|
|
|
* (Changing fd->events after adding an FD wasn't an anticipated use of
|
|
|
|
* this API, but it occurs in practice.) */
|
|
|
|
if (n_poll < n_fds)
|
2000-12-05 21:45:33 +01:00
|
|
|
{
|
2008-09-09 08:04:21 +02:00
|
|
|
fds[n_poll].fd = pollrec->fd->fd;
|
|
|
|
/* In direct contradiction to the Unix98 spec, IRIX runs into
|
|
|
|
* difficulty if you pass in POLLERR, POLLHUP or POLLNVAL
|
|
|
|
* flags in the events field of the pollfd while it should
|
|
|
|
* just ignoring them. So we mask them out here.
|
|
|
|
*/
|
|
|
|
fds[n_poll].events = pollrec->fd->events & ~(G_IO_ERR|G_IO_HUP|G_IO_NVAL);
|
|
|
|
fds[n_poll].revents = 0;
|
2000-12-05 21:45:33 +01:00
|
|
|
}
|
2008-09-09 08:04:21 +02:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
pollrec = pollrec->next;
|
2008-09-09 08:04:21 +02:00
|
|
|
n_poll++;
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
2000-12-13 05:23:45 +01:00
|
|
|
#ifdef G_THREADS_ENABLED
|
2000-12-05 21:45:33 +01:00
|
|
|
context->poll_changed = FALSE;
|
2000-12-13 05:23:45 +01:00
|
|
|
#endif
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
if (timeout)
|
|
|
|
{
|
|
|
|
*timeout = context->timeout;
|
2004-04-22 16:32:58 +02:00
|
|
|
if (*timeout != 0)
|
2010-10-22 18:40:08 +02:00
|
|
|
{
|
|
|
|
context->time_is_fresh = FALSE;
|
2010-11-03 03:03:08 +01:00
|
|
|
context->real_time_is_fresh = FALSE;
|
2010-10-22 18:40:08 +02:00
|
|
|
}
|
2000-12-05 21:45:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
UNLOCK_CONTEXT (context);
|
1998-12-02 15:55:27 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
return n_poll;
|
|
|
|
}
|
1998-12-02 15:55:27 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_main_context_check:
|
|
|
|
* @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()
|
|
|
|
*
|
2001-12-16 20:31:36 +01:00
|
|
|
* Passes the results of polling back to the main loop.
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
|
|
|
* Return value: %TRUE if some sources are ready to be dispatched.
|
|
|
|
**/
|
|
|
|
gboolean
|
|
|
|
g_main_context_check (GMainContext *context,
|
|
|
|
gint max_priority,
|
|
|
|
GPollFD *fds,
|
|
|
|
gint n_fds)
|
|
|
|
{
|
|
|
|
GSource *source;
|
|
|
|
GPollRec *pollrec;
|
|
|
|
gint n_ready = 0;
|
|
|
|
gint i;
|
2008-09-09 08:04:21 +02:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
LOCK_CONTEXT (context);
|
1998-12-02 15:55:27 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
if (context->in_check_or_prepare)
|
|
|
|
{
|
|
|
|
g_warning ("g_main_context_check() called recursively from within a source's check() or "
|
|
|
|
"prepare() member.");
|
2001-05-27 20:28:58 +02:00
|
|
|
UNLOCK_CONTEXT (context);
|
2000-12-05 21:45:33 +01:00
|
|
|
return FALSE;
|
|
|
|
}
|
1998-12-02 15:55:27 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
#ifdef G_THREADS_ENABLED
|
|
|
|
if (!context->poll_waiting)
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
#ifndef G_OS_WIN32
|
2004-02-14 01:23:36 +01:00
|
|
|
gchar a;
|
|
|
|
read (context->wake_up_pipe[0], &a, 1);
|
2000-12-05 21:45:33 +01:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else
|
|
|
|
context->poll_waiting = FALSE;
|
1998-12-02 15:55:27 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/* If the set of poll file descriptors changed, bail out
|
|
|
|
* and let the main loop rerun
|
|
|
|
*/
|
|
|
|
if (context->poll_changed)
|
2001-05-27 20:28:58 +02:00
|
|
|
{
|
|
|
|
UNLOCK_CONTEXT (context);
|
Fix warnings from sparse. (#487491, Kjartan Maraas)
2007-10-21 Behdad Esfahbod <behdad@gnome.org>
* glib/gdate.c (g_date_strftime):
* glib/gmain.c (g_main_context_check):
* glib/gregex.c (g_match_info_fetch_all), (g_regex_split_full):
* glib/gthread.c (g_once_init_enter_impl), (g_once_init_leave):
* glib/gthread.h:
* glib/gutf8.c (g_utf16_to_utf8), (g_utf16_to_ucs4):
* tests/errorcheck-mutex-test.c (lock_locked_mutex),
(trylock_locked_mutex), (unlock_unlocked_mutex),
(free_locked_mutex), (wait_on_unlocked_mutex),
(wait_on_otherwise_locked_mutex), (timed_wait_on_unlocked_mutex),
(timed_wait_on_otherwise_locked_mutex):
Fix warnings from sparse. (#487491, Kjartan Maraas)
svn path=/trunk/; revision=5792
2007-10-21 19:01:29 +02:00
|
|
|
return FALSE;
|
2001-05-27 20:28:58 +02:00
|
|
|
}
|
2000-12-13 05:23:45 +01:00
|
|
|
#endif /* G_THREADS_ENABLED */
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
pollrec = context->poll_records;
|
|
|
|
i = 0;
|
|
|
|
while (i < n_fds)
|
|
|
|
{
|
|
|
|
if (pollrec->fd->events)
|
2008-09-09 08:04:21 +02:00
|
|
|
pollrec->fd->revents = fds[i].revents;
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
pollrec = pollrec->next;
|
2008-09-09 08:04:21 +02:00
|
|
|
i++;
|
2000-12-05 21:45:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
source = next_valid_source (context, NULL);
|
|
|
|
while (source)
|
|
|
|
{
|
|
|
|
if ((n_ready > 0) && (source->priority > max_priority))
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
SOURCE_UNREF (source, context);
|
|
|
|
break;
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
2004-03-19 21:25:03 +01:00
|
|
|
if (SOURCE_BLOCKED (source))
|
2000-12-05 21:45:33 +01:00
|
|
|
goto next;
|
1998-12-02 15:55:27 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
if (!(source->flags & G_SOURCE_READY))
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
gboolean result;
|
|
|
|
gboolean (*check) (GSource *source);
|
1999-02-02 02:04:41 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
check = source->source_funcs->check;
|
1999-02-02 02:04:41 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
context->in_check_or_prepare++;
|
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
result = (*check) (source);
|
|
|
|
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
context->in_check_or_prepare--;
|
|
|
|
|
|
|
|
if (result)
|
2010-11-06 15:11:15 +01:00
|
|
|
{
|
|
|
|
GSource *ready_source = source;
|
|
|
|
|
|
|
|
while (ready_source)
|
|
|
|
{
|
|
|
|
ready_source->flags |= G_SOURCE_READY;
|
|
|
|
ready_source = ready_source->priv ? ready_source->priv->parent_source : NULL;
|
|
|
|
}
|
|
|
|
}
|
1999-02-02 02:04:41 +01:00
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
if (source->flags & G_SOURCE_READY)
|
1999-02-02 02:04:41 +01:00
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
source->ref_count++;
|
|
|
|
g_ptr_array_add (context->pending_dispatches, source);
|
1998-12-18 03:23:33 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
n_ready++;
|
2002-02-13 23:37:26 +01:00
|
|
|
|
|
|
|
/* never dispatch sources with less priority than the first
|
|
|
|
* one we choose to dispatch
|
|
|
|
*/
|
|
|
|
max_priority = source->priority;
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
2000-12-05 21:45:33 +01:00
|
|
|
|
|
|
|
next:
|
|
|
|
source = next_valid_source (context, source);
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
return n_ready > 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_main_context_dispatch:
|
|
|
|
* @context: a #GMainContext
|
|
|
|
*
|
2001-12-16 20:31:36 +01:00
|
|
|
* Dispatches all pending sources.
|
2000-12-05 21:45:33 +01:00
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_main_context_dispatch (GMainContext *context)
|
|
|
|
{
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
if (context->pending_dispatches->len > 0)
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
g_main_dispatch (context);
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
1998-12-15 06:28:02 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
}
|
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
/* HOLDS context lock */
|
2000-12-05 21:45:33 +01:00
|
|
|
static gboolean
|
|
|
|
g_main_context_iterate (GMainContext *context,
|
|
|
|
gboolean block,
|
2001-06-30 21:56:47 +02:00
|
|
|
gboolean dispatch,
|
|
|
|
GThread *self)
|
2000-12-05 21:45:33 +01:00
|
|
|
{
|
|
|
|
gint max_priority;
|
|
|
|
gint timeout;
|
|
|
|
gboolean some_ready;
|
2001-06-30 21:56:47 +02:00
|
|
|
gint nfds, allocated_nfds;
|
|
|
|
GPollFD *fds = NULL;
|
2009-05-27 18:01:14 +02:00
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
UNLOCK_CONTEXT (context);
|
2000-12-05 21:45:33 +01:00
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
#ifdef G_THREADS_ENABLED
|
|
|
|
if (!g_main_context_acquire (context))
|
2000-12-05 21:45:33 +01:00
|
|
|
{
|
2001-06-30 21:56:47 +02:00
|
|
|
gboolean got_ownership;
|
2009-05-27 18:01:14 +02:00
|
|
|
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
g_return_val_if_fail (g_thread_supported (), FALSE);
|
2000-12-05 21:45:33 +01:00
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
if (!block)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (!context->cond)
|
|
|
|
context->cond = g_cond_new ();
|
2009-05-27 18:01:14 +02:00
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
got_ownership = g_main_context_wait (context,
|
|
|
|
context->cond,
|
|
|
|
g_static_mutex_get_mutex (&context->mutex));
|
|
|
|
|
|
|
|
if (!got_ownership)
|
2009-05-27 18:01:14 +02:00
|
|
|
return FALSE;
|
2001-06-30 21:56:47 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
LOCK_CONTEXT (context);
|
2001-07-11 22:08:50 +02:00
|
|
|
#endif /* G_THREADS_ENABLED */
|
2001-06-30 21:56:47 +02:00
|
|
|
|
|
|
|
if (!context->cached_poll_array)
|
|
|
|
{
|
|
|
|
context->cached_poll_array_size = context->n_poll_records;
|
|
|
|
context->cached_poll_array = g_new (GPollFD, context->n_poll_records);
|
|
|
|
}
|
2000-12-05 21:45:33 +01:00
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
allocated_nfds = context->cached_poll_array_size;
|
|
|
|
fds = context->cached_poll_array;
|
2000-12-05 21:45:33 +01:00
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
|
2003-11-01 14:57:48 +01:00
|
|
|
g_main_context_prepare (context, &max_priority);
|
2001-06-30 21:56:47 +02:00
|
|
|
|
|
|
|
while ((nfds = g_main_context_query (context, max_priority, &timeout, fds,
|
|
|
|
allocated_nfds)) > allocated_nfds)
|
|
|
|
{
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
g_free (fds);
|
|
|
|
context->cached_poll_array_size = allocated_nfds = nfds;
|
|
|
|
context->cached_poll_array = fds = g_new (GPollFD, nfds);
|
|
|
|
UNLOCK_CONTEXT (context);
|
2000-12-05 21:45:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!block)
|
|
|
|
timeout = 0;
|
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
g_main_context_poll (context, timeout, max_priority, fds, nfds);
|
2000-12-05 21:45:33 +01:00
|
|
|
|
2003-11-01 14:57:48 +01:00
|
|
|
some_ready = g_main_context_check (context, max_priority, fds, nfds);
|
2000-12-05 21:45:33 +01:00
|
|
|
|
|
|
|
if (dispatch)
|
|
|
|
g_main_context_dispatch (context);
|
2001-06-30 21:56:47 +02:00
|
|
|
|
|
|
|
#ifdef G_THREADS_ENABLED
|
|
|
|
g_main_context_release (context);
|
|
|
|
#endif /* G_THREADS_ENABLED */
|
|
|
|
|
|
|
|
LOCK_CONTEXT (context);
|
2000-12-05 21:45:33 +01:00
|
|
|
|
|
|
|
return some_ready;
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_main_context_pending:
|
|
|
|
* @context: a #GMainContext (if %NULL, the default context will be used)
|
|
|
|
*
|
2001-12-16 20:31:36 +01:00
|
|
|
* Checks if any sources have pending events for the given context.
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
|
|
|
* Return value: %TRUE if events are pending.
|
|
|
|
**/
|
1998-12-02 15:55:27 +01:00
|
|
|
gboolean
|
2000-12-05 21:45:33 +01:00
|
|
|
g_main_context_pending (GMainContext *context)
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
2001-06-30 21:56:47 +02:00
|
|
|
gboolean retval;
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
if (!context)
|
|
|
|
context = g_main_context_default();
|
2001-06-30 21:56:47 +02:00
|
|
|
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
retval = g_main_context_iterate (context, FALSE, FALSE, G_THREAD_SELF);
|
|
|
|
UNLOCK_CONTEXT (context);
|
2000-12-05 21:45:33 +01:00
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
return retval;
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_main_context_iteration:
|
|
|
|
* @context: a #GMainContext (if %NULL, the default context will be used)
|
|
|
|
* @may_block: whether the call may block.
|
|
|
|
*
|
2001-12-16 20:31:36 +01:00
|
|
|
* Runs a single iteration for the given main loop. This involves
|
2000-12-05 21:45:33 +01:00
|
|
|
* 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
|
2007-11-10 01:23:16 +01:00
|
|
|
* 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.
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
|
|
|
* Return value: %TRUE if events were dispatched.
|
|
|
|
**/
|
1998-12-02 15:55:27 +01:00
|
|
|
gboolean
|
2000-12-05 21:45:33 +01:00
|
|
|
g_main_context_iteration (GMainContext *context, gboolean may_block)
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
2001-06-30 21:56:47 +02:00
|
|
|
gboolean retval;
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
if (!context)
|
|
|
|
context = g_main_context_default();
|
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
retval = g_main_context_iterate (context, may_block, TRUE, G_THREAD_SELF);
|
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
return retval;
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_main_loop_new:
|
|
|
|
* @context: a #GMainContext (if %NULL, the default context will be used).
|
2001-12-16 20:31:36 +01:00
|
|
|
* @is_running: set to %TRUE to indicate that the loop is running. This
|
2002-02-03 02:04:32 +01:00
|
|
|
* is not very important since calling g_main_loop_run() will set this to
|
2001-12-16 20:31:36 +01:00
|
|
|
* %TRUE anyway.
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
2001-12-16 20:31:36 +01:00
|
|
|
* Creates a new #GMainLoop structure.
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
2001-12-16 20:31:36 +01:00
|
|
|
* Return value: a new #GMainLoop.
|
2000-12-05 21:45:33 +01:00
|
|
|
**/
|
|
|
|
GMainLoop *
|
|
|
|
g_main_loop_new (GMainContext *context,
|
|
|
|
gboolean is_running)
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
1998-12-18 03:23:33 +01:00
|
|
|
GMainLoop *loop;
|
2008-05-17 04:25:59 +02:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
if (!context)
|
|
|
|
context = g_main_context_default();
|
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
g_main_context_ref (context);
|
|
|
|
|
1998-12-18 03:23:33 +01:00
|
|
|
loop = g_new0 (GMainLoop, 1);
|
2000-12-05 21:45:33 +01:00
|
|
|
loop->context = context;
|
1998-12-18 03:23:33 +01:00
|
|
|
loop->is_running = is_running != FALSE;
|
2001-01-03 21:18:40 +01:00
|
|
|
loop->ref_count = 1;
|
|
|
|
|
1998-12-18 03:23:33 +01:00
|
|
|
return loop;
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
2001-01-03 21:18:40 +01:00
|
|
|
/**
|
|
|
|
* g_main_loop_ref:
|
|
|
|
* @loop: a #GMainLoop
|
|
|
|
*
|
2001-12-16 20:31:36 +01:00
|
|
|
* Increases the reference count on a #GMainLoop object by one.
|
2001-01-03 21:18:40 +01:00
|
|
|
*
|
|
|
|
* Return value: @loop
|
|
|
|
**/
|
|
|
|
GMainLoop *
|
|
|
|
g_main_loop_ref (GMainLoop *loop)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (loop != NULL, NULL);
|
2004-03-04 11:12:55 +01:00
|
|
|
g_return_val_if_fail (g_atomic_int_get (&loop->ref_count) > 0, NULL);
|
2001-01-03 21:18:40 +01:00
|
|
|
|
2004-03-04 11:12:55 +01:00
|
|
|
g_atomic_int_inc (&loop->ref_count);
|
2001-01-03 21:18:40 +01:00
|
|
|
|
|
|
|
return loop;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_main_loop_unref:
|
|
|
|
* @loop: a #GMainLoop
|
|
|
|
*
|
|
|
|
* Decreases the reference count on a #GMainLoop object by one. If
|
|
|
|
* the result is zero, free the loop and free all associated memory.
|
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_main_loop_unref (GMainLoop *loop)
|
|
|
|
{
|
|
|
|
g_return_if_fail (loop != NULL);
|
2004-03-04 11:12:55 +01:00
|
|
|
g_return_if_fail (g_atomic_int_get (&loop->ref_count) > 0);
|
2001-01-03 21:18:40 +01:00
|
|
|
|
2004-03-04 11:12:55 +01:00
|
|
|
if (!g_atomic_int_dec_and_test (&loop->ref_count))
|
|
|
|
return;
|
|
|
|
|
|
|
|
g_main_context_unref (loop->context);
|
|
|
|
g_free (loop);
|
2001-01-03 21:18:40 +01:00
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_main_loop_run:
|
|
|
|
* @loop: a #GMainLoop
|
|
|
|
*
|
2002-02-03 02:04:32 +01:00
|
|
|
* Runs a main loop until g_main_loop_quit() is called on the loop.
|
2000-12-05 21:45:33 +01:00
|
|
|
* If this is called for the thread of the loop's #GMainContext,
|
|
|
|
* it will process events from the loop, otherwise it will
|
|
|
|
* simply wait.
|
|
|
|
**/
|
1998-12-02 15:55:27 +01:00
|
|
|
void
|
2000-12-05 21:45:33 +01:00
|
|
|
g_main_loop_run (GMainLoop *loop)
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
2001-06-30 21:56:47 +02:00
|
|
|
GThread *self = G_THREAD_SELF;
|
|
|
|
|
1998-12-18 03:23:33 +01:00
|
|
|
g_return_if_fail (loop != NULL);
|
2004-03-04 11:12:55 +01:00
|
|
|
g_return_if_fail (g_atomic_int_get (&loop->ref_count) > 0);
|
1998-12-18 03:23:33 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
#ifdef G_THREADS_ENABLED
|
2001-06-30 21:56:47 +02:00
|
|
|
if (!g_main_context_acquire (loop->context))
|
2000-12-05 21:45:33 +01:00
|
|
|
{
|
2001-06-30 21:56:47 +02:00
|
|
|
gboolean got_ownership = FALSE;
|
2001-01-03 21:18:40 +01:00
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
/* Another thread owns this context */
|
2000-12-05 21:45:33 +01:00
|
|
|
if (!g_thread_supported ())
|
|
|
|
{
|
2002-02-03 02:05:55 +01:00
|
|
|
g_warning ("g_main_loop_run() was called from second thread but "
|
2000-12-05 21:45:33 +01:00
|
|
|
"g_thread_init() was never called.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
LOCK_CONTEXT (loop->context);
|
2000-12-05 21:45:33 +01:00
|
|
|
|
2004-03-04 11:12:55 +01:00
|
|
|
g_atomic_int_inc (&loop->ref_count);
|
2001-06-30 21:56:47 +02:00
|
|
|
|
|
|
|
if (!loop->is_running)
|
|
|
|
loop->is_running = TRUE;
|
|
|
|
|
|
|
|
if (!loop->context->cond)
|
|
|
|
loop->context->cond = g_cond_new ();
|
|
|
|
|
2003-04-07 19:50:53 +02:00
|
|
|
while (loop->is_running && !got_ownership)
|
2001-06-30 21:56:47 +02:00
|
|
|
got_ownership = g_main_context_wait (loop->context,
|
|
|
|
loop->context->cond,
|
|
|
|
g_static_mutex_get_mutex (&loop->context->mutex));
|
|
|
|
|
|
|
|
if (!loop->is_running)
|
2000-12-05 21:45:33 +01:00
|
|
|
{
|
2001-07-17 10:49:23 +02:00
|
|
|
UNLOCK_CONTEXT (loop->context);
|
2001-06-30 21:56:47 +02:00
|
|
|
if (got_ownership)
|
|
|
|
g_main_context_release (loop->context);
|
2001-07-17 10:49:23 +02:00
|
|
|
g_main_loop_unref (loop);
|
2001-06-30 21:56:47 +02:00
|
|
|
return;
|
2000-12-05 21:45:33 +01:00
|
|
|
}
|
2001-06-30 21:56:47 +02:00
|
|
|
|
|
|
|
g_assert (got_ownership);
|
2001-01-03 21:18:40 +01:00
|
|
|
}
|
2001-06-30 21:56:47 +02:00
|
|
|
else
|
|
|
|
LOCK_CONTEXT (loop->context);
|
|
|
|
#endif /* G_THREADS_ENABLED */
|
2001-01-03 21:18:40 +01:00
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
if (loop->context->in_check_or_prepare)
|
2001-01-03 21:18:40 +01:00
|
|
|
{
|
2002-02-03 02:05:55 +01:00
|
|
|
g_warning ("g_main_loop_run(): called recursively from within a source's "
|
2002-02-03 02:04:32 +01:00
|
|
|
"check() or prepare() member, iteration not possible.");
|
2001-06-30 21:56:47 +02:00
|
|
|
return;
|
1999-01-17 05:49:43 +01:00
|
|
|
}
|
2001-06-30 21:56:47 +02:00
|
|
|
|
2004-03-04 11:12:55 +01:00
|
|
|
g_atomic_int_inc (&loop->ref_count);
|
2001-06-30 21:56:47 +02:00
|
|
|
loop->is_running = TRUE;
|
|
|
|
while (loop->is_running)
|
|
|
|
g_main_context_iterate (loop->context, TRUE, TRUE, self);
|
|
|
|
|
2001-11-01 01:01:50 +01:00
|
|
|
UNLOCK_CONTEXT (loop->context);
|
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
#ifdef G_THREADS_ENABLED
|
|
|
|
g_main_context_release (loop->context);
|
|
|
|
#endif /* G_THREADS_ENABLED */
|
|
|
|
|
2001-11-01 01:01:50 +01:00
|
|
|
g_main_loop_unref (loop);
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_main_loop_quit:
|
|
|
|
* @loop: a #GMainLoop
|
|
|
|
*
|
|
|
|
* Stops a #GMainLoop from running. Any calls to g_main_loop_run()
|
2007-11-09 03:46:41 +01:00
|
|
|
* for the loop will return.
|
|
|
|
*
|
|
|
|
* Note that sources that have already been dispatched when
|
|
|
|
* g_main_loop_quit() is called will still be executed.
|
2000-12-05 21:45:33 +01:00
|
|
|
**/
|
1998-12-02 15:55:27 +01:00
|
|
|
void
|
2000-12-05 21:45:33 +01:00
|
|
|
g_main_loop_quit (GMainLoop *loop)
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
1998-12-18 03:23:33 +01:00
|
|
|
g_return_if_fail (loop != NULL);
|
2004-03-04 11:12:55 +01:00
|
|
|
g_return_if_fail (g_atomic_int_get (&loop->ref_count) > 0);
|
1998-12-18 03:23:33 +01:00
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
LOCK_CONTEXT (loop->context);
|
1998-12-18 03:23:33 +01:00
|
|
|
loop->is_running = FALSE;
|
2001-06-30 21:56:47 +02:00
|
|
|
g_main_context_wakeup_unlocked (loop->context);
|
2000-12-05 21:45:33 +01:00
|
|
|
|
2001-10-29 12:09:53 +01:00
|
|
|
#ifdef G_THREADS_ENABLED
|
2001-06-30 21:56:47 +02:00
|
|
|
if (loop->context->cond)
|
|
|
|
g_cond_broadcast (loop->context->cond);
|
2001-10-29 12:09:53 +01:00
|
|
|
#endif /* G_THREADS_ENABLED */
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
UNLOCK_CONTEXT (loop->context);
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_main_loop_is_running:
|
|
|
|
* @loop: a #GMainLoop.
|
|
|
|
*
|
2002-02-03 02:04:32 +01:00
|
|
|
* Checks to see if the main loop is currently being run via g_main_loop_run().
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
|
|
|
* Return value: %TRUE if the mainloop is currently being run.
|
|
|
|
**/
|
1998-12-18 03:23:33 +01:00
|
|
|
gboolean
|
2000-12-05 21:45:33 +01:00
|
|
|
g_main_loop_is_running (GMainLoop *loop)
|
1998-12-18 03:23:33 +01:00
|
|
|
{
|
|
|
|
g_return_val_if_fail (loop != NULL, FALSE);
|
2004-03-04 11:12:55 +01:00
|
|
|
g_return_val_if_fail (g_atomic_int_get (&loop->ref_count) > 0, FALSE);
|
1998-12-18 03:23:33 +01:00
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
return loop->is_running;
|
|
|
|
}
|
2000-12-05 21:45:33 +01:00
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
/**
|
|
|
|
* g_main_loop_get_context:
|
|
|
|
* @loop: a #GMainLoop.
|
|
|
|
*
|
|
|
|
* Returns the #GMainContext of @loop.
|
|
|
|
*
|
|
|
|
* Return value: the #GMainContext of @loop
|
|
|
|
**/
|
|
|
|
GMainContext *
|
|
|
|
g_main_loop_get_context (GMainLoop *loop)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (loop != NULL, NULL);
|
2004-03-04 11:12:55 +01:00
|
|
|
g_return_val_if_fail (g_atomic_int_get (&loop->ref_count) > 0, NULL);
|
2001-06-30 21:56:47 +02:00
|
|
|
|
|
|
|
return loop->context;
|
1998-12-18 03:23:33 +01:00
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/* HOLDS: context's lock */
|
1998-12-02 15:55:27 +01:00
|
|
|
static void
|
2000-12-05 21:45:33 +01:00
|
|
|
g_main_context_poll (GMainContext *context,
|
|
|
|
gint timeout,
|
|
|
|
gint priority,
|
|
|
|
GPollFD *fds,
|
|
|
|
gint n_fds)
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
1999-07-24 20:50:58 +02:00
|
|
|
#ifdef G_MAIN_POLL_DEBUG
|
|
|
|
GTimer *poll_timer;
|
1998-12-02 15:55:27 +01:00
|
|
|
GPollRec *pollrec;
|
|
|
|
gint i;
|
2000-12-05 21:45:33 +01:00
|
|
|
#endif
|
1999-07-24 20:50:58 +02:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
GPollFunc poll_func;
|
1998-12-15 06:28:02 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
if (n_fds || timeout != 0)
|
1999-07-24 20:50:58 +02:00
|
|
|
{
|
|
|
|
#ifdef G_MAIN_POLL_DEBUG
|
2008-09-25 21:59:49 +02:00
|
|
|
if (_g_main_poll_debug)
|
2008-05-17 04:25:59 +02:00
|
|
|
{
|
|
|
|
g_print ("polling context=%p n=%d timeout=%d\n",
|
|
|
|
context, n_fds, timeout);
|
|
|
|
poll_timer = g_timer_new ();
|
|
|
|
}
|
1999-07-24 20:50:58 +02:00
|
|
|
#endif
|
2000-12-05 21:45:33 +01:00
|
|
|
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
poll_func = context->poll_func;
|
1999-07-24 20:50:58 +02:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
if ((*poll_func) (fds, n_fds, timeout) < 0 && errno != EINTR)
|
2003-06-06 00:18:27 +02:00
|
|
|
{
|
|
|
|
#ifndef G_OS_WIN32
|
|
|
|
g_warning ("poll(2) failed due to: %s.",
|
|
|
|
g_strerror (errno));
|
|
|
|
#else
|
|
|
|
/* If g_poll () returns -1, it has already called g_warning() */
|
|
|
|
#endif
|
|
|
|
}
|
1999-07-24 20:50:58 +02:00
|
|
|
|
|
|
|
#ifdef G_MAIN_POLL_DEBUG
|
2008-09-25 21:59:49 +02:00
|
|
|
if (_g_main_poll_debug)
|
1999-07-24 20:50:58 +02:00
|
|
|
{
|
2008-05-17 04:25:59 +02:00
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
g_print ("g_main_poll(%d) timeout: %d - elapsed %12.10f seconds",
|
|
|
|
n_fds,
|
|
|
|
timeout,
|
|
|
|
g_timer_elapsed (poll_timer, NULL));
|
|
|
|
g_timer_destroy (poll_timer);
|
|
|
|
pollrec = context->poll_records;
|
|
|
|
|
|
|
|
while (pollrec != NULL)
|
1999-07-24 20:50:58 +02:00
|
|
|
{
|
2008-05-17 04:25:59 +02:00
|
|
|
i = 0;
|
|
|
|
while (i < n_fds)
|
1999-07-24 20:50:58 +02:00
|
|
|
{
|
2008-05-17 04:25:59 +02:00
|
|
|
if (fds[i].fd == pollrec->fd->fd &&
|
|
|
|
pollrec->fd->events &&
|
|
|
|
fds[i].revents)
|
|
|
|
{
|
2008-09-23 18:32:30 +02:00
|
|
|
g_print (" [" G_POLLFD_FORMAT " :", fds[i].fd);
|
2008-05-17 04:25:59 +02:00
|
|
|
if (fds[i].revents & G_IO_IN)
|
|
|
|
g_print ("i");
|
|
|
|
if (fds[i].revents & G_IO_OUT)
|
|
|
|
g_print ("o");
|
|
|
|
if (fds[i].revents & G_IO_PRI)
|
|
|
|
g_print ("p");
|
|
|
|
if (fds[i].revents & G_IO_ERR)
|
|
|
|
g_print ("e");
|
|
|
|
if (fds[i].revents & G_IO_HUP)
|
|
|
|
g_print ("h");
|
|
|
|
if (fds[i].revents & G_IO_NVAL)
|
|
|
|
g_print ("n");
|
|
|
|
g_print ("]");
|
|
|
|
}
|
|
|
|
i++;
|
1999-07-24 20:50:58 +02:00
|
|
|
}
|
2008-05-17 04:25:59 +02:00
|
|
|
pollrec = pollrec->next;
|
1999-07-24 20:50:58 +02:00
|
|
|
}
|
2008-05-17 04:25:59 +02:00
|
|
|
g_print ("\n");
|
|
|
|
|
|
|
|
UNLOCK_CONTEXT (context);
|
1999-07-24 20:50:58 +02:00
|
|
|
}
|
|
|
|
#endif
|
2000-12-05 21:45:33 +01:00
|
|
|
} /* if (n_fds || timeout != 0) */
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_main_context_add_poll:
|
|
|
|
* @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.
|
|
|
|
*
|
2001-12-16 20:31:36 +01:00
|
|
|
* Adds a file descriptor to the set of file descriptors polled for
|
|
|
|
* this context. This will very seldomly be used directly. Instead
|
2000-12-05 21:45:33 +01:00
|
|
|
* a typical event source will use g_source_add_poll() instead.
|
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_main_context_add_poll (GMainContext *context,
|
|
|
|
GPollFD *fd,
|
|
|
|
gint priority)
|
1998-12-15 06:28:02 +01:00
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
if (!context)
|
|
|
|
context = g_main_context_default ();
|
|
|
|
|
2004-03-04 11:12:55 +01:00
|
|
|
g_return_if_fail (g_atomic_int_get (&context->ref_count) > 0);
|
2001-06-30 21:56:47 +02:00
|
|
|
g_return_if_fail (fd);
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
g_main_context_add_poll_unlocked (context, priority, fd);
|
|
|
|
UNLOCK_CONTEXT (context);
|
1998-12-15 06:28:02 +01:00
|
|
|
}
|
|
|
|
|
1998-12-16 10:34:30 +01:00
|
|
|
/* HOLDS: main_loop_lock */
|
1998-12-15 06:28:02 +01:00
|
|
|
static void
|
2000-12-05 21:45:33 +01:00
|
|
|
g_main_context_add_poll_unlocked (GMainContext *context,
|
|
|
|
gint priority,
|
|
|
|
GPollFD *fd)
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
2005-11-01 19:10:31 +01:00
|
|
|
GPollRec *lastrec, *pollrec;
|
|
|
|
GPollRec *newrec = g_slice_new (GPollRec);
|
1998-12-15 06:28:02 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/* This file descriptor may be checked before we ever poll */
|
|
|
|
fd->revents = 0;
|
1998-12-02 15:55:27 +01:00
|
|
|
newrec->fd = fd;
|
|
|
|
newrec->priority = priority;
|
|
|
|
|
|
|
|
lastrec = NULL;
|
2000-12-05 21:45:33 +01:00
|
|
|
pollrec = context->poll_records;
|
1998-12-02 15:55:27 +01:00
|
|
|
while (pollrec && priority >= pollrec->priority)
|
|
|
|
{
|
|
|
|
lastrec = pollrec;
|
|
|
|
pollrec = pollrec->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (lastrec)
|
|
|
|
lastrec->next = newrec;
|
|
|
|
else
|
2000-12-05 21:45:33 +01:00
|
|
|
context->poll_records = newrec;
|
1998-12-02 15:55:27 +01:00
|
|
|
|
|
|
|
newrec->next = pollrec;
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
context->n_poll_records++;
|
1999-07-24 20:50:58 +02:00
|
|
|
|
|
|
|
#ifdef G_THREADS_ENABLED
|
2000-12-05 21:45:33 +01:00
|
|
|
context->poll_changed = TRUE;
|
1999-07-24 20:50:58 +02:00
|
|
|
|
|
|
|
/* Now wake up the main loop if it is waiting in the poll() */
|
2001-06-30 21:56:47 +02:00
|
|
|
g_main_context_wakeup_unlocked (context);
|
1999-07-24 20:50:58 +02:00
|
|
|
#endif
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_main_context_remove_poll:
|
|
|
|
* @context:a #GMainContext
|
|
|
|
* @fd: a #GPollFD descriptor previously added with g_main_context_add_poll()
|
|
|
|
*
|
2001-12-16 20:31:36 +01:00
|
|
|
* Removes file descriptor from the set of file descriptors to be
|
2000-12-05 21:45:33 +01:00
|
|
|
* polled for a particular context.
|
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_main_context_remove_poll (GMainContext *context,
|
|
|
|
GPollFD *fd)
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
if (!context)
|
|
|
|
context = g_main_context_default ();
|
|
|
|
|
2004-03-04 11:12:55 +01:00
|
|
|
g_return_if_fail (g_atomic_int_get (&context->ref_count) > 0);
|
2001-06-30 21:56:47 +02:00
|
|
|
g_return_if_fail (fd);
|
1998-12-02 15:55:27 +01:00
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
LOCK_CONTEXT (context);
|
2000-12-05 21:45:33 +01:00
|
|
|
g_main_context_remove_poll_unlocked (context, fd);
|
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
g_main_context_remove_poll_unlocked (GMainContext *context,
|
|
|
|
GPollFD *fd)
|
|
|
|
{
|
|
|
|
GPollRec *pollrec, *lastrec;
|
|
|
|
|
1998-12-02 15:55:27 +01:00
|
|
|
lastrec = NULL;
|
2000-12-05 21:45:33 +01:00
|
|
|
pollrec = context->poll_records;
|
1998-12-02 15:55:27 +01:00
|
|
|
|
|
|
|
while (pollrec)
|
|
|
|
{
|
|
|
|
if (pollrec->fd == fd)
|
|
|
|
{
|
|
|
|
if (lastrec != NULL)
|
|
|
|
lastrec->next = pollrec->next;
|
|
|
|
else
|
2000-12-05 21:45:33 +01:00
|
|
|
context->poll_records = pollrec->next;
|
1998-12-02 15:55:27 +01:00
|
|
|
|
2005-11-01 19:10:31 +01:00
|
|
|
g_slice_free (GPollRec, pollrec);
|
1998-12-15 06:28:02 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
context->n_poll_records--;
|
1998-12-15 06:28:02 +01:00
|
|
|
break;
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
lastrec = pollrec;
|
|
|
|
pollrec = pollrec->next;
|
|
|
|
}
|
|
|
|
|
1999-07-24 20:50:58 +02:00
|
|
|
#ifdef G_THREADS_ENABLED
|
2000-12-05 21:45:33 +01:00
|
|
|
context->poll_changed = TRUE;
|
1999-07-24 20:50:58 +02:00
|
|
|
|
|
|
|
/* Now wake up the main loop if it is waiting in the poll() */
|
2001-06-30 21:56:47 +02:00
|
|
|
g_main_context_wakeup_unlocked (context);
|
1999-07-24 20:50:58 +02:00
|
|
|
#endif
|
2000-12-05 21:45:33 +01:00
|
|
|
}
|
1999-07-24 20:50:58 +02:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_source_get_current_time:
|
|
|
|
* @source: a #GSource
|
|
|
|
* @timeval: #GTimeVal structure in which to store current time.
|
|
|
|
*
|
|
|
|
* Gets the "current time" to be used when checking
|
|
|
|
* this source. The advantage of calling this function over
|
|
|
|
* calling g_get_current_time() directly is that when
|
|
|
|
* checking multiple sources, GLib can cache a single value
|
|
|
|
* instead of having to repeatedly get the system time.
|
2010-10-22 19:20:14 +02:00
|
|
|
*
|
|
|
|
* Deprecated: 2.28: use g_source_get_time() instead
|
2000-12-05 21:45:33 +01:00
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_source_get_current_time (GSource *source,
|
|
|
|
GTimeVal *timeval)
|
|
|
|
{
|
|
|
|
GMainContext *context;
|
|
|
|
|
|
|
|
g_return_if_fail (source->context != NULL);
|
|
|
|
|
|
|
|
context = source->context;
|
|
|
|
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
|
2010-11-03 03:03:08 +01:00
|
|
|
if (!context->real_time_is_fresh)
|
2000-12-05 21:45:33 +01:00
|
|
|
{
|
2010-11-03 03:03:08 +01:00
|
|
|
context->real_time = g_get_real_time ();
|
|
|
|
context->real_time_is_fresh = TRUE;
|
2000-12-05 21:45:33 +01:00
|
|
|
}
|
|
|
|
|
2010-11-03 03:03:08 +01:00
|
|
|
timeval->tv_sec = context->real_time / 1000000;
|
|
|
|
timeval->tv_usec = context->real_time % 1000000;
|
2000-12-05 21:45:33 +01:00
|
|
|
|
|
|
|
UNLOCK_CONTEXT (context);
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
2010-10-22 18:40:08 +02:00
|
|
|
/**
|
|
|
|
* g_source_get_time:
|
|
|
|
* @source: a #GSource
|
|
|
|
*
|
|
|
|
* Gets the time to be used when checking this source. The advantage of
|
|
|
|
* calling this function over calling g_get_monotonic_time() directly is
|
|
|
|
* that when checking multiple sources, GLib can cache a single value
|
|
|
|
* instead of having to repeatedly get the system monotonic time.
|
|
|
|
*
|
|
|
|
* The time here is the system monotonic time, if available, or some
|
|
|
|
* other reasonable alternative otherwise. See g_get_monotonic_time().
|
|
|
|
*
|
2010-11-01 20:46:35 +01:00
|
|
|
* Returns: the monotonic time in microseconds
|
|
|
|
*
|
2010-10-22 18:40:08 +02:00
|
|
|
* Since: 2.28
|
|
|
|
**/
|
2010-11-01 20:46:35 +01:00
|
|
|
gint64
|
|
|
|
g_source_get_time (GSource *source)
|
2010-10-22 18:40:08 +02:00
|
|
|
{
|
|
|
|
GMainContext *context;
|
2010-11-01 20:46:35 +01:00
|
|
|
gint64 result;
|
|
|
|
|
|
|
|
g_return_val_if_fail (source->context != NULL, 0);
|
|
|
|
|
2010-10-22 18:40:08 +02:00
|
|
|
context = source->context;
|
|
|
|
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
|
|
|
|
if (!context->time_is_fresh)
|
|
|
|
{
|
2010-11-01 20:46:35 +01:00
|
|
|
context->time = g_get_monotonic_time ();
|
2010-10-22 18:40:08 +02:00
|
|
|
context->time_is_fresh = TRUE;
|
|
|
|
}
|
2010-11-01 20:46:35 +01:00
|
|
|
|
|
|
|
result = context->time;
|
|
|
|
|
2010-10-22 18:40:08 +02:00
|
|
|
UNLOCK_CONTEXT (context);
|
2010-11-01 20:46:35 +01:00
|
|
|
|
|
|
|
return result;
|
2010-10-22 18:40:08 +02:00
|
|
|
}
|
2010-11-01 20:46:35 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_main_context_set_poll_func:
|
|
|
|
* @context: a #GMainContext
|
|
|
|
* @func: the function to call to poll all file descriptors
|
|
|
|
*
|
|
|
|
* Sets the function to use to handle polling of file descriptors. It
|
2003-07-25 23:32:47 +02:00
|
|
|
* will be used instead of the poll() system call
|
2001-12-16 20:31:36 +01:00
|
|
|
* (or GLib's replacement function, which is used where
|
2003-07-25 23:32:47 +02:00
|
|
|
* poll() isn't available).
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
|
|
|
* This function could possibly be used to integrate the GLib event
|
|
|
|
* loop with an external event loop.
|
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_main_context_set_poll_func (GMainContext *context,
|
|
|
|
GPollFunc func)
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
if (!context)
|
|
|
|
context = g_main_context_default ();
|
|
|
|
|
2004-03-04 11:12:55 +01:00
|
|
|
g_return_if_fail (g_atomic_int_get (&context->ref_count) > 0);
|
2001-06-30 21:56:47 +02:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
|
1998-12-02 15:55:27 +01:00
|
|
|
if (func)
|
2000-12-05 21:45:33 +01:00
|
|
|
context->poll_func = func;
|
1998-12-02 15:55:27 +01:00
|
|
|
else
|
2008-09-23 18:32:30 +02:00
|
|
|
context->poll_func = g_poll;
|
2000-12-05 21:45:33 +01:00
|
|
|
|
|
|
|
UNLOCK_CONTEXT (context);
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_main_context_get_poll_func:
|
|
|
|
* @context: a #GMainContext
|
|
|
|
*
|
2001-12-16 20:31:36 +01:00
|
|
|
* Gets the poll function set by g_main_context_set_poll_func().
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
|
|
|
* Return value: the poll function
|
|
|
|
**/
|
Finally, a new and improved IO Channel and condition watch implementation
2000-07-30 Tor Lillqvist <tml@iki.fi>
Finally, a new and improved IO Channel and condition watch
implementation for Win32. Based on code provided by Craig Setera.
When watching file descriptors, for which there is no select()
like functionality on Win32 that would work on all Win32 platforms
for all types of file descriptors (including anonymous pipes), we
start a new thread that blocks while trying to read from the file
descriptor. When the read returns, a Win32 Event is signalled that
the polling routine eventually notices. Meanwhile, the data being
read is stored in a circular buffer, from where the IO channel's
read() method picks it up.
If the buffer fills up the reading thread has to wait for space
becoming available. For this another Win32 Event is used. The IO
Channel's read() method signals this when it has read some data
out of the buffer.
The separate reader thread(s), and the circular buffer(s) with
associated events mean lots of possibilities for fun parallellism
errors. But it seems to work OK, i.e. GIMP runs.
* gmain.c: Small changes to the Win32 polling function.
(g_main_win32_get_poll_func): New function. Perhaps it would be a
good idea to provide this on all platforms.
* giowin32.c: The bulk of the new implementation.
(g_io_channel_win32_wait_for_condition): New function. To be used
where on Unix one does a select() on the channel's fd, like
libgimp's gimp_extension_process(). Could be provided on all
platforms.
* glib.h: Update documentation for IO Channels on Win32. Remove
the declarations for the as of now obsolete old functions related
to IO Channels for pipes with "wakeup" messages.
* glib.def: Some new functions.
* tests/gio-test.c: New file, to test GIOChannel and main loop.
* tests/Makefile.am
* tests/makefile.mingw.in: Add it.
2000-07-29 22:59:07 +02:00
|
|
|
GPollFunc
|
2000-12-05 21:45:33 +01:00
|
|
|
g_main_context_get_poll_func (GMainContext *context)
|
Finally, a new and improved IO Channel and condition watch implementation
2000-07-30 Tor Lillqvist <tml@iki.fi>
Finally, a new and improved IO Channel and condition watch
implementation for Win32. Based on code provided by Craig Setera.
When watching file descriptors, for which there is no select()
like functionality on Win32 that would work on all Win32 platforms
for all types of file descriptors (including anonymous pipes), we
start a new thread that blocks while trying to read from the file
descriptor. When the read returns, a Win32 Event is signalled that
the polling routine eventually notices. Meanwhile, the data being
read is stored in a circular buffer, from where the IO channel's
read() method picks it up.
If the buffer fills up the reading thread has to wait for space
becoming available. For this another Win32 Event is used. The IO
Channel's read() method signals this when it has read some data
out of the buffer.
The separate reader thread(s), and the circular buffer(s) with
associated events mean lots of possibilities for fun parallellism
errors. But it seems to work OK, i.e. GIMP runs.
* gmain.c: Small changes to the Win32 polling function.
(g_main_win32_get_poll_func): New function. Perhaps it would be a
good idea to provide this on all platforms.
* giowin32.c: The bulk of the new implementation.
(g_io_channel_win32_wait_for_condition): New function. To be used
where on Unix one does a select() on the channel's fd, like
libgimp's gimp_extension_process(). Could be provided on all
platforms.
* glib.h: Update documentation for IO Channels on Win32. Remove
the declarations for the as of now obsolete old functions related
to IO Channels for pipes with "wakeup" messages.
* glib.def: Some new functions.
* tests/gio-test.c: New file, to test GIOChannel and main loop.
* tests/Makefile.am
* tests/makefile.mingw.in: Add it.
2000-07-29 22:59:07 +02:00
|
|
|
{
|
Warn if no callback. Call callback correctly. (g_io_win32_create_watch):
2000-12-14 Tor Lillqvist <tml@iki.fi>
* giowin32.c (g_io_win32_dispatch): Warn if no callback. Call
callback correctly.
(g_io_win32_create_watch): Fix typo.
(g_io_win32_fd_create_watch): Ditto.
(g_io_channel_unix_new): If it is a file descriptor (i.e., a Unix
fd lookalike provided by the C library), call
g_io_channel_win32_new_fd(). If it is a socket (from WinSock),
call g_io_cahnnel_win32_new_stream_socket(). Hopefully sockets and
fds don't overlap. TODO: Implement also datagram sockets.
(g_io_channel_win32_poll): Call g_main_context_get_poll_func().
* gcompletion.h: Include <unistd.h> only on Unix. Is this
inclusion really needed here? OTOH, do include <stddef.h>, for
size_t.
* gmessages.c: (Win32) Don't define a function called "write" that
might clash with the prototype from <io.h>, use a #define.
* glib.def: Update.
* gmain.c (g_source_add_poll): Don't return a value from void
function.
(g_main_context_get_poll_func): Compile also for non-Win32, as
presumably was intended. The result var is a GPollFunc, not a
GPollFunc*. Return the result!
gobject:
2000-12-14 Tor Lillqvist <tml@iki.fi>
* makefile.mingw.in: Update, include parts from Makefile.am to
build gmarshal.[ch]. Some day, we won't need these separate
makefiles for Win32 compilation. I hope.
* makefile.msc.in: Update. No use trying to build gmarshal.[ch]
here, it would require Unixish tools. MSVC users building from CVS
sources are out of luck.
* gobject.def: Update.
2000-12-14 22:02:20 +01:00
|
|
|
GPollFunc result;
|
2000-12-05 21:45:33 +01:00
|
|
|
|
|
|
|
if (!context)
|
|
|
|
context = g_main_context_default ();
|
|
|
|
|
2004-03-04 11:12:55 +01:00
|
|
|
g_return_val_if_fail (g_atomic_int_get (&context->ref_count) > 0, NULL);
|
2001-06-30 21:56:47 +02:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
result = context->poll_func;
|
|
|
|
UNLOCK_CONTEXT (context);
|
Finally, a new and improved IO Channel and condition watch implementation
2000-07-30 Tor Lillqvist <tml@iki.fi>
Finally, a new and improved IO Channel and condition watch
implementation for Win32. Based on code provided by Craig Setera.
When watching file descriptors, for which there is no select()
like functionality on Win32 that would work on all Win32 platforms
for all types of file descriptors (including anonymous pipes), we
start a new thread that blocks while trying to read from the file
descriptor. When the read returns, a Win32 Event is signalled that
the polling routine eventually notices. Meanwhile, the data being
read is stored in a circular buffer, from where the IO channel's
read() method picks it up.
If the buffer fills up the reading thread has to wait for space
becoming available. For this another Win32 Event is used. The IO
Channel's read() method signals this when it has read some data
out of the buffer.
The separate reader thread(s), and the circular buffer(s) with
associated events mean lots of possibilities for fun parallellism
errors. But it seems to work OK, i.e. GIMP runs.
* gmain.c: Small changes to the Win32 polling function.
(g_main_win32_get_poll_func): New function. Perhaps it would be a
good idea to provide this on all platforms.
* giowin32.c: The bulk of the new implementation.
(g_io_channel_win32_wait_for_condition): New function. To be used
where on Unix one does a select() on the channel's fd, like
libgimp's gimp_extension_process(). Could be provided on all
platforms.
* glib.h: Update documentation for IO Channels on Win32. Remove
the declarations for the as of now obsolete old functions related
to IO Channels for pipes with "wakeup" messages.
* glib.def: Some new functions.
* tests/gio-test.c: New file, to test GIOChannel and main loop.
* tests/Makefile.am
* tests/makefile.mingw.in: Add it.
2000-07-29 22:59:07 +02:00
|
|
|
|
Warn if no callback. Call callback correctly. (g_io_win32_create_watch):
2000-12-14 Tor Lillqvist <tml@iki.fi>
* giowin32.c (g_io_win32_dispatch): Warn if no callback. Call
callback correctly.
(g_io_win32_create_watch): Fix typo.
(g_io_win32_fd_create_watch): Ditto.
(g_io_channel_unix_new): If it is a file descriptor (i.e., a Unix
fd lookalike provided by the C library), call
g_io_channel_win32_new_fd(). If it is a socket (from WinSock),
call g_io_cahnnel_win32_new_stream_socket(). Hopefully sockets and
fds don't overlap. TODO: Implement also datagram sockets.
(g_io_channel_win32_poll): Call g_main_context_get_poll_func().
* gcompletion.h: Include <unistd.h> only on Unix. Is this
inclusion really needed here? OTOH, do include <stddef.h>, for
size_t.
* gmessages.c: (Win32) Don't define a function called "write" that
might clash with the prototype from <io.h>, use a #define.
* glib.def: Update.
* gmain.c (g_source_add_poll): Don't return a value from void
function.
(g_main_context_get_poll_func): Compile also for non-Win32, as
presumably was intended. The result var is a GPollFunc, not a
GPollFunc*. Return the result!
gobject:
2000-12-14 Tor Lillqvist <tml@iki.fi>
* makefile.mingw.in: Update, include parts from Makefile.am to
build gmarshal.[ch]. Some day, we won't need these separate
makefiles for Win32 compilation. I hope.
* makefile.msc.in: Update. No use trying to build gmarshal.[ch]
here, it would require Unixish tools. MSVC users building from CVS
sources are out of luck.
* gobject.def: Update.
2000-12-14 22:02:20 +01:00
|
|
|
return result;
|
|
|
|
}
|
Finally, a new and improved IO Channel and condition watch implementation
2000-07-30 Tor Lillqvist <tml@iki.fi>
Finally, a new and improved IO Channel and condition watch
implementation for Win32. Based on code provided by Craig Setera.
When watching file descriptors, for which there is no select()
like functionality on Win32 that would work on all Win32 platforms
for all types of file descriptors (including anonymous pipes), we
start a new thread that blocks while trying to read from the file
descriptor. When the read returns, a Win32 Event is signalled that
the polling routine eventually notices. Meanwhile, the data being
read is stored in a circular buffer, from where the IO channel's
read() method picks it up.
If the buffer fills up the reading thread has to wait for space
becoming available. For this another Win32 Event is used. The IO
Channel's read() method signals this when it has read some data
out of the buffer.
The separate reader thread(s), and the circular buffer(s) with
associated events mean lots of possibilities for fun parallellism
errors. But it seems to work OK, i.e. GIMP runs.
* gmain.c: Small changes to the Win32 polling function.
(g_main_win32_get_poll_func): New function. Perhaps it would be a
good idea to provide this on all platforms.
* giowin32.c: The bulk of the new implementation.
(g_io_channel_win32_wait_for_condition): New function. To be used
where on Unix one does a select() on the channel's fd, like
libgimp's gimp_extension_process(). Could be provided on all
platforms.
* glib.h: Update documentation for IO Channels on Win32. Remove
the declarations for the as of now obsolete old functions related
to IO Channels for pipes with "wakeup" messages.
* glib.def: Some new functions.
* tests/gio-test.c: New file, to test GIOChannel and main loop.
* tests/Makefile.am
* tests/makefile.mingw.in: Add it.
2000-07-29 22:59:07 +02:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/* HOLDS: context's lock */
|
1999-07-24 20:50:58 +02:00
|
|
|
/* Wake the main loop up from a poll() */
|
|
|
|
static void
|
2001-06-30 21:56:47 +02:00
|
|
|
g_main_context_wakeup_unlocked (GMainContext *context)
|
1999-07-24 20:50:58 +02:00
|
|
|
{
|
|
|
|
#ifdef G_THREADS_ENABLED
|
2000-12-05 21:45:33 +01:00
|
|
|
if (g_thread_supported() && context->poll_waiting)
|
1999-07-24 20:50:58 +02:00
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
context->poll_waiting = FALSE;
|
1999-10-04 04:32:50 +02:00
|
|
|
#ifndef G_OS_WIN32
|
2000-12-05 21:45:33 +01:00
|
|
|
write (context->wake_up_pipe[1], "A", 1);
|
1999-07-24 20:50:58 +02:00
|
|
|
#else
|
Warn if no callback. Call callback correctly. (g_io_win32_create_watch):
2000-12-14 Tor Lillqvist <tml@iki.fi>
* giowin32.c (g_io_win32_dispatch): Warn if no callback. Call
callback correctly.
(g_io_win32_create_watch): Fix typo.
(g_io_win32_fd_create_watch): Ditto.
(g_io_channel_unix_new): If it is a file descriptor (i.e., a Unix
fd lookalike provided by the C library), call
g_io_channel_win32_new_fd(). If it is a socket (from WinSock),
call g_io_cahnnel_win32_new_stream_socket(). Hopefully sockets and
fds don't overlap. TODO: Implement also datagram sockets.
(g_io_channel_win32_poll): Call g_main_context_get_poll_func().
* gcompletion.h: Include <unistd.h> only on Unix. Is this
inclusion really needed here? OTOH, do include <stddef.h>, for
size_t.
* gmessages.c: (Win32) Don't define a function called "write" that
might clash with the prototype from <io.h>, use a #define.
* glib.def: Update.
* gmain.c (g_source_add_poll): Don't return a value from void
function.
(g_main_context_get_poll_func): Compile also for non-Win32, as
presumably was intended. The result var is a GPollFunc, not a
GPollFunc*. Return the result!
gobject:
2000-12-14 Tor Lillqvist <tml@iki.fi>
* makefile.mingw.in: Update, include parts from Makefile.am to
build gmarshal.[ch]. Some day, we won't need these separate
makefiles for Win32 compilation. I hope.
* makefile.msc.in: Update. No use trying to build gmarshal.[ch]
here, it would require Unixish tools. MSVC users building from CVS
sources are out of luck.
* gobject.def: Update.
2000-12-14 22:02:20 +01:00
|
|
|
ReleaseSemaphore (context->wake_up_semaphore, 1, NULL);
|
1999-07-24 20:50:58 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2001-06-30 21:56:47 +02:00
|
|
|
/**
|
|
|
|
* g_main_context_wakeup:
|
|
|
|
* @context: a #GMainContext
|
|
|
|
*
|
2003-07-25 23:32:47 +02:00
|
|
|
* If @context is currently waiting in a poll(), interrupt
|
|
|
|
* the poll(), and continue the iteration process.
|
2001-06-30 21:56:47 +02:00
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_main_context_wakeup (GMainContext *context)
|
|
|
|
{
|
|
|
|
if (!context)
|
|
|
|
context = g_main_context_default ();
|
|
|
|
|
2004-03-04 11:12:55 +01:00
|
|
|
g_return_if_fail (g_atomic_int_get (&context->ref_count) > 0);
|
2001-06-30 21:56:47 +02:00
|
|
|
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
g_main_context_wakeup_unlocked (context);
|
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
}
|
|
|
|
|
2006-01-03 11:24:07 +01:00
|
|
|
/**
|
|
|
|
* g_main_context_is_owner:
|
|
|
|
* @context: a #GMainContext
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
2006-01-03 13:01:05 +01:00
|
|
|
* Returns: %TRUE if current thread is owner of @context.
|
|
|
|
*
|
|
|
|
* Since: 2.10
|
2006-01-03 11:24:07 +01:00
|
|
|
**/
|
|
|
|
gboolean
|
|
|
|
g_main_context_is_owner (GMainContext *context)
|
|
|
|
{
|
|
|
|
gboolean is_owner;
|
|
|
|
|
|
|
|
if (!context)
|
|
|
|
context = g_main_context_default ();
|
|
|
|
|
|
|
|
#ifdef G_THREADS_ENABLED
|
|
|
|
LOCK_CONTEXT (context);
|
|
|
|
is_owner = context->owner == G_THREAD_SELF;
|
|
|
|
UNLOCK_CONTEXT (context);
|
|
|
|
#else
|
|
|
|
is_owner = TRUE;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return is_owner;
|
|
|
|
}
|
|
|
|
|
1998-12-02 15:55:27 +01:00
|
|
|
/* Timeouts */
|
|
|
|
|
2000-03-01 10:44:10 +01:00
|
|
|
static void
|
2000-12-05 21:45:33 +01:00
|
|
|
g_timeout_set_expiration (GTimeoutSource *timeout_source,
|
2010-11-01 20:46:35 +01:00
|
|
|
gint64 current_time)
|
2000-03-01 10:44:10 +01:00
|
|
|
{
|
2011-02-15 05:21:14 +01:00
|
|
|
timeout_source->expiration = current_time +
|
|
|
|
(guint64) timeout_source->interval * 1000;
|
2000-03-01 10:44:10 +01:00
|
|
|
|
2010-11-01 18:31:37 +01:00
|
|
|
if (timeout_source->seconds)
|
2006-09-10 07:44:46 +02:00
|
|
|
{
|
2011-03-07 04:37:01 +01:00
|
|
|
gint64 remainder;
|
2010-11-01 18:31:37 +01:00
|
|
|
static gint timer_perturb = -1;
|
2006-09-10 07:44:46 +02:00
|
|
|
|
2010-11-01 18:31:37 +01:00
|
|
|
if (timer_perturb == -1)
|
2007-07-22 02:34:21 +02:00
|
|
|
{
|
2010-11-01 18:31:37 +01:00
|
|
|
/*
|
|
|
|
* we want a per machine/session unique 'random' value; try the dbus
|
|
|
|
* address first, that has a UUID in it. If there is no dbus, use the
|
|
|
|
* hostname for hashing.
|
|
|
|
*/
|
|
|
|
const char *session_bus_address = g_getenv ("DBUS_SESSION_BUS_ADDRESS");
|
|
|
|
if (!session_bus_address)
|
|
|
|
session_bus_address = g_getenv ("HOSTNAME");
|
|
|
|
if (session_bus_address)
|
|
|
|
timer_perturb = ABS ((gint) g_str_hash (session_bus_address)) % 1000000;
|
|
|
|
else
|
|
|
|
timer_perturb = 0;
|
2007-07-22 02:34:21 +02:00
|
|
|
}
|
2006-09-10 07:44:46 +02:00
|
|
|
|
2010-11-01 18:31:37 +01:00
|
|
|
/* We want the microseconds part of the timeout to land on the
|
|
|
|
* 'timer_perturb' mark, but we need to make sure we don't try to
|
|
|
|
* set the timeout in the past. We do this by ensuring that we
|
|
|
|
* always only *increase* the expiration time by adding a full
|
|
|
|
* second in the case that the microsecond portion decreases.
|
|
|
|
*/
|
2011-03-07 04:37:01 +01:00
|
|
|
timeout_source->expiration -= timer_perturb;
|
|
|
|
|
|
|
|
remainder = timeout_source->expiration % 1000000;
|
|
|
|
if (remainder >= 1000000/4)
|
2010-11-01 20:46:35 +01:00
|
|
|
timeout_source->expiration += 1000000;
|
2006-09-10 07:44:46 +02:00
|
|
|
|
2011-03-07 04:37:01 +01:00
|
|
|
timeout_source->expiration -= remainder;
|
|
|
|
timeout_source->expiration += timer_perturb;
|
2006-09-10 07:44:46 +02:00
|
|
|
}
|
2000-03-01 10:44:10 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
2007-07-22 02:34:21 +02:00
|
|
|
g_timeout_prepare (GSource *source,
|
2010-11-01 20:46:35 +01:00
|
|
|
gint *timeout)
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
2010-11-01 20:46:35 +01:00
|
|
|
GTimeoutSource *timeout_source = (GTimeoutSource *) source;
|
|
|
|
gint64 now = g_source_get_time (source);
|
2000-12-05 21:45:33 +01:00
|
|
|
|
2010-11-01 20:46:35 +01:00
|
|
|
if (now < timeout_source->expiration)
|
2000-03-01 10:44:10 +01:00
|
|
|
{
|
2010-11-01 20:46:35 +01:00
|
|
|
/* Round up to ensure that we don't try again too early */
|
|
|
|
*timeout = (timeout_source->expiration - now + 999) / 1000;
|
|
|
|
return FALSE;
|
2000-03-01 10:44:10 +01:00
|
|
|
}
|
Changes for 64-bit cleanliness, loosely based on patch from Mark Murnane.
Wed Jun 20 12:00:54 2001 Owen Taylor <otaylor@redhat.com>
Changes for 64-bit cleanliness, loosely based on patch
from Mark Murnane.
* gconvert.c (g_convert/g_convert_with_fallback): Remove
workarounds for since-fixed GNU libc bugs. Minor
doc fix.
* gconvert.[ch]: Change gint to gsize/gssize as
appropriate.
* gconvert.c (g_locale/filename_to/from_utf8): Fix incorrect
computation of bytes_read / bytes_written.
* gfileutils.[ch] (g_file_get_contents): Make length
out parameter 'gsize *len'.
* ghook.c (g_hook_compare_ids): Don't compare a
and b as 'a - b'.
* gmacros.h (GSIZE_TO_POINTER): Add GPOINTER_TO_SIZE,
GSIZE_TO_POINTER.
* gmain.c (g_timeout_prepare): Rewrite to avoid
overflows. (Fixes bug when system clock skews
backwards more than 24 days.)
* gmarkup.[ch]: Make lengths passed to callbacks
gsize, length for g_markup_parse-context_parse(),
g_markup_escape_text() gssize.
* gmessages.[ch] (g_printf_string_upper_bound): Change
return value to gsize.
* gmessages.c (printf_string_upper_bound): Remove
a ridiculous use of 'inline' on a 300 line function.
* gstring.[ch]: Represent size of string as a gsize,
not gint. Make parameters to functions take gsize,
or gssize where -1 is allowed.
* gstring.c (g_string_erase): Make
g_string_erase (string, pos, -1) a synonym for
g_string_truncate for consistency with other G*
APIs.
* gstrfuncs.[ch]: Make all functions taking a string
length, take a gsize, or gssize if -1 is allowed.
(g_strstr_len, g_strrstr_len). Also fix some boundary
conditions in g_str[r]str[_len].
* gutf8.c tests/unicode-encoding.c: Make parameters that
are byte lengths gsize, gssize as appropriate. Make
character offsets, other counts, glong.
* gasyncqueue.c gcompletion.c
timeloop.c timeloop-basic.c gutils.c gspawn.c.
Small 64 bit cleanliness fixups.
* glist.c (g_list_sort2, g_list_sort_real): Fix functions
that should have been static.
* gdate.c (g_date_fill_parse_tokens): Fix extra
declaration that was shadowing another.
* tests/module-test.c: Include string.h
Mon Jun 18 15:43:29 2001 Owen Taylor <otaylor@redhat.com>
* gutf8.c (g_get_charset): Make argument
G_CONST_RETURN char **.
2001-06-23 15:55:09 +02:00
|
|
|
|
2010-11-01 20:46:35 +01:00
|
|
|
*timeout = 0;
|
|
|
|
return TRUE;
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
2007-07-22 02:34:21 +02:00
|
|
|
g_timeout_check (GSource *source)
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
2010-11-01 20:46:35 +01:00
|
|
|
GTimeoutSource *timeout_source = (GTimeoutSource *) source;
|
|
|
|
gint64 now = g_source_get_time (source);
|
2000-12-05 21:45:33 +01:00
|
|
|
|
2010-11-01 20:46:35 +01:00
|
|
|
return timeout_source->expiration <= now;
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
2007-07-22 02:34:21 +02:00
|
|
|
g_timeout_dispatch (GSource *source,
|
2010-11-01 20:46:35 +01:00
|
|
|
GSourceFunc callback,
|
|
|
|
gpointer user_data)
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
GTimeoutSource *timeout_source = (GTimeoutSource *)source;
|
2010-11-01 20:46:35 +01:00
|
|
|
gboolean again;
|
1998-12-02 15:55:27 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
if (!callback)
|
|
|
|
{
|
|
|
|
g_warning ("Timeout source dispatched without callback\n"
|
2010-11-01 20:46:35 +01:00
|
|
|
"You must call g_source_set_callback().");
|
2000-12-05 21:45:33 +01:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2010-11-01 20:46:35 +01:00
|
|
|
again = callback (user_data);
|
1998-12-18 18:52:18 +01:00
|
|
|
|
2010-11-01 20:46:35 +01:00
|
|
|
if (again)
|
|
|
|
g_timeout_set_expiration (timeout_source, g_source_get_time (source));
|
|
|
|
|
|
|
|
return again;
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_timeout_source_new:
|
|
|
|
* @interval: the timeout interval in milliseconds.
|
|
|
|
*
|
2001-12-16 20:31:36 +01:00
|
|
|
* Creates a new timeout source.
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
2001-12-16 20:31:36 +01:00
|
|
|
* Return value: the newly-created timeout source
|
2000-12-05 21:45:33 +01:00
|
|
|
**/
|
|
|
|
GSource *
|
|
|
|
g_timeout_source_new (guint interval)
|
|
|
|
{
|
2001-09-04 00:12:51 +02:00
|
|
|
GSource *source = g_source_new (&g_timeout_funcs, sizeof (GTimeoutSource));
|
2000-12-05 21:45:33 +01:00
|
|
|
GTimeoutSource *timeout_source = (GTimeoutSource *)source;
|
|
|
|
|
|
|
|
timeout_source->interval = interval;
|
2010-11-01 20:46:35 +01:00
|
|
|
g_timeout_set_expiration (timeout_source, g_get_monotonic_time ());
|
2000-12-05 21:45:33 +01:00
|
|
|
|
|
|
|
return source;
|
|
|
|
}
|
|
|
|
|
2006-09-10 07:44:46 +02:00
|
|
|
/**
|
|
|
|
* g_timeout_source_new_seconds:
|
|
|
|
* @interval: the timeout interval in seconds
|
|
|
|
*
|
|
|
|
* 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.
|
2007-07-22 02:34:21 +02:00
|
|
|
*
|
2006-09-10 07:44:46 +02:00
|
|
|
* The scheduling granularity/accuracy of this timeout source will be
|
|
|
|
* in seconds.
|
|
|
|
*
|
|
|
|
* Return value: the newly-created timeout source
|
|
|
|
*
|
|
|
|
* Since: 2.14
|
|
|
|
**/
|
|
|
|
GSource *
|
|
|
|
g_timeout_source_new_seconds (guint interval)
|
|
|
|
{
|
|
|
|
GSource *source = g_source_new (&g_timeout_funcs, sizeof (GTimeoutSource));
|
|
|
|
GTimeoutSource *timeout_source = (GTimeoutSource *)source;
|
|
|
|
|
2010-11-01 18:31:37 +01:00
|
|
|
timeout_source->interval = 1000 * interval;
|
|
|
|
timeout_source->seconds = TRUE;
|
2006-09-10 07:44:46 +02:00
|
|
|
|
2010-11-01 20:46:35 +01:00
|
|
|
g_timeout_set_expiration (timeout_source, g_get_monotonic_time ());
|
2006-09-10 07:44:46 +02:00
|
|
|
|
|
|
|
return source;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_timeout_add_full:
|
2007-06-18 18:55:50 +02:00
|
|
|
* @priority: the priority of the timeout source. Typically this will be in
|
|
|
|
* the range between #G_PRIORITY_DEFAULT and #G_PRIORITY_HIGH.
|
2000-12-05 21:45:33 +01:00
|
|
|
* @interval: the time between calls to the function, in milliseconds
|
2001-12-16 20:31:36 +01:00
|
|
|
* (1/1000ths of a second)
|
2000-12-05 21:45:33 +01:00
|
|
|
* @function: function to call
|
|
|
|
* @data: data to pass to @function
|
2007-06-18 18:55:50 +02:00
|
|
|
* @notify: function to call when the timeout is removed, or %NULL
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
|
|
|
* Sets a function to be called at regular intervals, with the given
|
|
|
|
* priority. The function is called repeatedly until it returns
|
2001-12-16 20:31:36 +01:00
|
|
|
* %FALSE, at which point the timeout is automatically destroyed and
|
2000-12-05 21:45:33 +01:00
|
|
|
* 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).
|
2008-09-26 16:05:03 +02:00
|
|
|
*
|
|
|
|
* This internally creates a main loop source using g_timeout_source_new()
|
|
|
|
* and attaches it to the main loop context using g_source_attach(). You can
|
|
|
|
* do these steps manually if you need greater control.
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
2006-01-05 22:07:55 +01:00
|
|
|
* Return value: the ID (greater than 0) of the event source.
|
2000-12-05 21:45:33 +01:00
|
|
|
**/
|
2000-03-01 10:44:10 +01:00
|
|
|
guint
|
1998-12-02 15:55:27 +01:00
|
|
|
g_timeout_add_full (gint priority,
|
2000-03-01 10:44:10 +01:00
|
|
|
guint interval,
|
1998-12-02 15:55:27 +01:00
|
|
|
GSourceFunc function,
|
|
|
|
gpointer data,
|
|
|
|
GDestroyNotify notify)
|
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
GSource *source;
|
2000-12-07 21:29:58 +01:00
|
|
|
guint id;
|
2000-12-05 21:45:33 +01:00
|
|
|
|
|
|
|
g_return_val_if_fail (function != NULL, 0);
|
1998-12-02 15:55:27 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
source = g_timeout_source_new (interval);
|
1998-12-18 18:52:18 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
if (priority != G_PRIORITY_DEFAULT)
|
|
|
|
g_source_set_priority (source, priority);
|
1998-12-02 15:55:27 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
g_source_set_callback (source, function, data, notify);
|
2000-12-07 21:29:58 +01:00
|
|
|
id = g_source_attach (source, NULL);
|
|
|
|
g_source_unref (source);
|
|
|
|
|
|
|
|
return id;
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_timeout_add:
|
|
|
|
* @interval: the time between calls to the function, in milliseconds
|
2001-12-16 20:31:36 +01:00
|
|
|
* (1/1000ths of a second)
|
2000-12-05 21:45:33 +01:00
|
|
|
* @function: function to call
|
|
|
|
* @data: data to pass to @function
|
|
|
|
*
|
|
|
|
* Sets a function to be called at regular intervals, with the default
|
|
|
|
* priority, #G_PRIORITY_DEFAULT. The function is called repeatedly
|
2001-12-16 20:31:36 +01:00
|
|
|
* until it returns %FALSE, at which point the timeout is automatically
|
2002-07-26 01:04:22 +02:00
|
|
|
* destroyed and the function will not be called again. The first call
|
2000-12-05 21:45:33 +01:00
|
|
|
* 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).
|
2006-09-10 07:44:46 +02:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
2008-09-26 16:05:03 +02:00
|
|
|
* This internally creates a main loop source using g_timeout_source_new()
|
|
|
|
* and attaches it to the main loop context using g_source_attach(). You can
|
|
|
|
* do these steps manually if you need greater control.
|
|
|
|
*
|
2006-01-05 22:07:55 +01:00
|
|
|
* Return value: the ID (greater than 0) of the event source.
|
2000-12-05 21:45:33 +01:00
|
|
|
**/
|
2006-09-10 07:44:46 +02:00
|
|
|
guint
|
1998-12-02 15:55:27 +01:00
|
|
|
g_timeout_add (guint32 interval,
|
|
|
|
GSourceFunc function,
|
|
|
|
gpointer data)
|
|
|
|
{
|
1998-12-19 23:21:39 +01:00
|
|
|
return g_timeout_add_full (G_PRIORITY_DEFAULT,
|
|
|
|
interval, function, data, NULL);
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
2006-09-10 07:44:46 +02:00
|
|
|
/**
|
2007-06-18 18:55:50 +02:00
|
|
|
* g_timeout_add_seconds_full:
|
|
|
|
* @priority: the priority of the timeout source. Typically this will be in
|
|
|
|
* the range between #G_PRIORITY_DEFAULT and #G_PRIORITY_HIGH.
|
2006-09-10 07:44:46 +02:00
|
|
|
* @interval: the time between calls to the function, in seconds
|
|
|
|
* @function: function to call
|
|
|
|
* @data: data to pass to @function
|
2007-06-18 18:55:50 +02:00
|
|
|
* @notify: function to call when the timeout is removed, or %NULL
|
2006-09-10 07:44:46 +02:00
|
|
|
*
|
2007-06-18 18:55:50 +02:00
|
|
|
* 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.
|
2006-09-10 07:44:46 +02:00
|
|
|
*
|
|
|
|
* 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
|
2007-06-14 15:52:01 +02:00
|
|
|
* 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().
|
2006-09-10 07:44:46 +02:00
|
|
|
*
|
2008-09-26 16:05:03 +02:00
|
|
|
* This internally creates a main loop source using
|
|
|
|
* g_timeout_source_new_seconds() and attaches it to the main loop context
|
|
|
|
* using g_source_attach(). You can do these steps manually if you need
|
|
|
|
* greater control.
|
|
|
|
*
|
2006-09-10 07:44:46 +02:00
|
|
|
* Return value: the ID (greater than 0) of the event source.
|
|
|
|
*
|
|
|
|
* Since: 2.14
|
|
|
|
**/
|
|
|
|
guint
|
2007-06-18 18:55:50 +02:00
|
|
|
g_timeout_add_seconds_full (gint priority,
|
|
|
|
guint32 interval,
|
|
|
|
GSourceFunc function,
|
|
|
|
gpointer data,
|
|
|
|
GDestroyNotify notify)
|
2006-09-10 07:44:46 +02:00
|
|
|
{
|
|
|
|
GSource *source;
|
|
|
|
guint id;
|
|
|
|
|
|
|
|
g_return_val_if_fail (function != NULL, 0);
|
|
|
|
|
|
|
|
source = g_timeout_source_new_seconds (interval);
|
|
|
|
|
2007-06-18 18:55:50 +02:00
|
|
|
if (priority != G_PRIORITY_DEFAULT)
|
|
|
|
g_source_set_priority (source, priority);
|
|
|
|
|
|
|
|
g_source_set_callback (source, function, data, notify);
|
2006-09-10 07:44:46 +02:00
|
|
|
id = g_source_attach (source, NULL);
|
|
|
|
g_source_unref (source);
|
|
|
|
|
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
2007-06-18 18:55:50 +02:00
|
|
|
/**
|
|
|
|
* g_timeout_add_seconds:
|
|
|
|
* @interval: the time between calls to the function, in seconds
|
|
|
|
* @function: function to call
|
|
|
|
* @data: data to pass to @function
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
2008-09-26 16:05:03 +02:00
|
|
|
* This internally creates a main loop source using
|
|
|
|
* g_timeout_source_new_seconds() and attaches it to the main loop context
|
|
|
|
* using g_source_attach(). You can do these steps manually if you need
|
|
|
|
* greater control. Also see g_timout_add_seconds_full().
|
|
|
|
*
|
2007-06-18 18:55:50 +02:00
|
|
|
* Return value: the ID (greater than 0) of the event source.
|
|
|
|
*
|
|
|
|
* Since: 2.14
|
|
|
|
**/
|
|
|
|
guint
|
|
|
|
g_timeout_add_seconds (guint interval,
|
|
|
|
GSourceFunc function,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (function != NULL, 0);
|
|
|
|
|
|
|
|
return g_timeout_add_seconds_full (G_PRIORITY_DEFAULT, interval, function, data, NULL);
|
|
|
|
}
|
2006-09-10 07:44:46 +02:00
|
|
|
|
2004-02-14 01:23:36 +01:00
|
|
|
/* Child watch functions */
|
|
|
|
|
2004-02-28 14:03:55 +01:00
|
|
|
#ifdef G_OS_WIN32
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
g_child_watch_prepare (GSource *source,
|
|
|
|
gint *timeout)
|
|
|
|
{
|
|
|
|
*timeout = -1;
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
g_child_watch_check (GSource *source)
|
|
|
|
{
|
|
|
|
GChildWatchSource *child_watch_source;
|
|
|
|
gboolean child_exited;
|
|
|
|
|
|
|
|
child_watch_source = (GChildWatchSource *) source;
|
|
|
|
|
|
|
|
child_exited = child_watch_source->poll.revents & G_IO_IN;
|
|
|
|
|
|
|
|
if (child_exited)
|
|
|
|
{
|
|
|
|
DWORD child_status;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Note: We do _not_ check for the special value of STILL_ACTIVE
|
|
|
|
* since we know that the process has exited and doing so runs into
|
|
|
|
* problems if the child process "happens to return STILL_ACTIVE(259)"
|
|
|
|
* as Microsoft's Platform SDK puts it.
|
|
|
|
*/
|
|
|
|
if (!GetExitCodeProcess (child_watch_source->pid, &child_status))
|
|
|
|
{
|
|
|
|
gchar *emsg = g_win32_error_message (GetLastError ());
|
|
|
|
g_warning (G_STRLOC ": GetExitCodeProcess() failed: %s", emsg);
|
|
|
|
g_free (emsg);
|
|
|
|
|
|
|
|
child_watch_source->child_status = -1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
child_watch_source->child_status = child_status;
|
|
|
|
}
|
|
|
|
|
|
|
|
return child_exited;
|
|
|
|
}
|
|
|
|
|
|
|
|
#else /* G_OS_WIN32 */
|
|
|
|
|
2004-03-01 21:56:47 +01:00
|
|
|
static gboolean
|
2004-02-14 01:23:36 +01:00
|
|
|
check_for_child_exited (GSource *source)
|
|
|
|
{
|
|
|
|
GChildWatchSource *child_watch_source;
|
|
|
|
gint count;
|
|
|
|
|
|
|
|
/* protect against another SIGCHLD in the middle of this call */
|
|
|
|
count = child_watch_count;
|
|
|
|
|
|
|
|
child_watch_source = (GChildWatchSource *) source;
|
|
|
|
|
2004-03-01 21:56:47 +01:00
|
|
|
if (child_watch_source->child_exited)
|
|
|
|
return TRUE;
|
|
|
|
|
2004-02-14 01:23:36 +01:00
|
|
|
if (child_watch_source->count < count)
|
|
|
|
{
|
|
|
|
gint child_status;
|
|
|
|
|
|
|
|
if (waitpid (child_watch_source->pid, &child_status, WNOHANG) > 0)
|
|
|
|
{
|
|
|
|
child_watch_source->child_status = child_status;
|
|
|
|
child_watch_source->child_exited = TRUE;
|
|
|
|
}
|
|
|
|
child_watch_source->count = count;
|
|
|
|
}
|
2004-03-01 21:56:47 +01:00
|
|
|
|
|
|
|
return child_watch_source->child_exited;
|
2004-02-14 01:23:36 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
g_child_watch_prepare (GSource *source,
|
|
|
|
gint *timeout)
|
|
|
|
{
|
|
|
|
*timeout = -1;
|
|
|
|
|
2004-03-01 21:56:47 +01:00
|
|
|
return check_for_child_exited (source);
|
2004-02-14 01:23:36 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
g_child_watch_check (GSource *source)
|
|
|
|
{
|
2004-03-01 21:56:47 +01:00
|
|
|
return check_for_child_exited (source);
|
2004-02-14 01:23:36 +01:00
|
|
|
}
|
|
|
|
|
2004-02-28 14:03:55 +01:00
|
|
|
#endif /* G_OS_WIN32 */
|
|
|
|
|
2004-02-14 01:23:36 +01:00
|
|
|
static gboolean
|
|
|
|
g_child_watch_dispatch (GSource *source,
|
|
|
|
GSourceFunc callback,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
GChildWatchSource *child_watch_source;
|
|
|
|
GChildWatchFunc child_watch_callback = (GChildWatchFunc) callback;
|
|
|
|
|
|
|
|
child_watch_source = (GChildWatchSource *) source;
|
|
|
|
|
|
|
|
if (!callback)
|
|
|
|
{
|
|
|
|
g_warning ("Child watch source dispatched without callback\n"
|
|
|
|
"You must call g_source_set_callback().");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
(child_watch_callback) (child_watch_source->pid, child_watch_source->child_status, user_data);
|
|
|
|
|
|
|
|
/* We never keep a child watch source around as the child is gone */
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2004-02-28 14:03:55 +01:00
|
|
|
#ifndef G_OS_WIN32
|
|
|
|
|
2004-02-14 01:23:36 +01:00
|
|
|
static void
|
|
|
|
g_child_watch_signal_handler (int signum)
|
|
|
|
{
|
|
|
|
child_watch_count ++;
|
|
|
|
|
|
|
|
if (child_watch_init_state == CHILD_WATCH_INITIALIZED_THREADED)
|
|
|
|
{
|
|
|
|
write (child_watch_wake_up_pipe[1], "B", 1);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* We count on the signal interrupting the poll in the same thread.
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
g_child_watch_source_init_single (void)
|
|
|
|
{
|
2004-11-08 16:34:26 +01:00
|
|
|
struct sigaction action;
|
|
|
|
|
2004-02-14 01:23:36 +01:00
|
|
|
g_assert (! g_thread_supported());
|
|
|
|
g_assert (child_watch_init_state == CHILD_WATCH_UNINITIALIZED);
|
|
|
|
|
|
|
|
child_watch_init_state = CHILD_WATCH_INITIALIZED_SINGLE;
|
|
|
|
|
2004-11-08 16:34:26 +01:00
|
|
|
action.sa_handler = g_child_watch_signal_handler;
|
|
|
|
sigemptyset (&action.sa_mask);
|
2005-04-01 23:40:43 +02:00
|
|
|
action.sa_flags = SA_NOCLDSTOP;
|
2004-11-08 16:34:26 +01:00
|
|
|
sigaction (SIGCHLD, &action, NULL);
|
2004-02-14 01:23:36 +01:00
|
|
|
}
|
|
|
|
|
2008-12-31 06:51:47 +01:00
|
|
|
G_GNUC_NORETURN static gpointer
|
|
|
|
child_watch_helper_thread (gpointer data)
|
2004-02-14 01:23:36 +01:00
|
|
|
{
|
|
|
|
while (1)
|
|
|
|
{
|
|
|
|
gchar b[20];
|
|
|
|
GSList *list;
|
|
|
|
|
|
|
|
read (child_watch_wake_up_pipe[0], b, 20);
|
|
|
|
|
|
|
|
/* We were woken up. Wake up all other contexts in all other threads */
|
2004-03-04 11:12:55 +01:00
|
|
|
G_LOCK (main_context_list);
|
2004-02-14 01:23:36 +01:00
|
|
|
for (list = main_context_list; list; list = list->next)
|
|
|
|
{
|
|
|
|
GMainContext *context;
|
|
|
|
|
|
|
|
context = list->data;
|
2004-03-04 11:12:55 +01:00
|
|
|
if (g_atomic_int_get (&context->ref_count) > 0)
|
|
|
|
/* Due to racing conditions we can find ref_count == 0, in
|
|
|
|
* that case, however, the context is still not destroyed
|
|
|
|
* and no poll can be active, otherwise the ref_count
|
|
|
|
* wouldn't be 0 */
|
|
|
|
g_main_context_wakeup (context);
|
2004-02-14 01:23:36 +01:00
|
|
|
}
|
2004-03-04 11:12:55 +01:00
|
|
|
G_UNLOCK (main_context_list);
|
2004-02-14 01:23:36 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
g_child_watch_source_init_multi_threaded (void)
|
|
|
|
{
|
|
|
|
GError *error = NULL;
|
2004-11-08 16:34:26 +01:00
|
|
|
struct sigaction action;
|
2004-02-14 01:23:36 +01:00
|
|
|
|
|
|
|
g_assert (g_thread_supported());
|
|
|
|
|
|
|
|
if (pipe (child_watch_wake_up_pipe) < 0)
|
|
|
|
g_error ("Cannot create wake up pipe: %s\n", g_strerror (errno));
|
|
|
|
fcntl (child_watch_wake_up_pipe[1], F_SETFL, O_NONBLOCK | fcntl (child_watch_wake_up_pipe[1], F_GETFL));
|
|
|
|
|
|
|
|
/* We create a helper thread that polls on the wakeup pipe indefinitely */
|
|
|
|
/* FIXME: Think this through for races */
|
|
|
|
if (g_thread_create (child_watch_helper_thread, NULL, FALSE, &error) == NULL)
|
|
|
|
g_error ("Cannot create a thread to monitor child exit status: %s\n", error->message);
|
|
|
|
child_watch_init_state = CHILD_WATCH_INITIALIZED_THREADED;
|
2004-11-08 16:34:26 +01:00
|
|
|
|
|
|
|
action.sa_handler = g_child_watch_signal_handler;
|
|
|
|
sigemptyset (&action.sa_mask);
|
|
|
|
action.sa_flags = SA_RESTART | SA_NOCLDSTOP;
|
|
|
|
sigaction (SIGCHLD, &action, NULL);
|
2004-02-14 01:23:36 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
g_child_watch_source_init_promote_single_to_threaded (void)
|
|
|
|
{
|
|
|
|
g_child_watch_source_init_multi_threaded ();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
g_child_watch_source_init (void)
|
|
|
|
{
|
|
|
|
if (g_thread_supported())
|
|
|
|
{
|
|
|
|
if (child_watch_init_state == CHILD_WATCH_UNINITIALIZED)
|
|
|
|
g_child_watch_source_init_multi_threaded ();
|
|
|
|
else if (child_watch_init_state == CHILD_WATCH_INITIALIZED_SINGLE)
|
|
|
|
g_child_watch_source_init_promote_single_to_threaded ();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (child_watch_init_state == CHILD_WATCH_UNINITIALIZED)
|
|
|
|
g_child_watch_source_init_single ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-02-28 14:03:55 +01:00
|
|
|
#endif /* !G_OS_WIN32 */
|
|
|
|
|
2004-02-14 01:23:36 +01:00
|
|
|
/**
|
|
|
|
* g_child_watch_source_new:
|
2008-06-11 08:57:22 +02:00
|
|
|
* @pid: process to watch. On POSIX the pid of a child process. On
|
|
|
|
* Windows a handle for a process (which doesn't have to be a child).
|
2004-02-14 01:23:36 +01:00
|
|
|
*
|
|
|
|
* Creates a new child_watch 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.
|
|
|
|
*
|
2005-11-04 21:33:34 +01:00
|
|
|
* Note that child watch sources can only be used in conjunction with
|
|
|
|
* <literal>g_spawn...</literal> when the %G_SPAWN_DO_NOT_REAP_CHILD
|
|
|
|
* flag is used.
|
|
|
|
*
|
2006-10-01 07:35:29 +02:00
|
|
|
* Note that on platforms where #GPid must be explicitly closed
|
2004-03-01 21:47:49 +01:00
|
|
|
* (see g_spawn_close_pid()) @pid must not be closed while the
|
|
|
|
* source is still active. Typically, you will want to call
|
|
|
|
* g_spawn_close_pid() in the callback function for the source.
|
2005-04-01 23:40:43 +02:00
|
|
|
*
|
|
|
|
* Note further that using g_child_watch_source_new() is not
|
|
|
|
* compatible with calling <literal>waitpid(-1)</literal> in
|
|
|
|
* the application. Calling waitpid() for individual pids will
|
|
|
|
* still work fine.
|
2004-03-01 21:47:49 +01:00
|
|
|
*
|
2004-02-14 01:23:36 +01:00
|
|
|
* Return value: the newly-created child watch source
|
|
|
|
*
|
|
|
|
* Since: 2.4
|
|
|
|
**/
|
|
|
|
GSource *
|
|
|
|
g_child_watch_source_new (GPid pid)
|
|
|
|
{
|
|
|
|
GSource *source = g_source_new (&g_child_watch_funcs, sizeof (GChildWatchSource));
|
|
|
|
GChildWatchSource *child_watch_source = (GChildWatchSource *)source;
|
|
|
|
|
2004-02-28 14:03:55 +01:00
|
|
|
#ifdef G_OS_WIN32
|
2008-08-04 20:46:59 +02:00
|
|
|
child_watch_source->poll.fd = (gintptr) pid;
|
2004-02-28 14:03:55 +01:00
|
|
|
child_watch_source->poll.events = G_IO_IN;
|
|
|
|
|
|
|
|
g_source_add_poll (source, &child_watch_source->poll);
|
|
|
|
#else /* G_OS_WIN32 */
|
2004-02-14 01:23:36 +01:00
|
|
|
g_child_watch_source_init ();
|
2004-02-28 14:03:55 +01:00
|
|
|
#endif /* G_OS_WIN32 */
|
2004-02-14 01:23:36 +01:00
|
|
|
|
|
|
|
child_watch_source->pid = pid;
|
|
|
|
|
|
|
|
return source;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_child_watch_add_full:
|
|
|
|
* @priority: the priority of the idle source. Typically this will be in the
|
|
|
|
* range between #G_PRIORITY_DEFAULT_IDLE and #G_PRIORITY_HIGH_IDLE.
|
2008-06-11 08:57:22 +02:00
|
|
|
* @pid: process to watch. On POSIX the pid of a child process. On
|
|
|
|
* Windows a handle for a process (which doesn't have to be a child).
|
2004-02-14 01:23:36 +01:00
|
|
|
* @function: function to call
|
|
|
|
* @data: data to pass to @function
|
|
|
|
* @notify: function to call when the idle is removed, or %NULL
|
|
|
|
*
|
2006-12-15 04:42:24 +01:00
|
|
|
* Sets a function to be called when the child indicated by @pid
|
|
|
|
* exits, at the priority @priority.
|
|
|
|
*
|
|
|
|
* If you obtain @pid from g_spawn_async() or g_spawn_async_with_pipes()
|
|
|
|
* you will need to pass #G_SPAWN_DO_NOT_REAP_CHILD as flag to
|
|
|
|
* the spawn function for the child watching to work.
|
2004-02-14 01:23:36 +01:00
|
|
|
*
|
2006-10-01 07:35:29 +02:00
|
|
|
* Note that on platforms where #GPid must be explicitly closed
|
2004-03-01 21:47:49 +01:00
|
|
|
* (see g_spawn_close_pid()) @pid must not be closed while the
|
|
|
|
* source is still active. Typically, you will want to call
|
|
|
|
* g_spawn_close_pid() in the callback function for the source.
|
|
|
|
*
|
2004-10-08 16:20:56 +02:00
|
|
|
* GLib supports only a single callback per process id.
|
|
|
|
*
|
2008-09-26 16:05:03 +02:00
|
|
|
* This internally creates a main loop source using
|
|
|
|
* g_child_watch_source_new() and attaches it to the main loop context
|
|
|
|
* using g_source_attach(). You can do these steps manually if you
|
|
|
|
* need greater control.
|
|
|
|
*
|
2006-01-05 22:07:55 +01:00
|
|
|
* Return value: the ID (greater than 0) of the event source.
|
2004-02-14 01:23:36 +01:00
|
|
|
*
|
|
|
|
* Since: 2.4
|
|
|
|
**/
|
|
|
|
guint
|
|
|
|
g_child_watch_add_full (gint priority,
|
|
|
|
GPid pid,
|
|
|
|
GChildWatchFunc function,
|
|
|
|
gpointer data,
|
|
|
|
GDestroyNotify notify)
|
|
|
|
{
|
|
|
|
GSource *source;
|
|
|
|
guint id;
|
|
|
|
|
|
|
|
g_return_val_if_fail (function != NULL, 0);
|
|
|
|
|
|
|
|
source = g_child_watch_source_new (pid);
|
|
|
|
|
|
|
|
if (priority != G_PRIORITY_DEFAULT)
|
|
|
|
g_source_set_priority (source, priority);
|
|
|
|
|
|
|
|
g_source_set_callback (source, (GSourceFunc) function, data, notify);
|
|
|
|
id = g_source_attach (source, NULL);
|
|
|
|
g_source_unref (source);
|
|
|
|
|
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_child_watch_add:
|
2008-06-11 08:57:22 +02:00
|
|
|
* @pid: process id to watch. On POSIX the pid of a child process. On
|
|
|
|
* Windows a handle for a process (which doesn't have to be a child).
|
2004-02-14 01:23:36 +01:00
|
|
|
* @function: function to call
|
|
|
|
* @data: data to pass to @function
|
|
|
|
*
|
2006-12-15 04:42:24 +01:00
|
|
|
* Sets a function to be called when the child indicated by @pid
|
|
|
|
* exits, at a default priority, #G_PRIORITY_DEFAULT.
|
|
|
|
*
|
|
|
|
* If you obtain @pid from g_spawn_async() or g_spawn_async_with_pipes()
|
|
|
|
* you will need to pass #G_SPAWN_DO_NOT_REAP_CHILD as flag to
|
|
|
|
* the spawn function for the child watching to work.
|
2004-02-14 01:23:36 +01:00
|
|
|
*
|
2006-10-01 07:35:29 +02:00
|
|
|
* Note that on platforms where #GPid must be explicitly closed
|
2004-03-01 21:47:49 +01:00
|
|
|
* (see g_spawn_close_pid()) @pid must not be closed while the
|
|
|
|
* source is still active. Typically, you will want to call
|
|
|
|
* g_spawn_close_pid() in the callback function for the source.
|
|
|
|
*
|
2004-10-08 16:20:56 +02:00
|
|
|
* GLib supports only a single callback per process id.
|
|
|
|
*
|
2008-09-26 16:05:03 +02:00
|
|
|
* This internally creates a main loop source using
|
|
|
|
* g_child_watch_source_new() and attaches it to the main loop context
|
|
|
|
* using g_source_attach(). You can do these steps manually if you
|
|
|
|
* need greater control.
|
|
|
|
*
|
2006-01-05 22:07:55 +01:00
|
|
|
* Return value: the ID (greater than 0) of the event source.
|
2004-02-14 01:23:36 +01:00
|
|
|
*
|
|
|
|
* Since: 2.4
|
|
|
|
**/
|
|
|
|
guint
|
|
|
|
g_child_watch_add (GPid pid,
|
|
|
|
GChildWatchFunc function,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
return g_child_watch_add_full (G_PRIORITY_DEFAULT, pid, function, data, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
1998-12-02 15:55:27 +01:00
|
|
|
/* Idle functions */
|
|
|
|
|
|
|
|
static gboolean
|
2000-12-05 21:45:33 +01:00
|
|
|
g_idle_prepare (GSource *source,
|
|
|
|
gint *timeout)
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
2000-05-19 10:18:29 +02:00
|
|
|
*timeout = 0;
|
|
|
|
|
1998-12-02 15:55:27 +01:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
2000-12-05 21:45:33 +01:00
|
|
|
g_idle_check (GSource *source)
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
2000-12-05 21:45:33 +01:00
|
|
|
g_idle_dispatch (GSource *source,
|
|
|
|
GSourceFunc callback,
|
|
|
|
gpointer user_data)
|
1998-12-02 15:55:27 +01:00
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
if (!callback)
|
|
|
|
{
|
|
|
|
g_warning ("Idle source dispatched without callback\n"
|
|
|
|
"You must call g_source_set_callback().");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return callback (user_data);
|
|
|
|
}
|
1998-12-02 15:55:27 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_idle_source_new:
|
|
|
|
*
|
2001-12-16 20:31:36 +01:00
|
|
|
* Creates a new idle source.
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
|
|
|
* The source will not initially be associated with any #GMainContext
|
|
|
|
* and must be added to one with g_source_attach() before it will be
|
2003-08-07 20:19:23 +02:00
|
|
|
* 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.
|
2000-12-05 21:45:33 +01:00
|
|
|
*
|
2001-12-16 20:31:36 +01:00
|
|
|
* Return value: the newly-created idle source
|
2000-12-05 21:45:33 +01:00
|
|
|
**/
|
|
|
|
GSource *
|
|
|
|
g_idle_source_new (void)
|
|
|
|
{
|
2003-08-07 20:19:23 +02:00
|
|
|
GSource *source;
|
|
|
|
|
|
|
|
source = g_source_new (&g_idle_funcs, sizeof (GSource));
|
|
|
|
g_source_set_priority (source, G_PRIORITY_DEFAULT_IDLE);
|
|
|
|
|
|
|
|
return source;
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_idle_add_full:
|
|
|
|
* @priority: the priority of the idle source. Typically this will be in the
|
2009-03-18 00:03:33 +01:00
|
|
|
* range between #G_PRIORITY_DEFAULT_IDLE and #G_PRIORITY_HIGH_IDLE.
|
2000-12-05 21:45:33 +01:00
|
|
|
* @function: function to call
|
|
|
|
* @data: data to pass to @function
|
|
|
|
* @notify: function to call when the idle is removed, or %NULL
|
|
|
|
*
|
|
|
|
* Adds a function to be called whenever there are no higher priority
|
2001-12-16 20:31:36 +01:00
|
|
|
* events pending. If the function returns %FALSE it is automatically
|
2000-12-05 21:45:33 +01:00
|
|
|
* removed from the list of event sources and will not be called again.
|
|
|
|
*
|
2008-09-26 16:05:03 +02:00
|
|
|
* This internally creates a main loop source using g_idle_source_new()
|
|
|
|
* and attaches it to the main loop context using g_source_attach().
|
|
|
|
* You can do these steps manually if you need greater control.
|
|
|
|
*
|
2006-01-05 22:07:55 +01:00
|
|
|
* Return value: the ID (greater than 0) of the event source.
|
2000-12-05 21:45:33 +01:00
|
|
|
**/
|
1998-12-02 15:55:27 +01:00
|
|
|
guint
|
1999-01-17 05:49:43 +01:00
|
|
|
g_idle_add_full (gint priority,
|
1998-12-02 15:55:27 +01:00
|
|
|
GSourceFunc function,
|
|
|
|
gpointer data,
|
|
|
|
GDestroyNotify notify)
|
|
|
|
{
|
2000-12-05 21:45:33 +01:00
|
|
|
GSource *source;
|
2000-12-07 21:29:58 +01:00
|
|
|
guint id;
|
2000-12-05 21:45:33 +01:00
|
|
|
|
1999-01-17 05:49:43 +01:00
|
|
|
g_return_val_if_fail (function != NULL, 0);
|
1998-12-02 15:55:27 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
source = g_idle_source_new ();
|
|
|
|
|
2003-08-07 20:19:23 +02:00
|
|
|
if (priority != G_PRIORITY_DEFAULT_IDLE)
|
2000-12-05 21:45:33 +01:00
|
|
|
g_source_set_priority (source, priority);
|
|
|
|
|
|
|
|
g_source_set_callback (source, function, data, notify);
|
2000-12-07 21:29:58 +01:00
|
|
|
id = g_source_attach (source, NULL);
|
|
|
|
g_source_unref (source);
|
|
|
|
|
|
|
|
return id;
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_idle_add:
|
|
|
|
* @function: function to call
|
|
|
|
* @data: data to pass to @function.
|
|
|
|
*
|
|
|
|
* 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
|
2001-12-16 20:31:36 +01:00
|
|
|
* returns %FALSE it is automatically removed from the list of event
|
2000-12-05 21:45:33 +01:00
|
|
|
* sources and will not be called again.
|
|
|
|
*
|
2008-09-26 16:05:03 +02:00
|
|
|
* This internally creates a main loop source using g_idle_source_new()
|
|
|
|
* and attaches it to the main loop context using g_source_attach().
|
|
|
|
* You can do these steps manually if you need greater control.
|
|
|
|
*
|
2006-01-05 22:07:55 +01:00
|
|
|
* Return value: the ID (greater than 0) of the event source.
|
2000-12-05 21:45:33 +01:00
|
|
|
**/
|
1998-12-02 15:55:27 +01:00
|
|
|
guint
|
|
|
|
g_idle_add (GSourceFunc function,
|
|
|
|
gpointer data)
|
|
|
|
{
|
1998-12-19 23:21:39 +01:00
|
|
|
return g_idle_add_full (G_PRIORITY_DEFAULT_IDLE, function, data, NULL);
|
1998-12-02 15:55:27 +01:00
|
|
|
}
|
1999-01-17 05:49:43 +01:00
|
|
|
|
2000-12-05 21:45:33 +01:00
|
|
|
/**
|
|
|
|
* g_idle_remove_by_data:
|
|
|
|
* @data: the data for the idle source's callback.
|
|
|
|
*
|
|
|
|
* Removes the idle function with the given data.
|
|
|
|
*
|
|
|
|
* Return value: %TRUE if an idle source was found and removed.
|
|
|
|
**/
|
1999-01-17 05:49:43 +01:00
|
|
|
gboolean
|
|
|
|
g_idle_remove_by_data (gpointer data)
|
|
|
|
{
|
2001-09-04 00:12:51 +02:00
|
|
|
return g_source_remove_by_funcs_user_data (&g_idle_funcs, data);
|
1999-01-17 05:49:43 +01:00
|
|
|
}
|
2010-10-03 23:26:37 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* g_main_context_invoke:
|
|
|
|
* @context: a #GMainContext, or %NULL
|
|
|
|
* @function: function to call
|
|
|
|
* @data: data to pass to @function
|
|
|
|
*
|
|
|
|
* Invokes a function in such a way that @context is owned during the
|
|
|
|
* invocation of @function.
|
|
|
|
*
|
|
|
|
* If @context is %NULL then the global default main context — as
|
|
|
|
* returned by g_main_context_default() — is used.
|
|
|
|
*
|
|
|
|
* If @context is owned by the current thread, @function is called
|
|
|
|
* directly. Otherwise, if @context is the thread-default main context
|
|
|
|
* of the current thread and g_main_context_acquire() succeeds, then
|
|
|
|
* @function is called and g_main_context_release() is called
|
|
|
|
* afterwards.
|
|
|
|
*
|
|
|
|
* In any other case, an idle source is created to call @function and
|
|
|
|
* that source is attached to @context (presumably to be run in another
|
|
|
|
* thread). The idle source is attached with #G_PRIORITY_DEFAULT
|
|
|
|
* priority. If you want a different priority, use
|
|
|
|
* g_main_context_invoke_full().
|
|
|
|
*
|
|
|
|
* Note that, as with normal idle functions, @function should probably
|
|
|
|
* return %FALSE. If it returns %TRUE, it will be continuously run in a
|
|
|
|
* loop (and may prevent this call from returning).
|
|
|
|
*
|
|
|
|
* Since: 2.28
|
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_main_context_invoke (GMainContext *context,
|
|
|
|
GSourceFunc function,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
g_main_context_invoke_full (context,
|
|
|
|
G_PRIORITY_DEFAULT,
|
|
|
|
function, data, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* g_main_context_invoke_full:
|
|
|
|
* @context: a #GMainContext, or %NULL
|
|
|
|
* @priority: the priority at which to run @function
|
|
|
|
* @function: function to call
|
|
|
|
* @data: data to pass to @function
|
|
|
|
* @notify: a function to call when @data is no longer in use, or %NULL.
|
|
|
|
*
|
|
|
|
* Invokes a function in such a way that @context is owned during the
|
|
|
|
* invocation of @function.
|
|
|
|
*
|
|
|
|
* This function is the same as g_main_context_invoke() except that it
|
|
|
|
* lets you specify the priority incase @function ends up being
|
|
|
|
* scheduled as an idle and also lets you give a #GDestroyNotify for @data.
|
|
|
|
*
|
|
|
|
* @notify should not assume that it is called from any particular
|
|
|
|
* thread or with any particular context acquired.
|
|
|
|
*
|
|
|
|
* Since: 2.28
|
|
|
|
**/
|
|
|
|
void
|
|
|
|
g_main_context_invoke_full (GMainContext *context,
|
|
|
|
gint priority,
|
|
|
|
GSourceFunc function,
|
|
|
|
gpointer data,
|
|
|
|
GDestroyNotify notify)
|
|
|
|
{
|
|
|
|
g_return_if_fail (function != NULL);
|
|
|
|
|
|
|
|
if (!context)
|
|
|
|
context = g_main_context_default ();
|
|
|
|
|
|
|
|
if (g_main_context_is_owner (context))
|
|
|
|
{
|
|
|
|
while (function (data));
|
|
|
|
if (notify != NULL)
|
|
|
|
notify (data);
|
|
|
|
}
|
|
|
|
|
|
|
|
else
|
|
|
|
{
|
|
|
|
GMainContext *thread_default;
|
|
|
|
|
|
|
|
thread_default = g_main_context_get_thread_default ();
|
|
|
|
|
|
|
|
if (!thread_default)
|
|
|
|
thread_default = g_main_context_default ();
|
|
|
|
|
|
|
|
if (thread_default == context && g_main_context_acquire (context))
|
|
|
|
{
|
|
|
|
while (function (data));
|
|
|
|
|
|
|
|
g_main_context_release (context);
|
|
|
|
|
|
|
|
if (notify != NULL)
|
|
|
|
notify (data);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
GSource *source;
|
|
|
|
|
|
|
|
source = g_idle_source_new ();
|
|
|
|
g_source_set_priority (source, priority);
|
|
|
|
g_source_set_callback (source, function, data, notify);
|
|
|
|
g_source_attach (source, context);
|
|
|
|
g_source_unref (source);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|