os/ossrv/ofdbus/dbus-glib/dbus/dbus-gobject.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-gobject.c Exporting a GObject remotely
sl@0
     3
 *
sl@0
     4
 * Copyright (C) 2003, 2004, 2005 Red Hat, Inc.
sl@0
     5
 * Copyright (C) 2005 Nokia
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 <gobject/gvaluecollector.h>
sl@0
    31
#include <dbus/dbus-glib.h>
sl@0
    32
#include <dbus/dbus-glib-lowlevel.h>
sl@0
    33
#include "dbus-gtest.h"
sl@0
    34
#include "dbus-gutils.h"
sl@0
    35
#include "dbus-gobject.h"
sl@0
    36
#include "dbus-gsignature.h"
sl@0
    37
#include "dbus-gvalue.h"
sl@0
    38
#include "dbus-gmarshal.h"
sl@0
    39
#include "dbus-gvalue-utils.h"
sl@0
    40
#include <string.h>
sl@0
    41
sl@0
    42
#ifdef __SYMBIAN32__
sl@0
    43
#include "libdbus_glib_wsd_solution.h"
sl@0
    44
#endif
sl@0
    45
sl@0
    46
sl@0
    47
typedef struct
sl@0
    48
{
sl@0
    49
  char *default_iface;
sl@0
    50
  GType code_enum;
sl@0
    51
} DBusGErrorInfo;
sl@0
    52
sl@0
    53
sl@0
    54
sl@0
    55
#if EMULATOR
sl@0
    56
GET_STATIC_VAR_FROM_TLS(marshal_table,dbus_gobject,GHashTable *)
sl@0
    57
#define marshal_table (*GET_DBUS_WSD_VAR_NAME(marshal_table,dbus_gobject,s)())
sl@0
    58
GET_STATIC_VAR_FROM_TLS(error_metadata,dbus_gobject,GData *)
sl@0
    59
#define error_metadata (*GET_DBUS_WSD_VAR_NAME(error_metadata,dbus_gobject,s)())
sl@0
    60
GET_STATIC_VAR_FROM_TLS(globals_lock,dbus_gobject,GStaticRWLock )
sl@0
    61
#define globals_lock (*GET_DBUS_WSD_VAR_NAME(globals_lock,dbus_gobject,s)())
sl@0
    62
sl@0
    63
#else
sl@0
    64
static GStaticRWLock globals_lock = G_STATIC_RW_LOCK_INIT;
sl@0
    65
sl@0
    66
static GHashTable *marshal_table = NULL;
sl@0
    67
static GData *error_metadata = NULL;
sl@0
    68
sl@0
    69
#endif
sl@0
    70
sl@0
    71
sl@0
    72
sl@0
    73
sl@0
    74
sl@0
    75
static char*
sl@0
    76
uscore_to_wincaps (const char *uscore)
sl@0
    77
{
sl@0
    78
  const char *p;
sl@0
    79
  GString *str;
sl@0
    80
  gboolean last_was_uscore;
sl@0
    81
sl@0
    82
  last_was_uscore = TRUE;
sl@0
    83
  
sl@0
    84
  str = g_string_new (NULL);
sl@0
    85
  p = uscore;
sl@0
    86
  while (p && *p)
sl@0
    87
    {
sl@0
    88
      if (*p == '-' || *p == '_')
sl@0
    89
        {
sl@0
    90
          last_was_uscore = TRUE;
sl@0
    91
        }
sl@0
    92
      else
sl@0
    93
        {
sl@0
    94
          if (last_was_uscore)
sl@0
    95
            {
sl@0
    96
              g_string_append_c (str, g_ascii_toupper (*p));
sl@0
    97
              last_was_uscore = FALSE;
sl@0
    98
            }
sl@0
    99
          else
sl@0
   100
            g_string_append_c (str, *p);
sl@0
   101
        }
sl@0
   102
      ++p;
sl@0
   103
    }
sl@0
   104
sl@0
   105
  return g_string_free (str, FALSE);
sl@0
   106
}
sl@0
   107
sl@0
   108
static const char *
sl@0
   109
string_table_next (const char *table)
sl@0
   110
{
sl@0
   111
  return (table + (strlen (table) + 1));
sl@0
   112
}
sl@0
   113
sl@0
   114
static const char *
sl@0
   115
string_table_lookup (const char *table, int index)
sl@0
   116
{
sl@0
   117
  const char *ret;
sl@0
   118
sl@0
   119
  ret = table;
sl@0
   120
sl@0
   121
  while (index--)
sl@0
   122
    ret = string_table_next (ret);
sl@0
   123
sl@0
   124
  return ret;
sl@0
   125
}
sl@0
   126
sl@0
   127
static const char *
sl@0
   128
get_method_data (const DBusGObjectInfo *object,
sl@0
   129
		 const DBusGMethodInfo *method)
sl@0
   130
{
sl@0
   131
  return object->data + method->data_offset;
sl@0
   132
}
sl@0
   133
sl@0
   134
static char *
sl@0
   135
object_error_domain_prefix_from_object_info (const DBusGObjectInfo *info)
sl@0
   136
{
sl@0
   137
  /* FIXME */
sl@0
   138
  return NULL;
sl@0
   139
}
sl@0
   140
sl@0
   141
static char *
sl@0
   142
object_error_code_from_object_info (const DBusGObjectInfo *info, GQuark domain, gint code)
sl@0
   143
{
sl@0
   144
  /* FIXME */
sl@0
   145
  return NULL;
sl@0
   146
}
sl@0
   147
sl@0
   148
static const char *
sl@0
   149
method_interface_from_object_info (const DBusGObjectInfo *object,
sl@0
   150
			      const DBusGMethodInfo *method)
sl@0
   151
{
sl@0
   152
  return string_table_lookup (get_method_data (object, method), 0);
sl@0
   153
}
sl@0
   154
sl@0
   155
static const char *
sl@0
   156
method_name_from_object_info (const DBusGObjectInfo *object,
sl@0
   157
			      const DBusGMethodInfo *method)
sl@0
   158
{
sl@0
   159
  return string_table_lookup (get_method_data (object, method), 1);
sl@0
   160
}
sl@0
   161
sl@0
   162
static const char *
sl@0
   163
method_arg_info_from_object_info (const DBusGObjectInfo *object,
sl@0
   164
				  const DBusGMethodInfo *method)
sl@0
   165
{
sl@0
   166
  return string_table_lookup (get_method_data (object, method), 3);/*RB was 2*/
sl@0
   167
}
sl@0
   168
sl@0
   169
typedef enum
sl@0
   170
{
sl@0
   171
  RETVAL_NONE,    
sl@0
   172
  RETVAL_NOERROR,    
sl@0
   173
  RETVAL_ERROR
sl@0
   174
} RetvalType;
sl@0
   175
sl@0
   176
static const char *
sl@0
   177
arg_iterate (const char    *data,
sl@0
   178
	     const char   **name,
sl@0
   179
	     gboolean      *in,
sl@0
   180
	     gboolean      *constval,
sl@0
   181
	     RetvalType    *retval,
sl@0
   182
	     const char   **type)
sl@0
   183
{
sl@0
   184
  gboolean inarg;
sl@0
   185
sl@0
   186
  if (name)
sl@0
   187
    *name = data;
sl@0
   188
sl@0
   189
  data = string_table_next (data);
sl@0
   190
  switch (*data)
sl@0
   191
    {
sl@0
   192
    case 'I':
sl@0
   193
      inarg = TRUE;
sl@0
   194
      break;
sl@0
   195
    case 'O':
sl@0
   196
      inarg = FALSE;
sl@0
   197
      break;
sl@0
   198
    default:
sl@0
   199
      g_warning ("invalid arg direction '%c'", *data);
sl@0
   200
      inarg = FALSE;
sl@0
   201
      break;
sl@0
   202
    }
sl@0
   203
  if (in)
sl@0
   204
    *in = inarg;
sl@0
   205
sl@0
   206
  if (!inarg)
sl@0
   207
    {
sl@0
   208
      data = string_table_next (data);
sl@0
   209
      switch (*data)
sl@0
   210
	{
sl@0
   211
	case 'F':
sl@0
   212
	  if (constval)
sl@0
   213
	    *constval = FALSE;
sl@0
   214
	  break;
sl@0
   215
	case 'C':
sl@0
   216
	  if (constval)
sl@0
   217
	    *constval = TRUE;
sl@0
   218
	  break;
sl@0
   219
	default:
sl@0
   220
	  g_warning ("invalid arg const value '%c'", *data);
sl@0
   221
	  break;
sl@0
   222
	}
sl@0
   223
      data = string_table_next (data);
sl@0
   224
      switch (*data)
sl@0
   225
	{
sl@0
   226
	case 'N':
sl@0
   227
	  if (retval)
sl@0
   228
	    *retval = RETVAL_NONE;
sl@0
   229
	  break;
sl@0
   230
	case 'E':
sl@0
   231
	  if (retval)
sl@0
   232
	    *retval = RETVAL_ERROR;
sl@0
   233
	  break;
sl@0
   234
	case 'R':
sl@0
   235
	  if (retval)
sl@0
   236
	    *retval = RETVAL_NOERROR;
sl@0
   237
	  break;
sl@0
   238
	default:
sl@0
   239
	  g_warning ("invalid arg ret value '%c'", *data);
sl@0
   240
	  break;
sl@0
   241
	}
sl@0
   242
    }
sl@0
   243
  else
sl@0
   244
    {
sl@0
   245
      if (constval)
sl@0
   246
	*constval = FALSE;
sl@0
   247
      if (retval)
sl@0
   248
      #ifndef __SYMBIAN32__
sl@0
   249
      	*retval = FALSE;
sl@0
   250
      #else
sl@0
   251
	*retval = (RetvalType)FALSE;
sl@0
   252
	 #endif
sl@0
   253
    }
sl@0
   254
  
sl@0
   255
  data = string_table_next (data);
sl@0
   256
  if (type)
sl@0
   257
    *type = data;
sl@0
   258
sl@0
   259
  return string_table_next (data);
sl@0
   260
}
sl@0
   261
sl@0
   262
static char *
sl@0
   263
method_dir_signature_from_object_info (const DBusGObjectInfo *object,
sl@0
   264
				       const DBusGMethodInfo *method,
sl@0
   265
				       gboolean               in)
sl@0
   266
{
sl@0
   267
  const char *arg;
sl@0
   268
  GString *ret;
sl@0
   269
sl@0
   270
  arg = method_arg_info_from_object_info (object, method);
sl@0
   271
sl@0
   272
  ret = g_string_new (NULL);
sl@0
   273
sl@0
   274
  while (*arg)
sl@0
   275
    {
sl@0
   276
      const char *name;
sl@0
   277
      gboolean arg_in;
sl@0
   278
      const char *type;
sl@0
   279
sl@0
   280
      arg = arg_iterate (arg, &name, &arg_in, NULL, NULL, &type);
sl@0
   281
sl@0
   282
      if (arg_in == in)
sl@0
   283
	g_string_append (ret, type);
sl@0
   284
    }
sl@0
   285
sl@0
   286
  return g_string_free (ret, FALSE);
sl@0
   287
}
sl@0
   288
sl@0
   289
static char *
sl@0
   290
method_input_signature_from_object_info (const DBusGObjectInfo *object,
sl@0
   291
					 const DBusGMethodInfo *method)
sl@0
   292
{
sl@0
   293
  return method_dir_signature_from_object_info (object, method, TRUE);
sl@0
   294
}
sl@0
   295
sl@0
   296
static char *
sl@0
   297
method_output_signature_from_object_info (const DBusGObjectInfo *object,
sl@0
   298
					  const DBusGMethodInfo *method)
sl@0
   299
{
sl@0
   300
  return method_dir_signature_from_object_info (object, method, FALSE);
sl@0
   301
}
sl@0
   302
sl@0
   303
static const char *
sl@0
   304
propsig_iterate (const char *data, const char **iface, const char **name)
sl@0
   305
{
sl@0
   306
  *iface = data;
sl@0
   307
sl@0
   308
  data = string_table_next (data);
sl@0
   309
  *name = data;
sl@0
   310
sl@0
   311
  return string_table_next (data);
sl@0
   312
}
sl@0
   313
sl@0
   314
#if EMULATOR
sl@0
   315
GET_STATIC_VAR_FROM_TLS(quark,dbus_gobject,GQuark )
sl@0
   316
#define quark (*GET_DBUS_WSD_VAR_NAME(quark,dbus_gobject,s)())
sl@0
   317
sl@0
   318
#endif
sl@0
   319
sl@0
   320
static GQuark
sl@0
   321
dbus_g_object_type_dbus_metadata_quark (void)
sl@0
   322
{
sl@0
   323
#ifndef EMULATOR
sl@0
   324
  static GQuark quark;
sl@0
   325
#endif
sl@0
   326
  
sl@0
   327
  if (!quark)
sl@0
   328
    quark = g_quark_from_static_string ("DBusGObjectTypeDBusMetadataQuark");
sl@0
   329
  return quark;
sl@0
   330
}
sl@0
   331
sl@0
   332
static GList *
sl@0
   333
lookup_object_info (GObject *object)
sl@0
   334
{
sl@0
   335
  GType *interfaces, *p;
sl@0
   336
  GList *info_list = NULL;
sl@0
   337
  const DBusGObjectInfo *info;
sl@0
   338
  GType classtype;
sl@0
   339
sl@0
   340
  interfaces = g_type_interfaces (G_TYPE_FROM_INSTANCE (object), NULL);
sl@0
   341
sl@0
   342
  for (p = interfaces; *p != 0; p++)
sl@0
   343
    {
sl@0
   344
      info = g_type_get_qdata (*p, dbus_g_object_type_dbus_metadata_quark ());
sl@0
   345
      if (info != NULL && info->format_version >= 0)
sl@0
   346
          info_list = g_list_prepend (info_list, (gpointer) info);
sl@0
   347
    }
sl@0
   348
sl@0
   349
  g_free (interfaces);
sl@0
   350
sl@0
   351
  for (classtype = G_TYPE_FROM_INSTANCE (object); classtype != 0; classtype = g_type_parent (classtype))
sl@0
   352
    {
sl@0
   353
      info = g_type_get_qdata (classtype, dbus_g_object_type_dbus_metadata_quark ());
sl@0
   354
      if (info != NULL && info->format_version >= 0)
sl@0
   355
          info_list = g_list_prepend (info_list, (gpointer) info);
sl@0
   356
    }
sl@0
   357
sl@0
   358
  /* if needed only:
sl@0
   359
  return g_list_reverse (info_list);
sl@0
   360
  */
sl@0
   361
  
sl@0
   362
  return info_list;
sl@0
   363
}
sl@0
   364
sl@0
   365
static void
sl@0
   366
gobject_unregister_function (DBusConnection  *connection,
sl@0
   367
                             void            *user_data)
sl@0
   368
{
sl@0
   369
#ifdef WINSCW
sl@0
   370
sl@0
   371
  GObject *object;
sl@0
   372
sl@0
   373
  object = G_OBJECT (user_data);
sl@0
   374
#endif
sl@0
   375
  /* FIXME */
sl@0
   376
sl@0
   377
}
sl@0
   378
sl@0
   379
typedef struct
sl@0
   380
{
sl@0
   381
  GString *xml;
sl@0
   382
  GType gtype;
sl@0
   383
  const DBusGObjectInfo *object_info;
sl@0
   384
} DBusGLibWriteIterfaceData;
sl@0
   385
