os/ossrv/ofdbus/dbus-glib/dbus/dbus-gmain.c
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
/* -*- mode: C; c-file-style: "gnu" -*- */
sl@0
     2
/* dbus-gmain.c GLib main loop integration
sl@0
     3
 *
sl@0
     4
 * Copyright (C) 2002, 2003 CodeFactory AB
sl@0
     5
 * Copyright (C) 2005 Red Hat, Inc.
sl@0
     6
 * Portion Copyright © 2008 Nokia Corporation and/or its subsidiary(-ies). All rights reserved.
sl@0
     7
 * Licensed under the Academic Free License version 2.1
sl@0
     8
 * 
sl@0
     9
 * This program is free software; you can redistribute it and/or modify
sl@0
    10
 * it under the terms of the GNU General Public License as published by
sl@0
    11
 * the Free Software Foundation; either version 2 of the License, or
sl@0
    12
 * (at your option) any later version.
sl@0
    13
 *
sl@0
    14
 * This program is distributed in the hope that it will be useful,
sl@0
    15
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
sl@0
    16
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
sl@0
    17
 * GNU General Public License for more details.
sl@0
    18
 * 
sl@0
    19
 * You should have received a copy of the GNU General Public License
sl@0
    20
 * along with this program; if not, write to the Free Software
sl@0
    21
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
sl@0
    22
 *
sl@0
    23
 */
sl@0
    24
sl@0
    25
#ifndef __SYMBIAN32__
sl@0
    26
#include <config.h>
sl@0
    27
#else
sl@0
    28
#include "config.h"
sl@0
    29
#endif //__SYMBIAN32__
sl@0
    30
#include <dbus/dbus-glib.h>
sl@0
    31
#include <dbus/dbus-glib-lowlevel.h>
sl@0
    32
#include "dbus-gtest.h"
sl@0
    33
#include "dbus-gutils.h"
sl@0
    34
#include "dbus-gvalue.h"
sl@0
    35
#include "dbus-gobject.h"
sl@0
    36
#include "dbus-gvalue-utils.h"
sl@0
    37
#include "dbus-gsignature.h"
sl@0
    38
#include <string.h>
sl@0
    39
#ifndef __SYMBIAN32__
sl@0
    40
#include <libintl.h>
sl@0
    41
#define _(x) dgettext (GETTEXT_PACKAGE, x)
sl@0
    42
#define N_(x) x
sl@0
    43
#else
sl@0
    44
sl@0
    45
#define _(x) x
sl@0
    46
#define N_(x) x
sl@0
    47
#endif
sl@0
    48
sl@0
    49
#ifdef __SYMBIAN32__
sl@0
    50
#include "libdbus_glib_wsd_solution.h"
sl@0
    51
#endif
sl@0
    52
/**
sl@0
    53
 * @defgroup DBusGLibInternals GLib bindings implementation details
sl@0
    54
 * @ingroup  DBusInternals
sl@0
    55
 * @brief Implementation details of GLib bindings
sl@0
    56
 *
sl@0
    57
 * @{
sl@0
    58
 */
sl@0
    59
sl@0
    60
/**
sl@0
    61
 * DBusGMessageQueue:
sl@0
    62
 * A GSource subclass for dispatching DBusConnection messages.
sl@0
    63
 * We need this on top of the IO handlers, because sometimes
sl@0
    64
 * there are messages to dispatch queued up but no IO pending.
sl@0
    65
 */
sl@0
    66
typedef struct
sl@0
    67
{
sl@0
    68
  GSource source; /**< the parent GSource */
sl@0
    69
  DBusConnection *connection; /**< the connection to dispatch */
sl@0
    70
} DBusGMessageQueue;
sl@0
    71
sl@0
    72
static gboolean message_queue_prepare  (GSource     *source,
sl@0
    73
                                        gint        *timeout);
sl@0
    74
static gboolean message_queue_check    (GSource     *source);
sl@0
    75
static gboolean message_queue_dispatch (GSource     *source,
sl@0
    76
                                        GSourceFunc  callback,
sl@0
    77
                                        gpointer     user_data);
sl@0
    78
sl@0
    79
static const GSourceFuncs message_queue_funcs = {
sl@0
    80
  message_queue_prepare,
sl@0
    81
  message_queue_check,
sl@0
    82
  message_queue_dispatch,
sl@0
    83
  NULL
sl@0
    84
};
sl@0
    85
sl@0
    86
static gboolean
sl@0
    87
message_queue_prepare (GSource *source,
sl@0
    88
                       gint    *timeout)
sl@0
    89
{
sl@0
    90
  DBusConnection *connection = ((DBusGMessageQueue *)source)->connection;
sl@0
    91
  
sl@0
    92
  *timeout = -1;
sl@0
    93
sl@0
    94
  return (dbus_connection_get_dispatch_status (connection) == DBUS_DISPATCH_DATA_REMAINS);  
sl@0
    95
}
sl@0
    96
