1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/ossrv/ofdbus/dbus-glib/dbus/dbus-gmain.c Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,849 @@
1.4 +/* -*- mode: C; c-file-style: "gnu" -*- */
1.5 +/* dbus-gmain.c GLib main loop integration
1.6 + *
1.7 + * Copyright (C) 2002, 2003 CodeFactory AB
1.8 + * Copyright (C) 2005 Red Hat, Inc.
1.9 + * Portion Copyright © 2008 Nokia Corporation and/or its subsidiary(-ies). All rights reserved.
1.10 + * Licensed under the Academic Free License version 2.1
1.11 + *
1.12 + * This program is free software; you can redistribute it and/or modify
1.13 + * it under the terms of the GNU General Public License as published by
1.14 + * the Free Software Foundation; either version 2 of the License, or
1.15 + * (at your option) any later version.
1.16 + *
1.17 + * This program is distributed in the hope that it will be useful,
1.18 + * but WITHOUT ANY WARRANTY; without even the implied warranty of
1.19 + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
1.20 + * GNU General Public License for more details.
1.21 + *
1.22 + * You should have received a copy of the GNU General Public License
1.23 + * along with this program; if not, write to the Free Software
1.24 + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
1.25 + *
1.26 + */
1.27 +
1.28 +#ifndef __SYMBIAN32__
1.29 +#include <config.h>
1.30 +#else
1.31 +#include "config.h"
1.32 +#endif //__SYMBIAN32__
1.33 +#include <dbus/dbus-glib.h>
1.34 +#include <dbus/dbus-glib-lowlevel.h>
1.35 +#include "dbus-gtest.h"
1.36 +#include "dbus-gutils.h"
1.37 +#include "dbus-gvalue.h"
1.38 +#include "dbus-gobject.h"
1.39 +#include "dbus-gvalue-utils.h"
1.40 +#include "dbus-gsignature.h"
1.41 +#include <string.h>
1.42 +#ifndef __SYMBIAN32__
1.43 +#include <libintl.h>
1.44 +#define _(x) dgettext (GETTEXT_PACKAGE, x)
1.45 +#define N_(x) x
1.46 +#else
1.47 +
1.48 +#define _(x) x
1.49 +#define N_(x) x
1.50 +#endif
1.51 +
1.52 +#ifdef __SYMBIAN32__
1.53 +#include "libdbus_glib_wsd_solution.h"
1.54 +#endif
1.55 +/**
1.56 + * @defgroup DBusGLibInternals GLib bindings implementation details
1.57 + * @ingroup DBusInternals
1.58 + * @brief Implementation details of GLib bindings
1.59 + *
1.60 + * @{
1.61 + */
1.62 +
1.63 +/**
1.64 + * DBusGMessageQueue:
1.65 + * A GSource subclass for dispatching DBusConnection messages.
1.66 + * We need this on top of the IO handlers, because sometimes
1.67 + * there are messages to dispatch queued up but no IO pending.
1.68 + */
1.69 +typedef struct
1.70 +{
1.71 + GSource source; /**< the parent GSource */
1.72 + DBusConnection *connection; /**< the connection to dispatch */
1.73 +} DBusGMessageQueue;
1.74 +
1.75 +static gboolean message_queue_prepare (GSource *source,
1.76 + gint *timeout);
1.77 +static gboolean message_queue_check (GSource *source);
1.78 +static gboolean message_queue_dispatch (GSource *source,
1.79 + GSourceFunc callback,
1.80 + gpointer user_data);
1.81 +
1.82 +static const GSourceFuncs message_queue_funcs = {
1.83 + message_queue_prepare,
1.84 + message_queue_check,
1.85 + message_queue_dispatch,
1.86 + NULL
1.87 +};
1.88 +
1.89 +static gboolean
1.90 +message_queue_prepare (GSource *source,
1.91 + gint *timeout)
1.92 +{
1.93 + DBusConnection *connection = ((DBusGMessageQueue *)source)->connection;
1.94 +
1.95 + *timeout = -1;
1.96 +
1.97 + return (dbus_connection_get_dispatch_status (connection) == DBUS_DISPATCH_DATA_REMAINS);
1.98 +}
1.99 +
1.100 +static gboolean
1.101 +message_queue_check (GSource *source)
1.102 +{
1.103 + return FALSE;
1.104 +}
1.105 +
1.106 +static gboolean
1.107 +message_queue_dispatch (GSource *source,
1.108 + GSourceFunc callback,
1.109 + gpointer user_data)
1.110 +{
1.111 + DBusConnection *connection = ((DBusGMessageQueue *)source)->connection;
1.112 +
1.113 + dbus_connection_ref (connection);
1.114 +
1.115 + /* Only dispatch once - we don't want to starve other GSource */
1.116 + dbus_connection_dispatch (connection);
1.117 +
1.118 + dbus_connection_unref (connection);
1.119 +
1.120 + return TRUE;
1.121 +}
1.122 +
1.123 +typedef struct
1.124 +{
1.125 + GMainContext *context; /**< the main context */
1.126 + GSList *ios; /**< all IOHandler */
1.127 + GSList *timeouts; /**< all TimeoutHandler */
1.128 + DBusConnection *connection; /**< NULL if this is really for a server not a connection */
1.129 + GSource *message_queue_source; /**< DBusGMessageQueue */
1.130 +} ConnectionSetup;
1.131 +
1.132 +
1.133 +typedef struct
1.134 +{
1.135 + ConnectionSetup *cs;
1.136 + GSource *source;
1.137 + DBusWatch *watch;
1.138 +} IOHandler;
1.139 +
1.140 +typedef struct
1.141 +{
1.142 + ConnectionSetup *cs;
1.143 + GSource *source;
1.144 + DBusTimeout *timeout;
1.145 +} TimeoutHandler;
1.146 +
1.147 +
1.148 +#if EMULATOR
1.149 +GET_GLOBAL_VAR_FROM_TLS(_dbus_gmain_connection_slot,dbus_gmain,dbus_int32_t)
1.150 +#define _dbus_gmain_connection_slot (*GET_DBUS_WSD_VAR_NAME(_dbus_gmain_connection_slot,dbus_gmain,g)())
1.151 +
1.152 +GET_STATIC_VAR_FROM_TLS(server_slot,dbus_gmain,dbus_int32_t)
1.153 +#define server_slot (*GET_DBUS_WSD_VAR_NAME(server_slot,dbus_gmain,s)())
1.154 +
1.155 +#else
1.156 +
1.157 +dbus_int32_t _dbus_gmain_connection_slot = -1;
1.158 +
1.159 +static dbus_int32_t server_slot = -1;
1.160 +#endif
1.161 +
1.162 +
1.163 +
1.164 +static ConnectionSetup*
1.165 +connection_setup_new (GMainContext *context,
1.166 + DBusConnection *connection)
1.167 +{
1.168 + ConnectionSetup *cs;
1.169 +
1.170 + cs = g_new0 (ConnectionSetup, 1);
1.171 +
1.172 + g_assert (context != NULL);
1.173 +
1.174 + cs->context = context;
1.175 + g_main_context_ref (cs->context);
1.176 +
1.177 + if (connection)
1.178 + {
1.179 + cs->connection = connection;
1.180 +
1.181 + cs->message_queue_source = g_source_new ((GSourceFuncs *) &message_queue_funcs,
1.182 + sizeof (DBusGMessageQueue));
1.183 + ((DBusGMessageQueue*)cs->message_queue_source)->connection = connection;
1.184 + g_source_attach (cs->message_queue_source, cs->context);
1.185 + }
1.186 +
1.187 + return cs;
1.188 +}
1.189 +
1.190 +static void
1.191 +io_handler_source_finalized (gpointer data)
1.192 +{
1.193 + IOHandler *handler;
1.194 +
1.195 + handler = data;
1.196 +
1.197 + if (handler->watch)
1.198 + dbus_watch_set_data (handler->watch, NULL, NULL);
1.199 +
1.200 + g_free (handler);
1.201 +}
1.202 +
1.203 +static void
1.204 +io_handler_destroy_source (void *data)
1.205 +{
1.206 + IOHandler *handler;
1.207 +
1.208 + handler = data;
1.209 +
1.210 + if (handler->source)
1.211 + {
1.212 + GSource *source = handler->source;
1.213 + handler->source = NULL;
1.214 + handler->cs->ios = g_slist_remove (handler->cs->ios, handler);
1.215 + g_source_destroy (source);
1.216 + g_source_unref (source);
1.217 + }
1.218 +}
1.219 +
1.220 +static void
1.221 +io_handler_watch_freed (void *data)
1.222 +{
1.223 + IOHandler *handler;
1.224 +
1.225 + handler = data;
1.226 +
1.227 + handler->watch = NULL;
1.228 +
1.229 + io_handler_destroy_source (handler);
1.230 +}
1.231 +
1.232 +static gboolean
1.233 +io_handler_dispatch (GIOChannel *source,
1.234 + GIOCondition condition,
1.235 + gpointer data)
1.236 +{
1.237 + IOHandler *handler;
1.238 + guint dbus_condition = 0;
1.239 + DBusConnection *connection;
1.240 +
1.241 + handler = data;
1.242 +
1.243 + connection = handler->cs->connection;
1.244 +
1.245 + if (connection)
1.246 + dbus_connection_ref (connection);
1.247 +
1.248 + if (condition & G_IO_IN)
1.249 + dbus_condition |= DBUS_WATCH_READABLE;
1.250 + if (condition & G_IO_OUT)
1.251 + dbus_condition |= DBUS_WATCH_WRITABLE;
1.252 + if (condition & G_IO_ERR)
1.253 + dbus_condition |= DBUS_WATCH_ERROR;
1.254 + if (condition & G_IO_HUP)
1.255 + dbus_condition |= DBUS_WATCH_HANGUP;
1.256 +
1.257 + /* Note that we don't touch the handler after this, because
1.258 + * dbus may have disabled the watch and thus killed the
1.259 + * handler.
1.260 + */
1.261 + dbus_watch_handle (handler->watch, dbus_condition);
1.262 + handler = NULL;
1.263 +
1.264 + if (connection)
1.265 + dbus_connection_unref (connection);
1.266 +
1.267 + return TRUE;
1.268 +}
1.269 +
1.270 +static void
1.271 +connection_setup_add_watch (ConnectionSetup *cs,
1.272 + DBusWatch *watch)
1.273 +{
1.274 + guint flags;
1.275 + GIOCondition condition;
1.276 + GIOChannel *channel;
1.277 + IOHandler *handler;
1.278 +
1.279 + if (!dbus_watch_get_enabled (watch))
1.280 + return;
1.281 +
1.282 + g_assert (dbus_watch_get_data (watch) == NULL);
1.283 +
1.284 + flags = dbus_watch_get_flags (watch);
1.285 +
1.286 + condition = G_IO_ERR | G_IO_HUP;
1.287 + if (flags & DBUS_WATCH_READABLE)
1.288 + condition |= G_IO_IN;
1.289 + if (flags & DBUS_WATCH_WRITABLE)
1.290 + condition |= G_IO_OUT;
1.291 +
1.292 + handler = g_new0 (IOHandler, 1);
1.293 + handler->cs = cs;
1.294 + handler->watch = watch;
1.295 +
1.296 + channel = g_io_channel_unix_new (dbus_watch_get_fd (watch));
1.297 +
1.298 + handler->source = g_io_create_watch (channel, condition);
1.299 + g_source_set_callback (handler->source, (GSourceFunc) io_handler_dispatch, handler,
1.300 + io_handler_source_finalized);
1.301 + g_source_attach (handler->source, cs->context);
1.302 +
1.303 + cs->ios = g_slist_prepend (cs->ios, handler);
1.304 +
1.305 + dbus_watch_set_data (watch, handler, io_handler_watch_freed);
1.306 + g_io_channel_unref (channel);
1.307 +}
1.308 +
1.309 +static void
1.310 +connection_setup_remove_watch (ConnectionSetup *cs,
1.311 + DBusWatch *watch)
1.312 +{
1.313 + IOHandler *handler;
1.314 +
1.315 + handler = dbus_watch_get_data (watch);
1.316 +
1.317 + if (handler == NULL)
1.318 + return;
1.319 +
1.320 + io_handler_destroy_source (handler);
1.321 +}
1.322 +
1.323 +static void
1.324 +timeout_handler_source_finalized (gpointer data)
1.325 +{
1.326 + TimeoutHandler *handler;
1.327 +
1.328 + handler = data;
1.329 +
1.330 + if (handler->timeout)
1.331 + dbus_timeout_set_data (handler->timeout, NULL, NULL);
1.332 +
1.333 + g_free (handler);
1.334 +}
1.335 +
1.336 +static void
1.337 +timeout_handler_destroy_source (void *data)
1.338 +{
1.339 + TimeoutHandler *handler;
1.340 +
1.341 + handler = data;
1.342 +
1.343 + if (handler->source)
1.344 + {
1.345 + GSource *source = handler->source;
1.346 + handler->source = NULL;
1.347 + handler->cs->timeouts = g_slist_remove (handler->cs->timeouts, handler);
1.348 + g_source_destroy (source);
1.349 + g_source_unref (source);
1.350 + }
1.351 +}
1.352 +
1.353 +static void
1.354 +timeout_handler_timeout_freed (void *data)
1.355 +{
1.356 + TimeoutHandler *handler;
1.357 +
1.358 + handler = data;
1.359 +
1.360 + handler->timeout = NULL;
1.361 +
1.362 + timeout_handler_destroy_source (handler);
1.363 +}
1.364 +
1.365 +static gboolean
1.366 +timeout_handler_dispatch (gpointer data)
1.367 +{
1.368 + TimeoutHandler *handler;
1.369 +
1.370 + handler = data;
1.371 +
1.372 + dbus_timeout_handle (handler->timeout);
1.373 +
1.374 + return TRUE;
1.375 +}
1.376 +
1.377 +static void
1.378 +connection_setup_add_timeout (ConnectionSetup *cs,
1.379 + DBusTimeout *timeout)
1.380 +{
1.381 + TimeoutHandler *handler;
1.382 +
1.383 + if (!dbus_timeout_get_enabled (timeout))
1.384 + return;
1.385 +
1.386 + g_assert (dbus_timeout_get_data (timeout) == NULL);
1.387 +
1.388 + handler = g_new0 (TimeoutHandler, 1);
1.389 + handler->cs = cs;
1.390 + handler->timeout = timeout;
1.391 +
1.392 + handler->source = g_timeout_source_new (dbus_timeout_get_interval (timeout));
1.393 + g_source_set_callback (handler->source, timeout_handler_dispatch, handler,
1.394 + timeout_handler_source_finalized);
1.395 + g_source_attach (handler->source, handler->cs->context);
1.396 +
1.397 + cs->timeouts = g_slist_prepend (cs->timeouts, handler);
1.398 +
1.399 + dbus_timeout_set_data (timeout, handler, timeout_handler_timeout_freed);
1.400 +}
1.401 +
1.402 +static void
1.403 +connection_setup_remove_timeout (ConnectionSetup *cs,
1.404 + DBusTimeout *timeout)
1.405 +{
1.406 + TimeoutHandler *handler;
1.407 +
1.408 + handler = dbus_timeout_get_data (timeout);
1.409 +
1.410 + if (handler == NULL)
1.411 + return;
1.412 +
1.413 + timeout_handler_destroy_source (handler);
1.414 +}
1.415 +
1.416 +static void
1.417 +connection_setup_free (ConnectionSetup *cs)
1.418 +{
1.419 + while (cs->ios)
1.420 + io_handler_destroy_source (cs->ios->data);
1.421 +
1.422 + while (cs->timeouts)
1.423 + timeout_handler_destroy_source (cs->timeouts->data);
1.424 +
1.425 + if (cs->message_queue_source)
1.426 + {
1.427 + GSource *source;
1.428 +
1.429 + source = cs->message_queue_source;
1.430 + cs->message_queue_source = NULL;
1.431 +
1.432 + g_source_destroy (source);
1.433 + g_source_unref (source);
1.434 + }
1.435 +
1.436 + g_main_context_unref (cs->context);
1.437 + g_free (cs);
1.438 +}
1.439 +
1.440 +static dbus_bool_t
1.441 +add_watch (DBusWatch *watch,
1.442 + gpointer data)
1.443 +{
1.444 + ConnectionSetup *cs;
1.445 +
1.446 + cs = data;
1.447 +
1.448 + connection_setup_add_watch (cs, watch);
1.449 +
1.450 + return TRUE;
1.451 +}
1.452 +
1.453 +static void
1.454 +remove_watch (DBusWatch *watch,
1.455 + gpointer data)
1.456 +{
1.457 + ConnectionSetup *cs;
1.458 +
1.459 + cs = data;
1.460 +
1.461 + connection_setup_remove_watch (cs, watch);
1.462 +}
1.463 +
1.464 +static void
1.465 +watch_toggled (DBusWatch *watch,
1.466 + void *data)
1.467 +{
1.468 + /* Because we just exit on OOM, enable/disable is
1.469 + * no different from add/remove
1.470 + */
1.471 + if (dbus_watch_get_enabled (watch))
1.472 + add_watch (watch, data);
1.473 + else
1.474 + remove_watch (watch, data);
1.475 +}
1.476 +
1.477 +static dbus_bool_t
1.478 +add_timeout (DBusTimeout *timeout,
1.479 + void *data)
1.480 +{
1.481 + ConnectionSetup *cs;
1.482 +
1.483 + cs = data;
1.484 +
1.485 + if (!dbus_timeout_get_enabled (timeout))
1.486 + return TRUE;
1.487 +
1.488 + connection_setup_add_timeout (cs, timeout);
1.489 +
1.490 + return TRUE;
1.491 +}
1.492 +
1.493 +static void
1.494 +remove_timeout (DBusTimeout *timeout,
1.495 + void *data)
1.496 +{
1.497 + ConnectionSetup *cs;
1.498 +
1.499 + cs = data;
1.500 +
1.501 + connection_setup_remove_timeout (cs, timeout);
1.502 +}
1.503 +
1.504 +static void
1.505 +timeout_toggled (DBusTimeout *timeout,
1.506 + void *data)
1.507 +{
1.508 + /* Because we just exit on OOM, enable/disable is
1.509 + * no different from add/remove
1.510 + */
1.511 + if (dbus_timeout_get_enabled (timeout))
1.512 + add_timeout (timeout, data);
1.513 + else
1.514 + remove_timeout (timeout, data);
1.515 +}
1.516 +
1.517 +static void
1.518 +wakeup_main (void *data)
1.519 +{
1.520 + ConnectionSetup *cs = data;
1.521 +
1.522 + g_main_context_wakeup (cs->context);
1.523 +}
1.524 +
1.525 +
1.526 +/* Move to a new context */
1.527 +static ConnectionSetup*
1.528 +connection_setup_new_from_old (GMainContext *context,
1.529 + ConnectionSetup *old)
1.530 +{
1.531 + GSList *tmp;
1.532 + ConnectionSetup *cs;
1.533 +
1.534 + g_assert (old->context != context);
1.535 +
1.536 + cs = connection_setup_new (context, old->connection);
1.537 +
1.538 + tmp = old->ios;
1.539 + while (tmp != NULL)
1.540 + {
1.541 + IOHandler *handler = tmp->data;
1.542 +
1.543 + connection_setup_add_watch (cs, handler->watch);
1.544 +
1.545 + tmp = tmp->next;
1.546 + }
1.547 +
1.548 + tmp = old->timeouts;
1.549 + while (tmp != NULL)
1.550 + {
1.551 + TimeoutHandler *handler = tmp->data;
1.552 +
1.553 + connection_setup_add_timeout (cs, handler->timeout);
1.554 +
1.555 + tmp = tmp->next;
1.556 + }
1.557 +
1.558 + return cs;
1.559 +}
1.560 +
1.561 +/** @} */ /* End of GLib bindings internals */
1.562 +
1.563 +/** @addtogroup DBusGLib
1.564 + * @{
1.565 + */
1.566 +
1.567 +/**
1.568 + * dbus_connection_setup_with_g_main:
1.569 + * @connection: the connection
1.570 + * @context: the #GMainContext or #NULL for default context
1.571 + *
1.572 + * Sets the watch and timeout functions of a #DBusConnection
1.573 + * to integrate the connection with the GLib main loop.
1.574 + * Pass in #NULL for the #GMainContext unless you're
1.575 + * doing something specialized.
1.576 + *
1.577 + * If called twice for the same context, does nothing the second
1.578 + * time. If called once with context A and once with context B,
1.579 + * context B replaces context A as the context monitoring the
1.580 + * connection.
1.581 + */
1.582 + #ifdef __SYMBIAN32__
1.583 +EXPORT_C
1.584 +#endif
1.585 +void
1.586 +dbus_connection_setup_with_g_main (DBusConnection *connection,
1.587 + GMainContext *context)
1.588 +{
1.589 + ConnectionSetup *old_setup;
1.590 + ConnectionSetup *cs;
1.591 +
1.592 + /* FIXME we never free the slot, so its refcount just keeps growing,
1.593 + * which is kind of broken.
1.594 + */
1.595 + dbus_connection_allocate_data_slot (&_dbus_gmain_connection_slot);
1.596 + if (_dbus_gmain_connection_slot < 0)
1.597 + goto nomem;
1.598 +
1.599 + if (context == NULL)
1.600 + context = g_main_context_default ();
1.601 +
1.602 + cs = NULL;
1.603 +
1.604 + old_setup = dbus_connection_get_data (connection, _dbus_gmain_connection_slot);
1.605 + if (old_setup != NULL)
1.606 + {
1.607 + if (old_setup->context == context)
1.608 + return; /* nothing to do */
1.609 +
1.610 + cs = connection_setup_new_from_old (context, old_setup);
1.611 +
1.612 + /* Nuke the old setup */
1.613 + dbus_connection_set_data (connection, _dbus_gmain_connection_slot, NULL, NULL);
1.614 + old_setup = NULL;
1.615 + }
1.616 +
1.617 + if (cs == NULL)
1.618 + cs = connection_setup_new (context, connection);
1.619 +
1.620 + if (!dbus_connection_set_data (connection, _dbus_gmain_connection_slot, cs,
1.621 + (DBusFreeFunction)connection_setup_free))
1.622 + goto nomem;
1.623 +
1.624 + if (!dbus_connection_set_watch_functions (connection,
1.625 + add_watch,
1.626 + remove_watch,
1.627 + watch_toggled,
1.628 + cs, NULL))
1.629 + goto nomem;
1.630 +
1.631 + if (!dbus_connection_set_timeout_functions (connection,
1.632 + add_timeout,
1.633 + remove_timeout,
1.634 + timeout_toggled,
1.635 + cs, NULL))
1.636 + goto nomem;
1.637 +
1.638 + dbus_connection_set_wakeup_main_function (connection,
1.639 + wakeup_main,
1.640 + cs, NULL);
1.641 +
1.642 + return;
1.643 +
1.644 + nomem:
1.645 + g_error ("Not enough memory to set up DBusConnection for use with GLib");
1.646 +}
1.647 +
1.648 +/**
1.649 + * dbus_server_setup_with_g_main:
1.650 + * @server: the server
1.651 + * @context: the #GMainContext or #NULL for default
1.652 + *
1.653 + * Sets the watch and timeout functions of a #DBusServer
1.654 + * to integrate the server with the GLib main loop.
1.655 + * In most cases the context argument should be #NULL.
1.656 + *
1.657 + * If called twice for the same context, does nothing the second
1.658 + * time. If called once with context A and once with context B,
1.659 + * context B replaces context A as the context monitoring the
1.660 + * connection.
1.661 + */
1.662 + #ifdef __SYMBIAN32__
1.663 + EXPORT_C
1.664 + #endif
1.665 +void
1.666 +dbus_server_setup_with_g_main (DBusServer *server,
1.667 + GMainContext *context)
1.668 +{
1.669 + ConnectionSetup *old_setup;
1.670 + ConnectionSetup *cs;
1.671 +
1.672 + /* FIXME we never free the slot, so its refcount just keeps growing,
1.673 + * which is kind of broken.
1.674 + */
1.675 + dbus_server_allocate_data_slot (&server_slot);
1.676 + if (server_slot < 0)
1.677 + goto nomem;
1.678 +
1.679 + if (context == NULL)
1.680 + context = g_main_context_default ();
1.681 +
1.682 + cs = NULL;
1.683 +
1.684 + old_setup = dbus_server_get_data (server, server_slot);
1.685 + if (old_setup != NULL)
1.686 + {
1.687 + if (old_setup->context == context)
1.688 + return; /* nothing to do */
1.689 +
1.690 + cs = connection_setup_new_from_old (context, old_setup);
1.691 +
1.692 + /* Nuke the old setup */
1.693 + dbus_server_set_data (server, server_slot, NULL, NULL);
1.694 + old_setup = NULL;
1.695 + }
1.696 +
1.697 + if (cs == NULL)
1.698 + cs = connection_setup_new (context, NULL);
1.699 +
1.700 + if (!dbus_server_set_data (server, server_slot, cs,
1.701 + (DBusFreeFunction)connection_setup_free))
1.702 + goto nomem;
1.703 +
1.704 + if (!dbus_server_set_watch_functions (server,
1.705 + add_watch,
1.706 + remove_watch,
1.707 + watch_toggled,
1.708 + cs, NULL))
1.709 + goto nomem;
1.710 +
1.711 + if (!dbus_server_set_timeout_functions (server,
1.712 + add_timeout,
1.713 + remove_timeout,
1.714 + timeout_toggled,
1.715 + cs, NULL))
1.716 + goto nomem;
1.717 +
1.718 + return;
1.719 +
1.720 + nomem:
1.721 + g_error ("Not enough memory to set up DBusServer for use with GLib");
1.722 +}
1.723 +
1.724 +/**
1.725 + * dbus_g_connection_open:
1.726 + * @address: address of the connection to open
1.727 + * @error: address where an error can be returned.
1.728 + *
1.729 + * Returns a connection to the given address.
1.730 + *
1.731 + * (Internally, calls dbus_connection_open() then calls
1.732 + * dbus_connection_setup_with_g_main() on the result.)
1.733 + *
1.734 + * Returns: a DBusConnection
1.735 + */
1.736 + #ifdef __SYMBIAN32__
1.737 + EXPORT_C
1.738 + #endif
1.739 +DBusGConnection*
1.740 +dbus_g_connection_open (const gchar *address,
1.741 + GError **error)
1.742 +{
1.743 + DBusConnection *connection;
1.744 + DBusError derror;
1.745 +
1.746 + g_return_val_if_fail (error == NULL || *error == NULL, NULL);
1.747 +
1.748 + _dbus_g_value_types_init ();
1.749 +
1.750 + dbus_error_init (&derror);
1.751 +
1.752 + connection = dbus_connection_open (address, &derror);
1.753 + if (connection == NULL)
1.754 + {
1.755 + dbus_set_g_error (error, &derror);
1.756 + dbus_error_free (&derror);
1.757 + return NULL;
1.758 + }
1.759 +
1.760 + /* does nothing if it's already been done */
1.761 + dbus_connection_setup_with_g_main (connection, NULL);
1.762 +
1.763 + return DBUS_G_CONNECTION_FROM_CONNECTION (connection);
1.764 +}
1.765 +
1.766 +/**
1.767 + * dbus_g_bus_get:
1.768 + * @type: bus type
1.769 + * @error: address where an error can be returned.
1.770 + *
1.771 + * Returns a connection to the given bus. The connection is a global variable
1.772 + * shared with other callers of this function.
1.773 + *
1.774 + * (Internally, calls dbus_bus_get() then calls
1.775 + * dbus_connection_setup_with_g_main() on the result.)
1.776 + *
1.777 + * Returns: a DBusConnection
1.778 + */
1.779 + #ifdef __SYMBIAN32__
1.780 +EXPORT_C
1.781 +#endif
1.782 +DBusGConnection*
1.783 +dbus_g_bus_get (DBusBusType type,
1.784 + GError **error)
1.785 +{
1.786 + DBusConnection *connection;
1.787 + DBusError derror;
1.788 +
1.789 + g_return_val_if_fail (error == NULL || *error == NULL, NULL);
1.790 +
1.791 + _dbus_g_value_types_init ();
1.792 +
1.793 + dbus_error_init (&derror);
1.794 +
1.795 + connection = dbus_bus_get (type, &derror);
1.796 + if (connection == NULL)
1.797 + {
1.798 + dbus_set_g_error (error, &derror);
1.799 + dbus_error_free (&derror);
1.800 + return NULL;
1.801 + }
1.802 +
1.803 + /* does nothing if it's already been done */
1.804 + dbus_connection_setup_with_g_main (connection, NULL);
1.805 +
1.806 + return DBUS_G_CONNECTION_FROM_CONNECTION (connection);
1.807 +}
1.808 +
1.809 +/** @} */ /* end of public API */
1.810 +
1.811 +#ifdef DBUS_BUILD_TESTS
1.812 +
1.813 +/**
1.814 + * @ingroup DBusGLibInternals
1.815 + * Unit test for GLib main loop integration
1.816 + * Returns: #TRUE on success.
1.817 + */
1.818 + #ifdef __SYMBIAN32__
1.819 + EXPORT_C
1.820 + #endif
1.821 +gboolean
1.822 +_dbus_gmain_test (const char *test_data_dir)
1.823 +{
1.824 + GType type;
1.825 + GType rectype;
1.826 +
1.827 + g_type_init ();
1.828 + _dbus_g_value_types_init ();
1.829 +
1.830 + rectype = dbus_g_type_get_collection ("GArray", G_TYPE_UINT);
1.831 + g_assert (rectype != G_TYPE_INVALID);
1.832 + g_assert (!strcmp (g_type_name (rectype), "GArray_guint_"));
1.833 +
1.834 + type = _dbus_gtype_from_signature ("au", TRUE);
1.835 + g_assert (type == rectype);
1.836 +
1.837 + rectype = dbus_g_type_get_map ("GHashTable", G_TYPE_STRING, G_TYPE_STRING);
1.838 + g_assert (rectype != G_TYPE_INVALID);
1.839 + g_assert (!strcmp (g_type_name (rectype), "GHashTable_gchararray+gchararray_"));
1.840 +
1.841 + type = _dbus_gtype_from_signature ("a{ss}", TRUE);
1.842 + g_assert (type == rectype);
1.843 +
1.844 + type = _dbus_gtype_from_signature ("o", FALSE);
1.845 + g_assert (type == DBUS_TYPE_G_OBJECT_PATH);
1.846 + type = _dbus_gtype_from_signature ("o", TRUE);
1.847 + g_assert (type == DBUS_TYPE_G_OBJECT_PATH);
1.848 +
1.849 + return TRUE;
1.850 +}
1.851 +
1.852 +#endif /* DBUS_BUILD_TESTS */