sl@0
   386
typedef struct
sl@0
   387
{
sl@0
   388
  GSList *methods;
sl@0
   389
  GSList *signals;
sl@0
   390
  GSList *properties;
sl@0
   391
} DBusGLibWriteInterfaceValues;
sl@0
   392
sl@0
   393
static void
sl@0
   394
write_interface (gpointer key, gpointer val, gpointer user_data)
sl@0
   395
{
sl@0
   396
  const char *name;
sl@0
   397
  GSList *methods;
sl@0
   398
  GSList *signals;
sl@0
   399
  GSList *properties;
sl@0
   400
  GString *xml;
sl@0
   401
  const DBusGObjectInfo *object_info;
sl@0
   402
  DBusGLibWriteIterfaceData *data;
sl@0
   403
  DBusGLibWriteInterfaceValues *values;
sl@0
   404
sl@0
   405
  name = key;
sl@0
   406
sl@0
   407
  values = val;
sl@0
   408
  methods = values->methods;
sl@0
   409
  signals = values->signals;
sl@0
   410
  properties = values->properties;
sl@0
   411
sl@0
   412
  data = user_data;
sl@0
   413
  xml = data->xml;
sl@0
   414
  object_info = data->object_info;
sl@0
   415
sl@0
   416
  g_string_append_printf (xml, "  <interface name=\"%s\">\n", name);
sl@0
   417
sl@0
   418
  /* FIXME: recurse to parent types ? */
sl@0
   419
  for (; methods; methods = methods->next)
sl@0
   420
    {
sl@0
   421
      DBusGMethodInfo *method;
sl@0
   422
      const char *args;
sl@0
   423
      method = methods->data;
sl@0
   424
sl@0
   425
      g_string_append_printf (xml, "    <method name=\"%s\">\n",
sl@0
   426
			      method_name_from_object_info (object_info, method));
sl@0
   427
sl@0
   428
      args = method_arg_info_from_object_info (object_info, method);
sl@0
   429
sl@0
   430
      while (*args)
sl@0
   431
	{
sl@0
   432
	  const char *name;
sl@0
   433
	  gboolean arg_in;
sl@0
   434
	  const char *type;
sl@0
   435
	  
sl@0
   436
	  args = arg_iterate (args, &name, &arg_in, NULL, NULL, &type);
sl@0
   437
sl@0
   438
	  /* FIXME - handle container types */
sl@0
   439
	  g_string_append_printf (xml, "      <arg name=\"%s\" type=\"%s\" direction=\"%s\"/>\n",
sl@0
   440
				  name, type, arg_in ? "in" : "out");
sl@0
   441
sl@0
   442
	}
sl@0
   443
      g_string_append (xml, "    </method>\n");
sl@0
   444
sl@0
   445
    }
sl@0
   446
  g_slist_free (values->methods);
sl@0
   447
sl@0
   448
  for (; signals; signals = signals->next)
sl@0
   449
    {
sl@0
   450
      guint id;
sl@0
   451
      guint arg;
sl@0
   452
      const char *signame;
sl@0
   453
      GSignalQuery query;
sl@0
   454
      char *s;
sl@0
   455
sl@0
   456
      signame = signals->data;
sl@0
   457
sl@0
   458
      s = _dbus_gutils_wincaps_to_uscore (signame);
sl@0
   459
      
sl@0
   460
      id = g_signal_lookup (s, data->gtype);
sl@0
   461
      g_assert (id != 0);
sl@0
   462
sl@0
   463
      g_signal_query (id, &query);
sl@0
   464
      g_assert (query.return_type == G_TYPE_NONE);
sl@0
   465
sl@0
   466
      g_string_append_printf (xml, "    <signal name=\"%s\">\n", signame);
sl@0
   467
sl@0
   468
      for (arg = 0; arg < query.n_params; arg++)
sl@0
   469
	{
sl@0
   470
	  char *dbus_type = _dbus_gtype_to_signature (query.param_types[arg]);
sl@0
   471
sl@0
   472
	  g_assert (dbus_type != NULL);
sl@0
   473
sl@0
   474
          g_string_append (xml, "      <arg type=\"");
sl@0
   475
          g_string_append (xml, dbus_type);
sl@0
   476
          g_string_append (xml, "\"/>\n");
sl@0
   477
	  g_free (dbus_type);
sl@0
   478
	}
sl@0
   479
sl@0
   480
      g_string_append (xml, "    </signal>\n");
sl@0
   481
      g_free (s);
sl@0
   482
    }
sl@0
   483
  g_slist_free (values->signals);
sl@0
   484
sl@0
   485
  for (; properties; properties = properties->next)
sl@0
   486
    {
sl@0
   487
      const char *propname;
sl@0
   488
      GParamSpec *spec;
sl@0
   489
      char *dbus_type;
sl@0
   490
      gboolean can_set;
sl@0
   491
      gboolean can_get;
sl@0
   492
      char *s;
sl@0
   493
sl@0
   494
      propname = properties->data;
sl@0
   495
      spec = NULL;
sl@0
   496
sl@0
   497
      s = _dbus_gutils_wincaps_to_uscore (propname);
sl@0
   498
sl@0
   499
      spec = g_object_class_find_property (g_type_class_peek (data->gtype), s);
sl@0
   500
      g_assert (spec != NULL);
sl@0
   501
      g_free (s);
sl@0
   502
      
sl@0
   503
      dbus_type = _dbus_gtype_to_signature (G_PARAM_SPEC_VALUE_TYPE (spec));
sl@0
   504
      g_assert (dbus_type != NULL);
sl@0
   505
      
sl@0
   506
      can_set = ((spec->flags & G_PARAM_WRITABLE) != 0 &&
sl@0
   507
		 (spec->flags & G_PARAM_CONSTRUCT_ONLY) == 0);
sl@0
   508
      
sl@0
   509
      can_get = (spec->flags & G_PARAM_READABLE) != 0;
sl@0
   510
      
sl@0
   511
      if (can_set || can_get)
sl@0
   512
	{
sl@0
   513
	  g_string_append_printf (xml, "    <property name=\"%s\" ", propname);
sl@0
   514
	  g_string_append (xml, "type=\"");
sl@0
   515
	  g_string_append (xml, dbus_type);
sl@0
   516
	  g_string_append (xml, "\" access=\"");
sl@0
   517
sl@0
   518
	  if (can_set && can_get)
sl@0
   519
	    g_string_append (xml, "readwrite");
sl@0
   520
	  else if (can_get)
sl@0
   521
	    g_string_append (xml, "read");
sl@0
   522
	  else
sl@0
   523
	    {
sl@0
   524
	      g_assert (can_set);
sl@0
   525
	      g_string_append (xml, "write");
sl@0
   526
	    }
sl@0
   527
          
sl@0
   528
	  g_string_append (xml, "\"/>\n");
sl@0
   529
	}
sl@0
   530
      
sl@0
   531
      g_free (dbus_type);
sl@0
   532
sl@0
   533
      g_string_append (xml, "    </property>\n");
sl@0
   534
    }
sl@0
   535
  g_slist_free (values->properties);
sl@0
   536
sl@0
   537
  g_free (values);
sl@0
   538
  g_string_append (xml, "  </interface>\n");
sl@0
   539
}
sl@0
   540
sl@0
   541
static DBusGLibWriteInterfaceValues *
sl@0
   542
lookup_values (GHashTable *interfaces, const char *method_interface)
sl@0
   543
{
sl@0
   544
  DBusGLibWriteInterfaceValues *values;
sl@0
   545
  if ((values = g_hash_table_lookup (interfaces, (gpointer) method_interface)) == NULL)
sl@0
   546
    {
sl@0
   547
      values = g_new0 (DBusGLibWriteInterfaceValues, 1);
sl@0
   548
      g_hash_table_insert (interfaces, (gpointer) method_interface, values);
sl@0
   549
    }
sl@0
   550
  return values;
sl@0
   551
}
sl@0
   552
sl@0
   553
static void
sl@0
   554
introspect_interfaces (GObject *object, GString *xml)
sl@0
   555
{
sl@0
   556
  GList *info_list;
sl@0
   557
  const GList *info_list_walk;
sl@0
   558
  const DBusGObjectInfo *info;
sl@0
   559
  DBusGLibWriteIterfaceData data;
sl@0
   560
  int i;
sl@0
   561
  GHashTable *interfaces;
sl@0
   562
  DBusGLibWriteInterfaceValues *values;
sl@0
   563
  const char *propsig;
sl@0
   564
sl@0
   565
  info_list = lookup_object_info (object);
sl@0
   566
sl@0
   567
  g_assert (info_list != NULL);
sl@0
   568
sl@0
   569
  /* Gather a list of all interfaces, indexed into their methods */
sl@0
   570
  for (info_list_walk = info_list; info_list_walk != NULL; info_list_walk = g_list_next (info_list_walk))
sl@0
   571
    {
sl@0
   572
      info = (DBusGObjectInfo *) info_list_walk->data;
sl@0
   573
      interfaces = g_hash_table_new (g_str_hash, g_str_equal);
sl@0
   574
      
sl@0
   575
      g_assert (info != NULL);
sl@0
   576
sl@0
   577
      for (i = 0; i < info->n_method_infos; i++)
sl@0
   578
        {
sl@0
   579
          #ifdef WINSCW
sl@0
   580
          const char *method_name;
sl@0
   581
           const char *method_args;
sl@0
   582
          #endif
sl@0
   583
          const char *method_interface;
sl@0
   584
         
sl@0
   585
        
sl@0
   586
          const DBusGMethodInfo *method;
sl@0
   587
sl@0
   588
          method = &(info->method_infos[i]);
sl@0
   589
sl@0
   590
          method_interface = method_interface_from_object_info (info, method);
sl@0
   591
          #ifdef WINSCW
sl@0
   592
          method_name = method_name_from_object_info (info, method);
sl@0
   593
         
sl@0
   594
          method_args = method_arg_info_from_object_info (info, method);
sl@0
   595
          #endif
sl@0
   596
          values = lookup_values (interfaces, method_interface);
sl@0
   597
          values->methods = g_slist_prepend (values->methods, (gpointer) method);
sl@0
   598
        }
sl@0
   599
sl@0
   600
      propsig = info->exported_signals;
sl@0
   601
      while (*propsig)
sl@0
   602
        {
sl@0
   603
          const char *iface;
sl@0
   604
          const char *signame;
sl@0
   605
sl@0
   606
          propsig = propsig_iterate (propsig, &iface, &signame);
sl@0
   607
sl@0
   608
          values = lookup_values (interfaces, iface);
sl@0
   609
          values->signals = g_slist_prepend (values->signals, (gpointer) signame);
sl@0
   610
        }
sl@0
   611
sl@0
   612
      propsig = info->exported_properties;
sl@0
   613
      while (*propsig)
sl@0
   614
        {
sl@0
   615
          const char *iface;
sl@0
   616
          const char *propname;
sl@0
   617
sl@0
   618
          propsig = propsig_iterate (propsig, &iface, &propname);
sl@0
   619
sl@0
   620
          values = lookup_values (interfaces, iface);
sl@0
   621
          values->properties = g_slist_prepend (values->properties, (gpointer) propname);
sl@0
   622
        }
sl@0
   623
      
sl@0
   624
      memset (&data, 0, sizeof (data));
sl@0
   625
      data.xml = xml;
sl@0
   626
      data.gtype = G_TYPE_FROM_INSTANCE (object);
sl@0
   627
      data.object_info = info;
sl@0
   628
sl@0
   629
      g_hash_table_foreach (interfaces, write_interface, &data);
sl@0
   630
      g_hash_table_destroy (interfaces);
sl@0
   631
    }
sl@0
   632
sl@0
   633
  g_list_free (info_list);
sl@0
   634
}
sl@0
   635
sl@0
   636
static DBusHandlerResult
sl@0
   637
handle_introspect (DBusConnection *connection,
sl@0
   638
                   DBusMessage    *message,
sl@0
   639
                   GObject        *object)
sl@0
   640
{
sl@0
   641
  GString *xml;
sl@0
   642
  unsigned int i;
sl@0
   643
  DBusMessage *ret;
sl@0
   644
  char **children;
sl@0
   645
  
sl@0
   646
  if (!dbus_connection_list_registered (connection, 
sl@0
   647
                                        dbus_message_get_path (message),
sl@0
   648
                                        &children))
sl@0
   649
    g_error ("Out of memory");
sl@0
   650
  
sl@0
   651
  xml = g_string_new (NULL);
sl@0
   652
sl@0
   653
  g_string_append (xml, DBUS_INTROSPECT_1_0_XML_DOCTYPE_DECL_NODE);
sl@0
   654
  
sl@0
   655
  g_string_append (xml, "<node>\n");
sl@0
   656
sl@0
   657
  /* We are introspectable, though I guess that was pretty obvious */
sl@0
   658
  g_string_append_printf (xml, "  <interface name=\"%s\">\n", DBUS_INTERFACE_INTROSPECTABLE);
sl@0
   659
  g_string_append (xml, "    <method name=\"Introspect\">\n");
sl@0
   660
  g_string_append_printf (xml, "      <arg name=\"data\" direction=\"out\" type=\"%s\"/>\n", DBUS_TYPE_STRING_AS_STRING);
sl@0
   661
  g_string_append (xml, "    </method>\n");
sl@0
   662
  g_string_append (xml, "  </interface>\n");
sl@0
   663
sl@0
   664
  /* We support get/set properties */
sl@0
   665
  g_string_append_printf (xml, "  <interface name=\"%s\">\n", DBUS_INTERFACE_PROPERTIES);
sl@0
   666
  g_string_append (xml, "    <method name=\"Get\">\n");
sl@0
   667
  g_string_append_printf (xml, "      <arg name=\"interface\" direction=\"in\" type=\"%s\"/>\n", DBUS_TYPE_STRING_AS_STRING);
sl@0
   668
  g_string_append_printf (xml, "      <arg name=\"propname\" direction=\"in\" type=\"%s\"/>\n", DBUS_TYPE_STRING_AS_STRING);
sl@0
   669
  g_string_append_printf (xml, "      <arg name=\"value\" direction=\"out\" type=\"%s\"/>\n", DBUS_TYPE_VARIANT_AS_STRING);
sl@0
   670
  g_string_append (xml, "    </method>\n");
sl@0
   671
  g_string_append (xml, "    <method name=\"Set\">\n");
sl@0
   672
  g_string_append_printf (xml, "      <arg name=\"interface\" direction=\"in\" type=\"%s\"/>\n", DBUS_TYPE_STRING_AS_STRING);
sl@0
   673
  g_string_append_printf (xml, "      <arg name=\"propname\" direction=\"in\" type=\"%s\"/>\n", DBUS_TYPE_STRING_AS_STRING);
sl@0
   674
  g_string_append_printf (xml, "      <arg name=\"value\" direction=\"in\" type=\"%s\"/>\n", DBUS_TYPE_VARIANT_AS_STRING);
sl@0
   675
  g_string_append (xml, "    </method>\n");
sl@0
   676
  g_string_append (xml, "  </interface>\n");
sl@0
   677
  
sl@0
   678
  introspect_interfaces (object, xml);
sl@0
   679
sl@0
   680
  /* Append child nodes */
sl@0
   681
  for (i = 0; children[i]; i++)
sl@0
   682
      g_string_append_printf (xml, "  <node name=\"%s\"/>\n",
sl@0
   683
                              children[i]);
sl@0
   684
  
sl@0
   685
  /* Close the XML, and send it to the requesting app */
sl@0
   686
  g_string_append (xml, "</node>\n");
sl@0
   687
sl@0
   688
  ret = dbus_message_new_method_return (message);
sl@0
   689
  if (ret == NULL)
sl@0
   690
    g_error ("Out of memory");
sl@0
   691
sl@0
   692
  dbus_message_append_args (ret,
sl@0
   693
                            DBUS_TYPE_STRING, &xml->str,
sl@0
   694
                            DBUS_TYPE_INVALID);
sl@0
   695
sl@0
   696
  dbus_connection_send (connection, ret, NULL);
sl@0
   697
  dbus_message_unref (ret);
sl@0
   698
sl@0
   699
  g_string_free (xml, TRUE);
sl@0
   700
sl@0
   701
  dbus_free_string_array (children);
sl@0
   702
  
sl@0
   703
  return DBUS_HANDLER_RESULT_HANDLED;
sl@0
   704
}
sl@0
   705
