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