sl@0
    97
static gboolean
sl@0
    98
message_queue_check (GSource *source)
sl@0
    99
{
sl@0
   100
  return FALSE;
sl@0
   101
}
sl@0
   102
sl@0
   103
static gboolean
sl@0
   104
message_queue_dispatch (GSource     *source,
sl@0
   105
                        GSourceFunc  callback,
sl@0
   106
                        gpointer     user_data)
sl@0
   107
{
sl@0
   108
  DBusConnection *connection = ((DBusGMessageQueue *)source)->connection;
sl@0
   109
sl@0
   110
  dbus_connection_ref (connection);
sl@0
   111
sl@0
   112
  /* Only dispatch once - we don't want to starve other GSource */
sl@0
   113
  dbus_connection_dispatch (connection);
sl@0
   114
  
sl@0
   115
  dbus_connection_unref (connection);
sl@0
   116
sl@0
   117
  return TRUE;
sl@0
   118
}
sl@0
   119
sl@0
   120
typedef struct
sl@0
   121
{
sl@0
   122
  GMainContext *context;      /**< the main context */
sl@0
   123
  GSList *ios;                /**< all IOHandler */
sl@0
   124
  GSList *timeouts;           /**< all TimeoutHandler */
sl@0
   125
  DBusConnection *connection; /**< NULL if this is really for a server not a connection */
sl@0
   126
  GSource *message_queue_source; /**< DBusGMessageQueue */
sl@0
   127
} ConnectionSetup;
sl@0
   128
sl@0
   129
sl@0
   130
typedef struct
sl@0
   131
{
sl@0
   132
  ConnectionSetup *cs;
sl@0
   133
  GSource *source;
sl@0
   134
  DBusWatch *watch;
sl@0
   135
} IOHandler;
sl@0
   136
sl@0
   137
typedef struct
sl@0
   138
{
sl@0
   139
  ConnectionSetup *cs;
sl@0
   140
  GSource *source;
sl@0
   141
  DBusTimeout *timeout;
sl@0
   142
} TimeoutHandler;
sl@0
   143
sl@0
   144
sl@0
   145
#if EMULATOR
sl@0
   146
GET_GLOBAL_VAR_FROM_TLS(_dbus_gmain_connection_slot,dbus_gmain,dbus_int32_t)
sl@0
   147
#define _dbus_gmain_connection_slot (*GET_DBUS_WSD_VAR_NAME(_dbus_gmain_connection_slot,dbus_gmain,g)())
sl@0
   148
sl@0
   149
GET_STATIC_VAR_FROM_TLS(server_slot,dbus_gmain,dbus_int32_t)
sl@0
   150
#define server_slot (*GET_DBUS_WSD_VAR_NAME(server_slot,dbus_gmain,s)())
sl@0
   151
sl@0
   152
#else
sl@0
   153
sl@0
   154
dbus_int32_t _dbus_gmain_connection_slot = -1;
sl@0
   155
sl@0
   156
static dbus_int32_t server_slot = -1;
sl@0
   157
#endif
sl@0
   158
sl@0
   159
sl@0
   160
sl@0
   161
static ConnectionSetup*
sl@0
   162
connection_setup_new (GMainContext   *context,
sl@0
   163
                      DBusConnection *connection)
sl@0
   164
{
sl@0
   165
  ConnectionSetup *cs;
sl@0
   166
sl@0
   167
  cs = g_new0 (ConnectionSetup, 1);
sl@0
   168
sl@0
   169
  g_assert (context != NULL);
sl@0
   170
  
sl@0
   171
  cs->context = context;
sl@0
   172
  g_main_context_ref (cs->context);  
sl@0
   173
sl@0
   174
  if (connection)
sl@0
   175
    {
sl@0
   176
      cs->connection = connection;
sl@0
   177
sl@0
   178
      cs->message_queue_source = g_source_new ((GSourceFuncs *) &message_queue_funcs,
sl@0
   179
                                               sizeof (DBusGMessageQueue));
sl@0
   180
      ((DBusGMessageQueue*)cs->message_queue_source)->connection = connection;
sl@0
   181
      g_source_attach (cs->message_queue_source, cs->context);
sl@0
   182
    }
sl@0
   183
  
sl@0
   184
  return cs;
sl@0
   185
}
sl@0
   186
sl@0
   187
static void
sl@0
   188
io_handler_source_finalized (gpointer data)
sl@0
   189
{
sl@0
   190
  IOHandler *handler;
sl@0
   191
sl@0
   192
  handler = data;
sl@0
   193
sl@0
   194
  if (handler->watch)
sl@0
   195
    dbus_watch_set_data (handler->watch, NULL, NULL);
sl@0
   196
  
sl@0
   197
  g_free (handler);
sl@0
   198
}
sl@0
   199
sl@0
   200
static void
sl@0
   201