sl@0
   706
static DBusMessage*
sl@0
   707
set_object_property (DBusConnection  *connection,
sl@0
   708
                     DBusMessage     *message,
sl@0
   709
                     DBusMessageIter *iter,
sl@0
   710
                     GObject         *object,
sl@0
   711
                     GParamSpec      *pspec)
sl@0
   712
{
sl@0
   713
  GValue value = { 0, };
sl@0
   714
  DBusMessage *ret;
sl@0
   715
  DBusMessageIter sub;
sl@0
   716
  DBusGValueMarshalCtx context;
sl@0
   717
sl@0
   718
  dbus_message_iter_recurse (iter, &sub);
sl@0
   719
sl@0
   720
  context.gconnection = DBUS_G_CONNECTION_FROM_CONNECTION (connection);
sl@0
   721
  context.proxy = NULL;
sl@0
   722
sl@0
   723
  g_value_init (&value, pspec->value_type);
sl@0
   724
  if (_dbus_gvalue_demarshal (&context, &sub, &value, NULL))
sl@0
   725
    {
sl@0
   726
      g_object_set_property (object,
sl@0
   727
                             pspec->name,
sl@0
   728
                             &value);
sl@0
   729
sl@0
   730
      g_value_unset (&value);
sl@0
   731
sl@0
   732
      ret = dbus_message_new_method_return (message);
sl@0
   733
      if (ret == NULL)
sl@0
   734
        g_error ("out of memory");
sl@0
   735
    }
sl@0
   736
  else
sl@0
   737
    {
sl@0
   738
      ret = dbus_message_new_error (message,
sl@0
   739
                                    DBUS_ERROR_INVALID_ARGS,
sl@0
   740
                                    "Argument's D-BUS type can't be converted to a GType");
sl@0
   741
      if (ret == NULL)
sl@0
   742
        g_error ("out of memory");
sl@0
   743
    }
sl@0
   744
sl@0
   745
  return ret;
sl@0
   746
}
sl@0
   747
sl@0
   748
static DBusMessage*
sl@0
   749
get_object_property (DBusConnection *connection,
sl@0
   750
                     DBusMessage    *message,
sl@0
   751
                     GObject        *object,
sl@0
   752
                     GParamSpec     *pspec)
sl@0
   753
{
sl@0
   754
  GType value_gtype;
sl@0
   755
  GValue value = {0, };
sl@0
   756
  gchar *variant_sig;
sl@0
   757
  DBusMessage *ret;
sl@0
   758
  DBusMessageIter iter, subiter;
sl@0
   759
sl@0
   760
  ret = dbus_message_new_method_return (message);
sl@0
   761
  if (ret == NULL)
sl@0
   762
    g_error ("out of memory");
sl@0
   763
sl@0
   764
sl@0
   765
  g_value_init (&value, pspec->value_type);
sl@0
   766
  g_object_get_property (object, pspec->name, &value);
sl@0
   767
sl@0
   768
  variant_sig = _dbus_gvalue_to_signature (&value);
sl@0
   769
  if (variant_sig == NULL)
sl@0
   770
    {
sl@0
   771
      value_gtype = G_VALUE_TYPE (&value);
sl@0
   772
      g_warning ("Cannot marshal type \"%s\" in variant", g_type_name (value_gtype));
sl@0
   773
      g_value_unset (&value);
sl@0
   774
      return ret;
sl@0
   775
    }
sl@0
   776
sl@0
   777
  dbus_message_iter_init_append (ret, &iter);
sl@0
   778
  if (!dbus_message_iter_open_container (&iter,
sl@0
   779
					 DBUS_TYPE_VARIANT,
sl@0
   780
					 variant_sig,
sl@0
   781
					 &subiter))
sl@0
   782
    {
sl@0
   783
      g_free (variant_sig);
sl@0
   784
      g_value_unset (&value);
sl@0
   785
      return ret;
sl@0
   786
    }
sl@0
   787
sl@0
   788
  if (!_dbus_gvalue_marshal (&subiter, &value))
sl@0
   789
    {
sl@0
   790
      dbus_message_unref (ret);
sl@0
   791
      ret = dbus_message_new_error (message,
sl@0
   792
                                    DBUS_ERROR_UNKNOWN_METHOD,
sl@0
   793
                                    "Can't convert GType of object property to a D-BUS type");
sl@0
   794
    }
sl@0
   795
sl@0
   796
  dbus_message_iter_close_container (&iter, &subiter);
sl@0
   797
sl@0
   798
  g_value_unset (&value);
sl@0
   799
  g_free (variant_sig);
sl@0
   800
sl@0
   801
  return ret;
sl@0
   802
}
sl@0
   803
sl@0
   804
static gboolean
sl@0
   805
lookup_object_and_method (GObject      *object,
sl@0
   806
			  DBusMessage  *message,
sl@0
   807
			  const DBusGObjectInfo **object_ret,
sl@0
   808
			  const DBusGMethodInfo **method_ret)
sl@0
   809
{
sl@0
   810
  const char *interface;
sl@0
   811
  const char *member;
sl@0
   812
  const char *signature;
sl@0
   813
  GList *info_list;
sl@0
   814
  const GList *info_list_walk;
sl@0
   815
  const DBusGObjectInfo *info;
sl@0
   816
  int i;
sl@0
   817
sl@0
   818
  interface = dbus_message_get_interface (message);
sl@0
   819
  member = dbus_message_get_member (message);
sl@0
   820
  signature = dbus_message_get_signature (message);
sl@0
   821
sl@0
   822
  info_list = lookup_object_info (object);
sl@0
   823
  
sl@0
   824
  for (info_list_walk = info_list; info_list_walk != NULL; info_list_walk = g_list_next (info_list_walk))
sl@0
   825
    {
sl@0
   826
      info = (DBusGObjectInfo *) info_list_walk->data;
sl@0
   827
      *object_ret = info;
sl@0
   828
sl@0
   829
      for (i = 0; i < info->n_method_infos; i++)
sl@0
   830
        {
sl@0
   831
          const char *expected_member;
sl@0
   832
          const char *expected_interface;
sl@0
   833
          char *expected_signature;
sl@0
   834
          const DBusGMethodInfo *method;
sl@0
   835
sl@0
   836
          method = &(info->method_infos[i]);
sl@0
   837
sl@0
   838
          /* Check method interface/name and input signature */ 
sl@0
   839
          expected_interface = method_interface_from_object_info (*object_ret, method);
sl@0
   840
          expected_member = method_name_from_object_info (*object_ret, method);
sl@0
   841
          expected_signature = method_input_signature_from_object_info (*object_ret, method);
sl@0
   842
sl@0
   843
          if ((interface == NULL
sl@0
   844
              || strcmp (expected_interface, interface) == 0)
sl@0
   845
              && strcmp (expected_member, member) == 0
sl@0
   846
              && strcmp (expected_signature, signature) == 0)
sl@0
   847
            {
sl@0
   848
              g_free (expected_signature);
sl@0
   849
              *method_ret = method;
sl@0
   850
              g_list_free (info_list);
sl@0
   851
              return TRUE;
sl@0
   852
            }
sl@0
   853
            g_free (expected_signature);
sl@0
   854
        }
sl@0
   855
    }
sl@0
   856
sl@0
   857
  if (info_list)
sl@0
   858
    g_list_free (info_list);
sl@0
   859
sl@0
   860
  return FALSE;
sl@0
   861
}
sl@0
   862
sl@0
   863
static char *
sl@0
   864
gerror_domaincode_to_dbus_error_name (const DBusGObjectInfo *object_info,
sl@0
   865
				      const char *msg_interface,
sl@0
   866
				      GQuark domain, gint code)
sl@0
   867
{
sl@0
   868
  const char *domain_str;
sl@0
   869
  const char *code_str;
sl@0
   870
  GString *dbus_error_name;
sl@0
   871
sl@0
   872
  domain_str = object_error_domain_prefix_from_object_info (object_info);
sl@0
   873
  code_str = object_error_code_from_object_info (object_info, domain, code);
sl@0
   874
sl@0
   875
  if (!domain_str || !code_str)
sl@0
   876
    {
sl@0
   877
      DBusGErrorInfo *info;
sl@0
   878
sl@0
   879
      g_static_rw_lock_reader_lock (&globals_lock);
sl@0
   880
sl@0
   881
      if (error_metadata != NULL)
sl@0
   882
	info = g_datalist_id_get_data (&error_metadata, domain);
sl@0
   883
      else
sl@0
   884
	info = NULL;
sl@0
   885
sl@0
   886
      g_static_rw_lock_reader_unlock (&globals_lock);
sl@0
   887
sl@0
   888
      if (info)
sl@0
   889
	{
sl@0
   890
	  GEnumValue *value;
sl@0
   891
	  GEnumClass *klass;
sl@0
   892
sl@0
   893
	  klass = g_type_class_ref (info->code_enum);
sl@0
   894
	  value = g_enum_get_value (klass, code);
sl@0
   895
	  g_type_class_unref (klass);
sl@0
   896
sl@0
   897
	  domain_str = info->default_iface;
sl@0
   898
	  code_str = value->value_nick;
sl@0
   899
	}
sl@0
   900
    }
sl@0
   901
sl@0
   902
  if (!domain_str)
sl@0
   903
    domain_str = msg_interface;
sl@0
   904
sl@0
   905
  if (!domain_str || !code_str)
sl@0
   906
    {
sl@0
   907
      /* If we can't map it sensibly, make up an error name */
sl@0
   908
      char *domain_from_quark;
sl@0
   909
      
sl@0
   910
      dbus_error_name = g_string_new ("org.freedesktop.DBus.GLib.UnmappedError.");
sl@0
   911
sl@0
   912
      domain_from_quark = uscore_to_wincaps (g_quark_to_string (domain));
sl@0
   913
      g_string_append (dbus_error_name, domain_from_quark);
sl@0
   914
      g_free (domain_from_quark);
sl@0
   915
	
sl@0
   916
      g_string_append_printf (dbus_error_name, ".Code%d", code);
sl@0
   917
    }
sl@0
   918
  else
sl@0
   919
    {
sl@0
   920
      dbus_error_name = g_string_new (domain_str);
sl@0
   921
      g_string_append_c (dbus_error_name, '.');
sl@0
   922
      g_string_append (dbus_error_name, code_str);
sl@0
   923
    }
sl@0
   924
sl@0
   925
  return g_string_free (dbus_error_name, FALSE);
sl@0
   926
}
sl@0
   927
sl@0
   928
static DBusMessage *
sl@0
   929
gerror_to_dbus_error_message (const DBusGObjectInfo *object_info,
sl@0
   930
			      DBusMessage     *message,
sl@0
   931
			      GError          *error)
sl@0
   932
{
sl@0
   933
  DBusMessage *reply;
sl@0
   934
sl@0
   935
  if (!error)
sl@0
   936
    {
sl@0
   937
      char *error_msg;
sl@0
   938
      
sl@0
   939
      error_msg = g_strdup_printf ("Method invoked for %s returned FALSE but did not set error", dbus_message_get_member (message));
sl@0
   940
      reply = dbus_message_new_error (message, "org.freedesktop.DBus.GLib.ErrorError", error_msg);
sl@0
   941
      g_free (error_msg);
sl@0
   942
    }
sl@0
   943
  else
sl@0
   944
    {
sl@0
   945
      if (error->domain == DBUS_GERROR)
sl@0
   946
	reply = dbus_message_new_error (message,
sl@0
   947
					dbus_g_error_get_name (error),
sl@0
   948
					error->message);
sl@0
   949
      else
sl@0
   950
	{
sl@0
   951
	  char *error_name;
sl@0
   952
	  error_name = gerror_domaincode_to_dbus_error_name (object_info,
sl@0
   953
							     dbus_message_get_interface (message),
sl@0
   954
							     error->domain, error->code);
sl@0
   955
	  reply = dbus_message_new_error (message, error_name, error->message);
sl@0
   956
	  g_free (error_name); 
sl@0
   957
	}
sl@0
   958
    }
sl@0
   959
  return reply;
sl@0
   960
}
sl@0
   961
sl@0
   962
/**
sl@0
   963
 * SECTION:dbus-gmethod
sl@0
   964
 * @short_description: GMethod Info & Invocation
sl@0
   965
 * @see_also: #DBusGMessage
sl@0
   966
 * @stability: Stable
sl@0
   967
 *
sl@0
   968
 * These types are used to call methods on #GObject objects.
sl@0
   969
 */
sl@0
   970
sl@0
   971
/**
sl@0
   972
 * The context of an asynchronous method call.  See dbus_g_method_return() and
sl@0
   973
 * dbus_g_method_return_error().
sl@0
   974
 */
sl@0
   975
struct _DBusGMethodInvocation {
sl@0
   976
  DBusGConnection *connection; /**< The connection */
sl@0
   977
  DBusGMessage *message; /**< The message which generated the method call */
sl@0
   978
  const DBusGObjectInfo *object; /**< The object the method was called on */
sl@0
   979
  const DBusGMethodInfo *method; /**< The method called */
sl@0
   980
};
sl@0
   981
sl@0
   982
static DBusHandlerResult
sl@0
   983
invoke_object_method (GObject         *object,
sl@0
   984
		      const DBusGObjectInfo *object_info,
sl@0
   985
		      const DBusGMethodInfo *method,
sl@0
   986
		      DBusConnection  *connection,
sl@0
   987
		      DBusMessage     *message)
