Update contrib.
1 /* -*- mode: C; c-file-style: "gnu" -*- */
2 /* dbus-gmain.c GLib main loop integration
4 * Copyright (C) 2002, 2003 CodeFactory AB
5 * Copyright (C) 2005 Red Hat, Inc.
6 * Portion Copyright © 2008 Nokia Corporation and/or its subsidiary(-ies). All rights reserved.
7 * Licensed under the Academic Free License version 2.1
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; either version 2 of the License, or
12 * (at your option) any later version.
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
21 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
29 #endif //__SYMBIAN32__
30 #include <dbus/dbus-glib.h>
31 #include <dbus/dbus-glib-lowlevel.h>
32 #include "dbus-gtest.h"
33 #include "dbus-gutils.h"
34 #include "dbus-gvalue.h"
35 #include "dbus-gobject.h"
36 #include "dbus-gvalue-utils.h"
37 #include "dbus-gsignature.h"
41 #define _(x) dgettext (GETTEXT_PACKAGE, x)
50 #include "libdbus_glib_wsd_solution.h"
53 * @defgroup DBusGLibInternals GLib bindings implementation details
54 * @ingroup DBusInternals
55 * @brief Implementation details of GLib bindings
62 * A GSource subclass for dispatching DBusConnection messages.
63 * We need this on top of the IO handlers, because sometimes
64 * there are messages to dispatch queued up but no IO pending.
68 GSource source; /**< the parent GSource */
69 DBusConnection *connection; /**< the connection to dispatch */
72 static gboolean message_queue_prepare (GSource *source,
74 static gboolean message_queue_check (GSource *source);
75 static gboolean message_queue_dispatch (GSource *source,
79 static const GSourceFuncs message_queue_funcs = {
80 message_queue_prepare,
82 message_queue_dispatch,
87 message_queue_prepare (GSource *source,
90 DBusConnection *connection = ((DBusGMessageQueue *)source)->connection;
94 return (dbus_connection_get_dispatch_status (connection) == DBUS_DISPATCH_DATA_REMAINS);
98 message_queue_check (GSource *source)
104 message_queue_dispatch (GSource *source,
105 GSourceFunc callback,
108 DBusConnection *connection = ((DBusGMessageQueue *)source)->connection;
110 dbus_connection_ref (connection);
112 /* Only dispatch once - we don't want to starve other GSource */
113 dbus_connection_dispatch (connection);
115 dbus_connection_unref (connection);
122 GMainContext *context; /**< the main context */
123 GSList *ios; /**< all IOHandler */
124 GSList *timeouts; /**< all TimeoutHandler */
125 DBusConnection *connection; /**< NULL if this is really for a server not a connection */
126 GSource *message_queue_source; /**< DBusGMessageQueue */
141 DBusTimeout *timeout;
146 GET_GLOBAL_VAR_FROM_TLS(_dbus_gmain_connection_slot,dbus_gmain,dbus_int32_t)
147 #define _dbus_gmain_connection_slot (*GET_DBUS_WSD_VAR_NAME(_dbus_gmain_connection_slot,dbus_gmain,g)())
149 GET_STATIC_VAR_FROM_TLS(server_slot,dbus_gmain,dbus_int32_t)
150 #define server_slot (*GET_DBUS_WSD_VAR_NAME(server_slot,dbus_gmain,s)())
154 dbus_int32_t _dbus_gmain_connection_slot = -1;
156 static dbus_int32_t server_slot = -1;
161 static ConnectionSetup*
162 connection_setup_new (GMainContext *context,
163 DBusConnection *connection)
167 cs = g_new0 (ConnectionSetup, 1);
169 g_assert (context != NULL);
171 cs->context = context;
172 g_main_context_ref (cs->context);
176 cs->connection = connection;
178 cs->message_queue_source = g_source_new ((GSourceFuncs *) &message_queue_funcs,
179 sizeof (DBusGMessageQueue));
180 ((DBusGMessageQueue*)cs->message_queue_source)->connection = connection;
181 g_source_attach (cs->message_queue_source, cs->context);
188 io_handler_source_finalized (gpointer data)
195 dbus_watch_set_data (handler->watch, NULL, NULL);
201 io_handler_destroy_source (void *data)
209 GSource *source = handler->source;
210 handler->source = NULL;
211 handler->cs->ios = g_slist_remove (handler->cs->ios, handler);
212 g_source_destroy (source);
213 g_source_unref (source);
218 io_handler_watch_freed (void *data)
224 handler->watch = NULL;
226 io_handler_destroy_source (handler);
230 io_handler_dispatch (GIOChannel *source,
231 GIOCondition condition,
235 guint dbus_condition = 0;
236 DBusConnection *connection;
240 connection = handler->cs->connection;
243 dbus_connection_ref (connection);
245 if (condition & G_IO_IN)
246 dbus_condition |= DBUS_WATCH_READABLE;
247 if (condition & G_IO_OUT)
248 dbus_condition |= DBUS_WATCH_WRITABLE;
249 if (condition & G_IO_ERR)
250 dbus_condition |= DBUS_WATCH_ERROR;
251 if (condition & G_IO_HUP)
252 dbus_condition |= DBUS_WATCH_HANGUP;
254 /* Note that we don't touch the handler after this, because
255 * dbus may have disabled the watch and thus killed the
258 dbus_watch_handle (handler->watch, dbus_condition);
262 dbus_connection_unref (connection);
268 connection_setup_add_watch (ConnectionSetup *cs,
272 GIOCondition condition;
276 if (!dbus_watch_get_enabled (watch))
279 g_assert (dbus_watch_get_data (watch) == NULL);
281 flags = dbus_watch_get_flags (watch);
283 condition = G_IO_ERR | G_IO_HUP;
284 if (flags & DBUS_WATCH_READABLE)
285 condition |= G_IO_IN;
286 if (flags & DBUS_WATCH_WRITABLE)
287 condition |= G_IO_OUT;
289 handler = g_new0 (IOHandler, 1);
291 handler->watch = watch;
293 channel = g_io_channel_unix_new (dbus_watch_get_fd (watch));
295 handler->source = g_io_create_watch (channel, condition);
296 g_source_set_callback (handler->source, (GSourceFunc) io_handler_dispatch, handler,
297 io_handler_source_finalized);
298 g_source_attach (handler->source, cs->context);
300 cs->ios = g_slist_prepend (cs->ios, handler);
302 dbus_watch_set_data (watch, handler, io_handler_watch_freed);
303 g_io_channel_unref (channel);
307 connection_setup_remove_watch (ConnectionSetup *cs,
312 handler = dbus_watch_get_data (watch);
317 io_handler_destroy_source (handler);
321 timeout_handler_source_finalized (gpointer data)
323 TimeoutHandler *handler;
327 if (handler->timeout)
328 dbus_timeout_set_data (handler->timeout, NULL, NULL);
334 timeout_handler_destroy_source (void *data)
336 TimeoutHandler *handler;
342 GSource *source = handler->source;
343 handler->source = NULL;
344 handler->cs->timeouts = g_slist_remove (handler->cs->timeouts, handler);
345 g_source_destroy (source);
346 g_source_unref (source);
351 timeout_handler_timeout_freed (void *data)
353 TimeoutHandler *handler;
357 handler->timeout = NULL;
359 timeout_handler_destroy_source (handler);
363 timeout_handler_dispatch (gpointer data)
365 TimeoutHandler *handler;
369 dbus_timeout_handle (handler->timeout);
375 connection_setup_add_timeout (ConnectionSetup *cs,
376 DBusTimeout *timeout)
378 TimeoutHandler *handler;
380 if (!dbus_timeout_get_enabled (timeout))
383 g_assert (dbus_timeout_get_data (timeout) == NULL);
385 handler = g_new0 (TimeoutHandler, 1);
387 handler->timeout = timeout;
389 handler->source = g_timeout_source_new (dbus_timeout_get_interval (timeout));
390 g_source_set_callback (handler->source, timeout_handler_dispatch, handler,
391 timeout_handler_source_finalized);
392 g_source_attach (handler->source, handler->cs->context);
394 cs->timeouts = g_slist_prepend (cs->timeouts, handler);
396 dbus_timeout_set_data (timeout, handler, timeout_handler_timeout_freed);
400 connection_setup_remove_timeout (ConnectionSetup *cs,
401 DBusTimeout *timeout)
403 TimeoutHandler *handler;
405 handler = dbus_timeout_get_data (timeout);
410 timeout_handler_destroy_source (handler);
414 connection_setup_free (ConnectionSetup *cs)
417 io_handler_destroy_source (cs->ios->data);
420 timeout_handler_destroy_source (cs->timeouts->data);
422 if (cs->message_queue_source)
426 source = cs->message_queue_source;
427 cs->message_queue_source = NULL;
429 g_source_destroy (source);
430 g_source_unref (source);
433 g_main_context_unref (cs->context);
438 add_watch (DBusWatch *watch,
445 connection_setup_add_watch (cs, watch);
451 remove_watch (DBusWatch *watch,
458 connection_setup_remove_watch (cs, watch);
462 watch_toggled (DBusWatch *watch,
465 /* Because we just exit on OOM, enable/disable is
466 * no different from add/remove
468 if (dbus_watch_get_enabled (watch))
469 add_watch (watch, data);
471 remove_watch (watch, data);
475 add_timeout (DBusTimeout *timeout,
482 if (!dbus_timeout_get_enabled (timeout))
485 connection_setup_add_timeout (cs, timeout);
491 remove_timeout (DBusTimeout *timeout,
498 connection_setup_remove_timeout (cs, timeout);
502 timeout_toggled (DBusTimeout *timeout,
505 /* Because we just exit on OOM, enable/disable is
506 * no different from add/remove
508 if (dbus_timeout_get_enabled (timeout))
509 add_timeout (timeout, data);
511 remove_timeout (timeout, data);
515 wakeup_main (void *data)
517 ConnectionSetup *cs = data;
519 g_main_context_wakeup (cs->context);
523 /* Move to a new context */
524 static ConnectionSetup*
525 connection_setup_new_from_old (GMainContext *context,
526 ConnectionSetup *old)
531 g_assert (old->context != context);
533 cs = connection_setup_new (context, old->connection);
538 IOHandler *handler = tmp->data;
540 connection_setup_add_watch (cs, handler->watch);
548 TimeoutHandler *handler = tmp->data;
550 connection_setup_add_timeout (cs, handler->timeout);
558 /** @} */ /* End of GLib bindings internals */
560 /** @addtogroup DBusGLib
565 * dbus_connection_setup_with_g_main:
566 * @connection: the connection
567 * @context: the #GMainContext or #NULL for default context
569 * Sets the watch and timeout functions of a #DBusConnection
570 * to integrate the connection with the GLib main loop.
571 * Pass in #NULL for the #GMainContext unless you're
572 * doing something specialized.
574 * If called twice for the same context, does nothing the second
575 * time. If called once with context A and once with context B,
576 * context B replaces context A as the context monitoring the
583 dbus_connection_setup_with_g_main (DBusConnection *connection,
584 GMainContext *context)
586 ConnectionSetup *old_setup;
589 /* FIXME we never free the slot, so its refcount just keeps growing,
590 * which is kind of broken.
592 dbus_connection_allocate_data_slot (&_dbus_gmain_connection_slot);
593 if (_dbus_gmain_connection_slot < 0)
597 context = g_main_context_default ();
601 old_setup = dbus_connection_get_data (connection, _dbus_gmain_connection_slot);
602 if (old_setup != NULL)
604 if (old_setup->context == context)
605 return; /* nothing to do */
607 cs = connection_setup_new_from_old (context, old_setup);
609 /* Nuke the old setup */
610 dbus_connection_set_data (connection, _dbus_gmain_connection_slot, NULL, NULL);
615 cs = connection_setup_new (context, connection);
617 if (!dbus_connection_set_data (connection, _dbus_gmain_connection_slot, cs,
618 (DBusFreeFunction)connection_setup_free))
621 if (!dbus_connection_set_watch_functions (connection,
628 if (!dbus_connection_set_timeout_functions (connection,
635 dbus_connection_set_wakeup_main_function (connection,
642 g_error ("Not enough memory to set up DBusConnection for use with GLib");
646 * dbus_server_setup_with_g_main:
647 * @server: the server
648 * @context: the #GMainContext or #NULL for default
650 * Sets the watch and timeout functions of a #DBusServer
651 * to integrate the server with the GLib main loop.
652 * In most cases the context argument should be #NULL.
654 * If called twice for the same context, does nothing the second
655 * time. If called once with context A and once with context B,
656 * context B replaces context A as the context monitoring the
663 dbus_server_setup_with_g_main (DBusServer *server,
664 GMainContext *context)
666 ConnectionSetup *old_setup;
669 /* FIXME we never free the slot, so its refcount just keeps growing,
670 * which is kind of broken.
672 dbus_server_allocate_data_slot (&server_slot);
677 context = g_main_context_default ();
681 old_setup = dbus_server_get_data (server, server_slot);
682 if (old_setup != NULL)
684 if (old_setup->context == context)
685 return; /* nothing to do */
687 cs = connection_setup_new_from_old (context, old_setup);
689 /* Nuke the old setup */
690 dbus_server_set_data (server, server_slot, NULL, NULL);
695 cs = connection_setup_new (context, NULL);
697 if (!dbus_server_set_data (server, server_slot, cs,
698 (DBusFreeFunction)connection_setup_free))
701 if (!dbus_server_set_watch_functions (server,
708 if (!dbus_server_set_timeout_functions (server,
718 g_error ("Not enough memory to set up DBusServer for use with GLib");
722 * dbus_g_connection_open:
723 * @address: address of the connection to open
724 * @error: address where an error can be returned.
726 * Returns a connection to the given address.
728 * (Internally, calls dbus_connection_open() then calls
729 * dbus_connection_setup_with_g_main() on the result.)
731 * Returns: a DBusConnection
737 dbus_g_connection_open (const gchar *address,
740 DBusConnection *connection;
743 g_return_val_if_fail (error == NULL || *error == NULL, NULL);
745 _dbus_g_value_types_init ();
747 dbus_error_init (&derror);
749 connection = dbus_connection_open (address, &derror);
750 if (connection == NULL)
752 dbus_set_g_error (error, &derror);
753 dbus_error_free (&derror);
757 /* does nothing if it's already been done */
758 dbus_connection_setup_with_g_main (connection, NULL);
760 return DBUS_G_CONNECTION_FROM_CONNECTION (connection);
766 * @error: address where an error can be returned.
768 * Returns a connection to the given bus. The connection is a global variable
769 * shared with other callers of this function.
771 * (Internally, calls dbus_bus_get() then calls
772 * dbus_connection_setup_with_g_main() on the result.)
774 * Returns: a DBusConnection
780 dbus_g_bus_get (DBusBusType type,
783 DBusConnection *connection;
786 g_return_val_if_fail (error == NULL || *error == NULL, NULL);
788 _dbus_g_value_types_init ();
790 dbus_error_init (&derror);
792 connection = dbus_bus_get (type, &derror);
793 if (connection == NULL)
795 dbus_set_g_error (error, &derror);
796 dbus_error_free (&derror);
800 /* does nothing if it's already been done */
801 dbus_connection_setup_with_g_main (connection, NULL);
803 return DBUS_G_CONNECTION_FROM_CONNECTION (connection);
806 /** @} */ /* end of public API */
808 #ifdef DBUS_BUILD_TESTS
811 * @ingroup DBusGLibInternals
812 * Unit test for GLib main loop integration
813 * Returns: #TRUE on success.
819 _dbus_gmain_test (const char *test_data_dir)
825 _dbus_g_value_types_init ();
827 rectype = dbus_g_type_get_collection ("GArray", G_TYPE_UINT);
828 g_assert (rectype != G_TYPE_INVALID);
829 g_assert (!strcmp (g_type_name (rectype), "GArray_guint_"));
831 type = _dbus_gtype_from_signature ("au", TRUE);
832 g_assert (type == rectype);
834 rectype = dbus_g_type_get_map ("GHashTable", G_TYPE_STRING, G_TYPE_STRING);
835 g_assert (rectype != G_TYPE_INVALID);
836 g_assert (!strcmp (g_type_name (rectype), "GHashTable_gchararray+gchararray_"));
838 type = _dbus_gtype_from_signature ("a{ss}", TRUE);
839 g_assert (type == rectype);
841 type = _dbus_gtype_from_signature ("o", FALSE);
842 g_assert (type == DBUS_TYPE_G_OBJECT_PATH);
843 type = _dbus_gtype_from_signature ("o", TRUE);
844 g_assert (type == DBUS_TYPE_G_OBJECT_PATH);
849 #endif /* DBUS_BUILD_TESTS */