io_handler_destroy_source (void *data)
sl@0
   202
{
sl@0
   203
  IOHandler *handler;
sl@0
   204
sl@0
   205
  handler = data;
sl@0
   206
sl@0
   207
  if (handler->source)
sl@0
   208
    {
sl@0
   209
      GSource *source = handler->source;
sl@0
   210
      handler->source = NULL;
sl@0
   211
      handler->cs->ios = g_slist_remove (handler->cs->ios, handler);
sl@0
   212
      g_source_destroy (source);
sl@0
   213
      g_source_unref (source);
sl@0
   214
    }
sl@0
   215
}
sl@0
   216
sl@0
   217
static void
sl@0
   218
io_handler_watch_freed (void *data)
sl@0
   219
{
sl@0
   220
  IOHandler *handler;
sl@0
   221
sl@0
   222
  handler = data;
sl@0
   223
sl@0
   224
  handler->watch = NULL;
sl@0
   225
sl@0
   226
  io_handler_destroy_source (handler);
sl@0
   227
}
sl@0
   228
sl@0
   229
static gboolean
sl@0
   230
io_handler_dispatch (GIOChannel   *source,
sl@0
   231
                     GIOCondition  condition,
sl@0
   232
                     gpointer      data)
sl@0
   233
{
sl@0
   234
  IOHandler *handler;
sl@0
   235
  guint dbus_condition = 0;
sl@0
   236
  DBusConnection *connection;
sl@0
   237
sl@0
   238
  handler = data;
sl@0
   239
sl@0
   240
  connection = handler->cs->connection;
sl@0
   241
  
sl@0
   242
  if (connection)
sl@0
   243
    dbus_connection_ref (connection);
sl@0
   244
  
sl@0
   245
  if (condition & G_IO_IN)
sl@0
   246
    dbus_condition |= DBUS_WATCH_READABLE;
sl@0
   247
  if (condition & G_IO_OUT)
sl@0
   248
    dbus_condition |= DBUS_WATCH_WRITABLE;
sl@0
   249
  if (condition & G_IO_ERR)
sl@0
   250
    dbus_condition |= DBUS_WATCH_ERROR;
sl@0
   251
  if (condition & G_IO_HUP)
sl@0
   252
    dbus_condition |= DBUS_WATCH_HANGUP;
sl@0
   253
sl@0
   254
  /* Note that we don't touch the handler after this, because
sl@0
   255
   * dbus may have disabled the watch and thus killed the
sl@0
   256
   * handler.
sl@0
   257
   */
sl@0
   258
  dbus_watch_handle (handler->watch, dbus_condition);
sl@0
   259
  handler = NULL;
sl@0
   260
sl@0
   261
  if (connection)
sl@0
   262
    dbus_connection_unref (connection);
sl@0
   263
  
sl@0
   264
  return TRUE;
sl@0
   265
}
sl@0
   266
sl@0
   267
static void
sl@0
   268
connection_setup_add_watch (ConnectionSetup *cs,
sl@0
   269
                            DBusWatch       *watch)
sl@0
   270
{
sl@0
   271
  guint flags;
sl@0
   272
  GIOCondition condition;
sl@0
   273
  GIOChannel *channel;
sl@0
   274
  IOHandler *handler;
sl@0
   275
  
sl@0
   276
  if (!dbus_watch_get_enabled (watch))
sl@0
   277
    return;
sl@0
   278
  
sl@0
   279
  g_assert (dbus_watch_get_data (watch) == NULL);
sl@0
   280
  
sl@0
   281
  flags = dbus_watch_get_flags (watch);
sl@0
   282
sl@0
   283
  condition = G_IO_ERR | G_IO_HUP;
sl@0
   284
  if (flags & DBUS_WATCH_READABLE)
sl@0
   285
    condition |= G_IO_IN;
sl@0
   286
  if (flags & DBUS_WATCH_WRITABLE)
sl@0
   287
    condition |= G_IO_OUT;
sl@0
   288
sl@0
   289
  handler = g_new0 (IOHandler, 1);
sl@0
   290
  handler->cs = cs;
sl@0
   291
  handler->watch = watch;
sl@0
   292
  
sl@0
   293
  channel = g_io_channel_unix_new (dbus_watch_get_fd (watch));
sl@0
   294
  
sl@0
   295
  handler->source = g_io_create_watch (channel, condition);
sl@0
   296
  g_source_set_callback (handler->source, (GSourceFunc) io_handler_dispatch, handler,
sl@0
   297
                         io_handler_source_finalized);
sl@0
   298
  g_source_attach (handler->source, cs->context);
sl@0
   299
sl@0
   300
  cs->ios = g_slist_prepend (cs->ios, handler);
sl@0
   301
  
sl@0
   302
  dbus_watch_set_data (watch, handler, io_handler_watch_freed);
sl@0
   303
  g_io_channel_unref (channel);
sl@0
   304
}
sl@0
   305
sl@0
   306
static void
sl@0
   307