sl@0
   988
{
sl@0
   989
  gboolean had_error, call_only;
sl@0
   990
  GError *gerror;
sl@0
   991
  GValueArray *value_array;
sl@0
   992
  GValue return_value = {0,};
sl@0
   993
  GClosure closure;
sl@0
   994
  char *in_signature;
sl@0
   995
  GArray *out_param_values = NULL;
sl@0
   996
  GValueArray *out_param_gvalues = NULL;
sl@0
   997
  int out_param_count;
sl@0
   998
  int out_param_pos, out_param_gvalue_pos;
sl@0
   999
  DBusHandlerResult result;
sl@0
  1000
  DBusMessage *reply;
sl@0
  1001
  gboolean have_retval;
sl@0
  1002
  gboolean retval_signals_error;
sl@0
  1003
  gboolean retval_is_synthetic;
sl@0
  1004
  gboolean retval_is_constant;
sl@0
  1005
  const char *arg_metadata;
sl@0
  1006
sl@0
  1007
  gerror = NULL;
sl@0
  1008
sl@0
  1009
  /* Determine whether or not this method should be invoked in a new
sl@0
  1010
     thread
sl@0
  1011
   */
sl@0
  1012
  if (strcmp (string_table_lookup (get_method_data (object_info, method), 2), "A") == 0)
sl@0
  1013
    call_only = TRUE;
sl@0
  1014
  else
sl@0
  1015
    call_only = FALSE;
sl@0
  1016
sl@0
  1017
  have_retval = FALSE;
sl@0
  1018
  retval_signals_error = FALSE;
sl@0
  1019
  retval_is_synthetic = FALSE;
sl@0
  1020
  retval_is_constant = FALSE;
sl@0
  1021
sl@0
  1022
  /* This is evil.  We do this to work around the fact that
sl@0
  1023
   * the generated glib marshallers check a flag in the closure object
sl@0
  1024
   * which we don't care about.  We don't need/want to create
sl@0
  1025
   * a new closure for each invocation.
sl@0
  1026
   */
sl@0
  1027
  memset (&closure, 0, sizeof (closure));
sl@0
  1028
sl@0
  1029
  in_signature = method_input_signature_from_object_info (object_info, method); 
sl@0
  1030
  
sl@0
  1031
  /* Convert method IN parameters to GValueArray */
sl@0
  1032
  {
sl@0
  1033
    GArray *types_array;
sl@0
  1034
    guint n_params;
sl@0
  1035
    const GType *types;
sl@0
  1036
    DBusGValueMarshalCtx context;
sl@0
  1037
    GError *error = NULL;
sl@0
  1038
    
sl@0
  1039
    context.gconnection = DBUS_G_CONNECTION_FROM_CONNECTION (connection);
sl@0
  1040
    context.proxy = NULL;
sl@0
  1041
sl@0
  1042
    types_array = _dbus_gtypes_from_arg_signature (in_signature, FALSE);
sl@0
  1043
    n_params = types_array->len;
sl@0
  1044
    types = (const GType*) types_array->data;
sl@0
  1045
sl@0
  1046
    value_array = _dbus_gvalue_demarshal_message (&context, message, n_params, types, &error);
sl@0
  1047
    if (value_array == NULL)
sl@0
  1048
      {
sl@0
  1049
	g_free (in_signature); 
sl@0
  1050
	g_array_free (types_array, TRUE);
sl@0
  1051
	reply = dbus_message_new_error (message, "org.freedesktop.DBus.GLib.ErrorError", error->message);
sl@0
  1052
	dbus_connection_send (connection, reply, NULL);
sl@0
  1053
	dbus_message_unref (reply);
sl@0
  1054
	g_error_free (error);
sl@0
  1055
	return DBUS_HANDLER_RESULT_HANDLED;
sl@0
  1056
      }
sl@0
  1057
    g_array_free (types_array, TRUE);
sl@0
  1058
  }
sl@0
  1059
sl@0
  1060
  /* Prepend object as first argument */ 
sl@0
  1061
  g_value_array_prepend (value_array, NULL);
sl@0
  1062
  g_value_init (g_value_array_get_nth (value_array, 0), G_TYPE_OBJECT);
sl@0
  1063
  g_value_set_object (g_value_array_get_nth (value_array, 0), object);
sl@0
  1064
  
sl@0
  1065
  if (call_only)
sl@0
  1066
    {
sl@0
  1067
      GValue context_value = {0,};
sl@0
  1068
      DBusGMethodInvocation *context;
sl@0
  1069
      context = g_new (DBusGMethodInvocation, 1);
sl@0
  1070
      context->connection = dbus_g_connection_ref (DBUS_G_CONNECTION_FROM_CONNECTION (connection));
sl@0
  1071
      context->message = dbus_g_message_ref (DBUS_G_MESSAGE_FROM_MESSAGE (message));
sl@0
  1072
      context->object = object_info;
sl@0
  1073
      context->method = method;
sl@0
  1074
      g_value_init (&context_value, G_TYPE_POINTER);
sl@0
  1075
      g_value_set_pointer (&context_value, context);
sl@0
  1076
      g_value_array_append (value_array, &context_value);
sl@0
  1077
    }
sl@0
  1078
  else
sl@0
  1079
    {
sl@0
  1080
      RetvalType retval;
sl@0
  1081
      gboolean arg_in;
sl@0
  1082
      gboolean arg_const;
sl@0
  1083
      const char *argsig;
sl@0
  1084
sl@0
  1085
      arg_metadata = method_arg_info_from_object_info (object_info, method);
sl@0
  1086
      
sl@0
  1087
      /* Count number of output parameters, and look for a return value */
sl@0
  1088
      out_param_count = 0;
sl@0
  1089
      while (*arg_metadata)
sl@0
  1090
	{
sl@0
  1091
	  arg_metadata = arg_iterate (arg_metadata, NULL, &arg_in, &arg_const, &retval, &argsig);
sl@0
  1092
	  if (arg_in)
sl@0
  1093
	    continue;
sl@0
  1094
	  if (retval != RETVAL_NONE)
sl@0
  1095
	    {
sl@0
  1096
	      DBusSignatureIter tmp_sigiter;
sl@0
  1097
	      /* This is the function return value */
sl@0
  1098
	      g_assert (!have_retval);
sl@0
  1099
	      have_retval = TRUE;
sl@0
  1100
	      retval_is_synthetic = FALSE;
sl@0
  1101
sl@0
  1102
	      switch (retval)
sl@0
  1103
		{
sl@0
  1104
		case RETVAL_NONE:
sl@0
  1105
		  g_assert_not_reached ();
sl@0
  1106
		  break;
sl@0
  1107
		case RETVAL_NOERROR:
sl@0
  1108
		  retval_signals_error = FALSE;
sl@0
  1109
		  break;
sl@0
  1110
		case RETVAL_ERROR:
sl@0
  1111
		  retval_signals_error = TRUE;
sl@0
  1112
		  break;
sl@0
  1113
		}
sl@0
  1114
sl@0
  1115
	      retval_is_constant = arg_const;
sl@0
  1116
sl@0
  1117
	      /* Initialize our return GValue with the specified type */
sl@0
  1118
	      dbus_signature_iter_init (&tmp_sigiter, argsig);
sl@0
  1119
	      g_value_init (&return_value, _dbus_gtype_from_signature_iter (&tmp_sigiter, FALSE));
sl@0
  1120
	    }
sl@0
  1121
	  else
sl@0
  1122
	    {
sl@0
  1123
	      /* It's a regular output value */
sl@0
  1124
	      out_param_count++;
sl@0
  1125
	    }
sl@0
  1126
	}
sl@0
  1127
sl@0
  1128
      /* For compatibility, if we haven't found a return value, we assume
sl@0
  1129
       * the function returns a gboolean for signalling an error
sl@0
  1130
       * (and therefore also takes a GError).  We also note that it
sl@0
  1131
       * is a "synthetic" return value; i.e. we aren't going to be
sl@0
  1132
       * sending it over the bus, it's just to signal an error.
sl@0
  1133
       */
sl@0
  1134
      if (!have_retval)
sl@0
  1135
	{
sl@0
  1136
	  have_retval = TRUE;
sl@0
  1137
	  retval_is_synthetic = TRUE;
sl@0
  1138
	  retval_signals_error = TRUE;
sl@0
  1139
	  g_value_init (&return_value, G_TYPE_BOOLEAN);
sl@0
  1140
	}
sl@0
  1141
sl@0
  1142
      /* Create an array to store the actual values of OUT parameters
sl@0
  1143
       * (other than the real function return, if any).  Then, create
sl@0
  1144
       * a GValue boxed POINTER to each of those values, and append to
sl@0
  1145
       * the invocation, so the method can return the OUT parameters.
sl@0
  1146
       */
sl@0
  1147
      out_param_values = g_array_sized_new (FALSE, TRUE, sizeof (GTypeCValue), out_param_count);
sl@0
  1148
sl@0
  1149
      /* We have a special array of GValues for toplevel GValue return
sl@0
  1150
       * types.
sl@0
  1151
       */
sl@0
  1152
      out_param_gvalues = g_value_array_new (out_param_count);
sl@0
  1153
      out_param_pos = 0;
sl@0
  1154
      out_param_gvalue_pos = 0;
sl@0
  1155
sl@0
  1156
      /* Reset argument metadata pointer */
sl@0
  1157
      arg_metadata = method_arg_info_from_object_info (object_info, method);
sl@0
  1158
      
sl@0
  1159
      /* Iterate over output arguments again, this time allocating space for
sl@0
  1160
       * them as appopriate.
sl@0
  1161
       */
sl@0
  1162
      while (*arg_metadata)
sl@0
  1163
	{
sl@0
  1164
	  GValue value = {0, };
sl@0
  1165
	  GTypeCValue storage;
sl@0
  1166
	  DBusSignatureIter tmp_sigiter;
sl@0
  1167
	  GType current_gtype;
sl@0
  1168
sl@0
  1169
	  arg_metadata = arg_iterate (arg_metadata, NULL, &arg_in, NULL, &retval, &argsig);
sl@0
  1170
	  /* Skip over input arguments and the return value, if any */
sl@0
  1171
	  if (arg_in || retval != RETVAL_NONE)
sl@0
  1172
	    continue;
sl@0
  1173
sl@0
  1174
	  dbus_signature_iter_init (&tmp_sigiter, argsig);
sl@0
  1175
	  current_gtype = _dbus_gtype_from_signature_iter (&tmp_sigiter, FALSE);
sl@0
  1176
sl@0
  1177
	  g_value_init (&value, G_TYPE_POINTER);
sl@0
  1178
sl@0
  1179
	  /* We special case variants to make method invocation a bit nicer */
sl@0
  1180
	  if (current_gtype != G_TYPE_VALUE)
sl@0
  1181
	    {
sl@0
  1182
	      memset (&storage, 0, sizeof (storage));
sl@0
  1183
	      g_array_append_val (out_param_values, storage);
sl@0
  1184
	      g_value_set_pointer (&value, &(g_array_index (out_param_values, GTypeCValue, out_param_pos)));
sl@0
  1185
	      out_param_pos++;
sl@0
  1186
	    }
sl@0
  1187
	  else
sl@0
  1188
	    {
sl@0
  1189
	      g_value_array_append (out_param_gvalues, NULL);
sl@0
  1190
	      g_value_set_pointer (&value, out_param_gvalues->values + out_param_gvalue_pos);
sl@0
  1191
	      out_param_gvalue_pos++;
sl@0
  1192
	    }
sl@0
  1193
	  g_value_array_append (value_array, &value);
sl@0
  1194
	}
sl@0
  1195
    }
sl@0
  1196
sl@0
  1197
  /* Append GError as final argument if necessary */
sl@0
  1198
  if (retval_signals_error)
sl@0
  1199
    {
sl@0
  1200
      g_assert (have_retval);
sl@0
  1201
      g_value_array_append (value_array, NULL);
sl@0
  1202
      g_value_init (g_value_array_get_nth (value_array, value_array->n_values - 1), G_TYPE_POINTER);
sl@0
  1203
      g_value_set_pointer (g_value_array_get_nth (value_array, value_array->n_values - 1), &gerror);
sl@0
  1204
    }
sl@0
  1205
  
sl@0
  1206
  /* Actually invoke method */
sl@0
  1207
  method->marshaller (&closure, have_retval ? &return_value : NULL,
sl@0
  1208
		      value_array->n_values,
sl@0
  1209
		      value_array->values,
sl@0
  1210
		      NULL, 
sl@0
  1211
		      #ifdef WINSCW
sl@0
  1212
		      method->function);
sl@0
  1213
		      #else
sl@0
  1214
		      (gpointer)method->function);
sl@0
  1215
		      #endif
sl@0
  1216
		      
sl@0
  1217
  if (call_only)
sl@0
  1218
    {
sl@0
  1219
      result = DBUS_HANDLER_RESULT_HANDLED;
sl@0
  1220
      goto done;
sl@0
  1221
    }
sl@0
  1222
  if (retval_signals_error)
sl@0
  1223
    had_error = _dbus_gvalue_signals_error (&return_value);
sl@0
  1224
  else
sl@0
  1225
    had_error = FALSE;
sl@0
  1226
sl@0
  1227
  if (!had_error)
sl@0
  1228
    {
sl@0
  1229
      DBusMessageIter iter;
sl@0
  1230
sl@0
  1231
      reply = dbus_message_new_method_return (message);
sl@0
  1232
      if (reply == NULL)
sl@0
  1233
	goto nomem;
sl@0
  1234
sl@0
  1235
      /* Append output arguments to reply */
sl@0
  1236
      dbus_message_iter_init_append (reply, &iter);
sl@0
  1237
sl@0
  1238
      /* First, append the return value, unless it's synthetic */
sl@0
  1239
      if (have_retval && !retval_is_synthetic)
sl@0
  1240
	{
sl@0
  1241
	  if (!_dbus_gvalue_marshal (&iter, &return_value))
sl@0
  1242
	    goto nomem;
sl@0
  1243
	  if (!retval_is_constant)
sl@0
  1244
	    g_value_unset (&return_value);
sl@0
  1245
	}
sl@0
  1246
sl@0
  1247
      /* Grab the argument metadata and iterate over it */
sl@0
  1248
      arg_metadata = method_arg_info_from_object_info (object_info, method);
sl@0
  1249
      
sl@0
  1250
      /* Now append any remaining return values */
sl@0
  1251
      out_param_pos = 0;
sl@0
  1252
      out_param_gvalue_pos = 0;
sl@0
  1253
      while (*arg_metadata)
sl@0
  1254
	{
sl@0
  1255
	  GValue gvalue = {0, };
sl@0
  1256
	  const char *arg_name;
sl@0
  1257
	  gboolean arg_in;
sl@0
  1258
	  gboolean constval;
sl@0
  1259
	  RetvalType retval;
sl@0
  1260
	  const char *arg_signature;
sl@0
  1261
	  DBusSignatureIter argsigiter;
sl@0
  1262
sl@0
  1263
	  do
sl@0
  1264
	    {
sl@0
  1265
	      /* Iterate over only output values; skip over input
sl@0
  1266
		 arguments and the return value */
sl@0
  1267
	      arg_metadata = arg_iterate (arg_metadata, &arg_name, &arg_in, &constval, &retval, &arg_signature);
sl@0
  1268
	    }
sl@0
  1269
	  while ((arg_in || retval != RETVAL_NONE) && *arg_metadata);
sl@0
  1270
sl@0
  1271
	  /* If the last argument we saw was input or the return
sl@0
  1272
  	   * value, we must be done iterating over output arguments.
sl@0
  1273
	   */
sl@0
  1274
	  if (arg_in || retval != RETVAL_NONE)
sl@0
  1275
	    break;
sl@0
  1276
sl@0
  1277
	  dbus_signature_iter_init (&argsigiter, arg_signature);
sl@0
  1278
	  
sl@0
  1279
	  g_value_init (&gvalue, _dbus_gtype_from_signature_iter (&argsigiter, FALSE));
sl@0
  1280
	  if (G_VALUE_TYPE (&gvalue) != G_TYPE_VALUE)
sl@0
  1281
	    {
sl@0
  1282
	      if (!_dbus_gvalue_take (&gvalue,
sl@0
  1283
				     &(g_array_index (out_param_values, GTypeCValue, out_param_pos))))
sl@0
  1284
		g_assert_not_reached ();
sl@0
  1285
	      out_param_pos++;
sl@0
  1286
	    }
sl@0
  1287
	  else
sl@0
  1288
	    {
sl@0
  1289
	      g_value_set_static_boxed (&gvalue, out_param_gvalues->values + out_param_gvalue_pos);
sl@0
  1290
	      out_param_gvalue_pos++;
sl@0
  1291
	    }
sl@0
  1292
	      
sl@0
  1293
	  if (!_dbus_gvalue_marshal (&iter, &gvalue))
sl@0
  1294
	    goto nomem;
sl@0
  1295
	  /* Here we actually free the allocated value; we
sl@0
  1296
	   * took ownership of it with _dbus_gvalue_take, unless
sl@0
  1297
	   * an annotation has specified this value as constant.
sl@0
  1298
	   */
sl@0
  1299
	  if (!constval)
sl@0
  1300
	    g_value_unset (&gvalue);
sl@0
  1301
	}
sl@0
  1302
    }
sl@0
  1303
  else
sl@0
  1304
    reply = gerror_to_dbus_error_message (object_info, message, gerror);
sl@0
  1305
sl@0
  1306
  if (reply)
sl@0
  1307
    {
sl@0
  1308
      dbus_connection_send (connection, reply, NULL);
sl@0
  1309
      dbus_message_unref (reply);
sl@0
  1310
    }
sl@0
  1311
sl@0
  1312
  result = DBUS_HANDLER_RESULT_HANDLED;
sl@0
  1313
 done:
sl@0
  1314
  g_free (in_signature);
sl@0
  1315
  if (!call_only)
sl@0
  1316
    {
sl@0
  1317
      g_array_free (out_param_values, TRUE);
sl@0
  1318
      g_value_array_free (out_param_gvalues);
sl@0
  1319
    }
sl@0
  1320
  g_value_array_free (value_array);
sl@0
  1321
  return result;
sl@0
  1322
 nomem:
sl@0
  1323
  result = DBUS_HANDLER_RESULT_NEED_MEMORY;
sl@0
  1324
  goto done;
sl@0
  1325
}
sl@0
  1326
sl@0
  1327
static DBusHandlerResult
sl@0
  1328
gobject_message_function (DBusConnection  *connection,
sl@0
  1329
                          DBusMessage     *message,
sl@0
  1330
                          void            *user_data)
sl@0
  1331
{
sl@0
  1332
  GParamSpec *pspec;
sl@0
  1333
  GObject *object;
sl@0
  1334
  gboolean setter;
sl@0
  1335
  gboolean getter;
sl@0
  1336
  char *s;
sl@0
  1337
  const char *wincaps_propname;
sl@0
  1338
  /* const char *wincaps_propiface; */
sl@0
  1339
  DBusMessageIter iter;
sl@0
  1340
  const DBusGMethodInfo *method;
sl@0
  1341
  const DBusGObjectInfo *object_info;
sl@0
  1342
sl@0
  1343
  object = G_OBJECT (user_data);
sl@0
  1344
sl@0
  1345
  if (dbus_message_is_method_call (message,
sl@0
  1346
                                   DBUS_INTERFACE_INTROSPECTABLE,
sl@0
  1347
                                   "Introspect"))
sl@0
  1348
    return handle_introspect (connection, message, object);
sl@0
  1349
  
sl@0
  1350
  /* Try the metainfo, which lets us invoke methods */
sl@0
  1351
  if (lookup_object_and_method (object, message, &object_info, &method))
sl@0
  1352
    return invoke_object_method (object, object_info, method, connection, message);
sl@0
  1353
sl@0
  1354
  /* If no metainfo, we can still do properties and signals
sl@0
  1355
   * via standard GLib introspection
sl@0
  1356
   */
sl@0
  1357
  getter = FALSE;
sl@0
  1358
  setter = FALSE;
sl@0
  1359
  if (dbus_message_is_method_call (message,
sl@0
  1360
                                   DBUS_INTERFACE_PROPERTIES,
sl@0
  1361
                                   "Get"))
sl@0
  1362
    getter = TRUE;
sl@0
  1363
  else if (dbus_message_is_method_call (message,
sl@0
  1364
                                        DBUS_INTERFACE_PROPERTIES,
sl@0
  1365
                                        "Set"))
sl@0
  1366
    setter = TRUE;
sl@0
  1367
sl@0
  1368
  if (!(setter || getter))
sl@0
  1369
    return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
sl@0
  1370
sl@0
  1371
  dbus_message_iter_init (message, &iter);
sl@0
  1372
sl@0
  1373
  if (dbus_message_iter_get_arg_type (&iter) != DBUS_TYPE_STRING)
sl@0
  1374
    {
sl@0
  1375
      g_warning ("Property get or set does not have an interface string as first arg\n");
sl@0
  1376
      return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
sl@0
  1377
    }
sl@0
  1378
  /* We never use the interface name; if we did, we'd need to
sl@0
  1379
   * remember that it can be empty string for "pick one for me"
sl@0
  1380
   */
sl@0
  1381
  /* dbus_message_iter_get_basic (&iter, &wincaps_propiface); */
sl@0
  1382
  dbus_message_iter_next (&iter);
sl@0
  1383
sl@0
  1384
  if (dbus_message_iter_get_arg_type (&iter) != DBUS_TYPE_STRING)
sl@0
  1385
    {
sl@0
  1386
      g_warning ("Property get or set does not have a property name string as second arg\n");
sl@0
  1387
      return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
sl@0
  1388
    }
sl@0
  1389
  dbus_message_iter_get_basic (&iter, &wincaps_propname);
sl@0
  1390
  dbus_message_iter_next (&iter);
sl@0
  1391
  
sl@0
  1392
  s = _dbus_gutils_wincaps_to_uscore (wincaps_propname);
sl@0
  1393
sl@0
  1394
  pspec = g_object_class_find_property (G_OBJECT_GET_CLASS (object),
sl@0
  1395
                                        s);
sl@0
  1396
sl@0
  1397
  g_free (s);
sl@0
  1398
sl@0
  1399
  if (pspec != NULL)
sl@0
  1400
    {
sl@0
  1401
      DBusMessage *ret;
sl@0
  1402
sl@0
  1403
      if (setter)
sl@0
  1404
        {
sl@0
  1405
          if (dbus_message_iter_get_arg_type (&iter) != DBUS_TYPE_VARIANT)
sl@0
  1406
            {
sl@0
  1407
              g_warning ("Property set does not have a variant value as third arg\n");
sl@0
  1408
              return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
sl@0
  1409
            }
sl@0
  1410
          
sl@0
  1411
          ret = set_object_property (connection, message, &iter,
sl@0
  1412
                                     object, pspec);
sl@0
  1413
          dbus_message_iter_next (&iter);
sl@0
  1414
        }
sl@0
  1415
      else if (getter)
sl@0
  1416
        {
sl@0
  1417
          ret = get_object_property (connection, message,
sl@0
  1418
                                     object, pspec);
sl@0
  1419
        }
sl@0
  1420
      else
sl@0
  1421
        {
sl@0
  1422
          g_assert_not_reached ();
sl@0
  1423
          ret = NULL;
sl@0
  1424
        }
sl@0
  1425
sl@0
  1426
      g_assert (ret != NULL);
sl@0
  1427
sl@0
  1428
      if (dbus_message_iter_get_arg_type (&iter) != DBUS_TYPE_INVALID)
sl@0
  1429
        g_warning ("Property get or set had too many arguments\n");
sl@0
  1430
sl@0
  1431
      dbus_connection_send (connection, ret, NULL);
sl@0
  1432
      dbus_message_unref (ret);
sl@0
  1433
      return DBUS_HANDLER_RESULT_HANDLED;
sl@0
  1434
    }
sl@0
  1435
sl@0
  1436
  return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
sl@0
  1437
}
sl@0
  1438
sl@0
  1439
static const DBusObjectPathVTable gobject_dbus_vtable = {
sl@0
  1440
  gobject_unregister_function,
sl@0
  1441
  gobject_message_function,
sl@0
  1442
  NULL
sl@0
  1443
};
sl@0
  1444
sl@0
  1445
typedef struct {
sl@0
  1446
  GClosure         closure;
sl@0
  1447
  DBusGConnection *connection;
sl@0
  1448
  GObject         *object;
sl@0
  1449
  const char      *signame;
sl@0
  1450
  const char      *sigiface;
sl@0
  1451
} DBusGSignalClosure;
sl@0
  1452
sl@0
  1453
static GClosure *
sl@0
  1454
dbus_g_signal_closure_new (DBusGConnection *connection,
sl@0
  1455
			   GObject         *object,
sl@0
  1456
			   const char      *signame,
sl@0
  1457
			   const char      *sigiface)
sl@0
  1458
{
sl@0
  1459
  DBusGSignalClosure *closure;
sl@0
  1460
  
sl@0
  1461
  closure = (DBusGSignalClosure*) g_closure_new_simple (sizeof (DBusGSignalClosure), NULL);
sl@0
  1462
sl@0
  1463
  closure->connection = dbus_g_connection_ref (connection);
sl@0
  1464
  closure->object = object;
sl@0
  1465
  closure->signame = signame;
sl@0
  1466
  closure->sigiface = sigiface;
sl@0
  1467
  return (GClosure*) closure;
sl@0
  1468
}
sl@0
  1469
sl@0
  1470
static void
sl@0
  1471
dbus_g_signal_closure_finalize (gpointer data,
sl@0
  1472
				GClosure *closure)
sl@0
  1473
{
sl@0
  1474
  DBusGSignalClosure *sigclosure = (DBusGSignalClosure *) closure;
sl@0
  1475
sl@0
  1476
  dbus_g_connection_unref (sigclosure->connection);
sl@0
  1477
}
sl@0
  1478
sl@0
  1479
static void
sl@0
  1480
signal_emitter_marshaller (GClosure        *closure,
sl@0
  1481
			   GValue          *retval,
sl@0
  1482
			   guint            n_param_values,
sl@0
  1483
			   const GValue    *param_values,
sl@0
  1484
			   gpointer         invocation_hint,
sl@0
  1485
			   gpointer         marshal_data)
sl@0
  1486
{
sl@0
  1487
  DBusGSignalClosure *sigclosure;
sl@0
  1488
  DBusMessage *signal;
sl@0
  1489
  DBusMessageIter iter;
sl@0
  1490
  guint i;
sl@0
  1491
  const char *path;
sl@0
  1492
sl@0
  1493
  sigclosure = (DBusGSignalClosure *) closure;
sl@0
  1494
  
sl@0
  1495
  g_assert (retval == NULL);
sl@0
  1496
sl@0
  1497
  path = _dbus_gobject_get_path (sigclosure->object);
sl@0
  1498
sl@0
  1499
  g_assert (path != NULL);
sl@0
  1500
sl@0
  1501
  signal = dbus_message_new_signal (path,
sl@0
  1502
				    sigclosure->sigiface,
sl@0
  1503
				    sigclosure->signame);
sl@0
  1504
  if (!signal)
sl@0
  1505
    {
sl@0
  1506
      g_error ("out of memory");
sl@0
  1507
      return;
sl@0
  1508
    }
sl@0
  1509
sl@0
  1510
  dbus_message_iter_init_append (signal, &iter);
sl@0
  1511
sl@0
  1512
  /* First argument is the object itself, and we can't marshall that */
sl@0
  1513
  for (i = 1; i < n_param_values; i++)
sl@0
  1514
    {
sl@0
  1515
      if (!_dbus_gvalue_marshal (&iter,
sl@0
  1516
				(GValue *) (&(param_values[i]))))
sl@0
  1517
	{
sl@0
  1518
	  g_warning ("failed to marshal parameter %d for signal %s",
sl@0
  1519
		     i, sigclosure->signame);
sl@0
  1520
	  goto out;
sl@0
  1521
	}
sl@0
  1522
    }
sl@0
  1523
  dbus_connection_send (DBUS_CONNECTION_FROM_G_CONNECTION (sigclosure->connection),
sl@0
  1524
			signal, NULL);
sl@0
  1525
 out:
sl@0
  1526
  dbus_message_unref (signal);
sl@0
  1527
}
sl@0
  1528
sl@0
  1529
static void
sl@0
  1530
export_signals (DBusGConnection *connection, const GList *info_list, GObject *object)
sl@0
  1531
{
sl@0
  1532
  GType gtype;
sl@0
  1533
  const char *sigdata;
sl@0
  1534
  const char *iface;
sl@0
  1535
  const char *signame;
sl@0
  1536
  const DBusGObjectInfo *info;
sl@0
  1537
sl@0
  1538
  gtype = G_TYPE_FROM_INSTANCE (object);
sl@0
  1539
sl@0
  1540
  for (; info_list != NULL; info_list = g_list_next (info_list))
sl@0
  1541
    {
sl@0
  1542
      info = (DBusGObjectInfo *) info_list->data;
sl@0
  1543
      
sl@0
  1544
      sigdata = info->exported_signals;
sl@0
  1545
      
sl@0
  1546
      while (*sigdata != '\0')
sl@0
  1547
        {
sl@0
  1548
          guint id;
sl@0
  1549
          GSignalQuery query;
sl@0
  1550
          GClosure *closure;
sl@0
  1551
          char *s;
sl@0
  1552
sl@0
  1553
          sigdata = propsig_iterate (sigdata, &iface, &signame);
sl@0
  1554
          
sl@0
  1555
          s = _dbus_gutils_wincaps_to_uscore (signame);
sl@0
  1556
sl@0
  1557
          id = g_signal_lookup (s, gtype);
sl@0
  1558
          if (id == 0)
sl@0
  1559
            {
sl@0
  1560
              g_warning ("signal \"%s\" (from \"%s\") exported but not found in object class \"%s\"",
sl@0
  1561
                     s, signame, g_type_name (gtype));
sl@0
  1562
              g_free (s);
sl@0
  1563
              continue;
sl@0
  1564
            }
sl@0
  1565
sl@0
  1566
          g_signal_query (id, &query);
sl@0
  1567
sl@0
  1568
          if (query.return_type != G_TYPE_NONE)
sl@0
  1569
            {
sl@0
  1570
              g_warning ("Not exporting signal \"%s\" for object class \"%s\" as it has a return type \"%s\"",
sl@0
  1571
                     s, g_type_name (gtype), g_type_name (query.return_type));
sl@0
  1572
              g_free (s);
sl@0
  1573
              continue; /* FIXME: these could be listed as methods ? */
sl@0
  1574
            }
sl@0
  1575
          
sl@0
  1576
          closure = dbus_g_signal_closure_new (connection, object, signame, (char*) iface);
sl@0
  1577
          g_closure_set_marshal (closure, signal_emitter_marshaller);
sl@0
  1578
sl@0
  1579
          g_signal_connect_closure_by_id (object,
sl@0
  1580
                          id,
sl@0
  1581
                          0,
sl@0
  1582
                          closure,
sl@0
  1583
                          FALSE);
sl@0
  1584
sl@0
  1585
          g_closure_add_finalize_notifier (closure, NULL,
sl@0
  1586
                           dbus_g_signal_closure_finalize);
sl@0
  1587
          g_free (s);
sl@0
  1588
        }
sl@0
  1589
    }
sl@0
  1590
}
sl@0
  1591
sl@0
  1592
#include "dbus-glib-error-switch.h"
sl@0
  1593