connection_setup_remove_watch (ConnectionSetup *cs,
sl@0
   308
                               DBusWatch       *watch)
sl@0
   309
{
sl@0
   310
  IOHandler *handler;
sl@0
   311
sl@0
   312
  handler = dbus_watch_get_data (watch);
sl@0
   313
sl@0
   314
  if (handler == NULL)
sl@0
   315
    return;
sl@0
   316
  
sl@0
   317
  io_handler_destroy_source (handler);
sl@0
   318
}
sl@0
   319
sl@0
   320
static void
sl@0
   321
timeout_handler_source_finalized (gpointer data)
sl@0
   322
{
sl@0
   323
  TimeoutHandler *handler;
sl@0
   324
sl@0
   325
  handler = data;
sl@0
   326
sl@0
   327
  if (handler->timeout)
sl@0
   328
    dbus_timeout_set_data (handler->timeout, NULL, NULL);
sl@0
   329
  
sl@0
   330
  g_free (handler);
sl@0
   331
}
sl@0
   332
sl@0
   333
static void
sl@0
   334
timeout_handler_destroy_source (void *data)
sl@0
   335
{
sl@0
   336
  TimeoutHandler *handler;
sl@0
   337
sl@0
   338
  handler = data;
sl@0
   339
sl@0
   340
  if (handler->source)
sl@0
   341
    {
sl@0
   342
      GSource *source = handler->source;
sl@0
   343
      handler->source = NULL;
sl@0
   344
      handler->cs->timeouts = g_slist_remove (handler->cs->timeouts, handler);
sl@0
   345
      g_source_destroy (source);
sl@0
   346
      g_source_unref (source);
sl@0
   347
    }
sl@0
   348
}
sl@0
   349
sl@0
   350
static void
sl@0
   351
timeout_handler_timeout_freed (void *data)
sl@0
   352
{
sl@0
   353
  TimeoutHandler *handler;
sl@0
   354
sl@0
   355
  handler = data;
sl@0
   356
sl@0
   357
  handler->timeout = NULL;
sl@0
   358
sl@0
   359
  timeout_handler_destroy_source (handler);
sl@0
   360
}
sl@0
   361
sl@0
   362
static gboolean
sl@0
   363
timeout_handler_dispatch (gpointer      data)
sl@0
   364
{
sl@0
   365
  TimeoutHandler *handler;
sl@0
   366
sl@0
   367
  handler = data;
sl@0
   368
sl@0
   369
  dbus_timeout_handle (handler->timeout);
sl@0
   370
  
sl@0
   371
  return TRUE;
sl@0
   372
}
sl@0
   373
sl@0
   374
static void
sl@0
   375
connection_setup_add_timeout (ConnectionSetup *cs,
sl@0
   376
                              DBusTimeout     *timeout)
sl@0
   377
{
sl@0
   378
  TimeoutHandler *handler;
sl@0
   379
  
sl@0
   380
  if (!dbus_timeout_get_enabled (timeout))
sl@0
   381
    return;
sl@0
   382
  
sl@0
   383
  g_assert (dbus_timeout_get_data (timeout) == NULL);
sl@0
   384
sl@0
   385
  handler = g_new0 (TimeoutHandler, 1);
sl@0
   386
  handler->cs = cs;
sl@0
   387
  handler->timeout = timeout;
sl@0
   388
sl@0
   389
  handler->source = g_timeout_source_new (dbus_timeout_get_interval (timeout));
sl@0
   390
  g_source_set_callback (handler->source, timeout_handler_dispatch, handler,
sl@0
   391
                         timeout_handler_source_finalized);
sl@0
   392
  g_source_attach (handler->source, handler->cs->context);
sl@0
   393
sl@0
   394
  cs->timeouts = g_slist_prepend (cs->timeouts, handler);
sl@0
   395
sl@0
   396
  dbus_timeout_set_data (timeout, handler, timeout_handler_timeout_freed);
sl@0
   397
}
sl@0
   398
sl@0
   399
static void
sl@0
   400
connection_setup_remove_timeout (ConnectionSetup *cs,
sl@0
   401
                                 DBusTimeout       *timeout)
sl@0
   402
{
sl@0
   403
  TimeoutHandler *handler;
sl@0
   404
  
sl@0
   405
  handler = dbus_timeout_get_data (timeout);
sl@0
   406
sl@0
   407
  if (handler == NULL)
sl@0
   408
    return;
sl@0
   409
  
sl@0
   410
  timeout_handler_destroy_source (handler);
sl@0
   411
}
sl@0
   412
sl@0
   413
static void
sl@0
   414