sl@0
  1594
/**
sl@0
  1595
 * dbus_set_g_error:
sl@0
  1596
 * @gerror: an error
sl@0
  1597
 * @error: a #DBusError
sl@0
  1598
 *
sl@0
  1599
 * FIXME
sl@0
  1600
 */
sl@0
  1601
 	#ifdef __SYMBIAN32__
sl@0
  1602
	EXPORT_C
sl@0
  1603
	#endif
sl@0
  1604
void
sl@0
  1605
dbus_set_g_error (GError    **gerror,
sl@0
  1606
		  DBusError  *error)
sl@0
  1607
{
sl@0
  1608
  int code;
sl@0
  1609
sl@0
  1610
  code = dbus_error_to_gerror_code (error->name);
sl@0
  1611
  if (code != DBUS_GERROR_REMOTE_EXCEPTION)
sl@0
  1612
    g_set_error (gerror, DBUS_GERROR,
sl@0
  1613
		 code,
sl@0
  1614
		 "%s",
sl@0
  1615
		 error->message);
sl@0
  1616
  else
sl@0
  1617
    g_set_error (gerror, DBUS_GERROR,
sl@0
  1618
		 code,
sl@0
  1619
		 "%s%c%s",
sl@0
  1620
		 error->message ? error->message : "",
sl@0
  1621
		 '\0',
sl@0
  1622
		 error->name);
sl@0
  1623
}
sl@0
  1624
sl@0
  1625
static void
sl@0
  1626
dbus_g_error_info_free (gpointer p)
sl@0
  1627
{
sl@0
  1628
  DBusGErrorInfo *info;
sl@0
  1629
sl@0
  1630
  info = p;
sl@0
  1631
sl@0
  1632
  g_free (info->default_iface);
sl@0
  1633
  g_free (info);
sl@0
  1634
}
sl@0
  1635
sl@0
  1636
/**
sl@0
  1637
 * SECTION:dbus-gobject
sl@0
  1638
 * @short_description: Exporting a #GObject remotely
sl@0
  1639
 * @see_also: #GObject
sl@0
  1640
 * @stability: Stable
sl@0
  1641
 *
sl@0
  1642
 * FIXME
sl@0
  1643
 */
sl@0
  1644
sl@0
  1645
/**
sl@0
  1646
 * dbus_g_object_type_install_info:
sl@0
  1647
 * @object_type: #GType for the object
sl@0
  1648
 * @info: introspection data generated by #dbus-glib-tool
sl@0
  1649
 *
sl@0
  1650
 * Install introspection information about the given object #GType
sl@0
  1651
 * sufficient to allow methods on the object to be invoked by name.
sl@0
  1652
 * The introspection information is normally generated by
sl@0
  1653
 * dbus-glib-tool, then this function is called in the
sl@0
  1654
 * class_init() for the object class.
sl@0
  1655
 *
sl@0
  1656
 * Once introspection information has been installed, instances of the
sl@0
  1657
 * object registered with #dbus_g_connection_register_g_object() can have
sl@0
  1658
 * their methods invoked remotely.
sl@0
  1659
 */
sl@0
  1660
 	#ifdef __SYMBIAN32__
sl@0
  1661
	EXPORT_C
sl@0
  1662
	#endif
sl@0
  1663
void
sl@0
  1664
dbus_g_object_type_install_info (GType                  object_type,
sl@0
  1665
				 const DBusGObjectInfo *info)
sl@0
  1666
{
sl@0
  1667
  g_return_if_fail (G_TYPE_IS_CLASSED (object_type) || G_TYPE_IS_INTERFACE (object_type));
sl@0
  1668
sl@0
  1669
  _dbus_g_value_types_init ();
sl@0
  1670
sl@0
  1671
  g_type_set_qdata (object_type,
sl@0
  1672
		    dbus_g_object_type_dbus_metadata_quark (),
sl@0
  1673
		    (gpointer) info);
sl@0
  1674
}
sl@0
  1675
sl@0
  1676
/**
sl@0
  1677
 * dbus_g_error_domain_register:
sl@0
  1678
 * @domain: the #GError domain 
sl@0
  1679
 * @default_iface: the D-BUS interface used for error values by default, or #NULL
sl@0
  1680
 * @code_enum: a #GType for a #GEnum of the error codes
sl@0
  1681
 *
sl@0
  1682
 * Register a #GError domain and set of codes with D-BUS.  You must
sl@0
  1683
 * have created a GEnum for the error codes.  This function will not
sl@0
  1684
 * be needed with an introspection-capable GLib.
sl@0
  1685
 */
sl@0
  1686
 	#ifdef __SYMBIAN32__
sl@0
  1687
	EXPORT_C
sl@0
  1688
	#endif
sl@0
  1689
void
sl@0
  1690
dbus_g_error_domain_register (GQuark                domain,
sl@0
  1691
			      const char           *default_iface,
sl@0
  1692
			      GType                 code_enum)
sl@0
  1693
{
sl@0
  1694
  DBusGErrorInfo *info;
sl@0
  1695
  
sl@0
  1696
  g_return_if_fail (g_quark_to_string (domain) != NULL);
sl@0
  1697
  g_return_if_fail (code_enum != G_TYPE_INVALID);
sl@0
  1698
  g_return_if_fail (G_TYPE_FUNDAMENTAL (code_enum) == G_TYPE_ENUM);
sl@0
  1699
sl@0
  1700
  g_static_rw_lock_writer_lock (&globals_lock);
sl@0
  1701
sl@0
  1702
  if (error_metadata == NULL)
sl@0
  1703
    g_datalist_init (&error_metadata);
sl@0
  1704
sl@0
  1705
  info = g_datalist_id_get_data (&error_metadata, domain);
sl@0
  1706
sl@0
  1707
  if (info != NULL)
sl@0
  1708
    {
sl@0
  1709
      g_warning ("Metadata for error domain \"%s\" already registered\n",
sl@0
  1710
		 g_quark_to_string (domain));
sl@0
  1711
    }
sl@0
  1712
  else
sl@0
  1713
    {
sl@0
  1714
      info = g_new0 (DBusGErrorInfo, 1);
sl@0
  1715
      info->default_iface = g_strdup (default_iface);
sl@0
  1716
      info->code_enum = code_enum;
sl@0
  1717
sl@0
  1718
      g_datalist_id_set_data_full (&error_metadata,
sl@0
  1719
				   domain,
sl@0
  1720
				   info,
sl@0
  1721
				   dbus_g_error_info_free);
sl@0
  1722
    }
sl@0
  1723
sl@0
  1724
  g_static_rw_lock_writer_unlock (&globals_lock);
sl@0
  1725
}
sl@0
  1726
sl@0
  1727
static void
sl@0
  1728
unregister_gobject (DBusGConnection *connection, GObject *dead)
sl@0
  1729
{
sl@0
  1730
  char *path;
sl@0
  1731
  path = g_object_steal_data (dead, "dbus_glib_object_path");
sl@0
  1732
  dbus_connection_unregister_object_path (DBUS_CONNECTION_FROM_G_CONNECTION (connection), path);
sl@0
  1733
  g_free (path);
sl@0
  1734
}
sl@0
  1735
sl@0
  1736
/**
sl@0
  1737
 * dbus_g_connection_register_g_object:
sl@0
  1738
 * @connection: the D-BUS connection
sl@0
  1739
 * @at_path: the path where the object will live (the object's name)
sl@0
  1740
 * @object: the object
sl@0
  1741
 *
sl@0
  1742
 * Registers a #GObject at the given path. Properties, methods, and signals
sl@0
  1743
 * of the object can then be accessed remotely. Methods are only available
sl@0
  1744
 * if method introspection data has been added to the object's class
sl@0
  1745
 * with g_object_class_install_info().
sl@0
  1746
 *
sl@0
  1747
 * The registration will be cancelled if either the #DBusConnection or
sl@0
  1748
 * the #GObject gets finalized.
sl@0
  1749
 */
sl@0
  1750
 	#ifdef __SYMBIAN32__
sl@0
  1751
	EXPORT_C
sl@0
  1752
	#endif
sl@0
  1753
void
sl@0
  1754
dbus_g_connection_register_g_object (DBusGConnection       *connection,
sl@0
  1755
                                     const char            *at_path,
sl@0
  1756
                                     GObject               *object)
sl@0
  1757
{
sl@0
  1758
  GList *info_list;
sl@0
  1759
  g_return_if_fail (connection != NULL);
sl@0
  1760
  g_return_if_fail (at_path != NULL);
sl@0
  1761
  g_return_if_fail (G_IS_OBJECT (object));
sl@0
  1762
sl@0
  1763
  info_list = lookup_object_info (object);
sl@0
  1764
  if (info_list == NULL)
sl@0
  1765
    {
sl@0
  1766
      g_warning ("No introspection data registered for object class \"%s\"",
sl@0
  1767
		 g_type_name (G_TYPE_FROM_INSTANCE (object)));
sl@0
  1768
      return;
sl@0
  1769
    }
sl@0
  1770
sl@0
  1771
  if (!dbus_connection_register_object_path (DBUS_CONNECTION_FROM_G_CONNECTION (connection),
sl@0
  1772
                                             at_path,
sl@0
  1773
                                             &gobject_dbus_vtable,
sl@0
  1774
                                             object))
sl@0
  1775
    {
sl@0
  1776
      g_error ("Failed to register GObject with DBusConnection");
sl@0
  1777
      return;
sl@0
  1778
    }
sl@0
  1779
sl@0
  1780
  export_signals (connection, info_list, object);
sl@0
  1781
  g_list_free (info_list);
sl@0
  1782
sl@0
  1783
  g_object_set_data (object, "dbus_glib_object_path", g_strdup (at_path));
sl@0
  1784
  g_object_weak_ref (object, (GWeakNotify)unregister_gobject, connection);
sl@0
  1785
}
sl@0
  1786
sl@0
  1787
/**
sl@0
  1788
 * dbus_g_connection_lookup_g_object:
sl@0
  1789
 * @connection: a #DBusGConnection
sl@0
  1790
 * @at_path: path
sl@0
  1791
 *
sl@0
  1792
 * FIXME 
sl@0
  1793
 *
sl@0
  1794
 * Returns: the object at path @at_path
sl@0
  1795
 */
sl@0
  1796
 #ifdef __SYMBIAN32__
sl@0
  1797
EXPORT_C
sl@0
  1798
#endif
sl@0
  1799
GObject *
sl@0
  1800
dbus_g_connection_lookup_g_object (DBusGConnection       *connection,
sl@0
  1801
				   const char            *at_path)
sl@0
  1802
{
sl@0
  1803
  gpointer ret;
sl@0
  1804
  if (!dbus_connection_get_object_path_data (DBUS_CONNECTION_FROM_G_CONNECTION (connection), at_path, &ret))
sl@0
  1805
    return NULL;
sl@0
  1806
  return ret;
sl@0
  1807
}
sl@0
  1808
sl@0
  1809
typedef struct {
sl@0
  1810
  GType    rettype;
sl@0
  1811
  guint    n_params;
sl@0
  1812
  GType   *params;
sl@0
  1813
} DBusGFuncSignature;
sl@0
  1814
sl@0
  1815
static guint
sl@0
  1816
funcsig_hash (gconstpointer key)
sl@0
  1817
{
sl@0
  1818
  const DBusGFuncSignature *sig = key;
sl@0
  1819
  GType *types;
sl@0
  1820
  guint ret;
sl@0
  1821
  guint i;
sl@0
  1822
sl@0
  1823
  ret = sig->rettype;
sl@0
  1824
  types = sig->params;
sl@0
  1825
sl@0
  1826
  for (i = 0; i < sig->n_params; i++)
sl@0
  1827
    {
sl@0
  1828
      ret += (int) (*types);
sl@0
  1829
      types++;
sl@0
  1830
    }
sl@0
  1831
      
sl@0
  1832
  return ret;
sl@0
  1833
}
sl@0
  1834
sl@0
  1835
static gboolean
sl@0
  1836
funcsig_equal (gconstpointer aval,
sl@0
  1837
	       gconstpointer bval)
sl@0
  1838
{
sl@0
  1839
  const DBusGFuncSignature *a = aval;
sl@0
  1840
  const DBusGFuncSignature *b = bval;
sl@0
  1841
  const GType *atypes;
sl@0
  1842
  const GType *btypes;
sl@0
  1843
  guint i;
sl@0
  1844
sl@0
  1845
  if (a->rettype != b->rettype
sl@0
  1846
      || a->n_params != b->n_params)
sl@0
  1847
    return FALSE;
sl@0
  1848
sl@0
  1849
  atypes = a->params;
sl@0
  1850
  btypes = b->params;
sl@0
  1851
sl@0
  1852
  for (i = 0; i < a->n_params; i++)
sl@0
  1853
    {
sl@0
  1854
      if (*btypes != *atypes)
sl@0
  1855
	return FALSE;
sl@0
  1856
      atypes++;
sl@0
  1857
      btypes++;
sl@0
  1858
    }
sl@0
  1859
      
sl@0
  1860
  return TRUE;
sl@0
  1861
}
sl@0
  1862
sl@0
  1863
static void
sl@0
  1864
funcsig_free (DBusGFuncSignature *sig)
sl@0
  1865
{
sl@0
  1866
  g_free (sig->params);
sl@0
  1867
  g_free (sig);
sl@0
  1868
}
sl@0
  1869
sl@0
  1870
GClosureMarshal
sl@0
  1871
_dbus_gobject_lookup_marshaller (GType        rettype,
sl@0
  1872
				 guint        n_params,
sl@0
  1873
				 const GType *param_types)