connection_setup_free (ConnectionSetup *cs)
sl@0
   415
{
sl@0
   416
  while (cs->ios)
sl@0
   417
    io_handler_destroy_source (cs->ios->data);
sl@0
   418
sl@0
   419
  while (cs->timeouts)
sl@0
   420
    timeout_handler_destroy_source (cs->timeouts->data);
sl@0
   421
sl@0
   422
  if (cs->message_queue_source)
sl@0
   423
    {
sl@0
   424
      GSource *source;
sl@0
   425
sl@0
   426
      source = cs->message_queue_source;
sl@0
   427
      cs->message_queue_source = NULL;
sl@0
   428
sl@0
   429
      g_source_destroy (source);
sl@0
   430
      g_source_unref (source);
sl@0
   431
    }
sl@0
   432
  
sl@0
   433
  g_main_context_unref (cs->context);
sl@0
   434
  g_free (cs);
sl@0
   435
}
sl@0
   436
sl@0
   437
static dbus_bool_t
sl@0
   438
add_watch (DBusWatch *watch,
sl@0
   439
	   gpointer   data)
sl@0
   440
{
sl@0
   441
  ConnectionSetup *cs;
sl@0
   442
sl@0
   443
  cs = data;
sl@0
   444
sl@0
   445
  connection_setup_add_watch (cs, watch);
sl@0
   446
  
sl@0
   447
  return TRUE;
sl@0
   448
}
sl@0
   449
sl@0
   450
static void
sl@0
   451
remove_watch (DBusWatch *watch,
sl@0
   452
	      gpointer   data)
sl@0
   453
{
sl@0
   454
  ConnectionSetup *cs;
sl@0
   455
sl@0
   456
  cs = data;
sl@0
   457
sl@0
   458
  connection_setup_remove_watch (cs, watch);
sl@0
   459
}
sl@0
   460
sl@0
   461
static void
sl@0
   462
watch_toggled (DBusWatch *watch,
sl@0
   463
               void      *data)
sl@0
   464
{
sl@0
   465
  /* Because we just exit on OOM, enable/disable is
sl@0
   466
   * no different from add/remove
sl@0
   467
   */
sl@0
   468
  if (dbus_watch_get_enabled (watch))
sl@0
   469
    add_watch (watch, data);
sl@0
   470
  else
sl@0
   471
    remove_watch (watch, data);
sl@0
   472
}
sl@0
   473
sl@0
   474
static dbus_bool_t
sl@0
   475
add_timeout (DBusTimeout *timeout,
sl@0
   476
	     void        *data)
sl@0
   477
{
sl@0
   478
  ConnectionSetup *cs;
sl@0
   479
sl@0
   480
  cs = data;
sl@0
   481
  
sl@0
   482
  if (!dbus_timeout_get_enabled (timeout))
sl@0
   483
    return TRUE;
sl@0
   484
sl@0
   485
  connection_setup_add_timeout (cs, timeout);
sl@0
   486
sl@0
   487
  return TRUE;
sl@0
   488
}
sl@0
   489
sl@0
   490
static void
sl@0
   491
remove_timeout (DBusTimeout *timeout,
sl@0
   492
		void        *data)
sl@0
   493
{
sl@0
   494
  ConnectionSetup *cs;
sl@0
   495
sl@0
   496
  cs = data;
sl@0
   497
sl@0
   498
  connection_setup_remove_timeout (cs, timeout);
sl@0
   499
}
sl@0
   500
sl@0
   501
static void
sl@0
   502
timeout_toggled (DBusTimeout *timeout,
sl@0
   503
                 void        *data)
sl@0
   504
{
sl@0
   505
  /* Because we just exit on OOM, enable/disable is
sl@0
   506
   * no different from add/remove
sl@0
   507
   */
sl@0
   508
  if (dbus_timeout_get_enabled (timeout))
sl@0
   509
    add_timeout (timeout, data);
sl@0
   510
  else
sl@0
   511
    remove_timeout (timeout, data);
sl@0
   512
}
sl@0
   513
sl@0
   514
static void
sl@0
   515
wakeup_main (void *data)
sl@0
   516
{
sl@0
   517
  ConnectionSetup *cs = data;
sl@0
   518
sl@0
   519
  g_main_context_wakeup (cs->context);
sl@0
   520
}
sl@0
   521
sl@0
   522
sl@0
   523
/* Move to a new context */
sl@0
   524
static ConnectionSetup*
sl@0
   525
connection_setup_new_from_old (GMainContext    *context,
sl@0
   526
                               ConnectionSetup *old)
sl@0
   527
{
sl@0
   528
  GSList *tmp;
sl@0
   529
  ConnectionSetup *cs;
sl@0
   530
sl@0
   531
  g_assert (old->context != context);
sl@0
   532
  
sl@0
   533
  cs = connection_setup_new (context, old->connection);
sl@0
   534
  
sl@0
   535
  tmp = old->ios;
sl@0
   536
  while (tmp != NULL)
sl@0
   537
    {
sl@0
   538
      IOHandler *handler = tmp->data;
sl@0
   539
sl@0
   540
      connection_setup_add_watch (cs, handler->watch);
sl@0
   541
      
sl@0
   542
      tmp = tmp->next;
sl@0
   543
    }
sl@0
   544
sl@0
   545
  tmp = old->timeouts;
sl@0
   546
  while (tmp != NULL)
sl@0
   547
    {
sl@0
   548
      TimeoutHandler *handler = tmp->data;
sl@0
   549
sl@0
   550
      connection_setup_add_timeout (cs, handler->timeout);
sl@0
   551
      
sl@0
   552
      tmp = tmp->next;
sl@0
   553
    }
sl@0
   554
sl@0
   555
  return cs;
sl@0
   556
}
sl@0
   557
sl@0
   558
/** @} */ /* End of GLib bindings internals */
sl@0
   559
sl@0
   560
/** @addtogroup DBusGLib
sl@0
   561
 * @{
sl@0
   562
 */
sl@0
   563
sl@0
   564
/**
sl@0
   565
 * dbus_connection_setup_with_g_main:
sl@0
   566
 * @connection: the connection
sl@0
   567
 * @context: the #GMainContext or #NULL for default context
sl@0
   568
 *
sl@0
   569
 * Sets the watch and timeout functions of a #DBusConnection
sl@0
   570
 * to integrate the connection with the GLib main loop.
sl@0
   571
 * Pass in #NULL for the #GMainContext unless you're
sl@0
   572
 * doing something specialized.
sl@0
   573
 *
sl@0
   574
 * If called twice for the same context, does nothing the second
sl@0
   575
 * time. If called once with context A and once with context B,
sl@0
   576
 * context B replaces context A as the context monitoring the
sl@0
   577
 * connection.
sl@0
   578
 */
sl@0
   579
 #ifdef __SYMBIAN32__
sl@0
   580
EXPORT_C
sl@0
   581
#endif
sl@0
   582
void
sl@0
   583
dbus_connection_setup_with_g_main (DBusConnection *connection,
sl@0
   584
				   GMainContext   *context)
sl@0
   585
{
sl@0
   586
  ConnectionSetup *old_setup;
sl@0
   587
  ConnectionSetup *cs;
sl@0
   588
  
sl@0
   589
  /* FIXME we never free the slot, so its refcount just keeps growing,
sl@0
   590
   * which is kind of broken.
sl@0
   591
   */
sl@0
   592
  dbus_connection_allocate_data_slot (&_dbus_gmain_connection_slot);
sl@0
   593
  if (_dbus_gmain_connection_slot < 0)
sl@0
   594
    goto nomem;
sl@0
   595
sl@0
   596
  if (context == NULL)
sl@0
   597
    context = g_main_context_default ();
sl@0
   598
sl@0
   599
  cs = NULL;
sl@0
   600
  
sl@0
   601
  old_setup = dbus_connection_get_data (connection, _dbus_gmain_connection_slot);
sl@0
   602
  if (old_setup != NULL)
sl@0
   603
    {
sl@0
   604
      if (old_setup->context == context)
sl@0
   605
        return; /* nothing to do */
sl@0
   606
sl@0
   607
      cs = connection_setup_new_from_old (context, old_setup);
sl@0
   608
      
sl@0
   609
      /* Nuke the old setup */
sl@0
   610
      dbus_connection_set_data (connection, _dbus_gmain_connection_slot, NULL, NULL);
sl@0
   611
      old_setup = NULL;
sl@0
   612
    }
sl@0
   613
sl@0
   614
  if (cs == NULL)
sl@0
   615
    cs = connection_setup_new (context, connection);
sl@0
   616
sl@0
   617
  if (!dbus_connection_set_data (connection, _dbus_gmain_connection_slot, cs,
sl@0
   618
                                 (DBusFreeFunction)connection_setup_free))
sl@0
   619
    goto nomem;
sl@0
   620
  
sl@0
   621
  if (!dbus_connection_set_watch_functions (connection,
sl@0
   622
                                            add_watch,
sl@0
   623
                                            remove_watch,
sl@0
   624
                                            watch_toggled,
sl@0
   625
                                            cs, NULL))
sl@0
   626
    goto nomem;
sl@0
   627
sl@0
   628
  if (!dbus_connection_set_timeout_functions (connection,
sl@0
   629
                                              add_timeout,
sl@0
   630
                                              remove_timeout,
sl@0
   631
                                              timeout_toggled,
sl@0
   632
                                              cs, NULL))
sl@0
   633
    goto nomem;
sl@0
   634
    
sl@0
   635
  dbus_connection_set_wakeup_main_function (connection,
sl@0
   636
					    wakeup_main,
sl@0
   637
					    cs, NULL);
sl@0
   638
      
sl@0
   639
  return;
sl@0
   640
sl@0
   641
 nomem:
sl@0
   642
  g_error ("Not enough memory to set up DBusConnection for use with GLib");
sl@0
   643
}
sl@0
   644
sl@0
   645