sl@0
  1874
{
sl@0
  1875
  GClosureMarshal ret;
sl@0
  1876
  DBusGFuncSignature sig;
sl@0
  1877
  GType *params;
sl@0
  1878
  guint i;
sl@0
  1879
sl@0
  1880
  /* Convert to fundamental types */
sl@0
  1881
  rettype = G_TYPE_FUNDAMENTAL (rettype);
sl@0
  1882
  params = g_new (GType, n_params);
sl@0
  1883
  for (i = 0; i < n_params; i++)
sl@0
  1884
    params[i] = G_TYPE_FUNDAMENTAL (param_types[i]);
sl@0
  1885
sl@0
  1886
  sig.rettype = rettype;
sl@0
  1887
  sig.n_params = n_params;
sl@0
  1888
  sig.params = params;
sl@0
  1889
  
sl@0
  1890
  g_static_rw_lock_reader_lock (&globals_lock);
sl@0
  1891
sl@0
  1892
  if (marshal_table)
sl@0
  1893
  #ifdef WINSCW
sl@0
  1894
    ret = g_hash_table_lookup (marshal_table, &sig);
sl@0
  1895
  #else
sl@0
  1896
  ret = (GClosureMarshal)g_hash_table_lookup (marshal_table, &sig);
sl@0
  1897
  #endif  
sl@0
  1898
  else
sl@0
  1899
    ret = NULL;
sl@0
  1900
sl@0
  1901
  g_static_rw_lock_reader_unlock (&globals_lock);
sl@0
  1902
sl@0
  1903
  if (ret == NULL)
sl@0
  1904
    {
sl@0
  1905
      if (rettype == G_TYPE_NONE)
sl@0
  1906
	{
sl@0
  1907
	  if (n_params == 0)
sl@0
  1908
	    ret = g_cclosure_marshal_VOID__VOID;
sl@0
  1909
	  else if (n_params == 1)
sl@0
  1910
	    {
sl@0
  1911
	      switch (params[0])
sl@0
  1912
		{
sl@0
  1913
		case G_TYPE_BOOLEAN:
sl@0
  1914
		  ret = g_cclosure_marshal_VOID__BOOLEAN;
sl@0
  1915
		  break;
sl@0
  1916
		case G_TYPE_UCHAR:
sl@0
  1917
		  ret = g_cclosure_marshal_VOID__UCHAR;
sl@0
  1918
		  break;
sl@0
  1919
		case G_TYPE_INT:
sl@0
  1920
		  ret = g_cclosure_marshal_VOID__INT;
sl@0
  1921
		  break;
sl@0
  1922
		case G_TYPE_UINT:
sl@0
  1923
		  ret = g_cclosure_marshal_VOID__UINT;
sl@0
  1924
		  break;
sl@0
  1925
		case G_TYPE_DOUBLE:
sl@0
  1926
		  ret = g_cclosure_marshal_VOID__DOUBLE;
sl@0
  1927
		  break;
sl@0
  1928
		case G_TYPE_STRING:
sl@0
  1929
		  ret = g_cclosure_marshal_VOID__STRING;
sl@0
  1930
		  break;
sl@0
  1931
		case G_TYPE_BOXED:
sl@0
  1932
		  ret = g_cclosure_marshal_VOID__BOXED;
sl@0
  1933
		  break;
sl@0
  1934
		}
sl@0
  1935
	    }
sl@0
  1936
	  else if (n_params == 3
sl@0
  1937
		   && params[0] == G_TYPE_STRING
sl@0
  1938
		   && params[1] == G_TYPE_STRING
sl@0
  1939
		   && params[2] == G_TYPE_STRING)
sl@0
  1940
	    {
sl@0
  1941
	      ret = _dbus_g_marshal_NONE__STRING_STRING_STRING;
sl@0
  1942
	    }
sl@0
  1943
	}
sl@0
  1944
    }
sl@0
  1945
sl@0
  1946
  g_free (params);
sl@0
  1947
  return ret;
sl@0
  1948
}
sl@0
  1949
sl@0
  1950
/**
sl@0
  1951
 * dbus_g_object_register_marshaller:
sl@0
  1952
 * @marshaller: a GClosureMarshal to be used for invocation
sl@0
  1953
 * @rettype: a GType for the return type of the function
sl@0
  1954
 * @:... The parameter #GTypes, followed by %G_TYPE_INVALID
sl@0
  1955
 *
sl@0
  1956
 * Register a #GClosureMarshal to be used for signal invocations,
sl@0
  1957
 * giving its return type and a list of parameter types,
sl@0
  1958
 * followed by %G_TYPE_INVALID.
sl@0
  1959
 *
sl@0
  1960
 * This function will not be needed once GLib includes libffi.
sl@0
  1961
 */
sl@0
  1962
 	#ifdef __SYMBIAN32__
sl@0
  1963
	EXPORT_C
sl@0
  1964
	#endif
sl@0
  1965
void
sl@0
  1966
dbus_g_object_register_marshaller (GClosureMarshal  marshaller,
sl@0
  1967
				   GType            rettype,
sl@0
  1968
				   ...)
sl@0
  1969
{
sl@0
  1970
  va_list args;
sl@0
  1971
  GArray *types;
sl@0
  1972
  GType gtype;
sl@0
  1973
sl@0
  1974
  va_start (args, rettype);
sl@0
  1975
sl@0
  1976
  types = g_array_new (TRUE, TRUE, sizeof (GType));
sl@0
  1977
sl@0
  1978
  while ((gtype = va_arg (args, GType)) != G_TYPE_INVALID)
sl@0
  1979
    g_array_append_val (types, gtype);
sl@0
  1980
sl@0
  1981
  dbus_g_object_register_marshaller_array (marshaller, rettype,
sl@0
  1982
					   types->len, (GType*) types->data);
sl@0
  1983
sl@0
  1984
  g_array_free (types, TRUE);
sl@0
  1985
  va_end (args);
sl@0
  1986
}
sl@0
  1987
sl@0
  1988
/**
sl@0
  1989
 * dbus_g_object_register_marshaller_array:
sl@0
  1990
 * @marshaller: a #GClosureMarshal to be used for invocation
sl@0
  1991
 * @rettype: a #GType for the return type of the function
sl@0
  1992
 * @n_types: number of function parameters
sl@0
  1993
 * @types: a C array of GTypes values
sl@0
  1994
 *
sl@0
  1995
 * Register a #GClosureMarshal to be used for signal invocations.
sl@0
  1996
 * @see_also #dbus_g_object_register_marshaller
sl@0
  1997
 */
sl@0
  1998
 	#ifdef __SYMBIAN32__
sl@0
  1999
	EXPORT_C
sl@0
  2000
	#endif
sl@0
  2001
void
sl@0
  2002
dbus_g_object_register_marshaller_array (GClosureMarshal  marshaller,
sl@0
  2003
					 GType            rettype,
sl@0
  2004
					 guint            n_types,
sl@0
  2005
					 const GType*     types)
sl@0
  2006
{
sl@0
  2007
  DBusGFuncSignature *sig;
sl@0
  2008
  guint i;
sl@0
  2009
sl@0
  2010
  g_static_rw_lock_writer_lock (&globals_lock);
sl@0
  2011
sl@0
  2012
  if (marshal_table == NULL)
sl@0
  2013
    marshal_table = g_hash_table_new_full (funcsig_hash,
sl@0
  2014
					   funcsig_equal,
sl@0
  2015
					   (GDestroyNotify) funcsig_free,
sl@0
  2016
					   NULL);
sl@0
  2017
  sig = g_new0 (DBusGFuncSignature, 1);
sl@0
  2018
  sig->rettype = G_TYPE_FUNDAMENTAL (rettype);
sl@0
  2019
  sig->n_params = n_types;
sl@0
  2020
  sig->params = g_new (GType, n_types);
sl@0
  2021
  for (i = 0; i < n_types; i++)
sl@0
  2022
    sig->params[i] = G_TYPE_FUNDAMENTAL (types[i]);
sl@0
  2023
#ifdef WINSCW
sl@0
  2024
  g_hash_table_insert (marshal_table, sig, marshaller);
sl@0
  2025
  #else
sl@0
  2026
  g_hash_table_insert (marshal_table, sig, (gpointer)marshaller);
sl@0
  2027
  #endif
sl@0
  2028
sl@0
  2029
  g_static_rw_lock_writer_unlock (&globals_lock);
sl@0
  2030
}
sl@0
  2031
sl@0
  2032
/**
sl@0
  2033
 * dbus_g_method_get_sender:
sl@0
  2034
 * @context: the method context
sl@0
  2035
 *
sl@0
  2036
 * Get the sender of a message so we can send a
sl@0
  2037
 * "reply" later (i.e. send a message directly
sl@0
  2038
 * to a service which invoked the method at a
sl@0
  2039
 * later time).
sl@0
  2040
 *
sl@0
  2041
 * Returns: the unique name of the sender. It
sl@0
  2042
 * is up to the caller to free the returned string.
sl@0
  2043
 */
sl@0
  2044
 	#ifdef __SYMBIAN32__
sl@0
  2045
	EXPORT_C
sl@0
  2046
	#endif
sl@0
  2047
gchar *
sl@0
  2048
dbus_g_method_get_sender (DBusGMethodInvocation *context)
sl@0
  2049
{
sl@0
  2050
  const gchar *sender;
sl@0
  2051
sl@0
  2052
  sender = dbus_message_get_sender (dbus_g_message_get_message (context->message));
sl@0
  2053
sl@0
  2054
  if (sender == NULL)
sl@0
  2055
    return NULL;
sl@0
  2056
    
sl@0
  2057
  return strdup (sender);
sl@0
  2058
}
sl@0
  2059
sl@0
  2060
/**
sl@0
  2061
 * dbus_g_method_get_reply:
sl@0
  2062
 * @context: the method context
sl@0
  2063
 *
sl@0
  2064
 * Get the reply message to append reply values
sl@0
  2065
 * Used as a sidedoor when you can't generate dbus values
sl@0
  2066
 * of the correct type due to glib binding limitations
sl@0
  2067
 *
sl@0
  2068
 * Returns: a #DBusMessage with the reply
sl@0
  2069
 */
sl@0
  2070
 	#ifdef __SYMBIAN32__
sl@0
  2071
	EXPORT_C
sl@0
  2072
	#endif
sl@0
  2073
DBusMessage *
sl@0
  2074
dbus_g_method_get_reply (DBusGMethodInvocation *context)
sl@0
  2075
{
sl@0
  2076
  return dbus_message_new_method_return (dbus_g_message_get_message (context->message));
sl@0
  2077
}
sl@0
  2078
sl@0
  2079
/**
sl@0
  2080
 * dbus_g_method_send_reply:
sl@0
  2081
 * Send a manually created reply message
sl@0
  2082
 * @context: the method context
sl@0
  2083
 * @reply: the reply message, will be unreffed
sl@0
  2084
 *
sl@0
  2085
 * Used as a sidedoor when you can't generate dbus values
sl@0
  2086
 * of the correct type due to glib binding limitations
sl@0
  2087
 */
sl@0
  2088
 	#ifdef __SYMBIAN32__
sl@0
  2089
	EXPORT_C
sl@0
  2090
	#endif
sl@0
  2091
void
sl@0
  2092
dbus_g_method_send_reply (DBusGMethodInvocation *context, DBusMessage *reply)
sl@0
  2093
{
sl@0
  2094
  dbus_connection_send (dbus_g_connection_get_connection (context->connection), reply, NULL);
sl@0
  2095
  dbus_message_unref (reply);
sl@0
  2096
sl@0
  2097
  dbus_g_connection_unref (context->connection);
sl@0
  2098
  dbus_g_message_unref (context->message);
sl@0
  2099
  g_free (context);
sl@0
  2100
}
sl@0
  2101
sl@0
  2102
sl@0
  2103
/**
sl@0
  2104
 * dbus_g_method_return:
sl@0
  2105
 * @context: the method context
sl@0
  2106
 *
sl@0
  2107
 * Send a return message for a given method invocation, with arguments.
sl@0
  2108
 * This function also frees the sending context.
sl@0
  2109
 */
sl@0
  2110
 	#ifdef __SYMBIAN32__
sl@0
  2111
	EXPORT_C
sl@0
  2112
	#endif
sl@0
  2113
void
sl@0
  2114
dbus_g_method_return (DBusGMethodInvocation *context, ...)
sl@0
  2115
{
sl@0
  2116
  DBusMessage *reply;
sl@0
  2117
  DBusMessageIter iter;
sl@0
  2118
  va_list args;
sl@0
  2119
  char *out_sig;
sl@0
  2120
  GArray *argsig;
sl@0
  2121
  guint i;
sl@0
  2122
sl@0
  2123
  reply = dbus_message_new_method_return (dbus_g_message_get_message (context->message));
sl@0
  2124
  out_sig = method_output_signature_from_object_info (context->object, context->method);
sl@0
  2125
  argsig = _dbus_gtypes_from_arg_signature (out_sig, FALSE);
sl@0
  2126
sl@0
  2127
  dbus_message_iter_init_append (reply, &iter);
sl@0
  2128
sl@0
  2129
  va_start (args, context);
sl@0
  2130
  for (i = 0; i < argsig->len; i++)
sl@0
  2131
    {
sl@0
  2132
      GValue value = {0,};
sl@0
  2133
      char *error;
sl@0
  2134
      g_value_init (&value, g_array_index (argsig, GType, i));
sl@0
  2135
      error = NULL;
sl@0
  2136
      G_VALUE_COLLECT (&value, args, G_VALUE_NOCOPY_CONTENTS, &error);
sl@0
  2137
      if (error)
sl@0
  2138
	{
sl@0
  2139
	  g_warning(error);
sl@0
  2140
	  g_free (error);
sl@0
  2141
	}
sl@0
  2142
      _dbus_gvalue_marshal (&iter, &value);
sl@0
  2143
    }
sl@0
  2144
  va_end (args);
sl@0
  2145
sl@0
  2146
  dbus_connection_send (dbus_g_connection_get_connection (context->connection), reply, NULL);
sl@0
  2147
  dbus_message_unref (reply);
sl@0
  2148
sl@0
  2149
  dbus_g_connection_unref (context->connection);
sl@0
  2150
  dbus_g_message_unref (context->message);
sl@0
  2151
  g_free (context);
sl@0
  2152
  g_free (out_sig);
sl@0
  2153
  g_array_free (argsig, TRUE);
sl@0
  2154
}
sl@0
  2155
sl@0
  2156
/**
sl@0
  2157
 * dbus_g_method_return_error:
sl@0
  2158
 * @context: the method context
sl@0
  2159
 * @error: the error to send
sl@0
  2160
 *
sl@0
  2161
 * Send a error message for a given method invocation.
sl@0
  2162
 * This function also frees the sending context.
sl@0
  2163
 */
sl@0
  2164
 	#ifdef __SYMBIAN32__
sl@0
  2165
	EXPORT_C
sl@0
  2166
	#endif
sl@0
  2167
void
sl@0
  2168
dbus_g_method_return_error (DBusGMethodInvocation *context, GError *error)
sl@0
  2169
{
sl@0
  2170
  DBusMessage *reply;
sl@0
  2171
  reply = gerror_to_dbus_error_message (context->object, dbus_g_message_get_message (context->message), error);
sl@0
  2172
  dbus_connection_send (dbus_g_connection_get_connection (context->connection), reply, NULL);
sl@0
  2173
  dbus_message_unref (reply);
sl@0
  2174
  g_free (context);
sl@0
  2175
}
sl@0
  2176
sl@0
  2177
const char * _dbus_gobject_get_path (GObject *obj)
sl@0
  2178
{
sl@0
  2179
  return g_object_get_data (obj, "dbus_glib_object_path");
sl@0
  2180
}
sl@0
  2181
sl@0
  2182
#ifdef DBUS_BUILD_TESTS
sl@0
  2183
#include <stdlib.h>
sl@0
  2184
sl@0
  2185
static void
sl@0
  2186
_dummy_function (void)
sl@0
  2187
{
sl@0
  2188
}
sl@0
  2189
sl@0
  2190
/* Data structures copied from one generated by current dbus-binding-tool;
sl@0
  2191
 * we need to support this layout forever
sl@0
  2192
 */
sl@0
  2193
static const DBusGMethodInfo dbus_glib_internal_test_methods[] = {
sl@0
  2194
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 0 },
sl@0
  2195
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 49 },
sl@0
  2196
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 117 },
sl@0
  2197
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 191 },
sl@0
  2198
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 270 },
sl@0
  2199
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 320 },
sl@0
  2200
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 391 },
sl@0
  2201
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 495 },
sl@0
  2202
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 623 },
sl@0
  2203
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 693 },
sl@0
  2204
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 765 },
sl@0
  2205
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 838 },
sl@0
  2206
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 911 },
sl@0
  2207
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 988 },
sl@0
  2208
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 1064 },
sl@0
  2209
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 1140 },
sl@0
  2210
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 1204 },
sl@0
  2211
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 1278 },
sl@0
  2212
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 1347 },
sl@0
  2213
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 1408 },
sl@0
  2214
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 1460 },
sl@0
  2215
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 1533 },
sl@0
  2216
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 1588 },
sl@0
  2217
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 1647 },
sl@0
  2218
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 1730 },
sl@0
  2219
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 1784 },
sl@0
  2220
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 1833 },
sl@0
  2221
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 1895 },
sl@0
  2222
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 1947 },
sl@0
  2223
  { (GCallback) _dummy_function, g_cclosure_marshal_VOID__VOID, 1999 },
sl@0
  2224
};
sl@0
  2225
sl@0
  2226
const DBusGObjectInfo dbus_glib_internal_test_object_info = {
sl@0
  2227
  0,
sl@0
  2228
  dbus_glib_internal_test_methods,
sl@0
  2229
  30,
sl@0
  2230
"org.freedesktop.DBus.Tests.MyObject\0DoNothing\0S\0\0org.freedesktop.DBus.Tests.MyObject\0Increment\0S\0x\0I\0u\0arg1\0O\0F\0N\0u\0\0org.freedesktop.DBus.Tests.MyObject\0IncrementRetval\0S\0x\0I\0u\0arg1\0O\0F\0R\0u\0\0org.freedesktop.DBus.Tests.MyObject\0IncrementRetvalError\0S\0x\0I\0u\0arg1\0O\0F\0E\0u\0\0org.freedesktop.DBus.Tests.MyObject\0ThrowError\0S\0\0org.freedesktop.DBus.Tests.MyObject\0Uppercase\0S\0arg0\0I\0s\0arg1\0O\0F\0N\0s\0\0org.freedesktop.DBus.Tests.MyObject\0ManyArgs\0S\0x\0I\0u\0str\0I\0s\0trouble\0I\0d\0d_ret\0O\0F\0N\0d\0str_ret\0O\0F\0N\0s\0\0org.freedesktop.DBus.Tests.MyObject\0ManyReturn\0S\0arg0\0O\0F\0N\0u\0arg1\0O\0F\0N\0s\0arg2\0O\0F\0N\0i\0arg3\0O\0F\0N\0u\0arg4\0O\0F\0N\0u\0arg5\0O\0C\0N\0s\0\0org.freedesktop.DBus.Tests.MyObject\0Stringify\0S\0val\0I\0v\0arg1\0O\0F\0N\0s\0\0org.freedesktop.DBus.Tests.MyObject\0Unstringify\0S\0val\0I\0s\0arg1\0O\0F\0N\0v\0\0org.freedesktop.DBus.Tests.MyObject\0Recursive1\0S\0arg0\0I\0au\0arg1\0O\0F\0N\0u\0\0org.freedesktop.DBus.Tests.MyObject\0Recursive2\0S\0arg0\0I\0u\0arg1\0O\0F\0N\0au\0\0org.freedesktop.DBus.Tests.MyObject\0ManyUppercase\0S\0arg0\0I\0as\0arg1\0O\0F\0N\0as\0\0org.freedesktop.DBus.Tests.MyObject\0StrHashLen\0S\0arg0\0I\0a{ss}\0arg1\0O\0F\0N\0u\0\0org.freedesktop.DBus.Tests.MyObject\0SendCar\0S\0arg0\0I\0(suv)\0arg1\0O\0F\0N\0(uo)\0\0org.freedesktop.DBus.Tests.MyObject\0GetHash\0S\0arg0\0O\0F\0N\0a{ss}\0\0org.freedesktop.DBus.Tests.MyObject\0RecArrays\0S\0val\0I\0aas\0arg1\0O\0F\0N\0aau\0\0org.freedesktop.DBus.Tests.MyObject\0Objpath\0S\0arg0\0I\0o\0arg1\0O\0C\0N\0o\0\0org.freedesktop.DBus.Tests.MyObject\0GetObjs\0S\0arg0\0O\0F\0N\0ao\0\0org.freedesktop.DBus.Tests.MyObject\0IncrementVal\0S\0\0org.freedesktop.DBus.Tests.MyObject\0AsyncIncrement\0A\0x\0I\0u\0arg1\0O\0F\0N\0u\0\0org.freedesktop.DBus.Tests.MyObject\0AsyncThrowError\0A\0\0org.freedesktop.DBus.Tests.MyObject\0GetVal\0S\0arg0\0O\0F\0N\0u\0\0org.freedesktop.DBus.Tests.MyObject\0ManyStringify\0S\0arg0\0I\0a{sv}\0arg1\0O\0F\0N\0a{sv}\0\0org.freedesktop.DBus.Tests.MyObject\0EmitFrobnicate\0S\0\0org.freedesktop.DBus.Tests.MyObject\0Terminate\0S\0\0org.freedesktop.DBus.Tests.FooObject\0GetValue\0S\0arg0\0O\0F\0N\0u\0\0org.freedesktop.DBus.Tests.FooObject\0EmitSignals\0S\0\0org.freedesktop.DBus.Tests.FooObject\0EmitSignal2\0S\0\0org.freedesktop.DBus.Tests.FooObject\0Terminate\0S\0\0\0",
sl@0
  2231
"org.freedesktop.DBus.Tests.MyObject\0Frobnicate\0org.freedesktop.DBus.Tests.FooObject\0Sig0\0org.freedesktop.DBus.Tests.FooObject\0Sig1\0org.freedesktop.DBus.Tests.FooObject\0Sig2\0\0",
sl@0
  2232
"\0"
sl@0
  2233
};
sl@0
  2234
sl@0
  2235
sl@0
  2236
/**
sl@0
  2237
 * @ingroup DBusGLibInternals
sl@0
  2238
 * Unit test for GLib GObject integration ("skeletons")
sl@0
  2239
 * Returns: #TRUE on success.
sl@0
  2240
 */
sl@0
  2241
   	#ifdef __SYMBIAN32__
sl@0
  2242
	EXPORT_C
sl@0
  2243
	#endif
sl@0
  2244
gboolean
sl@0
  2245
_dbus_gobject_test (const char *test_data_dir)
sl@0
  2246
{
sl@0
  2247
  int i;
sl@0
  2248
  const char *arg;
sl@0
  2249
  const char *arg_name;
sl@0
  2250
  gboolean arg_in;
sl@0
  2251
  gboolean constval;
sl@0
  2252
  RetvalType retval;
sl@0
  2253
  const char *arg_signature;
sl@0
  2254
  const char *sigdata;
sl@0
  2255
  const char *iface;
sl@0
  2256
  const char *signame;
sl@0
  2257
  
sl@0
  2258
  static struct { const char *wincaps; const char *uscore; } name_pairs[] = {
sl@0
  2259
    { "SetFoo", "set_foo" },
sl@0
  2260
    { "Foo", "foo" },
sl@0
  2261
    { "GetFooBar", "get_foo_bar" },
sl@0
  2262
    { "Hello", "hello" }
sl@0
  2263
    
sl@0
  2264
    /* Impossible-to-handle cases */
sl@0
  2265
    /* { "FrobateUIHandler", "frobate_ui_handler" } */
sl@0
  2266
  };
sl@0
  2267
sl@0
  2268
  /* Test lookup in our hardcoded object info; if these tests fail
sl@0
  2269
   * then it likely means you changed the generated object info in an
sl@0
  2270
   * incompatible way and broke the lookup functions.  In that case
sl@0
  2271
   * you need to bump the version and use a new structure instead. */
sl@0
  2272
  /* DoNothing */
sl@0
  2273
  arg = method_arg_info_from_object_info (&dbus_glib_internal_test_object_info,
sl@0
  2274
					  &(dbus_glib_internal_test_methods[0]));
sl@0
  2275
  g_assert (*arg == '\0');
sl@0
  2276
sl@0
  2277
  /* Increment */
sl@0
  2278
  arg = method_arg_info_from_object_info (&dbus_glib_internal_test_object_info,
sl@0
  2279
					  &(dbus_glib_internal_test_methods[1]));
sl@0
  2280
  g_assert (*arg != '\0');
sl@0
  2281
  arg = arg_iterate (arg, &arg_name, &arg_in, &constval, &retval, &arg_signature);
sl@0
  2282
  g_assert (!strcmp (arg_name, "x"));
sl@0
  2283
  g_assert (arg_in == TRUE);
sl@0
  2284
  g_assert (!strcmp (arg_signature, "u"));
sl@0
  2285
  g_assert (*arg != '\0');
sl@0
  2286
  arg = arg_iterate (arg, &arg_name, &arg_in, &constval, &retval, &arg_signature);
sl@0
  2287
  g_assert (arg_in == FALSE);
sl@0
  2288
  g_assert (retval == RETVAL_NONE);
sl@0
  2289
  g_assert (!strcmp (arg_signature, "u"));
sl@0
  2290
  g_assert (*arg == '\0');
sl@0
  2291
sl@0
  2292
  /* IncrementRetval */
sl@0
  2293
  arg = method_arg_info_from_object_info (&dbus_glib_internal_test_object_info,
sl@0
  2294
					  &(dbus_glib_internal_test_methods[2]));
sl@0
  2295
  g_assert (*arg != '\0');
sl@0
  2296
  arg = arg_iterate (arg, &arg_name, &arg_in, &constval, &retval, &arg_signature);
sl@0
  2297
  g_assert (!strcmp (arg_name, "x"));
sl@0
  2298
  g_assert (arg_in == TRUE);
sl@0
  2299
  g_assert (!strcmp (arg_signature, "u"));
sl@0
  2300
  g_assert (*arg != '\0');
sl@0
  2301
  arg = arg_iterate (arg, &arg_name, &arg_in, &constval, &retval, &arg_signature);
sl@0
  2302
  g_assert (retval == RETVAL_NOERROR);
sl@0
  2303
  g_assert (arg_in == FALSE);
sl@0
  2304
  g_assert (!strcmp (arg_signature, "u"));
sl@0
  2305
  g_assert (*arg == '\0');
sl@0
  2306
sl@0
  2307
  /* IncrementRetvalError */
sl@0
  2308
  arg = method_arg_info_from_object_info (&dbus_glib_internal_test_object_info,
sl@0
  2309
					  &(dbus_glib_internal_test_methods[3]));
sl@0
  2310
  g_assert (*arg != '\0');
sl@0
  2311
  arg = arg_iterate (arg, &arg_name, &arg_in, &constval, &retval, &arg_signature);
sl@0
  2312
  g_assert (!strcmp (arg_name, "x"));
sl@0
  2313
  g_assert (arg_in == TRUE);
sl@0
  2314
  g_assert (!strcmp (arg_signature, "u"));
sl@0
  2315
  g_assert (*arg != '\0');
sl@0
  2316
  arg = arg_iterate (arg, &arg_name, &arg_in, &constval, &retval, &arg_signature);
sl@0
  2317
  g_assert (retval == RETVAL_ERROR);
sl@0
  2318
  g_assert (arg_in == FALSE);
sl@0
  2319
  g_assert (!strcmp (arg_signature, "u"));
sl@0
  2320
  g_assert (*arg == '\0');
sl@0
  2321
  
sl@0
  2322
  /* Stringify */
sl@0
  2323
  arg = method_arg_info_from_object_info (&dbus_glib_internal_test_object_info,
sl@0
  2324
					  &(dbus_glib_internal_test_methods[8]));
sl@0
  2325
  g_assert (*arg != '\0');
sl@0
  2326
  arg = arg_iterate (arg, &arg_name, &arg_in, &constval, &retval, &arg_signature);
sl@0
  2327
  g_assert (!strcmp (arg_name, "val"));
sl@0
  2328
  g_assert (arg_in == TRUE);
sl@0
  2329
  g_assert (!strcmp (arg_signature, "v"));
sl@0
  2330
  g_assert (*arg != '\0');
sl@0
  2331
  arg = arg_iterate (arg, &arg_name, &arg_in, &constval, &retval, &arg_signature);
sl@0
  2332
  g_assert (retval == RETVAL_NONE);
sl@0
  2333
  g_assert (arg_in == FALSE);
sl@0
  2334
  g_assert (!strcmp (arg_signature, "s"));
sl@0
  2335
  g_assert (*arg == '\0');
sl@0
  2336
sl@0
  2337
  sigdata = dbus_glib_internal_test_object_info.exported_signals;
sl@0
  2338
  g_assert (*sigdata != '\0');
sl@0
  2339
  sigdata = propsig_iterate (sigdata, &iface, &signame);
sl@0
  2340
  g_assert (!strcmp (iface, "org.freedesktop.DBus.Tests.MyObject"));
sl@0
  2341
  g_assert (!strcmp (signame, "Frobnicate"));
sl@0
  2342
  g_assert (*sigdata != '\0');
sl@0
  2343
  sigdata = propsig_iterate (sigdata, &iface, &signame);
sl@0
  2344
  g_assert (!strcmp (iface, "org.freedesktop.DBus.Tests.FooObject"));
sl@0
  2345
  g_assert (!strcmp (signame, "Sig0"));
sl@0
  2346
  g_assert (*sigdata != '\0');
sl@0
  2347
  sigdata = propsig_iterate (sigdata, &iface, &signame);
sl@0
  2348
  g_assert (!strcmp (iface, "org.freedesktop.DBus.Tests.FooObject"));
sl@0
  2349
  g_assert (!strcmp (signame, "Sig1"));
sl@0
  2350
  g_assert (*sigdata != '\0');
sl@0
  2351
  sigdata = propsig_iterate (sigdata, &iface, &signame);
sl@0
  2352
  g_assert (!strcmp (iface, "org.freedesktop.DBus.Tests.FooObject"));
sl@0
  2353
  g_assert (!strcmp (signame, "Sig2"));
sl@0
  2354
  g_assert (*sigdata == '\0');
sl@0
  2355
  
sl@0
  2356
sl@0
  2357
  i = 0;
sl@0
  2358
  while (i < (int) G_N_ELEMENTS (name_pairs))
sl@0
  2359
    {
sl@0
  2360
      char *uscore;
sl@0
  2361
      char *wincaps;
sl@0
  2362
sl@0
  2363
      uscore = _dbus_gutils_wincaps_to_uscore (name_pairs[i].wincaps);
sl@0
  2364
      wincaps = uscore_to_wincaps (name_pairs[i].uscore);
sl@0
  2365
sl@0
  2366
      if (strcmp (uscore, name_pairs[i].uscore) != 0)
sl@0
  2367
        {
sl@0
  2368
          g_printerr ("\"%s\" should have been converted to \"%s\" not \"%s\"\n",
sl@0
  2369
                      name_pairs[i].wincaps, name_pairs[i].uscore,
sl@0
  2370
                      uscore);
sl@0
  2371
          exit (1);
sl@0
  2372
        }
sl@0
  2373
      
sl@0
  2374
      if (strcmp (wincaps, name_pairs[i].wincaps) != 0)
sl@0
  2375
        {
sl@0
  2376
          g_printerr ("\"%s\" should have been converted to \"%s\" not \"%s\"\n",
sl@0
  2377
                      name_pairs[i].uscore, name_pairs[i].wincaps,
sl@0
  2378
                      wincaps);
sl@0
  2379
          exit (1);
sl@0
  2380
        }
sl@0
  2381
      
sl@0
  2382
      g_free (uscore);
sl@0
  2383
      g_free (wincaps);
sl@0
  2384
sl@0
  2385
      ++i;
sl@0
  2386
    }
sl@0
  2387
  
sl@0
  2388
  return TRUE;
sl@0
  2389
}
sl@0
  2390
sl@0
  2391
#endif /* DBUS_BUILD_TESTS */