/**
sl@0
   646
 * dbus_server_setup_with_g_main:
sl@0
   647
 * @server: the server
sl@0
   648
 * @context: the #GMainContext or #NULL for default
sl@0
   649
 *
sl@0
   650
 * Sets the watch and timeout functions of a #DBusServer
sl@0
   651
 * to integrate the server with the GLib main loop.
sl@0
   652
 * In most cases the context argument should be #NULL.
sl@0
   653
 *
sl@0
   654
 * If called twice for the same context, does nothing the second
sl@0
   655
 * time. If called once with context A and once with context B,
sl@0
   656
 * context B replaces context A as the context monitoring the
sl@0
   657
 * connection.
sl@0
   658
 */
sl@0
   659
 	#ifdef __SYMBIAN32__
sl@0
   660
	EXPORT_C
sl@0
   661
	#endif
sl@0
   662
void
sl@0
   663
dbus_server_setup_with_g_main (DBusServer   *server,
sl@0
   664
                               GMainContext *context)
sl@0
   665
{
sl@0
   666
  ConnectionSetup *old_setup;
sl@0
   667
  ConnectionSetup *cs;
sl@0
   668
  
sl@0
   669
  /* FIXME we never free the slot, so its refcount just keeps growing,
sl@0
   670
   * which is kind of broken.
sl@0
   671
   */
sl@0
   672
  dbus_server_allocate_data_slot (&server_slot);
sl@0
   673
  if (server_slot < 0)
sl@0
   674
    goto nomem;
sl@0
   675
sl@0
   676
  if (context == NULL)
sl@0
   677
    context = g_main_context_default ();
sl@0
   678
sl@0
   679
  cs = NULL;
sl@0
   680
  
sl@0
   681
  old_setup = dbus_server_get_data (server, server_slot);
sl@0
   682
  if (old_setup != NULL)
sl@0
   683
    {
sl@0
   684
      if (old_setup->context == context)
sl@0
   685
        return; /* nothing to do */
sl@0
   686
sl@0
   687
      cs = connection_setup_new_from_old (context, old_setup);
sl@0
   688
      
sl@0
   689
      /* Nuke the old setup */
sl@0
   690
      dbus_server_set_data (server, server_slot, NULL, NULL);
sl@0
   691
      old_setup = NULL;
sl@0
   692
    }
sl@0
   693
sl@0
   694
  if (cs == NULL)
sl@0
   695
    cs = connection_setup_new (context, NULL);
sl@0
   696
sl@0
   697
  if (!dbus_server_set_data (server, server_slot, cs,
sl@0
   698
                             (DBusFreeFunction)connection_setup_free))
sl@0
   699
    goto nomem;
sl@0
   700
  
sl@0
   701
  if (!dbus_server_set_watch_functions (server,
sl@0
   702
                                        add_watch,
sl@0
   703
                                        remove_watch,
sl@0
   704
                                        watch_toggled,
sl@0
   705
                                        cs, NULL))
sl@0
   706
    goto nomem;
sl@0
   707
sl@0
   708
  if (!dbus_server_set_timeout_functions (server,
sl@0
   709
                                          add_timeout,
sl@0
   710
                                          remove_timeout,
sl@0
   711
                                          timeout_toggled,
sl@0
   712
                                          cs, NULL))
sl@0
   713
    goto nomem;
sl@0
   714
      
sl@0
   715
  return;
sl@0
   716
sl@0
   717
 nomem:
sl@0
   718
  g_error ("Not enough memory to set up DBusServer for use with GLib");
sl@0
   719
}
sl@0
   720
sl@0
   721
/**
sl@0
   722
 * dbus_g_connection_open:
sl@0
   723
 * @address: address of the connection to open
sl@0
   724
 * @error: address where an error can be returned.
sl@0
   725
 *
sl@0
   726
 * Returns a connection to the given address.
sl@0
   727
 *
sl@0
   728
 * (Internally, calls dbus_connection_open() then calls
sl@0
   729
 * dbus_connection_setup_with_g_main() on the result.)
sl@0
   730
 *
sl@0
   731
 * Returns: a DBusConnection
sl@0
   732
 */
sl@0
   733
 	#ifdef __SYMBIAN32__
sl@0
   734
	EXPORT_C
sl@0
   735
	#endif
sl@0
   736
DBusGConnection*
sl@0
   737
dbus_g_connection_open (const gchar  *address,
sl@0
   738
                        GError      **error)
sl@0
   739
{
sl@0
   740
  DBusConnection *connection;
sl@0
   741
  DBusError derror;
sl@0
   742
sl@0
   743
  g_return_val_if_fail (error == NULL || *error == NULL, NULL);
sl@0
   744
sl@0
   745
  _dbus_g_value_types_init ();
sl@0
   746
sl@0
   747
  dbus_error_init (&derror);
sl@0
   748
sl@0
   749
  connection = dbus_connection_open (address, &derror);
sl@0
   750
  if (connection == NULL)
sl@0
   751
    {
sl@0
   752
      dbus_set_g_error (error, &derror);
sl@0
   753
      dbus_error_free (&derror);
sl@0
   754
      return NULL;
sl@0
   755
    }
sl@0
   756
sl@0
   757
  /* does nothing if it's already been done */
sl@0
   758
  dbus_connection_setup_with_g_main (connection, NULL);
sl@0
   759
sl@0
   760
  return DBUS_G_CONNECTION_FROM_CONNECTION (connection);
sl@0
   761
}
sl@0
   762
sl@0
   763
/**
sl@0
   764
 * dbus_g_bus_get:
sl@0
   765
 * @type: bus type
sl@0
   766
 * @error: address where an error can be returned.
sl@0
   767
 *
sl@0
   768
 * Returns a connection to the given bus. The connection is a global variable
sl@0
   769
 * shared with other callers of this function.
sl@0
   770
 * 
sl@0
   771
 * (Internally, calls dbus_bus_get() then calls
sl@0
   772
 * dbus_connection_setup_with_g_main() on the result.)
sl@0
   773
 *
sl@0
   774
 * Returns: a DBusConnection
sl@0
   775
 */
sl@0
   776
 #ifdef __SYMBIAN32__
sl@0
   777
EXPORT_C
sl@0
   778
#endif
sl@0
   779
DBusGConnection*
sl@0
   780
dbus_g_bus_get (DBusBusType     type,
sl@0
   781
                GError        **error)
sl@0
   782
{
sl@0
   783
  DBusConnection *connection;
sl@0
   784
  DBusError derror;
sl@0
   785
sl@0
   786
  g_return_val_if_fail (error == NULL || *error == NULL, NULL);
sl@0
   787
sl@0
   788
  _dbus_g_value_types_init ();
sl@0
   789
  
sl@0
   790
  dbus_error_init (&derror);
sl@0
   791
sl@0
   792
  connection = dbus_bus_get (type, &derror);
sl@0
   793
  if (connection == NULL)
sl@0
   794
    {
sl@0
   795
      dbus_set_g_error (error, &derror);
sl@0
   796
      dbus_error_free (&derror);
sl@0
   797
      return NULL;
sl@0
   798
    }
sl@0
   799
sl@0
   800
  /* does nothing if it's already been done */
sl@0
   801
  dbus_connection_setup_with_g_main (connection, NULL);
sl@0
   802
sl@0
   803
  return DBUS_G_CONNECTION_FROM_CONNECTION (connection);
sl@0
   804
}
sl@0
   805
sl@0
   806
/** @} */ /* end of public API */
sl@0
   807
sl@0
   808
#ifdef DBUS_BUILD_TESTS
sl@0
   809
sl@0
   810
/**
sl@0
   811
 * @ingroup DBusGLibInternals
sl@0
   812
 * Unit test for GLib main loop integration
sl@0
   813
 * Returns: #TRUE on success.
sl@0
   814
 */
sl@0
   815
  	#ifdef __SYMBIAN32__
sl@0
   816
	EXPORT_C
sl@0
   817
	#endif
sl@0
   818
gboolean
sl@0
   819
_dbus_gmain_test (const char *test_data_dir)
sl@0
   820
{
sl@0
   821
  GType type;
sl@0
   822
  GType rectype;
sl@0
   823
sl@0
   824
  g_type_init ();
sl@0
   825
  _dbus_g_value_types_init ();
sl@0
   826
sl@0
   827
  rectype = dbus_g_type_get_collection ("GArray", G_TYPE_UINT);
sl@0
   828
  g_assert (rectype != G_TYPE_INVALID);
sl@0
   829
  g_assert (!strcmp (g_type_name (rectype), "GArray_guint_"));
sl@0
   830
sl@0
   831
  type = _dbus_gtype_from_signature ("au", TRUE);
sl@0
   832
  g_assert (type == rectype);
sl@0
   833
sl@0
   834
  rectype = dbus_g_type_get_map ("GHashTable", G_TYPE_STRING, G_TYPE_STRING);
sl@0
   835
  g_assert (rectype != G_TYPE_INVALID);
sl@0
   836
  g_assert (!strcmp (g_type_name (rectype), "GHashTable_gchararray+gchararray_"));
sl@0
   837
sl@0
   838
  type = _dbus_gtype_from_signature ("a{ss}", TRUE);
sl@0
   839
  g_assert (type == rectype);
sl@0
   840
sl@0
   841
  type = _dbus_gtype_from_signature ("o", FALSE);
sl@0
   842
  g_assert (type == DBUS_TYPE_G_OBJECT_PATH);
sl@0
   843
  type = _dbus_gtype_from_signature ("o", TRUE);
sl@0
   844
  g_assert (type == DBUS_TYPE_G_OBJECT_PATH);
sl@0
   845
  
sl@0
   846
  return TRUE;
sl@0
   847
}
sl@0
   848
sl@0
   849
#endif /* DBUS_BUILD_TESTS */